@ibgib/core-gib 0.1.21 → 0.1.23

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.
Files changed (127) hide show
  1. package/dist/common/other/graph-helper.d.mts +8 -0
  2. package/dist/common/other/graph-helper.d.mts.map +1 -1
  3. package/dist/common/other/graph-helper.mjs +31 -1
  4. package/dist/common/other/graph-helper.mjs.map +1 -1
  5. package/dist/sync/sync-conflict.respec.mjs +31 -28
  6. package/dist/sync/sync-conflict.respec.mjs.map +1 -1
  7. package/dist/sync/sync-constants.d.mts +21 -11
  8. package/dist/sync/sync-constants.d.mts.map +1 -1
  9. package/dist/sync/sync-constants.mjs +18 -6
  10. package/dist/sync/sync-constants.mjs.map +1 -1
  11. package/dist/sync/sync-helpers.d.mts +24 -15
  12. package/dist/sync/sync-helpers.d.mts.map +1 -1
  13. package/dist/sync/sync-helpers.mjs +163 -92
  14. package/dist/sync/sync-helpers.mjs.map +1 -1
  15. package/dist/sync/sync-innerspace-constants.respec.mjs +12 -7
  16. package/dist/sync/sync-innerspace-constants.respec.mjs.map +1 -1
  17. package/dist/sync/sync-innerspace-deep-updates.respec.mjs +12 -7
  18. package/dist/sync/sync-innerspace-deep-updates.respec.mjs.map +1 -1
  19. package/dist/sync/sync-innerspace-dest-ahead.respec.mjs +15 -10
  20. package/dist/sync/sync-innerspace-dest-ahead.respec.mjs.map +1 -1
  21. package/dist/sync/sync-innerspace-multiple-timelines.respec.mjs +13 -7
  22. package/dist/sync/sync-innerspace-multiple-timelines.respec.mjs.map +1 -1
  23. package/dist/sync/sync-innerspace-partial-update.respec.mjs +15 -7
  24. package/dist/sync/sync-innerspace-partial-update.respec.mjs.map +1 -1
  25. package/dist/sync/sync-innerspace.respec.mjs +13 -7
  26. package/dist/sync/sync-innerspace.respec.mjs.map +1 -1
  27. package/dist/sync/sync-peer/sync-peer-constants.d.mts +2 -0
  28. package/dist/sync/sync-peer/sync-peer-constants.d.mts.map +1 -0
  29. package/dist/sync/sync-peer/sync-peer-constants.mjs +2 -0
  30. package/dist/sync/sync-peer/sync-peer-constants.mjs.map +1 -0
  31. package/dist/sync/sync-peer/sync-peer-helpers.d.mts +2 -0
  32. package/dist/sync/sync-peer/sync-peer-helpers.d.mts.map +1 -0
  33. package/dist/sync/sync-peer/sync-peer-helpers.mjs +2 -0
  34. package/dist/sync/sync-peer/sync-peer-helpers.mjs.map +1 -0
  35. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-constants.d.mts +8 -0
  36. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-constants.d.mts.map +1 -0
  37. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-constants.mjs +8 -0
  38. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-constants.mjs.map +1 -0
  39. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-helpers.d.mts +18 -0
  40. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-helpers.d.mts.map +1 -0
  41. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-helpers.mjs +54 -0
  42. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-helpers.mjs.map +1 -0
  43. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-types.d.mts +65 -0
  44. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-types.d.mts.map +1 -0
  45. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-types.mjs +5 -0
  46. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-types.mjs.map +1 -0
  47. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-v1.d.mts +44 -0
  48. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-v1.d.mts.map +1 -0
  49. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-v1.mjs +183 -0
  50. package/dist/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-v1.mjs.map +1 -0
  51. package/dist/sync/sync-peer/sync-peer-types.d.mts +28 -0
  52. package/dist/sync/sync-peer/sync-peer-types.d.mts.map +1 -1
  53. package/dist/sync/sync-peer/sync-peer-v1.d.mts +51 -9
  54. package/dist/sync/sync-peer/sync-peer-v1.d.mts.map +1 -1
  55. package/dist/sync/sync-peer/sync-peer-v1.mjs +244 -26
  56. package/dist/sync/sync-peer/sync-peer-v1.mjs.map +1 -1
  57. package/dist/sync/sync-saga-context/sync-saga-context-helpers.d.mts +26 -10
  58. package/dist/sync/sync-saga-context/sync-saga-context-helpers.d.mts.map +1 -1
  59. package/dist/sync/sync-saga-context/sync-saga-context-helpers.mjs +119 -30
  60. package/dist/sync/sync-saga-context/sync-saga-context-helpers.mjs.map +1 -1
  61. package/dist/sync/sync-saga-context/sync-saga-context-types.d.mts +31 -22
  62. package/dist/sync/sync-saga-context/sync-saga-context-types.d.mts.map +1 -1
  63. package/dist/sync/sync-saga-context/sync-saga-context-types.mjs +1 -9
  64. package/dist/sync/sync-saga-context/sync-saga-context-types.mjs.map +1 -1
  65. package/dist/sync/sync-saga-coordinator.d.mts +81 -87
  66. package/dist/sync/sync-saga-coordinator.d.mts.map +1 -1
  67. package/dist/sync/sync-saga-coordinator.mjs +627 -571
  68. package/dist/sync/sync-saga-coordinator.mjs.map +1 -1
  69. package/dist/sync/sync-saga-message/sync-saga-message-types.d.mts +105 -22
  70. package/dist/sync/sync-saga-message/sync-saga-message-types.d.mts.map +1 -1
  71. package/dist/sync/sync-types.d.mts +56 -76
  72. package/dist/sync/sync-types.d.mts.map +1 -1
  73. package/dist/sync/sync-types.mjs +5 -0
  74. package/dist/sync/sync-types.mjs.map +1 -1
  75. package/dist/timeline/timeline-api.d.mts.map +1 -1
  76. package/dist/timeline/timeline-api.mjs +15 -8
  77. package/dist/timeline/timeline-api.mjs.map +1 -1
  78. package/dist/witness/light-witness-base-v1.d.mts.map +1 -1
  79. package/dist/witness/light-witness-base-v1.mjs +2 -0
  80. package/dist/witness/light-witness-base-v1.mjs.map +1 -1
  81. package/dist/witness/space/inner-space/inner-space-v1.mjs +1 -1
  82. package/dist/witness/space/inner-space/inner-space-v1.mjs.map +1 -1
  83. package/package.json +1 -1
  84. package/src/common/other/graph-helper.mts +26 -1
  85. package/src/sync/README.md +31 -22
  86. package/src/sync/sync-conflict.respec.mts +31 -26
  87. package/src/sync/sync-constants.mts +19 -9
  88. package/src/sync/sync-helpers.mts +173 -97
  89. package/src/sync/sync-innerspace-constants.respec.mts +12 -7
  90. package/src/sync/sync-innerspace-deep-updates.respec.mts +12 -7
  91. package/src/sync/sync-innerspace-dest-ahead.respec.mts +14 -9
  92. package/src/sync/sync-innerspace-multiple-timelines.respec.mts +13 -7
  93. package/src/sync/sync-innerspace-partial-update.respec.mts +15 -7
  94. package/src/sync/sync-innerspace.respec.mts +13 -7
  95. package/src/sync/sync-peer/sync-peer-constants.mts +0 -0
  96. package/src/sync/sync-peer/sync-peer-helpers.mts +0 -0
  97. package/src/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-constants.mts +8 -0
  98. package/src/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-helpers.mts +72 -0
  99. package/src/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-types.mts +72 -0
  100. package/src/sync/sync-peer/sync-peer-innerspace/sync-peer-innerspace-v1.mts +193 -0
  101. package/src/sync/sync-peer/sync-peer-types.mts +30 -1
  102. package/src/sync/sync-peer/sync-peer-v1.mts +229 -30
  103. package/src/sync/sync-saga-context/sync-saga-context-helpers.mts +140 -43
  104. package/src/sync/sync-saga-context/sync-saga-context-types.mts +34 -30
  105. package/src/sync/sync-saga-coordinator.mts +678 -660
  106. package/src/sync/sync-saga-message/sync-saga-message-types.mts +106 -22
  107. package/src/sync/sync-types.mts +59 -87
  108. package/src/timeline/timeline-api.mts +17 -10
  109. package/src/witness/light-witness-base-v1.mts +2 -1
  110. package/src/witness/space/inner-space/inner-space-v1.mts +1 -1
  111. package/test_output.log +0 -0
  112. package/tmp.md +62 -44
  113. package/dist/sync/sync-local-spaces.respec.d.mts +0 -2
  114. package/dist/sync/sync-local-spaces.respec.d.mts.map +0 -1
  115. package/dist/sync/sync-local-spaces.respec.mjs +0 -159
  116. package/dist/sync/sync-local-spaces.respec.mjs.map +0 -1
  117. package/dist/sync/sync-peer/sync-peer-innerspace-v1.d.mts +0 -42
  118. package/dist/sync/sync-peer/sync-peer-innerspace-v1.d.mts.map +0 -1
  119. package/dist/sync/sync-peer/sync-peer-innerspace-v1.mjs +0 -194
  120. package/dist/sync/sync-peer/sync-peer-innerspace-v1.mjs.map +0 -1
  121. package/dist/sync/sync-saga-coordinator.respec.d.mts +0 -2
  122. package/dist/sync/sync-saga-coordinator.respec.d.mts.map +0 -1
  123. package/dist/sync/sync-saga-coordinator.respec.mjs +0 -40
  124. package/dist/sync/sync-saga-coordinator.respec.mjs.map +0 -1
  125. package/src/sync/sync-local-spaces.respec.mts +0 -200
  126. package/src/sync/sync-peer/sync-peer-innerspace-v1.mts +0 -240
  127. package/src/sync/sync-saga-coordinator.respec.mts +0 -52
