Skip to content

Commit 5857050

Browse files
committed
chore: type attr decorator
1 parent 3073e92 commit 5857050

10 files changed

+118
-23
lines changed

packages/model/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -145,6 +145,7 @@
145145
"@warp-drive/internal-config": "workspace:5.4.0-alpha.28",
146146
"ember-inflector": "^4.0.2",
147147
"ember-source": "~5.6.0",
148+
"expect-type": "^0.18.0",
148149
"rollup": "^4.9.6",
149150
"typescript": "^5.3.3",
150151
"walk-sync": "^3.0.0",

packages/model/src/-private.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
1-
export { default as attr } from './-private/attr';
1+
export { attr } from './-private/attr';
22
export { default as belongsTo } from './-private/belongs-to';
33
export { default as hasMany } from './-private/has-many';
4-
export { default as Model } from './-private/model';
4+
export { Model } from './-private/model';
55
export { default as Errors } from './-private/errors';
66

77
export { default as ManyArray } from './-private/many-array';

packages/model/src/-private/attr.js packages/model/src/-private/attr.ts

+96-12
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,63 @@
11
import { assert } from '@ember/debug';
22
import { computed } from '@ember/object';
33

4+
/**
5+
@module @ember-data/model
6+
*/
7+
import { expectTypeOf } from 'expect-type';
8+
49
import { DEBUG } from '@ember-data/env';
510
import { recordIdentifierFor } from '@ember-data/store';
611
import { peekCache } from '@ember-data/store/-private';
12+
import type { Value } from '@warp-drive/core-types/json/raw';
713

8-
import { computedMacroWithOptionalParams } from './util';
14+
import type { Model } from './model';
15+
import type { DataDecorator, DecoratorPropertyDescriptor } from './util';
16+
import { isElementDescriptor } from './util';
917

