@aws-amplify/datastore-storage-adapter 2.0.52 → 2.0.54

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 (30) hide show
  1. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/ExpoSQLiteAdapter/ExpoSQLiteAdapter.js +9 -0
  2. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/ExpoSQLiteAdapter/ExpoSQLiteDatabase.js +195 -0
  3. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/SQLiteAdapter/SQLiteAdapter.js +9 -0
  4. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/SQLiteAdapter/SQLiteDatabase.js +114 -0
  5. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/common/CommonSQLiteAdapter.js +240 -0
  6. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/common/SQLiteUtils.js +349 -0
  7. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/common/constants.js +6 -0
  8. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/common/types.js +2 -0
  9. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/cjs/index.js +8 -0
  10. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/ExpoSQLiteAdapter/ExpoSQLiteAdapter.d.ts +3 -0
  11. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/ExpoSQLiteAdapter/ExpoSQLiteAdapter.js +6 -0
  12. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/ExpoSQLiteAdapter/ExpoSQLiteDatabase.d.ts +17 -0
  13. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/ExpoSQLiteAdapter/ExpoSQLiteDatabase.js +193 -0
  14. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/SQLiteAdapter/SQLiteAdapter.d.ts +3 -0
  15. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/SQLiteAdapter/SQLiteAdapter.js +6 -0
  16. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/SQLiteAdapter/SQLiteDatabase.d.ts +17 -0
  17. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/SQLiteAdapter/SQLiteDatabase.js +111 -0
  18. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/CommonSQLiteAdapter.d.ts +23 -0
  19. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/CommonSQLiteAdapter.js +236 -0
  20. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/SQLiteUtils.d.ts +36 -0
  21. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/SQLiteUtils.js +331 -0
  22. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/constants.d.ts +1 -0
  23. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/constants.js +3 -0
  24. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/types.d.ts +16 -0
  25. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/common/types.js +1 -0
  26. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/index.d.ts +2 -0
  27. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/esm/index.js +4 -0
  28. package/.rollup.cache/home/runner/work/amplify-js/amplify-js/amplify-js/packages/datastore-storage-adapter/dist/meta/cjs.tsbuildinfo +1 -0
  29. package/CHANGELOG.md +8 -0
  30. package/package.json +4 -4
