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

  1. Create it in the root directoryindex.htmlfile

      
<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
  1. Modify theserver/app.jsTo 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)