Skip to content

Commit c3d77e3

Browse files
authored
feat: ensure data utils work well with legacy relationship proxies (#9317)
* feat: ensure data utils work well with legacy relationship proxies * fixup types * fixup * fixup * fix lint
1 parent 91283cd commit c3d77e3

File tree

7 files changed

+148
-8
lines changed

7 files changed

+148
-8
lines changed

packages/diagnostic/src/-define.ts

+9-3
Original file line numberDiff line numberDiff line change
@@ -64,9 +64,11 @@ export function test<TC extends TestContext = TestContext>(name: string, cb: Tes
6464
assert(`Cannot add the same test name twice: ${name}`, !currentModule.tests.byName.has(name));
6565
Config.totals.tests++;
6666

67+
const testName = currentModule.moduleName + ' > ' + name;
6768
const testInfo = {
68-
id: generateHash(currentModule.moduleName + ' > ' + name),
69+
id: generateHash(testName),
6970
name,
71+
testName,
7072
cb,
7173
skip: false,
7274
todo: false,
@@ -83,9 +85,11 @@ export function todo<TC extends TestContext = TestContext>(name: string, cb: Tes
8385
assert(`Cannot add the same test name twice: ${name}`, !currentModule.tests.byName.has(name));
8486
Config.totals.todo++;
8587

88+
const testName = currentModule.moduleName + ' > ' + name;
8689
const testInfo = {
87-
id: generateHash(currentModule.moduleName + ' > ' + name),
90+
id: generateHash(testName),
8891
name,
92+
testName,
8993
cb,
9094
skip: false,
9195
todo: true,
@@ -102,9 +106,11 @@ export function skip<TC extends TestContext = TestContext>(name: string, cb: Tes
102106
assert(`Cannot add the same test name twice: ${name}`, !currentModule.tests.byName.has(name));
103107
Config.totals.skipped++;
104108

109+
const testName = currentModule.moduleName + ' > ' + name;
105110
const testInfo = {
106-
id: generateHash(currentModule.moduleName + ' > ' + name),
111+
id: generateHash(testName),
107112
name,
113+
testName,
108114
cb,
109115
skip: true,
110116
todo: false,

packages/diagnostic/src/-types.ts

+4
Original file line numberDiff line numberDiff line change
@@ -100,8 +100,12 @@ export type ModuleCallback<TC extends TestContext> = ((hooks: Hooks<TC>) => void
100100
export type TestCallback<TC extends TestContext> = (this: TC, assert: Diagnostic) => void | Promise<void>;
101101

102102
export interface TestInfo<TC extends TestContext> {
103+
/* A unique id for the test based on the hash of the full testName */
103104
id: string;
105+
/* The name of the test, not including moduleName */
104106
name: string;
107+
/* The full name of the test including moduleName */
108+
testName: string;
105109
cb: TestCallback<TC>;
106110
skip: boolean;
107111
todo: boolean;

packages/diagnostic/src/internals/run.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ export async function runTest<TC extends TestContext>(
1616
const testContext = {
1717
[PublicTestInfo]: {
1818
id: test.id,
19-
name: test.name,
19+
name: test.testName,
2020
},
2121
} as unknown as TC;
2222
const testReport: TestReport = {

packages/ember/src/-private/promise-state.ts

+15-3
Original file line numberDiff line numberDiff line change
@@ -44,12 +44,24 @@ export class PromiseState<T = unknown, E = unknown> {
4444
}
4545
}
4646

47+
const LegacyPromiseProxy = Symbol.for('LegacyPromiseProxy');
48+
type LegacyAwaitable<T, E> = { promise: Promise<T> | Awaitable<T, E>; [LegacyPromiseProxy]: true };
49+
50+
function isLegacyAwaitable<T, E>(promise: object): promise is LegacyAwaitable<T, E> {
51+
return LegacyPromiseProxy in promise && 'promise' in promise && promise[LegacyPromiseProxy] === true;
52+
}
53+
54+
function getPromise<T, E>(promise: Promise<T> | Awaitable<T, E> | LegacyAwaitable<T, E>): Promise<T> | Awaitable<T, E> {
55+
return isLegacyAwaitable(promise) ? promise.promise : promise;
56+
}
57+
4758
export function getPromiseState<T = unknown, E = unknown>(promise: Promise<T> | Awaitable<T, E>): PromiseState<T, E> {
48-
let state = PromiseCache.get(promise) as PromiseState<T, E> | undefined;
59+
const _promise = getPromise(promise);
60+
let state = PromiseCache.get(_promise) as PromiseState<T, E> | undefined;
4961

5062
if (!state) {
51-
state = new PromiseState(promise);
52-
PromiseCache.set(promise, state);
63+
state = new PromiseState(_promise);
64+
PromiseCache.set(_promise, state);
5365
}
5466

5567
return state;

packages/model/src/-private/promise-belongs-to.ts

+4
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ export interface BelongsToProxyCreateArgs<T = unknown> {
2323
_belongsToState: BelongsToProxyMeta<T>;
2424
}
2525

26+
export const LegacyPromiseProxy = Symbol.for('LegacyPromiseProxy');
27+
2628
interface PromiseObjectType<T> extends PromiseProxyMixin<T | null>, ObjectProxy<T> {
2729
// eslint-disable-next-line @typescript-eslint/no-misused-new
2830
new <PT>(...args: unknown[]): PromiseObjectType<PT>;
@@ -80,6 +82,8 @@ class PromiseBelongsTo<T = unknown> extends Extended<T> {
8082
await legacySupport.reloadBelongsTo(key, options);
8183
return this;
8284
}
85+
86+
[LegacyPromiseProxy] = true as const;
8387
}
8488

8589
export { PromiseBelongsTo };

packages/model/src/-private/promise-many-array.ts

+3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import { compat } from '@ember-data/tracking';
66
import { defineSignal } from '@ember-data/tracking/-private';
77

88
import type ManyArray from './many-array';
9+
import { LegacyPromiseProxy } from './promise-belongs-to';
910

1011
export interface HasManyProxyCreateArgs<T = unknown> {
1112
promise: Promise<ManyArray<T>>;
@@ -195,6 +196,8 @@ export default class PromiseManyArray<T = unknown> {
195196
static create<T>({ promise, content }: HasManyProxyCreateArgs<T>): PromiseManyArray<T> {
196197
return new this(promise, content);
197198
}
199+
200+
[LegacyPromiseProxy] = true as const;
198201
}
199202
defineSignal(PromiseManyArray.prototype, 'content', null);
200203
defineSignal(PromiseManyArray.prototype, 'isPending', false);

tests/warp-drive__ember/tests/integration/get-promise-state-test.gts

+112-1
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,38 @@
11
import { rerender, settled } from '@ember/test-helpers';
22

3-
import { createDeferred, setPromiseResult } from '@ember-data/request';
3+
import { createDeferred, setPromiseResult, type Awaitable } from '@ember-data/request';
44
import type { RenderingTestContext } from '@warp-drive/diagnostic/ember';
55
import { module, setupRenderingTest, test } from '@warp-drive/diagnostic/ember';
66
import { getPromiseState } from '@warp-drive/ember';
77

88
type PromiseState<T, E> = ReturnType<typeof getPromiseState<T, E>>;
9+
const SecretSymbol = Symbol.for('LegacyPromiseProxy');
10+
11+
interface PromiseProxy<T, E> extends Promise<T> {}
12+
class PromiseProxy<T, E> {
13+
[SecretSymbol]: true;
14+
promise: Awaitable<T, E>;
15+
16+
constructor(promise: Awaitable<T, E>) {
17+
this[SecretSymbol] = true;
18+
this.promise = promise;
19+
}
20+
21+
then<T1, T2>(
22+
onFulfilled?: ((value: T) => unknown) | undefined | null,
23+
onRejected?: ((error: E) => T2 | Promise<T2>) | undefined | null
24+
): Promise<T1 | T2> {
25+
return this.promise.then(onFulfilled!, onRejected!) as Promise<T1 | T2>;
26+
}
27+
28+
catch<T2>(onRejected: ((error: E) => T2 | Promise<T2>) | undefined | null): Promise<T2> {
29+
return this.promise.catch(onRejected!) as Promise<T2>;
30+
}
31+
32+
finally(onFinally: () => void): Promise<T> {
33+
return this.promise.finally(onFinally) as Promise<T>;
34+
}
35+
}
936

1037
module('Integration | get-promise-state', function (hooks) {
1138
setupRenderingTest(hooks);
@@ -196,4 +223,88 @@ module('Integration | get-promise-state', function (hooks) {
196223
assert.equal(counter, 1);
197224
assert.equal(this.element.textContent?.trim(), 'Our Error\n Count:\n 1');
198225
});
226+
227+
test('it unwraps promise-proxies that utilize the secret symbol for error states', async function (this: RenderingTestContext, assert) {
228+
const _promise = Promise.resolve().then(() => {
229+
throw new Error('Our Error');
230+
});
231+
const promise = new PromiseProxy<never, Error>(_promise);
232+
233+
try {
234+
getPromiseState(promise);
235+
await promise;
236+
} catch {
237+
// do nothing
238+
}
239+
240+
let state: PromiseState<string, Error>;
241+
function _getPromiseState<T>(p: Promise<T>): PromiseState<T, Error> {
242+
state = getPromiseState(p) as PromiseState<string, Error>;
243+
return state as PromiseState<T, Error>;
244+
}
245+
let counter = 0;
246+
function countFor(_result: unknown, _error: unknown) {
247+
return ++counter;
248+
}
249+
250+
await this.render(
251+
<template>
252+
{{#let (_getPromiseState promise) as |state|}}
253+
{{#if state.isPending}}
254+
Pending
255+
{{else if state.isError}}
256+
{{state.error.message}}
257+
{{else if state.isSuccess}}
258+
Invalid Success Reached
259+
{{/if}}
260+
<br />Count:
261+
{{countFor state.result state.error}}{{/let}}
262+
</template>
263+
);
264+
265+
assert.equal(state!.result, null);
266+
assert.true(state!.error instanceof Error);
267+
assert.equal((state!.error as Error | undefined)?.message, 'Our Error');
268+
assert.equal(counter, 1);
269+
assert.equal(this.element.textContent?.trim(), 'Our Error\n Count:\n 1');
270+
await rerender();
271+
assert.equal(state!.result, null);
272+
assert.true(state!.error instanceof Error);
273+
assert.equal((state!.error as Error | undefined)?.message, 'Our Error');
274+
assert.equal(counter, 1);
275+
assert.equal(this.element.textContent?.trim(), 'Our Error\n Count:\n 1');
276+
assert.equal(state!, getPromiseState(_promise));
277+
});
278+
279+
test('it unwraps promise-proxies that utilize the secret symbol for success states', async function (this: RenderingTestContext, assert) {
280+
const _promise = Promise.resolve().then(() => 'Our Data');
281+
const promise = new PromiseProxy<string, Error>(_promise);
282+
getPromiseState(promise);
283+
await promise;
284+
285+
let state: PromiseState<string, Error>;
286+
function _getPromiseState<T>(p: Promise<T>): PromiseState<T, Error> {
287+
state = getPromiseState(p) as PromiseState<string, Error>;
288+
return state as PromiseState<T, Error>;
289+
}
290+
let counter = 0;
291+
function countFor(_result: unknown) {
292+
return ++counter;
293+
}
294+
295+
await this.render(
296+
<template>
297+
{{#let (_getPromiseState promise) as |state|}}
298+
{{state.result}}<br />Count:
299+
{{countFor state.result}}
300+
{{/let}}
301+
</template>
302+
);
303+
304+
assert.equal(this.element.textContent?.trim(), 'Our DataCount:\n 1');
305+
await settled();
306+
307+
assert.equal(this.element.textContent?.trim(), 'Our DataCount:\n 1');
308+
assert.equal(state!, getPromiseState(_promise));
309+
});
199310
});

0 commit comments

Comments
 (0)