const path = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');
const CopyPlugin = require('copy-webpack-plugin');
const webpack = require('webpack');
const dotenv = require('dotenv');
module.exports = (env, argv) => {
const mode = argv && argv.mode ? argv.mode : 'development';
// Load environment variables from .env.development only in development mode
if (mode === 'development') {
dotenv.config({path: '.env.development'});
}
return {
entry: './src/index.js',
output: {
filename: 'main.[contenthash].js',
path: path.resolve(__dirname, 'build'),
// Use relative URLs so assets load under any prefix (e.g., /admin)
publicPath: '',
// Clean the output directory before emit to avoid stale files (e.g., js/js duplicates)
clean: true
},
devServer: {
static: path.join(__dirname, 'public'),
port: 3000,
open: '/admin/',
hot: true,
historyApiFallback: {
index: '/admin/index.html'
},
devMiddleware: {
publicPath: '/admin'
},
setupMiddlewares: (middlewares, devServer) => {
// Redirect /admin -> /admin/ (only exact match without trailing slash)
devServer.app.use((req, res, next) => {
if (req.path === '/admin' && !req.originalUrl.endsWith('/')) {
return res.redirect(302, '/admin/');
}
next();
});
return middlewares;
}
},
plugins: [
new HtmlWebpackPlugin({
template: path.join(__dirname, 'public', 'index.html')
}),
new CopyPlugin({
patterns: [
{
context: path.resolve(__dirname, 'public'),
from: 'images/*.*',
to: 'images/[name][ext]'
},
{
context: path.resolve(__dirname, 'src/assets'),
from: '*.svg',
to: 'static/[name][ext]'
},
{
context: path.resolve(__dirname, 'src', 'pages', 'AiIntegration', 'css'),
from: '**/*',
to: 'css'
},
{
context: path.resolve(__dirname, 'src', 'pages', 'AiIntegration', 'js'),
from: '**/*',
to: 'js'
},
{
context: path.resolve(__dirname, 'src', 'pages', 'AiIntegration', 'ai'),
from: '**/*',
to: 'ai'
},
{
context: path.resolve(__dirname, '../../../document-templates/sample'),
from: 'sample.docx',
to: 'assets/sample.docx',
noErrorOnMissing: true
}
]
}),
new webpack.DefinePlugin({
'process.env.REACT_APP_BACKEND_URL': JSON.stringify(process.env.REACT_APP_BACKEND_URL)
})
],
module: {
rules: [
{
test: /\.(js)$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
options: {
presets: [['@babel/preset-react', {runtime: 'automatic'}], '@babel/preset-env']
}
}
},
{
test: /\.module\.(css|scss)$/i,
use: [
'style-loader',
{
loader: 'css-loader',
options: {
modules: {
localIdentName: '[local]-[hash:base64:5]'
}
}
},
{
loader: 'sass-loader',
options: {
api: 'modern'
}
}
]
},
{
test: /\.(css|scss)$/i,
exclude: /\.module\.(css|scss)$/i,
use: [
'style-loader',
'css-loader',
{
loader: 'sass-loader',
options: {
api: 'modern'
}
}
]
},
{
test: /\.(png|svg|jpg|jpeg|gif)$/i,
type: 'asset/resource',
generator: {
filename: 'static/[hash][ext]'
}
}
]
},
resolve: {
extensions: ['', '.js'],
alias: {
'@components': path.resolve(__dirname, 'src/components'),
'@screen': path.resolve(__dirname, 'src/screen'),
'@services': path.resolve(__dirname, 'src/services'),
'@store': path.resolve(__dirname, 'src/store'),
'@utility': path.resolve(__dirname, 'src/utility'),
'@assets': path.resolve(__dirname, 'src/assets')
}
}
};
};