Skip to content

Commit c9d7adb

Browse files
committed
chore: begin work on consumer types
1 parent 7b03579 commit c9d7adb

File tree

5 files changed

+100
-23
lines changed

5 files changed

+100
-23
lines changed

packages/core-types/src/record.ts

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
/**
2+
* @module @warp-drive/core-types
3+
*/
4+
import type { ResourceType } from 'symbols';
5+
6+
/**
7+
* Records may be anything, They don't even
8+
* have to be objects.
9+
*
10+
* Whatever they are, if they have a ResourceType
11+
* property, that property will be used by EmberData
12+
* and WarpDrive to provide better type safety and
13+
* intellisense.
14+
*
15+
* @class TypedRecordInstance
16+
*/
17+
export interface TypedRecordInstance {
18+
/**
19+
* The type of the resource.
20+
*
21+
* This is an optional feature that can be used by
22+
* record implementations to provide a typescript
23+
* hint for the type of the resource.
24+
*
25+
* When used, EmberData and WarpDrive APIs can
26+
* take advantage of this to provide better type
27+
* safety and intellisense.
28+
*
29+
* @property {ResourceType} [ResourceType]
30+
* @type {string}
31+
*
32+
*/
33+
[ResourceType]?: string;
34+
}
35+
36+
export type TypeFromInstance<T> = T extends TypedRecordInstance ? T[typeof ResourceType] : never;

packages/core-types/src/spec/raw.ts

