PR #172 fixed the jsonb double-encoding for `tool_allowlist` but the same class of bug, and the same re-derived workaround, remained elsewhere. 1. model_config (agent roles): jsonbObject still used the buggy `::jsonb` bind, so `ai_agent_roles.model_config` round-tripped as a jsonb STRING SCALAR. The read-path `typeof === 'object'` check then failed and the model override was SILENTLY dropped (role fell back to the default model). Fixed to `::text::jsonb` and added `parseModelConfig` + `normalizeRow` so every read self-heals already-corrupted rows (no migration). 2. Centralized the write workaround as `jsonbBind()` in database/utils.ts — one implementation with one explanation of the quirk — replacing the per-repo `jsonbArray` (mcp) and `jsonbObject` (roles). 3. Integration coverage (the fix is a DB round-trip a unit test cannot see; the read-side parser MASKS a write regression): new ai-mcp-server-repo.int-spec asserts `jsonb_typeof(tool_allowlist)='array'` after insert + heals a seeded string-scalar row; ai-agent-roles-repo int-spec gains the same for `model_config` (`'object'` + heal). 4. Updated the stale `ai-mcp-servers.types.ts` comment (the driver returns a JSON string for legacy rows; the repo normalizes every read). 5. Fail-open logging: a corrupt tool_allowlist degrades to "no restriction" (agent gets ALL tools) — normalizeRow now warns (server id only, never contents) so the silent widening leaves a trace. 6. Simplified parseToolAllowlist (normalize the string once, then a single array-of-strings check) — identical behaviour, all 12 cases still pass. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -1,4 +1,5 @@
|
||||
import { Kysely } from 'kysely';
|
||||
import { Kysely, sql } from 'kysely';
|
||||
import { randomUUID } from 'node:crypto';
|
||||
import { AiAgentRoleRepo } from '@docmost/db/repos/ai-agent-roles/ai-agent-roles.repo';
|
||||
import { getTestDb, destroyTestDb, createWorkspace } from './db';
|
||||
|
||||
@@ -25,8 +26,16 @@ describe('AiAgentRoleRepo isolation + partial unique index [integration]', () =>
|
||||
});
|
||||
|
||||
it('findById / listByWorkspace exclude soft-deleted rows', async () => {
|
||||
const live = await repo.insert({ workspaceId: w1, name: 'Live', instructions: 'x' });
|
||||
const dead = await repo.insert({ workspaceId: w1, name: 'Dead', instructions: 'x' });
|
||||
const live = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: 'Live',
|
||||
instructions: 'x',
|
||||
});
|
||||
const dead = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: 'Dead',
|
||||
instructions: 'x',
|
||||
});
|
||||
await repo.softDelete(dead.id, w1);
|
||||
|
||||
expect(await repo.findById(live.id, w1)).toBeDefined();
|
||||
@@ -38,7 +47,11 @@ describe('AiAgentRoleRepo isolation + partial unique index [integration]', () =>
|
||||
});
|
||||
|
||||
it('findById of a W2 role from W1 context returns undefined (tenant isolation)', async () => {
|
||||
const w2role = await repo.insert({ workspaceId: w2, name: 'W2Role', instructions: 'x' });
|
||||
const w2role = await repo.insert({
|
||||
workspaceId: w2,
|
||||
name: 'W2Role',
|
||||
instructions: 'x',
|
||||
});
|
||||
|
||||
expect(await repo.findById(w2role.id, w2)).toBeDefined();
|
||||
// Same id, wrong workspace context -> not visible.
|
||||
@@ -58,21 +71,100 @@ describe('AiAgentRoleRepo isolation + partial unique index [integration]', () =>
|
||||
});
|
||||
|
||||
it('same name is reusable after softDelete (partial unique index WHERE deleted_at IS NULL)', async () => {
|
||||
const first = await repo.insert({ workspaceId: w1, name: 'Reusable', instructions: 'x' });
|
||||
const first = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: 'Reusable',
|
||||
instructions: 'x',
|
||||
});
|
||||
await repo.softDelete(first.id, w1);
|
||||
|
||||
// Now inserting the same name must succeed because the soft-deleted row is
|
||||
// excluded from the partial unique index.
|
||||
const second = await repo.insert({ workspaceId: w1, name: 'Reusable', instructions: 'x' });
|
||||
const second = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: 'Reusable',
|
||||
instructions: 'x',
|
||||
});
|
||||
expect(second.id).toBeDefined();
|
||||
expect(second.id).not.toBe(first.id);
|
||||
});
|
||||
|
||||
it('same name in W1 and W2 is allowed (unique is per-workspace)', async () => {
|
||||
const a = await repo.insert({ workspaceId: w1, name: 'CrossTenant', instructions: 'x' });
|
||||
const b = await repo.insert({ workspaceId: w2, name: 'CrossTenant', instructions: 'x' });
|
||||
const a = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: 'CrossTenant',
|
||||
instructions: 'x',
|
||||
});
|
||||
const b = await repo.insert({
|
||||
workspaceId: w2,
|
||||
name: 'CrossTenant',
|
||||
instructions: 'x',
|
||||
});
|
||||
expect(a.id).toBeDefined();
|
||||
expect(b.id).toBeDefined();
|
||||
expect(a.id).not.toBe(b.id);
|
||||
});
|
||||
|
||||
// model_config jsonb round-trip (issue #173 §1): the same double-encoding bug
|
||||
// PR #172 fixed for tool_allowlist lived in jsonbObject. A DB round-trip is the
|
||||
// only way to observe it — the write must land as a real jsonb OBJECT, and a
|
||||
// legacy string-scalar row must self-heal on read (else the model override is
|
||||
// silently dropped and the role falls back to the default model).
|
||||
const jsonbTypeof = async (id: string): Promise<string | null> => {
|
||||
const res = await sql<{ t: string | null }>`
|
||||
SELECT jsonb_typeof(model_config) AS t
|
||||
FROM ai_agent_roles WHERE id = ${id}
|
||||
`.execute(db);
|
||||
return res.rows[0]?.t ?? null;
|
||||
};
|
||||
|
||||
it('insert stores model_config as a jsonb OBJECT and reads it back as an object', async () => {
|
||||
const role = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: `Model-${randomUUID()}`,
|
||||
instructions: 'x',
|
||||
modelConfig: { driver: 'gemini', chatModel: 'gemini-2.0-flash' },
|
||||
});
|
||||
expect(await jsonbTypeof(role.id)).toBe('object');
|
||||
// The returned row is already normalized to an object.
|
||||
expect(role.modelConfig).toEqual({
|
||||
driver: 'gemini',
|
||||
chatModel: 'gemini-2.0-flash',
|
||||
});
|
||||
const found = await repo.findById(role.id, w1);
|
||||
expect(found?.modelConfig).toEqual({
|
||||
driver: 'gemini',
|
||||
chatModel: 'gemini-2.0-flash',
|
||||
});
|
||||
});
|
||||
|
||||
it('an empty model_config is normalized to null (no override)', async () => {
|
||||
const role = await repo.insert({
|
||||
workspaceId: w1,
|
||||
name: `Empty-${randomUUID()}`,
|
||||
instructions: 'x',
|
||||
modelConfig: {},
|
||||
});
|
||||
// The column is SQL NULL, so jsonb_typeof returns SQL NULL (JS null).
|
||||
expect(await jsonbTypeof(role.id)).toBeNull();
|
||||
expect((await repo.findById(role.id, w1))?.modelConfig).toBeNull();
|
||||
});
|
||||
|
||||
it('repairs a legacy double-encoded (string scalar) model_config on read', async () => {
|
||||
const id = randomUUID();
|
||||
// Seed the corrupt string-scalar shape the old `::jsonb` bind produced.
|
||||
await sql`
|
||||
INSERT INTO ai_agent_roles (id, workspace_id, name, instructions, model_config)
|
||||
VALUES (
|
||||
${id}, ${w1}, ${`Legacy-${id}`}, 'x',
|
||||
to_jsonb(${'{"driver":"openai","chatModel":"gpt"}'}::text)
|
||||
)
|
||||
`.execute(db);
|
||||
expect(await jsonbTypeof(id)).toBe('string'); // sanity: really corrupt
|
||||
|
||||
expect((await repo.findById(id, w1))?.modelConfig).toEqual({
|
||||
driver: 'openai',
|
||||
chatModel: 'gpt',
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
94
apps/server/test/integration/ai-mcp-server-repo.int-spec.ts
Normal file
94
apps/server/test/integration/ai-mcp-server-repo.int-spec.ts
Normal file
@@ -0,0 +1,94 @@
|
||||
import { Kysely, sql } from 'kysely';
|
||||
import { randomUUID } from 'node:crypto';
|
||||
import { AiMcpServerRepo } from '@docmost/db/repos/ai-chat/ai-mcp-server.repo';
|
||||
import { getTestDb, destroyTestDb, createWorkspace } from './db';
|
||||
|
||||
/**
|
||||
* AiMcpServerRepo `tool_allowlist` jsonb round-trip (PR #172 / issue #173 §3).
|
||||
*
|
||||
* The fix under test is a DB round-trip, so a unit test cannot observe it: the
|
||||
* write must land as a real jsonb ARRAY (not a double-encoded string scalar),
|
||||
* and the read must repair any legacy string-scalar rows. The read-side
|
||||
* `parseToolAllowlist` MASKS a write regression (it parses the string back), so
|
||||
* without this integration check, reverting `::text::jsonb` to `::jsonb` would
|
||||
* keep every unit test green while silently corrupting the column again.
|
||||
*/
|
||||
describe('AiMcpServerRepo tool_allowlist jsonb round-trip [integration]', () => {
|
||||
let db: Kysely<any>;
|
||||
let repo: AiMcpServerRepo;
|
||||
let ws: string;
|
||||
|
||||
beforeAll(async () => {
|
||||
db = getTestDb();
|
||||
repo = new AiMcpServerRepo(db as any);
|
||||
ws = (await createWorkspace(db)).id;
|
||||
});
|
||||
|
||||
afterAll(async () => {
|
||||
await destroyTestDb();
|
||||
});
|
||||
|
||||
const jsonbTypeof = async (id: string): Promise<string | null> => {
|
||||
const res = await sql<{ t: string | null }>`
|
||||
SELECT jsonb_typeof(tool_allowlist) AS t
|
||||
FROM ai_mcp_servers WHERE id = ${id}
|
||||
`.execute(db);
|
||||
return res.rows[0]?.t ?? null;
|
||||
};
|
||||
|
||||
it('insert stores the allowlist as a jsonb ARRAY (not a string scalar)', async () => {
|
||||
const row = await repo.insert({
|
||||
workspaceId: ws,
|
||||
name: `srv-${randomUUID()}`,
|
||||
transport: 'http',
|
||||
url: 'https://example.com/mcp',
|
||||
toolAllowlist: ['search', 'crawl'],
|
||||
});
|
||||
|
||||
// The column holds a real jsonb array — the whole point of ::text::jsonb.
|
||||
expect(await jsonbTypeof(row.id)).toBe('array');
|
||||
|
||||
// And the read returns a genuine string[], not a JSON string.
|
||||
const found = await repo.findById(row.id, ws);
|
||||
expect(found?.toolAllowlist).toEqual(['search', 'crawl']);
|
||||
expect(Array.isArray(found?.toolAllowlist)).toBe(true);
|
||||
});
|
||||
|
||||
it('an empty allowlist is normalized to null (no restriction), not []', async () => {
|
||||
const row = await repo.insert({
|
||||
workspaceId: ws,
|
||||
name: `srv-${randomUUID()}`,
|
||||
transport: 'http',
|
||||
url: 'https://example.com/mcp',
|
||||
toolAllowlist: [],
|
||||
});
|
||||
// The column is SQL NULL, so jsonb_typeof returns SQL NULL (JS null).
|
||||
expect(await jsonbTypeof(row.id)).toBeNull();
|
||||
expect((await repo.findById(row.id, ws))?.toolAllowlist).toBeNull();
|
||||
});
|
||||
|
||||
it('repairs a legacy double-encoded (string scalar) row on read (self-heal)', async () => {
|
||||
// Seed a row whose tool_allowlist is a jsonb STRING SCALAR holding the JSON
|
||||
// text — exactly what the old `::jsonb` double-encoding produced.
|
||||
const id = randomUUID();
|
||||
await sql`
|
||||
INSERT INTO ai_mcp_servers (id, workspace_id, name, transport, url, tool_allowlist)
|
||||
VALUES (
|
||||
${id}, ${ws}, ${`srv-${id}`}, 'http', 'https://example.com/mcp',
|
||||
to_jsonb(${'["alpha","beta"]'}::text)
|
||||
)
|
||||
`.execute(db);
|
||||
|
||||
// Sanity: the seeded column really IS the corrupt string-scalar shape.
|
||||
expect(await jsonbTypeof(id)).toBe('string');
|
||||
|
||||
// The repo read heals it back to a real string[].
|
||||
expect((await repo.findById(id, ws))?.toolAllowlist).toEqual([
|
||||
'alpha',
|
||||
'beta',
|
||||
]);
|
||||
const enabled = await repo.listEnabled(ws);
|
||||
const healed = enabled.find((r) => r.id === id);
|
||||
expect(healed?.toolAllowlist).toEqual(['alpha', 'beta']);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user