0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2025-01-20 22:12:38 -05:00

feat: migrations now working!

This commit is contained in:
bholmesdev 2024-01-26 22:10:01 -05:00
parent 49b82a0e7d
commit f1fb0259ae
2 changed files with 41 additions and 28 deletions

View file

@ -1,11 +1,11 @@
import type { InArgs, InStatement } from '@libsql/client'; import { createClient, type InArgs, type InStatement } from '@libsql/client';
import type { AstroConfig } from 'astro'; import type { AstroConfig } from 'astro';
import deepDiff from 'deep-diff'; import deepDiff from 'deep-diff';
import { type SQL, eq, type Query, sql } from 'drizzle-orm'; import { type SQL, eq, sql } from 'drizzle-orm';
import { SQLiteAsyncDialect } from 'drizzle-orm/sqlite-core'; import { SQLiteAsyncDialect } from 'drizzle-orm/sqlite-core';
import type { Arguments } from 'yargs-parser'; import type { Arguments } from 'yargs-parser';
import { appTokenError } from '../../../errors.js'; import { appTokenError } from '../../../errors.js';
import { collectionToTable, createLocalDatabaseClient } from '../../../internal.js'; import { drizzle } from 'drizzle-orm/sqlite-proxy';
import { import {
createCurrentSnapshot, createCurrentSnapshot,
createEmptySnapshot, createEmptySnapshot,
@ -14,7 +14,7 @@ import {
loadInitialSnapshot, loadInitialSnapshot,
loadMigration, loadMigration,
} from '../../../migrations.js'; } from '../../../migrations.js';
import type { DBCollections, DBSnapshot } from '../../../types.js'; import type { DBSnapshot } from '../../../types.js';
import { import {
STUDIO_ADMIN_TABLE_ROW_ID, STUDIO_ADMIN_TABLE_ROW_ID,
adminTable, adminTable,
@ -24,6 +24,9 @@ import {
migrationsTable, migrationsTable,
} from '../../../utils.js'; } from '../../../utils.js';
import { getMigrationQueries } from '../../queries.js'; import { getMigrationQueries } from '../../queries.js';
import type { AstroConfigWithDB } from '../../../config.js';
import { setupDbTables } from '../../../internal.js';
const { diff } = deepDiff; const { diff } = deepDiff;
const sqliteDialect = new SQLiteAsyncDialect(); const sqliteDialect = new SQLiteAsyncDialect();
@ -119,38 +122,48 @@ async function pushSchema({
.where(eq(adminTable.id, STUDIO_ADMIN_TABLE_ROW_ID)); .where(eq(adminTable.id, STUDIO_ADMIN_TABLE_ROW_ID));
} }
/** TODO: refine with migration changes */
async function pushData({ async function pushData({
config, config,
appToken, appToken,
isDryRun, isDryRun,
}: { }: {
config: AstroConfig; config: AstroConfigWithDB;
appToken: string; appToken: string;
isDryRun?: boolean; isDryRun?: boolean;
}) { }) {
const db = await createLocalDatabaseClient({ const queries: InStatement[] = [];
collections: config.db!.collections! as DBCollections, if (config.db?.data) {
dbUrl: ':memory:', const libsqlClient = createClient({ url: ':memory:' });
seeding: true, // Use proxy to trace all queries to queue up in a batch
const db = await drizzle(async (sqlQuery, params, method) => {
const stmt: InStatement = { sql: sqlQuery, args: params };
queries.push(stmt);
// Use in-memory database to generate results for `returning()`
const { rows } = await libsqlClient.execute(stmt);
// Drizzle expects each row as an array of its values
const rowValues: unknown[][] = [];
for (const row of rows) {
if (row != null && typeof row === 'object') {
rowValues.push(Object.values(row));
}
}
if (method === 'get') {
return { rows: rowValues[0] };
}
return { rows: rowValues };
}); });
const queries: Query[] = []; await setupDbTables({
db,
mode: 'build',
collections: config.db.collections ?? {},
data: config.db.data,
});
}
// TODO: update migration seeding
// for (const [name, collection] of Object.entries(config.db!.collections! as DBCollections)) {
// if (collection.writable || !collection.data) continue;
// const table = collectionToTable(name, collection);
// const insert = db.insert(table).values(await collection.data());
// queries.push(insert.toSQL());
// }
const url = new URL('/db/query', getRemoteDatabaseUrl()); const url = new URL('/db/query', getRemoteDatabaseUrl());
const requestBody: InStatement[] = queries.map((q) => ({
sql: q.sql,
args: q.params as InArgs,
}));
if (isDryRun) { if (isDryRun) {
console.info('[DRY RUN] Batch data seed:', JSON.stringify(requestBody, null, 2)); console.info('[DRY RUN] Batch data seed:', JSON.stringify(queries, null, 2));
return new Response(null, { status: 200 }); return new Response(null, { status: 200 });
} }
@ -159,7 +172,7 @@ async function pushData({
headers: new Headers({ headers: new Headers({
Authorization: `Bearer ${appToken}`, Authorization: `Bearer ${appToken}`,
}), }),
body: JSON.stringify(requestBody), body: JSON.stringify(queries),
}); });
} }

View file

@ -89,10 +89,10 @@ export async function setupDbTables({
logger, logger,
mode, mode,
}: { }: {
db: LibSQLDatabase; db: SqliteRemoteDatabase;
data?: DBUserConfig['data']; data?: DBUserConfig['data'];
collections: DBCollections; collections: DBCollections;
logger: AstroIntegrationLogger; logger?: AstroIntegrationLogger;
mode: 'dev' | 'build'; mode: 'dev' | 'build';
}) { }) {
const setupQueries: SQL[] = []; const setupQueries: SQL[] = [];
@ -129,7 +129,7 @@ export async function setupDbTables({
mode, mode,
}); });
} catch (e) { } catch (e) {
logger.error( (logger ?? console).error(
`Failed to seed data. Did you update to match recent schema changes? Full error:\n\n${e}` `Failed to seed data. Did you update to match recent schema changes? Full error:\n\n${e}`
); );
} }