commit 3241338b1d409e8bdafd50f2bd97a49087993720 Author: Luk Lu Date: Tue Sep 10 12:06:56 2019 +0800 第一次导入 diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3d3c395 --- /dev/null +++ b/.gitignore @@ -0,0 +1,17 @@ +.DS_Store +node_modules/ +npm-debug.log* +yarn-debug.log* +yarn-error.log* +/test/unit/coverage/ +/test/e2e/reports/ +selenium-debug.log + +# Editor directories and files +.idea +.vscode +*.suo +*.ntvs* +*.njsproj +*.sln +/package-lock.json diff --git a/deploy.js b/deploy.js new file mode 100644 index 0000000..d071980 --- /dev/null +++ b/deploy.js @@ -0,0 +1,123 @@ +const fs = require('fs') +const path = require('path') +const ssh = new (require('node-ssh'))() + +/** ******************* 读取命令行以及配置文件里的参数 ******************** **/ +const commander = require('commander') +const deepmerge = require('deepmerge') + +var Config = {} + +// 读取配置文件 +try { + if (fs.existsSync('./ConfigBasic.js')) { + Config = require('./ConfigBasic.js') + console.info('ConfigBasic loaded') + } + if (fs.existsSync('./ConfigCustom.js')) { // 如果存在,覆盖掉 ConfigBasic 里的默认参数 + Config = deepmerge(Config, require('./ConfigCustom.js')) // 注意,objectMerge后,产生了一个新的对象,而不是在原来的Config里添加 + console.info('ConfigCustom loaded') + } + if (fs.existsSync('./ConfigSecret.js')) { // 如果存在,覆盖掉 ConfigBasic 和 ConfigCustom 里的参数 + Config = deepmerge(Config, require('./ConfigSecret.js')) + console.info('ConfigSecret loaded') + } +} catch (err) { + console.error('Loading config files failed: ' + err.message) +} + +commander + .version('1.0', '-v, --version') // 默认是 -V。如果要 -v,就要加 '-v --version' + .option('-H, --host ', `Host IP or domain name of the target server. Default to ${Config.deploy.host}`) + .option('-P, --port ', `Ssh port number of the target server. Default to ${Config.deploy.port}`) + .option('-r, --root ', `Path to deploy on the target server. Default to ${Config.deploy.root}`) + .option('-d, --dist ', `Folder to deploy on the target server. Default to ${Config.deploy.dist}`) + .option('-u, --user ', `User id to login the target server. Default to ${Config.deploy.user}`) + .option('-k, --key ', `User private key file to login the target server. Default to ${Config.deploy.key}`) + .option('-p, --password ', `User password to login the target server. You may have to enclose it in "". Default to ${Config.deploy.password}`) + .option('-l, --local ', `Local folder to copy from. Default to ${Config.deploy.local}`) + .parse(process.argv) + +const root = commander.root || Config.deploy.root // 本地的项目目录。似乎该目录必须已经存在于服务器上 +console.log(` root = ${root} `) +const dist = commander.dist || Config.deploy.dist || 'dist' // 新系统将发布在这个目录里。建议为dist,和npm run build产生的目录一致,这样既可以远程自动部署,也可以直接登录服务器手动部署。 +console.log(` dist = ${dist} `) +const privateKeyFile = commander.key || Config.deploy.key || `${process.env.HOME}/.ssh/id_rsa` +console.log(` privateKeyFile = ${privateKeyFile} `) +const local = commander.local || Config.deploy.local || 'dist' +console.log(` local = ${local} `) + +const connection = { + host: commander.host || Config.deploy.host, + port: commander.port || Config.deploy.port || 22, + username: commander.user || Config.deploy.user, + privateKey: fs.existsSync(privateKeyFile) ? privateKeyFile : undefined, + password: commander.password || Config.deploy.password, + tryKeyboard: true, + onKeyboardInteractive: (name, instructions, instructionsLang, prompts, finish) => { // 不起作用 + if (prompts.length > 0 && prompts[0].prompt.toLowerCase().includes('password')) { + finish([password]) + } + }, +} +console.log(` connection = ${JSON.stringify(connection)}`) + +/** ********************** 连接到待部署的主机,拷贝文件到指定路径 ************* **/ +function subDirs (path) { + const dirs = [path] + if (fs.statSync(path).isFile()) { + return dirs + } + fs.readdirSync(path).forEach(item => { + const stat = fs.statSync(`${path}/${item}`) + if (stat.isDirectory()) { + dirs.push(...subDirs(`${path}/${item}`)) + } + }) + return dirs +} + +const necessaryPath = (path) => { + return subDirs(path) + .map(it => it.replace(path, '')) + .filter(it => it) + .map(it => it.split('/').filter(it => it)) +} + +ssh.connect(connection).then(async () => { + console.log(`[ mv ${dist} ${dist}-backup-${new Date().toISOString()} ... ]`) + await ssh.execCommand(`mv ${dist} ${dist}-backup-${new Date().toISOString()}`, { cwd: root }) + console.log(`[ mkdir ${dist} ... ]`) + await ssh.execCommand(`mkdir ${dist}`, { cwd: root }) + const toCreate = necessaryPath('./'+local) + for (const name of toCreate) { + console.log(`[ mkdir ${dist}/${name.join('/')} ... ]`) + await ssh.execCommand(`mkdir ${dist}/${name.join('/')}`, { cwd: root }) + } + + let err + console.log(`[ Upload to ${root}/${dist} ... ]`) + await ssh.putDirectory('./'+local, `${root}/${dist}`, { + concurrency: 10, + recursive: true, + validate: itemPath => { + const baseName = path.basename(itemPath) + return !baseName.endsWith('.map') + }, + tick: (localPath, remotePath, error) => { + console.log(`Uploading "${localPath}" ===> "${remotePath}" ${error || 'succeeded!'}`) + err = error + }, + }) + ssh.dispose() + if (err) { + console.log('[ Uploaded with error! ]') + process.exit(1) + } else { + console.log('[ Uploaded successfully! ]') + } +}).catch(err => { + console.error(err) + ssh.dispose() + process.exit(1) +}) diff --git a/package.json b/package.json new file mode 100644 index 0000000..c981bd7 --- /dev/null +++ b/package.json @@ -0,0 +1,20 @@ +{ + "name": "deployer", + "version": "1.0.0", + "description": "Deploy files to SSH or GIT servers", + "main": "deploy.js", + "scripts": { + "test": "echo \"Error: no test specified\" && exit 1" + }, + "repository": { + "type": "git", + "url": "https://git.faronear.org/npm/deployer" + }, + "author": "", + "license": "ISC", + "dependencies": { + "commander": "^3.0.1", + "deepmerge": "^4.0.0", + "node-ssh": "^6.0.0" + } +}