+7-7
Original file line numberDiff line numberDiff line change
@@ -27,9 +27,9 @@ export interface PaginationLinks extends Links {
2727
* [JSON:API Spec](https://jsonapi.org/format/#document-resource-identifier-objects)
2828
* @internal
2929
*/
30-
export interface ExistingResourceIdentifierObject {
30+
export interface ExistingResourceIdentifierObject<T extends string = string> {
3131
id: string;
32-
type: string;
32+
type: T;
3333

3434
/**
3535
* While not officially part of the `JSON:API` spec,
@@ -65,7 +65,7 @@ export interface ExistingResourceIdentifierObject {
6565
*
6666
* @internal
6767
*/
68-
export interface NewResourceIdentifierObject {
68+
export interface NewResourceIdentifierObject<T extends string = string> {
6969
/**
7070
* Resources newly created on the client _may_
7171
* not have an `id` available to them prior
@@ -76,7 +76,7 @@ export interface NewResourceIdentifierObject {
7676
* @internal
7777
*/
7878
id: string | null;
79-
type: string;
79+
type: T;
8080

8181
/**
8282
* Resources newly created on the client _will always_
@@ -90,10 +90,10 @@ export interface ResourceIdentifier {
9090
lid: string;
9191
}
9292

93-
export type ResourceIdentifierObject =
93+
export type ResourceIdentifierObject<T extends string = string> =
9494
| ResourceIdentifier
95-
| ExistingResourceIdentifierObject
96-
| NewResourceIdentifierObject;
95+
| ExistingResourceIdentifierObject<T>
96+
| NewResourceIdentifierObject<T>;
9797

9898
// TODO disallow NewResource, make narrowable
9999
export interface SingleResourceRelationship {

packages/core-types/src/symbols.ts

+18
Original file line numberDiff line numberDiff line change
@@ -1 +1,19 @@
1+
/**
2+
* @module @warp-drive/core-types
3+
*/
14
export const RecordStore = Symbol('Store');
5+
6+
/**
7+
* Symbol for the type of a resource.
8+
*
9+
* This is an optional feature that can be used by
10+
* record implementations to provide a typescript
11+
* hint for the type of the resource.
12+
*
13+
* When used, EmberData and WarpDrive APIs can
14+
* take advantage of this to provide better type
15+
* safety and intellisense.
16+
*
17+
* @type {Symbol}
18+
*/
19+
export const ResourceType = Symbol('$type');

packages/store/src/-private/store-service.ts

+38-15
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import type {
1515
StableExistingRecordIdentifier,
1616
StableRecordIdentifier,
1717
} from '@warp-drive/core-types/identifier';
18+
import type { TypedRecordInstance, TypeFromInstance } from '@warp-drive/core-types/record';
1819
import { EnableHydration, SkipCache } from '@warp-drive/core-types/request';
1920
import type { ResourceDocument } from '@warp-drive/core-types/spec/document';
2021
import type {
@@ -24,6 +25,7 @@ import type {
2425
ResourceIdentifierObject,
2526
SingleResourceDocument,
2627
} from '@warp-drive/core-types/spec/raw';
28+
import type { ResourceType } from '@warp-drive/core-types/symbols';
2729

2830
import type { Cache, CacheV1 } from '../-types/q/cache';
2931
import type { CacheCapabilitiesManager } from '../-types/q/cache-store-wrapper';
@@ -64,10 +66,26 @@ type CompatStore = Store & {
6466
};
6567
function upgradeStore(store: Store): asserts store is CompatStore {}
6668

67-
export interface CreateRecordProperties {
68-
id?: string | null;
69-
[key: string]: unknown;
70-
}
69+
type MaybeHasId = { id?: string };
70+
/**
71+
* Currently only records that extend object can be created via
72+
* store.createRecord. This is a limitation of the current API,
73+
* but can be worked around by creating a new identifier, running
74+
* the cache.clientDidCreate method, and then peeking the record
75+
* for the identifier.
76+
*
77+
* To assign primary key to a record during creation, only `id` will
78+
* work correctly for `store.createRecord`, other primary key may be
79+
* handled by updating the record after creation or using the flow
80+
* described above.
81+
*
82+
* TODO: These are limitations we want to (and can) address. If you
83+
* have need of lifting these limitations, please open an issue.
84+
*
85+
* @typedoc
86+
*/
87+
export type CreateRecordProperties<T extends MaybeHasId = MaybeHasId & Record<string, unknown>> =
88+
T extends TypedRecordInstance ? Partial<Omit<T, typeof ResourceType>> : MaybeHasId & Record<string, unknown>;
7189

7290
/**
7391
* A Store coordinates interaction between your application, a [Cache](https://api.emberjs.com/ember-data/release/classes/%3CInterface%3E%20Cache),
@@ -672,7 +690,9 @@ class Store extends EmberObject {
672690
newly created record.
673691
@return {Model} record
674692
*/
675-
createRecord(modelName: string, inputProperties: CreateRecordProperties): RecordInstance {
693+
createRecord<T extends MaybeHasId>(modelName: TypeFromInstance<T>, inputProperties: CreateRecordProperties<T>): T;
694+
createRecord(modelName: string, inputProperties: CreateRecordProperties): unknown;
695+
createRecord(modelName: string, inputProperties: CreateRecordProperties): unknown {
676696
if (DEBUG) {
677697
assertDestroyingStore(this, 'createRecord');
678698
}
@@ -696,21 +716,22 @@ class Store extends EmberObject {
696716
// give the adapter an opportunity to generate one. Typically,
697717
// client-side ID generators will use something like uuid.js
698718
// to avoid conflicts.
719+
let id: string | null = null;
699720

700721
if (properties.id === null || properties.id === undefined) {
701722
upgradeStore(this);
702723
const adapter = this.adapterFor?.(modelName, true);
703724

704725
if (adapter && adapter.generateIdForRecord) {
705-
properties.id = adapter.generateIdForRecord(this, modelName, properties);
726+
id = properties.id = coerceId(adapter.generateIdForRecord(this, modelName, properties));
706727
} else {
707-
properties.id = null;
728+
id = properties.id = null;
708729
}
730+
} else {
731+
id = properties.id = coerceId(properties.id);
709732
}
710733

711-
// Coerce ID to a string
712-
properties.id = coerceId(properties.id);
713-
const resource = { type: normalizedModelName, id: properties.id };
734+
const resource = { type: normalizedModelName, id };
714735

715736
if (resource.id) {
716737
const identifier = this.identifierCache.peekRecordIdentifier(resource as ResourceIdentifierObject);
@@ -749,7 +770,7 @@ class Store extends EmberObject {
749770
@public
750771
@param {Model} record
751772
*/
752-
deleteRecord(record: RecordInstance): void {
773+
deleteRecord<T>(record: T): void {
753774
if (DEBUG) {
754775
assertDestroyingStore(this, 'deleteRecord');
755776
}
@@ -782,7 +803,7 @@ class Store extends EmberObject {
782803
@public
783804
@param {Model} record
784805
*/
785-
unloadRecord(record: RecordInstance): void {
806+
unloadRecord<T>(record: T): void {
786807
if (DEBUG) {
787808
assertDestroyingStore(this, 'unloadRecord');
788809
}
@@ -1163,13 +1184,15 @@ class Store extends EmberObject {
11631184
@param {Object} [options] - if the first param is a string this will be the optional options for the request. See examples for available options.
11641185
@return {Promise} promise
11651186
*/
1166-
findRecord(resource: string, id: string | number, options?: FindOptions): Promise<RecordInstance>;
1167-
findRecord(resource: ResourceIdentifierObject, id?: FindOptions): Promise<RecordInstance>;
1187+
findRecord<T>(resource: TypeFromInstance<T>, id: string | number, options?: FindOptions): Promise<T>;
1188+
findRecord(resource: string, id: string | number, options?: FindOptions): Promise<unknown>;
1189+
findRecord<T>(resource: ResourceIdentifierObject<TypeFromInstance<T>>, id?: FindOptions): Promise<T>;
1190+
findRecord(resource: ResourceIdentifierObject, id?: FindOptions): Promise<unknown>;
11681191
findRecord(
11691192
resource: string | ResourceIdentifierObject,
11701193
id?: string | number | FindOptions,
11711194
options?: FindOptions
1172-
): Promise<RecordInstance> {
1195+
): Promise<unknown> {
11731196
if (DEBUG) {
11741197
assertDestroyingStore(this, 'findRecord');
11751198
}

packages/store/src/-private/utils/coerce-id.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ import { DEPRECATE_NON_STRICT_ID } from '@ember-data/deprecations';
1414
// corresponding record, we will not know if it is a string or a number.
1515
type Coercable = string | number | boolean | null | undefined | symbol;
1616

17-
function coerceId(id: Coercable): string | null {
17+
function coerceId(id: unknown): string | null {
1818
if (DEPRECATE_NON_STRICT_ID) {
1919
let normalized: string | null;
2020
if (id === null || id === undefined || id === '') {

0 commit comments

Comments
 (0)