feat: 初步实现 vite dev 构建

This commit is contained in:
winixt 2022-04-03 19:26:49 +08:00
parent 35218f0e42
commit 31ff10532e
40 changed files with 331 additions and 354 deletions

View File

@ -24,6 +24,7 @@
"access": "public"
},
"dependencies": {
"@fesjs/utils": "^2.0.4",
"@vitejs/plugin-vue": "^2.2.4",
"@vitejs/plugin-vue-jsx": "^1.3.8",
"postcss-flexbugs-fixes": "^5.0.2",

View File

@ -1,3 +1,11 @@
/**
* outputPath
* inlineLimit: 如何实现
* publicPath: 就是 base
* polyfill @vitejs/plugin-legacy
* 确认 css 最终构建实现 autoprefixer postcss-safe-parser postcss-flexbugs-fixes
*/
export default () => {
// console.log(api, 'TODO: 实现 vite build');
};

View File

@ -0,0 +1,40 @@
import { getInnerCommonConfig } from '../../common/getConfig';
import viteMiddlewarePlugin from './viteMiddlewarePlugin';
export default async (api, args) => {
const { deepmerge, getPort, changePort, getHostName } = api.utils;
const port = await getPort(args.port || api.config.viteOption.server?.port);
changePort(port);
const hostname = getHostName(api.config.viteOption.server?.host);
const { server } = api.config.viteOption;
const beforeMiddlewares = await api.applyPlugins({
key: 'addBeforeMiddlewares',
type: api.ApplyPluginsType.add,
initialValue: [],
args: {},
});
const middlewares = await api.applyPlugins({
key: 'addMiddlewares',
type: api.ApplyPluginsType.add,
initialValue: [],
args: {},
});
return deepmerge(
{
mode: 'development',
plugins: [viteMiddlewarePlugin(beforeMiddlewares, middlewares)],
server: {
...server,
port,
host: hostname,
https: process.env.HTTPS || args.https,
},
},
getInnerCommonConfig(api),
);
};

View File

@ -1,49 +1,21 @@
import { createServer } from 'vite';
import vue from '@vitejs/plugin-vue';
import vueJsx from '@vitejs/plugin-vue-jsx';
import { createHtmlPlugin } from 'vite-plugin-html';
import { join } from 'path';
import SFCConfigBlockPlugin from '../SFCConfigBlockPlugin';
import getDevConfig from './getDevConfig';
/**
* TODO
* 如何处理 html 比较麻烦晚点再看看有无更好的方案
* exportStatic vite 如何针对不同的路径输出不同的 html
*
* 动态更改页面内容
*
* 可以支持的能力
*
* port
* host
* https
* alias
* mountElementId
* outputPath
* inlineLimit: 如何实现
* publicPath: 就是 base
* typescript 的支持tsx 的支持: vite 默认支持
* css modules: vite 默认支持
* 确认 css 最终构建实现 autoprefixer postcss-safe-parser postcss-flexbugs-fixes
* babel-plugin-import 的支持: vite-plugin-babel-import
* define and resolveDefine webpack 一致
* polyfill @vitejs/plugin-legacy
*
* proxy
* createRouteMiddleware 能力
* 确认 mock 能用
* 可以给 server 添加 middlewares
*
*
* analyze: rollup-plugin-visualizer
*
* 其他插件如何对内部配置进行修改
*
* 共享 webpack vite 的部分配置降低熟悉 vite 的成本
*/
export default (api) => {
const {
paths,
utils: { chalk, rimraf, getPort, changePort, getHostName },
utils: { chalk, rimraf },
} = api;
let server;
@ -68,11 +40,6 @@ export default (api) => {
async fn({ args = {} }) {
rimraf.sync(paths.absTmpPath);
const port = await getPort(args.port || api.config.viteOption?.server?.port);
changePort(port);
const hostname = getHostName(api.config.viteOption?.server?.host);
await api.applyPlugins({
key: 'onGenerateFiles',
type: api.ApplyPluginsType.event,
@ -80,41 +47,7 @@ export default (api) => {
api.startWatch();
server = await createServer({
mode: 'development',
define: {
PAGE_TITLE: 'Vite',
},
plugins: [
vue(),
SFCConfigBlockPlugin,
vueJsx(),
createHtmlPlugin({
minify: true,
entry: join(api.paths.absTmpPath, 'fes.js'),
template: 'public/index.html',
inject: {
data: {
title: 'Vite',
},
},
}),
],
configFile: false,
cacheDir: join(api.cwd, '.cache'),
resolve: {
alias: {
'@': paths.absSrcPath,
'@@': paths.absTmpPath,
'@fesInner': '/',
},
},
server: {
port,
host: hostname,
https: process.env.HTTPS || args.https,
},
});
server = await createServer(await getDevConfig(api, args));
await server.listen();
server.printUrls();

View File

@ -0,0 +1,14 @@
export default (beforeMiddlewares = [], afterMiddlewares = []) => ({
name: 'server-middleware-plugin',
configureServer(server) {
beforeMiddlewares.forEach((middleware) => {
server.middlewares.use(middleware);
});
return () => {
afterMiddlewares.forEach((middleware) => {
server.middlewares.use(middleware);
});
};
},
});

View File

@ -1,3 +1,48 @@
export function getConfig() {
return {};
import { join } from 'path';
import vue from '@vitejs/plugin-vue';
import vueJsx from '@vitejs/plugin-vue-jsx';
import { createHtmlPlugin } from 'vite-plugin-html';
import SFCConfigBlockPlugin from './SFCConfigBlockPlugin';
import getDefine from './getDefine';
// TODO
// * 如何处理 html (改 mountId or title 等)(比较麻烦,晚点再看看有无更好的方案)
export function getInnerCommonConfig(api) {
const { deepmerge } = api.utils;
const { server, build, define, base, ...otherViteOption } = api.config.viteOption;
const publicPath = base || api.config.publicPath;
return deepmerge(
{
configFile: false,
define: getDefine(api, publicPath),
cacheDir: join(api.cwd, '.cache'),
plugins: [
vue(api.config.viteVuePlugin || {}),
SFCConfigBlockPlugin,
vueJsx(api.config.viteVueJsx || {}),
createHtmlPlugin({
minify: true,
entry: join(api.paths.absTmpPath, 'fes.js'),
template: 'public/index.html',
inject: {
data: {
title: 'Fes.js',
mountElementId: api.config.mountElementId,
},
},
}),
],
resolve: {
alias: {
'@': api.paths.absSrcPath,
'@@': api.paths.absTmpPath,
[api.builder.innerDepPrefix]: '/',
},
},
},
otherViteOption,
);
}

View File

@ -0,0 +1,16 @@
import { resolveRuntimeEnv, stringifyObjValue } from '@fesjs/utils';
export default (api, publicPath) => {
const viteOption = api.config.viteOption;
const env = resolveRuntimeEnv(publicPath);
const define = stringifyObjValue({
...api.config.define,
...viteOption.define,
});
return {
'process.env': env,
...define,
};
};

View File

@ -0,0 +1,11 @@
export default (api) => {
api.describe({
key: 'viteOption',
config: {
schema(joi) {
return joi.object();
},
default: {},
},
});
};

View File

@ -0,0 +1,11 @@
export default (api) => {
api.describe({
key: 'viteVueJsx',
config: {
schema(joi) {
return joi.object();
},
default: {},
},
});
};

View File

@ -0,0 +1,11 @@
export default (api) => {
api.describe({
key: 'viteVuePlugin',
config: {
schema(joi) {
return joi.object();
},
default: {},
},
});
};

View File

@ -2,6 +2,9 @@ export default function () {
return {
plugins: [
// bundle configs
require.resolve('./features/viteOption'),
require.resolve('./features/viteVueJsx'),
require.resolve('./features/viteVuePlugin'),
// commands
require.resolve('./commands/build'),

10
packages/fes-build-vite/types.d.ts vendored Normal file
View File

@ -0,0 +1,10 @@
import type {UserConfig} from 'vite';
import type {Options} from '@vitejs/plugin-vue'
import createPlugin from '@vitejs/plugin-vue-jsx'
export interface ViteBuildConfig {
viteOption: UserConfig;
viteVuePlugin: Options;
viteVueJsx: Parameters<typeof createPlugin>[0];
}

View File

@ -42,7 +42,6 @@
"babel-plugin-import": "1.13.3",
"cli-highlight": "^2.1.4",
"cliui": "7.0.4",
"connect-history-api-fallback": "^1.6.0",
"copy-webpack-plugin": "^7.0.0",
"css-loader": "^5.0.1",
"css-minimizer-webpack-plugin": "^3.0.0",

View File

@ -1,19 +0,0 @@
import { extname, join } from 'path';
import historyFallback from 'connect-history-api-fallback';
const ASSET_EXT_NAMES = ['.ico', '.png', '.jpg', '.jpeg', '.gif', '.svg'];
export default (api) => (req, res, next) => {
const proxyConfig = api.config.proxy;
if (proxyConfig && Object.keys(proxyConfig).some((path) => req.path.startsWith(path))) {
return next();
}
if (req.path === '/favicon.ico') {
return res.sendFile(join(__dirname, 'fes.png'));
}
if (ASSET_EXT_NAMES.includes(extname(req.path))) {
return next();
}
const history = historyFallback();
history(req, res, next);
};

Binary file not shown.

Before

Width:  |  Height:  |  Size: 50 KiB

View File

@ -36,9 +36,7 @@ export default (api) => {
],
async fn({ args = {} }) {
const { cleanTmpPathExceptCache, getBundleAndConfigs } = require('../buildDevUtils');
const createRouteMiddleware = require('./createRouteMiddleware').default;
console.log(args.port || api.config.devServer?.port);
port = await getPort(args.port || api.config.devServer?.port);
changePort(port);
@ -82,7 +80,7 @@ export default (api) => {
port,
proxy: api.config.proxy,
https: isHTTPS,
beforeMiddlewares: [...beforeMiddlewares, createRouteMiddleware(api)],
beforeMiddlewares: [...beforeMiddlewares],
afterMiddlewares: [...middlewares],
customerDevServerConfig: api.config.devServer,
});

View File

@ -1,6 +1,19 @@
import webpack from 'webpack';
import resolveDefine from './resolveDefine';
import { resolveRuntimeEnv, stringifyObjValue } from '@fesjs/utils';
export default function createDefineWebpackConfig({ config, webpackConfig }) {
webpackConfig.plugin('define').use(webpack.DefinePlugin, [resolveDefine(config)]);
export default function createDefineWebpackConfig({ config, publicPath, webpackConfig }) {
const env = stringifyObjValue(resolveRuntimeEnv(publicPath));
const define = stringifyObjValue({
__VUE_OPTIONS_API__: true,
__VUE_PROD_DEVTOOLS__: false,
...config.define,
});
webpackConfig.plugin('define').use(webpack.DefinePlugin, [
{
'process.env': env,
...define,
},
]);
}

View File

@ -1,14 +1,13 @@
import { join, resolve } from 'path';
import { existsSync } from 'fs';
import { winPath } from '@fesjs/utils';
import resolveDefine from './resolveDefine';
import { winPath, resolveRuntimeEnv } from '@fesjs/utils';
export default async function createHtmlWebpackConfig({ api, cwd, config, webpackConfig, headScripts, isProd }) {
export default async function createHtmlWebpackConfig({ api, cwd, config, webpackConfig, headScripts, isProd, publicPath }) {
const htmlOptions = {
title: 'fes.js',
filename: '[name].html',
...config.html,
templateParameters: resolveDefine(config, true),
templateParameters: resolveRuntimeEnv(publicPath),
mountElementId: config.mountElementId,
};
@ -48,7 +47,7 @@ export default async function createHtmlWebpackConfig({ api, cwd, config, webpac
...config.html,
title: route?.meta?.title || config.html.title || 'fes.js',
filename: _fileName,
templateParameters: resolveDefine(config, true),
templateParameters: resolveRuntimeEnv(publicPath),
mountElementId: config.mountElementId,
};
webpackConfig.plugin(_fileName).use(require.resolve('html-webpack-plugin'), [_htmlOptions]);

View File

@ -214,6 +214,7 @@ export default async function getConfig({ api, cwd, config, env, entry = {}, mod
webpackConfig,
headScripts,
isProd,
publicPath,
});
// --------------- copy -----------
@ -256,6 +257,7 @@ export default async function getConfig({ api, cwd, config, env, entry = {}, mod
// --------------- define -----------
createDefineWebpackConfig({
config,
publicPath,
webpackConfig,
});

View File

@ -1,41 +0,0 @@
const prefixRE = /^FES_APP_/;
const ENV_SHOULD_PASS = ['NODE_ENV', 'FES_ENV'];
export default function resolveDefine(opts = {}, raw) {
const env = {};
Object.keys(process.env).forEach((key) => {
if (prefixRE.test(key) || ENV_SHOULD_PASS.includes(key)) {
env[key] = process.env[key];
}
});
env.BASE_URL = opts.publicPath;
if (raw) {
return env;
}
for (const key in env) {
if (Object.prototype.hasOwnProperty.call(env, key)) {
env[key] = JSON.stringify(env[key]);
}
}
const define = {
__VUE_OPTIONS_API__: true,
__VUE_PROD_DEVTOOLS__: false,
...opts.define,
};
for (const key in define) {
if (Object.prototype.hasOwnProperty.call(define, key)) {
define[key] = JSON.stringify(define[key]);
}
}
return {
'process.env': env,
...define,
};
}

View File

@ -1,7 +1,6 @@
export default function (api) {
[
'addHTMLHeadScripts',
'addMiddlewares',
'modifyBundleConfigOpts',
'modifyBundleConfig',
'modifyBabelOpts',

5
packages/fes-build-webpack/types.d.ts vendored Normal file
View File

@ -0,0 +1,5 @@
export interface WebpackBuildConfig {
}

View File

@ -1,6 +1,5 @@
import { deepmerge, lodash } from '@fesjs/utils';
export default ({ defaultConfig, config }) => {
if (lodash.isPlainObject(defaultConfig) && lodash.isPlainObject(config)) {
return deepmerge(defaultConfig, config);

View File

@ -28,11 +28,14 @@
"@fesjs/compiler": "^2.0.5",
"@fesjs/utils": "^2.0.4",
"body-parser": "^1.19.0",
"connect-history-api-fallback": "^1.6.0",
"cookie": "^0.4.2",
"cookie-parser": "^1.4.5",
"envinfo": "^7.7.3",
"mime": "^3.0.0",
"mockjs": "^1.1.0"
},
"peerDependencies": {
"@vue/compiler-sfc": "^3.0.5"
}
}
}

View File

@ -3,7 +3,28 @@ import { resolve } from 'path';
import { chokidar, lodash, parseRequireDeps } from '@fesjs/utils';
import bodyParser from 'body-parser';
import cookieParser from 'cookie-parser';
import cookie from 'cookie';
import mockjs from 'mockjs';
import mime from 'mime';
function getContentType(type) {
return type.indexOf('/') === -1 ? mime.getType(type) : type;
}
function setCookie(res, name, value, opts = {}) {
const val = typeof value === 'object' ? `j:${JSON.stringify(value)}` : String(value);
if ('maxAge' in opts) {
opts.expires = new Date(Date.now() + opts.maxAge);
opts.maxAge /= 1000;
}
if (opts.path == null) {
opts.path = '/';
}
res.setHeader('Set-Cookie', cookie.serialize(name, String(val), opts));
}
export default (api) => {
let mockFlag = false; // mock 开关flag
@ -132,40 +153,44 @@ export default (api) => {
return (req, res, next) => {
// 如果请求不是以 cgiMock.prefix 开头,直接 next
if (!req.path.startsWith(mockPrefix)) {
if (!req.url.startsWith(mockPrefix)) {
return next();
}
// 请求以 cgiMock.prefix 开头,匹配处理
const matchRequet = requestList.find((item) => req.path.search(item.url) !== -1);
const matchRequet = requestList.find((item) => req.url.search(item.url) !== -1);
if (!matchRequet) {
return next();
}
const sendData = () => {
// set header
res.set(matchRequet.headers);
// set Content-Type
matchRequet.type && res.type(matchRequet.type);
if (matchRequet.headers) {
for (const [key, value] of Object.entries(matchRequet.headers)) {
res.setHeader(key, value);
}
}
if (matchRequet.type) {
res.setHeader('Content-Type', getContentType(matchRequet.type));
}
// set status code
res.status(matchRequet.statusCode);
res.statusCode = matchRequet.statusCode;
// set cookie
traversalHandler(matchRequet.cookies, (item) => {
const name = item.name;
const value = item.value;
delete item.name;
delete item.value;
res.cookie(name, value, item);
setCookie(res, name, value, item);
});
// do result
if (lodash.isFunction(matchRequet.result)) {
matchRequet.result(req, res);
} else if (lodash.isArray(matchRequet.result) || lodash.isPlainObject(matchRequet.result)) {
!matchRequet.type && res.type('json');
res.json(matchRequet.result);
!matchRequet.type && res.setHeader('Content-Type', getContentType('json'));
res.end(JSON.stringify(matchRequet.result));
} else {
!matchRequet.type && res.type('text');
res.send(matchRequet.result.toString());
!matchRequet.type && res.setHeader('Content-Type', getContentType('text'));
res.end(matchRequet.result.toString());
}
};

View File

@ -1,3 +1,21 @@
import { extname } from 'path';
import historyFallback from 'connect-history-api-fallback';
const ASSET_EXT_NAMES = ['.ico', '.png', '.jpg', '.jpeg', '.gif', '.svg'];
const proxyMiddleware = (api) => (req, res, next) => {
const proxyConfig = api.config.proxy;
if (proxyConfig && Object.keys(proxyConfig).some((path) => req.url.startsWith(path))) {
return next();
}
if (ASSET_EXT_NAMES.includes(extname(req.url))) {
return next();
}
const history = historyFallback();
history(req, res, next);
};
export default (api) => {
api.describe({
key: 'proxy',
@ -10,4 +28,6 @@ export default (api) => {
},
},
});
api.addBeforeMiddlewares(() => proxyMiddleware(api));
};

View File

@ -22,6 +22,7 @@ export default function (api) {
'addTmpGenerateWatcherPaths',
'addBeforeMiddlewares',
'addMiddlewares',
].forEach((name) => {
api.registerMethod({ name });
});

View File

@ -3,9 +3,11 @@ import pxtoviewport from '@ttou/postcss-px-to-viewport';
import { defineBuildConfig } from '@fesjs/fes';
export default defineBuildConfig({
define: {
// __VUE_OPTIONS_API__: true,
// __VUE_PROD_DEVTOOLS__: false
proxy: {
'/v2': {
'target': 'https://api.douban.com/',
'changeOrigin': true,
}
},
publicPath: '/',
request: {

View File

@ -26,31 +26,11 @@ export default {
const clickIcon = () => {
console.log('click Icon');
};
console.log(process.env.NODE_ENV, process.env.HELLO);
console.log(useRoute());
// request('/api', null, {
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// mergeRequest: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// mergeRequest: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
request('/v2/movie/in_theaters_proxy', (res) => {
console.log(res);
});
const get = (id) => {
request(
@ -74,61 +54,9 @@ export default {
});
};
get(1);
// get(2);
// get(3);
// get(1);
// post(3);
// post(1);
// post(2);
post(3);
// setTimeout(() => {
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// }, 1000);
// setTimeout(() => {
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// }, 3200);
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// // skipErrorHandler: [500]
// }).then((res) => {
// console.log(res);
// }).catch((err) => {
// console.log('inner error', err);
// });
return {
fes,
rotate,

View File

@ -3,6 +3,12 @@ import pxtoviewport from '@ttou/postcss-px-to-viewport';
import { defineBuildConfig } from '@fesjs/fes';
export default defineBuildConfig({
proxy: {
'/v2': {
'target': 'https://api.douban.com/',
'changeOrigin': true,
}
},
request: {
dataField: 'result'
},

View File

@ -1,25 +0,0 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
<meta name="format-detection" content="telephone=no" />
<meta name="format-detection" content="email=no" />
<meta name="viewport"
content="viewport-fit=cover,width=device-width,initial-scale=1,maximum-scale=1,minimum-scale=1,user-scalable=no">
<title>
<%= htmlWebpackPlugin.options.title %>
</title>
<link rel="shortcut icon" type="image/x-icon" href="./logo.png">
</head>
<body ontouchstart="">
<div id="app"></div>
<script type="module" src="/src/.fes/fes.js"></script>
<!-- built files will be auto injected -->
</body>
</html>

View File

@ -11,13 +11,13 @@
<meta name="viewport"
content="viewport-fit=cover,width=device-width,initial-scale=1,maximum-scale=1,minimum-scale=1,user-scalable=no">
<title>
<%= PAGE_TITLE %>
<%= title %>
</title>
<link rel="shortcut icon" type="image/x-icon" href="./logo.png">
</head>
<body ontouchstart="">
<div id="app"></div>
<div id="<%= mountElementId %>"></div>
<script type="module" src="/src/.fes/fes.js"></script>
<!-- built files will be auto injected -->
</body>

View File

@ -11,8 +11,6 @@ export default defineRuntimeConfig({
},
default(error) {
console.log(error);
const msg = error?.data?.msg || error?.msg;
console.log(msg);
},
},
},

View File

@ -30,83 +30,10 @@ export default {
console.log('click Icon');
};
const get = (id) => {
request(
'/get/api',
{ id },
{
method: 'get',
},
);
};
request('/v2/movie/in_theaters_proxy', (res) => {
console.log(res);
});
const post = (id) => {
request(
'/api',
{ id },
{
responseType: 'blob',
},
).then((data) => {
console.log(data);
});
};
get(1);
// get(2);
// get(3);
// post(1);
// post(2);
post(3);
// setTimeout(() => {
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// }, 1000);
// setTimeout(() => {
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// throttle: 3000,
// cache: true
// }).then((res) => {
// console.log(res);
// });
// }, 3200);
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// cache: true
// }).then((res) => {
// console.log(res);
// });
// request('/api', null, {
// // skipErrorHandler: [500]
// }).then((res) => {
// console.log(res);
// }).catch((err) => {
// console.log('inner error', err);
// });
return {
fes,
rotate,

View File

@ -1,7 +1,5 @@
<template>
<div>
mock and proxy
</div>
<div>mock and proxy</div>
</template>
<config>
{
@ -37,8 +35,8 @@ request(
{ a: 1 },
{
method: 'get',
headers: { Accept: '*/*' }
}
headers: { Accept: '*/*' },
},
)
.then((resp) => {
console.log(resp);

View File

@ -29,6 +29,8 @@ import getAppPath from './getAppPath';
import getPort from './getPort';
import changePort from './changePort';
import getHostName from './getHostName';
import resolveRuntimeEnv from './resolveRuntimeEnv';
import stringifyObjValue from './stringifyObjValue';
export {
chalk,
@ -64,4 +66,6 @@ export {
getPort,
changePort,
getHostName,
resolveRuntimeEnv,
stringifyObjValue,
};

View File

@ -0,0 +1,16 @@
const prefixRE = /^FES_APP_/;
const ENV_SHOULD_PASS = ['NODE_ENV', 'FES_ENV'];
export default (publicPath) => {
const env = {};
Object.keys(process.env).forEach((key) => {
if (prefixRE.test(key) || ENV_SHOULD_PASS.includes(key)) {
env[key] = process.env[key];
}
});
env.BASE_URL = publicPath;
return env;
};

View File

@ -0,0 +1,12 @@
import lodash from 'lodash';
export default (obj) => {
const newObj = lodash.cloneDeep(obj);
for (const key in newObj) {
if (Object.prototype.hasOwnProperty.call(newObj, key)) {
newObj[key] = JSON.stringify(newObj[key]);
}
}
return newObj;
};

View File

@ -5057,7 +5057,7 @@ cookie@0.4.1:
resolved "https://registry.npmmirror.com/cookie/-/cookie-0.4.1.tgz#afd713fe26ebd21ba95ceb61f9a8116e50a537d1"
integrity sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA==
cookie@0.4.2:
cookie@0.4.2, cookie@^0.4.2:
version "0.4.2"
resolved "https://registry.npmmirror.com/cookie/-/cookie-0.4.2.tgz#0e41f24de5ecf317947c82fc789e06a884824432"
integrity sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA==
@ -9401,6 +9401,11 @@ mime@^2.4.4:
resolved "https://registry.npmmirror.com/mime/-/mime-2.6.0.tgz#a2a682a95cd4d0cb1d6257e28f83da7e35800367"
integrity sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==
mime@^3.0.0:
version "3.0.0"
resolved "https://registry.npmmirror.com/mime/-/mime-3.0.0.tgz#b374550dca3a0c18443b0c950a6a58f1931cf7a7"
integrity sha512-jSCU7/VB1loIWBZe14aEYHU/+1UMEHoaO7qxCOVJOw9GgH72VAWppxNcjU+x9a2k3GSIBXNKxXQFqRvvZ7vr3A==
mimic-fn@^1.0.0:
version "1.2.0"
resolved "https://registry.npmmirror.com/mimic-fn/-/mimic-fn-1.2.0.tgz#820c86a39334640e99516928bd03fca88057d022"