mirror of
https://github.com/doocs/md.git
synced 2024-11-24 19:10:34 +08:00
5dec24f819
* fix: Update the command line parameter passing method * chore: Upgrade mockm * chore: Only use mockm with md-cli * chore: Update the version number of md-cli * feat: Optimize process management * fix: Optimize space paths
219 lines
6.3 KiB
JavaScript
219 lines
6.3 KiB
JavaScript
const fetch = (...args) => import(`node-fetch`).then(({default: fetch}) => fetch(...args))
|
|
const FormData = require(`form-data`)
|
|
|
|
|
|
/**
|
|
* 判断端口是否可用
|
|
* @param {string|array} port 多个端口用数组
|
|
*/
|
|
function portIsOk (port) {
|
|
if(typeof(port) === `object`) { // 判断多个端口
|
|
return Promise.all(port.map(item => portIsOk(item)))
|
|
}
|
|
return new Promise(resolve => {
|
|
const net = require(`net`)
|
|
const server = net.createServer().listen(port)
|
|
server.on(`listening`, () => server.close(resolve(true)))
|
|
server.on(`error`, () => resolve(port))
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 处理不同系统的命令行空格差异, 在 cp.spawn 中的参数中, 如果包含空格, win 平台需要使用双引号包裹, unix 不需要
|
|
* @param {string} str
|
|
*/
|
|
function handleSpace(str = ``) {
|
|
const newStr = require('os').type() === 'Windows_NT' && str.match(` `) ? `"${str}"` : str
|
|
return newStr
|
|
}
|
|
|
|
/**
|
|
* 自定义控制台颜色
|
|
* https://stackoverflow.com/questions/9781218/how-to-change-node-jss-console-font-color
|
|
* nodejs 内置颜色: https://nodejs.org/api/util.html#util_foreground_colors
|
|
*/
|
|
function colors () {
|
|
const util = require('util')
|
|
|
|
function colorize (color, text) {
|
|
const codes = util.inspect.colors[color]
|
|
return `\x1b[${codes[0]}m${text}\x1b[${codes[1]}m`
|
|
}
|
|
|
|
let returnValue = {}
|
|
Object.keys(util.inspect.colors).forEach((color) => {
|
|
returnValue[color] = (text) => colorize(color, text)
|
|
})
|
|
|
|
const colorTable = new Proxy(returnValue, {
|
|
get (obj, prop) {
|
|
// 在没有对应的具名颜色函数时, 返回空函数作为兼容处理
|
|
const res = obj[prop] ? obj[prop] : (arg => arg)
|
|
return res
|
|
}
|
|
})
|
|
|
|
// 取消下行注释, 查看所有的颜色和名字:
|
|
// Object.keys(returnValue).forEach((color) => console.log(returnValue[color](color)))
|
|
return colorTable
|
|
}
|
|
|
|
/**
|
|
* 以 Promise 方式运行 spawn
|
|
* @param {*} cmd 主程序
|
|
* @param {*} args 程序参数数组
|
|
* @param {*} opts spawn 选项
|
|
*/
|
|
function spawn (cmd, args, opts) {
|
|
opts = { stdio: `inherit`, ...opts }
|
|
opts.shell = opts.shell || process.platform === 'win32'
|
|
return new Promise((resolve, reject) => {
|
|
const cp = require('child_process')
|
|
const child = cp.spawn(cmd, args, opts)
|
|
let stdout = ''
|
|
let stderr = ''
|
|
child.stdout && child.stdout.on('data', d => { stdout += d })
|
|
child.stderr && child.stderr.on('data', d => { stderr += d })
|
|
child.on('error', reject)
|
|
child.on('close', code => {
|
|
resolve({code, stdout, stderr})
|
|
})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 解析命令行参数
|
|
* @param {*} arr
|
|
* @returns
|
|
*/
|
|
function parseArgv(arr) {
|
|
return (arr || process.argv.slice(2)).reduce((acc, arg) => {
|
|
let [k, ...v] = arg.split('=')
|
|
v = v.join(`=`) // 把带有 = 的值合并为字符串
|
|
acc[k] = v === '' // 没有值时, 则表示为 true
|
|
? true
|
|
: (
|
|
/^(true|false)$/.test(v) // 转换指明的 true/false
|
|
? v === 'true'
|
|
: (
|
|
/[\d|.]+/.test(v)
|
|
? (isNaN(Number(v)) ? v : Number(v)) // 如果转换为数字失败, 则使用原始字符
|
|
: v
|
|
)
|
|
)
|
|
return acc
|
|
}, {})
|
|
}
|
|
|
|
function dcloud(spaceInfo) {
|
|
if(Boolean(spaceInfo.spaceId && spaceInfo.clientSecret) === false) {
|
|
throw new Error(`请填写 spaceInfo`)
|
|
}
|
|
|
|
function sign(data, secret) {
|
|
const hmac = require(`crypto`).createHmac(`md5`, secret)
|
|
// 排序 obj 再转换为 key=val&key=val 的格式
|
|
const str = Object.keys(data).sort().reduce((acc, cur) => `${acc}&${cur}=${data[cur]}`, ``).slice(1)
|
|
hmac.update(str)
|
|
return hmac.digest(`hex`)
|
|
}
|
|
|
|
async function anonymousAuthorize() {
|
|
const data = {
|
|
method: `serverless.auth.user.anonymousAuthorize`,
|
|
params: `{}`,
|
|
spaceId: spaceInfo.spaceId,
|
|
timestamp: Date.now(),
|
|
}
|
|
return await fetch(`https://api.bspapp.com/client`, {
|
|
headers: {
|
|
'x-serverless-sign': sign(data, spaceInfo.clientSecret),
|
|
},
|
|
body: `{"method":"serverless.auth.user.anonymousAuthorize","params":"{}","spaceId":"${spaceInfo.spaceId}","timestamp":${data.timestamp}}`,
|
|
method: `POST`,
|
|
}).then((res) => res.json())
|
|
}
|
|
|
|
async function report({ id, token }) {
|
|
const reportReq = {
|
|
method: `serverless.file.resource.report`,
|
|
params: `{"id":"${id}"}`,
|
|
spaceId: spaceInfo.spaceId,
|
|
timestamp: Date.now(),
|
|
token: token,
|
|
}
|
|
return await fetch(`https://api.bspapp.com/client`, {
|
|
headers: {
|
|
'x-basement-token': reportReq.token,
|
|
'x-serverless-sign': sign(reportReq, spaceInfo.clientSecret),
|
|
},
|
|
body: JSON.stringify(reportReq),
|
|
method: `POST`,
|
|
}).then((res) => res.json())
|
|
}
|
|
|
|
async function generateProximalSign({ name, token }) {
|
|
const data = {
|
|
method: `serverless.file.resource.generateProximalSign`,
|
|
params: `{"env":"public","filename":"${name}"}`,
|
|
spaceId: spaceInfo.spaceId,
|
|
timestamp: Date.now(),
|
|
token,
|
|
}
|
|
const res = await fetch(`https://api.bspapp.com/client`, {
|
|
headers: {
|
|
'x-basement-token': data.token,
|
|
'x-serverless-sign': sign(data, spaceInfo.clientSecret),
|
|
},
|
|
body: JSON.stringify(data),
|
|
method: `POST`,
|
|
}).then((res) => res.json())
|
|
return res
|
|
}
|
|
|
|
async function upload({ data, file }) {
|
|
const formdata = new FormData()
|
|
Object.entries({
|
|
'Cache-Control': `max-age=2592000`,
|
|
'Content-Disposition': `attachment`,
|
|
OSSAccessKeyId: data.accessKeyId,
|
|
Signature: data.signature,
|
|
host: data.host,
|
|
id: data.id,
|
|
key: data.ossPath,
|
|
policy: data.policy,
|
|
success_action_status: 200,
|
|
file,
|
|
}).forEach(([key, val]) => formdata.append(key, val))
|
|
|
|
return await fetch(`https://${data.host}`, {
|
|
headers: {
|
|
'X-OSS-server-side-encrpytion': `AES256`,
|
|
},
|
|
body: formdata,
|
|
method: `POST`,
|
|
})
|
|
}
|
|
|
|
async function uploadFile({ name = `unnamed.file`, file }) {
|
|
const token = (await anonymousAuthorize()).data.accessToken
|
|
const res = await generateProximalSign({ name, token })
|
|
await upload({ data: res.data, file })
|
|
await report({ id: res.data.id, token })
|
|
const fileUrl = `https://${res.data.cdnDomain}/${res.data.ossPath}`
|
|
return fileUrl
|
|
}
|
|
|
|
return uploadFile
|
|
|
|
}
|
|
|
|
module.exports = {
|
|
portIsOk,
|
|
handleSpace,
|
|
colors: colors(),
|
|
spawn,
|
|
parseArgv,
|
|
dcloud,
|
|
}
|