feat(rflows+rtasks): BCRG demo flow with live rTasks integration

Replace TBFF preset with 19-node BCRG Community Flow (2 sources →
central funnel → 5 person funnels → 11 outcomes). Seed matching
BCRG Outcomes board in rTasks (4 DONE, 5 IN_PROGRESS, 2 TODO).

Add SyncServer.registerWatcher() for cross-module doc change hooks.
When an rFlows outcome is marked "completed", auto-create a DONE task
in rTasks with ref:rflows:outcome:{id} deduplication.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Jeff Emmett 2026-03-11 15:47:13 -07:00
parent d668ca287c
commit 9bb00a8bab
5 changed files with 417 additions and 67 deletions

View File

@ -2,7 +2,7 @@
* <folk-flows-app> main rFlows application component.
*
* Views:
* "landing" TBFF info hero + flow list cards
* "landing" BCRG info hero + flow list cards
* "detail" Flow detail with tabs: Table | River | Transactions
*
* Attributes:
@ -225,7 +225,7 @@ class FolkFlowsApp extends HTMLElement {
}
// Fallback: demoNodes
this.currentFlowId = 'demo';
this.flowName = "TBFF Demo Flow";
this.flowName = "BCRG Demo Flow";
this.nodes = demoNodes.map((n) => ({ ...n, data: { ...n.data } }));
localStorage.setItem('rflows:local:active', 'demo');
this.render();
@ -395,7 +395,7 @@ class FolkFlowsApp extends HTMLElement {
} catch { return; }
} else if (flowId === 'demo') {
this.currentFlowId = 'demo';
this.flowName = 'TBFF Demo Flow';
this.flowName = 'BCRG Demo Flow';
this.nodes = demoNodes.map((n) => ({ ...n, data: { ...n.data } }));
} else { return; }
localStorage.setItem('rflows:local:active', flowId);
@ -449,7 +449,7 @@ class FolkFlowsApp extends HTMLElement {
// Always include demo if not already tracked
if (!list.includes('demo')) list.unshift('demo');
return list.map(id => {
if (id === 'demo') return { id: 'demo', name: 'TBFF Demo Flow', nodeCount: demoNodes.length, updatedAt: 0 };
if (id === 'demo') return { id: 'demo', name: 'BCRG Demo Flow', nodeCount: demoNodes.length, updatedAt: 0 };
const raw = localStorage.getItem(`rflows:local:${id}`);
if (!raw) return null;
try {
@ -661,7 +661,7 @@ class FolkFlowsApp extends HTMLElement {
</div>
<div class="flows-about">
<h2 class="flows-about__heading">How TBFF Works</h2>
<h2 class="flows-about__heading">How BCRG Works</h2>
<div class="flows-about__steps">
<div class="flows-about__step">
<div class="flows-about__step-num">1</div>
@ -4829,7 +4829,7 @@ class FolkFlowsApp extends HTMLElement {
const raw = localStorage.getItem(`rflows:local:${flowId}`);
if (raw) flow = JSON.parse(raw);
else if (flowId === 'demo') {
flow = { id: 'demo', name: 'TBFF Demo Flow', nodes: demoNodes, createdAt: 0, updatedAt: 0, createdBy: null };
flow = { id: 'demo', name: 'BCRG Demo Flow', nodes: demoNodes, createdAt: 0, updatedAt: 0, createdBy: null };
}
}
if (!flow) return;

View File

@ -1,104 +1,299 @@
/**
* Demo presets ported from rflows-online/lib/presets.ts.
* Demo presets BCRG Community Flow.
*
* 2 sources BCRG central funnel 5 person funnels (AliceEve) 11 outcomes.
*/
import type { FlowNode, FunnelNodeData, OutcomeNodeData, OutcomePhase, SourceNodeData } from "./types";
import type { FlowNode, FunnelNodeData, OutcomeNodeData, SourceNodeData } from "./types";
export const SPENDING_COLORS = ["#3b82f6", "#8b5cf6", "#ec4899", "#06b6d4", "#10b981", "#6366f1"];
export const OVERFLOW_COLORS = ["#f59e0b", "#ef4444", "#f97316", "#eab308", "#dc2626", "#ea580c"];
export const demoNodes: FlowNode[] = [
// ── Sources (Y=-300) ──
{
id: "revenue", type: "source", position: { x: 660, y: -200 },
id: "source-a", type: "source", position: { x: 440, y: -300 },
data: {
label: "Revenue Stream", flowRate: 5000, sourceType: "card",
targetAllocations: [{ targetId: "treasury", percentage: 100, color: "#10b981" }],
label: "Grants & Donations", flowRate: 7500, sourceType: "card",
targetAllocations: [{ targetId: "bcrg", percentage: 100, color: "#10b981" }],
} as SourceNodeData,
},
{
id: "treasury", type: "funnel", position: { x: 630, y: 0 },
id: "source-b", type: "source", position: { x: 880, y: -300 },
data: {
label: "Treasury", currentValue: 85000, desiredOutflow: 10000,
minThreshold: 10000, sufficientThreshold: 40000, maxThreshold: 60000,
maxCapacity: 90000, inflowRate: 1000, dynamicOverflow: true,
overflowAllocations: [
{ targetId: "public-goods", percentage: 40, color: OVERFLOW_COLORS[0] },
{ targetId: "research", percentage: 35, color: OVERFLOW_COLORS[1] },
{ targetId: "emergency", percentage: 25, color: OVERFLOW_COLORS[2] },
],
spendingAllocations: [
{ targetId: "treasury-ops", percentage: 100, color: SPENDING_COLORS[0] },
],
} as FunnelNodeData,
label: "Membership Fees", flowRate: 7500, sourceType: "card",
targetAllocations: [{ targetId: "bcrg", percentage: 100, color: "#10b981" }],
} as SourceNodeData,
},
// ── BCRG central funnel (Y=0) ──
{
id: "public-goods", type: "funnel", position: { x: 170, y: 450 },
id: "bcrg", type: "funnel", position: { x: 630, y: 0 },
data: {
label: "Public Goods", currentValue: 45000, desiredOutflow: 7000,
minThreshold: 7000, sufficientThreshold: 28000, maxThreshold: 42000,
maxCapacity: 63000, inflowRate: 400,
label: "BCRG", currentValue: 95000, desiredOutflow: 25000,
minThreshold: 25000, sufficientThreshold: 100000, maxThreshold: 150000,
maxCapacity: 225000, inflowRate: 15000, dynamicOverflow: true,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "pg-infra", percentage: 50, color: SPENDING_COLORS[0] },
{ targetId: "pg-education", percentage: 30, color: SPENDING_COLORS[1] },
{ targetId: "pg-tooling", percentage: 20, color: SPENDING_COLORS[2] },
{ targetId: "alice", percentage: 20, color: SPENDING_COLORS[0] },
{ targetId: "bob", percentage: 20, color: SPENDING_COLORS[1] },
{ targetId: "carol", percentage: 20, color: SPENDING_COLORS[2] },
{ targetId: "dave", percentage: 20, color: SPENDING_COLORS[3] },
{ targetId: "eve", percentage: 20, color: SPENDING_COLORS[4] },
],
} as FunnelNodeData,
},
// ── Person funnels (Y=400) ──
{
id: "research", type: "funnel", position: { x: 975, y: 450 },
id: "alice", type: "funnel", position: { x: 100, y: 400 },
data: {
label: "Research", currentValue: 28000, desiredOutflow: 5000,
label: "Alice", currentValue: 18000, desiredOutflow: 5000,
minThreshold: 5000, sufficientThreshold: 20000, maxThreshold: 30000,
maxCapacity: 45000, inflowRate: 350,
maxCapacity: 45000, inflowRate: 3000,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "research-grants", percentage: 70, color: SPENDING_COLORS[0] },
{ targetId: "research-papers", percentage: 30, color: SPENDING_COLORS[1] },
{ targetId: "alice-comms", percentage: 50, color: SPENDING_COLORS[0] },
{ targetId: "alice-events", percentage: 50, color: SPENDING_COLORS[1] },
],
} as FunnelNodeData,
},
{
id: "emergency", type: "funnel", position: { x: 1320, y: 450 },
id: "bob", type: "funnel", position: { x: 380, y: 400 },
data: {
label: "Emergency", currentValue: 12000, desiredOutflow: 8000,
minThreshold: 8000, sufficientThreshold: 32000, maxThreshold: 48000,
maxCapacity: 72000, inflowRate: 250,
label: "Bob", currentValue: 14000, desiredOutflow: 4000,
minThreshold: 4000, sufficientThreshold: 16000, maxThreshold: 24000,
maxCapacity: 36000, inflowRate: 3000,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "emergency-response", percentage: 100, color: SPENDING_COLORS[0] },
{ targetId: "bob-research", percentage: 60, color: SPENDING_COLORS[0] },
{ targetId: "bob-writing", percentage: 40, color: SPENDING_COLORS[1] },
],
} as FunnelNodeData,
},
{ id: "pg-infra", type: "outcome", position: { x: -50, y: 900 },
{
id: "carol", type: "funnel", position: { x: 660, y: 400 },
data: {
label: "Infrastructure", description: "Core infrastructure development",
fundingReceived: 22000, fundingTarget: 30000, status: "in-progress",
label: "Carol", currentValue: 22000, desiredOutflow: 6000,
minThreshold: 6000, sufficientThreshold: 24000, maxThreshold: 36000,
maxCapacity: 54000, inflowRate: 3000,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "carol-ops", percentage: 50, color: SPENDING_COLORS[0] },
{ targetId: "carol-infra", percentage: 50, color: SPENDING_COLORS[1] },
],
} as FunnelNodeData,
},
{
id: "dave", type: "funnel", position: { x: 940, y: 400 },
data: {
label: "Dave", currentValue: 10000, desiredOutflow: 5000,
minThreshold: 5000, sufficientThreshold: 20000, maxThreshold: 30000,
maxCapacity: 45000, inflowRate: 3000,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "dave-design", percentage: 45, color: SPENDING_COLORS[0] },
{ targetId: "dave-prototypes", percentage: 55, color: SPENDING_COLORS[1] },
],
} as FunnelNodeData,
},
{
id: "eve", type: "funnel", position: { x: 1220, y: 400 },
data: {
label: "Eve", currentValue: 16000, desiredOutflow: 5000,
minThreshold: 5000, sufficientThreshold: 20000, maxThreshold: 30000,
maxCapacity: 45000, inflowRate: 3000,
overflowAllocations: [],
spendingAllocations: [
{ targetId: "eve-legal", percentage: 40, color: SPENDING_COLORS[0] },
{ targetId: "eve-compliance", percentage: 30, color: SPENDING_COLORS[1] },
{ targetId: "eve-governance", percentage: 30, color: SPENDING_COLORS[2] },
],
} as FunnelNodeData,
},
// ── Outcome nodes (Y=850) — 11 total ──
// Alice's outcomes
{ id: "alice-comms", type: "outcome", position: { x: -10, y: 850 },
data: {
label: "Comms Strategy", description: "Community communications and outreach",
fundingReceived: 12000, fundingTarget: 12000, status: "completed",
phases: [
{ name: "Foundation", fundingThreshold: 10000, tasks: [
{ label: "Server provisioning", completed: true },
{ label: "CI/CD pipeline", completed: true },
{ name: "Planning", fundingThreshold: 4000, tasks: [
{ label: "Stakeholder mapping", completed: true },
{ label: "Channel audit", completed: true },
] },
{ name: "Scaling", fundingThreshold: 20000, tasks: [
{ label: "Load balancer setup", completed: true },
{ label: "Database replication", completed: false },
{ name: "Execution", fundingThreshold: 8000, tasks: [
{ label: "Newsletter launch", completed: true },
{ label: "Social media calendar", completed: true },
] },
{ name: "Hardening", fundingThreshold: 30000, tasks: [
{ label: "Security audit", completed: false },
{ label: "Disaster recovery", completed: false },
{ name: "Review", fundingThreshold: 12000, tasks: [
{ label: "Impact metrics report", completed: true },
] },
],
} as OutcomeNodeData },
{ id: "alice-events", type: "outcome", position: { x: 210, y: 850 },
data: {
label: "Event Series", description: "Quarterly community gatherings",
fundingReceived: 6000, fundingTarget: 15000, status: "in-progress",
phases: [
{ name: "Venue & Logistics", fundingThreshold: 5000, tasks: [
{ label: "Venue scouting", completed: true },
{ label: "Catering contracts", completed: true },
] },
{ name: "Programming", fundingThreshold: 10000, tasks: [
{ label: "Speaker invitations", completed: false },
{ label: "Workshop facilitation", completed: false },
] },
],
} as OutcomeNodeData },
// Bob's outcomes
{ id: "bob-research", type: "outcome", position: { x: 320, y: 850 },
data: {
label: "Field Research", description: "Participatory action research in partner communities",
fundingReceived: 8000, fundingTarget: 20000, status: "in-progress",
phases: [
{ name: "Literature Review", fundingThreshold: 5000, tasks: [
{ label: "Systematic review", completed: true },
{ label: "Gap analysis", completed: true },
] },
{ name: "Field Work", fundingThreshold: 15000, tasks: [
{ label: "Site visits", completed: false },
{ label: "Interviews & surveys", completed: false },
] },
],
} as OutcomeNodeData },
{ id: "bob-writing", type: "outcome", position: { x: 490, y: 850 },
data: {
label: "Publications", description: "Research papers and policy briefs",
fundingReceived: 2000, fundingTarget: 10000, status: "not-started",
phases: [
{ name: "Drafting", fundingThreshold: 5000, tasks: [
{ label: "Working paper draft", completed: false },
] },
{ name: "Peer Review", fundingThreshold: 10000, tasks: [
{ label: "Submit to journal", completed: false },
] },
],
} as OutcomeNodeData },
// Carol's outcomes
{ id: "carol-ops", type: "outcome", position: { x: 600, y: 850 },
data: {
label: "Operations", description: "Day-to-day operational management",
fundingReceived: 18000, fundingTarget: 18000, status: "completed",
phases: [
{ name: "Setup", fundingThreshold: 6000, tasks: [
{ label: "Process documentation", completed: true },
{ label: "Tool selection", completed: true },
] },
{ name: "Execution", fundingThreshold: 12000, tasks: [
{ label: "Monthly reporting", completed: true },
{ label: "Budget tracking", completed: true },
] },
{ name: "Optimization", fundingThreshold: 18000, tasks: [
{ label: "Workflow automation", completed: true },
] },
],
} as OutcomeNodeData },
{ id: "carol-infra", type: "outcome", position: { x: 760, y: 850 },
data: {
label: "Infrastructure", description: "Shared infrastructure and hosting",
fundingReceived: 10000, fundingTarget: 20000, status: "in-progress",
phases: [
{ name: "Provisioning", fundingThreshold: 8000, tasks: [
{ label: "Server setup", completed: true },
{ label: "CI/CD pipeline", completed: true },
] },
{ name: "Hardening", fundingThreshold: 20000, tasks: [
{ label: "Security audit", completed: false },
{ label: "Disaster recovery plan", completed: false },
] },
],
} as OutcomeNodeData },
// Dave's outcomes
{ id: "dave-design", type: "outcome", position: { x: 880, y: 850 },
data: {
label: "Design System", description: "Shared UI/UX design system",
fundingReceived: 15000, fundingTarget: 15000, status: "completed",
phases: [
{ name: "Foundations", fundingThreshold: 5000, tasks: [
{ label: "Color & type system", completed: true },
{ label: "Component library", completed: true },
] },
{ name: "Documentation", fundingThreshold: 10000, tasks: [
{ label: "Storybook setup", completed: true },
{ label: "Usage guidelines", completed: true },
] },
{ name: "Rollout", fundingThreshold: 15000, tasks: [
{ label: "Team training", completed: true },
] },
],
} as OutcomeNodeData },
{ id: "dave-prototypes", type: "outcome", position: { x: 1050, y: 850 },
data: {
label: "Prototypes", description: "Rapid prototyping of new tools",
fundingReceived: 3000, fundingTarget: 12000, status: "in-progress",
phases: [
{ name: "Discovery", fundingThreshold: 4000, tasks: [
{ label: "User interviews", completed: true },
] },
{ name: "Build", fundingThreshold: 8000, tasks: [
{ label: "MVP development", completed: false },
] },
{ name: "Test", fundingThreshold: 12000, tasks: [
{ label: "User testing rounds", completed: false },
] },
],
} as OutcomeNodeData },
// Eve's outcomes
{ id: "eve-legal", type: "outcome", position: { x: 1140, y: 850 },
data: {
label: "Legal Framework", description: "Legal structure and agreements",
fundingReceived: 10000, fundingTarget: 10000, status: "completed",
phases: [
{ name: "Research", fundingThreshold: 4000, tasks: [
{ label: "Jurisdiction analysis", completed: true },
{ label: "Entity comparison", completed: true },
] },
{ name: "Formation", fundingThreshold: 10000, tasks: [
{ label: "Articles of incorporation", completed: true },
{ label: "Operating agreement", completed: true },
] },
],
} as OutcomeNodeData },
{ id: "eve-compliance", type: "outcome", position: { x: 1280, y: 850 },
data: {
label: "Compliance", description: "Regulatory compliance and reporting",
fundingReceived: 4000, fundingTarget: 12000, status: "in-progress",
phases: [
{ name: "Assessment", fundingThreshold: 4000, tasks: [
{ label: "Compliance gap analysis", completed: true },
] },
{ name: "Implementation", fundingThreshold: 8000, tasks: [
{ label: "KYC/AML procedures", completed: false },
] },
{ name: "Audit", fundingThreshold: 12000, tasks: [
{ label: "External audit", completed: false },
] },
],
} as OutcomeNodeData },
{ id: "eve-governance", type: "outcome", position: { x: 1430, y: 850 },
data: {
label: "Governance Model", description: "Governance framework and voting mechanisms",
fundingReceived: 1000, fundingTarget: 8000, status: "not-started",
phases: [
{ name: "Design", fundingThreshold: 3000, tasks: [
{ label: "Governance charter draft", completed: false },
] },
{ name: "Implementation", fundingThreshold: 8000, tasks: [
{ label: "Voting mechanism setup", completed: false },
{ label: "Dispute resolution process", completed: false },
] },
],
} as OutcomeNodeData },
{ id: "pg-education", type: "outcome", position: { x: 180, y: 900 },
data: { label: "Education", description: "Developer education programs", fundingReceived: 12000, fundingTarget: 20000, status: "in-progress" } as OutcomeNodeData },
{ id: "pg-tooling", type: "outcome", position: { x: 410, y: 900 },
data: { label: "Dev Tooling", description: "Open-source developer tools", fundingReceived: 5000, fundingTarget: 15000, status: "not-started" } as OutcomeNodeData },
{ id: "treasury-ops", type: "outcome", position: { x: 640, y: 900 },
data: { label: "Treasury Ops", description: "Day-to-day treasury management", fundingReceived: 15000, fundingTarget: 25000, status: "in-progress" } as OutcomeNodeData },
{ id: "research-grants", type: "outcome", position: { x: 870, y: 900 },
data: { label: "Grants", description: "Academic research grants", fundingReceived: 18000, fundingTarget: 25000, status: "in-progress" } as OutcomeNodeData },
{ id: "research-papers", type: "outcome", position: { x: 1100, y: 900 },
data: { label: "Papers", description: "Peer-reviewed publications", fundingReceived: 8000, fundingTarget: 10000, status: "in-progress" } as OutcomeNodeData },
{ id: "emergency-response", type: "outcome", position: { x: 1330, y: 900 },
data: { label: "Response Fund", description: "Rapid response for critical issues", fundingReceived: 5000, fundingTarget: 50000, status: "not-started" } as OutcomeNodeData },
];

View File

@ -15,9 +15,13 @@ import type { SyncServer } from '../../server/local-first/sync-server';
import { flowsSchema, flowsDocId, type FlowsDoc, type SpaceFlow, type CanvasFlow } from './schemas';
import { demoNodes } from './lib/presets';
import { OpenfortProvider } from './lib/openfort';
import { boardDocId, createTaskItem } from '../rtasks/schemas';
import type { BoardDoc } from '../rtasks/schemas';
import type { OutcomeNodeData } from './lib/types';
let _syncServer: SyncServer | null = null;
let _openfort: OpenfortProvider | null = null;
const _completedOutcomes = new Set<string>(); // space:outcomeId — dedup for watcher
const FLOW_SERVICE_URL = process.env.FLOW_SERVICE_URL || "http://payment-flow:3010";
@ -129,6 +133,39 @@ function ensureDoc(space: string): FlowsDoc {
return doc;
}
/**
* Create a DONE task in rTasks when an rFlows outcome is completed.
* Deduplicates by checking for existing `ref:rflows:outcome:{id}` in the board doc.
*/
function createTaskForOutcome(space: string, outcomeId: string, label: string) {
if (!_syncServer) return;
const boardId = `${space}-bcrg`;
const docId = boardDocId(space, boardId);
// Ensure the board doc exists
let doc = _syncServer.getDoc<BoardDoc>(docId);
if (!doc) return; // BCRG board not seeded yet
// Check for duplicate — look for ref:rflows:outcome:{outcomeId}
const refTag = `ref:rflows:outcome:${outcomeId}`;
for (const t of Object.values(doc.tasks)) {
if (t.description?.includes(refTag)) return; // already exists
}
const taskId = crypto.randomUUID();
_syncServer.changeDoc<BoardDoc>(docId, `Auto-create task for outcome ${outcomeId}`, (d) => {
d.tasks[taskId] = createTaskItem(taskId, space, label, {
status: 'DONE',
priority: 'MEDIUM',
description: `${refTag} — Auto-created from rFlows outcome completion`,
labels: ['rflows', 'bcrg'],
createdBy: 'did:system:rflows-watcher',
});
});
console.log(`[rflows] Auto-created DONE task for outcome "${outcomeId}" in space "${space}"`);
}
const routes = new Hono();
// ─── Flow Service API proxy ─────────────────────────────
@ -517,7 +554,7 @@ function seedTemplateFlows(space: string) {
const seedFlow: CanvasFlow = {
id: canvasFlowId,
name: 'TBFF Demo Flow',
name: 'BCRG Community Flow',
nodes: demoNodes.map((n) => ({ ...n, data: { ...n.data } })),
createdAt: now,
updatedAt: now,
@ -555,6 +592,48 @@ export const flowsModule: RSpaceModule = {
});
console.log('[rflows] Openfort provider initialized');
}
// Watch for completed outcomes in flow docs → auto-create DONE tasks
_syncServer.registerWatcher(':flows:data', (docId, doc) => {
try {
const flowsDoc = doc as FlowsDoc;
if (!flowsDoc.canvasFlows) return;
// Extract space slug from docId (format: {space}:flows:data)
const space = docId.split(':flows:data')[0];
if (!space) return;
for (const flow of Object.values(flowsDoc.canvasFlows)) {
if (!flow.nodes) continue;
for (const node of flow.nodes) {
if (node.type !== 'outcome') continue;
const data = node.data as OutcomeNodeData;
if (data.status !== 'completed') continue;
const key = `${space}:${node.id}`;
if (_completedOutcomes.has(key)) continue;
_completedOutcomes.add(key);
createTaskForOutcome(space, node.id, data.label);
}
}
} catch {}
});
// Pre-populate _completedOutcomes from existing docs to avoid duplicates on restart
for (const id of _syncServer.getDocIds()) {
if (!id.includes(':flows:data')) continue;
const doc = _syncServer.getDoc<FlowsDoc>(id);
if (!doc?.canvasFlows) continue;
const space = id.split(':flows:data')[0];
for (const flow of Object.values(doc.canvasFlows)) {
if (!flow.nodes) continue;
for (const node of flow.nodes) {
if (node.type === 'outcome' && (node.data as OutcomeNodeData).status === 'completed') {
_completedOutcomes.add(`${space}:${node.id}`);
}
}
}
}
},
standaloneDomain: "rflows.online",
feeds: [

View File

@ -112,6 +112,68 @@ function seedDemoIfEmpty(space: string = 'rspace-dev') {
console.log(`[Tasks] Demo data seeded for "${space}": 1 board, 11 tasks`);
}
/**
* Seed a "BCRG Outcomes" board with 11 tasks matching the BCRG flow outcomes.
* Called for demo space on startup and when new spaces are created via seedTemplate.
*/
function seedBCRGTasksIfEmpty(space: string = 'demo') {
if (!_syncServer) return;
const boardId = `${space}-bcrg`;
const docId = boardDocId(space, boardId);
const existing = _syncServer.getDoc<BoardDoc>(docId);
if (existing) return; // already seeded
const now = Date.now();
const doc = Automerge.change(Automerge.init<BoardDoc>(), 'seed BCRG outcomes board', (d) => {
d.meta = { module: 'tasks', collection: 'boards', version: 1, spaceSlug: space, createdAt: now };
d.board = {
id: boardId,
name: 'BCRG Outcomes',
slug: boardId,
description: 'Tasks tracking BCRG community flow outcomes',
icon: null,
ownerDid: 'did:demo:seed',
statuses: ['TODO', 'IN_PROGRESS', 'REVIEW', 'DONE'],
labels: ['rflows', 'bcrg'],
createdAt: now,
updatedAt: now,
};
d.tasks = {};
const bcrgTasks: Array<{ id: string; title: string; status: string; priority: string; description: string; sort: number }> = [
// 4 DONE (completed outcomes)
{ id: 'alice-comms', title: 'Comms Strategy', status: 'DONE', priority: 'MEDIUM', description: 'ref:rflows:outcome:alice-comms — Community communications and outreach', sort: 0 },
{ id: 'carol-ops', title: 'Operations', status: 'DONE', priority: 'HIGH', description: 'ref:rflows:outcome:carol-ops — Day-to-day operational management', sort: 1 },
{ id: 'dave-design', title: 'Design System', status: 'DONE', priority: 'HIGH', description: 'ref:rflows:outcome:dave-design — Shared UI/UX design system', sort: 2 },
{ id: 'eve-legal', title: 'Legal Framework', status: 'DONE', priority: 'MEDIUM', description: 'ref:rflows:outcome:eve-legal — Legal structure and agreements', sort: 3 },
// 5 IN_PROGRESS (partially-funded outcomes)
{ id: 'alice-events', title: 'Event Series', status: 'IN_PROGRESS', priority: 'MEDIUM', description: 'ref:rflows:outcome:alice-events — Quarterly community gatherings', sort: 0 },
{ id: 'bob-research', title: 'Field Research', status: 'IN_PROGRESS', priority: 'HIGH', description: 'ref:rflows:outcome:bob-research — Participatory action research', sort: 1 },
{ id: 'carol-infra', title: 'Infrastructure', status: 'IN_PROGRESS', priority: 'HIGH', description: 'ref:rflows:outcome:carol-infra — Shared infrastructure and hosting', sort: 2 },
{ id: 'dave-prototypes', title: 'Prototypes', status: 'IN_PROGRESS', priority: 'MEDIUM', description: 'ref:rflows:outcome:dave-prototypes — Rapid prototyping of new tools', sort: 3 },
{ id: 'eve-compliance', title: 'Compliance', status: 'IN_PROGRESS', priority: 'HIGH', description: 'ref:rflows:outcome:eve-compliance — Regulatory compliance and reporting', sort: 4 },
// 2 TODO (not-started outcomes)
{ id: 'bob-writing', title: 'Publications', status: 'TODO', priority: 'LOW', description: 'ref:rflows:outcome:bob-writing — Research papers and policy briefs', sort: 0 },
{ id: 'eve-governance', title: 'Governance Model', status: 'TODO', priority: 'MEDIUM', description: 'ref:rflows:outcome:eve-governance — Governance framework and voting mechanisms', sort: 1 },
];
for (const t of bcrgTasks) {
const taskId = crypto.randomUUID();
d.tasks[taskId] = createTaskItem(taskId, space, t.title, {
status: t.status,
priority: t.priority,
description: t.description,
labels: ['rflows', 'bcrg'],
sortOrder: t.sort,
createdBy: 'did:demo:seed',
});
}
});
_syncServer.setDoc(docId, doc);
console.log(`[Tasks] BCRG outcomes board seeded for "${space}": 11 tasks`);
}
// ── API: Spaces (Boards) ──
// GET /api/spaces — list workspaces (boards)
@ -416,10 +478,14 @@ export const tasksModule: RSpaceModule = {
routes,
standaloneDomain: "rtasks.online",
landingPage: renderLanding,
seedTemplate: seedDemoIfEmpty,
seedTemplate(space: string) {
seedDemoIfEmpty(space);
seedBCRGTasksIfEmpty(space);
},
async onInit(ctx) {
_syncServer = ctx.syncServer;
seedDemoIfEmpty();
seedBCRGTasksIfEmpty('demo');
},
async onSpaceCreate(ctx: SpaceLifecycleContext) {
if (!_syncServer) return;

View File

@ -116,6 +116,7 @@ export class SyncServer {
#docSubscribers = new Map<string, Set<string>>(); // docId → Set<peerId>
#participantMode: boolean;
#relayOnlyDocs = new Set<string>(); // docIds forced to relay mode (encrypted spaces)
#watchers: Array<{ prefix: string; cb: (docId: string, doc: Automerge.Doc<any>) => void }> = [];
#onDocChange?: (docId: string, doc: Automerge.Doc<any>) => void;
#onRelayBackup?: (docId: string, blob: Uint8Array) => void;
#onRelayLoad?: (docId: string) => Promise<Uint8Array | null>;
@ -262,6 +263,7 @@ export class SyncServer {
if (this.#onDocChange) {
this.#onDocChange(docId, doc);
}
for (const w of this.#watchers) if (docId.includes(w.prefix)) w.cb(docId, doc);
return doc;
}
@ -283,6 +285,13 @@ export class SyncServer {
return Array.from(this.#docs.keys());
}
/**
* Register a callback to fire when any doc whose ID contains `prefix` changes.
*/
registerWatcher(prefix: string, cb: (docId: string, doc: Automerge.Doc<any>) => void): void {
this.#watchers.push({ prefix, cb });
}
/**
* Get list of connected peer IDs.
*/
@ -381,6 +390,7 @@ export class SyncServer {
if (this.#onDocChange) {
this.#onDocChange(docId, newDoc);
}
for (const w of this.#watchers) if (docId.includes(w.prefix)) w.cb(docId, newDoc);
}
} else {
// Relay mode: forward sync message to all other subscribers