-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
Copy pathsave-record.ts
94 lines (82 loc) · 3.56 KB
/
save-record.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
/**
* @module @ember-data/legacy-compat/builders
*/
import { assert } from '@ember/debug';
import type Model from '@ember-data/model';
import { SkipCache } from '@ember-data/request';
import type { ImmutableRequestInfo } from '@ember-data/request/-private/types';
import { recordIdentifierFor, storeFor } from '@ember-data/store';
import type { InstanceCache } from '@ember-data/store/-private/caches/instance-cache';
import type { Cache } from '@ember-data/types/cache/cache';
import type { StableRecordIdentifier } from '@ember-data/types/q/identifier';
type SaveRecordRequestInput<T extends string = string, RT = unknown> = ImmutableRequestInfo & {
op: 'createRecord' | 'deleteRecord' | 'updateRecord';
data: {
record: StableRecordIdentifier;
options: SaveRecordBuilderOptions;
};
records: [StableRecordIdentifier];
};
type SaveRecordBuilderOptions = Record<string, unknown>;
function _resourceIsFullDeleted(identifier: StableRecordIdentifier, cache: Cache): boolean {
return cache.isDeletionCommitted(identifier) || (cache.isNew(identifier) && cache.isDeleted(identifier));
}
function resourceIsFullyDeleted(instanceCache: InstanceCache, identifier: StableRecordIdentifier): boolean {
const cache = instanceCache.cache;
return !cache || _resourceIsFullDeleted(identifier, cache);
}
/**
This function builds a request config for saving the given record (e.g. creating, updating, or deleting the record).
When passed to `store.request`, this config will result in the same behavior as a legacy `store.saveRecord` request.
Additionally, it takes the same options as `store.saveRecord`.
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.
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.
To that end, these builders are deprecated and will be removed in a future version of Ember Data.
@method saveRecord
@deprecated
@public
@static
@for @ember-data/legacy-compat/builders
@param {object} record a record to save
@param {SaveRecordBuilderOptions} options optional, may include `adapterOptions` hash which will be passed to adapter.saveRecord
@return {SaveRecordRequestInput} request config
*/
export function saveRecordBuilder<T extends Model>(
record: T,
options: Record<string, unknown> = {}
): SaveRecordRequestInput<string, T> {
const store = storeFor(record);
assert(`Unable to initiate save for a record in a disconnected state`, store);
const identifier = recordIdentifierFor(record);
if (!identifier) {
// this commonly means we're disconnected
// but just in case we throw here to prevent bad things.
throw new Error(`Record Is Disconnected`);
}
assert(
`Cannot initiate a save request for an unloaded record: ${identifier.lid}`,
store._instanceCache.recordIsLoaded(identifier)
);
if (resourceIsFullyDeleted(store._instanceCache, identifier)) {
throw new Error('cannot build saveRecord request for deleted record');
}
if (!options) {
options = {};
}
let operation: 'createRecord' | 'deleteRecord' | 'updateRecord' = 'updateRecord';
const cache = store.cache;
if (cache.isNew(identifier)) {
operation = 'createRecord';
} else if (cache.isDeleted(identifier)) {
operation = 'deleteRecord';
}
return {
op: operation,
data: {
options,
record: identifier,
},
records: [identifier],
cacheOptions: { [SkipCache as symbol]: true },
};
}