const path = require('path')
|
|
const getDependency = err => err.dependencies && err.dependencies.length && err.dependencies[0]
|
|
const LOCAL_RESOURCE_REGEX = /Module parse failed:\s*(.*)\s*Unexpected character/
|
|
const LINENO_REGEX = /\(([0-9]+):[0-9]+\)/
|
|
const FILE_REGEX = /in\s(.*)\s\(line\s([0-9]+),\scolumn\s[0-9]+\)/
|
|
function formatMessage (msg) {
|
if (msg) {
|
const matches = msg.match(FILE_REGEX)
|
if (matches && matches.length === 3) {
|
const filePath = path.relative(process.env.UNI_INPUT_DIR, path.resolve(matches[1]))
|
msg = msg.replace(matches[0], 'at ' + filePath.split('?')[0] + ':' + matches[2])
|
}
|
return msg.replace('Module build failed: ', '模块编译失败:')
|
}
|
return ''
|
}
|
|
const installPreprocessorTips = {}
|
|
function ModuleBuildError (err) {
|
const lines = err.message.split('\n')
|
let firstLineMessage = lines[0]
|
if (lines.length > 1) {
|
firstLineMessage = lines[1]
|
}
|
if (~firstLineMessage.indexOf('Module build failed: ModuleBuildError: Module build failed:')) {
|
// 移除调用栈错误
|
return false
|
}
|
if (~firstLineMessage.indexOf('Failed to find')) { // css 引用路径错误
|
return {
|
line: 1,
|
message: '文件查找失败:' + firstLineMessage.split('Failed to find')[1]
|
}
|
} else if (~firstLineMessage.indexOf('SyntaxError:') || ~firstLineMessage.indexOf('Syntax Error')) {
|
if (~firstLineMessage.indexOf('ModuleBuildError: Module build failed: Syntax Error')) {
|
return false
|
}
|
|
if (err.error && err.error.loc) { // babel
|
let message = (err.message + '\n').replace('SyntaxError:', '语法错误:')
|
message = message.replace(/^\s*at\s.*:\d+:\d+[\s)]*\n/gm, '') + ' '
|
return {
|
line: err.error.loc.line || 1,
|
message
|
}
|
} else { // css
|
let message = (err.message).replace('Syntax Error', '语法错误:')
|
message = message.replace(/^\s*at\s.*:\d+:\d+[\s)]*\n/gm, '') + ' '
|
const matches = message.match(LINENO_REGEX)
|
if (matches && matches.length === 2) {
|
return {
|
line: matches[1],
|
message
|
}
|
}
|
}
|
} else if (~err.message.indexOf('Cannot find module')) {
|
let builtinCompile = ''
|
let name = ''
|
if (~err.message.indexOf('compile-less')) {
|
name = 'compile-less'
|
builtinCompile = 'less'
|
} else if (~err.message.indexOf('compile-node-sass')) {
|
name = 'compile-node-sass'
|
builtinCompile = 'scss/sass'
|
} else if (~err.message.indexOf('compile-stylus')) {
|
name = 'compile-stylus'
|
builtinCompile = 'stylus'
|
} else if (~err.message.indexOf('compile-typescript')) {
|
name = 'compile-typescript'
|
builtinCompile = 'typescript'
|
} else if (~err.message.indexOf('compile-pug-cli')) {
|
name = 'compile-pug-cli'
|
builtinCompile = 'pug/jade'
|
}
|
if (builtinCompile) {
|
if (installPreprocessorTips[name]) {
|
return false
|
}
|
installPreprocessorTips[name] = true
|
installHBuilderXPlugin(name)
|
return {
|
message: '预编译器错误:代码使用了' + builtinCompile +
|
'语言,但未安装相应的编译器插件,' + (supportAutoInstallPlugin() ? '正在从' : '请前往') +
|
'插件市场安装该插件:\nhttps://ext.dcloud.net.cn/plugin?name=' +
|
name
|
}
|
}
|
} else if (~firstLineMessage.indexOf('Module parse failed')) {
|
const matches = firstLineMessage.match(LOCAL_RESOURCE_REGEX)
|
if (matches && matches.length === 2) {
|
return {
|
message: '资源引用失败:暂不支持引用本地资源\'' + matches[1].trim() + '\',可更换为网络地址或base64'
|
}
|
}
|
}
|
return formatMessage(err.message)
|
}
|
|
function supportAutoInstallPlugin () {
|
// 只要有 HBuilderX 版本号,就一定支持自动安装
|
return !!process.env.HX_Version
|
}
|
|
function installHBuilderXPlugin (lang) {
|
if (supportAutoInstallPlugin()) {
|
return console.error(
|
`%HXRunUniAPPPluginName%${lang}%HXRunUniAPPPluginName%`
|
)
|
}
|
}
|
|
function ModuleNotFoundError (err) {
|
const matches = err.message.match(/Can't resolve '(.*loader)'/)
|
if (matches && matches.length > 0) {
|
return {
|
line: 1,
|
message: `
|
Failed to resolve loader: ${matches[1]}
|
You may need to install it.
|
`
|
}
|
}
|
const dependency = getDependency(err)
|
if (dependency) {
|
let line = 1
|
if (dependency.loc.start && dependency.loc.start.line) {
|
line = dependency.loc.start.line
|
}
|
return {
|
line: line,
|
message: '文件查找失败:\'' + dependency.userRequest + '\''
|
}
|
}
|
}
|
|
function ModuleParseError (err) {
|
const firstLineMessage = err.message.split('\n')[0]
|
const matches = firstLineMessage.match(LOCAL_RESOURCE_REGEX)
|
if (matches && matches.length === 2) {
|
return {
|
message: `资源引用失败:暂不支持引用此类型资源(${err.module.rawRequest})`
|
}
|
}
|
}
|
|
module.exports = {
|
ModuleBuildError,
|
ModuleNotFoundError,
|
ModuleParseError
|
// ChunkRenderError(err) {},
|
// EntryModuleNotFoundError(err) {},
|
// ModuleDependencyError(err) {},
|
// ModuleError(err) {},
|
// ModuleDependencyWarning(warn) {},
|
// ModuleWarning(warn) {}
|
}
|