@warp-drive/legacy 5.8.0-beta.0 → 5.8.0-beta.2

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 (214) hide show
  1. package/README.md +16 -26
  2. package/declarations/adapter/error.d.ts +7 -7
  3. package/declarations/adapter/json-api.d.ts +6 -8
  4. package/declarations/adapter/rest.d.ts +26 -112
  5. package/declarations/adapter.d.ts +6 -8
  6. package/declarations/compat/-private.d.ts +1 -1
  7. package/declarations/compat/builders/find-all.d.ts +6 -6
  8. package/declarations/compat/builders/find-record.d.ts +8 -8
  9. package/declarations/compat/builders/query.d.ts +12 -12
  10. package/declarations/compat/extensions.d.ts +1 -1
  11. package/declarations/compat/legacy-network-handler/minimum-adapter-interface.d.ts +7 -9
  12. package/declarations/compat/legacy-network-handler/minimum-serializer-interface.d.ts +20 -30
  13. package/declarations/compat/utils.d.ts +17 -17
  14. package/declarations/compat.d.ts +35 -11
  15. package/declarations/index.d.ts +102 -0
  16. package/declarations/model/-private/attr.d.ts +5 -6
  17. package/declarations/model/-private/belongs-to.d.ts +4 -5
  18. package/declarations/model/-private/has-many.d.ts +4 -5
  19. package/declarations/model/-private/hooks.d.ts +1 -1
  20. package/declarations/model/-private/legacy-relationships-support.d.ts +2 -2
  21. package/declarations/model/-private/model.d.ts +18 -59
  22. package/declarations/model/-private/promise-many-array.d.ts +0 -18
  23. package/declarations/model/-private/record-state.d.ts +1 -1
  24. package/declarations/model/-private/references/belongs-to.d.ts +19 -29
  25. package/declarations/model/-private/references/has-many.d.ts +14 -16
  26. package/declarations/model/migration-support.d.ts +46 -21
  27. package/declarations/model-fragments/extensions/fragment-array.d.ts +16 -0
  28. package/declarations/model-fragments/extensions/fragment.d.ts +15 -0
  29. package/declarations/model-fragments/hooks/model-for.d.ts +20 -0
  30. package/declarations/model-fragments/index.d.ts +5 -0
  31. package/declarations/model-fragments/instance-initializers/fragment-extensions.d.ts +9 -0
  32. package/declarations/model-fragments/utilities/with-array-defaults.d.ts +15 -0
  33. package/declarations/model-fragments/utilities/with-fragment-array-defaults.d.ts +20 -0
  34. package/declarations/model-fragments/utilities/with-fragment-defaults.d.ts +19 -0
  35. package/declarations/model-fragments/utilities/with-legacy.d.ts +3 -0
  36. package/declarations/model-fragments.d.ts +9 -0
  37. package/declarations/model.d.ts +2 -2
  38. package/declarations/serializer/-private/embedded-records-mixin.d.ts +1 -6
  39. package/declarations/serializer/-private/transforms/boolean.d.ts +2 -2
  40. package/declarations/serializer/-private/transforms/date.d.ts +2 -2
  41. package/declarations/serializer/-private/transforms/number.d.ts +1 -1
  42. package/declarations/serializer/-private/transforms/string.d.ts +1 -1
  43. package/declarations/serializer/json-api.d.ts +11 -12
  44. package/declarations/serializer/json.d.ts +9 -11
  45. package/declarations/serializer/rest.d.ts +4 -6
  46. package/declarations/serializer.d.ts +9 -12
  47. package/dist/{-private-8UmnAf9J.js → -private-BG3bMiKp.js} +3 -2
  48. package/dist/adapter/-private.js +1 -1
  49. package/dist/adapter/error.js +14 -15
  50. package/dist/adapter/json-api.js +4 -1
  51. package/dist/adapter/rest.js +38 -125
  52. package/dist/adapter.js +6 -8
  53. package/dist/compat/-private.js +1 -1
  54. package/dist/compat/builders.js +26 -26
  55. package/dist/compat/utils.js +17 -18
  56. package/dist/compat.js +61 -43
  57. package/dist/{errors-8kD2mSe_.js → errors-Cz5KrzBk.js} +115 -122
  58. package/dist/hooks-D6diaM34.js +74 -0
  59. package/dist/index.js +195 -0
  60. package/dist/{json-Et4mt_LM.js → json-ChdEfB0X.js} +18 -35
  61. package/dist/model/-private.js +1 -1
  62. package/dist/model/migration-support.js +59 -27
  63. package/dist/model-for-CqXsIKws.js +221 -0
  64. package/dist/model-fragments.js +76 -0
  65. package/dist/model.js +18 -90
  66. package/dist/{schema-provider-DQu4Rjco.js → schema-provider-DJCV_6AF.js} +50 -95
  67. package/dist/{serialize-into-hash-CS0MIv4F.js → serialize-into-hash-DPZYoF-i.js} +1 -1
  68. package/dist/serializer/json-api.js +18 -45
  69. package/dist/serializer/json.js +1 -1
  70. package/dist/serializer/rest.js +14 -21
  71. package/dist/serializer/transform.js +15 -6
  72. package/dist/serializer.js +9 -13
  73. package/dist/store.js +5 -1
  74. package/dist/unpkg/dev/-private-DtjBbEgy.js +1206 -0
  75. package/dist/unpkg/dev/adapter/-private.js +1 -0
  76. package/dist/unpkg/dev/adapter/error.js +335 -0
  77. package/dist/unpkg/dev/adapter/json-api.js +271 -0
  78. package/dist/unpkg/dev/adapter/rest.js +1171 -0
  79. package/dist/unpkg/dev/adapter.js +1252 -0
  80. package/dist/unpkg/dev/compat/-private.js +1 -0
  81. package/dist/unpkg/dev/compat/builders.js +275 -0
  82. package/dist/unpkg/dev/compat/extensions.js +242 -0
  83. package/dist/unpkg/dev/compat/utils.js +223 -0
  84. package/dist/unpkg/dev/compat.js +1147 -0
  85. package/dist/unpkg/dev/errors-DmGGJr3T.js +2562 -0
  86. package/dist/unpkg/dev/hooks-CkYiE6Ud.js +73 -0
  87. package/dist/unpkg/dev/index.js +197 -0
  88. package/dist/unpkg/dev/json-Cu1LNgmQ.js +1256 -0
  89. package/dist/unpkg/dev/model/-private.js +1 -0
  90. package/dist/unpkg/dev/model/migration-support.js +553 -0
  91. package/dist/unpkg/dev/model-for-CqXsIKws.js +221 -0
  92. package/dist/unpkg/dev/model-fragments.js +76 -0
  93. package/dist/unpkg/dev/model.js +678 -0
  94. package/dist/unpkg/dev/runtime-BPCpkOf1-BKOwiRJp.js +65 -0
  95. package/dist/unpkg/dev/schema-provider-DDVYxmUV.js +2186 -0
  96. package/dist/unpkg/dev/serialize-into-hash-B2xDbuo5.js +259 -0
  97. package/dist/unpkg/dev/serializer/json-api.js +649 -0
  98. package/dist/unpkg/dev/serializer/json.js +4 -0
  99. package/dist/unpkg/dev/serializer/rest.js +1242 -0
  100. package/dist/unpkg/dev/serializer/transform.js +278 -0
  101. package/dist/unpkg/dev/serializer.js +248 -0
  102. package/dist/unpkg/dev/store.js +637 -0
  103. package/dist/unpkg/dev/util-DvanW33H.js +20 -0
  104. package/dist/unpkg/dev/utils-BhvS1iTS.js +8 -0
  105. package/dist/unpkg/dev-deprecated/-private-DtjBbEgy.js +1206 -0
  106. package/dist/unpkg/dev-deprecated/adapter/-private.js +1 -0
  107. package/dist/unpkg/dev-deprecated/adapter/error.js +335 -0
  108. package/dist/unpkg/dev-deprecated/adapter/json-api.js +271 -0
  109. package/dist/unpkg/dev-deprecated/adapter/rest.js +1171 -0
  110. package/dist/unpkg/dev-deprecated/adapter.js +1252 -0
  111. package/dist/unpkg/dev-deprecated/compat/-private.js +1 -0
  112. package/dist/unpkg/dev-deprecated/compat/builders.js +275 -0
  113. package/dist/unpkg/dev-deprecated/compat/extensions.js +242 -0
  114. package/dist/unpkg/dev-deprecated/compat/utils.js +223 -0
  115. package/dist/unpkg/dev-deprecated/compat.js +1147 -0
  116. package/dist/unpkg/dev-deprecated/errors-Spt6ubMd.js +2565 -0
  117. package/dist/unpkg/dev-deprecated/hooks-DOXegvhL.js +73 -0
  118. package/dist/unpkg/dev-deprecated/index.js +196 -0
  119. package/dist/unpkg/dev-deprecated/json-Cu1LNgmQ.js +1256 -0
  120. package/dist/unpkg/dev-deprecated/model/-private.js +1 -0
  121. package/dist/unpkg/dev-deprecated/model/migration-support.js +570 -0
  122. package/dist/unpkg/dev-deprecated/model-for-CqXsIKws.js +221 -0
  123. package/dist/unpkg/dev-deprecated/model-fragments.js +76 -0
  124. package/dist/unpkg/dev-deprecated/model.js +682 -0
  125. package/dist/unpkg/dev-deprecated/runtime-BPCpkOf1-BKOwiRJp.js +65 -0
  126. package/dist/unpkg/dev-deprecated/schema-provider-BP6_8N-V.js +2211 -0
  127. package/dist/unpkg/dev-deprecated/serialize-into-hash-B2xDbuo5.js +259 -0
  128. package/dist/unpkg/dev-deprecated/serializer/json-api.js +649 -0
  129. package/dist/unpkg/dev-deprecated/serializer/json.js +4 -0
  130. package/dist/unpkg/dev-deprecated/serializer/rest.js +1242 -0
  131. package/dist/unpkg/dev-deprecated/serializer/transform.js +278 -0
  132. package/dist/unpkg/dev-deprecated/serializer.js +248 -0
  133. package/dist/unpkg/dev-deprecated/store.js +637 -0
  134. package/dist/unpkg/dev-deprecated/util-CWr5WQOT.js +24 -0
  135. package/dist/unpkg/dev-deprecated/utils-C9PJehtL.js +12 -0
  136. package/dist/unpkg/prod/-private-BdyZaGEh.js +971 -0
  137. package/dist/unpkg/prod/adapter/-private.js +1 -0
  138. package/dist/unpkg/prod/adapter/error.js +330 -0
  139. package/dist/unpkg/prod/adapter/json-api.js +266 -0
  140. package/dist/unpkg/prod/adapter/rest.js +1134 -0
  141. package/dist/unpkg/prod/adapter.js +1219 -0
  142. package/dist/unpkg/prod/compat/-private.js +1 -0
  143. package/dist/unpkg/prod/compat/builders.js +210 -0
  144. package/dist/unpkg/prod/compat/extensions.js +232 -0
  145. package/dist/unpkg/prod/compat/utils.js +218 -0
  146. package/dist/unpkg/prod/compat.js +727 -0
  147. package/dist/unpkg/prod/errors-BGVFCBmi.js +2314 -0
  148. package/dist/unpkg/prod/hooks-BztVA_x0.js +41 -0
  149. package/dist/unpkg/prod/index.js +151 -0
  150. package/dist/unpkg/prod/json-BWrZ5546.js +1243 -0
  151. package/dist/unpkg/prod/model/-private.js +1 -0
  152. package/dist/unpkg/prod/model/migration-support.js +546 -0
  153. package/dist/unpkg/prod/model-for-CqXsIKws.js +221 -0
  154. package/dist/unpkg/prod/model-fragments.js +76 -0
  155. package/dist/unpkg/prod/model.js +593 -0
  156. package/dist/unpkg/prod/runtime-BPCpkOf1-BKOwiRJp.js +65 -0
  157. package/dist/unpkg/prod/schema-provider-DJtD_8jZ.js +1861 -0
  158. package/dist/unpkg/prod/serialize-into-hash-DGlzQteF.js +215 -0
  159. package/dist/unpkg/prod/serializer/json-api.js +592 -0
  160. package/dist/unpkg/prod/serializer/json.js +4 -0
  161. package/dist/unpkg/prod/serializer/rest.js +1210 -0
  162. package/dist/unpkg/prod/serializer/transform.js +278 -0
  163. package/dist/unpkg/prod/serializer.js +248 -0
  164. package/dist/unpkg/prod/store.js +505 -0
  165. package/dist/unpkg/prod/util-DvanW33H.js +20 -0
  166. package/dist/unpkg/prod/utils-BhvS1iTS.js +8 -0
  167. package/dist/unpkg/prod-deprecated/-private-BdyZaGEh.js +971 -0
  168. package/dist/unpkg/prod-deprecated/adapter/-private.js +1 -0
  169. package/dist/unpkg/prod-deprecated/adapter/error.js +330 -0
  170. package/dist/unpkg/prod-deprecated/adapter/json-api.js +266 -0
  171. package/dist/unpkg/prod-deprecated/adapter/rest.js +1134 -0
  172. package/dist/unpkg/prod-deprecated/adapter.js +1219 -0
  173. package/dist/unpkg/prod-deprecated/compat/-private.js +1 -0
  174. package/dist/unpkg/prod-deprecated/compat/builders.js +210 -0
  175. package/dist/unpkg/prod-deprecated/compat/extensions.js +232 -0
  176. package/dist/unpkg/prod-deprecated/compat/utils.js +218 -0
  177. package/dist/unpkg/prod-deprecated/compat.js +727 -0
  178. package/dist/unpkg/prod-deprecated/errors-CdDaK81x.js +2317 -0
  179. package/dist/unpkg/prod-deprecated/hooks-yId87yyG.js +41 -0
  180. package/dist/unpkg/prod-deprecated/index.js +150 -0
  181. package/dist/unpkg/prod-deprecated/json-BWrZ5546.js +1243 -0
  182. package/dist/unpkg/prod-deprecated/model/-private.js +1 -0
  183. package/dist/unpkg/prod-deprecated/model/migration-support.js +563 -0
  184. package/dist/unpkg/prod-deprecated/model-for-CqXsIKws.js +221 -0
  185. package/dist/unpkg/prod-deprecated/model-fragments.js +76 -0
  186. package/dist/unpkg/prod-deprecated/model.js +596 -0
  187. package/dist/unpkg/prod-deprecated/runtime-BPCpkOf1-BKOwiRJp.js +65 -0
  188. package/dist/unpkg/prod-deprecated/schema-provider-CjX55uSY.js +1904 -0
  189. package/dist/unpkg/prod-deprecated/serialize-into-hash-DGlzQteF.js +215 -0
  190. package/dist/unpkg/prod-deprecated/serializer/json-api.js +592 -0
  191. package/dist/unpkg/prod-deprecated/serializer/json.js +4 -0
  192. package/dist/unpkg/prod-deprecated/serializer/rest.js +1210 -0
  193. package/dist/unpkg/prod-deprecated/serializer/transform.js +278 -0
  194. package/dist/unpkg/prod-deprecated/serializer.js +248 -0
  195. package/dist/unpkg/prod-deprecated/store.js +505 -0
  196. package/dist/unpkg/prod-deprecated/util-B6cn-i93.js +23 -0
  197. package/dist/unpkg/prod-deprecated/utils-BUWwQwCh.js +11 -0
  198. package/logos/README.md +2 -2
  199. package/logos/logo-yellow-slab.svg +1 -0
  200. package/logos/word-mark-black.svg +1 -0
  201. package/logos/word-mark-white.svg +1 -0
  202. package/package.json +15 -7
  203. package/logos/NCC-1701-a-blue.svg +0 -4
  204. package/logos/NCC-1701-a-gold.svg +0 -4
  205. package/logos/NCC-1701-a-gold_100.svg +0 -1
  206. package/logos/NCC-1701-a-gold_base-64.txt +0 -1
  207. package/logos/NCC-1701-a.svg +0 -4
  208. package/logos/docs-badge.svg +0 -2
  209. package/logos/ember-data-logo-dark.svg +0 -12
  210. package/logos/ember-data-logo-light.svg +0 -12
  211. package/logos/social1.png +0 -0
  212. package/logos/social2.png +0 -0
  213. package/logos/warp-drive-logo-dark.svg +0 -4
  214. package/logos/warp-drive-logo-gold.svg +0 -4
