@@ -15,6 +15,7 @@ import type {
15
15
StableExistingRecordIdentifier ,
16
16
StableRecordIdentifier ,
17
17
} from '@warp-drive/core-types/identifier' ;
18
+ import type { TypedRecordInstance , TypeFromInstance } from '@warp-drive/core-types/record' ;
18
19
import { EnableHydration , SkipCache } from '@warp-drive/core-types/request' ;
19
20
import type { ResourceDocument } from '@warp-drive/core-types/spec/document' ;
20
21
import type {
@@ -24,6 +25,7 @@ import type {
24
25
ResourceIdentifierObject ,
25
26
SingleResourceDocument ,
26
27
} from '@warp-drive/core-types/spec/raw' ;
28
+ import type { ResourceType } from '@warp-drive/core-types/symbols' ;
27
29
28
30
import type { Cache , CacheV1 } from '../-types/q/cache' ;
29
31
import type { CacheCapabilitiesManager } from '../-types/q/cache-store-wrapper' ;
@@ -64,10 +66,26 @@ type CompatStore = Store & {
64
66
} ;
65
67
function upgradeStore ( store : Store ) : asserts store is CompatStore { }
66
68
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 > ;
71
89
72
90
/**
73
91
* 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 {
672
690
newly created record.
673
691
@return {Model } record
674
692
*/
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 {
676
696
if ( DEBUG ) {
677
697
assertDestroyingStore ( this , 'createRecord' ) ;
678
698
}
@@ -696,21 +716,22 @@ class Store extends EmberObject {
696
716
// give the adapter an opportunity to generate one. Typically,
697
717
// client-side ID generators will use something like uuid.js
698
718
// to avoid conflicts.
719
+ let id : string | null = null ;
699
720
700
721
if ( properties . id === null || properties . id === undefined ) {
701
722
upgradeStore ( this ) ;
702
723
const adapter = this . adapterFor ?.( modelName , true ) ;
703
724
704
725
if ( adapter && adapter . generateIdForRecord ) {
705
- properties . id = adapter . generateIdForRecord ( this , modelName , properties ) ;
726
+ id = properties . id = coerceId ( adapter . generateIdForRecord ( this , modelName , properties ) ) ;
706
727
} else {
707
- properties . id = null ;
728
+ id = properties . id = null ;
708
729
}
730
+ } else {
731
+ id = properties . id = coerceId ( properties . id ) ;
709
732
}
710
733
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 } ;
714
735
715
736
if ( resource . id ) {
716
737
const identifier = this . identifierCache . peekRecordIdentifier ( resource as ResourceIdentifierObject ) ;
@@ -749,7 +770,7 @@ class Store extends EmberObject {
749
770
@public
750
771
@param {Model } record
751
772
*/
752
- deleteRecord ( record : RecordInstance ) : void {
773
+ deleteRecord < T > ( record : T ) : void {
753
774
if ( DEBUG ) {
754
775
assertDestroyingStore ( this , 'deleteRecord' ) ;
755
776
}
@@ -782,7 +803,7 @@ class Store extends EmberObject {
782
803
@public
783
804
@param {Model } record
784
805
*/
785
- unloadRecord ( record : RecordInstance ) : void {
806
+ unloadRecord < T > ( record : T ) : void {
786
807
if ( DEBUG ) {
787
808
assertDestroyingStore ( this , 'unloadRecord' ) ;
788
809
}
@@ -1163,13 +1184,15 @@ class Store extends EmberObject {
1163
1184
@param {Object } [options] - if the first param is a string this will be the optional options for the request. See examples for available options.
1164
1185
@return {Promise } promise
1165
1186
*/
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 > ;
1168
1191
findRecord (
1169
1192
resource : string | ResourceIdentifierObject ,
1170
1193
id ?: string | number | FindOptions ,
1171
1194
options ?: FindOptions
1172
- ) : Promise < RecordInstance > {
1195
+ ) : Promise < unknown > {
1173
1196
if ( DEBUG ) {
1174
1197
assertDestroyingStore ( this , 'findRecord' ) ;
1175
1198
}
0 commit comments