2024-03-22 08:28:12 -04:00
|
|
|
import { createServer } from 'node:http';
|
2024-05-13 12:20:43 -04:00
|
|
|
import { LibsqlError, createClient } from '@libsql/client';
|
2024-03-22 08:28:12 -04:00
|
|
|
import { z } from 'zod';
|
|
|
|
import { cli } from '../dist/core/cli/index.js';
|
|
|
|
import { resolveDbConfig } from '../dist/core/load-file.js';
|
2024-05-03 11:08:50 -04:00
|
|
|
import { getCreateIndexQueries, getCreateTableQuery } from '../dist/core/queries.js';
|
2024-03-22 08:28:12 -04:00
|
|
|
|
|
|
|
const singleQuerySchema = z.object({
|
|
|
|
sql: z.string(),
|
|
|
|
args: z.array(z.any()).or(z.record(z.string(), z.any())),
|
|
|
|
});
|
|
|
|
|
|
|
|
const querySchema = singleQuerySchema.or(z.array(singleQuerySchema));
|
|
|
|
|
2024-04-02 16:07:18 -04:00
|
|
|
let portIncrementer = 8030;
|
2024-03-22 08:28:12 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {import('astro').AstroConfig} astroConfig
|
|
|
|
* @param {number | undefined} port
|
|
|
|
*/
|
|
|
|
export async function setupRemoteDbServer(astroConfig) {
|
|
|
|
const port = portIncrementer++;
|
|
|
|
process.env.ASTRO_STUDIO_REMOTE_DB_URL = `http://localhost:${port}`;
|
|
|
|
process.env.ASTRO_INTERNAL_TEST_REMOTE = true;
|
|
|
|
const server = createRemoteDbServer().listen(port);
|
|
|
|
|
|
|
|
const { dbConfig } = await resolveDbConfig(astroConfig);
|
|
|
|
const setupQueries = [];
|
|
|
|
for (const [name, table] of Object.entries(dbConfig?.tables ?? {})) {
|
|
|
|
const createQuery = getCreateTableQuery(name, table);
|
|
|
|
const indexQueries = getCreateIndexQueries(name, table);
|
|
|
|
setupQueries.push(createQuery, ...indexQueries);
|
|
|
|
}
|
|
|
|
await fetch(`http://localhost:${port}/db/query`, {
|
|
|
|
method: 'POST',
|
|
|
|
body: JSON.stringify(setupQueries.map((sql) => ({ sql, args: [] }))),
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
await cli({
|
|
|
|
config: astroConfig,
|
|
|
|
flags: {
|
|
|
|
_: [undefined, 'astro', 'db', 'execute', 'db/seed.ts'],
|
|
|
|
remote: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
server,
|
|
|
|
async stop() {
|
|
|
|
delete process.env.ASTRO_STUDIO_REMOTE_DB_URL;
|
|
|
|
delete process.env.ASTRO_INTERNAL_TEST_REMOTE;
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
server.close((err) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else {
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-09-03 09:53:29 -03:00
|
|
|
/**
|
|
|
|
* Clears the environment variables related to Astro DB and Astro Studio.
|
|
|
|
*/
|
|
|
|
export function clearEnvironment() {
|
|
|
|
const keys = Array.from(Object.keys(process.env));
|
|
|
|
for (const key of keys) {
|
|
|
|
if (key.startsWith('ASTRO_DB_') || key.startsWith('ASTRO_STUDIO_')) {
|
|
|
|
delete process.env[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-22 08:28:12 -04:00
|
|
|
function createRemoteDbServer() {
|
|
|
|
const dbClient = createClient({
|
|
|
|
url: ':memory:',
|
|
|
|
});
|
|
|
|
const server = createServer((req, res) => {
|
|
|
|
if (
|
|
|
|
!req.url.startsWith('/db/query') ||
|
|
|
|
req.method !== 'POST' ||
|
|
|
|
req.headers['content-type'] !== 'application/json'
|
|
|
|
) {
|
|
|
|
res.writeHead(404, { 'Content-Type': 'application/json' });
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
success: false,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const rawBody = [];
|
|
|
|
req.on('data', (chunk) => {
|
|
|
|
rawBody.push(chunk);
|
|
|
|
});
|
|
|
|
req.on('end', async () => {
|
|
|
|
let json;
|
|
|
|
try {
|
|
|
|
json = JSON.parse(Buffer.concat(rawBody).toString());
|
2024-08-08 05:54:04 -04:00
|
|
|
} catch {
|
2024-03-22 08:28:12 -04:00
|
|
|
applyParseError(res);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const parsed = querySchema.safeParse(json);
|
|
|
|
if (parsed.success === false) {
|
|
|
|
applyParseError(res);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const body = parsed.data;
|
|
|
|
try {
|
|
|
|
const result = Array.isArray(body)
|
|
|
|
? await dbClient.batch(body)
|
|
|
|
: await dbClient.execute(body);
|
|
|
|
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
|
|
res.end(JSON.stringify(result));
|
|
|
|
} catch (e) {
|
|
|
|
res.writeHead(500, { 'Content-Type': 'application/json' });
|
|
|
|
res.statusMessage = e.message;
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
success: false,
|
2024-05-13 12:20:43 -04:00
|
|
|
error: {
|
|
|
|
code: e instanceof LibsqlError ? e.code : 'SQLITE_QUERY_FAILED',
|
|
|
|
details: e.message,
|
2024-05-13 16:21:32 +00:00
|
|
|
},
|
2024-03-22 08:28:12 -04:00
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
server.on('close', () => {
|
|
|
|
dbClient.close();
|
|
|
|
});
|
|
|
|
|
|
|
|
return server;
|
|
|
|
}
|
|
|
|
|
|
|
|
function applyParseError(res) {
|
|
|
|
res.writeHead(400, { 'Content-Type': 'application/json' });
|
|
|
|
res.statusMessage = 'Invalid request body';
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
// Use JSON response with `success: boolean` property
|
|
|
|
// to match remote error responses.
|
|
|
|
success: false,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|