# Конфигурация сервера

Для структуры кода и конфигурации webpack, описанных ранее, также потребуются и некоторые изменения в коде сервера Express.

  • необходимо создать приложение с собранным entry-server.js из итоговой сборки. Путь к нему можно найти с помощью манифеста webpack:

    // server.js
    const path = require('path')
    const manifest = require('./dist/server/ssr-manifest.json')
    
    // Имя 'app.js' получается из имени точки входа с добавленным суффиксом `.js`
    const appPath = path.join(__dirname, './dist', 'server', manifest['app.js'])
    const createApp = require(appPath).default
    
    1
    2
    3
    4
    5
    6
    7
  • требуется определить правильные пути ко всем используемым ресурсам:

    // server.js
    server.use(
      '/img',
      express.static(path.join(__dirname, './dist/client', 'img'))
    )
    server.use(
      '/js',
      express.static(path.join(__dirname, './dist/client', 'js'))
    )
    server.use(
      '/css',
      express.static(path.join(__dirname, './dist/client', 'css'))
    )
    server.use(
      '/favicon.ico',
      express.static(path.join(__dirname, './dist/client', 'favicon.ico'))
    )
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
  • нужно заменять содержимое index.html на содержимое приложения с отрисовкой на стороне сервера:

    // server.js
    const indexTemplate = fs.readFileSync(
      path.join(__dirname, '/dist/client/index.html'),
      'utf-8'
    )
    
    server.get('*', async (req, res) => {
      const { app } = createApp()
    
      const appContent = await renderToString(app)
    
      const html = indexTemplate
        .toString()
        .replace('<div id="app">', `<div id="app">${appContent}`)
    
      res.setHeader('Content-Type', 'text/html')
      res.send(html)
    })
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18

Полный код сервера Express представлен ниже:

const path = require('path')
const express = require('express')
const fs = require('fs')
const { renderToString } = require('@vue/server-renderer')
const manifest = require('./dist/server/ssr-manifest.json')

const server = express()

const appPath = path.join(__dirname, './dist', 'server', manifest['app.js'])
const createApp = require(appPath).default

server.use('/img', express.static(path.join(__dirname, './dist/client', 'img')))
server.use('/js', express.static(path.join(__dirname, './dist/client', 'js')))
server.use('/css', express.static(path.join(__dirname, './dist/client', 'css')))
server.use('/favicon.ico', express.static(path.join(__dirname, './dist/client', 'favicon.ico')))

server.get('*', async (req, res) => {
  const { app } = createApp()

  const appContent = await renderToString(app)

  fs.readFile(path.join(__dirname, '/dist/client/index.html'), (err, html) => {
    if (err) {
      throw err
    }

    html = html
      .toString()
      .replace('<div id="app">', `<div id="app">${appContent}`)
    res.setHeader('Content-Type', 'text/html')
    res.send(html)
  })
})

console.log('Сервер запущен по адресу: http://localhost:8080')

server.listen(8080)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37