preface
Recently, the company requested to make an application similar to the official website. In order to solve the SEO problem, it is prepared to use Node to do server-side rendering. I also tried several ready-made application-level frameworks, but in the spirit of cooking chicken need to learn the mood, ready to come up with a simple server side rendering application from scratch, this time I choose Koa as the back-end development framework, first tried template rendering, and then tried Vue SSR
One item runs
Create an empty folder, NPM init
Create server/app.js as the file to start the service
// 1. The installation depends on NPM I KOA
// 2. Modify the properties of scripts in package.json as follows
"scripts": {
"start": "node server/app.js"
}
// 3. App.js write the following code
const Koa = require('koa');
let app = new Koa();
app.use(ctx= > {
ctx.body = 'hello node! '
});
app.listen(3000, () = > {console.log('Server start http://127.0.0.1:3000');
});
// 4 NPM start Visit http://127.0.0.1:3000 to view the result
Copy the code
The second route
Create server/routes.js and use koa-Router middleware to configure routes
/ / package
const router = require('koa-router') ()// Create a routing rule
router.get('/', (ctx, next) => {
ctx.body = 'home'
});
// Export routes for backup
module.exports = router
Copy the code
// server/app.js
/ / the introduction of koa
const Koa = require('koa')
const routers = require('./routes.js')
// Instantiate the KOA object
const app = new Koa()
// Initialize the routing middleware
app.use(routers.routes()).use(routers.allowedMethods());
// Listen on port 3000
app.listen(3000, () = > {console.log('Server start http://127.0.0.1:3000')})Copy the code
Routing can also be said to be nested
Create a server/ Router folder and create a user.js routing file
// server/router.js
const router = require('koa-router') ()const user = require('./router/user')
// Create a routing rule
router.get('/', (ctx, next) => {
ctx.body = 'home'
});
// Mount the user route
router.use('/user', user.routes(), user.allowedMethods())
module.exports = router
Copy the code
// server/router/user.js
const userRouter = require("koa-router") (); userRouter.get("/", (ctx, next) => {
ctx.body = "user";
});
module.exports = userRouter;
/ / NPM start browser visit http://127.0.0.1:3000/user to check the effect
Copy the code
Three template rendering
Koa-views middleware and EJS are used to implement template rendering
Install the KOA template using middleware
npm install --save koa-views
Install ejS template engine
npm install --save ejs
Copy the code
Create server/views/index.ejs file
<html>
<head>
<title><% = title% ></title>
</head>
<body>
<h1><% = title% ></h1>
<p>EJS Welcome to <% = title% ></p>
</body>
</html>
Copy the code
// Add middleware configuration to server/app.js
const path = require("path");
const views = require('koa-views')
// Configure server-side template rendering engine middleware
app.use(views(path.join(__dirname, './views'), {
extension: 'ejs'
}))
// Change to server/router/user.js in the above
userRouter.get("/".async (ctx, next) => {
// ctx.body = "user";
const title = 'hello koa2'
await ctx.render('index', {
title,
})
});
module.exports = userRouter;
/ / NPM start browser visit http://127.0.0.1:3000/user to check the effect
Copy the code
Four Vue SSR
At this time, as a seasoned vUE/React framework worker, I deeply felt the malicious template, which was really not used to. So I decided to give it a try after looking at the Vue SSR documentation.
First, let’s take a look at Vue SSR
As shown in the figure, this is actually an isomorphic concept. After creating the Vue application, webPack is used to package it, and the Server bundle and client bundle are generated. The Server bundle is used for server rendering to return HTML strings, and the client code activates the page so that the server rendering application can be developed directly using VUE. (You can also try nuxt.js.)
Early experience
Vue SSR as long as it relies on the official vue-server-renderer, let’s use it to write a simple ๐ฐ:
// Step 1: Create a Vue instance
const Vue = require('vue')
const app = new Vue({
template: `<div>Hello Vue SSR</div>`
})
// Step 2: Create a renderer
const renderer = require('vue-server-renderer').createRenderer()
// Step 3: Render the Vue instance as HTML
renderer.renderToString(app, (err, html) => {
if (err) throw err
console.log(html) // <div data-server-rendered="true">Hello Vue SSR</div>
})
Copy the code
Use templates in the Node environment
- Create it in the root directory
index.html
file
<html lang="en">
<head>
<title>Hello</title>
</head>
<body>
<! --vue-ssr-outlet-->
<! This is where the application's HTML tags will be injected.
</body>
</html>
Copy the code
- Modify the
server/app.js
To join
const router = require("koa-router") ();const Vue = require('vue')
// Create a renderer
const renderer = require('vue-server-renderer').createRenderer({
// Read the index.html file to form the template
template: require("fs").readFileSync(path.join(__dirname, '.. /index.html'), "utf-8")});// Create a route
router.get("/", ctx => {
const app = new Vue({
data: {
msg: 'Hello Vue SSR'
},
template: `<div>{{msg}}</div>`
})
renderer.renderToString(app, (err, html) => {
ctx.res.end(html);
});
});
// Initialize the routing middleware
app.use(router.routes()).use(router.allowedMethods());
// NPM start Visit http://127.0.0.1:3000 to view the result
Copy the code
Webpack packages the code
Next, write the client code in VUE and use WebPack to package the files
app # Client codeโโ app.vue โโ app.js โโ entry-client.jsRun in browser onlyโ โ โ entry - server. JsRun only on the server
Copy the code
Add client code
// app/App.vue
<template>
<div id="app">
<span>times: {{times}}</span>
<button @click="add">+</button>
<button @click="sub">-</button>
</div>
</template>
<script>
export default {
name: "app".data: function () {
return {
times: 0}},methods: {
add: function () {
this.times = this.times + 1;
},
sub: function () {
this.times = this.times - 1; }}}</script>
Copy the code
The above section is a very simple Vue component that is also common code for both server and client rendering.
In server rendering, app.js only exposes a factory function that returns a new component instance for rendering each time it is called. The rest of the logic is transferred to the client – and browser-side entry files.
// Client render app/app.js
import Vue from 'vue'
import App from './App.vue'
export function createApp() {
return new Vue({
render: h= > h(App)
})
}
Copy the code
// entry-server.js
import { createApp } from './app'
export default context => {
const app = createApp()
return app
}
Copy the code
Client-server. js is used to mount it into the DOM structure with the id app.
// client-server.js
import { createApp } from './app'
var app = createApp();
app.$mount('#app')
Copy the code
Webpack configuration
Build โ โ โ webpack. Base. Config. JsBase common configurationโ โ โ webpack. Client. Config. JsClient package configurationโ โ โ webpack. Server config. JsServer side package configuration
Copy the code
Webpack.base.config.js stores the basic configuration
const path = require("path");
const { VueLoaderPlugin } = require("vue-loader");
module.exports = {
output: {
// Package to root folder 'dist'
path: path.resolve(__dirname, ".. /dist"),
publicPath: "/".filename: "[name].[chunkhash].js"
},
module: {
// Parse the.vue file
rules: [
{
test: /\.vue$/.loader: "vue-loader"
},
{
test: /\.js$/.loader: "babel-loader".exclude: /node_modules/}},plugins: [new VueLoaderPlugin()]
};
Copy the code
// webpack.server.config.js
const path = require("path");
const merge = require("webpack-merge");
const base = require("./webpack.base.config");
// Used to package the generated server-side bundle
// You can end up packing all the files into a JSON file that can be passed to the server renderer.
const VueSSRServerPlugin = require("vue-server-renderer/server-plugin");
function resolve(name) {
return path.resolve(__dirname, "..", name);
}
module.exports = merge(base, {
target: "node".mode: 'production'.// entry-server.js as the entry
entry: resolve("app/entry-server.js"),
output: {
libraryTarget: "commonjs2"
},
plugins: [new VueSSRServerPlugin()]
});
Copy the code
// webpack.client.config.js
const webpack = require("webpack");
const merge = require("webpack-merge");
const path = require("path")
const base = require("./webpack.base.config");
// Similar to VueSSRServerPlugin
// The main function is to package the front-end code into bundle.json and pass the value to the renderer
const VueSSRClientPlugin = require("vue-server-renderer/client-plugin");
function resolve(name) {
return path.resolve(__dirname, "..", name);
}
module.exports = merge(base, {
mode: 'production'.entry: {
app: [resolve("app/entry-client.js")]},plugins: [
// extract vendor chunks for better caching
new VueSSRClientPlugin(),
],
optimization: {
// We need to extract the runtime environment into a separate manifest file
runtimeChunk: {
name: "mainifest"
},
// Equivalent to the previous CommonsChunkPlugin
// Split the node_modules code to form vendors.[hash].js
splitChunks: {
chunks: "async".minSize: 30000.minChunks: 1.maxAsyncRequests: 5.maxInitialRequests: 3.automaticNameDelimiter: "~".name: true.cacheGroups: {
vendor: {
test: /node_modules\/(.*)\.js/.name: "vendors".chunks: "initial".priority: - 10.reuseExistingChunk: false}}}}});Copy the code
Pack the two webpack files respectively and generate the dist folder. The following files appear in the folder respectively:
server/app.js
const statics = require("koa-static");
const bundle = require(".. /dist/vue-ssr-server-bundle.json");
// Add middleware for static files pointing to the dist folder
app.use(statics(path.join(__dirname, ".. /dist")));
const renderer = createBundleRenderer(bundle, {
template: require("fs").readFileSync(resolve(".. /index.html"), "utf-8"),
clientManifest: require(".. /dist/vue-ssr-client-manifest.json")}); router.get("/", ctx => {
renderer.renderToString({}, (err, html) => {
ctx.res.end(html);
});
});
Copy the code
NPM start starts the server, and the rendered index.html file is shown below
<! DOCTYPE html> <html lang="en">
<head>
<title>Hello</title>
<link rel="preload" href="/mainifest.bfe7c0725f559a154244.js" as="script"><link rel="preload" href="/vendors.4617e00e6036ba4dec5e.js" as="script"><link rel="preload" href="/app.645788af3f428d46b800.js" as="script"></head>
<body>
<div id="app" data-server-rendered="true"><span>times: 0</span> <button>-</button> <button>+</button>
<div id="app"><span>times: 0</span> <button>+</button> <button>-</button> <div>312312</div> <title>312312</title></div></div>
<script src="/mainifest.bfe7c0725f559a154244.js" defer></script>
<script src="/vendors.4617e00e6036ba4dec5e.js" defer></script>
<script src="/app.645788af3f428d46b800.js"defer></script> <! This is where the application HTML tags will be injected --> </body> </ HTML >Copy the code
This is just a simple example of using the simplest Vue server rendering and activating it on the client side. Other parts of server rendering such as routing, state management, etc. can be Google by themselves and not covered in this article.
Five GET and POST requests
Now that you’ve built the above project, try calling the Ajax request in the.vue component.
<script>
import axios from "axios";
created() {
axios
.post("http://localhost:3001/user", {
name: "you".age: 100
})
.then(res= > {
console.log("\n [api-get data]");
console.log(res);
this.name = res.data.title;
})
.catch(function(err) {
console.log(err);
});
},
<script>
Copy the code
Next, add the relevant code to the /user route in the server code. At this time, the back end mimics the traditional MVC pattern to transform the server folder
โ โ โ server# server codeโ โ โ controllersThe operation layer performs server-side template rendering, json interface returns data, page jumpโ โ โ the user - info. Js โ โ โ modelsThe data model layer performs data operationsโ โโ โโ โโ exercisesThe routing layer controls routingโ โโ โโ exercisesThe business layer realizes the encapsulation of the data layer model to the operational layer Controllerโ โ โ โ the user - info. JsCopy the code
Locate the router/user.js route
const userRouter = require("koa-router") ();const userController = require(".. /controllers/user-info");
userRouter.get("/", userController.getUserInfo);
module.exports = userRouter;
Copy the code
// controllers/user-info.js
module.exports = {
async getUserInfo(ctx) {
const formData = ctx.request.body;
ctx.body = {
success: false.message: ' '.data: { ...formData },
}
}
};
Copy the code
For POST request processing, the KOA-BodyParser middleware parses formData from the KOA2 context into ctx.request.body
// server/app.js add const bodyParser = require('koa-bodyparser')
app.use(bodyParser())
Copy the code
After starting the server, view the request
Six connection MySQL
1. Install and start MySQL locally, and create table user
CREATE TABLE IF NOT EXISTS `user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) DEFAULT NULL.`age` varchar(255) DEFAULT NULL,
PRIMARY KEY (`id`))ENGINE=InnoDB DEFAULT CHARSET=utf8;
Copy the code
2. Create SQL /index.js to connect to the MySQL database
First test the connection
var mysql = require("mysql");
var connection = mysql.createConnection({
host: "localhost".user: "root"./ / user name
password: "* * * * *"./ / password
database: "node_ssr" // Database name
});
// Resume link
connection.connect();
/ / query
connection.query("SELECT * FROM user".function(error, results, fields) {
if (error) throw error;
console.log(results);
});
// Insert data
connection.query(
"INSERT INTO user(id, name, age) VALUES(0, ? ,?) "["xiatian"."24"].function(err, res) {
if (err) {
console.log("New error:");
console.log(err);
return;
} else {
console.log("Add success:");
console.log(res); }});// Close the connection
connection.end();
Copy the code
After the connection is normal, change it to the following
var mysql = require("mysql");
var pool = mysql.createPool({
host: "localhost".user: "root".password: "* * * *".database: "node_ssr"
});
// Expose the query method for use by the model module
exports.query = (sql, values) = > {
return new Promise((resolve, reject) = > {
pool.getConnection((err, connection) = > {
if (err) {
console.log(err);
reject(err);
}
connection.query(sql, values, (error, results) => {
if (error) {
console.log(error);
reject(error);
} else {
resolve(results);
}
connection.release();
});
});
});
};
Copy the code
Go to the server/model/user-info.js file and import query
const { query } = require(".. /.. /sql/index");
const user = {
// Create a user
async create(body) {
const _sql = "INSERT INTO ?? SET ?";
const result = await query(_sql, ["user", body]);
returnresult; }};module.exports = user
Copy the code
// server/service/user-info.js
const model = require('.. /model/user-info')
const user = {
async create(user) {
// Call the model template and return the result
const result = await model.create(user)
return result
}
}
module.exports = user
Copy the code
// server/controllers/user-info.js
// Request database -> get database return data, and return
const userInfoService = require(".. /service/user-info");
module.exports = {
async SignUp(ctx) {
const formData = ctx.request.body;
let result = {
success: false.message: ' '.data: null
}
let userResult = await userInfoService.create({
id: 0.name: formData.name,
age: formData.age,
});
if ( userResult && userResult.insertId * 1 > 0) {
result.success = true
}
ctx.body = result
}
};
Copy the code
Routing and request methods need to change as well
Start the server again and view the request
conclusion
The above is just the simplest way to try to develop a server-side rendering application. It is just a simple introduction and not difficult, but it is also a good opportunity for me to get out of the SPA application every day and learn a little bit about Node.
reference
Nodejs constructs multi-page server rendering
Koa2 Advanced Study Notes
Take you closer to Vue Server-side Rendering (Vue SSR)