0
Fork 0
mirror of https://github.com/logto-io/logto.git synced 2024-12-30 20:33:54 -05:00

refactor(core): remove connector commands

This commit is contained in:
Gao Sun 2022-10-11 18:10:57 +08:00
parent 4ccbe4ac65
commit 64a0936a77
No known key found for this signature in database
GPG key ID: 13EBE123E4773688
11 changed files with 12 additions and 195 deletions

View file

@ -12,6 +12,7 @@ import pRetry from 'p-retry';
import tar from 'tar'; import tar from 'tar';
import { z } from 'zod'; import { z } from 'zod';
import { connectorDirectory } from '../../constants';
import { log } from '../../utilities'; import { log } from '../../utilities';
import { defaultPath } from '../install/utils'; import { defaultPath } from '../install/utils';
@ -86,7 +87,7 @@ export const normalizePackageName = (name: string) =>
.join('/'); .join('/');
export const addConnectors = async (instancePath: string, packageNames: string[]) => { export const addConnectors = async (instancePath: string, packageNames: string[]) => {
const cwd = path.join(instancePath, coreDirectory, 'connectors'); const cwd = path.join(instancePath, coreDirectory, connectorDirectory);
if (!existsSync(cwd)) { if (!existsSync(cwd)) {
await mkdir(cwd); await mkdir(cwd);

View file

@ -0,0 +1 @@
export const connectorDirectory = 'connectors';

View file

@ -14,8 +14,8 @@
"lint:report": "pnpm lint --format json --output-file report.json", "lint:report": "pnpm lint --format json --output-file report.json",
"dev": "rm -rf build/ && pnpm run copyfiles && nodemon", "dev": "rm -rf build/ && pnpm run copyfiles && nodemon",
"start": "NODE_ENV=production node build/index.js", "start": "NODE_ENV=production node build/index.js",
"add-connector": "node build/cli/add-connector.js", "add-connector": "logto connector add -p ../../",
"add-official-connectors": "node build/cli/add-official-connectors.js", "add-official-connectors": "logto connector add -o -p ../../",
"alteration": "logto db alt", "alteration": "logto db alt",
"cli": "logto", "cli": "logto",
"test": "jest", "test": "jest",
@ -45,7 +45,6 @@
"hash-wasm": "^4.9.0", "hash-wasm": "^4.9.0",
"i18next": "^21.8.16", "i18next": "^21.8.16",
"iconv-lite": "0.6.3", "iconv-lite": "0.6.3",
"inquirer": "^8.2.2",
"jose": "^4.0.0", "jose": "^4.0.0",
"js-yaml": "^4.1.0", "js-yaml": "^4.1.0",
"koa": "^2.13.1", "koa": "^2.13.1",
@ -62,14 +61,12 @@
"oidc-provider": "^7.11.3", "oidc-provider": "^7.11.3",
"p-retry": "^4.6.1", "p-retry": "^4.6.1",
"query-string": "^7.0.1", "query-string": "^7.0.1",
"rimraf": "^3.0.2",
"roarr": "^7.11.0", "roarr": "^7.11.0",
"slonik": "^30.0.0", "slonik": "^30.0.0",
"slonik-interceptor-preset": "^1.2.10", "slonik-interceptor-preset": "^1.2.10",
"slonik-sql-tag-raw": "^1.1.4", "slonik-sql-tag-raw": "^1.1.4",
"snake-case": "^3.0.4", "snake-case": "^3.0.4",
"snakecase-keys": "^5.1.0", "snakecase-keys": "^5.1.0",
"tar": "^6.1.11",
"zod": "^3.18.0" "zod": "^3.18.0"
}, },
"devDependencies": { "devDependencies": {
@ -81,7 +78,6 @@
"@types/etag": "^1.8.1", "@types/etag": "^1.8.1",
"@types/fs-extra": "^9.0.13", "@types/fs-extra": "^9.0.13",
"@types/http-errors": "^1.8.2", "@types/http-errors": "^1.8.2",
"@types/inquirer": "^8.2.1",
"@types/jest": "^28.1.6", "@types/jest": "^28.1.6",
"@types/js-yaml": "^4.0.5", "@types/js-yaml": "^4.0.5",
"@types/koa": "^2.13.3", "@types/koa": "^2.13.3",
@ -92,9 +88,7 @@
"@types/lodash.pick": "^4.4.6", "@types/lodash.pick": "^4.4.6",
"@types/node": "^16.3.1", "@types/node": "^16.3.1",
"@types/oidc-provider": "^7.11.1", "@types/oidc-provider": "^7.11.1",
"@types/rimraf": "^3.0.2",
"@types/supertest": "^2.0.11", "@types/supertest": "^2.0.11",
"@types/tar": "^6.1.2",
"copyfiles": "^2.4.1", "copyfiles": "^2.4.1",
"eslint": "^8.21.0", "eslint": "^8.21.0",
"http-errors": "^1.6.3", "http-errors": "^1.6.3",

View file

@ -1,24 +0,0 @@
import 'module-alias/register';
import { getEnv } from '@silverhand/essentials';
import chalk from 'chalk';
import { addConnector } from '@/connectors/add-connectors';
import { defaultConnectorDirectory } from '@/env-set';
import { configDotEnv } from '@/env-set/dot-env';
configDotEnv();
const addConnectorCli = async (packageName: string) => {
const connectorDirectory = getEnv('CONNECTOR_DIRECTORY', defaultConnectorDirectory);
await addConnector(packageName, connectorDirectory);
console.log(`${chalk.blue(packageName)} added successfully.`);
};
const packageName = process.argv[2];
if (!packageName) {
throw new Error('Please provide a package name');
}
void addConnectorCli(packageName);

View file

@ -1,15 +0,0 @@
import 'module-alias/register';
import { getEnv } from '@silverhand/essentials';
import { addOfficialConnectors } from '@/connectors/add-connectors';
import { defaultConnectorDirectory } from '@/env-set';
import { configDotEnv } from '@/env-set/dot-env';
configDotEnv();
const addOfficialConnectorsCli = async () => {
const connectorDirectory = getEnv('CONNECTOR_DIRECTORY', defaultConnectorDirectory);
await addOfficialConnectors(connectorDirectory);
};
void addOfficialConnectorsCli();

View file

@ -1,67 +0,0 @@
import { exec } from 'child_process';
import { existsSync } from 'fs';
import { mkdir, rename, unlink } from 'fs/promises';
import path from 'path';
import { promisify } from 'util';
import chalk from 'chalk';
import rimraf from 'rimraf';
import tar from 'tar';
import { z } from 'zod';
import { npmPackResultGuard } from './types';
const execPromise = promisify(exec);
const npmConnectorFilter = '@logto/connector-';
const excludedPackages = new Set([
'@logto/connector-kit',
'@logto/connector-mock-sms',
'@logto/connector-mock-social',
'@logto/connector-mock-email',
]);
const fetchOfficialConnectorList = async () => {
const { stdout } = await execPromise(`npm search ${npmConnectorFilter} --json`);
const packages = z.object({ name: z.string() }).array().parse(JSON.parse(stdout));
return packages.filter(({ name }) => !excludedPackages.has(name));
};
export const addConnector = async (packageName: string, cwd: string) => {
if (!existsSync(cwd)) {
await mkdir(cwd);
}
const { stdout } = await execPromise(`npm pack ${packageName} --json`, { cwd });
const result = npmPackResultGuard.parse(JSON.parse(stdout));
if (!result[0]) {
throw new Error(`Failed to download package: ${packageName}`);
}
const { filename, name } = result[0];
const escapedFilename = filename.replace(/\//g, '-').replace(/@/g, '');
const filePath = path.join(cwd, escapedFilename);
await tar.extract({ cwd, file: filePath });
await unlink(filePath);
const packageFolder = path.join(cwd, name.replace(/\//g, '-').replace(/@/g, ''));
await promisify(rimraf)(packageFolder);
await rename(path.join(cwd, 'package'), packageFolder);
};
export const addOfficialConnectors = async (directory: string) => {
console.log(`${chalk.blue('[add-connectors]')} Fetching official connectors list`);
const packages = await fetchOfficialConnectorList();
// The await inside the loop is intended for better debugging experience and rate limitation.
for (const [index, { name }] of packages.entries()) {
console.log(
`${chalk.blue('[add-connectors]')} ${index + 1}/${
packages.length
} Adding connector package: ${name}`
);
// eslint-disable-next-line no-await-in-loop
await addConnector(name, directory);
}
};

View file

@ -2,10 +2,10 @@ import { existsSync } from 'fs';
import { readdir } from 'fs/promises'; import { readdir } from 'fs/promises';
import path from 'path'; import path from 'path';
import { connectorDirectory } from '@logto/cli/lib/constants';
import { AllConnector, CreateConnector, validateConfig } from '@logto/connector-kit'; import { AllConnector, CreateConnector, validateConfig } from '@logto/connector-kit';
import chalk from 'chalk'; import chalk from 'chalk';
import envSet from '@/env-set';
import RequestError from '@/errors/RequestError'; import RequestError from '@/errors/RequestError';
import { findAllConnectors, insertConnector } from '@/queries/connector'; import { findAllConnectors, insertConnector } from '@/queries/connector';
@ -21,10 +21,6 @@ const loadConnectors = async () => {
return cachedConnectors; return cachedConnectors;
} }
const {
values: { connectorDirectory },
} = envSet;
if (!existsSync(connectorDirectory)) { if (!existsSync(connectorDirectory)) {
return []; return [];
} }

View file

@ -29,11 +29,3 @@ export type LoadConnector<T extends AllConnector = AllConnector> = T & {
export type LogtoConnector<T extends AllConnector = AllConnector> = LoadConnector<T> & { export type LogtoConnector<T extends AllConnector = AllConnector> = LoadConnector<T> & {
dbEntry: Connector; dbEntry: Connector;
}; };
export const npmPackResultGuard = z
.object({
name: z.string(),
version: z.string(),
filename: z.string(),
})
.array();

View file

@ -1,30 +0,0 @@
import { existsSync } from 'fs';
import { mkdir } from 'fs/promises';
import inquirer from 'inquirer';
import { addOfficialConnectors } from '@/connectors/add-connectors';
import { allYes } from './parameters';
export const addConnectors = async (directory: string) => {
if (existsSync(directory)) {
return;
}
if (!allYes) {
const add = await inquirer.prompt({
type: 'confirm',
name: 'value',
message: `Would you like to add built-in connectors?`,
});
if (!add.value) {
await mkdir(directory);
return;
}
}
await addOfficialConnectors(directory);
};

View file

@ -1,12 +1,9 @@
import path from 'path';
import { getEnv, getEnvAsStringArray, Optional } from '@silverhand/essentials'; import { getEnv, getEnvAsStringArray, Optional } from '@silverhand/essentials';
import { DatabasePool } from 'slonik'; import { DatabasePool } from 'slonik';
import { getOidcConfigs } from '@/lib/logto-config'; import { getOidcConfigs } from '@/lib/logto-config';
import { appendPath } from '@/utils/url'; import { appendPath } from '@/utils/url';
import { addConnectors } from './add-connectors';
import { checkAlterationState } from './check-alteration-state'; import { checkAlterationState } from './check-alteration-state';
import createPoolByEnv from './create-pool-by-env'; import createPoolByEnv from './create-pool-by-env';
import loadOidcValues from './oidc'; import loadOidcValues from './oidc';
@ -20,9 +17,6 @@ export enum MountedApps {
Welcome = 'welcome', Welcome = 'welcome',
} }
// eslint-disable-next-line unicorn/prefer-module
export const defaultConnectorDirectory = path.join(__dirname, '../../connectors');
const loadEnvValues = async () => { const loadEnvValues = async () => {
const isProduction = getEnv('NODE_ENV') === 'production'; const isProduction = getEnv('NODE_ENV') === 'production';
const isTest = getEnv('NODE_ENV') === 'test'; const isTest = getEnv('NODE_ENV') === 'test';
@ -46,7 +40,6 @@ const loadEnvValues = async () => {
developmentUserId: getEnv('DEVELOPMENT_USER_ID'), developmentUserId: getEnv('DEVELOPMENT_USER_ID'),
trustProxyHeader: isTrue(getEnv('TRUST_PROXY_HEADER')), trustProxyHeader: isTrue(getEnv('TRUST_PROXY_HEADER')),
adminConsoleUrl: appendPath(endpoint, '/console'), adminConsoleUrl: appendPath(endpoint, '/console'),
connectorDirectory: getEnv('CONNECTOR_DIRECTORY', defaultConnectorDirectory),
}); });
}; };
@ -93,8 +86,6 @@ function createEnvSet() {
const [, oidcConfigs] = await Promise.all([checkAlterationState(pool), getOidcConfigs(pool)]); const [, oidcConfigs] = await Promise.all([checkAlterationState(pool), getOidcConfigs(pool)]);
oidc = await loadOidcValues(appendPath(values.endpoint, '/oidc').toString(), oidcConfigs); oidc = await loadOidcValues(appendPath(values.endpoint, '/oidc').toString(), oidcConfigs);
await addConnectors(values.connectorDirectory);
}, },
}; };
} }

