Files
vue-cli/packages/@vue/cli/lib/Generator.js
2018-01-09 19:49:00 -05:00

91 lines
2.3 KiB
JavaScript

const ejs = require('ejs')
const slash = require('slash')
const debug = require('debug')
const GeneratorAPI = require('./GeneratorAPI')
const sortObject = require('./util/sortObject')
const writeFileTree = require('./util/writeFileTree')
module.exports = class Generator {
constructor (context, pkg, plugins, completeCbs = []) {
this.context = context
this.plugins = plugins
this.pkg = pkg
this.completeCbs = completeCbs
// for conflict resolution
this.depSources = {}
// virtual file tree
this.files = {}
this.fileMiddlewares = []
this.postProcessFilesCbs = []
const cliService = plugins.find(p => p.id === '@vue/cli-service')
const rootOptions = cliService && cliService.options
// apply generators from plugins
plugins.forEach(({ id, apply, options }) => {
const api = new GeneratorAPI(id, this, options, rootOptions || {})
apply(api, options, rootOptions)
})
}
async generate () {
// wait for file resolve
await this.resolveFiles()
// set package.json
this.sortPkg()
this.files['package.json'] = JSON.stringify(this.pkg, null, 2)
// write file tree to disk
await writeFileTree(this.context, this.files)
}
sortPkg () {
// ensure package.json keys has readable order
this.pkg.dependencies = sortObject(this.pkg.dependencies)
this.pkg.devDependencies = sortObject(this.pkg.devDependencies)
this.pkg.scripts = sortObject(this.pkg.scripts, [
'serve',
'build',
'test',
'e2e',
'lint',
'deploy'
])
this.pkg = sortObject(this.pkg, [
'name',
'version',
'private',
'scripts',
'dependencies',
'devDependencies',
'vue',
'babel',
'eslintConfig',
'prettier',
'postcss',
'browserslist',
'jest'
])
debug('vue:cli-pkg')(this.pkg)
}
async resolveFiles () {
const files = this.files
for (const middleware of this.fileMiddlewares) {
await middleware(files, ejs.render)
}
// normalize paths
Object.keys(files).forEach(file => {
const normalized = slash(file)
if (file !== normalized) {
files[normalized] = files[file]
delete files[file]
}
})
for (const postProcess of this.postProcessFilesCbs) {
await postProcess(files)
}
debug('vue:cli-files')(this.files)
}
}