1 | ;
|
2 |
|
3 | var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
|
4 |
|
5 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
|
6 |
|
7 | var __awaiter = undefined && undefined.__awaiter || function (thisArg, _arguments, P, generator) {
|
8 | return new (P || (P = Promise))(function (resolve, reject) {
|
9 | function fulfilled(value) {
|
10 | try {
|
11 | step(generator.next(value));
|
12 | } catch (e) {
|
13 | reject(e);
|
14 | }
|
15 | }
|
16 | function rejected(value) {
|
17 | try {
|
18 | step(generator.throw(value));
|
19 | } catch (e) {
|
20 | reject(e);
|
21 | }
|
22 | }
|
23 | function step(result) {
|
24 | result.done ? resolve(result.value) : new P(function (resolve) {
|
25 | resolve(result.value);
|
26 | }).then(fulfilled, rejected);
|
27 | }
|
28 | step((generator = generator.apply(thisArg, _arguments)).next());
|
29 | });
|
30 | };
|
31 | var dexie_1 = require('dexie');
|
32 | var typestore_1 = require('typestore');
|
33 | var IndexedDBRepoPlugin_1 = require("./IndexedDBRepoPlugin");
|
34 | var log = typestore_1.Log.create(__filename);
|
35 | /**
|
36 | * Default options
|
37 | */
|
38 | exports.LocalStorageOptionDefaults = {
|
39 | databaseName: 'typestore-db',
|
40 | version: 1
|
41 | };
|
42 | /**
|
43 | * Uses dexie under the covers - its a mature library - and i'm lazy
|
44 | */
|
45 |
|
46 | var IndexedDBPlugin = function () {
|
47 | function IndexedDBPlugin() {
|
48 | var opts = arguments.length <= 0 || arguments[0] === undefined ? {} : arguments[0];
|
49 |
|
50 | _classCallCheck(this, IndexedDBPlugin);
|
51 |
|
52 | this.opts = opts;
|
53 | this.type = typestore_1.PluginType.Store;
|
54 | this.repoPlugins = {};
|
55 | this.opts = Object.assign({}, exports.LocalStorageOptionDefaults, opts);
|
56 |
|
57 | for (var _len = arguments.length, supportedModels = Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
|
58 | supportedModels[_key - 1] = arguments[_key];
|
59 | }
|
60 |
|
61 | this.supportedModels = supportedModels;
|
62 | }
|
63 |
|
64 | _createClass(IndexedDBPlugin, [{
|
65 | key: 'newDexie',
|
66 | value: function newDexie() {
|
67 | return new dexie_1.default(this.opts.databaseName, this.opts.provider);
|
68 | }
|
69 | }, {
|
70 | key: 'open',
|
71 | value: function open() {
|
72 | this.internalDb = this.newDexie();
|
73 | return this.internalDb;
|
74 | }
|
75 | }, {
|
76 | key: 'handle',
|
77 | value: function handle(eventType) {
|
78 | switch (eventType) {
|
79 | case typestore_1.PluginEventType.RepoInit:
|
80 | return typestore_1.repoAttachIfSupported(arguments.length <= 1 ? undefined : arguments[1], this);
|
81 | }
|
82 | return false;
|
83 | }
|
84 | }, {
|
85 | key: 'table',
|
86 | value: function table(modelType) {
|
87 | var table = this.tables[modelType.name];
|
88 | if (!table) throw new Error('Unable to find a table definition for ' + modelType.name);
|
89 | return table;
|
90 | }
|
91 | }, {
|
92 | key: 'init',
|
93 | value: function init(coordinator, opts) {
|
94 | return __awaiter(this, void 0, void 0, regeneratorRuntime.mark(function _callee() {
|
95 | return regeneratorRuntime.wrap(function _callee$(_context) {
|
96 | while (1) {
|
97 | switch (_context.prev = _context.next) {
|
98 | case 0:
|
99 | this.coordinator = coordinator;
|
100 | return _context.abrupt('return', coordinator);
|
101 |
|
102 | case 2:
|
103 | case 'end':
|
104 | return _context.stop();
|
105 | }
|
106 | }
|
107 | }, _callee, this);
|
108 | }));
|
109 | }
|
110 | }, {
|
111 | key: 'start',
|
112 | value: function start() {
|
113 | return __awaiter(this, void 0, void 0, regeneratorRuntime.mark(function _callee2() {
|
114 | var _this = this;
|
115 |
|
116 | var models, schemaAttrNameMap, schema,
|
117 | // Check for an existing database, version, schema
|
118 | version;
|
119 | return regeneratorRuntime.wrap(function _callee2$(_context2) {
|
120 | while (1) {
|
121 | switch (_context2.prev = _context2.next) {
|
122 | case 0:
|
123 | models = this.coordinator.getModels();
|
124 | // 1. Create the current schema config
|
125 | // TODO: Should only use indexed attributes for schema
|
126 |
|
127 | schemaAttrNameMap = {};
|
128 | schema = models.reduce(function (newSchema, modelType) {
|
129 | // Get all the known attributes for the table
|
130 | var attrs = modelType.options.attrs.filter(function (attr) {
|
131 | return !attr.transient;
|
132 | });
|
133 | schemaAttrNameMap[modelType.name] = attrs.map(function (attr) {
|
134 | return attr.name;
|
135 | });
|
136 | var attrDescs = attrs.map(function (attr) {
|
137 | var index = attr.index;
|
138 | var name = attr.name;
|
139 | var primaryKey = attr.primaryKey;
|
140 | var isArray = attr.isArray;var unique = primaryKey || index && index.unique;var prefix = unique ? '&' : isArray ? '*' : '';
|
141 | return '' + prefix + name;
|
142 | });
|
143 | // Added the attribute descriptor to the new schema
|
144 | newSchema[modelType.name] = attrDescs.join(',');
|
145 | log.debug('Created schema for ' + modelType.name, newSchema[modelType.name]);
|
146 | return newSchema;
|
147 | }, {});
|
148 | version = this.opts.version;
|
149 | _context2.next = 6;
|
150 | return new Promise(function (resolve, reject) {
|
151 | var db = _this.newDexie();
|
152 | db.open().then(function () {
|
153 | log.info('Opened existing database', db.name, ' with existing version ', db.verno);
|
154 | var tables = db.tables,
|
155 | tableNames = tables.map(function (table) {
|
156 | return table.name;
|
157 | }),
|
158 | newTableNames = Object.keys(schema),
|
159 |
|
160 | // New table defined
|
161 | newTable = !newTableNames.every(function (tableName) {
|
162 | return tableNames.includes(tableName);
|
163 | }),
|
164 |
|
165 | // Table removed??
|
166 | removedTable = !tableNames.every(function (tableName) {
|
167 | return newTableNames.includes(tableName);
|
168 | });
|
169 | var attrChanged = false;
|
170 | // If no new tables then check indexes
|
171 | if (!newTable && !removedTable) {
|
172 | var _iteratorNormalCompletion = true;
|
173 | var _didIteratorError = false;
|
174 | var _iteratorError = undefined;
|
175 |
|
176 | try {
|
177 | var _loop = function _loop() {
|
178 | var table = _step.value;
|
179 | var newAttrNames = schemaAttrNameMap[table.name];var _table$schema = table.schema;
|
180 | var indexes = _table$schema.indexes;
|
181 | var primKey = _table$schema.primKey;var oldAttrNames = indexes.map(function (index) {
|
182 | return index.name;
|
183 | }).concat([primKey.name]);
|
184 | if (newAttrNames.length !== oldAttrNames.length || !oldAttrNames.every(function (attrName) {
|
185 | return newAttrNames.includes(attrName);
|
186 | })) {
|
187 | log.info('Attributes have changed on table, bumping version. New attrs ', newAttrNames, ' old attr names ', oldAttrNames);
|
188 | attrChanged = true;
|
189 | return 'break';
|
190 | }
|
191 | };
|
192 |
|
193 | for (var _iterator = tables[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
194 | var _ret = _loop();
|
195 |
|
196 | if (_ret === 'break') break;
|
197 | }
|
198 | } catch (err) {
|
199 | _didIteratorError = true;
|
200 | _iteratorError = err;
|
201 | } finally {
|
202 | try {
|
203 | if (!_iteratorNormalCompletion && _iterator.return) {
|
204 | _iterator.return();
|
205 | }
|
206 | } finally {
|
207 | if (_didIteratorError) {
|
208 | throw _iteratorError;
|
209 | }
|
210 | }
|
211 | }
|
212 | }
|
213 | if (attrChanged || newTable || removedTable) {
|
214 | log.info('Schema changes detected, bumping version, everntually auto-upgrade', attrChanged, newTable, removedTable);
|
215 | version = db.verno + 1;
|
216 | }
|
217 | log.debug('Closing db check');
|
218 | db.close();
|
219 | resolve(true);
|
220 | }).catch('NoSuchDatabaseError', function (e) {
|
221 | log.info('Database does not exist, creating: ', _this.opts.databaseName);
|
222 | resolve(false);
|
223 | }).catch(function (e) {
|
224 | log.error("Unknown error", e);
|
225 | reject(e);
|
226 | });
|
227 | });
|
228 |
|
229 | case 6:
|
230 | // Table needs to be created
|
231 | log.debug('Creating schema', schema);
|
232 | this.open().version(version).stores(schema);
|
233 | _context2.next = 10;
|
234 | return new Promise(function (resolve, reject) {
|
235 | _this.internalDb.open().then(resolve).catch(reject);
|
236 | });
|
237 |
|
238 | case 10:
|
239 | this.tables = models.reduce(function (newTables, modelType) {
|
240 | newTables[modelType.name] = _this.internalDb.table(modelType.name);
|
241 | return newTables;
|
242 | }, {});
|
243 | log.debug('IndexedDB store is ready');
|
244 | return _context2.abrupt('return', this.coordinator);
|
245 |
|
246 | case 13:
|
247 | case 'end':
|
248 | return _context2.stop();
|
249 | }
|
250 | }
|
251 | }, _callee2, this);
|
252 | }));
|
253 | }
|
254 | }, {
|
255 | key: 'stop',
|
256 | value: function stop() {
|
257 | return __awaiter(this, void 0, void 0, regeneratorRuntime.mark(function _callee3() {
|
258 | return regeneratorRuntime.wrap(function _callee3$(_context3) {
|
259 | while (1) {
|
260 | switch (_context3.prev = _context3.next) {
|
261 | case 0:
|
262 | if (!this.internalDb) {
|
263 | _context3.next = 3;
|
264 | break;
|
265 | }
|
266 |
|
267 | _context3.next = 3;
|
268 | return this.internalDb.close();
|
269 |
|
270 | case 3:
|
271 | return _context3.abrupt('return', this.coordinator);
|
272 |
|
273 | case 4:
|
274 | case 'end':
|
275 | return _context3.stop();
|
276 | }
|
277 | }
|
278 | }, _callee3, this);
|
279 | }));
|
280 | }
|
281 | }, {
|
282 | key: 'syncModels',
|
283 | value: function syncModels() {
|
284 | log.debug('Currently the localstorage plugin does not sync models');
|
285 | return Promise.resolve(this.coordinator);
|
286 | }
|
287 | /**
|
288 | * Initialize a new repo
|
289 | * TODO: verify this logic works - just reading it makes me think we could be
|
290 | * asked to init a repo a second time with the same type and do nothing
|
291 | *
|
292 | * @param repo
|
293 | * @returns {T}
|
294 | */
|
295 |
|
296 | }, {
|
297 | key: 'initRepo',
|
298 | value: function initRepo(repo) {
|
299 | var plugin = this.repoPlugins[repo.modelType.name];
|
300 | if (plugin) return plugin.repo;
|
301 | plugin = new IndexedDBRepoPlugin_1.IndexedDBRepoPlugin(this, repo);
|
302 | return plugin.repo;
|
303 | }
|
304 | }, {
|
305 | key: 'db',
|
306 | get: function get() {
|
307 | return this.internalDb;
|
308 | }
|
309 | }]);
|
310 |
|
311 | return IndexedDBPlugin;
|
312 | }();
|
313 |
|
314 | exports.IndexedDBPlugin = IndexedDBPlugin;
|
315 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["IndexedDBPlugin.js"],"names":[],"mappings":"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","file":"IndexedDBPlugin.js","sourcesContent":["\"use strict\";\nvar __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {\n    return new (P || (P = Promise))(function (resolve, reject) {\n        function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n        function rejected(value) { try { step(generator.throw(value)); } catch (e) { reject(e); } }\n        function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }\n        step((generator = generator.apply(thisArg, _arguments)).next());\n    });\n};\nconst dexie_1 = require('dexie');\nconst typestore_1 = require('typestore');\nconst IndexedDBRepoPlugin_1 = require(\"./IndexedDBRepoPlugin\");\nconst log = typestore_1.Log.create(__filename);\n/**\n * Default options\n */\nexports.LocalStorageOptionDefaults = {\n    databaseName: 'typestore-db',\n    version: 1\n};\n/**\n * Uses dexie under the covers - its a mature library - and i'm lazy\n */\nclass IndexedDBPlugin {\n    constructor(opts = {}, ...supportedModels) {\n        this.opts = opts;\n        this.type = typestore_1.PluginType.Store;\n        this.repoPlugins = {};\n        this.opts = Object.assign({}, exports.LocalStorageOptionDefaults, opts);\n        this.supportedModels = supportedModels;\n    }\n    newDexie() {\n        return new dexie_1.default(this.opts.databaseName, this.opts.provider);\n    }\n    open() {\n        this.internalDb = this.newDexie();\n        return this.internalDb;\n    }\n    get db() {\n        return this.internalDb;\n    }\n    handle(eventType, ...args) {\n        switch (eventType) {\n            case typestore_1.PluginEventType.RepoInit:\n                return typestore_1.repoAttachIfSupported(args[0], this);\n        }\n        return false;\n    }\n    table(modelType) {\n        let table = this.tables[modelType.name];\n        if (!table)\n            throw new Error(`Unable to find a table definition for ${modelType.name}`);\n        return table;\n    }\n    init(coordinator, opts) {\n        return __awaiter(this, void 0, void 0, function* () {\n            this.coordinator = coordinator;\n            return coordinator;\n        });\n    }\n    start() {\n        return __awaiter(this, void 0, void 0, function* () {\n            const models = this.coordinator.getModels();\n            // 1. Create the current schema config\n            // TODO: Should only use indexed attributes for schema\n            const schemaAttrNameMap = {};\n            const schema = models.reduce((newSchema, modelType) => {\n                // Get all the known attributes for the table\n                const attrs = modelType.options.attrs\n                    .filter(attr => !attr.transient);\n                schemaAttrNameMap[modelType.name] = attrs.map(attr => attr.name);\n                const attrDescs = attrs.map((attr) => {\n                    const { index, name, primaryKey, isArray } = attr, unique = primaryKey || (index && index.unique), prefix = ((unique) ? '&' : (isArray) ? '*' : '');\n                    return `${prefix}${name}`;\n                });\n                // Added the attribute descriptor to the new schema\n                newSchema[modelType.name] = attrDescs.join(',');\n                log.debug(`Created schema for ${modelType.name}`, newSchema[modelType.name]);\n                return newSchema;\n            }, {});\n            // Check for an existing database, version, schema\n            let { version } = this.opts;\n            yield new Promise((resolve, reject) => {\n                const db = this.newDexie();\n                db.open()\n                    .then(() => {\n                    log.info('Opened existing database', db.name, ' with existing version ', db.verno);\n                    const tables = db.tables, tableNames = tables.map(table => table.name), newTableNames = Object.keys(schema), \n                    // New table defined\n                    newTable = !newTableNames.every(tableName => tableNames.includes(tableName)), \n                    // Table removed??\n                    removedTable = !tableNames.every(tableName => newTableNames.includes(tableName));\n                    let attrChanged = false;\n                    // If no new tables then check indexes\n                    if (!newTable && !removedTable) {\n                        for (let table of tables) {\n                            const newAttrNames = schemaAttrNameMap[table.name], { indexes, primKey } = table.schema, oldAttrNames = indexes.map(index => index.name).concat([primKey.name]);\n                            if (newAttrNames.length !== oldAttrNames.length || !oldAttrNames.every(attrName => newAttrNames.includes(attrName))) {\n                                log.info('Attributes have changed on table, bumping version.  New attrs ', newAttrNames, ' old attr names ', oldAttrNames);\n                                attrChanged = true;\n                                break;\n                            }\n                        }\n                    }\n                    if (attrChanged || newTable || removedTable) {\n                        log.info('Schema changes detected, bumping version, everntually auto-upgrade', attrChanged, newTable, removedTable);\n                        version = db.verno + 1;\n                    }\n                    log.debug('Closing db check');\n                    db.close();\n                    resolve(true);\n                })\n                    .catch('NoSuchDatabaseError', (e) => {\n                    log.info('Database does not exist, creating: ', this.opts.databaseName);\n                    resolve(false);\n                })\n                    .catch((e) => {\n                    log.error(\"Unknown error\", e);\n                    reject(e);\n                });\n            });\n            // Table needs to be created\n            log.debug(`Creating schema`, schema);\n            this.open()\n                .version(version)\n                .stores(schema);\n            yield new Promise((resolve, reject) => {\n                this.internalDb.open().then(resolve).catch(reject);\n            });\n            this.tables = models.reduce((newTables, modelType) => {\n                newTables[modelType.name] = this.internalDb.table(modelType.name);\n                return newTables;\n            }, {});\n            log.debug('IndexedDB store is ready');\n            return this.coordinator;\n        });\n    }\n    stop() {\n        return __awaiter(this, void 0, void 0, function* () {\n            if (this.internalDb)\n                yield this.internalDb.close();\n            return this.coordinator;\n        });\n    }\n    syncModels() {\n        log.debug('Currently the localstorage plugin does not sync models');\n        return Promise.resolve(this.coordinator);\n    }\n    /**\n     * Initialize a new repo\n     * TODO: verify this logic works - just reading it makes me think we could be\n     *  asked to init a repo a second time with the same type and do nothing\n     *\n     * @param repo\n     * @returns {T}\n     */\n    initRepo(repo) {\n        let plugin = this.repoPlugins[repo.modelType.name];\n        if (plugin)\n            return plugin.repo;\n        plugin = new IndexedDBRepoPlugin_1.IndexedDBRepoPlugin(this, repo);\n        return plugin.repo;\n    }\n}\nexports.IndexedDBPlugin = IndexedDBPlugin;\n"]}
|
316 | //# sourceMappingURL=IndexedDBPlugin.js.map
|