285 lines
8.2 KiB
JavaScript
285 lines
8.2 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
import fs from 'fs';
|
|
import { Command } from 'commander';
|
|
import Table from 'cli-table';
|
|
import chalk from 'chalk';
|
|
import inquirer from 'inquirer';
|
|
import isEmpty from 'lodash/isEmpty';
|
|
import { createStrapi, compileStrapi } from '@strapi/strapi';
|
|
import gitDiff from 'git-diff';
|
|
import tsUtils from '@strapi/typescript-utils';
|
|
|
|
import warnings from './warnings';
|
|
import packageJSON from '../package.json';
|
|
|
|
const program = new Command();
|
|
|
|
const getStrapiApp = async () => {
|
|
process.env.CONFIG_SYNC_CLI = 'true';
|
|
|
|
const appDir = process.cwd();
|
|
const isTSProject = await tsUtils.isUsingTypeScript(appDir);
|
|
const outDir = await tsUtils.resolveOutDir(appDir);
|
|
const alreadyCompiled = await fs.existsSync(outDir);
|
|
|
|
let appContext;
|
|
if (!isTSProject || !alreadyCompiled) {
|
|
appContext = await compileStrapi();
|
|
} else {
|
|
const distDir = isTSProject ? outDir : appDir;
|
|
appContext = { appDir, distDir };
|
|
}
|
|
|
|
const app = await createStrapi(appContext).load();
|
|
return app;
|
|
};
|
|
|
|
const initTable = (head) => {
|
|
return new Table({
|
|
head: [chalk.green('Name'), chalk.green(head || 'State')],
|
|
colWidths: [65, 20],
|
|
chars: { top: '═',
|
|
'top-mid': '╤',
|
|
'top-left': '╔',
|
|
'top-right': '╗',
|
|
bottom: '═',
|
|
'bottom-mid': '╧',
|
|
'bottom-left': '╚',
|
|
'bottom-right': '╝',
|
|
left: '║',
|
|
'left-mid': '╟',
|
|
mid: '─',
|
|
'mid-mid': '┼',
|
|
right: '║',
|
|
'right-mid': '╢',
|
|
middle: '│',
|
|
},
|
|
});
|
|
};
|
|
|
|
const getConfigState = (diff, configName, syncType) => {
|
|
if (
|
|
diff.fileConfig[configName]
|
|
&& diff.databaseConfig[configName]
|
|
) {
|
|
return chalk.yellow(syncType ? 'Update' : 'Different');
|
|
} else if (
|
|
diff.fileConfig[configName]
|
|
&& !diff.databaseConfig[configName]
|
|
) {
|
|
if (syncType === 'import') {
|
|
return chalk.green('Create');
|
|
} else if (syncType === 'export') {
|
|
return chalk.red('Delete');
|
|
} else {
|
|
return chalk.red('Only in sync dir');
|
|
}
|
|
} else if (
|
|
!diff.fileConfig[configName]
|
|
&& diff.databaseConfig[configName]
|
|
) {
|
|
if (syncType === 'import') {
|
|
return chalk.red('Delete');
|
|
} else if (syncType === 'export') {
|
|
return chalk.green('Create');
|
|
} else {
|
|
return chalk.green('Only in DB');
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleAction = async (syncType, skipConfirm, configType, partials, force) => {
|
|
const app = await getStrapiApp();
|
|
const hasSyncDir = fs.existsSync(app.config.get('plugin::config-sync.syncDir'));
|
|
|
|
// No import with empty sync dir.
|
|
if (!hasSyncDir && syncType === 'import') {
|
|
console.log(`${chalk.yellow.bold('[warning]')} You can't import an empty sync directory. Please export before continuing.`);
|
|
process.exit(0);
|
|
}
|
|
|
|
const diff = await app.plugin('config-sync').service('main').getFormattedDiff();
|
|
|
|
// No changes.
|
|
if (isEmpty(diff.diff)) {
|
|
console.log(`${chalk.cyan.bold('[notice]')} There are no changes to ${syncType}.`);
|
|
process.exit(0);
|
|
}
|
|
|
|
// Init table.
|
|
const table = initTable('Action');
|
|
const configNames = partials && partials.split(',');
|
|
const partialDiff = {};
|
|
|
|
// Fill partialDiff with arguments.
|
|
if (configNames) {
|
|
configNames.map((name) => {
|
|
if (diff.diff[name]) partialDiff[name] = diff.diff[name];
|
|
});
|
|
}
|
|
if (configType) {
|
|
Object.keys(diff.diff).map((name) => {
|
|
if (configType === name.split('.')[0]) {
|
|
partialDiff[name] = diff.diff[name];
|
|
}
|
|
});
|
|
}
|
|
|
|
// No changes for partial diff.
|
|
if ((partials || configType) && isEmpty(partialDiff)) {
|
|
console.log(`${chalk.cyan.bold('[notice]')} There are no changes for the specified config.`);
|
|
process.exit(0);
|
|
}
|
|
|
|
const finalDiff = (partials || configType) && partialDiff ? partialDiff : diff.diff;
|
|
|
|
// Add diff to table.
|
|
Object.keys(finalDiff).map((configName) => {
|
|
table.push([configName, getConfigState(diff, configName, syncType)]);
|
|
});
|
|
|
|
// Print table.
|
|
if (hasSyncDir) {
|
|
console.log(table.toString(), '\n');
|
|
}
|
|
|
|
// Prompt to confirm.
|
|
let answer = {};
|
|
if (!skipConfirm) {
|
|
answer = await inquirer.prompt([{
|
|
type: 'confirm',
|
|
name: 'confirm',
|
|
message: `Are you sure you want to ${syncType} the config changes?`,
|
|
}]);
|
|
console.log('');
|
|
}
|
|
|
|
// Preform the action.
|
|
if (skipConfirm || answer.confirm) {
|
|
if (syncType === 'import') {
|
|
const onSuccess = (name) => console.log(`${chalk.cyan.bold('[notice]')} Imported ${name}`);
|
|
try {
|
|
await Promise.all(Object.keys(finalDiff).map(async (name) => {
|
|
let warning;
|
|
if (
|
|
getConfigState(diff, name, syncType) === chalk.red('Delete')
|
|
&& warnings.delete[name]
|
|
) warning = warnings.delete[name];
|
|
|
|
await app.plugin('config-sync').service('main').importSingleConfig(name, onSuccess, force);
|
|
if (warning) console.log(`${chalk.yellow.bold('[warning]')} ${warning}`);
|
|
}));
|
|
console.log(`${chalk.green.bold('[success]')} Finished import`);
|
|
} catch (e) {
|
|
console.log(`${chalk.red.bold('[error]')} ${e}`);
|
|
}
|
|
}
|
|
if (syncType === 'export') {
|
|
const onSuccess = (name) => console.log(`${chalk.cyan.bold('[notice]')} Exported ${name}`);
|
|
|
|
try {
|
|
await Promise.all(Object.keys(finalDiff).map(async (name) => {
|
|
await app.plugin('config-sync').service('main').exportSingleConfig(name, onSuccess);
|
|
}));
|
|
console.log(`${chalk.green.bold('[success]')} Finished export`);
|
|
} catch (e) {
|
|
console.log(`${chalk.red.bold('[error]')} ${e}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
process.exit(0);
|
|
};
|
|
|
|
// Initial program setup
|
|
program.storeOptionsAsProperties(false).allowUnknownOption(true);
|
|
|
|
program.helpOption('-h, --help', 'Display help for command');
|
|
program.addHelpCommand('help [command]', 'Display help for command');
|
|
|
|
// `$ config-sync version` (--version synonym)
|
|
program.version(packageJSON.version, '-v, --version', 'Output the version number');
|
|
program
|
|
.command('version')
|
|
.description('Output your version of the config-sync plugin')
|
|
.action(() => {
|
|
process.stdout.write(`${packageJSON.version}\n`);
|
|
process.exit(0);
|
|
});
|
|
|
|
// `$ config-sync import`
|
|
program
|
|
.command('import')
|
|
.alias('i')
|
|
.option('-t, --type <type>', 'The type of config')
|
|
.option('-p, --partial <partials>', 'A comma separated string of configs')
|
|
.option('-y, --yes', 'Skip the confirm prompt')
|
|
.option('-f, --force', 'Ignore the soft setting')
|
|
.description('Import the config')
|
|
.action(async ({ yes, type, partial, force }) => {
|
|
return handleAction('import', yes, type, partial, force);
|
|
});
|
|
|
|
// `$ config-sync export`
|
|
program
|
|
.command('export')
|
|
.alias('e')
|
|
.option('-t, --type <type>', 'The type of config')
|
|
.option('-p, --partial <partials>', 'A comma separated string of configs')
|
|
.option('-y, --yes', 'Skip the confirm prompt')
|
|
.description('Export the config')
|
|
.action(async ({ yes, type, partial }) => {
|
|
return handleAction('export', yes, type, partial);
|
|
});
|
|
|
|
// `$ config-sync diff`
|
|
program
|
|
.command('diff')
|
|
.alias('d')
|
|
.description('The config diff')
|
|
.action(async (options, { args }) => {
|
|
const single = args[0];
|
|
const app = await getStrapiApp();
|
|
const diff = await app.plugin('config-sync').service('main').getFormattedDiff();
|
|
|
|
// No changes.
|
|
if (isEmpty(diff.diff)) {
|
|
console.log(`${chalk.cyan.bold('[notice]')} No differences between DB and sync directory.`);
|
|
process.exit(0);
|
|
}
|
|
|
|
// Single config diff.
|
|
if (single) {
|
|
// No changes.
|
|
if (!diff.fileConfig[single] && !diff.databaseConfig[single]) {
|
|
console.log(`${chalk.cyan.bold('[notice]')} No differences between DB and sync directory for ${single}.`);
|
|
process.exit(0);
|
|
}
|
|
|
|
// Git diff.
|
|
console.log(gitDiff(
|
|
JSON.stringify(diff.fileConfig[single], null, 2),
|
|
JSON.stringify(diff.databaseConfig[single], null, 2),
|
|
{ color: true },
|
|
));
|
|
|
|
process.exit(1);
|
|
}
|
|
|
|
// Init table.
|
|
const table = initTable();
|
|
|
|
// Add diff to table.
|
|
Object.keys(diff.diff).map((configName) => {
|
|
table.push([configName, getConfigState(diff, configName)]);
|
|
});
|
|
|
|
// Print table.
|
|
console.log(table.toString());
|
|
|
|
process.exit(1);
|
|
});
|
|
|
|
program.parseAsync(process.argv);
|