Schema: - records.overrides JSONB column (ALTER TABLE, already applied) - apply_transformations merges overrides on top: data || rules || overrides - generate_source_view always includes id and _overridden columns - set_record_overrides(id, overrides): stores and immediately merges into transformed - clear_record_overrides(id): clears overrides then reprocesses record API: - PUT /records/:id/overrides — set overrides - DELETE /records/:id/overrides — clear and reprocess UI (Records page): - Rows are clickable; overridden rows highlighted amber - Side panel shows all transformed fields as editable inputs - Overridden fields highlighted amber with pencil indicator - Save stores overrides; Clear removes them and restores computed values - id and _overridden hidden from table display Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
109 lines
4.1 KiB
JavaScript
109 lines
4.1 KiB
JavaScript
/**
|
|
* Records Routes
|
|
* Query and manage imported records
|
|
*/
|
|
|
|
const express = require('express');
|
|
const { lit } = require('../lib/sql');
|
|
|
|
module.exports = (pool) => {
|
|
const router = express.Router();
|
|
|
|
// List records for a source
|
|
router.get('/source/:source_name', async (req, res, next) => {
|
|
try {
|
|
const { limit = 100, offset = 0, transformed_only } = req.query;
|
|
const result = await pool.query(
|
|
`SELECT * FROM list_records(${lit(req.params.source_name)}, ${lit(parseInt(limit))}, ${lit(parseInt(offset))}, ${lit(transformed_only === 'true')})`
|
|
);
|
|
res.json(result.rows);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Get single record
|
|
router.get('/:id', async (req, res, next) => {
|
|
try {
|
|
const result = await pool.query(`SELECT * FROM get_record(${lit(parseInt(req.params.id))})`);
|
|
if (result.rows.length === 0) return res.status(404).json({ error: 'Record not found' });
|
|
res.json(result.rows[0]);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Search records
|
|
router.post('/search', async (req, res, next) => {
|
|
try {
|
|
const { source_name, query, limit = 100 } = req.body;
|
|
if (!source_name || !query) {
|
|
return res.status(400).json({ error: 'Missing required fields: source_name, query' });
|
|
}
|
|
const result = await pool.query(
|
|
`SELECT * FROM search_records(${lit(source_name)}, ${lit(query)}, ${lit(parseInt(limit))})`
|
|
);
|
|
res.json(result.rows);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Set overrides for a record and immediately merge into transformed
|
|
router.put('/:id/overrides', async (req, res, next) => {
|
|
try {
|
|
const { overrides } = req.body;
|
|
if (!overrides || typeof overrides !== 'object')
|
|
return res.status(400).json({ error: 'overrides object required' });
|
|
const result = await pool.query(
|
|
`SELECT * FROM set_record_overrides(${lit(parseInt(req.params.id))}, ${lit(overrides)})`
|
|
);
|
|
if (result.rows.length === 0) return res.status(404).json({ error: 'Record not found' });
|
|
res.json(result.rows[0]);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Clear overrides and reprocess that record to restore computed values
|
|
router.delete('/:id/overrides', async (req, res, next) => {
|
|
try {
|
|
const rec = await pool.query(
|
|
`SELECT * FROM clear_record_overrides(${lit(parseInt(req.params.id))})`
|
|
);
|
|
if (rec.rows.length === 0) return res.status(404).json({ error: 'Record not found' });
|
|
// Reprocess this record so transformed reflects rules/mappings without overrides
|
|
await pool.query(
|
|
`SELECT apply_transformations(${lit(rec.rows[0].source_name)}, ARRAY[${lit(parseInt(req.params.id))}::int], true)`
|
|
);
|
|
const updated = await pool.query(`SELECT * FROM get_record(${lit(parseInt(req.params.id))})`);
|
|
res.json(updated.rows[0]);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Delete record
|
|
router.delete('/:id', async (req, res, next) => {
|
|
try {
|
|
const result = await pool.query(`SELECT * FROM delete_record(${lit(parseInt(req.params.id))})`);
|
|
if (result.rows.length === 0) return res.status(404).json({ error: 'Record not found' });
|
|
res.json({ success: true, deleted: result.rows[0].id });
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
// Delete all records for a source
|
|
router.delete('/source/:source_name/all', async (req, res, next) => {
|
|
try {
|
|
const result = await pool.query(`SELECT * FROM delete_source_records(${lit(req.params.source_name)})`);
|
|
res.json({ success: true, deleted_count: result.rows[0].deleted_count });
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
});
|
|
|
|
return router;
|
|
};
|