feat: add config set and npm config
This commit is contained in:
parent
f11810220e
commit
89e4107800
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@kevisual/envision-cli",
|
"name": "@kevisual/envision-cli",
|
||||||
"version": "0.0.4",
|
"version": "0.0.5",
|
||||||
"description": "envision command tools",
|
"description": "envision command tools",
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
@ -2,16 +2,22 @@ import { program as app, Command } from '@/program.ts';
|
|||||||
import { checkFileExists, getConfig, writeConfig } from '@/module/index.ts';
|
import { checkFileExists, getConfig, writeConfig } from '@/module/index.ts';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import fs from 'fs';
|
import fs from 'fs';
|
||||||
|
import { chalk } from '@/module/chalk.ts';
|
||||||
|
import inquirer from 'inquirer';
|
||||||
|
|
||||||
const command = new Command('config')
|
const command = new Command('config')
|
||||||
.description('')
|
.description('')
|
||||||
.option('-d, --dev <dev>', 'Specify dev')
|
.option('-d, --dev <dev>', 'Specify dev')
|
||||||
.option('-l --list', 'list config')
|
.option('-s --set <set>', 'set config')
|
||||||
|
.option('-g --get <get>', 'get config')
|
||||||
|
.option('-r --remove <remove>', 'remove config')
|
||||||
|
.option('-v --value <value>', 'value')
|
||||||
.option('-w --workdir <path>', 'web config')
|
.option('-w --workdir <path>', 'web config')
|
||||||
.action(async (options) => {
|
.action(async (options) => {
|
||||||
const { dev, list, workdir } = options || {};
|
const { dev, list, workdir } = options || {};
|
||||||
let config = getConfig();
|
let config = getConfig();
|
||||||
let flag = false;
|
let flag = false;
|
||||||
|
const execPath = process.cwd();
|
||||||
if (dev === 'true' || dev === 'false') {
|
if (dev === 'true' || dev === 'false') {
|
||||||
flag = true;
|
flag = true;
|
||||||
const config = getConfig();
|
const config = getConfig();
|
||||||
@ -21,9 +27,8 @@ const command = new Command('config')
|
|||||||
config.dev = false;
|
config.dev = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (options.workdir) {
|
if (workdir) {
|
||||||
let finalPath: string;
|
let finalPath: string;
|
||||||
flag = true;
|
|
||||||
const workdir = options.workdir;
|
const workdir = options.workdir;
|
||||||
|
|
||||||
if (workdir.startsWith('/')) {
|
if (workdir.startsWith('/')) {
|
||||||
@ -31,21 +36,118 @@ const command = new Command('config')
|
|||||||
finalPath = workdir;
|
finalPath = workdir;
|
||||||
} else {
|
} else {
|
||||||
// 否则,处理为相对路径
|
// 否则,处理为相对路径
|
||||||
finalPath = path.resolve(workdir);
|
finalPath = path.join(execPath, workdir);
|
||||||
}
|
}
|
||||||
if (!checkFileExists(finalPath)) {
|
if (!checkFileExists(finalPath)) {
|
||||||
console.log('路径不存在');
|
console.log('路径不存在');
|
||||||
fs.mkdirSync(finalPath, { recursive: true });
|
fs.mkdirSync(finalPath, { recursive: true });
|
||||||
}
|
}
|
||||||
config.workdir = finalPath;
|
const answers = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'confirm',
|
||||||
|
name: 'confirm',
|
||||||
|
message: `Are you sure you want to set the workdir to: ${finalPath}?`,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
if (answers.confirm) {
|
||||||
|
flag = true;
|
||||||
|
config.workdir = finalPath;
|
||||||
|
console.log(chalk.green(`set workdir success:`, finalPath));
|
||||||
|
} else {
|
||||||
|
console.log('Cancel set workdir');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (list) {
|
if (options.set) {
|
||||||
const config = getConfig();
|
const key = options.set;
|
||||||
console.log('config', config);
|
let value = options.value;
|
||||||
|
const answer = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'input',
|
||||||
|
name: 'value',
|
||||||
|
message: `Enter your ${key}:(current: ${config[key]})`,
|
||||||
|
when: () => !value,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
value = answer.value || value;
|
||||||
|
if (key && value) {
|
||||||
|
flag = true;
|
||||||
|
config[key] = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (options.remove) {
|
||||||
|
const key = options.remove;
|
||||||
|
if (key) {
|
||||||
|
flag = true;
|
||||||
|
delete config[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flag) {
|
||||||
|
writeConfig(config);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const setCommand = new Command('set')
|
||||||
|
.argument('<key>')
|
||||||
|
.argument('[value]', 'value', 'not_input')
|
||||||
|
.description('set config')
|
||||||
|
.action(async (key, value) => {
|
||||||
|
const config = getConfig();
|
||||||
|
if (!key) {
|
||||||
|
console.log('key is empty');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let flag = false;
|
||||||
|
const answer = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'input',
|
||||||
|
name: 'value',
|
||||||
|
message: `Enter your ${key}:(current: ${config[key]})`,
|
||||||
|
when: () => value === 'not_input',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
value = answer.value || value;
|
||||||
|
if (key && value) {
|
||||||
|
flag = true;
|
||||||
|
console.log(chalk.green(`set ${key} success:`, value));
|
||||||
|
config[key] = value;
|
||||||
}
|
}
|
||||||
if (flag) {
|
if (flag) {
|
||||||
writeConfig(config);
|
writeConfig(config);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
command.addCommand(setCommand);
|
||||||
|
const getCommand = new Command('get')
|
||||||
|
.argument('[key]', 'key')
|
||||||
|
.description('get config')
|
||||||
|
.action(async (key) => {
|
||||||
|
const config = getConfig();
|
||||||
|
const keys = Object.keys(config);
|
||||||
|
const answer = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'input',
|
||||||
|
name: 'key',
|
||||||
|
message: `Enter your key:(keys: ${JSON.stringify(keys)})`,
|
||||||
|
when: () => !key,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
key = answer.key || key;
|
||||||
|
|
||||||
|
if (config[key]) {
|
||||||
|
console.log(chalk.green(`get ${key}:`));
|
||||||
|
console.log(config[key]);
|
||||||
|
} else {
|
||||||
|
console.log(chalk.red(`not found ${key}`));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
command.addCommand(getCommand);
|
||||||
|
|
||||||
|
const list = new Command('list').action(async () => {
|
||||||
|
const config = getConfig();
|
||||||
|
console.log(chalk.green('config', JSON.stringify(config, null, 2)));
|
||||||
|
});
|
||||||
|
command.addCommand(list);
|
||||||
|
|
||||||
app.addCommand(command);
|
app.addCommand(command);
|
||||||
|
214
src/command/npm.ts
Normal file
214
src/command/npm.ts
Normal file
@ -0,0 +1,214 @@
|
|||||||
|
import { program, Command } from '@/program.ts';
|
||||||
|
import { chalk } from '../module/chalk.ts';
|
||||||
|
import path from 'path';
|
||||||
|
import { spawn } from 'child_process';
|
||||||
|
import { fileIsExist } from '@/uitls/file.ts';
|
||||||
|
import { getConfig } from '@/module/get-config.ts';
|
||||||
|
import fs from 'fs';
|
||||||
|
import inquirer from 'inquirer';
|
||||||
|
|
||||||
|
const command = new Command('npm')
|
||||||
|
.description('npm command show publish and set .npmrc')
|
||||||
|
.option('-p --publish <publish>', 'publish')
|
||||||
|
.action(async (options) => {
|
||||||
|
const { publish } = options || {};
|
||||||
|
const config = getConfig();
|
||||||
|
let cmd = '';
|
||||||
|
const execPath = process.cwd();
|
||||||
|
if (publish) {
|
||||||
|
const packageJson = path.resolve(execPath, 'package.json');
|
||||||
|
switch (publish) {
|
||||||
|
case 'me':
|
||||||
|
cmd = 'npm publish --registry https://npm.xiongxiao.me';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
case 'npm':
|
||||||
|
cmd = 'npm publish --registry https://registry.npmjs.org';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
cmd = 'npm publish --registry https://npm.xiongxiao.me';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (fileIsExist(packageJson)) {
|
||||||
|
const keys = Object.keys(config).filter((key) => key.includes('NPM_TOKEN'));
|
||||||
|
const tokenEnv = keys.reduce((prev, key) => {
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
[key]: config[key],
|
||||||
|
};
|
||||||
|
}, {});
|
||||||
|
const child = spawn(cmd, {
|
||||||
|
shell: true,
|
||||||
|
cwd: execPath,
|
||||||
|
env: {
|
||||||
|
...process.env, // 保留当前环境变量
|
||||||
|
...tokenEnv,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
child.stdout.on('data', (data) => {
|
||||||
|
console.log(chalk.green(`${data}`));
|
||||||
|
});
|
||||||
|
child.stderr.on('data', (data) => {
|
||||||
|
// 过滤掉 'npm notice' 或者其他信息
|
||||||
|
if (data.toString().includes('npm notice')) {
|
||||||
|
console.log(chalk.yellow(`notice: ${data}`));
|
||||||
|
} else {
|
||||||
|
console.error(`stderr: ${data}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
child.on('close', (code) => {
|
||||||
|
// console.log(`child process exited with code ${code}`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.error(chalk.red('package.json not found'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const publish = new Command('publish')
|
||||||
|
.argument('[registry]')
|
||||||
|
.description('publish npm')
|
||||||
|
.action(async (registry) => {
|
||||||
|
const answer = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'list',
|
||||||
|
name: 'publish',
|
||||||
|
message: 'Select the registry to publish',
|
||||||
|
choices: [
|
||||||
|
{
|
||||||
|
name: 'me',
|
||||||
|
value: 'me',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'npm',
|
||||||
|
value: 'npm',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
when: !registry,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
registry = registry || answer.publish;
|
||||||
|
const config = getConfig();
|
||||||
|
let cmd = '';
|
||||||
|
const execPath = process.cwd();
|
||||||
|
if (registry) {
|
||||||
|
const packageJson = path.resolve(execPath, 'package.json');
|
||||||
|
switch (registry) {
|
||||||
|
case 'me':
|
||||||
|
cmd = 'npm publish --registry https://npm.xiongxiao.me';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
case 'npm':
|
||||||
|
cmd = 'npm publish --registry https://registry.npmjs.org';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
cmd = 'npm publish --registry https://npm.xiongxiao.me';
|
||||||
|
console.log(chalk.green(cmd));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (fileIsExist(packageJson)) {
|
||||||
|
const keys = Object.keys(config).filter((key) => key.includes('NPM_TOKEN'));
|
||||||
|
const tokenEnv = keys.reduce((prev, key) => {
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
[key]: config[key],
|
||||||
|
};
|
||||||
|
}, {});
|
||||||
|
const child = spawn(cmd, {
|
||||||
|
shell: true,
|
||||||
|
cwd: execPath,
|
||||||
|
env: {
|
||||||
|
...process.env, // 保留当前环境变量
|
||||||
|
...tokenEnv,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
child.stdout.on('data', (data) => {
|
||||||
|
console.log(chalk.green(`${data}`));
|
||||||
|
});
|
||||||
|
child.stderr.on('data', (data) => {
|
||||||
|
// 过滤掉 'npm notice' 或者其他信息
|
||||||
|
if (data.toString().includes('npm notice')) {
|
||||||
|
console.log(chalk.yellow(`notice: ${data}`));
|
||||||
|
} else {
|
||||||
|
console.error(`stderr: ${data}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
child.on('close', (code) => {
|
||||||
|
// console.log(`child process exited with code ${code}`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.error(chalk.red('package.json not found'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
command.addCommand(publish);
|
||||||
|
|
||||||
|
const getnpmrc = new Command('get').action(async () => {
|
||||||
|
const execPath = process.cwd();
|
||||||
|
const npmrcPath = path.resolve(execPath, '.npmrc');
|
||||||
|
if (fileIsExist(npmrcPath)) {
|
||||||
|
const npmrcContent = fs.readFileSync(npmrcPath, 'utf-8');
|
||||||
|
// console.log(chalk.green('get .npmrc success'));
|
||||||
|
console.log(npmrcContent);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
command.addCommand(getnpmrc);
|
||||||
|
|
||||||
|
const npmrc = new Command('set')
|
||||||
|
.description('set .npmrc')
|
||||||
|
.option('-f <force>')
|
||||||
|
.action(async (options) => {
|
||||||
|
const config = getConfig();
|
||||||
|
const npmrcContent =
|
||||||
|
config?.npmrc ||
|
||||||
|
`//npm.xiongxiao.me/:_authToken=\${ME_NPM_TOKEN}
|
||||||
|
@abearxiong:registry=https://npm.pkg.github.com
|
||||||
|
//registry.npmjs.org/:_authToken=\${NPM_TOKEN}
|
||||||
|
@kevisual:registry=https://npm.xiongxiao.me`;
|
||||||
|
const execPath = process.cwd();
|
||||||
|
const npmrcPath = path.resolve(execPath, '.npmrc');
|
||||||
|
let writeFlag = false;
|
||||||
|
if (fileIsExist(npmrcPath)) {
|
||||||
|
if (options.force) {
|
||||||
|
writeFlag = true;
|
||||||
|
} else {
|
||||||
|
const answer = await inquirer.prompt([
|
||||||
|
{
|
||||||
|
type: 'confirm',
|
||||||
|
name: 'confirm',
|
||||||
|
message: `Are you sure you want to overwrite the .npmrc file?`,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
if (answer.confirm) {
|
||||||
|
writeFlag = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
writeFlag = true;
|
||||||
|
}
|
||||||
|
if (writeFlag) {
|
||||||
|
fs.writeFileSync(npmrcPath, npmrcContent);
|
||||||
|
console.log(chalk.green('write .npmrc success'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
command.addCommand(npmrc);
|
||||||
|
|
||||||
|
const remove = new Command('remove').description('remove .npmrc').action(async () => {
|
||||||
|
const execPath = process.cwd();
|
||||||
|
const npmrcPath = path.resolve(execPath, '.npmrc');
|
||||||
|
if (fileIsExist(npmrcPath)) {
|
||||||
|
fs.unlinkSync(npmrcPath);
|
||||||
|
console.log(chalk.green('remove .npmrc success'));
|
||||||
|
} else {
|
||||||
|
console.log(chalk.green('.npmrc success'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
command.addCommand(remove);
|
||||||
|
|
||||||
|
//
|
||||||
|
program.addCommand(command);
|
@ -8,6 +8,7 @@ import './command/serve.ts';
|
|||||||
import './command/config.ts';
|
import './command/config.ts';
|
||||||
import './command/web.ts';
|
import './command/web.ts';
|
||||||
import './command/router.ts';
|
import './command/router.ts';
|
||||||
|
import './command/npm.ts';
|
||||||
|
|
||||||
// program.parse(process.argv);
|
// program.parse(process.argv);
|
||||||
|
|
||||||
|
2
src/module/chalk.ts
Normal file
2
src/module/chalk.ts
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
import { Chalk } from 'chalk';
|
||||||
|
export const chalk = new Chalk({ level: 3 });
|
9
src/uitls/file.ts
Normal file
9
src/uitls/file.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import fs from 'fs';
|
||||||
|
export const fileIsExist = (filePath: string) => {
|
||||||
|
try {
|
||||||
|
fs.accessSync(filePath, fs.constants.F_OK);
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
Loading…
x
Reference in New Issue
Block a user