Skip to content

Commit 10e0e6e

Browse files
committed
move shared module-scoped state to globals
This is a followup to the change that switch to Symbol.for, with the intent of making it not matter which loader you retrive cardApi from. It turns out we also need to make sure you're sharing the same module-scoped WeakMaps, etc.
1 parent 32637cf commit 10e0e6e

File tree

4 files changed

+68
-13
lines changed

4 files changed

+68
-13
lines changed

packages/base/card-api.gts

+17-4
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ import {
4848
type RealmInfo,
4949
} from '@cardstack/runtime-common';
5050
import type { ComponentLike } from '@glint/template';
51+
import { initSharedState } from './shared-state';
5152

5253
export { primitive, isField, type BoxComponent };
5354
export const serialize = Symbol.for('cardstack-serialize');
@@ -170,10 +171,22 @@ function isStaleValue(value: any): value is StaleValue {
170171
return false;
171172
}
172173
}
173-
const deserializedData = new WeakMap<BaseDef, Map<string, any>>();
174-
const recomputePromises = new WeakMap<BaseDef, Promise<any>>();
175-
const identityContexts = new WeakMap<BaseDef, IdentityContext>();
176-
const subscribers = new WeakMap<BaseDef, Set<CardChangeSubscriber>>();
174+
const deserializedData = initSharedState(
175+
'deserializedData',
176+
() => new WeakMap<BaseDef, Map<string, any>>(),
177+
);
178+
const recomputePromises = initSharedState(
179+
'recomputePromises',
180+
() => new WeakMap<BaseDef, Promise<any>>(),
181+
);
182+
const identityContexts = initSharedState(
183+
'identityContexts',
184+
() => new WeakMap<BaseDef, IdentityContext>(),
185+
);
186+
const subscribers = initSharedState(
187+
'subscribers',
188+
() => new WeakMap<BaseDef, Set<CardChangeSubscriber>>(),
189+
);
177190

178191
// our place for notifying Glimmer when a card is ready to re-render (which will
179192
// involve rerunning async computed fields)

packages/base/field-component.gts

+5-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import { getField } from '@cardstack/runtime-common';
1616
import type { ComponentLike } from '@glint/template';
1717
import { CardContainer } from '@cardstack/boxel-ui/components';
1818
import Modifier from 'ember-modifier';
19+
import { initSharedState } from './shared-state';
1920

2021
interface BoxComponentSignature {
2122
Args: { Named: { format?: Format } };
@@ -24,7 +25,10 @@ interface BoxComponentSignature {
2425

2526
export type BoxComponent = ComponentLike<BoxComponentSignature>;
2627

27-
const componentCache = new WeakMap<Box<BaseDef>, BoxComponent>();
28+
const componentCache = initSharedState(
29+
'componentCache',
30+
() => new WeakMap<Box<BaseDef>, BoxComponent>(),
31+
);
2832

2933
export function getBoxComponent(
3034
card: typeof BaseDef,

packages/base/room.gts

+21-8
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ import {
2121
} from '@cardstack/runtime-common';
2222
//@ts-expect-error cached type not available yet
2323
import { cached } from '@glimmer/tracking';
24+
import { initSharedState } from './shared-state';
2425

2526
// this is so we can have triple equals equivalent room member cards
2627
function upsertRoomMember({
@@ -275,14 +276,26 @@ interface RoomState {
275276

276277
// in addition to acting as a cache, this also ensures we have
277278
// triple equal equivalence for the interior cards of RoomField
278-
const eventCache = new WeakMap<RoomField, Map<string, MatrixEvent>>();
279-
const messageCache = new WeakMap<RoomField, Map<string, MessageField>>();
280-
const roomMemberCache = new WeakMap<RoomField, Map<string, RoomMemberField>>();
281-
const roomStateCache = new WeakMap<RoomField, RoomState>();
282-
const fragmentCache = new WeakMap<
283-
RoomField,
284-
Map<string, CardFragmentContent>
285-
>();
279+
const eventCache = initSharedState(
280+
'eventCache',
281+
() => new WeakMap<RoomField, Map<string, MatrixEvent>>(),
282+
);
283+
const messageCache = initSharedState(
284+
'messageCache',
285+
() => new WeakMap<RoomField, Map<string, MessageField>>(),
286+
);
287+
const roomMemberCache = initSharedState(
288+
'roomMemberCache',
289+
() => new WeakMap<RoomField, Map<string, RoomMemberField>>(),
290+
);
291+
const roomStateCache = initSharedState(
292+
'roomStateCache',
293+
() => new WeakMap<RoomField, RoomState>(),
294+
);
295+
const fragmentCache = initSharedState(
296+
'fragmentCache',
297+
() => new WeakMap<RoomField, Map<string, CardFragmentContent>>(),
298+
);
286299

287300
export class RoomField extends FieldDef {
288301
static displayName = 'Room';

packages/base/shared-state.ts

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
/*
2+
This module needs to exist so long as we have multiple loaders cooperating in
3+
the same environment. It ensures that any shared global state is really
4+
global. Ultimately we would like to get to the point where the loader itself
5+
is scoped so broadly that there's only one, and module-scoped state is safe to
6+
treat as global.
7+
*/
8+
9+
const bucket: Map<string, unknown> = (() => {
10+
let g = globalThis as unknown as {
11+
__card_api_shared_state: Map<string, unknown> | undefined;
12+
};
13+
if (!g.__card_api_shared_state) {
14+
g.__card_api_shared_state = new Map();
15+
}
16+
return g.__card_api_shared_state;
17+
})();
18+
19+
export function initSharedState<T>(key: string, fn: () => T): T {
20+
if (bucket.has(key)) {
21+
return bucket.get(key) as T;
22+
}
23+
bucket.set(key, fn());
24+
return bucket.get(key) as T;
25+
}

0 commit comments

Comments
 (0)