1018
/**
11-
@module @ember-data/model
12-
*/
19+
* Options provided to the attr decorator are
20+
* supplied to the associated transform. Any
21+
* key-value pair is valid; however, it is highly
22+
* recommended to only use statically defined values
23+
* that could be serialized to JSON.
24+
*
25+
* If no transform is provided, the only valid
26+
* option is `defaultValue`.
27+
*
28+
* Examples:
29+
*
30+
* ```ts
31+
* class User extends Model {
32+
* @attr('string', { defaultValue: 'Anonymous' }) name;
33+
* @attr('date', { defaultValue: () => new Date() }) createdAt;
34+
* @attr({ defaultValue: () => ({}) }) preferences;
35+
* @attr('boolean') hasVerifiedEmail;
36+
* @attr address;
37+
* }
38+
*
39+
* @typedoc
40+
*/
41+
type AttrOptions = {
42+
/**
43+
* The default value for this attribute.
44+
*
45+
* Default values can be provided as a value or a function that will be
46+
* executed to generate the default value.
47+
*
48+
* Default values *should not* be stateful (object, arrays, etc.) as
49+
* they will be shared across all instances of the record.
50+
*
51+
* @typedoc
52+
*/
53+
defaultValue?: string | number | boolean | null | (() => unknown);
54+
};
1355

1456
/**
1557
`attr` defines an attribute on a [Model](/ember-data/release/classes/Model).
1658
By default, attributes are passed through as-is, however you can specify an
1759
optional type to have the value automatically transformed.
18-
Ember Data ships with four basic transform types: `string`, `number`,
60+
EmberData ships with four basic transform types: `string`, `number`,
1961
`boolean` and `date`. You can define your own transforms by subclassing
2062
[Transform](/ember-data/release/classes/Transform).
2163
@@ -101,7 +143,7 @@ import { computedMacroWithOptionalParams } from './util';
101143
@param {Object} options a hash of options
102144
@return {Attribute}
103145
*/
104-
function attr(type, options) {
146+
function _attr(type?: string | AttrOptions, options?: AttrOptions & object) {
105147
if (typeof type === 'object') {
106148
options = type;
107149
type = undefined;
@@ -118,9 +160,9 @@ function attr(type, options) {
118160
};
119161

120162
return computed({
121-
get(key) {
163+
get(this: Model, key: string) {
122164
if (DEBUG) {
123-
if (['currentState'].indexOf(key) !== -1) {
165+
if (['currentState'].includes(key)) {
124166
throw new Error(
125167
`'${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()}`
126168
);
@@ -131,28 +173,31 @@ function attr(type, options) {
131173
}
132174
return peekCache(this).getAttr(recordIdentifierFor(this), key);
133175
},
134-
set(key, value) {
176+
set(this: Model, key: string, value: Value) {
135177
if (DEBUG) {
136-
if (['currentState'].indexOf(key) !== -1) {
178+
if (['currentState'].includes(key)) {
137179
throw new Error(
138180
`'${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()}`
139181
);
140182
}
141183
}
184+
const identifier = recordIdentifierFor(this);
142185
assert(
143-
`Attempted to set '${key}' on the deleted record ${recordIdentifierFor(this)}`,
186+
`Attempted to set '${key}' on the deleted record ${identifier.type}:${identifier.id} (${identifier.lid})`,
144187
!this.currentState.isDeleted
145188
);
146-
const identifier = recordIdentifierFor(this);
147189
const cache = peekCache(this);
148190

149191
const currentValue = cache.getAttr(identifier, key);
150192
if (currentValue !== value) {
151193
cache.setAttr(identifier, key, value);
152194

153195
if (!this.isValid) {
196+
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
154197
const { errors } = this;
198+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
155199
if (errors.get(key)) {
200+
// eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
156201
errors.remove(key);
157202
this.currentState.cleanErrorRequests();
158203
}
@@ -164,4 +209,43 @@ function attr(type, options) {
164209
}).meta(meta);
165210
}
166211

167-
export default computedMacroWithOptionalParams(attr);
212+
export function attr(): DataDecorator;
213+
export function attr(type: string): DataDecorator;
214+
export function attr(options: AttrOptions): DataDecorator;
215+
export function attr(type: string, options?: AttrOptions & object): DataDecorator;
216+
export function attr(target: object, key: string, desc: PropertyDescriptor): DecoratorPropertyDescriptor;
217+
export function attr(
218+
type?: string | AttrOptions | object,
219+
options?: (AttrOptions & object) | string,
220+
desc?: PropertyDescriptor
221+
): DataDecorator | DecoratorPropertyDescriptor {
222+
const args = [type, options, desc];
223+
return isElementDescriptor(args) ? _attr()(...args) : _attr(type, options as object);
224+
}
225+
226+
// positive tests
227+
expectTypeOf(attr({}, 'key', {})).toEqualTypeOf<DecoratorPropertyDescriptor>();
228+
expectTypeOf(attr('string')).toEqualTypeOf<DataDecorator>();
229+
expectTypeOf(attr({})).toEqualTypeOf<DataDecorator>();
230+
expectTypeOf(attr('string', {})).toEqualTypeOf<DataDecorator>();
231+
expectTypeOf(attr()).toEqualTypeOf<DataDecorator>();
232+
233+
expectTypeOf(attr('string', { defaultValue: 'hello' })).toEqualTypeOf<DataDecorator>();
234+
expectTypeOf(attr({ defaultValue: 'hello' })).toEqualTypeOf<DataDecorator>();
235+
expectTypeOf(attr('string', { defaultValue: () => {} })).toEqualTypeOf<DataDecorator>();
236+
expectTypeOf(attr({ defaultValue: () => {} })).toEqualTypeOf<DataDecorator>();
237+
238+
/* prettier-ignore */
239+
expectTypeOf(
240+
// @ts-expect-error
241+
attr(
242+
{ defaultValue: {} }
243+
)
244+
).toBeNever;
245+
expectTypeOf(
246+
attr(
247+
// @ts-expect-error
248+
1,
249+
{ defaultValue: 'hello' }
250+
)
251+
).toBeNever;

packages/model/src/-private/hooks.ts

+1-2
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,7 @@ import type { StableRecordIdentifier } from '@warp-drive/core-types';
66
import type { Cache } from '@warp-drive/core-types/cache';
77
import type { TypeFromInstance, TypeFromInstanceOrString } from '@warp-drive/core-types/record';
88

9-
import type { ModelStore } from './model';
10-
import type Model from './model';
9+
import type { Model, ModelStore } from './model';
1110
import { getModelFactory } from './schema-provider';
1211
import { normalizeModelName } from './util';
1312

packages/model/src/-private/model.d.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ interface Model {
8484
constructor: typeof Model;
8585
}
8686

87-
export default Model;
87+
export { Model };
8888

8989
export type StaticModel = typeof Model;
9090

packages/model/src/-private/model.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -2109,4 +2109,4 @@ if (DEBUG) {
21092109
};
21102110
}
21112111

2112-
export default Model;
2112+
export { Model };

packages/model/src/-private/notify-changes.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import type { StableRecordIdentifier } from '@warp-drive/core-types';
77
import type { RelationshipSchema } from '@warp-drive/core-types/schema';
88

99
import { LEGACY_SUPPORT } from './legacy-relationships-support';
10-
import type Model from './model';
10+
import type { Model } from './model';
1111

1212
export default function notifyChanges(
1313
identifier: StableRecordIdentifier,

packages/model/src/-private/schema-provider.ts

+1-2
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,7 @@ import type { FieldSchema } from '@ember-data/store/-types/q/schema-service';
55
import type { RecordIdentifier } from '@warp-drive/core-types/identifier';
66
import type { AttributesSchema, RelationshipsSchema } from '@warp-drive/core-types/schema';
77

8-
import type { FactoryCache, ModelFactory, ModelStore } from './model';
9-
import type Model from './model';
8+
import type { FactoryCache, Model, ModelFactory, ModelStore } from './model';
109
import _modelForMixin from './model-for-mixin';
1110
import { normalizeModelName } from './util';
1211

packages/model/src/-private/util.ts

+6-2
Original file line numberDiff line numberDiff line change
@@ -25,8 +25,12 @@ export function isElementDescriptor(args: unknown[]): args is [object, string, D
2525
);
2626
}
2727

28-
type DataDecorator = (target: object, key: string, desc?: DecoratorPropertyDescriptor) => DecoratorPropertyDescriptor;
29-
type DataDecoratorFactory = (...args: unknown[]) => DataDecorator;
28+
export type DataDecorator = (
29+
target: object,
30+
key: string,
31+
desc?: DecoratorPropertyDescriptor
32+
) => DecoratorPropertyDescriptor;
33+
export type DataDecoratorFactory = (...args: unknown[]) => DataDecorator;
3034

3135
export function computedMacroWithOptionalParams(fn: DataDecorator | DataDecoratorFactory) {
3236
return (...maybeDesc: unknown[]) =>

pnpm-lock.yaml

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)