0
Fork 0
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:
Gao Sun 2022-10-08 17:24:05 +08:00 committed by GitHub
commit ff900c3a65
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
12 changed files with 214 additions and 297 deletions

View file

@ -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

View file

@ -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.'
);

View file

@ -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);

View file

@ -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)}`

View file

@ -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)}`);
};

View file

@ -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> = {

View file

@ -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

View file

@ -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",

View file

@ -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 };
};

View file

@ -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) {

View file

@ -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;
}
};

View file

@ -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