0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2025-01-13 21:30:30 -05:00

feat(cli): remove connectors

This commit is contained in:
Gao Sun 2022-10-14 17:01:32 +08:00
parent dcb91428e6
commit 7d257c45bf
No known key found for this signature in database
GPG key ID: 13EBE123E4773688
5 changed files with 103 additions and 26 deletions

View file

@ -5,17 +5,17 @@ import { addConnectors, addOfficialConnectors, inquireInstancePath } from './uti
const add: CommandModule<
{ path?: string },
{ packages: string[]; path?: string; official: boolean }
{ packages?: string[]; path?: string; official: boolean }
> = {
command: ['add [packages...]', 'a', 'install', 'i'],
describe: 'Add specific Logto connectors',
builder: (yargs) =>
yargs
.positional('packages', {
describe: 'The additional connector package names',
describe: 'The connector package names to add',
type: 'string',
array: true,
default: [],
default: undefined,
})
.option('official', {
alias: 'o',
@ -30,9 +30,12 @@ const add: CommandModule<
if (official) {
await addOfficialConnectors(instancePath);
} else {
if (!packageNames?.length) {
log.error('No connector name provided');
}
await addConnectors(instancePath, packageNames);
}
log.info('Restart your Logto instance to get the changes reflected.');
},

View file

@ -3,15 +3,21 @@ import { CommandModule } from 'yargs';
import add from './add';
import list from './list';
import remove from './remove';
const connector: CommandModule = {
command: ['connector', 'c'],
describe: 'Command for Logto connectors',
builder: (yargs) =>
yargs
.option('path', { alias: 'p', type: 'string', describe: 'The path to your Logto instance' })
.option('path', {
alias: 'p',
type: 'string',
describe: 'The path to your Logto instance directory',
})
.command(add)
.command(list)
.command(remove)
.demandCommand(1),
handler: noop,
};

View file

@ -1,15 +1,7 @@
import { readdir } from 'fs/promises';
import path from 'path';
import chalk from 'chalk';
import { CommandModule } from 'yargs';
import {
getConnectorDirectory,
getConnectorPackageName,
inquireInstancePath,
isOfficialConnector,
} from './utils';
import { getConnectorPackagesFrom, isOfficialConnector } from './utils';
const logConnectorNames = (type: string, names: string[]) => {
if (names.length === 0) {
@ -25,14 +17,10 @@ const list: CommandModule<{ path?: string }, { path?: string }> = {
command: ['list', 'l'],
describe: 'List added Logto connectors',
handler: async ({ path: inputPath }) => {
const directory = getConnectorDirectory(await inquireInstancePath(inputPath));
const content = await readdir(directory, 'utf8');
const rawNames = await Promise.all(
content.map(async (value) => getConnectorPackageName(path.join(directory, value)))
);
const names = rawNames.filter((value): value is string => typeof value === 'string');
const officialPackages = names.filter((name) => isOfficialConnector(name));
const thirdPartyPackages = names.filter((name) => !isOfficialConnector(name));
const packages = await getConnectorPackagesFrom(inputPath);
const packageNames = packages.map(({ name }) => name);
const officialPackages = packageNames.filter((name) => isOfficialConnector(name));
const thirdPartyPackages = packageNames.filter((name) => !isOfficialConnector(name));
logConnectorNames('official'.toUpperCase(), officialPackages);
logConnectorNames('3rd-party'.toUpperCase(), thirdPartyPackages);

View file

@ -0,0 +1,59 @@
import chalk from 'chalk';
import fsExtra from 'fs-extra';
import { CommandModule } from 'yargs';
import { log } from '../../utilities';
import { getConnectorPackagesFrom } from './utils';
const remove: CommandModule<{ path?: string }, { path?: string; packages?: string[] }> = {
command: ['remove [packages...]', 'rm', 'delete'],
describe: 'Remove existing Logto connectors',
builder: (yargs) =>
yargs.positional('packages', {
describe: 'The connector package names to remove',
type: 'string',
array: true,
default: undefined,
}),
handler: async ({ path: inputPath, packages: packageNames }) => {
if (!packageNames?.length) {
log.error('No connector name provided');
}
const existingPackages = await getConnectorPackagesFrom(inputPath);
const notFoundPackageNames = packageNames.filter(
(current) => !existingPackages.some(({ name }) => current === name)
);
if (notFoundPackageNames.length > 0) {
log.error(
`Cannot remove ${notFoundPackageNames
.map((name) => chalk.green(name))
.join(', ')}: not found in your Logto instance directory`
);
}
const okSymbol = Symbol('Connector removed');
const result = await Promise.all(
packageNames.map(async (current) => {
const packageInfo = existingPackages.find(({ name }) => name === current);
try {
await fsExtra.remove(packageInfo?.path ?? '');
return okSymbol;
} catch (error: unknown) {
log.warn(`Error while removing ${chalk.green(packageInfo?.name)}`);
log.warn(error);
return error;
}
})
);
const errorCount = result.filter((value) => value !== okSymbol).length;
log.info(`Removed ${result.length - errorCount} connectors`);
},
};
export default remove;

View file

@ -1,6 +1,6 @@
import { exec } from 'child_process';
import { existsSync } from 'fs';
import { readFile, mkdir, unlink } from 'fs/promises';
import { readFile, mkdir, unlink, readdir } from 'fs/promises';
import path from 'path';
import { promisify } from 'util';
@ -86,13 +86,13 @@ export const normalizePackageName = (name: string) =>
)
.join('/');
export const getConnectorDirectory = (instancePath: string) =>
const getConnectorDirectory = (instancePath: string) =>
path.join(instancePath, coreDirectory, connectorDirectory);
export const isOfficialConnector = (packageName: string) =>
packageName.startsWith('@logto/connector-');
export const getConnectorPackageName = async (directory: string) => {
const getConnectorPackageName = async (directory: string) => {
const filePath = path.join(directory, 'package.json');
if (!existsSync(filePath)) {
@ -107,6 +107,27 @@ export const getConnectorPackageName = async (directory: string) => {
}
};
export type ConnectorPackage = {
name: string;
path: string;
};
export const getConnectorPackagesFrom = async (instancePath?: string) => {
const directory = getConnectorDirectory(await inquireInstancePath(instancePath));
const content = await readdir(directory, 'utf8');
const rawPackages = await Promise.all(
content.map(async (value) => {
const currentDirectory = path.join(directory, value);
return { name: await getConnectorPackageName(currentDirectory), path: currentDirectory };
})
);
return rawPackages.filter(
(packageInfo): packageInfo is ConnectorPackage => typeof packageInfo.name === 'string'
);
};
export const addConnectors = async (instancePath: string, packageNames: string[]) => {
const cwd = getConnectorDirectory(instancePath);