An egg is introduced
What is an egg?
Egg is a Node.js back-end Web framework from Alibaba, based on KOA packaging and some conventions.
Why is it called an egg?
Egg is conceived because egg is positioned as an enterprise-level Web infrastructure designed to help developers develop a framework suitable for their team.
Which products were developed with egg?
Egg is used to develop the whisperfinch, and the architecture diagram is as follows:
Which companies use eggs?
Hema, PingWest, PingWest, 58.com, etc. (Technology stack selection reference link)
Does Egg support Typescript?
While the egg itself is written in JavaScript, the egg application can be written in Typescript by creating a project using the following command (see link) :
$ npx egg-init --type=ts showcase
Copy the code
Is there an intelligent tip for writing an egg in JavaScript?
The typings directory will be dynamically generated in the project root directory by adding the following declaration to package.json, which contains the type declarations for various models (see link) :
"egg": {
"declarations": true
}
Copy the code
What is the relationship between egg and KOA?
Koa is the basic framework of egg, and egg is an enhancement of KOA.
Do I need koA to learn egg?
You can also start with eggs without KOA, but knowing KOA will help you understand them in a deeper way.
Create a project
We created an Egg project using the basic template and selected the domestic image:
$ npm init egg --type=simple --registry=china
Yarn create egg --type=simple --registry= China
Copy the code
To explain the NPM init egg syntax:
npm@6 introduces the npm-init
syntax, which is equivalent to the NPX create-
command. The NPX command will look for an executable file named create-
in $PATH and node_modules/. Bin. If it is not found, it will be executed.
In other words, the NPM init egg will find or download the create-egg executable, and the create-egg package is the alias for the egg-init package, which is called egg-init.
After the creation, the directory structure is as follows (ignore the README file and test directory) :
├ ─ ─ app │ ├ ─ ─ controller │ │ └ ─ ─ home. Js │ └ ─ ─ the router. The js ├ ─ ─ the config │ ├ ─ ─ config. The default. The js │ └ ─ ─ plugin. Js ├ ─ ─ package.jsonCopy the code
This is the minimal egg project. After installing the dependency with NPM I or yarn, run the start command:
$NPM run dev [master] node version v14.15.1 [master] egg version 2.29.1 [master] agent_worker#1:23135 started (842ms)
[master] egg started on http://127.0.0.1:7001 (1690ms)
Copy the code
Open http://127.0.0.1:7001/ and you’ll see a page that says hi, egg.
Directory conventions
The project created above is just a minimal structure. A typical egg project has the following directory structure:
An egg - project ├ ─ ─ package. Json ├ ─ ─ app. Js (optional) ├ ─ ─ agent. The js (optional) ├ ─ ─ app / | ├ ─ ─ the router, js# Is used to configure URL routing rules│ ├ ─ ─ the controller /# Store controller (parse user input, process, return result)│ ├── Model/(Optional)# to store the database model│ ├── Service/(Optional)# for writing the business logic layer│ ├── Middleware/(Optional)# for writing middleware│ ├── Schedule/(Optional)# used to set scheduled tasks│ ├── Public Exercises/(Optional)# for static resources│ ├── View/(Optional)# to place the template file│ ├ ─ extension/(optional)# For extensions to the framework│ ├ ─ ─ helper. Js (optional) │ ├ ─ ─ request. Js (optional) │ ├ ─ ─ the response. The js (optional) │ ├ ─ ─ the context, js (optional) │ ├ ─ ─ application. Js (optional) │ └ ─ ─ Agent. Js (optional) ├ ─ ─ the config / | ├ ─ ─ plugin. Js# to configure the plug-in to be loaded| ├ ─ ─ config. {env}. Js# used to write configuration files (env can be default, prod, test, local, unittest)
Copy the code
This is agreed upon by the Egg framework or the built-in plugin, and is a best practice summarized by Ali. Although the framework does provide the ability for users to customize the directory structure, it is recommended to follow The Ali approach. In the following chapters, the functions of these conventions directories and files will be explained one by one.
Router
The route defines the mapping between the request path (URL) and the Controller (Controller), that is, which Controller should handle the URL accessed by the user. Let’s open app/router.js and take a look:
module.exports = app= > {
const { router, controller } = app
router.get('/', controller.home.index)
};
Copy the code
As you can see, the routing file exports a function that takes an APP object as a parameter and defines the mapping with the following syntax:
router.verb('path-match', controllerAction)
Copy the code
Verb is usually a lower case of an HTTP verb, such as:
- HEAD –
router.head
- OPTIONS –
router.options
- GET –
router.get
- PUT –
router.put
- POST –
router.post
- PATCH –
router.patch
- DELETE –
router.delete
或router.del
On top of that, there’s a special verb called router. Redirect.
The controllerAction function specifies a specific function in a file in the controller directory using the dot (·) syntax. For example:
controller.home.index // Map the index method to the controller/home.js file
controller.v1.user.create // Controller /v1/user.js file create method
Copy the code
Here are some examples and their explanations:
module.exports = app= > {
const { router, controller } = app
// When the user accesses the news, the index method of controller/news.js will handle it
router.get('/news', controller.news.index)
// Capture the named parameter x in the URL with the colon ':x' and put it into ctx.params.x
router.get('/user/:id/:name', controller.user.info)
// Capture the grouping parameters in the URL by custom re and put them into ctx.params
router.get(/^\/package\/([\w-.]+\/[\w-.]+)$/, controller.package.detail)
}
Copy the code
In addition to using verbs to create routes, egg provides the following syntax to quickly generate CRUD routes:
// Map the RESTful style of posts to the controller/posts.js
router.resources('posts'.'/posts', controller.posts)
Copy the code
The following routes are automatically generated:
HTTP method | Request path | The name of the routing | Controller function |
---|---|---|---|
GET | /posts | posts | app.controller.posts.index |
GET | /posts/new | new_post | app.controller.posts.new |
GET | /posts/:id | post | app.controller.posts.show |
GET | /posts/:id/edit | edit_post | app.controller.posts.edit |
POST | /posts | posts | app.controller.posts.create |
PATCH | /posts/:id | post | app.controller.posts.update |
DELETE | /posts/:id | post | app.controller.posts.destroy |
You just go to controller and implement the corresponding method. |
As the project gets bigger and bigger, there will be more and more route mappings. We may want to be able to split the route mappings into files. There are two ways to do this:
-
Manual import, that is, write the routing file to the app/router directory, and then import these files into app/router.js. Example code:
// app/router.js module.exports = app= > { require('./router/news')(app) require('./router/admin')(app) }; // app/router/news.js module.exports = app= > { app.router.get('/news/list', app.controller.news.list) app.router.get('/news/detail', app.controller.news.detail) }; // app/router/admin.js module.exports = app= > { app.router.get('/admin/user', app.controller.admin.user) app.router.get('/admin/log', app.controller.admin.log) }; Copy the code
-
Use egg-router-plus plug-in to automatically import app/router/**/*.js and provide namespace function:
// app/router.js module.exports = app= > { const subRouter = app.router.namespace('/sub') subRouter.get('/test', app.controller.sub.test) // The final path is /sub/test } Copy the code
In addition to HTTP Verb, the Router also provides a redirect method for internal redirection, such as:
module.exports = app= > {
app.router.get('index'.'/home/index', app.controller.home.index)
app.router.redirect('/'.'/home/index'.302)}Copy the code
Middleware
Egg: A middleware is a separate file placed in the app/ Middleware directory that exports a common function that takes two arguments:
- Options: Middleware configuration items that the framework will
app.config[${middlewareName}]
Pass it in. - App: indicates the instance of the current Application.
Let’s create a new middleware/slow.js and print a log if the request time exceeds our specified threshold. The code is:
module.exports = (options, app) = > {
return async function (ctx, next) {
const startTime = Date.now()
await next()
const consume = Date.now() - startTime
const { threshold = 0 } = options || {}
if (consume > threshold) {
console.log(`${ctx.url}Request time${consume}Ms `)}}}Copy the code
Then use it in config.default.js:
module.exports = {
// Configure the required middleware. The array order is the loading order of the middleware
middleware: [ 'slow'].// Slow middleware options parameter
slow: {
enable: true}},Copy the code
If you only want to use the middleware in the specified route, for example, if you only want to use the middleware for url requests beginning with/API prefix, there are two ways to use the middleware:
-
Set the match or ignore property in the config.default.js configuration:
module.exports = { middleware: [ 'slow'].slow: { threshold: 1.match: '/api'}};Copy the code
-
Import in the routing file router.js
module.exports = app= > { const { router, controller } = app // Add any middleware before controller processing router.get('/api/home', app.middleware.slow({ threshold: 1 }), controller.home.index) } Copy the code
An egg divides the middleware application layer definition of middleware (app. Config. AppMiddleware) and the default middleware framework (app. Config. CoreMiddleware), we look to print:
module.exports = app= > {
const { router, controller } = app
console.log(app.config.appMiddleware)
console.log(app.config.coreMiddleware)
router.get('/api/home', app.middleware.slow({ threshold: 1 }), controller.home.index)
}
Copy the code
The result is:
// appMiddleware
[ 'slow' ]
// coreMiddleware
[
'meta'.'siteFile'.'notfound'.'static'.'bodyParser'.'overrideMethod'.'session'.'securities'.'i18n'.'eggLoaderTrace'
]
Copy the code
CoreMiddleware is one of the pieces of middleware built into egg and is enabled by default. If you don’t want to use coreMiddleware, you can configure it to turn it off:
module.exports = {
i18n: {
enable: false}}Copy the code
Controller
The Controller parses the user’s input and returns the result. The simplest example of helloWorld is:
const { Controller } = require('egg');
class HomeController extends Controller {
async index() {
const { ctx } = this;
ctx.body = 'hi, egg'; }}module.exports = HomeController;
Copy the code
Of course, our code in a real project is not as simple as this, but in general, the Controller does several things:
- Receive, verify, and process HTTP request parameters
- The Service is called down to process the business
- The result is responded to the user over HTTP
Here’s a real case:
const { Controller } = require('egg');
class PostController extends Controller {
async create() {
const { ctx, service } = this;
const createRule = {
title: { type: 'string' },
content: { type: 'string'}};// Check and assemble parameters
ctx.validate(createRule);
const data = Object.assign(ctx.request.body, { author: ctx.session.userId });
// Call Service for business processing
const res = await service.post.create(data);
// Respond to the client data
ctx.body = { id: res.id };
ctx.status = 201; }}module.exports = PostController;
Copy the code
Since Controller is a class, it is possible to encapsulate common methods using a custom base class. For example:
// app/core/base_controller.js
const { Controller } = require('egg');
class BaseController extends Controller {
get user() {
return this.ctx.session.user;
}
success(data) {
this.ctx.body = { success: true, data };
}
notFound(msg) {
this.ctx.throw(404, msg || 'not found'); }}module.exports = BaseController;
Copy the code
Then let all controllers inherit from this custom BaseController:
// app/controller/post.js
const Controller = require('.. /core/base_controller');
class PostController extends Controller {
async list() {
const posts = await this.service.listByUser(this.user);
this.success(posts); }}Copy the code
This. CTX allows you to get the context object in the Controller, making it easy to get and set parameters. For example:
ctx.query
: Request parameter in URL (ignore duplicate key)ctx.quries
: Request parameters in the URL (duplicate keys are put into the array)ctx.params
: Named parameter on the Routerctx.request.body
: Indicates the content in the HTTP request bodyctx.request.files
: File object uploaded by the front endctx.getFileStream()
: Obtains the uploaded file streamctx.multipart()
: getmultipart/form-data
datactx.cookies
: Reads and sets cookiesctx.session
: Reads and sets the sessionctx.service.xxx
: Get an instance of the specified service object (lazy loading)ctx.status
: Sets the status codectx.body
: Sets the response bodyctx.set
: Sets the response headerctx.redirect(url)
: redirectctx.render(template)
: Render template
This. CTX context object is one of the most important objects in the Egg framework and koA framework. We need to find out what this object does. Object.keys(CTX)
[
'request'.'response'.'app'.'req'.'res'.'onerror'.'originalUrl'.'starttime'.'matched'.'_matchedRoute'.'_matchedRouteName'.'captures'.'params'.'routerName'.'routerPath'
]
Copy the code
Service (Service)
A Service is an implementation of specific business logic. A encapsulated Service can be called by multiple controllers, and multiple services can be called from a single Controller, although business logic can also be written to the Controller. This is not recommended, however, and the Controller logic should be kept simple in your code and only serve as a “bridge.”
The Controller can call any method on any Service. Note that the Service is lazily loaded, meaning that the framework instantiates it only when it is accessed.
In general, a Service does several things:
- Handle complex business logic
- Calling databases or third party services (e.g. GitHub Information capture, etc.)
A simple example of a Service that returns the results of a query in a database:
// app/service/user.js
const { Service } = require('egg').Service;
class UserService extends Service {
async find(uid) {
const user = await this.ctx.db.query('select * from user where uid = ? ', uid);
returnuser; }}module.exports = UserService;
Copy the code
In Controller you can directly call:
class UserController extends Controller {
async info() {
const { ctx } = this;
const userId = ctx.params.id;
const userInfo = awaitctx.service.user.find(userId); ctx.body = userInfo; }}Copy the code
Note that the Service file must be placed in the app/ Service directory, and can be accessed by cascading directory names:
app/service/biz/user.js => ctx.service.biz.user
app/service/sync_user.js => ctx.service.syncUser
app/service/HackerNews.js => ctx.service.hackerNews
Copy the code
Functions in a Service can be understood as the smallest unit of a specific business logic. Services can also call other services. It is worth noting that: Service is not a singleton, but a request-level object. The framework delays instantiation of ctx.service.xx when it first accesses ctx.ctx in each request, so the context of the current request can be obtained in the Service through this.ctx.
Template rendering
The Egg framework has built-in Egg-View as a template solution and supports multiple template rendering engines such as EJS, Handlebars, Nunjunks, etc. Each template engine is introduced as a plug-in. By default, all plug-ins will find files in the app/ View directory. Then select a different template engine based on the suffix mapping defined in config\config.default.js:
config.view = {
defaultExtension: '.nj'.defaultViewEngine: 'nunjucks'.mapping: {
'.nj': 'nunjucks'.'.hbs': 'handlebars'.'.ejs': 'ejs',}}Copy the code
The above configuration indicates when the file:
- The suffix is
.nj
Nunjunks template engine is used - The suffix is
.hbs
When using handlebars template engine - The suffix is
.ejs
Using the EJS template engine - Default value when no suffix is specified
.html
- Default is nunjunks when no template engine is specified
Next we install the template engine plug-in:
$ npm i egg-view-nunjucks egg-view-ejs egg-view-handlebars --save
# 或者 yarn add egg-view-nunjucks egg-view-ejs egg-view-handlebars
Copy the code
Then enable the plugin in config/plugin.js:
exports.nunjucks = {
enable: true.package: 'egg-view-nunjucks',}exports.handlebars = {
enable: true.package: 'egg-view-handlebars',}exports.ejs = {
enable: true.package: 'egg-view-ejs',}Copy the code
Then add the app/ View directory and add several files to it:
├── handlebars. HBS ├── nunjunks.njCopy the code
The codes are:
<! -- ejs.ejs file code -->
<h1>ejs</h1>
<ul>
<% items.forEach(function(item){ %>
<li><%= item.title %></li>The < %}); % ></ul>
<! -- handlebars. HBS -->
<h1>handlebars</h1>
{{#each items}}
<li>{{title}}</li>
{{~/each}}
<! -- nunjunks.nj file code -->
<h1>nunjunks</h1>
<ul>
{% for item in items %}
<li>{{ item.title }}</li>
{% endfor %}
</ul>
Copy the code
Then configure the route in the Router:
module.exports = app= > {
const { router, controller } = app
router.get('/ejs', controller.home.ejs)
router.get('/handlebars', controller.home.handlebars)
router.get('/nunjunks', controller.home.nunjunks)
}
Copy the code
Next, implement the Controller logic:
const Controller = require('egg').Controller
class HomeController extends Controller {
async ejs() {
const { ctx } = this
const items = await ctx.service.view.getItems()
await ctx.render('ejs.ejs', {items})
}
async handlebars() {
const { ctx } = this
const items = await ctx.service.view.getItems()
await ctx.render('handlebars.hbs', {items})
}
async nunjunks() {
const { ctx } = this
const items = await ctx.service.view.getItems()
await ctx.render('nunjunks.nj', {items})
}
}
module.exports = HomeController
Copy the code
We put the data in the Service:
const { Service } = require('egg')
class ViewService extends Service {
getItems() {
return[{title: 'foo'.id: 1 },
{ title: 'bar'.id: 2}}},]module.exports = ViewService
Copy the code
To see the results of the different template engines, visit the following addresses:
GET http://localhost:7001/nunjunks
GET http://localhost:7001/handlebars
GET http://localhost:7001/ejs
Copy the code
Where does the ctx.render method come from, you may ask? Render, renderView, and renderString methods are added to the CTX context object.
const ContextView = require('.. /.. /lib/context_view')
const VIEW = Symbol('Context#view')
module.exports = {
render(. args) {
return this.renderView(... args).then(body= > {
this.body = body; })},renderView(. args) {
return this.view.render(... args); },renderString(. args) {
return this.view.renderString(... args); },get view() {
if (this[VIEW]) return this[VIEW]
return this[VIEW] = new ContextView(this)}}Copy the code
Internally, it will eventually forward the call to the render method on the ContextView instance, which is a class that will help us find the rendering engine based on the mapping defined in the configuration.
The plug-in
In the last video on template rendering, we saw how to use plugins, which are declared in the config/plugin.js of the application or framework:
exports.myPlugin = {
enable: true.// Whether to enable
package: 'egg-myPlugin'.// From node_modules
path: path.join(__dirname, '.. /lib/plugin/egg-mysql'), // Import from the local directory
env: ['local'.'unittest'.'prod'].// This function can be enabled only in the specified runtime environment
}
Copy the code
After the plug-in is enabled, you can use the functions provided by the plug-in:
app.myPlugin.xxx()
Copy the code
If the plug-in contains user-defined configurations, you can specify them in config.default.js, for example:
exports.myPlugin = {
hello: 'world'
}
Copy the code
A plug-in is a “mini-application” that contains services, middleware, configurations, framework extensions, etc., but does not have a separate Router or Controller, and cannot define its own plugin.js.
In the development must be connected to the database, the most practical plug-in is the plug-in database integration.
Integrated mongo
First, make sure the MongoDB database is installed and started on your computer. On a Mac, use the following command to quickly install and start the MongoDB database:
$ brew install mongodb-community
$ brew services start mongodb/brew/mongodb-community # Background startup
# or use mongod -- config/usr/local/etc/mongod. Conf start at the front desk
Copy the code
Then install the Egg-Mongoose plugin:
$ npm i egg-mongoose
# 或者 yarn add egg-mongoose
Copy the code
Open the plugin in config/plugin.js:
exports.mongoose = {
enable: true.package: 'egg-mongoose',}Copy the code
Define the connection parameters in config/config.default.js:
config.mongoose = {
client: {
url: 'mongo: / / 127.0.0.1 / example'.options: {}}}Copy the code
Then define the model in model/user.js:
module.exports = app= > {
const mongoose = app.mongoose
const UserSchema = new mongoose.Schema(
{
username: {type: String.required: true.unique: true}, / / user name
password: {type: String.required: true}, / / password
},
{ timestamps: true } // Generate the createdAt and updatedAt timestamps automatically
)
return mongoose.model('user', UserSchema)
}
Copy the code
Call mongoose’s methods in the controller:
const {Controller} = require('egg')
class UserController extends Controller {
// User list GET /users
async index() {
const {ctx} = this
ctx.body = await ctx.model.User.find({})
}
// User details GET /users/:id
async show() {
const {ctx} = this
ctx.body = await ctx.model.User.findById(ctx.params.id)
}
// Create user POST /users
async create() {
const {ctx} = this
ctx.body = await ctx.model.User.create(ctx.request.body)
}
// Update user PUT /users/:id
async update() {
const {ctx} = this
ctx.body = await ctx.model.User.findByIdAndUpdate(ctx.params.id, ctx.request.body)
}
// DELETE the user. DELETE /users/:id
async destroy() {
const {ctx} = this
ctx.body = await ctx.model.User.findByIdAndRemove(ctx.params.id)
}
}
module.exports = UserController
Copy the code
Finally, configure RESTful route mapping:
module.exports = app= > {
const {router, controller} = app
router.resources('users'.'/users', controller.user)
}
Copy the code
Integration of the MySQL
Make sure you have the MySQL database installed on your computer. If you are on a Mac, run the following command to quickly install and start MySQL:
$ brew install mysql
$ brew services start mysql # Background startup
Mysql. server start
$ mysql_secure_installation # Set password
Copy the code
You can connect to the mysql database by using the egg-mysql plugin.
$ npm i egg-mysql
# yarn add egg-mysql
Copy the code
Open the plugin in config/plugin.js:
exports.mysql = {
enable: true.package: 'egg-mysql',}Copy the code
Define the connection parameters in config/config.default.js:
config.mysql = {
client: {
host: 'localhost'.port: '3306'.user: 'root'.password: 'root'.database: 'cms',}}Copy the code
Mysql = app.mysql; mysql = app.mysql; mysql = app.mysql;
class UserService extends Service {
async find(uid) {
const user = await this.app.mysql.get('users', { id: 11 });
return { user }
}
}
Copy the code
If an error occurs during boot:
ERROR 5954 nodejs.ER_NOT_SUPPORTED_AUTH_MODEError: ER_NOT_SUPPORTED_AUTH_MODE: Client does not support authentication protocol requested by server; consider upgrading MySQL client
Copy the code
This is because you are using MySQL 8.x and egg-mysql relies on ali-rds, which is a package encapsulated by Ali. This package is deprecated and does not support the caching_sha2_password encryption method. This can be resolved by running the following command from MySQL Workbench:
ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'password'
flush privileges
Copy the code
But a better way to integrate MySQL is to use an ORM framework to help you manage your data layer code. Sequelize is one of the most popular ORM frameworks. It supports multiple data sources such as MySQL, PostgreSQL, SQLite, and MSSQL. Next we use sequelize to connect to the MySQL database.
npm install egg-sequelize mysql2 --save
yarn add egg-sequelize mysql2
Copy the code
Then open the egg-sequelize plugin in config/plugin.js:
exports.sequelize = {
enable: true.package: 'egg-sequelize',}Copy the code
Again, write the Sequelize configuration in config/config.default.js
config.sequelize = {
dialect: 'mysql'.host: '127.0.0.1'.port: 3306.database: 'example',}Copy the code
Then create the books table in the egg_Example library:
CREATE TABLE `books` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT 'primary key',
`name` varchar(30) DEFAULT NULL COMMENT 'book name',
`created_at` datetime DEFAULT NULL COMMENT 'created time',
`updated_at` datetime DEFAULT NULL COMMENT 'updated time'.PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='book';
Copy the code
Create model/book.js with the following code:
module.exports = app= > {
const { STRING, INTEGER } = app.Sequelize
const Book = app.model.define('book', {
id: { type: INTEGER, primaryKey: true.autoIncrement: true },
name: STRING(30})),return Book
}
Copy the code
Add controller/book.js controller:
const Controller = require('egg').Controller
class BookController extends Controller {
async index() {
const ctx = this.ctx
ctx.body = await ctx.model.Book.findAll({})
}
async show() {
const ctx = this.ctx
ctx.body = await ctx.model.Book.findByPk(+ctx.params.id)
}
async create() {
const ctx = this.ctx
ctx.body = await ctx.model.Book.create(ctx.request.body)
}
async update() {
const ctx = this.ctx
const book = await ctx.model.Book.findByPk(+ctx.params.id)
if(! book)return (ctx.status = 404)
await book.update(ctx.request.body)
ctx.body = book
}
async destroy() {
const ctx = this.ctx
const book = await ctx.model.Book.findByPk(+ctx.params.id)
if(! book)return (ctx.status = 404)
await book.destroy()
ctx.body = book
}
}
module.exports = BookController
Copy the code
Finally, configure RESTful route mapping:
module.exports = app= > {
const {router, controller} = app
router.resources('books'.'/books', controller.book)
}
Copy the code
Custom plug-in
Now that you have mastered the use of plug-ins, it’s time to write your own plug-ins. First, create a plug-in project based on the plugin scaffold template:
npm init egg --type=plugin
# yarn create egg --type=plugin
Copy the code
The default directory structure is:
├── config │ ├─ ├─ config.default.js ├── package.jsonCopy the code
The plugin does not have a separate router or controller, and you need to specify plugin-specific information in the eggPlugin node in package.json, for example:
{
"eggPlugin": {
"name": "myPlugin"."dependencies": [ "registry"]."optionalDependencies": [ "vip"]."env": [ "local"."test"."unittest"."prod"]}}Copy the code
The meanings of the preceding fields are:
name
– Plug-in name: Specifies the name of the dependent plug-in when you configure the dependency relationship.dependencies
– List of plug-ins that the current plug-in strongly depends on (If the dependent plug-in is not found, the application fails to start).optionalDependencies
– List of optional dependencies of the current plug-in (If the dependent plug-in is not enabled, only a Warning is displayed, which does not affect the application startup).env
– Specifies that the current plug-in is enabled only in certain runtime environments
What can be done in the plug-in?
-
Extend built-in objects: Define request.js, response.js, and other files in app/extend/, just like the application.
For example, the egg-bcrypt library simply extends extend.js:
Call ctx.genhash (plainText) and ctx.compare(plainText, hash) directly in the project.
-
Insert custom middleware: Write middleware in app/ Middleware and use it in app.js
For example, the Egg-Cors library defines a core.js middleware that uses KOA-CORS as it is
Config /config.default.js
exports.cors = { origin: The '*'.allowMethods: 'GET,HEAD,PUT,POST,DELETE,PATCH' } Copy the code
-
Do some initialization at startup: add synchronous or asynchronous initialization code in app.js
For example, egg- ElasticSearch code:
The beforeStart method can also define asynchronous start logic. It doesn’t matter whether the beforeStart method is synchronized or not, but if there is asynchronous logic, you can wrap an async function.
-
Set a scheduled task: Add a scheduled task to the app/schedule/ directory. We’ll see more about scheduled tasks in the next section.
Timing task
In a complex business scenario, there will inevitably be the need for scheduled tasks, such as:
- Check every day to see if a user has a birthday and send birthday wishes automatically
- Back up the database every day to prevent data loss caused by improper operations
- Delete temporary files once a week to release disk space
- Periodically obtain data from the remote interface and update the local cache
In the app/schedule directory, each file is an independent scheduled task. You can configure the properties of the scheduled task and the method to execute it. For example, create an update_cache.js update cache task and execute it every minute:
const Subscription = require('egg').Subscription
class UpdateCache extends Subscription {
// Use the schedule attribute to set the execution interval of scheduled tasks
static get schedule() {
return {
interval: '1m'.// 1 minute interval
type: 'all'.// Specify that all workers need to execute}}// Subscribe is the function that is run when a truly scheduled task is executed
async subscribe() {
const res = await this.ctx.curl('http://www.api.com/cache', {
dataType: 'json',})this.ctx.app.cache = res.data
}
}
module.exports = UpdateCache
Copy the code
That is, the egg gets the configuration of the scheduled task from the static accessor attribute schedule, and then executes the subscribe method according to the configuration. The time to execute a task can be specified by interval or cron:
-
Interval can be a number or string. If it is a number, it represents the number of milliseconds. For example, 5000 is 5 seconds. If it is a character type, it is converted to the number of milliseconds through the ms package.
-
Cron expressions are parsed using cron-parser, with the syntax:
* * * * * * ┬ ┬ ┬ ┬ ┬ ┬ │ │ │ │ │ | │ │ │ │ │ └ day of week (0 to 7) (0 or 7 is Sun │ │ │ │ └ ─ ─ ─ ─ ─ the month (1-12) │ │ │ └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ day of the month (1-31) │ │ └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ the adrenaline-charged (0-23) │ └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ minute (0-59) └ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ ─ the second (0 to 59, optional)Copy the code
There are two types of tasks to perform:
worker
Type: Only one worker will execute this scheduled task (randomly selected)all
Type: Each worker performs this scheduled task
Which type to use depends on the specific business, for example, the task of updating the cache must be selected all, while the task of backing up the database should be selected worker, otherwise the backup will be repeated.
There are some scenarios where we may need to perform scheduled tasks manually, such as initializing tasks when an application is started, which can be run through app.runschedule (schedulePath). App.runschedule accepts a scheduled task file path (either a relative path or a complete absolute path in the app/schedule directory). In app.js, the code is:
module.exports = app= > {
app.beforeStart(async() = > {// Make sure the cache is updated before the program starts
await app.runSchedule('update_cache')})}Copy the code
Error handling
In a development environment, it provides a very friendly visual interface to help developers locate problems. For example, we call this method when we change model.User to lowercase:
Directly locate the error line, convenient for developers to fast debugging. In a production environment, egg does not expose the error stack to the user, but returns the following error message:
Internal Server Error, real status: 500
Copy the code
If our project is separate from the front end and all returns are JSON, we can do this in config/plugin.js:
module.exports = {
onerror: {
accepts: () => 'json',
},
};
Copy the code
The stack of error calls is returned as JSON:
{
"message": "Cannot read property 'find' of undefined"."stack": "TypeError: Cannot read property 'find' of undefined\n at UserController.index (/Users/keliq/code/egg-project/app/controller/user.js:7:37)"."name": "TypeError"."status": 500
}
Copy the code
Accept function is a concrete implementation of the idea of content negotiation, that is, let the user decide which format to return, which also reflects the great flexibility of egg, for example, we want to return JSON format when the content-type is’ ‘, In other cases it returns HTML, which can be written like this:
module.exports = {
onerror: {
accepts: (ctx) = > {
if (ctx.get('content-type') = = ='application/json') return 'json';
return 'html'; }}};Copy the code
However, we can also customize the error in config/config.default.js:
module.exports = {
onerror: {
errorPageUrl: '/public/error.html',}};Copy the code
Errors in the production environment are redirected to this path, followed by an argument? Real_status = 500. In fact, egg errors are handled by the built-in egg-onError plugin, which catches any exceptions thrown by the Middleware, Controller, Service methods of a request and automatically returns different types of errors depending on the type the request is trying to get:
module.exports = {
onerror: {
all(err, ctx) {
// This is where error handling is defined for all response types
// Note that after config.all is defined, other error handling methods will not take effect
ctx.body = 'error'
ctx.status = 500
},
html(err, ctx) { // Handle the HTML hander
ctx.body = '<h3>error</h3>'
ctx.status = 500
},
json(err, ctx) { // json hander
ctx.body = {message: 'error'}
ctx.status = 500}},}Copy the code
The framework does not treat the 404 status returned by the server as an exception. If the status code is 404 and there is no body, the default response is as follows:
-
JSON: {“message”: “Not Found”}
-
404 Not Found
Many factories write their own 404 pages. If you need this, you can also write your own HTML and specify it in config/config.default.js:
module.exports = {
notfound: {
pageUrl: '/404.html',}}Copy the code
If you want to fully customize the server’s 404 response, including the JSON return, as with custom exception handling, you will be able to do so. Just add a piece of middleware/ Notfound_handler.js:
module.exports = () = > {
return async function (ctx, next) {
await next()
if (ctx.status === 404 && !ctx.body) {
ctx.body = ctx.acceptJSON ? { error: 'Not Found' } : '<h1>Page Not Found</h1>'}}}Copy the code
Of course, don’t forget to introduce this middleware in config/config.default.js:
config.middleware = ['notfoundHandler']
Copy the code
The life cycle
The following lifecycle hooks are provided for you to call during egg startup:
- The configuration file is about to load, and this is the last time to dynamically modify the configuration (
configWillLoad
) - The configuration file is loaded (
configDidLoad
) - File loading completed (
didLoad
) - Plug-in startup is complete (
willReady
) - Worker is ready (
didReady
) - Application startup completed (
serverDidReady
) - The application is about to close (
beforeClose
)
Simply create app.js in the project root directory, add and export a class:
class AppBootHook {
constructor(app) {
this.app = app
}
configWillLoad() {
// The config file has been read and merged, but has not yet taken effect. This is the last time for the application layer to modify the configuration
// Note: This function supports synchronous calls only
}
configDidLoad() {
// All the configuration is loaded and can be used to load the application custom files and start the custom service
}
async didLoad() {
// All the configuration is loaded and can be used to load the application custom files and start the custom service
}
async willReady() {
// All plugins are started, but the application is not ready
// You can perform some operations such as data initialization. The application will be started only if these operations are successful
}
async didReady() {
// The application has been started
}
async serverDidReady() {
// HTTP/HTTPS The server is started and begins to accept external requests
// We can get the server instance from app.server
}
async beforeClose() {
// The application is about to close}}module.exports = AppBootHook
Copy the code
The illustration
Framework extension
The Egg framework provides the following extension points:
- Application: The global Application object (Application level) of Koa. There is only one global object and it is created when the Application is started
- Context: Koa’s request Context object (request level), which generates one Context instance per request
- Request: Koa’s Request object (Request level), which provides properties and methods related to the Request
- Response: Koa’s Response object (request level), which provides the properties and methods associated with the Response
- Helper: Provides some useful utility functions
That is, developers can extend the framework built-in objects as much as they want. The extension is written as:
const BAR = Symbol('bar')
module.exports = {
foo(param) {}, // Extend the method
get bar() { // Extended attributes
if (!this[BAR]) {
this[BAR] = this.get('x-bar')}return this[BAR]
},
}
Copy the code
This in the extension point method refers to the extension point object itself. The essence of an extension is to merge user-defined objects onto the Koa extension point object’s prototype, i.e.
- Extending Application is just a way to
app/extend/application.js
Is merged with the Koa Application’s Prototype object and generated based on the extended prototype when the Application is startedapp
Object that can be passedctx.app.xxx
To conduct an interview: - Extending Context is just going to
app/extend/context.js
Is merged with the Prototype object of Koa Context, and CTX objects are generated based on the extended prototype when processing requests. - Extending Request/Response is a way to
app/extend/<request|response>.js
Object defined with the built-inrequest
或response
The extended Prototype object is generated when a request is processedrequest
或response
Object. - The extension Helper is just that
app/extend/helper.js
Object defined with the built-inhelper
The extended Prototype object is generated when a request is processedhelper
Object.
Custom framework
One of the most powerful features of Egg is that it allows teams to customize the framework, which means that you can encapsulate the upper framework based on the egg, just by extending two classes:
- Application: The App Worker instantiates Application, a singleton, when started
- Agent: The Agent Worker instantiates the Agent when it starts, singleton
Custom framework steps:
npm init egg --type=framework --registry=china
Yarn create egg --type=framework --registry= China
Copy the code
Generate the following directory structure:
├ ─ ─ app │ ├ ─ ─ the extend │ │ ├ ─ ─ application. Js │ │ └ ─ ─ the context, js │ └ ─ ─ service │ └ ─ ─ test. The js ├ ─ ─ the config │ ├ ─ ─ │ ├── index.js │ ├── index.js │ ├── index.js │ ├── lib │ ├─ framework.js │ ├── package.jsonCopy the code
As you can see, the structure is the same as a normal egg except for the addition of a lib directory. Take a look at the code in lib/framework.js:
const path = require('path')
const egg = require('egg')
const EGG_PATH = Symbol.for('egg#eggPath')
class Application extends egg.Application {
get [EGG_PATH]() {
return path.dirname(__dirname)
}
}
class Agent extends egg.Agent {
get [EGG_PATH]() {
return path.dirname(__dirname)
}
}
module.exports = Object.assign(egg, {
Application,
Agent,
})
Copy the code
As you can see, you just customize the Application and Agent classes and mount them to the egg object. These custom classes assign the accessor property Symbol. For (‘egg#eggPath’) to path.dirName (__dirName), which is the root directory of the framework. To be able to test the custom framework locally, we first run it under the framework project (let’s say my-Framework) :
npm link # 或者 yarn link
Copy the code
Then run it under the egg project:
npm link my-framework
Copy the code
Finally, add the following code to the package.json of the egg project:
"egg": {
"framework": "my-framework"
},
Copy the code
The implementation of custom frameworks is based on class inheritance. Each layer of frameworks must inherit from the previous layer of frameworks and specify eggPath. Then the framework path of each layer can be obtained by traversing the prototype chain. Department > Enterprise > Egg
const Application = require('egg').Application
// Inherit the Application from the egg
class Enterprise extends Application {
get [EGG_PATH]() {
return '/path/to/enterprise'}}const Application = require('enterprise').Application
// Inherit the Application from enterprise
class Department extends Application {
get [EGG_PATH]() {
return '/path/to/department'}}Copy the code
The advantage of the timing framework is that the business logic can be reused successively. Different departmental frameworks directly use the business logic written in the corporate framework and then complement their own business logic. Although plug-ins can achieve the effect of code reuse, it is not easy to encapsulate business logic into plug-ins. It is better to encapsulate business logic into frameworks. Here are the differences between applications, frameworks and plug-ins:
file | application | The framework | The plug-in |
---|---|---|---|
package.json | ✅ | ✅ | ✅ |
config/plugin.{env}.js | ✅ | ✅ | ❌ |
config/config.{env}.js | ✅ | ✅ | ✅ |
app/extend/application.js | ✅ | ✅ | ✅ |
app/extend/request.js | ✅ | ✅ | ✅ |
app/extend/response.js | ✅ | ✅ | ✅ |
app/extend/context.js | ✅ | ✅ | ✅ |
app/extend/helper.js | ✅ | ✅ | ✅ |
agent.js | ✅ | ✅ | ✅ |
app.js | ✅ | ✅ | ✅ |
app/service | ✅ | ✅ | ✅ |
app/middleware | ✅ | ✅ | ✅ |
app/controller | ✅ | ❌ | ❌ |
app/router.js | ✅ | ❌ | ❌ |
In addition to using Symbol. For (‘egg#eggPath’) to specify path implementation inheritance for the current framework, you can also customize the loader by providing the Symbol. For (‘egg#loader’) accessor property and customizing the AppWorkerLoader:
const path = require('path')
const egg = require('egg')
const EGG_PATH = Symbol.for('egg#eggPath')
const EGG_LOADER = Symbol.for('egg#loader')
class MyAppWorkerLoader extends egg.AppWorkerLoader {
// Customize AppWorkerLoader
}
class Application extends egg.Application {
get [EGG_PATH]() {
return path.dirname(__dirname)
}
get [EGG_LOADER]() {
return MyAppWorkerLoader
}
}
Copy the code
AppWorkerLoader inherits EggLoader from Egg-core. It is a base class that provides built-in methods based on the rules of file loading. AppWorkerLoader does not call these methods itself, but is called by the inherited class.
- loadPlugin()
- loadConfig()
- loadAgentExtend()
- loadApplicationExtend()
- loadRequestExtend()
- loadResponseExtend()
- loadContextExtend()
- loadHelperExtend()
- loadCustomAgent()
- loadCustomApp()
- loadService()
- loadMiddleware()
- loadController()
- loadRouter()
We can override these methods in our custom AppWorkerLoader:
const {AppWorkerLoader} = require('egg')
const {EggLoader} = require('egg-core')
// If you need to change the loading sequence, you must inherit EggLoader; otherwise, you can inherit AppWorkerLoader
class MyAppWorkerLoader extends AppWorkerLoader {
constructor(options) {
super(options)
}
load() {
super.load()
console.log('Custom load logic')}loadPlugin() {
super.loadPlugin()
console.log('Custom plugin loading logic')}loadConfig() {
super.loadConfig()
console.log('Custom Config loading logic')}loadAgentExtend() {
super.loadAgentExtend()
console.log('Custom Agent extend load logic')}loadApplicationExtend() {
super.loadApplicationExtend()
console.log('Custom Application extend loading logic')}loadRequestExtend() {
super.loadRequestExtend()
console.log('Custom Request extend loading logic')}loadResponseExtend() {
super.loadResponseExtend()
console.log('Custom Response extend load logic')}loadContextExtend() {
super.loadContextExtend()
console.log('Custom context extend load logic')}loadHelperExtend() {
super.loadHelperExtend()
console.log('Custom Helper extend load logic')}loadCustomAgent() {
super.loadCustomAgent()
console.log('Custom Agent loading logic')}loadCustomApp() {
super.loadCustomApp()
console.log('Custom App loading logic')}loadService() {
super.loadService()
console.log('Custom Service loading logic')}loadMiddleware() {
super.loadMiddleware()
console.log('Custom Middleware loading logic')}loadController() {
super.loadController()
console.log('Custom Controller loading logic')}loadRouter() {
super.loadRouter()
console.log('Custom Router loading logic')}}Copy the code
The final output is:
Custom Plugin loading logic custom Config loading logic Custom Application Extend loading logic Custom Request Extend loading logic Custom Response Extend loading logic Custom Context Extend loading logic Custom Helper extend Load logic Custom App load logic Custom Service load logic Custom Middleware load logic Custom Controller load logic Router load logic Custom Load logicCopy the code
The output shows the load order by default. This way, the loading logic of the framework is left entirely up to the developer, how to load the Controller, Service, Router, etc.