From aae093904c9d1bc57489f5a5f093f1ee2ad4cd3a Mon Sep 17 00:00:00 2001 From: OpenClaw Bot Date: Wed, 27 May 2026 23:34:45 +0200 Subject: [PATCH] Accept reader-friendly Scattermind build order aliases --- scripts/check-rank-feedback.mjs | 57 ++++++++++++++++++++++++++++++++- server.js | 14 ++++---- 2 files changed, 63 insertions(+), 8 deletions(-) diff --git a/scripts/check-rank-feedback.mjs b/scripts/check-rank-feedback.mjs index bd0e084..4e7bd7f 100644 --- a/scripts/check-rank-feedback.mjs +++ b/scripts/check-rank-feedback.mjs @@ -2313,6 +2313,61 @@ try { assert.equal(scalarBuildOrderSections.handoff.readiness.status, 'needs-source-context'); assert.ok(scalarBuildOrderSections.handoff.warnings.includes('missing evidence needed for active item feature-1')); + const snakeReaderFriendlyBuildOrderResponse = await fetch(`${base}/api/rank-feedback`, { + method: 'POST', + headers: { 'Content-Type': 'application/json' }, + body: JSON.stringify({ + sourceName: 'Scattermind', + referenceCode: 'SM-SNAKE-READER-BUILD-ORDER-1', + working_name: 'Reader-friendly snake Build Order', + ideaText: 'Scattermind exported snake_case Build Order keys and visible labels like Test this manually / Do not build this yet.', + context: 'Solo builder. Manual proof first. Avoid account dashboards and saved workspaces until one user acts.', + mode: 'mvp', + conceptMap: { + build_order_preview: { + build_first: [{ id: 'snake-source-preview', action: 'Snake-case source preview', evidence_needed: 'Can the snake_case build order still name one active source-traced move?', source_item_id: 'snake-build-1', source_title: 'Reader-friendly Build Order', ranker_hints: { value: 9, effort: 2, confidence: 8, urgency: 8, risk: 2 } }], + test_next: [{ id: 'snake-manual-test', action: 'Snake-case manual test', evidence_needed: 'Does the user know what signal to collect?', source_item_id: 'snake-test-1', source_title: 'Reader-friendly Build Order' }], + do_not_build_yet: [{ id: 'snake-saved-workspace', action: 'Snake-case saved workspace dashboard', evidence_needed: 'Not before proof.', source_item_id: 'snake-defer-1', source_title: 'Reader-friendly Build Order' }], + probably_noise: [{ id: 'snake-billing-admin', action: 'Snake-case billing and admin layer', evidence_needed: 'Not part of the bridge proof.', source_item_id: 'snake-park-1', source_title: 'Reader-friendly Build Order' }], + }, + lenses: { + channel: { title: 'Build Order', content: 'Build first: Source preview from Concept Map. Test this manually: Ask one tired user to explain the first move. Do not build this yet: saved workspace dashboard. Probably noise: billing admin before proof.' }, + }, + }, + }), + }); + assert.equal(snakeReaderFriendlyBuildOrderResponse.status, 200); + const snakeReaderFriendlyBuildOrder = await snakeReaderFriendlyBuildOrderResponse.json(); + assert.equal(snakeReaderFriendlyBuildOrder.input.optionCount, 4, 'snake_case Build Order keys should beat falling back to reader-facing lens text'); + assert.equal(snakeReaderFriendlyBuildOrder.ranked[0].id, 'snake-source-preview'); + assert.equal(snakeReaderFriendlyBuildOrder.ranked[0].provenance.sourceSection, 'concept-map.buildOrderPreview.doFirst'); + assert.equal(snakeReaderFriendlyBuildOrder.ranked.find(item => item.id === 'snake-manual-test').lane.id, 'test'); + assert.equal(snakeReaderFriendlyBuildOrder.ranked.find(item => item.id === 'snake-saved-workspace').lane.id, 'defer'); + assert.equal(snakeReaderFriendlyBuildOrder.ranked.find(item => item.id === 'snake-billing-admin').lane.id, 'park'); + assert.equal(snakeReaderFriendlyBuildOrder.handoff.readiness.status, 'ready'); + assert.deepEqual(snakeReaderFriendlyBuildOrder.handoff.warnings, []); + + const readerFriendlyLensLabelResponse = await fetch(`${base}/api/rank-feedback`, { + method: 'POST', + headers: { 'Content-Type': 'application/json' }, + body: JSON.stringify({ + sourceName: 'Scattermind', + referenceCode: 'SM-THIS-LABELS-1', + working_name: 'This-label Build Order', + ideaText: 'A paid Concept Map was copied as visible Build Order prose only.', + mode: 'mvp', + lenses: { + channel: 'Build first: One-page source receipt. Test this manually: Ask one tired user what to do next. Do not build this yet: saved account workspace. Probably noise: subscription dashboard before proof.', + }, + }), + }); + assert.equal(readerFriendlyLensLabelResponse.status, 200); + const readerFriendlyLensLabel = await readerFriendlyLensLabelResponse.json(); + assert.equal(readerFriendlyLensLabel.input.optionCount, 4); + assert.equal(readerFriendlyLensLabel.ranked.find(item => item.id === 'build-order-2').lane.id, 'test'); + assert.equal(readerFriendlyLensLabel.ranked.find(item => item.id === 'build-order-3').lane.id, 'defer'); + assert.equal(readerFriendlyLensLabel.ranked.find(item => item.id === 'build-order-4').lane.id, 'park'); + const gameRouteGuardrailResponse = await fetch(`${base}/api/rank-feedback`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, @@ -2350,7 +2405,7 @@ try { assert.equal(gameRouteGuardrail.handoff.readiness.status, 'ready'); assert.deepEqual(gameRouteGuardrail.handoff.warnings, []); - console.log(JSON.stringify({ ok: true, top: data.ranked[0].id, hintedTop: hinted.ranked[0].id, actionTop: actions.ranked[0].id, nestedConceptTop: nestedConcept.ranked[0].id, nonGoalTop: nonGoal.ranked[0].id, structuredContextTop: structuredContext.ranked[0].id, lensOnlyTop: lensOnly.ranked[0].id, scattermindPaidShapeTop: scattermindPaidShape.ranked[0].id, mergedContextTop: mergedContext.ranked[0].id, embeddedJsonTop: embeddedJson.ranked[0].id, fencedJsonTop: fencedJson.ranked[0].id, embeddedSnapshotTop: embeddedSnapshot.ranked[0].id, sourceExcerptTop: sourceExcerpt.ranked[0].id, snakeCaseBridgeTop: snakeCaseBridge.ranked[0].id, nextStepsAliasTop: nextStepsAlias.ranked[0].id, summaryGuardrailTop: summaryGuardrail.ranked[0].id, bridgeEnvelopeTop: bridgeEnvelope.ranked[0].id, directEnvelopeSectionsTop: directEnvelopeSections.ranked[0].id, softDirectLaneAliasesTop: softDirectLaneAliases.ranked[0].id, threadsFallbackTop: threadsFallback.ranked[0].id, questionsFallbackTop: questionsFallback.ranked[0].id, freeSnapshotTop: freeSnapshot.ranked[0].id, storedScattermindRowTop: storedScattermindRow.ranked[0].id, candidateActionsAliasTop: candidateActionsAlias.ranked[0].id, rankReadyActionsEnvelopeTop: rankReadyActionsEnvelope.ranked[0].id, continuationEnvelopeTop: continuationEnvelope.ranked[0].id, buildOrderPreviewTop: buildOrderPreview.ranked[0].id, scattermindRoadmapLanguageTop: scattermindRoadmapLanguage.ranked[0].id, scattermindRoadmapLensOnlyTop: scattermindRoadmapLensOnly.ranked[0].id, stringifiedRankerInputTop: stringifiedRankerInput.ranked[0].id, scalarBuildOrderSectionsTop: scalarBuildOrderSections.ranked[0].id, gameRouteGuardrailTop: gameRouteGuardrail.ranked[0].id, duplicateIds: duplicateIds.ranked.map(item => item.id), readiness: data.handoff.readiness.status, provenance: data.input.provenance, buildOrder: data.buildOrder }, null, 2)); + console.log(JSON.stringify({ ok: true, top: data.ranked[0].id, hintedTop: hinted.ranked[0].id, actionTop: actions.ranked[0].id, nestedConceptTop: nestedConcept.ranked[0].id, nonGoalTop: nonGoal.ranked[0].id, structuredContextTop: structuredContext.ranked[0].id, lensOnlyTop: lensOnly.ranked[0].id, scattermindPaidShapeTop: scattermindPaidShape.ranked[0].id, mergedContextTop: mergedContext.ranked[0].id, embeddedJsonTop: embeddedJson.ranked[0].id, fencedJsonTop: fencedJson.ranked[0].id, embeddedSnapshotTop: embeddedSnapshot.ranked[0].id, sourceExcerptTop: sourceExcerpt.ranked[0].id, snakeCaseBridgeTop: snakeCaseBridge.ranked[0].id, nextStepsAliasTop: nextStepsAlias.ranked[0].id, summaryGuardrailTop: summaryGuardrail.ranked[0].id, bridgeEnvelopeTop: bridgeEnvelope.ranked[0].id, directEnvelopeSectionsTop: directEnvelopeSections.ranked[0].id, softDirectLaneAliasesTop: softDirectLaneAliases.ranked[0].id, threadsFallbackTop: threadsFallback.ranked[0].id, questionsFallbackTop: questionsFallback.ranked[0].id, freeSnapshotTop: freeSnapshot.ranked[0].id, storedScattermindRowTop: storedScattermindRow.ranked[0].id, candidateActionsAliasTop: candidateActionsAlias.ranked[0].id, rankReadyActionsEnvelopeTop: rankReadyActionsEnvelope.ranked[0].id, continuationEnvelopeTop: continuationEnvelope.ranked[0].id, buildOrderPreviewTop: buildOrderPreview.ranked[0].id, scattermindRoadmapLanguageTop: scattermindRoadmapLanguage.ranked[0].id, scattermindRoadmapLensOnlyTop: scattermindRoadmapLensOnly.ranked[0].id, stringifiedRankerInputTop: stringifiedRankerInput.ranked[0].id, scalarBuildOrderSectionsTop: scalarBuildOrderSections.ranked[0].id, snakeReaderFriendlyBuildOrderTop: snakeReaderFriendlyBuildOrder.ranked[0].id, readerFriendlyLensLabelTop: readerFriendlyLensLabel.ranked[0].id, gameRouteGuardrailTop: gameRouteGuardrail.ranked[0].id, duplicateIds: duplicateIds.ranked.map(item => item.id), readiness: data.handoff.readiness.status, provenance: data.input.provenance, buildOrder: data.buildOrder }, null, 2)); } finally { server.kill('SIGTERM'); } diff --git a/server.js b/server.js index ac87935..7f6a8a9 100644 --- a/server.js +++ b/server.js @@ -1186,10 +1186,10 @@ function compactCandidateGroup(group = []) { function buildOrderSectionGroup(buildOrder = {}, baseSection = 'buildOrder') { const source = objectFrom(buildOrder); return compactCandidateGroup([ - { items: source.doFirst || source.do_first || source.buildFirst || source.buildNow || source.now || source.continueFirst || source.continue_first || source.makeTangible || source.make_tangible || source.startHere || source.start_here, sourceSection: `${baseSection}.doFirst`, defaultLane: 'do-first' }, - { items: source.validateNext || source.validate_next || source.validateManually || source.validate_manually || source.testNext || source.testManually || source.test_manually || source.manualProof || source.manual_proof || source.validation || source.evidenceNext || source.evidence_next || source.tryNext || source.try_next || source.learnNext || source.learn_next, sourceSection: `${baseSection}.validateNext`, defaultLane: 'validate-next' }, - { items: source.defer || source.deferred || source.later || source.afterProof || source.holdForLater || source.hold_for_later || source.notYet || source.not_yet, sourceSection: `${baseSection}.defer`, defaultLane: 'defer' }, - { items: source.park || source.parkingLot || source.parking_lot || source.parked || source.probablyNoise || source.probably_noise || source.noise || source.setAside || source.set_aside || source.outOfScope || source.out_of_scope, sourceSection: `${baseSection}.park`, defaultLane: 'park' }, + { items: source.doFirst || source.do_first || source.buildFirst || source.build_first || source.buildNow || source.build_now || source.now || source.continueFirst || source.continue_first || source.makeTangible || source.make_tangible || source.startHere || source.start_here || source.startNow || source.start_now || source.shipFirst || source.ship_first, sourceSection: `${baseSection}.doFirst`, defaultLane: 'do-first' }, + { items: source.validateNext || source.validate_next || source.validateManually || source.validate_manually || source.testNext || source.test_next || source.testManually || source.test_manually || source.manualProof || source.manual_proof || source.validation || source.evidenceNext || source.evidence_next || source.tryNext || source.try_next || source.learnNext || source.learn_next || source.proofStep || source.proof_step || source.proofSteps || source.proof_steps, sourceSection: `${baseSection}.validateNext`, defaultLane: 'validate-next' }, + { items: source.defer || source.deferred || source.later || source.afterProof || source.after_proof || source.holdForLater || source.hold_for_later || source.notYet || source.not_yet || source.leaveOut || source.leave_out || source.skipForNow || source.skip_for_now || source.doNotBuildYet || source.do_not_build_yet, sourceSection: `${baseSection}.defer`, defaultLane: 'defer' }, + { items: source.park || source.parkingLot || source.parking_lot || source.parked || source.probablyNoise || source.probably_noise || source.noise || source.setAside || source.set_aside || source.outOfScope || source.out_of_scope || source.cut || source.distractions, sourceSection: `${baseSection}.park`, defaultLane: 'park' }, ]); } @@ -1202,7 +1202,7 @@ function normalizeCandidateGroup(group = []) { } const buildOrderLabelSeparator = '\\s*(?::|,|[-–—])\\s*'; -const buildOrderLabelPattern = '(build first|build this first|start here|start with|start by|ship first|ship this first|first week|week one|first-week build order|continue first|make tangible first|make tangible|try next|test first|prove first|evidence next|learn next|test manually|validate manually|manual proof|validate next|hold for later|leave out|skip for now|not yet|defer|set aside|out of scope|probably noise|park|do not build yet|don\'t build yet)'; +const buildOrderLabelPattern = '(build first|build this first|start here|start with|start by|ship first|ship this first|first week|week one|first-week build order|continue first|make tangible first|make tangible|try next|test first|test this first|prove first|prove manually|evidence next|learn next|test manually|test this manually|validate manually|validate this manually|manual proof|validate next|hold for later|leave out|skip for now|not yet|defer|set aside|out of scope|probably noise|park|do not build yet|do not build this yet|don\'t build yet)'; const buildOrderLabelRegex = new RegExp(`^${buildOrderLabelPattern}${buildOrderLabelSeparator}`, 'i'); function normalizeBuildOrderFragment(fragment = '') { @@ -1229,8 +1229,8 @@ function titleFromBuildOrderFragment(value = '') { function laneFromBuildOrderLabel(fragment = '') { if (new RegExp(`^(build first|build this first|start here|start with|start by|ship first|ship this first|first week|week one|first-week build order|continue first|make tangible first|make tangible)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'do-first'; - if (new RegExp(`^(try next|test first|prove first|evidence next|learn next|test manually|validate manually|manual proof|validate next)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'validate-next'; - if (new RegExp(`^(hold for later|leave out|skip for now|not yet|defer|do not build yet|don't build yet)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'defer'; + if (new RegExp(`^(try next|test first|test this first|prove first|prove manually|evidence next|learn next|test manually|test this manually|validate manually|validate this manually|manual proof|validate next)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'validate-next'; + if (new RegExp(`^(hold for later|leave out|skip for now|not yet|defer|do not build yet|do not build this yet|don't build yet|don't build this yet)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'defer'; if (new RegExp(`^(set aside|out of scope|probably noise|park)${buildOrderLabelSeparator}`, 'i').test(fragment)) return 'park'; return ''; }