It is important to note that this article is not about how to use Next or Egg. It is recommended that you have some knowledge of Egg and Next. This article mainly wants to express some jokes about Next and how it has been used with Egg.
Recently, I was thinking about a problem. The front-end used to render the page to the browser through the template engine at the back end of PHP or Java, and now the MVC/MVVM framework like React, Vue and NG adopts asynchronous data request and client render the page. I think it’s actually progress. But since Node came out, it has made an SSR, or server isomorphism. It feels like you’re back to square one, just in a different language.
I also know the significance of SSR is conducive to optimize SEO, optimize the speed of the white screen, but at the same time if our website has requirements for SEO, then we have to use SSR technology. But the strain on the server actually increases the strain, so use SSR on demand. A little personal feeling, if there are strong requirements for SEO, feel the size of the company is limited, then the use of Node and how many small companies? For example, Taobao, Jingdong, and so on, rather than doing SEO, it is more direct to buy rankings for money search engine suppliers. Of course, this is a digression, is the so-called technology is not pressure body, for a front-end, the use of Node slowly became a just need, then of course to understand the use of SSR, in case of need.
After doing a few demos of Next, I came up with some issues.
- Next does a lot of webpack configuration for React builds, package cutting, etc. In theory, we don’t need to modify or extend anything. However, if your project is an old one rather than a new one, you might have a lot of webpack configuration too. You’ll probably need to spend some time working with Next’s WebPack configuration, which brings us to the first question: What does Next have in it, and what does it conflict with? After checking the documents, there seems to be no detailed explanation.
- The documentation only describes the basic usage and does not explain how the API is used. It is likely that the authors want developers to focus only on the usage and do not need to bother with the principle and how the API is used. Is that really good?
- The entire website does not specify how Next will use Express or Koa, and it is not possible for a project to use Next to replace Koa or Express. After all, Next was supposed to be responsible for view rendering.
Since I use Egg, do I expect the author to tell me how to use it with Egg, since there are no detailed instructions on how to use it with Koa? I’ll ask the Egg developers for a better example. And then… Since the reply is!
Ha ha! In this case I will find out for myself!
Next says how to start a custom Next. Here’s the official example:
// This file doesn't go through babel or webpack transformation. // Make sure the syntax and sources this file requires are compatible with the current node version you are running // See https://github.com/zeit/next.js/issues/1245 for discussions on Universal Webpack or universal Babel const { createServer } = require('http') const { parse } = require('url') const next = require('next') const dev = process.env.NODE_ENV ! == 'production' const app = next({ dev }) const handle = app.getRequestHandler() app.prepare().then(() => { createServer((req, res) => { // Be sure to pass `true` as the second argument to `url.parse`. // This tells it to parse the query portion of the URL. const parsedUrl = parse(req.url, true) const { pathname, query } = parsedUrl if (pathname === '/a') { app.render(req, res, '/b', query) } else if (pathname === '/b') { app.render(req, res, '/a', query) } else { handle(req, res, parsedUrl) } }).listen(3000, err => { if (err) throw err console.log('> Ready on http://localhost:3000') }) })Copy the code
May I level problem, green difficult to understand. Because an Egg starts differently, it encapsulates the start. Well, I’ll try to make it work. I looked in the Egg documentation, started some life cycles, and decided to instantiate Next in beforeStart.
When startup finds nothing wrong, try writing a Controller to render a page.
Write a route
Write a controller
Go you!
WTF!!!!
After the breakpoint, the render function returns a Promise, so I’ll add an await.
Render successfully, but after a breakpoint to find the return of undefined, so how it render to the page. Just tell me there’s something in Render! Holding the heart of learning I break into the code, found that there is really something! Break into the render function and find that the render function called sendHTML, and find that sendHTML is doing everything for us!
Here is the code for the breakpoints, next-server.js and render.js respectively
Next-server’s render function
Render’s sendHTML function
Github source code:
next-server
sendHTML
I realized how much Next had done for us. Incidentally, Next used res.end() to return data, and it was verified that Koa’s Onion model middleware fired properly. However, when using Egg, before using Next, we need to do some pre-injection of HTML template data, such as some template data, etc. We use egg-view plug-in, which renders an HTML string and puts it into ctx.body to return. Next render is not appropriate if you need to render an HTML string and need to do some special processing.
Egg template rendering (just a demo)
We didn’t find the API that returns HTML strings by reading the documentation, so we continue to look at the source code, and when we call render, we find a surprise.
A function called renderToHTML goes through the breakpoint and finds that it does return a compiled HTML string, which is fine! After looking at it, the API is not internal, but exposed, which means we can use it.
Making address:
zeit/next.js
In this case, let’s change it to something more like egg-view.
Verified!
RenderToHTML is called internally in Next’s render, and renderToHTML is not returned. It also handles the returned logic for us, adding some information in the response header. Therefore, there should be no difference between render and renderToHTML in theory. You can get an HTML string.
The following configuration must be added to the Egg route as Next builds files in the _next folder and imports them via pages that rely on files in the _next folder:
And use the handle function in the corresponding controller
At this point, you’re almost done plugging Next into the Egg. But I like to toss, how can so finish. What is the handle?
First, the Handle is mounted in the app by starting Next in the Egg and instantiating it.
So what is this thing? We point all missed routes to a controller that handles the return of the next build file. Then we don’t tell the Handle to return any path. We just call it and return the corresponding resource. Let’s look at the source again.
First we call getRequestHandler on startup, returning a handle.
And then what happens when we call handle in the Controller?
When we call, we need to pass in req and RES, and of course a third argument, which we can pass in the corresponding data. After some internal formatting, the URL value of req is taken and passed into a run function.
A router.match method was called immediately after the run was passed, which, according to the name, should match the url of the current REq through the Next internal route and then return the corresponding content. We all know that if we simply use Next, it actually has a routing system of its own. All pages are found in Pages by corresponding URLS, and then Next internally processes the URL starting with _NEXT to obtain resource files in the Next folder. So you can see why Next officially says that the _next path and pages should not be modified, because this series of internal configuration results. (Feels bad about scaling)
Router = router = router
router
This gives us a basic understanding of why we can use the Handle function to match the file resources we need. Because Handle internally matches itself to paths corresponding to _next and Pages files based on the current req. URL. It is commonly understood that when Egg does not match any of its routes, it writes a request that matches any of its routes, and then calls Handle to try to match Next’s own route configuration to see if it can match.
Look back at the custom launch demo in the documentation on the Next website and you’ll see exactly what it means.
At this point, Egg access to Next is basically complete, of course, if you need to elaborate, after Egg access, how to monitor Next error and so on. However, it has not been officially put into production, and the follow-up pit summary will be conducted after it is put into production in the future.
conclusion
- Next has to tease itself with documentation, basic usage, and no detailed API usage.
- The reason for the high level of encapsulation, for the diversity of business, can be compatible with so many different business scenarios need to question.
- How much impact will the pressure on the server have, it seems that there is no good data support.
- It’s not hard to get Egg in, but it might be nice to use for new projects, because there’s a lot of webPack configuration built in, and it might be a bit of a headache with older projects.
I have basically climbed out of a pit here, but I can’t find the combination of Egg and Next on any major websites. I don’t know whether it is appropriate for me to use it like this and whether there will be any problems. I hope you can give me some suggestions! Thanks a million!
Reprint of the self zhihu: zhuanlan.zhihu.com/p/54841918