principles-disciple 1.118.0 → 1.120.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -2,7 +2,7 @@
2
2
  "id": "principles-disciple",
3
3
  "name": "Principles Disciple",
4
4
  "description": "Evolutionary programming agent framework with strategic guardrails and reflection loops.",
5
- "version": "1.118.0",
5
+ "version": "1.120.0",
6
6
  "activation": {
7
7
  "onCapabilities": [
8
8
  "hook"
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "principles-disciple",
3
- "version": "1.118.0",
3
+ "version": "1.120.0",
4
4
  "description": "Native OpenClaw plugin for Principles Disciple",
5
5
  "type": "module",
6
6
  "main": "./dist/bundle.js",
@@ -262,4 +262,166 @@ describe('TrajectoryDatabase', () => {
262
262
  expect(secondStats.painEvents).toBe(1);
263
263
  reopened.dispose();
264
264
  });
265
+
266
+ describe('recordPainEvent with canonical_pain_id (PRI-406)', () => {
267
+ it('inserts pain event without canonical_pain_id successfully', () => {
268
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
269
+ const db = new TrajectoryDatabase({ workspaceDir });
270
+
271
+ const id = db.recordPainEvent({
272
+ sessionId: 's1',
273
+ source: 'test',
274
+ score: 50,
275
+ reason: 'test reason',
276
+ origin: 'test',
277
+ });
278
+
279
+ expect(id).toBeGreaterThan(0);
280
+ db.dispose();
281
+ });
282
+
283
+ it('inserts pain event with canonical_pain_id successfully', () => {
284
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
285
+ const db = new TrajectoryDatabase({ workspaceDir });
286
+
287
+ const id = db.recordPainEvent({
288
+ sessionId: 's1',
289
+ source: 'test',
290
+ score: 50,
291
+ reason: 'test reason',
292
+ origin: 'test',
293
+ canonicalPainId: 'pain-canonical-001',
294
+ });
295
+
296
+ expect(id).toBeGreaterThan(0);
297
+ db.dispose();
298
+ });
299
+
300
+ it('handles UNIQUE constraint violation on canonical_pain_id by updating instead of throwing', () => {
301
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
302
+ const db = new TrajectoryDatabase({ workspaceDir });
303
+
304
+ const canonicalId = 'pain-canonical-duplicate';
305
+
306
+ const id1 = db.recordPainEvent({
307
+ sessionId: 's1',
308
+ source: 'test',
309
+ score: 50,
310
+ reason: 'first',
311
+ origin: 'test',
312
+ canonicalPainId: canonicalId,
313
+ });
314
+ expect(id1).toBeGreaterThan(0);
315
+
316
+ const id2 = db.recordPainEvent({
317
+ sessionId: 's1',
318
+ source: 'test',
319
+ score: 60,
320
+ reason: 'second',
321
+ origin: 'test',
322
+ canonicalPainId: canonicalId,
323
+ });
324
+
325
+ expect(id2).toBe(id1);
326
+
327
+ db.dispose();
328
+ });
329
+
330
+ it('updates runtime_task_id when canonical_pain_id conflict occurs', () => {
331
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
332
+ const db = new TrajectoryDatabase({ workspaceDir });
333
+
334
+ const canonicalId = 'pain-canonical-update-rtid';
335
+
336
+ db.recordPainEvent({
337
+ sessionId: 's1',
338
+ source: 'test',
339
+ score: 50,
340
+ reason: 'no runtime task',
341
+ origin: 'test',
342
+ canonicalPainId: canonicalId,
343
+ });
344
+
345
+ db.recordPainEvent({
346
+ sessionId: 's1',
347
+ source: 'test',
348
+ score: 60,
349
+ reason: 'with runtime task',
350
+ origin: 'test',
351
+ canonicalPainId: canonicalId,
352
+ runtimeTaskId: 'task-123',
353
+ });
354
+
355
+ const queryResult = (db as any).db.prepare(
356
+ 'SELECT runtime_task_id FROM pain_events WHERE canonical_pain_id = ?'
357
+ ).get(canonicalId);
358
+
359
+ expect(queryResult.runtime_task_id).toBe('task-123');
360
+
361
+ db.dispose();
362
+ });
363
+
364
+ it('does not overwrite existing runtime_task_id when new one is null', () => {
365
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
366
+ const db = new TrajectoryDatabase({ workspaceDir });
367
+
368
+ const canonicalId = 'pain-canonical-coalesce';
369
+
370
+ db.recordPainEvent({
371
+ sessionId: 's1',
372
+ source: 'test',
373
+ score: 50,
374
+ reason: 'with runtime task',
375
+ origin: 'test',
376
+ canonicalPainId: canonicalId,
377
+ runtimeTaskId: 'original-task',
378
+ });
379
+
380
+ db.recordPainEvent({
381
+ sessionId: 's1',
382
+ source: 'test',
383
+ score: 60,
384
+ reason: 'without runtime task',
385
+ origin: 'test',
386
+ canonicalPainId: canonicalId,
387
+ });
388
+
389
+ const queryResult = (db as any).db.prepare(
390
+ 'SELECT runtime_task_id FROM pain_events WHERE canonical_pain_id = ?'
391
+ ).get(canonicalId);
392
+
393
+ expect(queryResult.runtime_task_id).toBe('original-task');
394
+
395
+ db.dispose();
396
+ });
397
+
398
+ it('throws for non-canonical_pain_id UNIQUE constraint violations', () => {
399
+ workspaceDir = fs.mkdtempSync(path.join(os.tmpdir(), 'pd-trajectory-'));
400
+ const db = new TrajectoryDatabase({ workspaceDir });
401
+
402
+ // Temporary unique index to simulate a non-canonical_pain_id conflict.
403
+ // Each test uses a fresh temp DB, so cleanup is not needed.
404
+ (db as any).db.exec('CREATE UNIQUE INDEX test_unique_source ON pain_events(source)');
405
+
406
+ db.recordPainEvent({
407
+ sessionId: 's1',
408
+ source: 'unique-source',
409
+ score: 50,
410
+ reason: 'first',
411
+ origin: 'test',
412
+ });
413
+
414
+ expect(() => {
415
+ db.recordPainEvent({
416
+ sessionId: 's1',
417
+ source: 'unique-source',
418
+ score: 60,
419
+ reason: 'second',
420
+ origin: 'test',
421
+ });
422
+ }).toThrow();
423
+
424
+ db.dispose();
425
+ });
426
+ });
265
427
  });