jazz-tools 0.19.8 → 0.19.11

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 (154) hide show
  1. package/.turbo/turbo-build.log +56 -50
  2. package/CHANGELOG.md +30 -3
  3. package/dist/{chunk-2S3Z2CN6.js → chunk-HX5S6W5E.js} +372 -103
  4. package/dist/chunk-HX5S6W5E.js.map +1 -0
  5. package/dist/index.js +1 -1
  6. package/dist/inspector/account-switcher.d.ts +4 -0
  7. package/dist/inspector/account-switcher.d.ts.map +1 -0
  8. package/dist/inspector/chunk-C6BJPHBQ.js +4096 -0
  9. package/dist/inspector/chunk-C6BJPHBQ.js.map +1 -0
  10. package/dist/inspector/contexts/node.d.ts +19 -0
  11. package/dist/inspector/contexts/node.d.ts.map +1 -0
  12. package/dist/inspector/{custom-element-P76EIWEV.js → custom-element-GJVBPZES.js} +1011 -884
  13. package/dist/inspector/custom-element-GJVBPZES.js.map +1 -0
  14. package/dist/inspector/{viewer/new-app.d.ts → in-app.d.ts} +3 -3
  15. package/dist/inspector/in-app.d.ts.map +1 -0
  16. package/dist/inspector/index.d.ts +0 -11
  17. package/dist/inspector/index.d.ts.map +1 -1
  18. package/dist/inspector/index.js +56 -3910
  19. package/dist/inspector/index.js.map +1 -1
  20. package/dist/inspector/pages/home.d.ts +2 -0
  21. package/dist/inspector/pages/home.d.ts.map +1 -0
  22. package/dist/inspector/register-custom-element.js +1 -1
  23. package/dist/inspector/router/context.d.ts +12 -0
  24. package/dist/inspector/router/context.d.ts.map +1 -0
  25. package/dist/inspector/router/hash-router.d.ts +7 -0
  26. package/dist/inspector/router/hash-router.d.ts.map +1 -0
  27. package/dist/inspector/router/in-memory-router.d.ts +7 -0
  28. package/dist/inspector/router/in-memory-router.d.ts.map +1 -0
  29. package/dist/inspector/router/index.d.ts +5 -0
  30. package/dist/inspector/router/index.d.ts.map +1 -0
  31. package/dist/inspector/standalone.d.ts +6 -0
  32. package/dist/inspector/standalone.d.ts.map +1 -0
  33. package/dist/inspector/standalone.js +420 -0
  34. package/dist/inspector/standalone.js.map +1 -0
  35. package/dist/inspector/tests/router/hash-router.test.d.ts +2 -0
  36. package/dist/inspector/tests/router/hash-router.test.d.ts.map +1 -0
  37. package/dist/inspector/tests/router/in-memory-router.test.d.ts +2 -0
  38. package/dist/inspector/tests/router/in-memory-router.test.d.ts.map +1 -0
  39. package/dist/inspector/ui/modal.d.ts +1 -0
  40. package/dist/inspector/ui/modal.d.ts.map +1 -1
  41. package/dist/inspector/viewer/breadcrumbs.d.ts +1 -7
  42. package/dist/inspector/viewer/breadcrumbs.d.ts.map +1 -1
  43. package/dist/inspector/viewer/header.d.ts +7 -0
  44. package/dist/inspector/viewer/header.d.ts.map +1 -0
  45. package/dist/inspector/viewer/page-stack.d.ts +4 -13
  46. package/dist/inspector/viewer/page-stack.d.ts.map +1 -1
  47. package/dist/inspector/viewer/page.d.ts.map +1 -1
  48. package/dist/react/hooks.d.ts +1 -1
  49. package/dist/react/hooks.d.ts.map +1 -1
  50. package/dist/react/index.d.ts +1 -1
  51. package/dist/react/index.d.ts.map +1 -1
  52. package/dist/react/index.js +5 -1
  53. package/dist/react/index.js.map +1 -1
  54. package/dist/react-core/hooks.d.ts +59 -0
  55. package/dist/react-core/hooks.d.ts.map +1 -1
  56. package/dist/react-core/index.js +124 -36
  57. package/dist/react-core/index.js.map +1 -1
  58. package/dist/react-core/tests/testUtils.d.ts +1 -0
  59. package/dist/react-core/tests/testUtils.d.ts.map +1 -1
  60. package/dist/react-core/tests/useSuspenseAccount.test.d.ts +2 -0
  61. package/dist/react-core/tests/useSuspenseAccount.test.d.ts.map +1 -0
  62. package/dist/react-core/tests/useSuspenseCoState.test.d.ts +2 -0
  63. package/dist/react-core/tests/useSuspenseCoState.test.d.ts.map +1 -0
  64. package/dist/react-core/use.d.ts +3 -0
  65. package/dist/react-core/use.d.ts.map +1 -0
  66. package/dist/react-native/index.js +5 -1
  67. package/dist/react-native/index.js.map +1 -1
  68. package/dist/react-native-core/crypto/RNCrypto.d.ts +2 -0
  69. package/dist/react-native-core/crypto/RNCrypto.d.ts.map +1 -0
  70. package/dist/react-native-core/crypto/RNCrypto.js +3 -0
  71. package/dist/react-native-core/crypto/RNCrypto.js.map +1 -0
  72. package/dist/react-native-core/hooks.d.ts +1 -1
  73. package/dist/react-native-core/hooks.d.ts.map +1 -1
  74. package/dist/react-native-core/index.js +5 -1
  75. package/dist/react-native-core/index.js.map +1 -1
  76. package/dist/react-native-core/platform.d.ts +2 -1
  77. package/dist/react-native-core/platform.d.ts.map +1 -1
  78. package/dist/testing.js +1 -1
  79. package/dist/testing.js.map +1 -1
  80. package/dist/tools/coValues/account.d.ts +7 -1
  81. package/dist/tools/coValues/account.d.ts.map +1 -1
  82. package/dist/tools/coValues/interfaces.d.ts +1 -1
  83. package/dist/tools/coValues/interfaces.d.ts.map +1 -1
  84. package/dist/tools/implementation/ContextManager.d.ts +3 -0
  85. package/dist/tools/implementation/ContextManager.d.ts.map +1 -1
  86. package/dist/tools/implementation/zodSchema/schemaTypes/AccountSchema.d.ts +8 -1
  87. package/dist/tools/implementation/zodSchema/schemaTypes/AccountSchema.d.ts.map +1 -1
  88. package/dist/tools/implementation/zodSchema/zodCo.d.ts.map +1 -1
  89. package/dist/tools/subscribe/CoValueCoreSubscription.d.ts +8 -22
  90. package/dist/tools/subscribe/CoValueCoreSubscription.d.ts.map +1 -1
  91. package/dist/tools/subscribe/SubscriptionCache.d.ts +51 -0
  92. package/dist/tools/subscribe/SubscriptionCache.d.ts.map +1 -0
  93. package/dist/tools/subscribe/SubscriptionScope.d.ts +17 -1
  94. package/dist/tools/subscribe/SubscriptionScope.d.ts.map +1 -1
  95. package/dist/tools/subscribe/utils.d.ts +9 -1
  96. package/dist/tools/subscribe/utils.d.ts.map +1 -1
  97. package/dist/tools/testing.d.ts +2 -2
  98. package/dist/tools/testing.d.ts.map +1 -1
  99. package/dist/tools/tests/SubscriptionCache.test.d.ts +2 -0
  100. package/dist/tools/tests/SubscriptionCache.test.d.ts.map +1 -0
  101. package/package.json +18 -6
  102. package/src/inspector/account-switcher.tsx +440 -0
  103. package/src/inspector/contexts/node.tsx +129 -0
  104. package/src/inspector/custom-element.tsx +2 -2
  105. package/src/inspector/in-app.tsx +61 -0
  106. package/src/inspector/index.tsx +2 -22
  107. package/src/inspector/pages/home.tsx +77 -0
  108. package/src/inspector/router/context.ts +21 -0
  109. package/src/inspector/router/hash-router.tsx +128 -0
  110. package/src/inspector/{viewer/use-page-path.ts → router/in-memory-router.tsx} +31 -29
  111. package/src/inspector/router/index.ts +4 -0
  112. package/src/inspector/standalone.tsx +60 -0
  113. package/src/inspector/tests/router/hash-router.test.tsx +847 -0
  114. package/src/inspector/tests/router/in-memory-router.test.tsx +724 -0
  115. package/src/inspector/ui/modal.tsx +5 -2
  116. package/src/inspector/viewer/breadcrumbs.tsx +5 -11
  117. package/src/inspector/viewer/header.tsx +67 -0
  118. package/src/inspector/viewer/page-stack.tsx +18 -26
  119. package/src/inspector/viewer/page.tsx +0 -1
  120. package/src/react/hooks.tsx +2 -0
  121. package/src/react/index.ts +1 -14
  122. package/src/react-core/hooks.ts +167 -18
  123. package/src/react-core/tests/createCoValueSubscriptionContext.test.tsx +18 -8
  124. package/src/react-core/tests/testUtils.tsx +67 -5
  125. package/src/react-core/tests/useCoState.test.ts +3 -7
  126. package/src/react-core/tests/useSubscriptionSelector.test.ts +3 -7
  127. package/src/react-core/tests/useSuspenseAccount.test.tsx +343 -0
  128. package/src/react-core/tests/useSuspenseCoState.test.tsx +1182 -0
  129. package/src/react-core/use.ts +46 -0
  130. package/src/react-native-core/crypto/RNCrypto.ts +1 -0
  131. package/src/react-native-core/hooks.tsx +2 -0
  132. package/src/react-native-core/platform.ts +2 -1
  133. package/src/tools/coValues/account.ts +13 -2
  134. package/src/tools/coValues/interfaces.ts +2 -3
  135. package/src/tools/implementation/ContextManager.ts +13 -0
  136. package/src/tools/implementation/zodSchema/schemaTypes/AccountSchema.ts +8 -1
  137. package/src/tools/subscribe/CoValueCoreSubscription.ts +71 -100
  138. package/src/tools/subscribe/SubscriptionCache.ts +272 -0
  139. package/src/tools/subscribe/SubscriptionScope.ts +113 -7
  140. package/src/tools/subscribe/utils.ts +77 -0
  141. package/src/tools/testing.ts +0 -3
  142. package/src/tools/tests/CoValueCoreSubscription.test.ts +46 -12
  143. package/src/tools/tests/ContextManager.test.ts +85 -0
  144. package/src/tools/tests/SubscriptionCache.test.ts +237 -0
  145. package/src/tools/tests/account.test.ts +11 -4
  146. package/src/tools/tests/coMap.test.ts +5 -7
  147. package/src/tools/tests/schema.resolved.test.ts +3 -3
  148. package/tsup.config.ts +2 -0
  149. package/dist/chunk-2S3Z2CN6.js.map +0 -1
  150. package/dist/inspector/custom-element-P76EIWEV.js.map +0 -1
  151. package/dist/inspector/viewer/new-app.d.ts.map +0 -1
  152. package/dist/inspector/viewer/use-page-path.d.ts +0 -10
  153. package/dist/inspector/viewer/use-page-path.d.ts.map +0 -1
  154. package/src/inspector/viewer/new-app.tsx +0 -156
