Skip to content

Commit 9779c67

Browse files
authored
Backport #9319 to lts-4-12: Add @ember-data/legacy-compat/builders (#9386)
* Copy-pasta from #9319 * Make it work
1 parent 690eb1c commit 9779c67

File tree

21 files changed

+1223
-23
lines changed

21 files changed

+1223
-23
lines changed

packages/legacy-compat/package.json

+7-1
Original file line numberDiff line numberDiff line change
@@ -47,11 +47,15 @@
4747
"dependenciesMeta": {
4848
"@ember-data/private-build-infra": {
4949
"injected": true
50+
},
51+
"@ember/string": {
52+
"injected": true
5053
}
5154
},
5255
"peerDependencies": {
5356
"@ember-data/graph": "workspace:4.12.7",
54-
"@ember-data/json-api": "workspace:4.12.7"
57+
"@ember-data/json-api": "workspace:4.12.7",
58+
"@ember/string": "^3.0.1"
5559
},
5660
"peerDependenciesMeta": {
5761
"@ember-data/graph": {
@@ -74,8 +78,10 @@
7478
"@babel/preset-typescript": "^7.21.4",
7579
"@babel/preset-env": "^7.21.4",
7680
"@babel/runtime": "^7.21.0",
81+
"@ember/string": "^3.0.1",
7782
"@rollup/plugin-babel": "^6.0.3",
7883
"@rollup/plugin-node-resolve": "^15.0.1",
84+
"@types/ember__string": "^3.0.15",
7985
"tslib": "^2.5.0",
8086
"walk-sync": "^3.0.0",
8187
"typescript": "^5.0.3"

packages/legacy-compat/rollup.config.mjs

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ export default {
1717
plugins: [
1818
// These are the modules that users should be able to import from your
1919
// addon. Anything not listed here may get optimized away.
20-
addon.publicEntrypoints(['index.js', '-private.js']),
20+
addon.publicEntrypoints(['index.js', 'builders.js', '-private.js']),
2121

2222
nodeResolve({ extensions: ['.ts'] }),
2323
babel({
+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
/**
2+
Builders for migrating from `store` methods to `store.request`.
3+
These builders enable you to migrate your codebase to using the correct syntax for `store.request` while temporarily preserving legacy behaviors.
4+
This is useful for quickly upgrading an entire app to a unified syntax while a longer incremental migration is made to shift off of adapters and serializers.
5+
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
6+
@module @ember-data/legacy-compat/builders
7+
@main @ember-data/legacy-compat/builders
8+
@deprecated
9+
*/
10+
11+
export { findAllBuilder as findAll } from './builders/find-all';
12+
13+
export { findRecordBuilder as findRecord } from './builders/find-record';
14+
15+
export { queryBuilder as query, queryRecordBuilder as queryRecord } from './builders/query';
16+
17+
export { saveRecordBuilder as saveRecord } from './builders/save-record';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
/**
2+
* @module @ember-data/legacy-compat/builders
3+
*/
4+
import { assert } from '@ember/debug';
5+
6+
import type Model from '@ember-data/model';
7+
import { SkipCache } from '@ember-data/request';
8+
import type { ImmutableRequestInfo } from '@ember-data/request/-private/types';
9+
import { normalizeModelName } from './utils';
10+
11+
// Keeping unused generics for consistency with 5x types
12+
type FindAllRequestInput<T extends string = string, RT = unknown[]> = ImmutableRequestInfo & {
13+
op: 'findAll';
14+
data: {
15+
type: T;
16+
options: FindAllBuilderOptions;
17+
};
18+
};
19+
20+
type FindAllBuilderOptions = {
21+
reload?: boolean;
22+
backgroundReload?: boolean;
23+
include?: string | string[];
24+
adapterOptions?: Record<string, unknown>;
25+
};
26+
27+
/**
28+
This function builds a request config to perform a `findAll` request for the given type.
29+
When passed to `store.request`, this config will result in the same behavior as a `store.findAll` request.
30+
Additionally, it takes the same options as `store.findAll`.
31+
32+
All `@ember-data/legacy-compat` builders exist to enable you to migrate your codebase to using the correct syntax for `store.request` while temporarily preserving legacy behaviors.
33+
This is useful for quickly upgrading an entire app to a unified syntax while a longer incremental migration is made to shift off of adapters and serializers.
34+
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
35+
36+
@method findAll
37+
@deprecated
38+
@public
39+
@static
40+
@for @ember-data/legacy-compat/builders
41+
@param {string} type the name of the resource
42+
@param {object} query a query to be used by the adapter
43+
@param {FindAllBuilderOptions} [options] optional, may include `adapterOptions` hash which will be passed to adapter.findAll
44+
@return {FindAllRequestInput} request config
45+
*/
46+
// Keeping this generic for consistency with 5x types
47+
export function findAllBuilder<T extends Model>(
48+
type: string,
49+
options?: FindAllBuilderOptions
50+
): FindAllRequestInput<string, Model[]>;
51+
export function findAllBuilder(type: string, options?: FindAllBuilderOptions): FindAllRequestInput;
52+
export function findAllBuilder(type: string, options: FindAllBuilderOptions = {}): FindAllRequestInput {
53+
assert(`You need to pass a model name to the findAll builder`, type);
54+
assert(
55+
`Model name passed to the findAll builder must be a dasherized string instead of ${type}`,
56+
typeof type === 'string'
57+
);
58+
59+
return {
60+
op: 'findAll',
61+
data: {
62+
type: normalizeModelName(type),
63+
options: options || {},
64+
},
65+
cacheOptions: { [SkipCache as symbol]: true },
66+
};
67+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
/**
2+
* @module @ember-data/legacy-compat/builders
3+
*/
4+
import { assert } from '@ember/debug';
5+
6+
import type Model from '@ember-data/model';
7+
import { SkipCache } from '@ember-data/request';
8+
import type { ImmutableRequestInfo } from '@ember-data/request/-private/types';
9+
import { constructResource, ensureStringId } from '@ember-data/store/-private';
10+
import type { ResourceIdentifierObject } from '@ember-data/types/q/ember-data-json-api';
11+
import { isMaybeIdentifier, normalizeModelName } from './utils';
12+
13+
// Keeping unused generics for consistency with 5x types
14+
type FindRecordRequestInput<T extends string = string, RT = unknown> = ImmutableRequestInfo & {
15+
op: 'findRecord';
16+
data: {
17+
record: ResourceIdentifierObject;
18+
options: FindRecordBuilderOptions;
19+
};
20+
};
21+
22+
type FindRecordBuilderOptions = {
23+
reload?: boolean;
24+
backgroundReload?: boolean;
25+
include?: string | string[];
26+
adapterOptions?: Record<string, unknown>;
27+
};
28+
29+
/**
30+
This function builds a request config to find the record for a given identifier or type and id combination.
31+
When passed to `store.request`, this config will result in the same behavior as a `store.findRecord` request.
32+
Additionally, it takes the same options as `store.findRecord`, with the exception of `preload` (which is unsupported).
33+
34+
**Example 1**
35+
36+
```ts
37+
import { findRecord } from '@ember-data/legacy-compat/builders';
38+
const { content: post } = await store.request<Post>(findRecord<Post>('post', '1'));
39+
```
40+
41+
**Example 2**
42+
43+
`findRecord` can be called with a single identifier argument instead of the combination
44+
of `type` (modelName) and `id` as separate arguments. You may recognize this combo as
45+
the typical pairing from [JSON:API](https://jsonapi.org/format/#document-resource-object-identification)
46+
47+
```ts
48+
import { findRecord } from '@ember-data/legacy-compat/builders';
49+
const { content: post } = await store.request<Post>(findRecord<Post>({ type: 'post', id }));
50+
```
51+
52+
All `@ember-data/legacy-compat` builders exist to enable you to migrate your codebase to using the correct syntax for `store.request` while temporarily preserving legacy behaviors.
53+
This is useful for quickly upgrading an entire app to a unified syntax while a longer incremental migration is made to shift off of adapters and serializers.
54+
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
55+
56+
@method findRecord
57+
@deprecated
58+
@public
59+
@static
60+
@for @ember-data/legacy-compat/builders
61+
@param {string|object} type - either a string representing the name of the resource or a ResourceIdentifier object containing both the type (a string) and the id (a string) for the record or an lid (a string) of an existing record
62+
@param {string|number|object} id - optional object with options for the request only if the first param is a ResourceIdentifier, else the string id of the record to be retrieved
63+
@param {FindRecordBuilderOptions} [options] - if the first param is a string this will be the optional options for the request. See examples for available options.
64+
@return {FindRecordRequestInput} request config
65+
*/
66+
export function findRecordBuilder<T extends Model>(
67+
resource: string,
68+
id: string,
69+
options?: FindRecordBuilderOptions
70+
): FindRecordRequestInput<string, T>;
71+
export function findRecordBuilder(
72+
resource: string,
73+
id: string,
74+
options?: FindRecordBuilderOptions
75+
): FindRecordRequestInput;
76+
export function findRecordBuilder<T extends Model>(
77+
resource: ResourceIdentifierObject,
78+
options?: FindRecordBuilderOptions
79+
): FindRecordRequestInput<string, T>;
80+
export function findRecordBuilder(
81+
resource: ResourceIdentifierObject,
82+
options?: FindRecordBuilderOptions
83+
): FindRecordRequestInput;
84+
export function findRecordBuilder(
85+
resource: string | ResourceIdentifierObject,
86+
idOrOptions?: string | FindRecordBuilderOptions,
87+
options?: FindRecordBuilderOptions
88+
): FindRecordRequestInput {
89+
assert(
90+
`You need to pass a modelName or resource identifier as the first argument to the findRecord builder`,
91+
resource
92+
);
93+
if (isMaybeIdentifier(resource)) {
94+
options = idOrOptions as FindRecordBuilderOptions | undefined;
95+
} else {
96+
assert(
97+
`You need to pass a modelName or resource identifier as the first argument to the findRecord builder (passed ${resource})`,
98+
typeof resource === 'string'
99+
);
100+
const type = normalizeModelName(resource);
101+
const normalizedId = ensureStringId(idOrOptions as string | number);
102+
resource = constructResource(type, normalizedId);
103+
}
104+
105+
options = options || {};
106+
107+
assert('findRecord builder does not support options.preload', !(options as any).preload);
108+
109+
return {
110+
op: 'findRecord' as const,
111+
data: {
112+
record: resource,
113+
options,
114+
},
115+
cacheOptions: { [SkipCache as symbol]: true },
116+
};
117+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,135 @@
1+
/**
2+
* @module @ember-data/legacy-compat/builders
3+
*/
4+
import { assert } from '@ember/debug';
5+
6+
import type Model from '@ember-data/model';
7+
import { SkipCache } from '@ember-data/request';
8+
import type { ImmutableRequestInfo } from '@ember-data/request/-private/types';
9+
import { normalizeModelName } from './utils';
10+
11+
type QueryRequestInput<T extends string = string, RT = unknown[]> = ImmutableRequestInfo & {
12+
op: 'query';
13+
data: {
14+
type: T;
15+
query: Record<string, unknown>;
16+
options: QueryBuilderOptions;
17+
};
18+
};
19+
20+
type QueryBuilderOptions = {
21+
[K in string | 'adapterOptions']?: K extends 'adapterOptions' ? Record<string, unknown> : unknown;
22+
};
23+
24+
/**
25+
This function builds a request config for a given type and query object.
26+
When passed to `store.request`, this config will result in the same behavior as a `store.query` request.
27+
Additionally, it takes the same options as `store.query`.
28+
29+
All `@ember-data/legacy-compat` builders exist to enable you to migrate your codebase to using the correct syntax for `store.request` while temporarily preserving legacy behaviors.
30+
This is useful for quickly upgrading an entire app to a unified syntax while a longer incremental migration is made to shift off of adapters and serializers.
31+
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
32+
33+
@method query
34+
@deprecated
35+
@public
36+
@static
37+
@for @ember-data/legacy-compat/builders
38+
@param {string} type the name of the resource
39+
@param {object} query a query to be used by the adapter
40+
@param {QueryBuilderOptions} [options] optional, may include `adapterOptions` hash which will be passed to adapter.query
41+
@return {QueryRequestInput} request config
42+
*/
43+
export function queryBuilder<T extends Model>(
44+
type: string,
45+
query: Record<string, unknown>,
46+
options?: QueryBuilderOptions
47+
): QueryRequestInput<string, T[]>;
48+
export function queryBuilder(
49+
type: string,
50+
query: Record<string, unknown>,
51+
options?: QueryBuilderOptions
52+
): QueryRequestInput;
53+
export function queryBuilder(
54+
type: string,
55+
query: Record<string, unknown>,
56+
options: QueryBuilderOptions = {}
57+
): QueryRequestInput {
58+
assert(`You need to pass a model name to the query builder`, type);
59+
assert(`You need to pass a query hash to the query builder`, query);
60+
assert(
61+
`Model name passed to the query builder must be a dasherized string instead of ${type}`,
62+
typeof type === 'string'
63+
);
64+
65+
return {
66+
op: 'query' as const,
67+
data: {
68+
type: normalizeModelName(type),
69+
query,
70+
options: options,
71+
},
72+
cacheOptions: { [SkipCache as symbol]: true },
73+
};
74+
}
75+
76+
type QueryRecordRequestInput<T extends string = string, RT = unknown> = ImmutableRequestInfo & {
77+
op: 'queryRecord';
78+
data: {
79+
type: T;
80+
query: Record<string, unknown>;
81+
options: QueryBuilderOptions;
82+
};
83+
};
84+
85+
/**
86+
This function builds a request config for a given type and query object.
87+
When passed to `store.request`, this config will result in the same behavior as a `store.queryRecord` request.
88+
Additionally, it takes the same options as `store.queryRecord`.
89+
90+
All `@ember-data/legacy-compat` builders exist to enable you to migrate your codebase to using the correct syntax for `store.request` while temporarily preserving legacy behaviors.
91+
This is useful for quickly upgrading an entire app to a unified syntax while a longer incremental migration is made to shift off of adapters and serializers.
92+
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
93+
94+
@method queryRecord
95+
@deprecated
96+
@public
97+
@static
98+
@for @ember-data/legacy-compat/builders
99+
@param {string} type the name of the resource
100+
@param {object} query a query to be used by the adapter
101+
@param {QueryBuilderOptions} [options] optional, may include `adapterOptions` hash which will be passed to adapter.query
102+
@return {QueryRecordRequestInput} request config
103+
*/
104+
export function queryRecordBuilder<T extends Model>(
105+
type: string,
106+
query: Record<string, unknown>,
107+
options?: QueryBuilderOptions
108+
): QueryRecordRequestInput<string, T | null>;
109+
export function queryRecordBuilder(
110+
type: string,
111+
query: Record<string, unknown>,
112+
options?: QueryBuilderOptions
113+
): QueryRecordRequestInput;
114+
export function queryRecordBuilder(
115+
type: string,
116+
query: Record<string, unknown>,
117+
options?: QueryBuilderOptions
118+
): QueryRecordRequestInput {
119+
assert(`You need to pass a model name to the queryRecord builder`, type);
120+
assert(`You need to pass a query hash to the queryRecord builder`, query);
121+
assert(
122+
`Model name passed to the queryRecord builder must be a dasherized string instead of ${type}`,
123+
typeof type === 'string'
124+
);
125+
126+
return {
127+
op: 'queryRecord',
128+
data: {
129+
type: normalizeModelName(type),
130+
query,
131+
options: options || {},
132+
},
133+
cacheOptions: { [SkipCache as symbol]: true },
134+
};
135+
}

0 commit comments

Comments
 (0)