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

Fix problems with local libSQL DB (#12089)

Co-authored-by:  Matthew Phillips <361671+matthewp@users.noreply.github.com>
Co-authored-by: Emanuele Stoppa <602478+ematipico@users.noreply.github.com>
This commit is contained in:
Luiz Ferraz 2024-10-07 10:17:26 -03:00 committed by GitHub
parent fef0b8cce1
commit 6e06e6ed4f
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
14 changed files with 214 additions and 11 deletions

View file

@ -0,0 +1,5 @@
---
'@astrojs/db': patch
---
Fixes initial schema push for local file and in-memory libSQL DB

View file

@ -0,0 +1,5 @@
---
'@astrojs/db': patch
---
Fixes relative local libSQL db URL

View file

@ -36,6 +36,7 @@ import type {
TextColumn, TextColumn,
} from '../types.js'; } from '../types.js';
import type { RemoteDatabaseInfo, Result } from '../utils.js'; import type { RemoteDatabaseInfo, Result } from '../utils.js';
import { LibsqlError } from '@libsql/client';
const sqlite = new SQLiteAsyncDialect(); const sqlite = new SQLiteAsyncDialect();
const genTempTableName = customAlphabet('abcdefghijklmnopqrstuvwxyz', 10); const genTempTableName = customAlphabet('abcdefghijklmnopqrstuvwxyz', 10);
@ -450,10 +451,19 @@ async function getDbCurrentSnapshot(
); );
return JSON.parse(res.snapshot); return JSON.parse(res.snapshot);
} catch (error: any) { } catch (error) {
if (error.code === 'SQLITE_UNKNOWN') { // Don't handle errors that are not from libSQL
if (error instanceof LibsqlError &&
// If the schema was never pushed to the database yet the table won't exist. // If the schema was never pushed to the database yet the table won't exist.
// Treat a missing snapshot table as an empty table. // Treat a missing snapshot table as an empty table.
(
// When connecting to a remote database in that condition
// the query will fail with the following error code and message.
(error.code === 'SQLITE_UNKNOWN' && error.message === 'SQLITE_UNKNOWN: SQLite error: no such table: _astro_db_snapshot') ||
// When connecting to a local or in-memory database that does not have a snapshot table yet
// the query will fail with the following error code and message.
(error.code === 'SQLITE_ERROR' && error.message === 'SQLITE_ERROR: no such table: _astro_db_snapshot'))
) {
return; return;
} }

View file

@ -168,10 +168,14 @@ export function getStudioVirtualModContents({
function dbUrlArg() { function dbUrlArg() {
const dbStr = JSON.stringify(dbInfo.url); const dbStr = JSON.stringify(dbInfo.url);
if (isBuild) {
// Allow overriding, mostly for testing // Allow overriding, mostly for testing
return dbInfo.type === 'studio' return dbInfo.type === 'studio'
? `import.meta.env.ASTRO_STUDIO_REMOTE_DB_URL ?? ${dbStr}` ? `import.meta.env.ASTRO_STUDIO_REMOTE_DB_URL ?? ${dbStr}`
: `import.meta.env.ASTRO_DB_REMOTE_URL ?? ${dbStr}`; : `import.meta.env.ASTRO_DB_REMOTE_URL ?? ${dbStr}`;
} else {
return dbStr;
}
} }
return ` return `

View file

@ -53,17 +53,38 @@ export function createRemoteDatabaseClient(options: RemoteDbClientOptions) {
return options.dbType === 'studio' return options.dbType === 'studio'
? createStudioDatabaseClient(options.appToken, remoteUrl) ? createStudioDatabaseClient(options.appToken, remoteUrl)
: createRemoteLibSQLClient(options.appToken, remoteUrl); : createRemoteLibSQLClient(options.appToken, remoteUrl, options.remoteUrl.toString());
} }
function createRemoteLibSQLClient(appToken: string, remoteDbURL: URL) { function createRemoteLibSQLClient(appToken: string, remoteDbURL: URL, rawUrl: string) {
const options: Partial<LibSQLConfig> = Object.fromEntries(remoteDbURL.searchParams.entries()); const options: Partial<LibSQLConfig> = Object.fromEntries(remoteDbURL.searchParams.entries());
remoteDbURL.search = ''; remoteDbURL.search = '';
let url = remoteDbURL.toString();
if (remoteDbURL.protocol === 'memory:') {
// libSQL expects a special string in place of a URL
// for in-memory DBs.
url = ':memory:';
} else if (
remoteDbURL.protocol === 'file:' &&
remoteDbURL.pathname.startsWith('/') &&
!rawUrl.startsWith('file:/')
) {
// libSQL accepts relative and absolute file URLs
// for local DBs. This doesn't match the URL specification.
// Parsing `file:some.db` and `file:/some.db` should yield
// the same result, but libSQL interprets the former as
// a relative path, and the latter as an absolute path.
// This detects when such a conversion happened during parsing
// and undoes it so that the URL given to libSQL is the
// same as given by the user.
url = 'file:' + remoteDbURL.pathname.substring(1);
}
const client = createClient({ const client = createClient({
...options, ...options,
authToken: appToken, authToken: appToken,
url: remoteDbURL.protocol === 'memory:' ? ':memory:' : remoteDbURL.toString(), url,
}); });
return drizzleLibsql(client); return drizzleLibsql(client);
} }

View file

@ -15,7 +15,7 @@ describe('astro:db', () => {
}); });
}); });
describe('development', () => { describe({ skip: process.platform === 'darwin' }, 'development', () => {
let devServer; let devServer;
before(async () => { before(async () => {
@ -94,7 +94,7 @@ describe('astro:db', () => {
}); });
}); });
describe('development --remote', () => { describe({ skip: process.platform === 'darwin' }, 'development --remote', () => {
let devServer; let devServer;
let remoteDbServer; let remoteDbServer;

View file

@ -0,0 +1,10 @@
import db from '@astrojs/db';
import { defineConfig } from 'astro/config';
// https://astro.build/config
export default defineConfig({
integrations: [db()],
devToolbar: {
enabled: false,
},
});

View file

@ -0,0 +1,13 @@
import { column, defineDb, defineTable } from 'astro:db';
const User = defineTable({
columns: {
id: column.text({ primaryKey: true, optional: false }),
username: column.text({ optional: false, unique: true }),
password: column.text({ optional: false }),
},
});
export default defineDb({
tables: { User },
});

View file

@ -0,0 +1,7 @@
import { User, db } from 'astro:db';
export default async function () {
await db.batch([
db.insert(User).values([{ id: 'mario', username: 'Mario', password: 'itsame' }]),
]);
}

View file

@ -0,0 +1,14 @@
{
"name": "@test/db-libsql-remote",
"version": "0.0.0",
"private": true,
"scripts": {
"dev": "astro dev",
"build": "astro build",
"preview": "astro preview"
},
"dependencies": {
"@astrojs/db": "workspace:*",
"astro": "workspace:*"
}
}

View file

@ -0,0 +1,11 @@
---
/// <reference path="../../.astro/db-types.d.ts" />
import { User, db } from 'astro:db';
const users = await db.select().from(User);
---
<h2>Users</h2>
<ul class="users-list">
{users.map((user) => <li>{user.name}</li>)}
</ul>

View file

@ -0,0 +1,77 @@
import assert from 'node:assert/strict';
import { relative } from 'node:path';
import { rm } from 'node:fs/promises';
import { after, before, describe, it } from 'node:test';
import { fileURLToPath } from 'node:url';
import testAdapter from '../../astro/test/test-adapter.js';
import { loadFixture } from '../../astro/test/test-utils.js';
import { clearEnvironment, initializeRemoteDb } from './test-utils.js';
describe('astro:db local database', () => {
let fixture;
before(async () => {
fixture = await loadFixture({
root: new URL('./fixtures/libsql-remote/', import.meta.url),
output: 'server',
adapter: testAdapter(),
});
});
describe('build --remote with local libSQL file (absolute path)', () => {
before(async () => {
clearEnvironment();
const absoluteFileUrl = new URL('./fixtures/libsql-remote/dist/absolute.db', import.meta.url);
// Remove the file if it exists to avoid conflict between test runs
await rm(absoluteFileUrl, { force: true });
process.env.ASTRO_INTERNAL_TEST_REMOTE = true;
process.env.ASTRO_DB_REMOTE_URL = absoluteFileUrl.toString();
await fixture.build();
await initializeRemoteDb(fixture.config);
});
after(async () => {
delete process.env.ASTRO_INTERNAL_TEST_REMOTE;
delete process.env.ASTRO_DB_REMOTE_URL;
});
it('Can render page', async () => {
const app = await fixture.loadTestAdapterApp();
const request = new Request('http://example.com/');
const response = await app.render(request);
assert.equal(response.status, 200);
});
});
describe('build --remote with local libSQL file (relative path)', () => {
before(async () => {
clearEnvironment();
const absoluteFileUrl = new URL('./fixtures/libsql-remote/dist/relative.db', import.meta.url);
const prodDbPath = relative(
fileURLToPath(fixture.config.root),
fileURLToPath(absoluteFileUrl),
);
// Remove the file if it exists to avoid conflict between test runs
await rm(prodDbPath, { force: true });
process.env.ASTRO_INTERNAL_TEST_REMOTE = true;
process.env.ASTRO_DB_REMOTE_URL = `file:${prodDbPath}`;
await fixture.build();
await initializeRemoteDb(fixture.config);
});
after(async () => {
delete process.env.ASTRO_INTERNAL_TEST_REMOTE;
delete process.env.ASTRO_DB_REMOTE_URL;
});
it('Can render page', async () => {
const app = await fixture.loadTestAdapterApp();
const request = new Request('http://example.com/');
const response = await app.render(request);
assert.equal(response.status, 200);
});
});
});

View file

@ -64,6 +64,23 @@ export async function setupRemoteDbServer(astroConfig) {
}; };
} }
export async function initializeRemoteDb(astroConfig) {
await cli({
config: astroConfig,
flags: {
_: [undefined, 'astro', 'db', 'push'],
remote: true,
},
});
await cli({
config: astroConfig,
flags: {
_: [undefined, 'astro', 'db', 'execute', 'db/seed.ts'],
remote: true,
},
});
}
/** /**
* Clears the environment variables related to Astro DB and Astro Studio. * Clears the environment variables related to Astro DB and Astro Studio.
*/ */

9
pnpm-lock.yaml generated
View file

@ -4408,6 +4408,15 @@ importers:
specifier: workspace:* specifier: workspace:*
version: link:../../../../astro version: link:../../../../astro
packages/db/test/fixtures/libsql-remote:
dependencies:
'@astrojs/db':
specifier: workspace:*
version: link:../../..
astro:
specifier: workspace:*
version: link:../../../../astro
packages/db/test/fixtures/local-prod: packages/db/test/fixtures/local-prod:
dependencies: dependencies:
'@astrojs/db': '@astrojs/db':