@@ -0,0 +1,46 @@
1
+ import React from "react";
2
+
3
+ // shim from https://github.com/pmndrs/jotai/blob/f287c5d665a807e676bc731e83174c62c1fe1fc9/src/react/useAtomValue.ts#L13C1-L56C1
4
+ const attachPromiseStatus = <T>(
5
+ promise: PromiseLike<T> & {
6
+ status?: "pending" | "fulfilled" | "rejected";
7
+ value?: T;
8
+ reason?: unknown;
9
+ },
10
+ ) => {
11
+ if (!promise.status) {
12
+ promise.status = "pending";
13
+ promise.then(
14
+ (v) => {
15
+ promise.status = "fulfilled";
16
+ promise.value = v;
17
+ },
18
+ (e) => {
19
+ promise.status = "rejected";
20
+ promise.reason = e;
21
+ },
22
+ );
23
+ }
24
+ };
25
+
26
+ export const use =
27
+ React.use ||
28
+ // A shim for older React versions
29
+ (<T>(
30
+ promise: PromiseLike<T> & {
31
+ status?: "pending" | "fulfilled" | "rejected";
32
+ value?: T;
33
+ reason?: unknown;
34
+ },
35
+ ): T => {
36
+ if (promise.status === "pending") {
37
+ throw promise;
38
+ } else if (promise.status === "fulfilled") {
39
+ return promise.value as T;
40
+ } else if (promise.status === "rejected") {
41
+ throw promise.reason;
42
+ } else {
43
+ attachPromiseStatus(promise);
44
+ throw promise;
45
+ }
46
+ });
@@ -0,0 +1 @@
1
+ export * from "cojson/crypto/RNCrypto";
@@ -19,6 +19,8 @@ export {
19
19
  useCoValueSubscription,
20
20
  useAccountSubscription,
21
21
  useSubscriptionSelector,
22
+ useSuspenseCoState,
23
+ useSuspenseAccount,
22
24
  } from "jazz-tools/react-core";