View file

@ -256,7 +256,6 @@ importers:
'@types/etag': ^1.8.1 '@types/etag': ^1.8.1
'@types/fs-extra': ^9.0.13 '@types/fs-extra': ^9.0.13
'@types/http-errors': ^1.8.2 '@types/http-errors': ^1.8.2
'@types/inquirer': ^8.2.1
'@types/jest': ^28.1.6 '@types/jest': ^28.1.6
'@types/js-yaml': ^4.0.5 '@types/js-yaml': ^4.0.5
'@types/koa': ^2.13.3 '@types/koa': ^2.13.3
@ -267,9 +266,7 @@ importers:
'@types/lodash.pick': ^4.4.6 '@types/lodash.pick': ^4.4.6
'@types/node': ^16.3.1 '@types/node': ^16.3.1
'@types/oidc-provider': ^7.11.1 '@types/oidc-provider': ^7.11.1
'@types/rimraf': ^3.0.2
'@types/supertest': ^2.0.11 '@types/supertest': ^2.0.11
'@types/tar': ^6.1.2
chalk: ^4 chalk: ^4
clean-deep: ^3.4.0 clean-deep: ^3.4.0
copyfiles: ^2.4.1 copyfiles: ^2.4.1
@ -286,7 +283,6 @@ importers:
http-errors: ^1.6.3 http-errors: ^1.6.3
i18next: ^21.8.16 i18next: ^21.8.16
iconv-lite: 0.6.3 iconv-lite: 0.6.3
inquirer: ^8.2.2
jest: ^28.1.3 jest: ^28.1.3
jest-matcher-specific-error: ^1.0.0 jest-matcher-specific-error: ^1.0.0
jose: ^4.0.0 jose: ^4.0.0
@ -310,7 +306,6 @@ importers:
p-retry: ^4.6.1 p-retry: ^4.6.1
prettier: ^2.7.1 prettier: ^2.7.1
query-string: ^7.0.1 query-string: ^7.0.1
rimraf: ^3.0.2
roarr: ^7.11.0 roarr: ^7.11.0
slonik: ^30.0.0 slonik: ^30.0.0
slonik-interceptor-preset: ^1.2.10 slonik-interceptor-preset: ^1.2.10
@ -318,7 +313,6 @@ importers:
snake-case: ^3.0.4 snake-case: ^3.0.4
snakecase-keys: ^5.1.0 snakecase-keys: ^5.1.0
supertest: ^6.2.2 supertest: ^6.2.2
tar: ^6.1.11
typescript: ^4.7.4 typescript: ^4.7.4
zod: ^3.18.0 zod: ^3.18.0
dependencies: dependencies:
@ -344,7 +338,6 @@ importers:
hash-wasm: 4.9.0 hash-wasm: 4.9.0
i18next: 21.8.16 i18next: 21.8.16
iconv-lite: 0.6.3 iconv-lite: 0.6.3
inquirer: 8.2.2
jose: 4.6.0 jose: 4.6.0
js-yaml: 4.1.0 js-yaml: 4.1.0
koa: 2.13.4 koa: 2.13.4
@ -361,14 +354,12 @@ importers:
oidc-provider: 7.11.3 oidc-provider: 7.11.3
p-retry: 4.6.1 p-retry: 4.6.1
query-string: 7.0.1 query-string: 7.0.1
rimraf: 3.0.2
roarr: 7.11.0 roarr: 7.11.0
slonik: 30.1.2 slonik: 30.1.2
slonik-interceptor-preset: 1.2.10 slonik-interceptor-preset: 1.2.10
slonik-sql-tag-raw: 1.1.4_roarr@7.11.0+slonik@30.1.2 slonik-sql-tag-raw: 1.1.4_roarr@7.11.0+slonik@30.1.2
snake-case: 3.0.4 snake-case: 3.0.4
snakecase-keys: 5.1.2 snakecase-keys: 5.1.2
tar: 6.1.11
zod: 3.18.0 zod: 3.18.0
devDependencies: devDependencies:
'@shopify/jest-koa-mocks': 5.0.0 '@shopify/jest-koa-mocks': 5.0.0
@ -379,7 +370,6 @@ importers:
'@types/etag': 1.8.1 '@types/etag': 1.8.1
'@types/fs-extra': 9.0.13 '@types/fs-extra': 9.0.13
'@types/http-errors': 1.8.2 '@types/http-errors': 1.8.2
'@types/inquirer': 8.2.1
'@types/jest': 28.1.6 '@types/jest': 28.1.6
'@types/js-yaml': 4.0.5 '@types/js-yaml': 4.0.5
'@types/koa': 2.13.4 '@types/koa': 2.13.4
@ -390,9 +380,7 @@ importers:
'@types/lodash.pick': 4.4.6 '@types/lodash.pick': 4.4.6
'@types/node': 16.11.12 '@types/node': 16.11.12
'@types/oidc-provider': 7.11.1 '@types/oidc-provider': 7.11.1
'@types/rimraf': 3.0.2
'@types/supertest': 2.0.11 '@types/supertest': 2.0.11
'@types/tar': 6.1.2
copyfiles: 2.4.1 copyfiles: 2.4.1
eslint: 8.21.0 eslint: 8.21.0
http-errors: 1.8.1 http-errors: 1.8.1
@ -4047,7 +4035,7 @@ packages:
'@jest/types': 28.1.3 '@jest/types': 28.1.3
deepmerge: 4.2.2 deepmerge: 4.2.2
identity-obj-proxy: 3.0.0 identity-obj-proxy: 3.0.0
jest: 28.1.3_k5ytkvaprncdyzidqqws5bqksq jest: 28.1.3_@types+node@16.11.12
jest-matcher-specific-error: 1.0.0 jest-matcher-specific-error: 1.0.0
jest-transform-stub: 2.0.0 jest-transform-stub: 2.0.0
ts-jest: 28.0.7_lhw3xkmzugq5tscs3x2ndm4sby ts-jest: 28.0.7_lhw3xkmzugq5tscs3x2ndm4sby
@ -4461,13 +4449,6 @@ packages:
'@types/node': 17.0.23 '@types/node': 17.0.23
dev: true dev: true
/@types/glob/8.0.0:
resolution: {integrity: sha512-l6NQsDDyQUVeoTynNpC9uRvCUint/gSUXQA2euwmTuWGvPY5LSDUu6tkCtJB2SvGQlJQzLaKqcGZP4//7EDveA==}
dependencies:
'@types/minimatch': 3.0.5
'@types/node': 17.0.23
dev: true
/@types/graceful-fs/4.1.5: /@types/graceful-fs/4.1.5:
resolution: {integrity: sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw==} resolution: {integrity: sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw==}
dependencies: dependencies:
@ -4765,13 +4746,6 @@ packages:
resolution: {integrity: sha512-xoDlM2S4ortawSWORYqsdU+2rxdh4LRW9ytc3zmT37RIKQh6IHyKwwtKhKis9ah8ol07DCkZxPt8BBvPjC6v4g==} resolution: {integrity: sha512-xoDlM2S4ortawSWORYqsdU+2rxdh4LRW9ytc3zmT37RIKQh6IHyKwwtKhKis9ah8ol07DCkZxPt8BBvPjC6v4g==}
dev: false dev: false
/@types/rimraf/3.0.2:
resolution: {integrity: sha512-F3OznnSLAUxFrCEu/L5PY8+ny8DtcFRjx7fZZ9bycvXRi3KPTRS9HOitGZwvPg0juRhXFWIeKX58cnX5YqLohQ==}
dependencies:
'@types/glob': 8.0.0
'@types/node': 17.0.23
dev: true
/@types/scheduler/0.16.2: /@types/scheduler/0.16.2:
resolution: {integrity: sha512-hppQEBDmlwhFAXKJX2KnWLYu5yMfi91yazPb2l+lbJiwW+wdo1gNeRA+3RgNSO39WYX2euey41KEwnqesU2Jew==} resolution: {integrity: sha512-hppQEBDmlwhFAXKJX2KnWLYu5yMfi91yazPb2l+lbJiwW+wdo1gNeRA+3RgNSO39WYX2euey41KEwnqesU2Jew==}
dev: true dev: true
@ -7872,6 +7846,7 @@ packages:
/fs.realpath/1.0.0: /fs.realpath/1.0.0:
resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==} resolution: {integrity: sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==}
dev: true
/fsevents/2.3.2: /fsevents/2.3.2:
resolution: {integrity: sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==} resolution: {integrity: sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==}
@ -8095,6 +8070,7 @@ packages:
minimatch: 3.1.2 minimatch: 3.1.2
once: 1.4.0 once: 1.4.0
path-is-absolute: 1.0.1 path-is-absolute: 1.0.1
dev: true
/glob/8.0.3: /glob/8.0.3:
resolution: {integrity: sha512-ull455NHSHI/Y1FqGaaYFaLGkNMMJbavMrEGFXG/PGrg6y7sutWHUHrz6gy6WEBH6akM1M414dWKCNs+IhKdiQ==} resolution: {integrity: sha512-ull455NHSHI/Y1FqGaaYFaLGkNMMJbavMrEGFXG/PGrg6y7sutWHUHrz6gy6WEBH6akM1M414dWKCNs+IhKdiQ==}
@ -8776,6 +8752,7 @@ packages:
dependencies: dependencies:
once: 1.4.0 once: 1.4.0
wrappy: 1.0.2 wrappy: 1.0.2
dev: true
/inherits/2.0.1: /inherits/2.0.1:
resolution: {integrity: sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA==} resolution: {integrity: sha512-8nWq2nLTAwd02jTqJExUYFSD/fKq6VH9Y/oG2accc/kdI0V98Bag8d5a4gi3XHz73rDWa2PvTtvcWYquKqSENA==}
@ -13973,6 +13950,7 @@ packages:
hasBin: true hasBin: true
dependencies: dependencies:
glob: 7.2.0 glob: 7.2.0
dev: true
/roarr/2.15.4: /roarr/2.15.4:
resolution: {integrity: sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A==} resolution: {integrity: sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A==}
@ -15237,7 +15215,7 @@ packages:
'@jest/types': 28.1.3 '@jest/types': 28.1.3
bs-logger: 0.2.6 bs-logger: 0.2.6
fast-json-stable-stringify: 2.1.0 fast-json-stable-stringify: 2.1.0
jest: 28.1.3_k5ytkvaprncdyzidqqws5bqksq jest: 28.1.3_@types+node@16.11.12
jest-util: 28.1.3 jest-util: 28.1.3
json5: 2.2.1 json5: 2.2.1
lodash.memoize: 4.1.2 lodash.memoize: 4.1.2