• 启动自定义

    启动自定义

    我们常常需要在应用启动期间进行一些初始化工作,等初始化完成后应用才可以启动成功,并开始对外提供服务。

    框架提供了统一的入口文件(app.js)进行启动过程自定义,这个文件返回一个 Boot 类,我们可以通过定义 Boot 类中的生命周期方法来执行启动应用过程中的初始化工作。

    框架提供了这些 生命周期函数供开发人员处理:

    • 配置文件即将加载,这是最后动态修改配置的时机(configWillLoad
    • 配置文件加载完成(configDidLoad
    • 文件加载完成(didLoad
    • 插件启动完毕(willReady
    • worker 准备就绪(didReady
    • 应用启动完成(serverDidReady
    • 应用即将关闭(beforeClose

    我们可以在 app.js 中定义这个 Boot 类,下面我们抽取几个在应用开发中常用的生命周期函数来举例:

    1. // app.js
    2. class AppBootHook {
    3. constructor(app) {
    4. this.app = app;
    5. }
    6. configWillLoad() {
    7. // 此时 config 文件已经被读取并合并,但是还并未生效
    8. // 这是应用层修改配置的最后时机
    9. // 注意:此函数只支持同步调用
    10. // 例如:参数中的密码是加密的,在此处进行解密
    11. this.app.config.mysql.password = decrypt(this.app.config.mysql.password);
    12. // 例如:插入一个中间件到框架的 coreMiddleware 之间
    13. const statusIdx = this.app.config.coreMiddleware.indexOf('status');
    14. this.app.config.coreMiddleware.splice(statusIdx + 1, 0, 'limit');
    15. }
    16. async didLoad() {
    17. // 所有的配置已经加载完毕
    18. // 可以用来加载应用自定义的文件,启动自定义的服务
    19. // 例如:创建自定义应用的示例
    20. this.app.queue = new Queue(this.app.config.queue);
    21. await this.app.queue.init();
    22. // 例如:加载自定义的目录
    23. this.app.loader.loadToContext(path.join(__dirname, 'app/tasks'), 'tasks', {
    24. fieldClass: 'tasksClasses',
    25. });
    26. }
    27. async willReady() {
    28. // 所有的插件都已启动完毕,但是应用整体还未 ready
    29. // 可以做一些数据初始化等操作,这些操作成功才会启动应用
    30. // 例如:从数据库加载数据到内存缓存
    31. this.app.cacheData = await this.app.model.query(QUERY_CACHE_SQL);
    32. }
    33. async didReady() {
    34. // 应用已经启动完毕
    35. const ctx = await this.app.createAnonymousContext();
    36. await ctx.service.Biz.request();
    37. }
    38. async serverDidReady() {
    39. // http / https server 已启动,开始接受外部请求
    40. // 此时可以从 app.server 拿到 server 的实例
    41. this.app.server.on('timeout', socket => {
    42. // handle socket timeout
    43. });
    44. }
    45. }
    46. module.exports = AppBootHook;

    注意:在自定义生命周期函数中不建议做太耗时的操作,框架会有启动的超时检测。

    如果你的 Egg 框架的生命周期函数是旧版本的,建议你升级到类方法模式;详情请查看升级你的生命周期事件函数。