23
25
 
24
26
  export function useAcceptInviteNative<S extends CoValueClassOrSchema>({
@@ -24,12 +24,13 @@ import { KvStore, KvStoreContext } from "./storage/kv-store-context.js";
24
24
  import { SQLiteDatabaseDriverAsync } from "cojson";
25
25
  import { WebSocketPeerWithReconnection } from "cojson-transport-ws";
26
26
  import type { RNQuickCrypto } from "jazz-tools/react-native-core/crypto";
27
+ import type { RNCrypto } from "cojson/crypto/RNCrypto";
27
28
 
28
29
  export type BaseReactNativeContextOptions = {
29
30
  sync: SyncConfig;
30
31
  reconnectionTimeout?: number;
31
32
  storage?: SQLiteDatabaseDriverAsync | "disabled";
32
- CryptoProvider?: typeof PureJSCrypto | typeof RNQuickCrypto;
33
+ CryptoProvider?: typeof PureJSCrypto | typeof RNQuickCrypto | typeof RNCrypto;
33
34
  authSecretStorage: AuthSecretStorage;
34
35
  };
35
36
 
@@ -273,7 +273,13 @@ export class Account extends CoValueBase implements CoValue {
273
273
  creationProps: { name: string };
274
274
  onCreate?: (account: A, worker: Account) => Promise<void>;
275
275
  },
276
- ) {
276
+ ): Promise<{
277
+ credentials: {
278
+ accountID: string;
279
+ accountSecret: AgentSecret;
280
+ };
281
+ account: A;
282
+ }> {
277
283
  const crypto = worker.$jazz.localNode.crypto;
278
284
 
279
285
  const connectedPeers = cojsonInternals.connectedPeers(
@@ -290,6 +296,11 @@ export class Account extends CoValueBase implements CoValue {
290
296
  peers: [connectedPeers[0]],
291
297
  });
292
298
 
299
+ const credentials = {
300
+ accountID: account.$jazz.id,
301
+ accountSecret: account.$jazz.localNode.getCurrentAgent().agentSecret,
302
+ };
303
+
293
304
  // Load the worker inside the account node
294
305
  const loadedWorker = await Account.load(worker.$jazz.id, {
295
306
  loadAs: account,
@@ -314,7 +325,7 @@ export class Account extends CoValueBase implements CoValue {
314
325
  // Close the account node, to avoid leaking memory
315
326
  account.$jazz.localNode.gracefulShutdown();
316
327
 
317
- return createdAccount;
328
+ return { credentials, account: createdAccount };
318
329
  }
319
330
 
320
331
  static fromNode<A extends Account>(
@@ -4,7 +4,7 @@ import {
4
4
  type CojsonInternalTypes,
5
5
  type RawCoValue,
6
6
  } from "cojson";
7
- import { AvailableCoValueCore } from "cojson/dist/coValueCore/coValueCore.js";
7
+ import { AvailableCoValueCore } from "cojson";
8
8
  import {
9
9
  Account,
10
10
  AnonymousJazzAgent,
@@ -23,7 +23,6 @@ import {
23
23
  ResolveQueryStrict,
24
24
  Resolved,
25
25
  SubscriptionScope,
26
- type SubscriptionValue,
27
26
  TypeSym,
28
27
  NotLoaded,
29
28
  activeAccountContext,
@@ -31,7 +30,7 @@ import {
31
30
  inspect,
32
31
  } from "../internal.js";
33
32
  import type { BranchDefinition } from "../subscribe/types.js";
34
- import { CoValueHeader } from "cojson/dist/coValueCore/verifiedState.js";
33
+ import { CoValueHeader } from "cojson";
35
34
 
36
35
  /** @category Abstract interfaces */
37
36
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
@@ -9,6 +9,7 @@ import { JazzContextType } from "../types.js";
9
9
  import { AnonymousJazzAgent } from "./anonymousJazzAgent.js";
10
10
  import { createAnonymousJazzContext } from "./createContext.js";
11
11
  import { InstanceOfSchema } from "./zodSchema/typeConverters/InstanceOfSchema.js";
12
+ import { SubscriptionCache } from "../subscribe/SubscriptionCache.js";
12
13
 
13
14
  export type JazzContextManagerAuthProps = {
14
15
  credentials?: AuthCredentials;
@@ -75,6 +76,7 @@ export class JazzContextManager<
75
76
  protected keepContextOpen = false;
76
77
  contextPromise: Promise<void> | undefined;
77
78
  protected authenticatingAccountID: string | null = null;
79
+ private subscriptionCache: SubscriptionCache;
78
80
 
79
81
  constructor(opts?: {
80
82
  useAnonymousFallback?: boolean;
@@ -82,6 +84,7 @@ export class JazzContextManager<
82
84
  }) {
83
85
  KvStoreContext.getInstance().initialize(this.getKvStore());
84
86
  this.authSecretStorage = new AuthSecretStorage(opts?.authSecretStorageKey);
87
+ this.subscriptionCache = new SubscriptionCache();
85
88
 
86
89
  if (opts?.useAnonymousFallback) {
87
90
  this.value = getAnonymousFallback();
@@ -130,6 +133,9 @@ export class JazzContextManager<
130
133
  context: PlatformSpecificContext<Acc>,
131
134
  authProps?: JazzContextManagerAuthProps,
132
135
  ) {
136
+ // Clear cache before updating context to prevent subscription leaks across authentication boundaries
137
+ this.subscriptionCache.clear();
138
+
133
139
  // When keepContextOpen we don't want to close the previous context
134
140
  // because we might need to handle the onAnonymousAccountDiscarded callback
135
141
  if (!this.keepContextOpen) {
@@ -178,6 +184,10 @@ export class JazzContextManager<
178
184
  return this.authenticatingAccountID;
179
185
  }
180
186
 
187
+ getSubscriptionScopeCache(): SubscriptionCache {
188
+ return this.subscriptionCache;
189
+ }
190
+
181
191
  logOut = async () => {
182
192
  if (!this.context || !this.props) {
183
193
  return;
@@ -185,6 +195,9 @@ export class JazzContextManager<
185
195
 
186
196
  this.authenticatingAccountID = null;
187
197
 
198
+ // Clear cache on logout to prevent subscription leaks across authentication boundaries
199
+ this.subscriptionCache.clear();
200
+
188
201
  await this.props.onLogOut?.();
189
202
 
190
203
  if (this.props.logOutReplacement) {
@@ -26,6 +26,7 @@ import {
26
26
  import { CoOptionalSchema } from "./CoOptionalSchema.js";
27
27
  import { CoreResolveQuery } from "./CoValueSchema.js";
28
28
  import { withSchemaResolveQuery } from "../../schemaUtils.js";
29
+ import { AgentSecret } from "cojson";
29
30
 
30
31
  export type BaseProfileShape = {
31
32
  name: z.core.$ZodString<string>;
@@ -103,8 +104,14 @@ export class AccountSchema<
103
104
  worker: Account,
104
105
  ) => Promise<void>;
105
106
  },
107
+ ): Promise<{
108
+ credentials: {
109
+ accountID: string;
110
+ accountSecret: AgentSecret;
111
+ };
106
112
  // @ts-expect-error we can't statically enforce the schema's resolve query is a valid resolve query, but in practice it is
107
- ): Promise<Loaded<AccountSchema<Shape>, DefaultResolveQuery>> {
113
+ account: Loaded<AccountSchema<Shape>, DefaultResolveQuery>;
114
+ }> {
108
115
  // @ts-expect-error
109
116
  return this.coValueClass.createAs(worker, options);
110
117
  }
@@ -1,6 +1,7 @@
1
1
  import {
2
2
  cojsonInternals,
3
3
  CoValueCore,
4
+ isRawCoID,
4
5
  LocalNode,
5
6
  RawCoID,
6
7
  RawCoValue,
@@ -67,121 +68,76 @@ export class CoValueCoreSubscription {
67
68
  private initializeSubscription(): void {
68
69
  const source = this.source;
69
70
 
70
- // If the CoValue is already available, handle it immediately
71
- if (source.isAvailable()) {
72
- this.handleAvailableSource();
71
+ // If the ID is not a valid raw CoID, we immediately emit an unavailable event
72
+ if (!isRawCoID(source.id)) {
73
+ this.emit(CoValueLoadingState.UNAVAILABLE);
73
74
  return;
74
75
  }
75
76
 
76
- // If a specific branch is requested while the source is not available, attempt to checkout that branch
77
+ // If we have a branch name, we handle branching
77
78
  if (this.branchName) {
78
- this.handleBranchCheckout();
79
+ this.handleBranching(this.branchName, this.branchOwnerId);
79
80
  return;
80
81
  }
81
82
 
82
- // If we don't have a branch requested, load the CoValue
83
- this.loadCoValue();
83
+ // If we don't have a branch name, we subscribe to the source directly
84
+ this.subscribe(this.source);
84
85
  }
85
86
 
86
- /**
87
- * Handles the case where the CoValue source is immediately available.
88
- * Either subscribes directly or attempts to get the requested branch.
89
- */
90
- private handleAvailableSource(): void {
91
- if (!this.branchName || !cojsonInternals.canBeBranched(this.source)) {
92
- this.subscribe(this.source.getCurrentContent());
87
+ private handleBranching(branchName: string, branchOwnerId?: RawCoID) {
88
+ const source = this.source;
89
+
90
+ // If the source is not available, we wait for it to become available and then try to branch
91
+ if (!source.isAvailable()) {
92
+ this.waitForSourceToBecomeAvailable(branchName, branchOwnerId);
93
93
  return;
94
94
  }
95
95
 
96
- // Try to get the specific branch from the available source
97
- const branch = this.source.getBranch(this.branchName, this.branchOwnerId);
98
-
99
- if (branch.isAvailable()) {
100
- // Branch is available, subscribe to it
101
- this.subscribe(branch.getCurrentContent());
96
+ // If the source is not branchable (e.g. it is a group), we subscribe to it directly
97
+ if (!cojsonInternals.canBeBranched(source)) {
98
+ this.subscribe(source);
102
99
  return;
103
- // If the branch hasn't been created, we create it directly so we can syncronously subscribe to it
104
- } else if (!this.source.hasBranch(this.branchName, this.branchOwnerId)) {
105
- this.source.createBranch(this.branchName, this.branchOwnerId);
106
- this.subscribe(branch.getCurrentContent());
107
- } else {
108
- // Branch not available, fall through to checkout logic
109
- this.handleBranchCheckout();
110
100
  }
111
- }
112
101
 
113
- /**
114
- * Attempts to checkout a specific branch of the CoValue.
115
- * This is called when the source isn't available but a branch is requested.
116
- */
117
- private handleBranchCheckout(): void {
118
- this.localNode
119
- .checkoutBranch(this.source.id, this.branchName!, this.branchOwnerId)
120
- .then((value) => {
121
- if (this.unsubscribed) return;
122
-
123
- if (value !== CoValueLoadingState.UNAVAILABLE) {
124
- // Branch checkout successful, subscribe to it
125
- this.subscribe(value);
126
- } else {
127
- // Branch checkout failed, handle the error
128
- this.handleUnavailableBranch();
129
- }
130
- })
131
- .catch((error) => {
132
- // Handle unexpected errors during branch checkout
133
- console.error(error);
102
+ // Try to get the specific branch from the available source
103
+ const branch = source.getBranch(branchName, branchOwnerId);
104
+
105
+ // If the branch hasn't been created, we create it directly so we can syncronously subscribe to it
106
+ if (!branch.isAvailable() && !source.hasBranch(branchName, branchOwnerId)) {
107
+ try {
108
+ source.createBranch(branchName, branchOwnerId);
109
+ } catch (error) {
110
+ // If the branch creation fails (provided group is not available), we emit an unavailable event
111
+ console.error("error creating branch", error);
134
112
  this.emit(CoValueLoadingState.UNAVAILABLE);
135
- });
136
- }
137
-
138
- /**
139
- * Handles the case where a branch checkout fails.
140
- * Determines whether to retry or report unavailability.
141
- */
142
- private handleUnavailableBranch(): void {
143
- const source = this.source;
144
- if (source.isAvailable()) {
145
- // This should be impossible - if source is available we can create the branch and it should be available
146
- throw new Error("Branch is unavailable");
113
+ return;
114
+ }
147
115
  }
148
116
 
149
- // Source isn't available either, subscribe to state changes and report unavailability
150
- this.subscribeToUnavailableSource();
151
- this.emit(CoValueLoadingState.UNAVAILABLE);
117
+ this.subscribe(branch);
152
118
  }
153
119
 
154
120
  /**
155
- * Loads the CoValue core from the network/storage.
156
- * This is the fallback strategy when immediate availability fails.
121
+ * Loads a CoValue core and emits an unavailable event if it is still unavailable after the retries.
157
122
  */
158
- private loadCoValue(): void {
123
+ load(value: CoValueCore) {
159
124
  this.localNode
160
- .loadCoValueCore(this.source.id, undefined, this.skipRetry)
161
- .then((value) => {
162
- if (this.unsubscribed) return;
163
-
164
- if (value.isAvailable()) {
165
- // Loading successful, subscribe to the loaded value
166
- this.subscribe(value.getCurrentContent());
167
- } else {
168
- // Loading failed, subscribe to state changes and report unavailability
169
- this.subscribeToUnavailableSource();
125
+ .loadCoValueCore(value.id, undefined, this.skipRetry)
126
+ .then(() => {
127
+ // If after the retries the value is still unavailable, we emit an unavailable event
128
+ if (!value.isAvailable()) {
170
129
  this.emit(CoValueLoadingState.UNAVAILABLE);
171
130
  }
172
- })
173
- .catch((error) => {
174
- // Handle unexpected errors during loading
175
- console.error(error);
176
- this.emit(CoValueLoadingState.UNAVAILABLE);
177
131
  });
178
132
  }
179
133
 
180
134
  /**
181
- * Subscribes to state changes of an unavailable CoValue source.
182
- * This allows the subscription to become active when the source becomes available after a first loading attempt.
135
+ * Waits for the source to become available and then tries to branch.
183
136
  */
184
- private subscribeToUnavailableSource(): void {
137
+ private waitForSourceToBecomeAvailable(
138
+ branchName: string,
139
+ branchOwnerId?: RawCoID,
140
+ ): void {
185
141
  const source = this.source;
186
142
 
187
143
  const handleStateChange = (
@@ -197,41 +153,60 @@ export class CoValueCoreSubscription {
197
153
 
198
154
  unsubFromStateChange();
199
155
 
200
- if (this.branchName) {
201
- // Branch was requested, attempt checkout again
202
- this.handleBranchCheckout();
203
- } else {
204
- // No branch requested, subscribe directly and cleanup state subscription
205
- this.subscribe(source.getCurrentContent());
206
- }
156
+ this.handleBranching(branchName, branchOwnerId);
207
157
  };
208
158
 
209
159
  // Subscribe to state changes and store the unsubscribe function
210
160
  this._unsubscribe = source.subscribe(handleStateChange);
161
+
162
+ this.load(source);
211
163
  }
212
164
 
213
165
  /**
214
166
  * Subscribes to a specific CoValue and notifies the listener.
215
167
  * This is the final step where we actually start receiving updates.
216
168
  */
217
- private subscribe(value: RawCoValue): void {
169
+ private subscribe(value: CoValueCore): void {
218
170
  if (this.unsubscribed) return;
219
171
 
220
172
  // Subscribe to the value and store the unsubscribe function
221
173
  this._unsubscribe = value.subscribe((value) => {
222
- this.emit(value);
174
+ if (value.isAvailable()) {
175
+ this.emit(value.getCurrentContent());
176
+ }
223
177
  });
178
+
179
+ if (!value.isAvailable()) {
180
+ this.load(value);
181
+ }
224
182
  }
225
183
 
184
+ lastState: CoValueLoadingState | undefined;
185
+
226
186
  emit(value: RawCoValue | typeof CoValueLoadingState.UNAVAILABLE): void {
227
187
  if (this.unsubscribed) return;
228
- if (!isReadyForEmit(value)) {
188
+ if (!this.isReadyForEmit(value)) {
229
189
  return;
230
190
  }
231
191
 
232
192
  this.listener(value);
233
193
  }
234
194
 
195
+ isReadyForEmit(
196
+ value: RawCoValue | typeof CoValueLoadingState.UNAVAILABLE,
197
+ ): boolean {
198
+ if (value === CoValueLoadingState.UNAVAILABLE) {
199
+ return true;
200
+ }
201
+
202
+ // If the value is not completely downloaded, we don't emit it to avoid providing partial data to the listener.
203
+ if (!isCompletelyDownloaded(value)) {
204
+ return false;
205
+ }
206
+
207
+ return true;
208
+ }
209
+
235
210
  /**
236
211
  * Unsubscribes from all active subscriptions and marks the instance as unsubscribed.
237
212
  * This prevents any further operations and ensures proper cleanup.
@@ -246,11 +221,7 @@ export class CoValueCoreSubscription {
246
221
  /**
247
222
  * This is true if the value is unavailable, or if the value is a binary coValue or a completely downloaded coValue.
248
223
  */
249
- function isReadyForEmit(value: RawCoValue | "unavailable") {
250
- if (value === "unavailable") {
251
- return true;
252
- }
253
-
224
+ function isCompletelyDownloaded(value: RawCoValue) {
254
225
  return (
255
226
  value.core.verified?.header.meta?.type === "binary" ||
256
227
  value.core.isCompletelyDownloaded()