Рендеринг на стороне сервера с Webpack и Express во время выполнения
Можно ли динамически отображать файлы шаблонов (например, Pug или Handlebars) во время выполнения, используя Webpack и Express?
Моя проблема при загрузке моей корневой страницы (index.pug
), загружается html, но никакие активы не загружаются.
Пример:
app
.set('views', path.join(__dirname, 'views'))
.set('view engine', 'pug')
.use('/', function(req, res) {
res.render('index', {some: 'param'})
})
Если я удалю '/'
обработчик маршрута, страница загружается со всеми активами просто отлично.
клиент webpack.config.js
файл:
module.exports = {
entry: {
main: ['webpack-hot-middleware/client?path=/__webpack_hmr&timeout=60000', './index.js', './css/main.css']
},
output: {
path: path.join(__dirname, 'dist'),
publicPath: '/',
filename: '[name].js'
},
mode: 'development',
target: 'web',
devtool: '#source-map',
module: {
rules: [
{
test: /\.pug$/,
use: ['html-loader?attrs=false', 'pug-html-loader']
}
]
},
plugins: [
new HtmlWebPackPlugin({
template: './views/index.pug',
filename: "./index.html",
excludeChunks: [ 'app' ]
})
]
}
сервер webpack.server.config.js
:
module.exports = (env, argv) => {
return ({
entry: {
app: 'app.js',
},
output: {
path: path.join(__dirname, 'dist'),
publicPath: '/',
filename: '[name].js'
},
target: 'node',
node: {
// Need this when working with express, otherwise the build fails
__dirname: false, // if you don't put this is, __dirname
__filename: false, // and __filename return blank or /
},
externals: [nodeExternals()], // Need this to avoid error when working with Express
module: {
rules: [
{
// Transpiles ES6-8 into ES5
test: /\.js$/,
exclude: /node_modules/,
use: {
loader: "babel-loader"
}
}
]
}
})
}
1 ответ
Вы можете заставить его работать в два простых шага (на основе кода выше).
- На сервере измените путь просмотра туда, где будет ваш созданный шаблон. В приведенном выше примере он будет внутри папки dist.
app
.set("views", "./dist");
.set('view engine', 'pug')
.use('/', function(req, res) {
res.render('index', {some: 'param'})
})
На этом этапе, после того, как мы запустим сценарий сборки и запустим сервер, мы получим сообщение о том, что в папке dist нет представления, что правильно, потому что движок ищет шаблон, а все, что у нас есть, это файлы html.
- В webpack.config.js в HtmlWebPackPlugin нам нужно оставить имя файла в качестве шаблона
plugins: [
new HtmlWebPackPlugin({
template: './views/index.pug',
filename: "./index.pug",
excludeChunks: [ 'app' ]
})
]
И это все. Теперь шаблонизатор найдет index.pug в dist с внедренным стилем и скриптом и сгенерирует из него html.
Примечание: в производстве важно установить путь для статических файлов, иначе вы не сможете отобразить свой css, даже если теги ссылок были введены правильно.
app.use(express.static(__dirname));
Приведенное выше решение сработало для меня, когда я столкнулся с той же проблемой, но с шаблоном ejs. Я нигде не мог найти на него ответа, так что, надеюсь, это избавит кого-то от многих часов разочарования.