mirror of
https://github.com/logto-io/logto.git
synced 2025-01-06 20:40:08 -05:00
Merge pull request #2051 from logto-io/gao-log-4312-core-remove-database-seed
refactor: remove database seed from core
This commit is contained in:
commit
ff900c3a65
12 changed files with 214 additions and 297 deletions
12
.github/workflows/integration-test.yml
vendored
12
.github/workflows/integration-test.yml
vendored
|
@ -83,13 +83,19 @@ jobs:
|
|||
- name: Extract
|
||||
run: tar -xzf logto.tar.gz
|
||||
|
||||
- name: Run Logto
|
||||
run: node . --from-root --all-yes &
|
||||
- name: Seed database
|
||||
working-directory: logto/packages/core
|
||||
run: |
|
||||
npm run cli db set-url postgres://postgres:postgres@localhost:5432
|
||||
npm run cli db seed
|
||||
|
||||
- name: Run Logto
|
||||
working-directory: logto/packages/core
|
||||
run: node . --from-root --all-yes &
|
||||
env:
|
||||
INTEGRATION_TEST: true
|
||||
NODE_ENV: production
|
||||
DB_URL_DEFAULT: postgres://postgres:postgres@localhost:5432
|
||||
DB_URL: postgres://postgres:postgres@localhost:5432
|
||||
|
||||
- name: Sleep for 5 seconds
|
||||
run: sleep 5
|
||||
|
|
|
@ -93,7 +93,7 @@ const deployAlteration = async (
|
|||
|
||||
await pool.end();
|
||||
log.error(
|
||||
`Error ocurred during running alteration ${chalk.green(filename)}.\n\n` +
|
||||
`Error ocurred during running alteration ${chalk.blue(filename)}.\n\n` +
|
||||
" This alteration didn't change anything since it was in a transaction.\n" +
|
||||
' Try to fix the error and deploy again.'
|
||||
);
|
||||
|
|
|
@ -2,62 +2,16 @@ import { readdir, readFile } from 'fs/promises';
|
|||
import path from 'path';
|
||||
|
||||
import { seeds } from '@logto/schemas';
|
||||
import {
|
||||
createPool,
|
||||
DatabasePool,
|
||||
DatabaseTransactionConnection,
|
||||
parseDsn,
|
||||
sql,
|
||||
stringifyDsn,
|
||||
} from 'slonik';
|
||||
import chalk from 'chalk';
|
||||
import { DatabasePool, DatabaseTransactionConnection, sql } from 'slonik';
|
||||
import { raw } from 'slonik-sql-tag-raw';
|
||||
import { CommandModule } from 'yargs';
|
||||
import { z } from 'zod';
|
||||
|
||||
import { createPoolFromConfig, getDatabaseUrlFromConfig, insertInto } from '../../database';
|
||||
import { createPoolAndDatabaseIfNeeded, insertInto } from '../../database';
|
||||
import { updateDatabaseTimestamp } from '../../queries/logto-config';
|
||||
import { buildApplicationSecret, getPathInModule, log } from '../../utilities';
|
||||
import { buildApplicationSecret, getPathInModule, log, oraPromise } from '../../utilities';
|
||||
import { getLatestAlterationTimestamp } from './alteration';
|
||||
|
||||
/**
|
||||
* Create a database pool with the database URL in config.
|
||||
* If the given database does not exists, it will try to create a new database by connecting to the maintenance database `postgres`.
|
||||
*
|
||||
* @returns A new database pool with the database URL in config.
|
||||
*/
|
||||
const createDatabasePool = async () => {
|
||||
try {
|
||||
return await createPoolFromConfig();
|
||||
} catch (error: unknown) {
|
||||
const result = z.object({ code: z.string() }).safeParse(error);
|
||||
|
||||
// Database does not exist, try to create one
|
||||
// https://www.postgresql.org/docs/14/errcodes-appendix.html
|
||||
if (!(result.success && result.data.code === '3D000')) {
|
||||
log.error(error);
|
||||
}
|
||||
|
||||
const databaseUrl = await getDatabaseUrlFromConfig();
|
||||
const dsn = parseDsn(databaseUrl);
|
||||
// It's ok to fall back to '?' since:
|
||||
// - Database name is required to connect in the previous pool
|
||||
// - It will throw error when creating database using '?'
|
||||
const databaseName = dsn.databaseName ?? '?';
|
||||
const maintenancePool = await createPool(stringifyDsn({ ...dsn, databaseName: 'postgres' }));
|
||||
await maintenancePool.query(sql`
|
||||
create database ${sql.identifier([databaseName])}
|
||||
with
|
||||
encoding = 'UTF8'
|
||||
connection_limit = -1;
|
||||
`);
|
||||
await maintenancePool.end();
|
||||
|
||||
log.info(`Database ${databaseName} successfully created.`);
|
||||
|
||||
return createPoolFromConfig();
|
||||
}
|
||||
};
|
||||
|
||||
const createTables = async (connection: DatabaseTransactionConnection) => {
|
||||
const tableDirectory = getPathInModule('@logto/schemas', 'tables');
|
||||
const directoryFiles = await readdir(tableDirectory);
|
||||
|
@ -70,10 +24,9 @@ const createTables = async (connection: DatabaseTransactionConnection) => {
|
|||
);
|
||||
|
||||
// Await in loop is intended for better error handling
|
||||
for (const [file, query] of queries) {
|
||||
for (const [, query] of queries) {
|
||||
// eslint-disable-next-line no-await-in-loop
|
||||
await connection.query(sql`${raw(query)}`);
|
||||
log.info(`Run ${file} succeeded.`);
|
||||
}
|
||||
};
|
||||
|
||||
|
@ -96,13 +49,18 @@ const seedTables = async (connection: DatabaseTransactionConnection) => {
|
|||
connection.query(insertInto(defaultRole, 'roles')),
|
||||
updateDatabaseTimestamp(connection, await getLatestAlterationTimestamp()),
|
||||
]);
|
||||
log.info('Seed tables succeeded.');
|
||||
};
|
||||
|
||||
export const seedByPool = async (pool: DatabasePool) => {
|
||||
await pool.transaction(async (connection) => {
|
||||
await createTables(connection);
|
||||
await seedTables(connection);
|
||||
await oraPromise(createTables(connection), {
|
||||
text: 'Create tables',
|
||||
prefixText: chalk.blue('[info]'),
|
||||
});
|
||||
await oraPromise(seedTables(connection), {
|
||||
text: 'Seed data',
|
||||
prefixText: chalk.blue('[info]'),
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
|
@ -110,7 +68,7 @@ const seed: CommandModule = {
|
|||
command: 'seed',
|
||||
describe: 'Create database and seed tables and data',
|
||||
handler: async () => {
|
||||
const pool = await createDatabasePool();
|
||||
const pool = await createPoolAndDatabaseIfNeeded();
|
||||
|
||||
try {
|
||||
await seedByPool(pool);
|
||||
|
|
|
@ -4,14 +4,17 @@ import { mkdir } from 'fs/promises';
|
|||
import os from 'os';
|
||||
import path from 'path';
|
||||
|
||||
import { conditional } from '@silverhand/essentials';
|
||||
import chalk from 'chalk';
|
||||
import { remove, writeFile } from 'fs-extra';
|
||||
import inquirer from 'inquirer';
|
||||
import ora from 'ora';
|
||||
import * as semver from 'semver';
|
||||
import tar from 'tar';
|
||||
import { CommandModule } from 'yargs';
|
||||
|
||||
import { downloadFile, log, safeExecSync } from '../utilities';
|
||||
import { createPoolAndDatabaseIfNeeded, getDatabaseUrlFromConfig } from '../database';
|
||||
import { downloadFile, log, oraPromise, safeExecSync } from '../utilities';
|
||||
import { seedByPool } from './database/seed';
|
||||
|
||||
export type InstallArgs = {
|
||||
path?: string;
|
||||
|
@ -36,12 +39,26 @@ const validateNodeVersion = () => {
|
|||
}
|
||||
};
|
||||
|
||||
const validatePath = (value: string) =>
|
||||
existsSync(path.resolve(value))
|
||||
? `The path ${chalk.green(value)} already exists, please try another.`
|
||||
: true;
|
||||
const inquireInstancePath = async (initialPath?: string) => {
|
||||
const { instancePath } = await inquirer.prompt<{ instancePath: string }>(
|
||||
{
|
||||
name: 'instancePath',
|
||||
message: 'Where should we create your Logto instance?',
|
||||
type: 'input',
|
||||
default: defaultPath,
|
||||
filter: (value: string) => value.trim(),
|
||||
validate: (value: string) =>
|
||||
existsSync(path.resolve(value))
|
||||
? `The path ${chalk.green(value)} already exists, please try another.`
|
||||
: true,
|
||||
},
|
||||
{ instancePath: initialPath }
|
||||
);
|
||||
|
||||
const getInstancePath = async () => {
|
||||
return instancePath;
|
||||
};
|
||||
|
||||
const validateDatabase = async () => {
|
||||
const { hasPostgresUrl } = await inquirer.prompt<{ hasPostgresUrl?: boolean }>({
|
||||
name: 'hasPostgresUrl',
|
||||
message: `Logto requires PostgreSQL >=${pgRequired.version} but cannot find in the current environment.\n Do you have a remote PostgreSQL instance ready?`,
|
||||
|
@ -59,17 +76,6 @@ const getInstancePath = async () => {
|
|||
if (hasPostgresUrl === false) {
|
||||
log.error('Logto requires a Postgres instance to run.');
|
||||
}
|
||||
|
||||
const { instancePath } = await inquirer.prompt<{ instancePath: string }>({
|
||||
name: 'instancePath',
|
||||
message: 'Where should we create your Logto instance?',
|
||||
type: 'input',
|
||||
default: defaultPath,
|
||||
filter: (value: string) => value.trim(),
|
||||
validate: validatePath,
|
||||
});
|
||||
|
||||
return instancePath;
|
||||
};
|
||||
|
||||
const downloadRelease = async () => {
|
||||
|
@ -85,38 +91,72 @@ const downloadRelease = async () => {
|
|||
};
|
||||
|
||||
const decompress = async (toPath: string, tarPath: string) => {
|
||||
const decompressSpinner = ora({
|
||||
text: `Decompress to ${toPath}`,
|
||||
prefixText: chalk.blue('[info]'),
|
||||
}).start();
|
||||
|
||||
try {
|
||||
await mkdir(toPath);
|
||||
await tar.extract({ file: tarPath, cwd: toPath, strip: 1 });
|
||||
} catch (error: unknown) {
|
||||
decompressSpinner.fail();
|
||||
log.error(error);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
decompressSpinner.succeed();
|
||||
};
|
||||
|
||||
const installLogto = async ({ path: pathArgument = defaultPath, silent = false }: InstallArgs) => {
|
||||
validateNodeVersion();
|
||||
|
||||
const instancePath = (!silent && (await getInstancePath())) || pathArgument;
|
||||
const isValidPath = validatePath(instancePath);
|
||||
// Get instance path
|
||||
const instancePath = await inquireInstancePath(conditional(silent && pathArgument));
|
||||
|
||||
if (isValidPath !== true) {
|
||||
log.error(isValidPath);
|
||||
// Validate database URL
|
||||
await validateDatabase();
|
||||
|
||||
// Download and decompress
|
||||
const tarPath = await downloadRelease();
|
||||
await oraPromise(
|
||||
decompress(instancePath, tarPath),
|
||||
{
|
||||
text: `Decompress to ${instancePath}`,
|
||||
prefixText: chalk.blue('[info]'),
|
||||
},
|
||||
true
|
||||
);
|
||||
|
||||
try {
|
||||
// Seed database
|
||||
const pool = await createPoolAndDatabaseIfNeeded(); // It will ask for database URL and save to config
|
||||
await seedByPool(pool);
|
||||
await pool.end();
|
||||
} catch (error: unknown) {
|
||||
console.error(error);
|
||||
|
||||
const { value } = await inquirer.prompt<{ value: boolean }>({
|
||||
name: 'value',
|
||||
type: 'confirm',
|
||||
message:
|
||||
'Error occurred during seeding your Logto database. Would you like to continue without seed?',
|
||||
default: false,
|
||||
});
|
||||
|
||||
if (!value) {
|
||||
await oraPromise(remove(instancePath), {
|
||||
text: 'Clean up',
|
||||
prefixText: chalk.blue('[info]'),
|
||||
});
|
||||
|
||||
// eslint-disable-next-line unicorn/no-process-exit
|
||||
process.exit(1);
|
||||
}
|
||||
|
||||
log.info(`You can use ${chalk.green('db seed')} command to seed when ready.`);
|
||||
}
|
||||
|
||||
const tarPath = await downloadRelease();
|
||||
|
||||
await decompress(instancePath, tarPath);
|
||||
// Save to dot env
|
||||
const databaseUrl = await getDatabaseUrlFromConfig();
|
||||
const dotEnvPath = path.resolve(instancePath, '.env');
|
||||
await writeFile(dotEnvPath, `DB_URL=${databaseUrl}`, {
|
||||
encoding: 'utf8',
|
||||
});
|
||||
log.info(`Saved database URL to ${chalk.blue(dotEnvPath)}`);
|
||||
|
||||
// Finale
|
||||
const startCommand = `cd ${instancePath} && npm start`;
|
||||
log.info(
|
||||
`Use the command below to start Logto. Happy hacking!\n\n ${chalk.green(startCommand)}`
|
||||
|
|
|
@ -41,5 +41,5 @@ export const getConfig = async () => {
|
|||
export const patchConfig = async (config: LogtoConfig) => {
|
||||
const configPath = await getConfigPath();
|
||||
await writeFile(configPath, JSON.stringify({ ...(await getConfig()), ...config }, undefined, 2));
|
||||
log.info(`Updated config in ${chalk.green(configPath)}`);
|
||||
log.info(`Updated config in ${chalk.blue(configPath)}`);
|
||||
};
|
||||
|
|
|
@ -1,19 +1,43 @@
|
|||
import { SchemaLike, SchemaValue, SchemaValuePrimitive } from '@logto/schemas';
|
||||
import chalk from 'chalk';
|
||||
import decamelize from 'decamelize';
|
||||
import { createPool, IdentifierSqlToken, sql, SqlToken } from 'slonik';
|
||||
import inquirer from 'inquirer';
|
||||
import { createPool, IdentifierSqlToken, parseDsn, sql, SqlToken, stringifyDsn } from 'slonik';
|
||||
import { createInterceptors } from 'slonik-interceptor-preset';
|
||||
import { z } from 'zod';
|
||||
|
||||
import { getConfig } from './config';
|
||||
import { getConfig, patchConfig } from './config';
|
||||
import { log } from './utilities';
|
||||
|
||||
export const defaultDatabaseUrl = 'postgresql://localhost:5432/logto';
|
||||
|
||||
export const getDatabaseUrlFromConfig = async () => {
|
||||
const { databaseUrl } = await getConfig();
|
||||
|
||||
if (!databaseUrl) {
|
||||
log.error(
|
||||
`No database URL configured. Set it via ${chalk.green('database set-url')} command first.`
|
||||
);
|
||||
const { value } = await inquirer
|
||||
.prompt<{ value: string }>({
|
||||
type: 'input',
|
||||
name: 'value',
|
||||
message: 'Enter your Logto database URL',
|
||||
default: defaultDatabaseUrl,
|
||||
})
|
||||
.catch(async (error) => {
|
||||
if (error.isTtyError) {
|
||||
log.error(
|
||||
`No database URL configured. Set it via ${chalk.green(
|
||||
'database set-url'
|
||||
)} command first.`
|
||||
);
|
||||
}
|
||||
|
||||
// The type definition does not give us type except `any`, throw it directly will honor the original behavior.
|
||||
// eslint-disable-next-line @typescript-eslint/no-throw-literal
|
||||
throw error;
|
||||
});
|
||||
await patchConfig({ databaseUrl: value });
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
return databaseUrl;
|
||||
|
@ -27,6 +51,45 @@ export const createPoolFromConfig = async () => {
|
|||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* Create a database pool with the database URL in config.
|
||||
* If the given database does not exists, it will try to create a new database by connecting to the maintenance database `postgres`.
|
||||
*
|
||||
* @returns A new database pool with the database URL in config.
|
||||
*/
|
||||
export const createPoolAndDatabaseIfNeeded = async () => {
|
||||
try {
|
||||
return await createPoolFromConfig();
|
||||
} catch (error: unknown) {
|
||||
const result = z.object({ code: z.string() }).safeParse(error);
|
||||
|
||||
// Database does not exist, try to create one
|
||||
// https://www.postgresql.org/docs/14/errcodes-appendix.html
|
||||
if (!(result.success && result.data.code === '3D000')) {
|
||||
log.error(error);
|
||||
}
|
||||
|
||||
const databaseUrl = await getDatabaseUrlFromConfig();
|
||||
const dsn = parseDsn(databaseUrl);
|
||||
// It's ok to fall back to '?' since:
|
||||
// - Database name is required to connect in the previous pool
|
||||
// - It will throw error when creating database using '?'
|
||||
const databaseName = dsn.databaseName ?? '?';
|
||||
const maintenancePool = await createPool(stringifyDsn({ ...dsn, databaseName: 'postgres' }));
|
||||
await maintenancePool.query(sql`
|
||||
create database ${sql.identifier([databaseName])}
|
||||
with
|
||||
encoding = 'UTF8'
|
||||
connection_limit = -1;
|
||||
`);
|
||||
await maintenancePool.end();
|
||||
|
||||
log.info(`${chalk.green('✔')} Created database ${databaseName}`);
|
||||
|
||||
return createPoolFromConfig();
|
||||
}
|
||||
};
|
||||
|
||||
// TODO: Move database utils to `core-kit`
|
||||
export type Table = { table: string; fields: Record<string, string> };
|
||||
export type FieldIdentifiers<Key extends string | number | symbol> = {
|
||||
|
|
|
@ -80,6 +80,29 @@ export const getPathInModule = (moduleName: string, relativePath = '/') =>
|
|||
relativePath
|
||||
);
|
||||
|
||||
export const oraPromise = async <T>(
|
||||
promise: PromiseLike<T>,
|
||||
options?: ora.Options,
|
||||
exitOnError = false
|
||||
) => {
|
||||
const spinner = ora(options).start();
|
||||
|
||||
try {
|
||||
const result = await promise;
|
||||
spinner.succeed();
|
||||
|
||||
return result;
|
||||
} catch (error: unknown) {
|
||||
spinner.fail();
|
||||
|
||||
if (exitOnError) {
|
||||
log.error(error);
|
||||
}
|
||||
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
|
||||
// TODO: Move to `@silverhand/essentials`
|
||||
// Intended
|
||||
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
||||
|
|
|
@ -17,11 +17,13 @@
|
|||
"add-connector": "node build/cli/add-connector.js",
|
||||
"add-official-connectors": "node build/cli/add-official-connectors.js",
|
||||
"alteration": "node build/cli/alteration.js",
|
||||
"cli": "logto",
|
||||
"test": "jest",
|
||||
"test:ci": "jest --coverage --silent",
|
||||
"test:report": "codecov -F core"
|
||||
},
|
||||
"dependencies": {
|
||||
"@logto/cli": "^1.0.0-beta.10",
|
||||
"@logto/connector-kit": "^1.0.0-beta.13",
|
||||
"@logto/core-kit": "^1.0.0-beta.13",
|
||||
"@logto/phrases": "^1.0.0-beta.10",
|
||||
|
|
|
@ -1,100 +0,0 @@
|
|||
import { readdir, readFile } from 'fs/promises';
|
||||
import path from 'path';
|
||||
|
||||
import { SchemaLike, seeds } from '@logto/schemas';
|
||||
import chalk from 'chalk';
|
||||
import decamelize from 'decamelize';
|
||||
import { createPool, parseDsn, sql, stringifyDsn } from 'slonik';
|
||||
import { createInterceptors } from 'slonik-interceptor-preset';
|
||||
import { raw } from 'slonik-sql-tag-raw';
|
||||
|
||||
import { updateDatabaseTimestamp } from '@/alteration';
|
||||
import { buildApplicationSecret } from '@/utils/id';
|
||||
|
||||
import { convertToPrimitiveOrSql } from './utils';
|
||||
|
||||
const {
|
||||
managementResource,
|
||||
defaultSignInExperience,
|
||||
createDefaultSetting,
|
||||
createDemoAppApplication,
|
||||
defaultRole,
|
||||
} = seeds;
|
||||
const tableDirectory = 'node_modules/@logto/schemas/tables';
|
||||
|
||||
export const replaceDsnDatabase = (dsn: string, databaseName: string): string =>
|
||||
stringifyDsn({ ...parseDsn(dsn), databaseName });
|
||||
|
||||
/**
|
||||
* Create a database.
|
||||
* @returns DSN with the created database name.
|
||||
*/
|
||||
export const createDatabase = async (dsn: string, databaseName: string): Promise<string> => {
|
||||
const pool = await createPool(replaceDsnDatabase(dsn, 'postgres'));
|
||||
|
||||
await pool.query(sql`
|
||||
create database ${sql.identifier([databaseName])}
|
||||
with
|
||||
encoding = 'UTF8'
|
||||
connection_limit = -1;
|
||||
`);
|
||||
await pool.end();
|
||||
|
||||
console.log(`${chalk.blue('[create]')} Database ${databaseName} successfully created.`);
|
||||
|
||||
return replaceDsnDatabase(dsn, databaseName);
|
||||
};
|
||||
|
||||
export const insertInto = <T extends SchemaLike>(object: T, table: string) => {
|
||||
const keys = Object.keys(object);
|
||||
|
||||
return sql`
|
||||
insert into ${sql.identifier([table])}
|
||||
(${sql.join(
|
||||
keys.map((key) => sql.identifier([decamelize(key)])),
|
||||
sql`, `
|
||||
)})
|
||||
values (${sql.join(
|
||||
keys.map((key) => convertToPrimitiveOrSql(key, object[key] ?? null)),
|
||||
sql`, `
|
||||
)})
|
||||
`;
|
||||
};
|
||||
|
||||
export const createDatabaseCli = async (dsn: string) => {
|
||||
const pool = await createPool(dsn, { interceptors: createInterceptors() });
|
||||
|
||||
const createTables = async () => {
|
||||
const directory = await readdir(tableDirectory);
|
||||
const tableFiles = directory.filter((file) => file.endsWith('.sql'));
|
||||
const queries = await Promise.all(
|
||||
tableFiles.map<Promise<[string, string]>>(async (file) => [
|
||||
file,
|
||||
await readFile(path.join(tableDirectory, file), 'utf8'),
|
||||
])
|
||||
);
|
||||
|
||||
// Await in loop is intended for better error handling
|
||||
for (const [file, query] of queries) {
|
||||
// eslint-disable-next-line no-await-in-loop
|
||||
await pool.query(sql`${raw(query)}`);
|
||||
console.log(`${chalk.blue('[create-tables]')} Run ${file} succeeded.`);
|
||||
}
|
||||
|
||||
await updateDatabaseTimestamp(pool);
|
||||
console.log(`${chalk.blue('[create-tables]')} Update alteration state succeeded.`);
|
||||
};
|
||||
|
||||
const seedTables = async () => {
|
||||
await Promise.all([
|
||||
pool.query(insertInto(managementResource, 'resources')),
|
||||
pool.query(insertInto(createDefaultSetting(), 'settings')),
|
||||
pool.query(insertInto(defaultSignInExperience, 'sign_in_experiences')),
|
||||
pool.query(insertInto(createDemoAppApplication(buildApplicationSecret()), 'applications')),
|
||||
pool.query(insertInto(defaultRole, 'roles')),
|
||||
]);
|
||||
console.log(`${chalk.blue('[seed-tables]')} Seed tables succeeded.`);
|
||||
};
|
||||
|
||||
return { createTables, seedTables, pool };
|
||||
};
|
|
@ -13,7 +13,7 @@ export const checkAlterationState = async (pool: DatabasePool) => {
|
|||
}
|
||||
|
||||
const error = new Error(
|
||||
`Found undeployed database alterations, you must deploy them first by "npm alteration deploy" command, reference: https://docs.logto.io/docs/recipes/deployment/#database-alteration`
|
||||
`Found undeployed database alterations, you must deploy them first by "npm run alteration deploy" command, reference: https://docs.logto.io/docs/recipes/deployment/#database-alteration`
|
||||
);
|
||||
|
||||
if (allYes) {
|
||||
|
|
|
@ -1,82 +1,7 @@
|
|||
import { assertEnv, conditional, getEnv, Optional } from '@silverhand/essentials';
|
||||
import inquirer from 'inquirer';
|
||||
import { assertEnv } from '@silverhand/essentials';
|
||||
import chalk from 'chalk';
|
||||
import { createPool } from 'slonik';
|
||||
import { createInterceptors } from 'slonik-interceptor-preset';
|
||||
import { z } from 'zod';
|
||||
|
||||
import { createDatabase, createDatabaseCli, replaceDsnDatabase } from '@/database/seed';
|
||||
|
||||
import { appendDotEnv } from './dot-env';
|
||||
import { allYes, noInquiry } from './parameters';
|
||||
|
||||
const defaultDatabaseUrl = getEnv('DB_URL_DEFAULT', 'postgres://@localhost:5432');
|
||||
const defaultDatabaseName = 'logto';
|
||||
|
||||
const initDatabase = async (dsn: string): Promise<[string, boolean]> => {
|
||||
try {
|
||||
return [await createDatabase(dsn, defaultDatabaseName), true];
|
||||
} catch (error: unknown) {
|
||||
const result = z.object({ code: z.string() }).safeParse(error);
|
||||
|
||||
// https://www.postgresql.org/docs/12/errcodes-appendix.html
|
||||
const databaseExists = result.success && result.data.code === '42P04';
|
||||
|
||||
if (!databaseExists) {
|
||||
throw error;
|
||||
}
|
||||
|
||||
if (allYes) {
|
||||
return [replaceDsnDatabase(dsn, defaultDatabaseName), false];
|
||||
}
|
||||
|
||||
const useCurrent = await inquirer.prompt({
|
||||
type: 'confirm',
|
||||
name: 'value',
|
||||
message: `A database named "${defaultDatabaseName}" already exists. Would you like to use it without filling the initial data?`,
|
||||
});
|
||||
|
||||
if (useCurrent.value) {
|
||||
return [replaceDsnDatabase(dsn, defaultDatabaseName), false];
|
||||
}
|
||||
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
|
||||
const inquireForLogtoDsn = async (key: string): Promise<[Optional<string>, boolean]> => {
|
||||
if (allYes) {
|
||||
return initDatabase(defaultDatabaseUrl);
|
||||
}
|
||||
|
||||
const setUp = await inquirer.prompt({
|
||||
type: 'confirm',
|
||||
name: 'value',
|
||||
message: `No Postgres DSN (${key}) found in env variables. Would you like to set up a new Logto database?`,
|
||||
});
|
||||
|
||||
if (!setUp.value) {
|
||||
const dsn = await inquirer.prompt({
|
||||
name: 'value',
|
||||
default: new URL(defaultDatabaseName, defaultDatabaseUrl).href,
|
||||
message: 'Please input the DSN which points to an existing Logto database:',
|
||||
});
|
||||
|
||||
return [conditional<string>(dsn.value && String(dsn.value)), false];
|
||||
}
|
||||
|
||||
const dsnAnswer = await inquirer.prompt({
|
||||
name: 'value',
|
||||
default: new URL(defaultDatabaseUrl).href,
|
||||
message: `Please input the DSN _WITHOUT_ database name:`,
|
||||
});
|
||||
const dsn = conditional<string>(dsnAnswer.value && String(dsnAnswer.value));
|
||||
|
||||
if (!dsn) {
|
||||
return [dsn, false];
|
||||
}
|
||||
|
||||
return initDatabase(dsn);
|
||||
};
|
||||
|
||||
const createPoolByEnv = async (isTest: boolean) => {
|
||||
// Database connection is disabled in unit test environment
|
||||
|
@ -92,26 +17,24 @@ const createPoolByEnv = async (isTest: boolean) => {
|
|||
|
||||
return await createPool(databaseDsn, { interceptors });
|
||||
} catch (error: unknown) {
|
||||
if (noInquiry) {
|
||||
throw error;
|
||||
if (error instanceof Error && error.message === `env variable ${key} not found`) {
|
||||
console.error(
|
||||
`${chalk.red('[error]')} No Postgres DSN (${chalk.green(
|
||||
key
|
||||
)}) found in env variables.\n\n` +
|
||||
` Either provide it in your env, or add it to the ${chalk.blue(
|
||||
'.env'
|
||||
)} file in the Logto project root.\n\n` +
|
||||
` If you want to set up a new Logto database, run ${chalk.green(
|
||||
'npm run cli db seed'
|
||||
)} before setting env ${chalk.green(key)}.\n\n` +
|
||||
` Visit ${chalk.blue(
|
||||
'https://docs.logto.io/docs/references/core/configuration'
|
||||
)} for more info about setting up env.\n`
|
||||
);
|
||||
}
|
||||
|
||||
const [dsn, needsSeed] = await inquireForLogtoDsn(key);
|
||||
|
||||
if (!dsn) {
|
||||
throw error;
|
||||
}
|
||||
|
||||
const cli = await createDatabaseCli(dsn);
|
||||
|
||||
if (needsSeed) {
|
||||
await cli.createTables();
|
||||
await cli.seedTables();
|
||||
}
|
||||
|
||||
appendDotEnv(key, dsn);
|
||||
|
||||
return cli.pool;
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
|
||||
|
|
|
@ -223,6 +223,7 @@ importers:
|
|||
|
||||
packages/core:
|
||||
specifiers:
|
||||
'@logto/cli': ^1.0.0-beta.10
|
||||
'@logto/connector-kit': ^1.0.0-beta.13
|
||||
'@logto/core-kit': ^1.0.0-beta.13
|
||||
'@logto/phrases': ^1.0.0-beta.10
|
||||
|
@ -301,6 +302,7 @@ importers:
|
|||
typescript: ^4.7.4
|
||||
zod: ^3.18.0
|
||||
dependencies:
|
||||
'@logto/cli': link:../cli
|
||||
'@logto/connector-kit': 1.0.0-beta.13
|
||||
'@logto/core-kit': 1.0.0-beta.13
|
||||
'@logto/phrases': link:../phrases
|
||||
|
|
Loading…
Reference in a new issue