@@ -0,0 +1,678 @@
1
+ import { computed } from '@ember/object';
2
+ import { recordIdentifierFor } from '@warp-drive/core';
3
+ import { RecordStore } from '@warp-drive/core/types/symbols';
4
+ import { i as isElementDescriptor, n as normalizeModelName } from "./util-DvanW33H.js";
5
+ import { assertPrivateStore } from '@warp-drive/core/store/-private';
6
+ import { l as lookupLegacySupport } from "./errors-DmGGJr3T.js";
7
+ import '@warp-drive/utilities/string';
8
+ export { M as Model, b as buildSchema, M as default, m as restoreDeprecatedModelRequestBehaviors } from "./schema-provider-DDVYxmUV.js";
9
+ export { i as instantiateRecord, m as modelFor, t as teardownRecord } from "./hooks-CkYiE6Ud.js";
10
+
11
+ /**
12
+ * Options provided to the attr decorator are
13
+ * supplied to the associated transform. Any
14
+ * key-value pair is valid; however, it is highly
15
+ * recommended to only use statically defined values
16
+ * that could be serialized to JSON.
17
+ *
18
+ * If no transform is provided, the only valid
19
+ * option is `defaultValue`.
20
+ *
21
+ * Examples:
22
+ *
23
+ * ```ts
24
+ * class User extends Model {
25
+ * @attr('string', { defaultValue: 'Anonymous' }) name;
26
+ * @attr('date', { defaultValue: () => new Date() }) createdAt;
27
+ * @attr({ defaultValue: () => ({}) }) preferences;
28
+ * @attr('boolean') hasVerifiedEmail;
29
+ * @attr address;
30
+ * }
31
+ *
32
+ * @class NOTATHING
33
+ */
34
+
35
+ function _attr(type, options) {
36
+ if (typeof type === 'object') {
37
+ options = type;
38
+ type = undefined;
39
+ } else {
40
+ options = options || {};
41
+ }
42
+ const meta = {
43
+ type: type,
44
+ kind: 'attribute',
45
+ isAttribute: true,
46
+ options: options,
47
+ key: null
48
+ };
49
+ return computed({
50
+ get(key) {
51
+ {
52
+ if (['currentState'].includes(key)) {
53
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your attr on ${this.constructor.toString()}`);
54
+ }
55
+ }
56
+ if (this.isDestroyed || this.isDestroying) {
57
+ return;
58
+ }
59
+ const cache = this[RecordStore].cache;
60
+ return cache.getAttr(recordIdentifierFor(this), key);
61
+ },
62
+ set(key, value) {
63
+ {
64
+ if (['currentState'].includes(key)) {
65
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your attr on ${this.constructor.toString()}`);
66
+ }
67
+ }
68
+ const identifier = recordIdentifierFor(this);
69
+ (test => {
70
+ if (!test) {
71
+ throw new Error(`Attempted to set '${key}' on the deleted record ${identifier.type}:${identifier.id} (${identifier.lid})`);
72
+ }
73
+ })(!this.currentState.isDeleted);
74
+ const cache = this[RecordStore].cache;
75
+ const currentValue = cache.getAttr(identifier, key);
76
+ if (currentValue !== value) {
77
+ cache.setAttr(identifier, key, value);
78
+ if (!this.isValid) {
79
+ const {
80
+ errors
81
+ } = this;
82
+ if (errors.get(key)) {
83
+ errors.remove(key);
84
+ this.currentState.cleanErrorRequests();
85
+ }
86
+ }
87
+ }
88
+ return value;
89
+ }
90
+ }).meta(meta);
91
+ }
92
+
93
+ // NOTE: Usage of Explicit ANY
94
+ // -------------------------------------------------------------------
95
+ // any is required here because we are the maximal not the minimal
96
+ // subset of options allowed. If we used unknown, object, or
97
+ // Record<string, unknown> we would get type errors when we try to
98
+ // assert against a more specific implementation with precise options.
99
+ // -------------------------------------------------------------------
100
+
101
+ // see note on Explicit ANY above
102
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
103
+
104
+ /**
105
+ * The return type of `void` is a lie to appease TypeScript. The actual return type
106
+ * is a descriptor, but typescript incorrectly insists that decorator functions return
107
+ * `void` or `any`.
108
+ *
109
+ */
110
+
111
+ /**
112
+ `attr` defines an attribute on a {@link Model}.
113
+ By default, attributes are passed through as-is, however you can specify an
114
+ optional type to have the value automatically transformed.
115
+ WarpDrive ships with four basic transform types: `string`, `number`,
116
+ `boolean` and `date`. You can define your own transforms by subclassing
117
+ {@link Transform}.
118
+
119
+ Note that you cannot use `attr` to define an attribute of `id`.
120
+
121
+ `attr` takes an optional hash as a second parameter, currently
122
+ supported options are:
123
+
124
+ - `defaultValue`: Pass a string or a function to be called to set the attribute
125
+ to a default value if and only if the key is absent from the payload response.
126
+
127
+ Example
128
+
129
+ ```js [app/models/user.js]
130
+ import { Model, attr } from '@warp-drive/legacy/model';
131
+
132
+ export default class UserModel extends Model {
133
+ @attr('string') username;
134
+ @attr('string') email;
135
+ @attr('boolean', { defaultValue: false }) verified;
136
+ }
137
+ ```
138
+
139
+ Default value can also be a function. This is useful it you want to return
140
+ a new object for each attribute.
141
+
142
+ ```js [app/models/user.js]
143
+ import { Model, attr } from '@warp-drive/legacy/model';
144
+
145
+ export default class UserModel extends Model {
146
+ @attr('string') username;
147
+ @attr('string') email;
148
+
149
+ @attr({
150
+ defaultValue() {
151
+ return {};
152
+ }
153
+ })
154
+ settings;
155
+ }
156
+ ```
157
+
158
+ The `options` hash is passed as second argument to a transforms'
159
+ `serialize` and `deserialize` method. This allows to configure a
160
+ transformation and adapt the corresponding value, based on the config:
161
+
162
+ ```js [app/models/post.js]
163
+ import { Model, attr } from '@warp-drive/legacy/model';
164
+
165
+ export default class PostModel extends Model {
166
+ @attr('text', {
167
+ uppercase: true
168
+ })
169
+ text;
170
+ }
171
+ ```
172
+
173
+ ```js [app/transforms/text.js]
174
+ export default class TextTransform {
175
+ serialize(value, options) {
176
+ if (options.uppercase) {
177
+ return value.toUpperCase();
178
+ }
179
+
180
+ return value;
181
+ }
182
+
183
+ deserialize(value) {
184
+ return value;
185
+ }
186
+
187
+ static create() {
188
+ return new this();
189
+ }
190
+ }
191
+ ```
192
+
193
+ @public
194
+ @param type the attribute type
195
+ @param options a hash of options
196
+ */
197
+
198
+ // see note on DataDecorator for why void
199
+ function attr(type, options, desc) {
200
+ const args = [type, options, desc];
201
+ // see note on DataDecorator for why void
202
+ return isElementDescriptor(args) ? _attr()(...args) : _attr(type, options);
203
+ }
204
+
205
+ // type BelongsToDecoratorObject<getT> = {
206
+ // get: () => getT;
207
+ // // set: (value: Awaited<getT>) => void;
208
+ // set: (value: getT) => void;
209
+ // // init: () => getT;
210
+ // };
211
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
212
+ // BelongsToDecoratorObject<getT>;
213
+
214
+ function _belongsTo(type, options) {
215
+ (test => {
216
+ if (!test) {
217
+ throw new Error(`Expected options.async from @belongsTo('${type}', options) to be a boolean`);
218
+ }
219
+ })(options && typeof options.async === 'boolean');
220
+ (test => {
221
+ if (!test) {
222
+ throw new Error(`Expected options.inverse from @belongsTo('${type}', options) to be either null or the string type of the related resource.`);
223
+ }
224
+ })(options.inverse === null || typeof options.inverse === 'string' && options.inverse.length > 0);
225
+ const meta = {
226
+ type: normalizeModelName(type),
227
+ options: options,
228
+ kind: 'belongsTo',
229
+ name: '<Unknown BelongsTo>'
230
+ };
231
+ return computed({
232
+ get(key) {
233
+ // this is a legacy behavior we may not carry into a new model setup
234
+ // it's better to error on disconnected records so users find errors
235
+ // in their logic.
236
+ if (this.isDestroying || this.isDestroyed) {
237
+ return null;
238
+ }
239
+ const support = lookupLegacySupport(this);
240
+ {
241
+ if (['currentState'].includes(key)) {
242
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your belongsTo on ${this.constructor.toString()}`);
243
+ }
244
+ if (Object.prototype.hasOwnProperty.call(options, 'serialize')) {
245
+ console.warn(`You provided a serialize option on the "${key}" property in the "${support.identifier.type}" class, this belongs in the serializer. See Serializer and it's implementations https://api.emberjs.com/ember-data/release/classes/Serializer`, false, {
246
+ id: 'ds.model.serialize-option-in-belongs-to'
247
+ });
248
+ }
249
+ if (Object.prototype.hasOwnProperty.call(options, 'embedded')) {
250
+ console.warn(`You provided an embedded option on the "${key}" property in the "${support.identifier.type}" class, this belongs in the serializer. See EmbeddedRecordsMixin https://api.emberjs.com/ember-data/release/classes/EmbeddedRecordsMixin`, false, {
251
+ id: 'ds.model.embedded-option-in-belongs-to'
252
+ });
253
+ }
254
+ }
255
+ return support.getBelongsTo(key);
256
+ },
257
+ set(key, value) {
258
+ (test => {
259
+ if (!test) {
260
+ throw new Error(`Cannot set belongsTo relationship ${key} on a destroyed record`);
261
+ }
262
+ })(!this.isDestroying && !this.isDestroyed);
263
+ if (this.isDestroying || this.isDestroyed) {
264
+ return null;
265
+ }
266
+ const support = lookupLegacySupport(this);
267
+ {
268
+ if (['currentState'].includes(key)) {
269
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your belongsTo on ${this.constructor.toString()}`);
270
+ }
271
+ }
272
+ assertPrivateStore(this[RecordStore]);
273
+ this[RecordStore]._join(() => {
274
+ support.setDirtyBelongsTo(key, value);
275
+ });
276
+ return support.getBelongsTo(key);
277
+ }
278
+ }).meta(meta);
279
+ }
280
+
281
+ /**
282
+ `belongsTo` is used to define One-To-One and One-To-Many, and One-To-None
283
+ relationships on a {@link Model}.
284
+
285
+ `belongsTo` takes a configuration hash as a second parameter, currently
286
+ supported options are:
287
+
288
+ - `async`: (*required*) A boolean value used to declare whether this is a sync (false) or async (true) relationship.
289
+ - `inverse`: (*required*) A string used to identify the inverse property on a related model, or `null`.
290
+ - `polymorphic`: (*optional*) A boolean value to mark the relationship as polymorphic
291
+ - `as`: (*optional*) A string used to declare the abstract type "this" record satisfies for polymorphism.
292
+
293
+ ### Examples
294
+
295
+ To declare a **one-to-many** (or many-to-many) relationship, use
296
+ `belongsTo` in combination with `hasMany`:
297
+
298
+ ```js
299
+ // app/models/comment.js
300
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
301
+
302
+ export default class Comment extends Model {
303
+ @belongsTo('post', { async: false, inverse: 'comments' }) post;
304
+ }
305
+
306
+ // app/models/post.js
307
+ import { Model, hasMany } from '@warp-drive/legacy/model';
308
+
309
+ export default class Post extends Model {
310
+ @hasMany('comment', { async: false, inverse: 'post' }) comments;
311
+ }
312
+ ```
313
+
314
+ To declare a **one-to-one** relationship with managed inverses, use `belongsTo` for both sides:
315
+
316
+ ```js
317
+ // app/models/author.js
318
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
319
+
320
+ export default class Author extends Model {
321
+ @belongsTo('address', { async: true, inverse: 'owner' }) address;
322
+ }
323
+
324
+ // app/models/address.js
325
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
326
+
327
+ export default class Address extends Model {
328
+ @belongsTo('author', { async: true, inverse: 'address' }) owner;
329
+ }
330
+ ```
331
+
332
+ To declare a **one-to-one** relationship without managed inverses, use `belongsTo` for both sides
333
+ with `null` as the inverse:
334
+
335
+ ```js
336
+ // app/models/author.js
337
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
338
+
339
+ export default class Author extends Model {
340
+ @belongsTo('address', { async: true, inverse: null }) address;
341
+ }
342
+
343
+ // app/models/address.js
344
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
345
+
346
+ export default class Address extends Model {
347
+ @belongsTo('author', { async: true, inverse: null }) owner;
348
+ }
349
+ ```
350
+
351
+ To declare a one-to-none relationship between two models, use
352
+ `belongsTo` with inverse set to `null` on just one side::
353
+
354
+ ```js
355
+ // app/models/person.js
356
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
357
+
358
+ export default class Person extends Model {
359
+ @belongsTo('person', { async: false, inverse: null }) bestFriend;
360
+ }
361
+ ```
362
+
363
+ #### Sync vs Async Relationships
364
+
365
+ WarpDrive fulfills relationships using resource data available in
366
+ the cache.
367
+
368
+ Sync relationships point directly to the known related resources.
369
+
370
+ When a relationship is declared as async, if any of the known related
371
+ resources have not been loaded, they will be fetched. The property
372
+ on the record when accessed provides a promise that resolves once
373
+ all resources are loaded.
374
+
375
+ Async relationships may take advantage of links. On access, if the related
376
+ link has not been loaded, or if any known resources are not available in
377
+ the cache, the fresh state will be fetched using the link.
378
+
379
+ In contrast to async relationship, accessing a sync relationship
380
+ will error on access when any of the known related resources have
381
+ not been loaded.
382
+
383
+ If you are using `links` with sync relationships, you have to use
384
+ the BelongsTo reference API to fetch or refresh related resources
385
+ that aren't loaded. For instance, for a `bestFriend` relationship:
386
+
387
+ ```js
388
+ person.belongsTo('bestFriend').reload();
389
+ ```
390
+
391
+ #### Polymorphic Relationships
392
+
393
+ To declare a polymorphic relationship, use `hasMany` with the `polymorphic`
394
+ option set to `true`:
395
+
396
+ ```js
397
+ // app/models/comment.js
398
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
399
+
400
+ export default class Comment extends Model {
401
+ @belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
402
+ }
403
+ ```
404
+
405
+ `'commentable'` here is referred to as the "abstract type" for the polymorphic
406
+ relationship.
407
+
408
+ Polymorphic relationships with `inverse: null` will accept any type of record as their content.
409
+ Polymorphic relationships with `inverse` set to a string will only accept records with a matching
410
+ inverse relationships declaring itself as satisfying the abstract type.
411
+
412
+ Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable'
413
+ for this relationship.
414
+
415
+ ```js
416
+ // app/models/post.js
417
+ import { Model, hasMany } from '@warp-drive/legacy/model';
418
+
419
+ export default class Post extends Model {
420
+ @hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
421
+ }
422
+ ```
423
+
424
+ Note: every Model that declares an inverse to a polymorphic relationship must
425
+ declare itself exactly the same. This is because polymorphism is based on structural
426
+ traits.
427
+
428
+ Polymorphic to polymorphic relationships are supported. Both sides of the relationship
429
+ must be declared as polymorphic, and the `as` option must be used to declare the abstract
430
+ type each record satisfies on both sides.
431
+
432
+ @public
433
+ @param type the name of the related resource
434
+ @param options a hash of options
435
+ */
436
+
437
+ // export function belongsTo<K extends Promise<unknown>, T extends Awaited<K> = Awaited<K>>(
438
+ // type: TypeFromInstance<NoNull<T>>,
439
+ // options: RelationshipOptions<T, true>
440
+ // ): RelationshipDecorator<K>;
441
+
442
+ function belongsTo(type, options) {
443
+ {
444
+ (test => {
445
+ if (!test) {
446
+ throw new Error(`belongsTo must be invoked with a type and options. Did you mean \`@belongsTo(${type}, { async: false, inverse: null })\`?`);
447
+ }
448
+ })(!isElementDescriptor(arguments));
449
+ }
450
+ return _belongsTo(type, options);
451
+ }
452
+ function normalizeType(type) {
453
+ return type;
454
+ }
455
+ function _hasMany(type, options) {
456
+ (test => {
457
+ if (!test) {
458
+ throw new Error(`Expected hasMany options.async to be a boolean`);
459
+ }
460
+ })(options && typeof options.async === 'boolean');
461
+
462
+ // Metadata about relationships is stored on the meta of
463
+ // the relationship. This is used for introspection and
464
+ // serialization. Note that `key` is populated lazily
465
+ // the first time the CP is called.
466
+ const meta = {
467
+ type: normalizeType(type),
468
+ options,
469
+ kind: 'hasMany',
470
+ name: '<Unknown BelongsTo>'
471
+ };
472
+ return computed({
473
+ get(key) {
474
+ {
475
+ if (['currentState'].includes(key)) {
476
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your hasMany on ${this.constructor.toString()}`);
477
+ }
478
+ }
479
+ if (this.isDestroying || this.isDestroyed) {
480
+ return [];
481
+ }
482
+ return lookupLegacySupport(this).getHasMany(key);
483
+ },
484
+ set(key, records) {
485
+ {
486
+ if (['currentState'].includes(key)) {
487
+ throw new Error(`'${key}' is a reserved property name on instances of classes extending Model. Please choose a different property name for your hasMany on ${this.constructor.toString()}`);
488
+ }
489
+ }
490
+ const support = lookupLegacySupport(this);
491
+ const manyArray = support.getManyArray(key);
492
+ (test => {
493
+ if (!test) {
494
+ throw new Error(`You must pass an array of records to set a hasMany relationship`);
495
+ }
496
+ })(Array.isArray(records));
497
+ assertPrivateStore(this[RecordStore]);
498
+ this[RecordStore]._join(() => {
499
+ manyArray.splice(0, manyArray.length, ...records);
500
+ });
501
+ return support.getHasMany(key);
502
+ }
503
+ }).meta(meta);
504
+ }
505
+
506
+ /**
507
+ `hasMany` is used to define Many-To-One and Many-To-Many, and Many-To-None
508
+ relationships on a {@link Model}.
509
+
510
+ `hasMany` takes a configuration hash as a second parameter, currently
511
+ supported options are:
512
+
513
+ - `async`: (*required*) A boolean value used to declare whether this is a sync (false) or async (true) relationship.
514
+ - `inverse`: (*required*) A string used to identify the inverse property on a related model, or `null`.
515
+ - `polymorphic`: (*optional*) A boolean value to mark the relationship as polymorphic
516
+ - `as`: (*optional*) A string used to declare the abstract type "this" record satisfies for polymorphism.
517
+
518
+ ### Examples
519
+
520
+ To declare a **many-to-one** (or one-to-many) relationship, use
521
+ `belongsTo` in combination with `hasMany`:
522
+
523
+ ```js
524
+ // app/models/post.js
525
+ import { Model, hasMany } from '@warp-drive/legacy/model';
526
+
527
+ export default class Post extends Model {
528
+ @hasMany('comment', { async: false, inverse: 'post' }) comments;
529
+ }
530
+
531
+
532
+ // app/models/comment.js
533
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
534
+
535
+ export default class Comment extends Model {
536
+ @belongsTo('post', { async: false, inverse: 'comments' }) post;
537
+ }
538
+ ```
539
+
540
+ To declare a **many-to-many** relationship with managed inverses, use `hasMany` for both sides:
541
+
542
+ ```js
543
+ // app/models/post.js
544
+ import { Model, hasMany } from '@warp-drive/legacy/model';
545
+
546
+ export default class Post extends Model {
547
+ @hasMany('tag', { async: true, inverse: 'posts' }) tags;
548
+ }
549
+
550
+ // app/models/tag.js
551
+ import { Model, hasMany } from '@warp-drive/legacy/model';
552
+
553
+ export default class Tag extends Model {
554
+ @hasMany('post', { async: true, inverse: 'tags' }) posts;
555
+ }
556
+ ```
557
+
558
+ To declare a **many-to-many** relationship without managed inverses, use `hasMany` for both sides
559
+ with `null` as the inverse:
560
+
561
+ ```js
562
+ // app/models/post.js
563
+ import { Model, hasMany } from '@warp-drive/legacy/model';
564
+
565
+ export default class Post extends Model {
566
+ @hasMany('tag', { async: true, inverse: null }) tags;
567
+ }
568
+
569
+ // app/models/tag.js
570
+ import { Model, hasMany } from '@warp-drive/legacy/model';
571
+
572
+ export default class Tag extends Model {
573
+ @hasMany('post', { async: true, inverse: null }) posts;
574
+ }
575
+ ```
576
+
577
+ To declare a many-to-none relationship between two models, use
578
+ `hasMany` with inverse set to `null` on just one side::
579
+
580
+ ```js
581
+ // app/models/post.js
582
+ import { Model, hasMany } from '@warp-drive/legacy/model';
583
+
584
+ export default class Post extends Model {
585
+ @hasMany('category', { async: true, inverse: null }) categories;
586
+ }
587
+ ```
588
+
589
+ #### Sync vs Async Relationships
590
+
591
+ WarpDrive fulfills relationships using resource data available in
592
+ the cache.
593
+
594
+ Sync relationships point directly to the known related resources.
595
+
596
+ When a relationship is declared as async, if any of the known related
597
+ resources have not been loaded, they will be fetched. The property
598
+ on the record when accessed provides a promise that resolves once
599
+ all resources are loaded.
600
+
601
+ Async relationships may take advantage of links. On access, if the related
602
+ link has not been loaded, or if any known resources are not available in
603
+ the cache, the fresh state will be fetched using the link.
604
+
605
+ In contrast to async relationship, accessing a sync relationship
606
+ will error on access when any of the known related resources have
607
+ not been loaded.
608
+
609
+ If you are using `links` with sync relationships, you have to use
610
+ the HasMany reference API to fetch or refresh related resources
611
+ that aren't loaded. For instance, for a `comments` relationship:
612
+
613
+ ```js
614
+ post.hasMany('comments').reload();
615
+ ```
616
+
617
+ #### Polymorphic Relationships
618
+
619
+ To declare a polymorphic relationship, use `hasMany` with the `polymorphic`
620
+ option set to `true`:
621
+
622
+ ```js
623
+ // app/models/comment.js
624
+ import { Model, belongsTo } from '@warp-drive/legacy/model';
625
+
626
+ export default class Comment extends Model {
627
+ @belongsTo('commentable', { async: false, inverse: 'comments', polymorphic: true }) parent;
628
+ }
629
+ ```
630
+
631
+ `'commentable'` here is referred to as the "abstract type" for the polymorphic
632
+ relationship.
633
+
634
+ Polymorphic relationships with `inverse: null` will accept any type of record as their content.
635
+ Polymorphic relationships with `inverse` set to a string will only accept records with a matching
636
+ inverse relationships declaring itself as satisfying the abstract type.
637
+
638
+ Below, 'as' is used to declare the that 'post' record satisfies the abstract type 'commentable'
639
+ for this relationship.
640
+
641
+ ```js
642
+ // app/models/post.js
643
+ import { Model, hasMany } from '@warp-drive/legacy/model';
644
+
645
+ export default class Post extends Model {
646
+ @hasMany('comment', { async: false, inverse: 'parent', as: 'commentable' }) comments;
647
+ }
648
+ ```
649
+
650
+ Note: every Model that declares an inverse to a polymorphic relationship must
651
+ declare itself exactly the same. This is because polymorphism is based on structural
652
+ traits.
653
+
654
+ Polymorphic to polymorphic relationships are supported. Both sides of the relationship
655
+ must be declared as polymorphic, and the `as` option must be used to declare the abstract
656
+ type each record satisfies on both sides.
657
+
658
+ @public
659
+ @param type the name of the related resource
660
+ @param options a hash of options
661
+ */
662
+
663
+ // export function hasMany<K extends Promise<unknown>, T extends Awaited<K> = Awaited<K>>(
664
+ // type: TypeFromInstance<NoNull<T>>,
665
+ // options: RelationshipOptions<T, true>
666
+ // ): RelationshipDecorator<K>;
667
+
668
+ function hasMany(type, options) {
669
+ {
670
+ (test => {
671
+ if (!test) {
672
+ throw new Error(`hasMany must be invoked with a type and options. Did you mean \`@hasMany(${type}, { async: false, inverse: null })\`?`);
673
+ }
674
+ })(!isElementDescriptor(arguments));
675
+ }
676
+ return _hasMany(type, options);
677
+ }
678
+ export { attr, belongsTo, hasMany };