@@ -0,0 +1,331 @@
1
+ // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2
+ // SPDX-License-Identifier: Apache-2.0
3
+ import { isGraphQLScalarType, QueryOne, isPredicateObj, isPredicateGroup, isModelFieldType, isTargetNameAssociation, isModelAttributeAuth, utils, } from '@aws-amplify/datastore';
4
+ const { USER, isNonModelConstructor, isModelConstructor } = utils;
5
+ const keysFromModel = model => Object.keys(model)
6
+ .map(k => `"${k}"`)
7
+ .join(', ');
8
+ const valuesFromModel = (model) => {
9
+ const values = Object.values(model).map(prepareValueForDML);
10
+ const paramaterized = values.map(() => '?').join(', ');
11
+ return [paramaterized, values];
12
+ };
13
+ const updateSet = model => {
14
+ const values = [];
15
+ const paramaterized = Object.entries(model)
16
+ .filter(([k]) => k !== 'id')
17
+ .map(([k, v]) => {
18
+ values.push(prepareValueForDML(v));
19
+ return `"${k}"=?`;
20
+ })
21
+ .join(', ');
22
+ return [paramaterized, values];
23
+ };
24
+ function prepareValueForDML(value) {
25
+ const scalarTypes = ['string', 'number', 'boolean'];
26
+ const isScalarType = value === null || value === undefined || scalarTypes.includes(typeof value);
27
+ if (isScalarType) {
28
+ return value;
29
+ }
30
+ const isObjectType = typeof value === 'object' &&
31
+ (Object.getPrototypeOf(value).constructor === Object ||
32
+ isNonModelConstructor(Object.getPrototypeOf(value).constructor) ||
33
+ isModelConstructor(Object.getPrototypeOf(value).constructor));
34
+ if (Array.isArray(value) || isObjectType) {
35
+ return JSON.stringify(value);
36
+ }
37
+ return `${value}`;
38
+ }
39
+ export function getSQLiteType(scalar) {
40
+ switch (scalar) {
41
+ case 'Boolean':
42
+ case 'Int':
43
+ case 'AWSTimestamp':
44
+ return 'INTEGER';
45
+ case 'ID':
46
+ case 'String':
47
+ case 'AWSDate':
48
+ case 'AWSTime':
49
+ case 'AWSDateTime':
50
+ case 'AWSEmail':
51
+ case 'AWSJSON':
52
+ case 'AWSURL':
53
+ case 'AWSPhone':
54
+ case 'AWSIPAddress':
55
+ return 'TEXT';
56
+ case 'Float':
57
+ return 'REAL';
58
+ default:
59
+ const _ = scalar;
60
+ throw new Error(`unknown type ${scalar}`);
61
+ }
62
+ }
63
+ export function generateSchemaStatements(schema) {
64
+ return Object.keys(schema.namespaces).flatMap(namespaceName => {
65
+ const namespace = schema.namespaces[namespaceName];
66
+ const isUserModel = namespaceName === USER;
67
+ return Object.values(namespace.models).map(model => modelCreateTableStatement(model, isUserModel));
68
+ });
69
+ }
70
+ export const implicitAuthFieldsForModel = (model) => {
71
+ if (!model.attributes || !model.attributes.length) {
72
+ return [];
73
+ }
74
+ const authRules = model.attributes.find(isModelAttributeAuth);
75
+ if (!authRules) {
76
+ return [];
77
+ }
78
+ const authFieldsForModel = authRules.properties.rules
79
+ .filter((rule) => rule.ownerField || rule.groupsField)
80
+ .map((rule) => rule.ownerField || rule.groupsField);
81
+ return authFieldsForModel.filter((authField) => {
82
+ const authFieldExplicitlyDefined = Object.values(model.fields).find((f) => f.name === authField);
83
+ return !authFieldExplicitlyDefined;
84
+ });
85
+ };
86
+ export function modelCreateTableStatement(model, userModel = false) {
87
+ // implicitly defined auth fields, e.g., `owner`, `groupsField`, etc.
88
+ const implicitAuthFields = implicitAuthFieldsForModel(model);
89
+ let fields = Object.values(model.fields).reduce((acc, field) => {
90
+ if (isGraphQLScalarType(field.type)) {
91
+ if (field.name === 'id') {
92
+ return [...acc, '"id" PRIMARY KEY NOT NULL'];
93
+ }
94
+ let columnParam = `"${field.name}" ${getSQLiteType(field.type)}`;
95
+ if (field.isRequired) {
96
+ columnParam += ' NOT NULL';
97
+ }
98
+ return [...acc, `${columnParam}`];
99
+ }
100
+ if (isModelFieldType(field.type)) {
101
+ let columnParam = `"${field.name}" TEXT`;
102
+ // add targetName as well as field name for BELONGS_TO relations
103
+ if (isTargetNameAssociation(field.association)) {
104
+ // check if this field has been explicitly defined in the model
105
+ const fkDefinedInModel = Object.values(model.fields).find((f) => f.name === field?.association?.targetName);
106
+ // if the FK is not explicitly defined in the model, we have to add it here
107
+ if (!fkDefinedInModel) {
108
+ const required = field.isRequired ? ' NOT NULL' : '';
109
+ columnParam += `, "${field.association.targetName}" TEXT${required}`;
110
+ }
111
+ }
112
+ // ignore isRequired param for model fields, since they will not contain
113
+ // the related data locally
114
+ return [...acc, `${columnParam}`];
115
+ }
116
+ // default to TEXT
117
+ let columnParam = `"${field.name}" TEXT`;
118
+ if (field.isRequired) {
119
+ columnParam += ' NOT NULL';
120
+ }
121
+ return [...acc, `${columnParam}`];
122
+ }, []);
123
+ implicitAuthFields.forEach((authField) => {
124
+ fields.push(`${authField} TEXT`);
125
+ });
126
+ if (userModel) {
127
+ fields = [
128
+ ...fields,
129
+ `"_version" INTEGER`,
130
+ `"_lastChangedAt" INTEGER`,
131
+ `"_deleted" INTEGER`,
132
+ ];
133
+ }
134
+ const createTableStatement = `CREATE TABLE IF NOT EXISTS "${model.name}" (${fields.join(', ')});`;
135
+ return createTableStatement;
136
+ }
137
+ export function modelInsertStatement(model, tableName) {
138
+ const keys = keysFromModel(model);
139
+ const [paramaterized, values] = valuesFromModel(model);
140
+ const insertStatement = `INSERT INTO "${tableName}" (${keys}) VALUES (${paramaterized})`;
141
+ return [insertStatement, values];
142
+ }
143
+ export function modelUpdateStatement(model, tableName) {
144
+ const [paramaterized, values] = updateSet(model);
145
+ const updateStatement = `UPDATE "${tableName}" SET ${paramaterized} WHERE id=?`;
146
+ return [updateStatement, [...values, model.id]];
147
+ }
148
+ export function queryByIdStatement(id, tableName) {
149
+ return [`SELECT * FROM "${tableName}" WHERE "id" = ?`, [id]];
150
+ }
151
+ /*
152
+ Predicates supported by DataStore:
153
+
154
+ Strings: eq | ne | le | lt | ge | gt | contains | notContains | beginsWith | between
155
+ Numbers: eq | ne | le | lt | ge | gt | between
156
+ Lists: contains | notContains
157
+ */
158
+ const comparisonOperatorMap = {
159
+ eq: '=',
160
+ ne: '!=',
161
+ le: '<=',
162
+ lt: '<',
163
+ ge: '>=',
164
+ gt: '>',
165
+ };
166
+ const logicalOperatorMap = {
167
+ beginsWith: '= 1',
168
+ contains: '> 0',
169
+ notContains: '= 0',
170
+ between: 'BETWEEN',
171
+ };
172
+ /**
173
+ * If the given (operator, operand) indicate the need for a special `NULL` comparison,
174
+ * that `WHERE` clause condition will be returned. If not special `NULL` handling is
175
+ * needed, `null` will be returned, and the caller should construct the `WHERE`
176
+ * clause component using the normal operator map(s) and parameterization.
177
+ *
178
+ * @param operator "beginsWith" | "contains" | "notContains" | "between"
179
+ * | "eq" | "ne" | "le" | "lt" | "ge" | "gt"
180
+ * @param operand any
181
+ * @returns (string | null) The `WHERE` clause component or `null` if N/A.
182
+ */
183
+ function buildSpecialNullComparison(field, operator, operand) {
184
+ if (operand === null || operand === undefined) {
185
+ if (operator === 'eq') {
186
+ return `"${field}" IS NULL`;
187
+ }
188
+ else if (operator === 'ne') {
189
+ return `"${field}" IS NOT NULL`;
190
+ }
191
+ }
192
+ // no special null handling required
193
+ return null;
194
+ }
195
+ export const whereConditionFromPredicateObject = ({ field, operator, operand, }) => {
196
+ const specialNullClause = buildSpecialNullComparison(field, operator, operand);
197
+ if (specialNullClause) {
198
+ return [specialNullClause, []];
199
+ }
200
+ const comparisonOperator = comparisonOperatorMap[operator];
201
+ if (comparisonOperator) {
202
+ return [`"${field}" ${comparisonOperator} ?`, [operand]];
203
+ }
204
+ const logicalOperatorKey = operator;
205
+ const logicalOperator = logicalOperatorMap[logicalOperatorKey];
206
+ let statement;
207
+ if (logicalOperator) {
208
+ let rightExp = [];
209
+ switch (logicalOperatorKey) {
210
+ case 'between':
211
+ rightExp = operand; // operand is a 2-tuple
212
+ statement = [
213
+ `"${field}" ${logicalOperator} ${rightExp
214
+ .map(_ => '?')
215
+ .join(' AND ')}`,
216
+ rightExp,
217
+ ];
218
+ break;
219
+ case 'beginsWith':
220
+ case 'contains':
221
+ case 'notContains':
222
+ statement = [`instr("${field}", ?) ${logicalOperator}`, [operand]];
223
+ break;
224
+ default:
225
+ const _ = logicalOperatorKey;
226
+ // Incorrect WHERE clause can result in data loss
227
+ throw new Error('Cannot map predicate to a valid WHERE clause');
228
+ }
229
+ return statement;
230
+ }
231
+ };
232
+ export function whereClauseFromPredicate(predicate) {
233
+ const result = [];
234
+ const params = [];
235
+ recurse(predicate, result, params);
236
+ const whereClause = `WHERE ${result.join(' ')}`;
237
+ return [whereClause, params];
238
+ function recurse(predicate, result = [], params = []) {
239
+ if (isPredicateGroup(predicate)) {
240
+ const { type: groupType, predicates: groupPredicates } = predicate;
241
+ let filterType = '';
242
+ let isNegation = false;
243
+ switch (groupType) {
244
+ case 'not':
245
+ isNegation = true;
246
+ break;
247
+ case 'and':
248
+ filterType = 'AND';
249
+ break;
250
+ case 'or':
251
+ filterType = 'OR';
252
+ break;
253
+ default:
254
+ const _ = groupType;
255
+ throw new Error(`Invalid ${groupType}`);
256
+ }
257
+ const groupResult = [];
258
+ for (const p of groupPredicates) {
259
+ recurse(p, groupResult, params);
260
+ }
261
+ result.push(`${isNegation ? 'NOT' : ''}(${groupResult.join(` ${filterType} `)})`);
262
+ }
263
+ else if (isPredicateObj(predicate)) {
264
+ const [condition, conditionParams] = whereConditionFromPredicateObject(predicate);
265
+ result.push(condition);
266
+ params.push(...conditionParams);
267
+ }
268
+ }
269
+ }
270
+ const sortDirectionMap = {
271
+ ASCENDING: 'ASC',
272
+ DESCENDING: 'DESC',
273
+ };
274
+ export function orderByClauseFromSort(sortPredicate = []) {
275
+ const orderByParts = sortPredicate.map(({ field, sortDirection }) => `"${String(field)}" ${sortDirectionMap[sortDirection]}`);
276
+ // We always sort by _rowid_ last
277
+ orderByParts.push(`_rowid_ ${sortDirectionMap.ASCENDING}`);
278
+ return `ORDER BY ${orderByParts.join(', ')}`;
279
+ }
280
+ export function limitClauseFromPagination(limit, page = 0) {
281
+ const params = [limit];
282
+ let clause = 'LIMIT ?';
283
+ if (page) {
284
+ const offset = limit * page;
285
+ params.push(offset);
286
+ clause += ' OFFSET ?';
287
+ }
288
+ return [clause, params];
289
+ }
290
+ export function queryAllStatement(tableName, predicate, sort, limit, page) {
291
+ let statement = `SELECT * FROM "${tableName}"`;
292
+ const params = [];
293
+ if (predicate && predicate.predicates.length) {
294
+ const [whereClause, whereParams] = whereClauseFromPredicate(predicate);
295
+ statement += ` ${whereClause}`;
296
+ params.push(...whereParams);
297
+ }
298
+ const orderByClause = orderByClauseFromSort(sort);
299
+ statement += ` ${orderByClause}`;
300
+ if (limit) {
301
+ const [limitClause, limitParams] = limitClauseFromPagination(limit, page);
302
+ statement += ` ${limitClause}`;
303
+ params.push(...limitParams);
304
+ }
305
+ return [statement, params];
306
+ }
307
+ export function queryOneStatement(firstOrLast, tableName) {
308
+ if (firstOrLast === QueryOne.FIRST) {
309
+ // ORDER BY rowid will no longer work as expected if a customer has
310
+ // a field by that name in their schema. We may want to enforce it
311
+ // as a reserved keyword in Codegen
312
+ return [`SELECT * FROM ${tableName} ORDER BY _rowid_ LIMIT 1`, []];
313
+ }
314
+ else {
315
+ return [`SELECT * FROM ${tableName} ORDER BY _rowid_ DESC LIMIT 1`, []];
316
+ }
317
+ }
318
+ export function deleteByIdStatement(id, tableName) {
319
+ const deleteStatement = `DELETE FROM "${tableName}" WHERE "id"=?`;
320
+ return [deleteStatement, [id]];
321
+ }
322
+ export function deleteByPredicateStatement(tableName, predicate) {
323
+ let statement = `DELETE FROM "${tableName}"`;
324
+ const params = [];
325
+ if (predicate && predicate.predicates.length) {
326
+ const [whereClause, whereParams] = whereClauseFromPredicate(predicate);
327
+ statement += ` ${whereClause}`;
328
+ params.push(...whereParams);
329
+ }
330
+ return [statement, params];
331
+ }
@@ -0,0 +1,3 @@
1
+ // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2
+ // SPDX-License-Identifier: Apache-2.0
3
+ export const DB_NAME = 'AmplifyDatastore';
@@ -0,0 +1,16 @@
1
+ import { PersistentModel, ModelInstanceMetadata } from '@aws-amplify/datastore';
2
+ export interface CommonSQLiteDatabase {
3
+ init(): Promise<void>;
4
+ createSchema(statements: string[]): Promise<void>;
5
+ clear(): Promise<void>;
6
+ get<T extends PersistentModel>(statement: string, params: (string | number)[]): Promise<T>;
7
+ getAll<T extends PersistentModel>(statement: string, params: (string | number)[]): Promise<T[]>;
8
+ save(statement: string, params: (string | number)[]): Promise<void>;
9
+ batchQuery<T = any>(queryParameterizedStatement: Set<ParameterizedStatement>): Promise<T[]>;
10
+ batchSave(saveParameterizedStatements: Set<ParameterizedStatement>, deleteParameterizedStatements?: Set<ParameterizedStatement>): Promise<void>;
11
+ selectAndDelete<T = any>(queryParameterizedStatement: ParameterizedStatement, deleteParameterizedStatement: ParameterizedStatement): Promise<T[]>;
12
+ }
13
+ export type ParameterizedStatement = [string, any[]];
14
+ export type ModelInstanceMetadataWithId = ModelInstanceMetadata & {
15
+ id: string;
16
+ };
@@ -0,0 +1,2 @@
1
+ import SQLiteAdapter from './SQLiteAdapter/SQLiteAdapter';
2
+ export { SQLiteAdapter };
@@ -0,0 +1,4 @@
1
+ // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2
+ // SPDX-License-Identifier: Apache-2.0
3
+ import SQLiteAdapter from './SQLiteAdapter/SQLiteAdapter';
4
+ export { SQLiteAdapter };