feat(repos): resources (+ deps + creds) and source_docs

This commit is contained in:
root
2026-05-31 02:19:23 +10:00
parent 99d64221a0
commit c8649d753f
4 changed files with 198 additions and 0 deletions

94
lib/db/repos/resources.js Normal file
View File

@@ -0,0 +1,94 @@
import { pool } from '../pool.js';
import { recordAudit } from './audit_stub.js';
const FIELDS = ['space_id','slug','name','runtime_type','host','url','version','status','monitoring','metadata','last_check','maintenance_until'];
export async function create(input, actor) {
const cols = [], vals = [], ph = [];
let i = 1;
for (const f of FIELDS) {
if (input[f] !== undefined) { cols.push(f); vals.push(input[f]); ph.push(`$${i++}`); }
}
const { rows: [r] } = await pool.query(
`INSERT INTO resources(${cols.join(',')}) VALUES(${ph.join(',')}) RETURNING *`,
vals
);
await recordAudit(actor, 'create', 'resource', r.id, null, r);
return r;
}
export async function getById(id) {
const { rows: [r] } = await pool.query(`SELECT * FROM resources WHERE id=$1`, [id]);
return r;
}
export async function listBySpace(space_id) {
const { rows } = await pool.query(
`SELECT * FROM resources WHERE space_id=$1 ORDER BY name`, [space_id]
);
return rows;
}
export async function update(id, patch, actor) {
const before = await getById(id);
const sets = [], vals = [];
let i = 1;
for (const f of FIELDS) {
if (patch[f] !== undefined) { sets.push(`${f}=$${i++}`); vals.push(patch[f]); }
}
sets.push(`updated_at=now()`);
vals.push(id);
const { rows: [r] } = await pool.query(
`UPDATE resources SET ${sets.join(', ')} WHERE id=$${i} RETURNING *`,
vals
);
await recordAudit(actor, 'update', 'resource', id, before, r);
return r;
}
export async function del(id, actor) {
const before = await getById(id);
await pool.query(`DELETE FROM resources WHERE id=$1`, [id]);
await recordAudit(actor, 'delete', 'resource', id, before, null);
}
export async function addDependency(resource_id, depends_on, kind) {
if (resource_id === depends_on) throw new Error('resource cannot depend on itself');
await pool.query(
`INSERT INTO resource_dependencies(resource_id, depends_on, kind)
VALUES($1,$2,$3) ON CONFLICT DO NOTHING`,
[resource_id, depends_on, kind || null]
);
}
export async function removeDependency(resource_id, depends_on) {
await pool.query(
`DELETE FROM resource_dependencies WHERE resource_id=$1 AND depends_on=$2`,
[resource_id, depends_on]
);
}
export async function listDependencies(resource_id) {
const { rows } = await pool.query(
`SELECT * FROM resource_dependencies WHERE resource_id=$1`, [resource_id]
);
return rows;
}
export async function addCredential(resource_id, { label, vault_path, kind, notes }) {
const { rows: [r] } = await pool.query(
`INSERT INTO resource_credentials(resource_id, label, vault_path, kind, notes)
VALUES($1,$2,$3,$4,$5) RETURNING *`,
[resource_id, label, vault_path, kind || null, notes || null]
);
return r;
}
export async function listCredentials(resource_id) {
const { rows } = await pool.query(
`SELECT id, resource_id, label, vault_path, kind, notes, created_at
FROM resource_credentials WHERE resource_id=$1 ORDER BY label`,
[resource_id]
);
return rows;
}

View File

@@ -0,0 +1,53 @@
import { pool } from '../pool.js';
import { recordAudit } from './audit_stub.js';
const FIELDS = ['resource_id','name','upstream_url','version','format','sync_source','local_path','body_text','embedding','last_synced','metadata'];
export async function create(input, actor) {
const cols = [], vals = [], ph = [];
let i = 1;
for (const f of FIELDS) {
if (input[f] !== undefined) { cols.push(f); vals.push(input[f]); ph.push(`$${i++}`); }
}
const { rows: [r] } = await pool.query(
`INSERT INTO source_docs(${cols.join(',')}) VALUES(${ph.join(',')}) RETURNING *`,
vals
);
await recordAudit(actor, 'create', 'source_doc', r.id, null, r);
return r;
}
export async function getById(id) {
const { rows: [r] } = await pool.query(`SELECT * FROM source_docs WHERE id=$1`, [id]);
return r;
}
export async function listByResource(resource_id) {
const { rows } = await pool.query(
`SELECT * FROM source_docs WHERE resource_id=$1 ORDER BY name`, [resource_id]
);
return rows;
}
export async function update(id, patch, actor) {
const before = await getById(id);
const sets = [], vals = [];
let i = 1;
for (const f of FIELDS) {
if (patch[f] !== undefined) { sets.push(`${f}=$${i++}`); vals.push(patch[f]); }
}
sets.push(`updated_at=now()`);
vals.push(id);
const { rows: [r] } = await pool.query(
`UPDATE source_docs SET ${sets.join(', ')} WHERE id=$${i} RETURNING *`,
vals
);
await recordAudit(actor, 'update', 'source_doc', id, before, r);
return r;
}
export async function del(id, actor) {
const before = await getById(id);
await pool.query(`DELETE FROM source_docs WHERE id=$1`, [id]);
await recordAudit(actor, 'delete', 'source_doc', id, before, null);
}