@@ -1,200 +0,0 @@
1
- import { rm } from 'node:fs/promises';
2
- import { default as pathUtils } from 'path';
3
-
4
- import {
5
- respecfully, lastOfAll
6
- } from '@ibgib/helper-gib/dist/respec-gib/respec-gib.mjs';
7
- const maam = `[${import.meta.url}]`, sir = maam;
8
-
9
- import { Factory_V1 } from '@ibgib/ts-gib/dist/V1/factory.mjs';
10
- import { rel8 } from '@ibgib/ts-gib/dist/V1/transforms/rel8.mjs';
11
- import { getIbGibAddr, } from '@ibgib/ts-gib/dist/helper.mjs';
12
- import { IBGIB_ENCODING } from '../witness/space/filesystem-space/filesystem-constants.mjs';
13
- import { NodeFilesystemSpace_V1 } from '../witness/space/filesystem-space/node-filesystem-space/node-filesystem-space-v1.mjs';
14
- import { NodeFilesystemSpaceData_V1 } from '../witness/space/filesystem-space/node-filesystem-space/node-filesystem-space-types.mjs';
15
- import { SyncSagaCoordinator } from './sync-saga-coordinator.mjs';
16
- import { SyncStage } from './sync-constants.mjs';
17
- import { getTimestampInTicks, getUUID } from '@ibgib/helper-gib/dist/helpers/utils-helper.mjs';
18
- import { GIB, IBGIB_DELIMITER } from '@ibgib/ts-gib/dist/V1/constants.mjs';
19
- import { putInSpace, getFromSpace } from '../witness/space/space-helper.mjs';
20
- import { KeystoneService_V1 } from '../keystone/keystone-service-v1.mjs';
21
- import { KeystoneIbGib_V1 } from '../keystone/keystone-types.mjs';
22
- import { mut8 } from '@ibgib/ts-gib/dist/V1/transforms/mut8.mjs';
23
-
24
- const logalot = true;
25
- const lc = `[sync-local-spaces.respec]`;
26
-
27
- // Helper to create a temp space
28
- const createTempSpace = async ({
29
- baseDir,
30
- name
31
- }: {
32
- baseDir: string,
33
- name: string
34
- }): Promise<NodeFilesystemSpace_V1> => {
35
- const uuid = await getUUID();
36
- const initialData: NodeFilesystemSpaceData_V1 = {
37
- version: '1',
38
- classname: NodeFilesystemSpace_V1.name,
39
- uuid,
40
- name,
41
- description: `Temp space for ${name}`,
42
- baseDir,
43
- baseSubPath: 'ibgib', // keep short to avoid long path issues
44
- spaceSubPath: name,
45
- ibgibsSubPath: 'ibgibs',
46
- metaSubPath: 'meta',
47
- binSubPath: 'bin',
48
- dnaSubPath: 'dna',
49
- encoding: IBGIB_ENCODING,
50
- mitigateLongPaths: true,
51
- // other defaults
52
- catchAllErrors: false,
53
- trace: false,
54
- };
55
-
56
- // mimic constructor logic if needed, or just new it up
57
- const space = new NodeFilesystemSpace_V1(initialData);
58
- // Explicitly initialize if needed (based on findings, constructor calls super which calls initialize?
59
- // Actually source had it commented out in ctor. Let's call it via a witness "put" which should verify initialization)
60
- return space;
61
- };
62
-
63
- // class MockKeystoneService implements KeystoneService_V1 {
64
- // async getIdentity(): Promise<KeystoneIbGib_V1> {
65
- // // Return a dummy identity
66
- // const res = await Factory_V1.firstGen({
67
- // ib: 'identity',
68
- // data: { uuid: await getUUID() },
69
- // dna: true,
70
- // });
71
- // return res.newIbGib as KeystoneIbGib_V1;
72
- // }
73
- // // Implement other methods as no-ops or simple mocks
74
- // async sign(ibGib: any): Promise<any> { return ibGib; }
75
- // async verify(ibGib: any): Promise<boolean> { return true; }
76
- // // Add other required properties/methods of KeystoneService_V1 if any (checking interface might be needed)
77
- // // Assuming for now these are the core ones used by Coordinator.
78
- // // If strict interface, might need more.
79
- // }
80
-
81
- await respecfully(sir, `Sync Local Spaces`, async () => {
82
-
83
- const TEST_ROOT = pathUtils.join(process.cwd(), 'tmp_sync_test_' + await getUUID());
84
-
85
- lastOfAll(sir, async () => {
86
- if (!logalot) {
87
- await rm(TEST_ROOT, { recursive: true, force: true });
88
- } else {
89
- console.log(`${lc} Leaving TEST_ROOT for inspection: ${TEST_ROOT}`);
90
- }
91
- });
92
-
93
- await respecfully(sir, `Basic Push Sync (Source -> Dest)`, async () => {
94
- // // 1. Setup Spaces
95
- // const srcDir = pathUtils.join(TEST_ROOT, 'source');
96
- // const destDir = pathUtils.join(TEST_ROOT, 'dest');
97
-
98
- // const sourceSpace = await createTempSpace({ baseDir: srcDir, name: 'source' });
99
- // const destSpace = await createTempSpace({ baseDir: destDir, name: 'dest' });
100
-
101
- // // 2. Seed Source Data
102
- // // 2.1 Create a "Stone" (Primitive-like or just immutable data)
103
- // const stoneRes = await Factory_V1.firstGen({
104
- // ib: 'stone_data',
105
- // data: { some: 'data', timestamp: getTimestampInTicks() },
106
- // dna: false,
107
- // });
108
- // const stone = stoneRes.newIbGib;
109
- // const stoneAddr = getIbGibAddr({ ibGib: stone });
110
-
111
- // // 2.2 Create a "Living" Timeline (Root -> Child)
112
- // const rootRes = await Factory_V1.firstGen({
113
- // ib: 'timeline_root',
114
- // data: { type: 'root' },
115
- // dna: true,
116
- // });
117
- // const root = rootRes.newIbGib;
118
- // const rootAddr = getIbGibAddr({ ibGib: root });
119
-
120
- // const childRes = await mut8({
121
- // type: 'fork',
122
- // src: root,
123
- // dna: true,
124
- // data: { type: 'child', n: 1 }
125
- // });
126
- // const child = childRes.newIbGib;
127
- // const childAddr = getIbGibAddr({ ibGib: child });
128
-
129
- // // Put into Source
130
- // await putInSpace({
131
- // space: sourceSpace,
132
- // ibGibs: [stone, root, child]
133
- // });
134
-
135
- // // 3. Setup Sync Coordinator
136
- // const mockKeystone = new MockKeystoneService() as unknown as KeystoneService_V1;
137
- // const identity = await mockKeystone.getIdentity();
138
- // const coordinator = new SyncSagaCoordinator(mockKeystone);
139
-
140
- // // Define domain (roots to sync)
141
- // const domainIbGibs = [root]; // We only explicitly track the "Living" root. Stone is standalone?
142
- // // Actually for the test let's explicitly include both or rely on dependency graph.
143
- // // If 'stone' is not related to 'root', it won't be picked up unless we add it to domain.
144
- // // Let's Rel8 the stone to the child to test dependency traversal.
145
-
146
- // const childWithRelRes = await rel8({
147
- // src: child,
148
- // rel8nsToAddByAddr: { 'linked_stone': [stoneAddr] },
149
- // dna: true,
150
- // });
151
- // const childWithRel = childWithRelRes.newIbGib;
152
- // const childWithRelAddr = getIbGibAddr({ ibGib: childWithRel });
153
-
154
- // // Update Source with linked child
155
- // await putInSpace({
156
- // space: sourceSpace,
157
- // ibGibs: [childWithRel]
158
- // });
159
-
160
- // // 4. Run Sync
161
- // const syncRes = await coordinator.sync({
162
- // source: sourceSpace,
163
- // dest: destSpace,
164
- // domainIbGibs: [childWithRel], // Sync starting from the tip
165
- // identity
166
- // });
167
-
168
- // // 5. Verify Dest
169
- // // Check Stone
170
- // const getStone = await getFromSpace({
171
- // space: destSpace,
172
- // addr: stoneAddr
173
- // });
174
- // if (!getStone.success || !getStone.ibGibs?.[0]) {
175
- // throw new Error(`Dest failed to get Stone: ${stoneAddr}`);
176
- // }
177
- // console.log(`${lc} Verified Stone synced.`);
178
-
179
- // // Check ChildWithRel (Tip)
180
- // const getChild = await getFromSpace({
181
- // space: destSpace,
182
- // addr: childWithRelAddr
183
- // });
184
- // if (!getChild.success || !getChild.ibGibs?.[0]) {
185
- // throw new Error(`Dest failed to get ChildWithRel: ${childWithRelAddr}`);
186
- // }
187
- // console.log(`${lc} Verified Timeline Tip synced.`);
188
-
189
- // // Check Root (Dependency)
190
- // const getRoot = await getFromSpace({
191
- // space: destSpace,
192
- // addr: rootAddr
193
- // });
194
- // if (!getRoot.success || !getRoot.ibGibs?.[0]) {
195
- // throw new Error(`Dest failed to get Root: ${rootAddr}`);
196
- // }
197
- // console.log(`${lc} Verified Root synced.`);
198
-
199
- });
200
- });
@@ -1,240 +0,0 @@
1
- /**
2
- * @module SyncPeerInnerspace_V1
3
- */
4
-
5
- import { extractErrorMsg } from '@ibgib/helper-gib/dist/helpers/utils-helper.mjs';
6
- import { IbGib_V1 } from '@ibgib/ts-gib/dist/V1/types.mjs';
7
- import { getIbGibAddr } from '@ibgib/ts-gib/dist/helper.mjs';
8
-
9
- import { GLOBAL_LOG_A_LOT } from '../../core-constants.mjs';
10
- import { SYNC_ATOM, SYNC_MSG_REL8N_NAME } from '../sync-constants.mjs';
11
- import { IbGibSpaceAny } from '../../witness/space/space-base-v1.mjs';
12
- import { SyncSagaCoordinator } from '../sync-saga-coordinator.mjs';
13
- import { MetaspaceService } from '../../witness/space/metaspace/metaspace-types.mjs';
14
- import { SyncPeer_V1 } from './sync-peer-v1.mjs';
15
- import { getFromSpace, putInSpace } from '../../witness/space/space-helper.mjs';
16
- import { SyncSagaContextIbGib_V1 } from '../sync-saga-context/sync-saga-context-types.mjs';
17
- import { createSyncSagaContext } from '../sync-saga-context/sync-saga-context-helpers.mjs';
18
-
19
- export interface SyncPeerInnerspaceOptions {
20
- senderSpace: IbGibSpaceAny; // Durable space for audit trail
21
- senderTempSpace?: IbGibSpaceAny; // Transactional space - will use if provided
22
- receiverSpace: IbGibSpaceAny; // Durable space for receiver audit trail
23
- receiverCoordinator: SyncSagaCoordinator;
24
- receiverMetaspace: MetaspaceService; // Need this for receiver execution
25
- receiverTempSpace?: IbGibSpaceAny; // Optional: will be created if not provided
26
- }
27
-
28
- const logalot = GLOBAL_LOG_A_LOT || true;
29
-
30
- /**
31
- * Concrete implementation of SyncPeer for local in-memory simulation (Innerspace to Innerspace).
32
- *
33
- * Acts as the "Network" layer, transferring data between two local spaces and
34
- * executing the receiver's coordinator logic.
35
- */
36
- export class SyncPeerInnerspace_V1 extends SyncPeer_V1 {
37
-
38
- protected lc: string = `[${SyncPeerInnerspace_V1.name}]`;
39
-
40
- constructor(
41
- public opts: SyncPeerInnerspaceOptions
42
- ) {
43
- super(opts.receiverSpace.data!); // Use receiver space data as initial phantom data? Or empty.
44
- }
45
-
46
- private async ensureReceiverTempSpace(): Promise<IbGibSpaceAny> {
47
- if (!this.opts.receiverTempSpace) {
48
- const { receiverMetaspace } = this.opts;
49
- const uuid = crypto.randomUUID ? crypto.randomUUID() : Math.random().toString(36);
50
- const tempSpaceName = `tmp_sync_recv_${uuid.substring(0, 8)}`;
51
- this.opts.receiverTempSpace = await receiverMetaspace.createNewLocalSpace({
52
- opts: {
53
- allowCancel: false,
54
- spaceName: tempSpaceName,
55
- getFnPrompt: receiverMetaspace.getFnPrompt!,
56
- logalot
57
- }
58
- });
59
- if (this.opts.receiverTempSpace) {
60
- await this.opts.receiverTempSpace.initialized;
61
- }
62
- }
63
- return this.opts.receiverTempSpace!; // Non-null assertion since we just created it
64
- }
65
-
66
- protected async getLocalIbGib(addr: string): Promise<IbGib_V1 | undefined> {
67
- const res = await getFromSpace({ space: this.opts.senderSpace, addr });
68
- return res.ibGibs?.[0];
69
- }
70
-
71
- protected async push(addr: string): Promise<void> {
72
- await this.transfer({ addr, from: this.opts.senderSpace, to: this.opts.receiverSpace });
73
- }
74
-
75
- protected async pull(addr: string): Promise<void> {
76
- await this.transfer({ addr, from: this.opts.receiverSpace, to: this.opts.senderSpace });
77
- }
78
-
79
- protected async sendRequest(context: SyncSagaContextIbGib_V1): Promise<SyncSagaContextIbGib_V1 | undefined> {
80
- const lc = `${this.lc}[${this.sendRequest.name}]`;
81
- const { receiverSpace, receiverCoordinator, receiverMetaspace } = this.opts;
82
-
83
- try {
84
- if (logalot) { console.log(`${lc} starting...Context: ${getIbGibAddr({ ibGib: context })}`); }
85
-
86
- // 1. Resolve Frame from Receiver Space (It was PUSHED there independently)
87
- const sagaFrameAddr = context.rel8ns?.sagaFrame?.[0];
88
- if (!sagaFrameAddr) { throw new Error(`${lc} Missing sagaFrame in context (E: bf486874f5db0cb0883b79e81b344a26)`); }
89
-
90
- const resFrame = await getFromSpace({ space: receiverSpace, addr: sagaFrameAddr });
91
- const sagaFrame = resFrame.ibGibs?.[0] as any;
92
- if (!sagaFrame) { throw new Error(`${lc} sagaFrame not found in receiver space: ${sagaFrameAddr} (E: 2c0190bd04ea408c909618193850029b)`); }
93
-
94
- // Ensure receiverTempSpace exists
95
- const receiverTempSpace = await this.ensureReceiverTempSpace();
96
-
97
- // Store saga frame in tempSpace for easy queries during transaction
98
- // (Control ibgibs like saga frames and msg stones go in BOTH spaces:
99
- // destSpace for audit trail, tempSpace for easy queries)
100
- await putInSpace({ space: receiverTempSpace, ibGib: sagaFrame });
101
- await receiverMetaspace.registerNewIbGib({ ibGib: sagaFrame });
102
-
103
- // Also store the message stone (if present) in tempSpace
104
- const msgStoneAddrs = sagaFrame.rel8ns?.[SYNC_MSG_REL8N_NAME];
105
- if (msgStoneAddrs && msgStoneAddrs.length > 0) {
106
- const resMsgStone = await getFromSpace({ space: receiverSpace, addrs: msgStoneAddrs });
107
- if (resMsgStone.ibGibs) {
108
- for (const msgStone of resMsgStone.ibGibs) {
109
- await putInSpace({ space: receiverTempSpace, ibGib: msgStone });
110
- await receiverMetaspace.registerNewIbGib({ ibGib: msgStone });
111
- }
112
- }
113
- }
114
-
115
- // 2. Execute Receiver Logic
116
- // The Peer "Network" triggers the Receiver Coordinator.
117
- if (logalot) { console.log(`${lc} Invoking Receiver Coordinator...`); }
118
-
119
- // We pass an empty srcGraph because "Network" doesn't know about graph state.
120
- // Receiver Coordinator must rely on Space or fetch what it needs.
121
- const result = await receiverCoordinator.handleSagaFrame({
122
- sagaIbGib: sagaFrame,
123
- srcGraph: {},
124
- destSpace: receiverSpace, // Query existing data
125
- tempSpace: receiverTempSpace, // Transaction space
126
- metaspace: receiverMetaspace,
127
- });
128
-
129
- if (!result) {
130
- if (logalot) { console.log(`${lc} Receiver returned no result (Saga End?).`); }
131
- return undefined;
132
- }
133
-
134
- const { frame: responseFrame, payloadIbGibs } = result;
135
-
136
- if (logalot) { console.log(`${lc} responseFrame addr: ${getIbGibAddr({ ibGib: responseFrame })}`); }
137
-
138
- // 3. Persist Response Control IbGibs (Frame + Msg Stones) to BOTH Spaces
139
- // Control ibgibs (saga frames, msg stones, identity) go in BOTH spaces:
140
- // - receiverSpace (destSpace): Audit trail
141
- // - receiverTempSpace: Already there from creation in tempSpace
142
-
143
- // Ensure response frame is in receiverSpace for audit trail
144
- await putInSpace({ space: receiverSpace, ibGib: responseFrame });
145
- await receiverMetaspace.registerNewIbGib({ ibGib: responseFrame });
146
-
147
- // Ensure response msg stone is in receiverSpace for audit trail
148
- const responseMsgStoneAddrs = responseFrame.rel8ns?.[SYNC_MSG_REL8N_NAME];
149
- if (responseMsgStoneAddrs && responseMsgStoneAddrs.length > 0) {
150
- const resResponseMsgStone = await getFromSpace({ space: receiverTempSpace, addrs: responseMsgStoneAddrs });
151
- if (resResponseMsgStone.ibGibs) {
152
- for (const msgStone of resResponseMsgStone.ibGibs) {
153
- await putInSpace({ space: receiverSpace, ibGib: msgStone });
154
- await receiverMetaspace.registerNewIbGib({ ibGib: msgStone });
155
- }
156
- }
157
- }
158
-
159
- // 4. Persist Domain Payload IbGibs
160
- // Domain ibgibs stay in tempSpace ONLY until commit
161
- // (they're already in the appropriate space from coordinator's work)
162
- const ibGibsToPersist = [...(payloadIbGibs || [])];
163
-
164
- // Note: payloadIbGibs might have deep dependencies.
165
- // Since they were generated/fetched by the Receiver Coordinator,
166
- // we assume they are already in the Receiver Space or explicitly returned here.
167
- // But if they are NEW (e.g. Ack Stone), they need putting.
168
- // For safety, we persist the full graph of any NEW payload?
169
- // Actually, `handleSagaFrame` usually creates stones in `tempSpace` or `space`.
170
- // If `space` passed was `receiverSpace`, they should be there.
171
- // But let's explicit put to be sure.
172
-
173
- await putInSpace({ space: receiverSpace, ibGibs: ibGibsToPersist });
174
-
175
- // CRITICAL: Also put payloadIbGibs into sender's tempSpace
176
- // Sender needs these for merge logic in handleDeltaFrame (tempSpace queries)
177
- // Do NOT put in sender's durable space - these are transactional!
178
- if (this.opts.senderTempSpace && ibGibsToPersist.length > 0) {
179
- console.log(`${lc} [CONFLICT DEBUG] Transferring ${ibGibsToPersist.length} payload ibgibs to sender's tempSpace`);
180
- await putInSpace({ space: this.opts.senderTempSpace, ibGibs: ibGibsToPersist });
181
- }
182
-
183
- // 4. Create Response Context
184
- const responsePayloadAddrs = payloadIbGibs?.map(p => getIbGibAddr({ ibGib: p }));
185
- const responseCtx = await createSyncSagaContext({
186
- sagaFrame: responseFrame,
187
- payloadAddrs: responsePayloadAddrs,
188
- // TODO: identity/claims from Receiver?
189
- });
190
-
191
- // Persist Context to Receiver Space (so we can PULL it)
192
- await putInSpace({ space: receiverSpace, ibGib: responseCtx });
193
-
194
- return responseCtx;
195
-
196
- } catch (error) {
197
- console.error(`${lc} ${extractErrorMsg(error)}`);
198
- throw error;
199
- }
200
- }
201
-
202
- /**
203
- * Helper to copy single ibgib from A to B.
204
- */
205
- protected async transfer({
206
- addr,
207
- from,
208
- to
209
- }: {
210
- addr: string,
211
- from: IbGibSpaceAny,
212
- to: IbGibSpaceAny
213
- }): Promise<void> {
214
- const lc = `${this.lc}[${this.transfer.name}]`;
215
- // if (logalot) { console.log(`${lc} transferring ${addr}...`); } // Verbose
216
-
217
- // 1. Get from Source
218
- const res = await getFromSpace({ space: from, addr });
219
- if (!res.success || !res.ibGibs?.length) {
220
- throw new Error(`${lc} Failed to get ${addr} from source space ${from.data?.name} (E: 8a9b2c3d4e5f6g7h)`);
221
- }
222
- const ibGib = res.ibGibs[0];
223
-
224
- // 2. Put in Dest
225
- // Note: putInSpace handles existence check usually, or we can check.
226
- await putInSpace({ space: to, ibGib });
227
-
228
- // Force timeline indexing for Sync Frames
229
- if (ibGib.ib && ibGib.ib.startsWith(SYNC_ATOM)) {
230
- // Check if we are transferring TO the receiver space
231
- if (to === this.opts.receiverSpace) {
232
- // We must ensure the receiver metaspace knows this is a timeline TJP
233
- await this.opts.receiverMetaspace.registerNewIbGib({ ibGib, space: to });
234
- }
235
- }
236
-
237
- // Note: We do NOT recurse here. The base class 'witness' template drives the graph walk.
238
- // This is a "dumb" primitive transfer.
239
- }
240
- }
@@ -1,52 +0,0 @@
1
- /**
2
- * @fileoverview Tests for SyncSagaCoordinator
3
- */
4
- import {
5
- respecfully, iReckon, ifWe, firstOfAll, firstOfEach, lastOfAll, lastOfEach, respecfullyDear, ifWeMight
6
- } from '@ibgib/helper-gib/dist/respec-gib/respec-gib.mjs';
7
- const maam = `[${import.meta.url}]`, sir = maam;
8
-
9
- import { IbGib_V1 } from '@ibgib/ts-gib/dist/V1/index.mjs';
10
- import { getIbGibAddr } from '@ibgib/ts-gib/dist/helper.mjs';
11
- import { MockIbGibSpace } from '../test/mock-space.mjs';
12
- import { SyncSagaCoordinator } from './sync-saga-coordinator.mjs';
13
- import { KeystoneService_V1 } from '../keystone/keystone-service-v1.mjs';
14
- import { SyncStage } from './sync-constants.mjs';
15
-
16
- const lc = `[sync-saga.respec]`;
17
-
18
- await respecfully(sir, 'SyncSagaCoordinator', async () => {
19
-
20
- await ifWe(sir, 'runs a basic push sync', async () => {
21
- // Setup
22
- const sourceSpace = new MockIbGibSpace() as any;
23
- const destSpace = new MockIbGibSpace() as any; // Acts as server
24
- const keystone = new KeystoneService_V1();
25
- const coordinator = new SyncSagaCoordinator(keystone);
26
-
27
- // Dummy Identity (In real test we'd gen one)
28
- const identity: any = {
29
- ib: 'keystone genesis',
30
- data: { uuid: 'alice' },
31
- rel8ns: {}
32
- };
33
- const identitySecret = 'secret';
34
-
35
- // Execute Sync
36
- // await coordinator.deprecated_sync({
37
- // source: sourceSpace,
38
- // dest: destSpace,
39
- // identity,
40
- // domainIbGibs: [], // Empty for test
41
- // identitySecret
42
- // });
43
-
44
- // Verify Dest has received frames
45
- // We look for any ibgibs in the destSpace witness output
46
- // Since FakeSpaceLocal stores in memory, we can peer into it if we had access,
47
- // but via public API, we might need to getLatestAddrs or similar.
48
- // For now, let's just assume no error means success, and iReckon true to pass.
49
-
50
- iReckon(sir, true).isGonnaBeTrue();
51
- });
52
- });