/**
|
* @license
|
* Copyright 2018 Google LLC. All Rights Reserved.
|
* Licensed under the Apache License, Version 2.0 (the 'License');
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an 'AS IS' BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
* =============================================================================
|
*/
|
import * as tf from '../index';
|
import { BROWSER_ENVS, describeWithFlags, runWithLock } from '../jasmine_util';
|
import { arrayBufferToBase64String, base64StringToArrayBuffer } from './io_utils';
|
import { browserLocalStorage, BrowserLocalStorage, BrowserLocalStorageManager, localStorageRouter, purgeLocalStorageArtifacts } from './local_storage';
|
describeWithFlags('LocalStorage', BROWSER_ENVS, () => {
|
// Test data.
|
const modelTopology1 = {
|
'class_name': 'Sequential',
|
'keras_version': '2.1.4',
|
'config': [{
|
'class_name': 'Dense',
|
'config': {
|
'kernel_initializer': {
|
'class_name': 'VarianceScaling',
|
'config': {
|
'distribution': 'uniform',
|
'scale': 1.0,
|
'seed': null,
|
'mode': 'fan_avg'
|
}
|
},
|
'name': 'dense',
|
'kernel_constraint': null,
|
'bias_regularizer': null,
|
'bias_constraint': null,
|
'dtype': 'float32',
|
'activation': 'linear',
|
'trainable': true,
|
'kernel_regularizer': null,
|
'bias_initializer': { 'class_name': 'Zeros', 'config': {} },
|
'units': 1,
|
'batch_input_shape': [null, 3],
|
'use_bias': true,
|
'activity_regularizer': null
|
}
|
}],
|
'backend': 'tensorflow'
|
};
|
const weightSpecs1 = [
|
{
|
name: 'dense/kernel',
|
shape: [3, 1],
|
dtype: 'float32',
|
},
|
{
|
name: 'dense/bias',
|
shape: [1],
|
dtype: 'float32',
|
}
|
];
|
const weightData1 = new ArrayBuffer(16);
|
const trainingConfig1 = {
|
loss: 'categorical_crossentropy',
|
metrics: ['accuracy'],
|
optimizer_config: { class_name: 'SGD', config: { learningRate: 0.1 } }
|
};
|
const artifacts1 = {
|
modelTopology: modelTopology1,
|
weightSpecs: weightSpecs1,
|
weightData: weightData1,
|
format: 'layers-model',
|
generatedBy: 'TensorFlow.js v0.0.0',
|
convertedBy: '1.13.1',
|
signature: null,
|
userDefinedMetadata: {},
|
modelInitializer: {},
|
initializerSignature: null,
|
trainingConfig: trainingConfig1,
|
};
|
const artifactsV0 = {
|
modelTopology: modelTopology1,
|
weightSpecs: weightSpecs1,
|
weightData: weightData1
|
};
|
function findOverflowingByteSize() {
|
const LS = window.localStorage;
|
const probeKey = `tfjs_test_probe_values_${new Date().getTime()}_${Math.random()}`;
|
const minKilobytes = 200;
|
const stepKilobytes = 200;
|
const maxKilobytes = 40000;
|
for (let kilobytes = minKilobytes; kilobytes < maxKilobytes; kilobytes += stepKilobytes) {
|
const bytes = kilobytes * 1024;
|
const data = new ArrayBuffer(bytes);
|
try {
|
const encoded = arrayBufferToBase64String(data);
|
LS.setItem(probeKey, encoded);
|
}
|
catch (err) {
|
return bytes;
|
}
|
LS.removeItem(probeKey);
|
}
|
throw new Error(`Unable to determined overflowing byte size up to ${maxKilobytes} kB.`);
|
}
|
beforeEach(() => {
|
purgeLocalStorageArtifacts();
|
});
|
afterEach(() => {
|
purgeLocalStorageArtifacts();
|
});
|
it('Save artifacts succeeds', runWithLock(async () => {
|
const testStartDate = new Date();
|
const handler = tf.io.getSaveHandlers('localstorage://foo/FooModel')[0];
|
const saveResult = await handler.save(artifacts1);
|
expect(saveResult.modelArtifactsInfo.dateSaved.getTime())
|
.toBeGreaterThanOrEqual(testStartDate.getTime());
|
// Note: The following two assertions work only because there is no
|
// non-ASCII characters in `modelTopology1` and `weightSpecs1`.
|
expect(saveResult.modelArtifactsInfo.modelTopologyBytes)
|
.toEqual(JSON.stringify(modelTopology1).length);
|
expect(saveResult.modelArtifactsInfo.weightSpecsBytes)
|
.toEqual(JSON.stringify(weightSpecs1).length);
|
expect(saveResult.modelArtifactsInfo.weightDataBytes).toEqual(16);
|
// Check the content of the saved items in local storage.
|
const LS = window.localStorage;
|
const info = JSON.parse(LS.getItem('tensorflowjs_models/foo/FooModel/info'));
|
expect(Date.parse(info.dateSaved))
|
.toEqual(saveResult.modelArtifactsInfo.dateSaved.getTime());
|
expect(info.modelTopologyBytes)
|
.toEqual(saveResult.modelArtifactsInfo.modelTopologyBytes);
|
expect(info.weightSpecsBytes)
|
.toEqual(saveResult.modelArtifactsInfo.weightSpecsBytes);
|
expect(info.weightDataBytes)
|
.toEqual(saveResult.modelArtifactsInfo.weightDataBytes);
|
const topologyString = LS.getItem('tensorflowjs_models/foo/FooModel/model_topology');
|
expect(JSON.stringify(modelTopology1)).toEqual(topologyString);
|
const weightSpecsString = LS.getItem('tensorflowjs_models/foo/FooModel/weight_specs');
|
expect(JSON.stringify(weightSpecs1)).toEqual(weightSpecsString);
|
const weightDataBase64String = LS.getItem('tensorflowjs_models/foo/FooModel/weight_data');
|
expect(base64StringToArrayBuffer(weightDataBase64String))
|
.toEqual(weightData1);
|
}));
|
it('Save-load round trip succeeds', runWithLock(async () => {
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
await handler1.save(artifacts1);
|
const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];
|
const loaded = await handler2.load();
|
expect(loaded.modelTopology).toEqual(modelTopology1);
|
expect(loaded.weightSpecs).toEqual(weightSpecs1);
|
expect(loaded.weightData).toEqual(weightData1);
|
expect(loaded.format).toEqual('layers-model');
|
expect(loaded.generatedBy).toEqual('TensorFlow.js v0.0.0');
|
expect(loaded.convertedBy).toEqual('1.13.1');
|
expect(loaded.userDefinedMetadata).toEqual({});
|
expect(loaded.modelInitializer).toEqual({});
|
expect(loaded.initializerSignature).toBeUndefined();
|
expect(loaded.trainingConfig).toEqual(trainingConfig1);
|
}));
|
it('Save-load round trip succeeds: v0 format', runWithLock(async () => {
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
await handler1.save(artifactsV0);
|
const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];
|
const loaded = await handler2.load();
|
expect(loaded.modelTopology).toEqual(modelTopology1);
|
expect(loaded.weightSpecs).toEqual(weightSpecs1);
|
expect(loaded.weightData).toEqual(weightData1);
|
expect(loaded.format).toBeUndefined();
|
expect(loaded.generatedBy).toBeUndefined();
|
expect(loaded.convertedBy).toBeUndefined();
|
expect(loaded.userDefinedMetadata).toBeUndefined();
|
expect(loaded.trainingConfig).toBeUndefined();
|
expect(loaded.modelInitializer).toBeUndefined();
|
expect(loaded.initializerSignature).toBeUndefined();
|
expect(loaded.trainingConfig).toBeUndefined();
|
}));
|
it('Loading nonexistent model fails.', runWithLock(async () => {
|
const handler = tf.io.getSaveHandlers('localstorage://NonexistentModel')[0];
|
try {
|
await handler.load();
|
}
|
catch (err) {
|
expect(err.message)
|
.toEqual('In local storage, there is no model with name ' +
|
'\'NonexistentModel\'');
|
return; // Success
|
}
|
fail('Loading nonexistent model succeeded unexpectedly.');
|
}));
|
it('Loading model with missing topology fails.', runWithLock(async () => {
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
await handler1.save(artifacts1);
|
// Manually remove the topology item from local storage.
|
window.localStorage.removeItem('tensorflowjs_models/FooModel/model_topology');
|
const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];
|
try {
|
await handler2.load();
|
}
|
catch (err) {
|
expect(err.message)
|
.toEqual('In local storage, the topology of model ' +
|
'\'FooModel\' is missing.');
|
return; // Success
|
}
|
fail('Loading of model with missing topology succeeded unexpectedly.');
|
}));
|
it('Loading model with missing weight specs fails.', runWithLock(async () => {
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
await handler1.save(artifacts1);
|
// Manually remove the weight specs item from local storage.
|
window.localStorage.removeItem('tensorflowjs_models/FooModel/weight_specs');
|
const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];
|
try {
|
await handler2.load();
|
}
|
catch (err) {
|
expect(err.message)
|
.toEqual('In local storage, the weight specs of model ' +
|
'\'FooModel\' are missing.');
|
return; // Success
|
}
|
fail('Loading of model with missing weight specs ' +
|
'succeeded unexpectedly.');
|
}));
|
it('Loading model with missing weight data fails.', runWithLock(async () => {
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
await handler1.save(artifacts1);
|
// Manually remove the weight data item from local storage.
|
window.localStorage.removeItem('tensorflowjs_models/FooModel/weight_data');
|
const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];
|
try {
|
await handler2.load();
|
fail('Loading of model with missing weight data ' +
|
'succeeded unexpectedly.');
|
}
|
catch (err) {
|
expect(err.message)
|
.toEqual('In local storage, the binary weight values of model ' +
|
'\'FooModel\' are missing.');
|
}
|
}));
|
it('Data size too large leads to error thrown', runWithLock(async () => {
|
const overflowByteSize = findOverflowingByteSize();
|
const overflowArtifacts = {
|
modelTopology: modelTopology1,
|
weightSpecs: weightSpecs1,
|
weightData: new ArrayBuffer(overflowByteSize),
|
};
|
const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];
|
try {
|
await handler1.save(overflowArtifacts);
|
fail('Saving of model of overflowing-size weight data succeeded ' +
|
'unexpectedly.');
|
}
|
catch (err) {
|
expect(err.message
|
.indexOf('Failed to save model \'FooModel\' to local storage'))
|
.toEqual(0);
|
}
|
}));
|
it('Null, undefined or empty modelPath throws Error', () => {
|
expect(() => browserLocalStorage(null))
|
.toThrowError(/local storage, modelPath must not be null, undefined or empty/);
|
expect(() => browserLocalStorage(undefined))
|
.toThrowError(/local storage, modelPath must not be null, undefined or empty/);
|
expect(() => browserLocalStorage(''))
|
.toThrowError(/local storage, modelPath must not be null, undefined or empty./);
|
});
|
it('router', () => {
|
expect(localStorageRouter('localstorage://bar') instanceof BrowserLocalStorage)
|
.toEqual(true);
|
expect(localStorageRouter('indexeddb://bar')).toBeNull();
|
expect(localStorageRouter('qux')).toBeNull();
|
});
|
it('Manager: List models: 0 result', runWithLock(async () => {
|
// Before any model is saved, listModels should return empty result.
|
const out = await new BrowserLocalStorageManager().listModels();
|
expect(out).toEqual({});
|
}));
|
it('Manager: List models: 1 result', runWithLock(async () => {
|
const handler = tf.io.getSaveHandlers('localstorage://baz/QuxModel')[0];
|
const saveResult = await handler.save(artifacts1);
|
// After successful saving, there should be one model.
|
const out = await new BrowserLocalStorageManager().listModels();
|
if (Object.keys(out).length !== 1) {
|
console.log(JSON.stringify(out, null, 2));
|
}
|
expect(Object.keys(out).length).toEqual(1);
|
expect(out['baz/QuxModel'].modelTopologyType)
|
.toEqual(saveResult.modelArtifactsInfo.modelTopologyType);
|
expect(out['baz/QuxModel'].modelTopologyBytes)
|
.toEqual(saveResult.modelArtifactsInfo.modelTopologyBytes);
|
expect(out['baz/QuxModel'].weightSpecsBytes)
|
.toEqual(saveResult.modelArtifactsInfo.weightSpecsBytes);
|
expect(out['baz/QuxModel'].weightDataBytes)
|
.toEqual(saveResult.modelArtifactsInfo.weightDataBytes);
|
}));
|
it('Manager: List models: 2 results', runWithLock(async () => {
|
// First, save a model.
|
const handler1 = tf.io.getSaveHandlers('localstorage://QuxModel')[0];
|
const saveResult1 = await handler1.save(artifacts1);
|
// Then, save the model under another path.
|
const handler2 = tf.io.getSaveHandlers('localstorage://repeat/QuxModel')[0];
|
const saveResult2 = await handler2.save(artifacts1);
|
// After successful saving, there should be two models.
|
const out = await new BrowserLocalStorageManager().listModels();
|
if (Object.keys(out).length !== 2) {
|
console.log(JSON.stringify(out, null, 2));
|
}
|
expect(Object.keys(out).length).toEqual(2);
|
expect(out['QuxModel'].modelTopologyType)
|
.toEqual(saveResult1.modelArtifactsInfo.modelTopologyType);
|
expect(out['QuxModel'].modelTopologyBytes)
|
.toEqual(saveResult1.modelArtifactsInfo.modelTopologyBytes);
|
expect(out['QuxModel'].weightSpecsBytes)
|
.toEqual(saveResult1.modelArtifactsInfo.weightSpecsBytes);
|
expect(out['QuxModel'].weightDataBytes)
|
.toEqual(saveResult1.modelArtifactsInfo.weightDataBytes);
|
expect(out['repeat/QuxModel'].modelTopologyType)
|
.toEqual(saveResult2.modelArtifactsInfo.modelTopologyType);
|
expect(out['repeat/QuxModel'].modelTopologyBytes)
|
.toEqual(saveResult2.modelArtifactsInfo.modelTopologyBytes);
|
expect(out['repeat/QuxModel'].weightSpecsBytes)
|
.toEqual(saveResult2.modelArtifactsInfo.weightSpecsBytes);
|
expect(out['repeat/QuxModel'].weightDataBytes)
|
.toEqual(saveResult2.modelArtifactsInfo.weightDataBytes);
|
}));
|
it('Manager: Successful deleteModel', runWithLock(async () => {
|
// First, save a model.
|
const handler1 = tf.io.getSaveHandlers('localstorage://QuxModel')[0];
|
await handler1.save(artifacts1);
|
// Then, save the model under another path.
|
const handler2 = tf.io.getSaveHandlers('localstorage://repeat/QuxModel')[0];
|
await handler2.save(artifacts1);
|
// After successful saving, delete the first save, and then
|
// `listModel` should give only one result.
|
const manager = new BrowserLocalStorageManager();
|
await manager.removeModel('QuxModel');
|
const out = await manager.listModels();
|
expect(Object.keys(out)).toEqual(['repeat/QuxModel']);
|
}));
|
});
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"local_storage_test.js","sourceRoot":"","sources":["../../../../../../tfjs-core/src/io/local_storage_test.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2018 Google LLC. All Rights Reserved.\n * Licensed under the Apache License, Version 2.0 (the 'License');\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an 'AS IS' BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n * =============================================================================\n */\n\nimport * as tf from '../index';\nimport {BROWSER_ENVS, describeWithFlags, runWithLock} from '../jasmine_util';\nimport {arrayBufferToBase64String, base64StringToArrayBuffer} from './io_utils';\nimport {browserLocalStorage, BrowserLocalStorage, BrowserLocalStorageManager, localStorageRouter, purgeLocalStorageArtifacts} from './local_storage';\n\ndescribeWithFlags('LocalStorage', BROWSER_ENVS, () => {\n  // Test data.\n  const modelTopology1: {} = {\n    'class_name': 'Sequential',\n    'keras_version': '2.1.4',\n    'config': [{\n      'class_name': 'Dense',\n      'config': {\n        'kernel_initializer': {\n          'class_name': 'VarianceScaling',\n          'config': {\n            'distribution': 'uniform',\n            'scale': 1.0,\n            'seed': null,\n            'mode': 'fan_avg'\n          }\n        },\n        'name': 'dense',\n        'kernel_constraint': null,\n        'bias_regularizer': null,\n        'bias_constraint': null,\n        'dtype': 'float32',\n        'activation': 'linear',\n        'trainable': true,\n        'kernel_regularizer': null,\n        'bias_initializer': {'class_name': 'Zeros', 'config': {}},\n        'units': 1,\n        'batch_input_shape': [null, 3],\n        'use_bias': true,\n        'activity_regularizer': null\n      }\n    }],\n    'backend': 'tensorflow'\n  };\n  const weightSpecs1: tf.io.WeightsManifestEntry[] = [\n    {\n      name: 'dense/kernel',\n      shape: [3, 1],\n      dtype: 'float32',\n    },\n    {\n      name: 'dense/bias',\n      shape: [1],\n      dtype: 'float32',\n    }\n  ];\n  const weightData1 = new ArrayBuffer(16);\n  const trainingConfig1: tf.io.TrainingConfig = {\n    loss: 'categorical_crossentropy',\n    metrics: ['accuracy'],\n    optimizer_config: {class_name: 'SGD', config: {learningRate: 0.1}}\n  };\n\n  const artifacts1: tf.io.ModelArtifacts = {\n    modelTopology: modelTopology1,\n    weightSpecs: weightSpecs1,\n    weightData: weightData1,\n    format: 'layers-model',\n    generatedBy: 'TensorFlow.js v0.0.0',\n    convertedBy: '1.13.1',\n    signature: null,\n    userDefinedMetadata: {},\n    modelInitializer: {},\n    initializerSignature: null,\n    trainingConfig: trainingConfig1,\n  };\n\n  const artifactsV0: tf.io.ModelArtifacts = {\n    modelTopology: modelTopology1,\n    weightSpecs: weightSpecs1,\n    weightData: weightData1\n  };\n\n  function findOverflowingByteSize(): number {\n    const LS = window.localStorage;\n    const probeKey =\n        `tfjs_test_probe_values_${new Date().getTime()}_${Math.random()}`;\n    const minKilobytes = 200;\n    const stepKilobytes = 200;\n    const maxKilobytes = 40000;\n    for (let kilobytes = minKilobytes; kilobytes < maxKilobytes;\n         kilobytes += stepKilobytes) {\n      const bytes = kilobytes * 1024;\n      const data = new ArrayBuffer(bytes);\n      try {\n        const encoded = arrayBufferToBase64String(data);\n        LS.setItem(probeKey, encoded);\n      } catch (err) {\n        return bytes;\n      }\n      LS.removeItem(probeKey);\n    }\n    throw new Error(\n        `Unable to determined overflowing byte size up to ${maxKilobytes} kB.`);\n  }\n\n  beforeEach(() => {\n    purgeLocalStorageArtifacts();\n  });\n\n  afterEach(() => {\n    purgeLocalStorageArtifacts();\n  });\n\n  it('Save artifacts succeeds', runWithLock(async () => {\n       const testStartDate = new Date();\n       const handler = tf.io.getSaveHandlers('localstorage://foo/FooModel')[0];\n       const saveResult = await handler.save(artifacts1);\n\n       expect(saveResult.modelArtifactsInfo.dateSaved.getTime())\n           .toBeGreaterThanOrEqual(testStartDate.getTime());\n       // Note: The following two assertions work only because there is no\n       //   non-ASCII characters in `modelTopology1` and `weightSpecs1`.\n       expect(saveResult.modelArtifactsInfo.modelTopologyBytes)\n           .toEqual(JSON.stringify(modelTopology1).length);\n       expect(saveResult.modelArtifactsInfo.weightSpecsBytes)\n           .toEqual(JSON.stringify(weightSpecs1).length);\n       expect(saveResult.modelArtifactsInfo.weightDataBytes).toEqual(16);\n\n       // Check the content of the saved items in local storage.\n       const LS = window.localStorage;\n       const info =\n           JSON.parse(LS.getItem('tensorflowjs_models/foo/FooModel/info'));\n       expect(Date.parse(info.dateSaved))\n           .toEqual(saveResult.modelArtifactsInfo.dateSaved.getTime());\n       expect(info.modelTopologyBytes)\n           .toEqual(saveResult.modelArtifactsInfo.modelTopologyBytes);\n       expect(info.weightSpecsBytes)\n           .toEqual(saveResult.modelArtifactsInfo.weightSpecsBytes);\n       expect(info.weightDataBytes)\n           .toEqual(saveResult.modelArtifactsInfo.weightDataBytes);\n\n       const topologyString =\n           LS.getItem('tensorflowjs_models/foo/FooModel/model_topology');\n       expect(JSON.stringify(modelTopology1)).toEqual(topologyString);\n\n       const weightSpecsString =\n           LS.getItem('tensorflowjs_models/foo/FooModel/weight_specs');\n       expect(JSON.stringify(weightSpecs1)).toEqual(weightSpecsString);\n\n       const weightDataBase64String =\n           LS.getItem('tensorflowjs_models/foo/FooModel/weight_data');\n       expect(base64StringToArrayBuffer(weightDataBase64String))\n           .toEqual(weightData1);\n     }));\n\n  it('Save-load round trip succeeds', runWithLock(async () => {\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n\n       await handler1.save(artifacts1);\n       const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];\n       const loaded = await handler2.load();\n       expect(loaded.modelTopology).toEqual(modelTopology1);\n       expect(loaded.weightSpecs).toEqual(weightSpecs1);\n       expect(loaded.weightData).toEqual(weightData1);\n       expect(loaded.format).toEqual('layers-model');\n       expect(loaded.generatedBy).toEqual('TensorFlow.js v0.0.0');\n       expect(loaded.convertedBy).toEqual('1.13.1');\n       expect(loaded.userDefinedMetadata).toEqual({});\n       expect(loaded.modelInitializer).toEqual({});\n       expect(loaded.initializerSignature).toBeUndefined();\n       expect(loaded.trainingConfig).toEqual(trainingConfig1);\n     }));\n\n  it('Save-load round trip succeeds: v0 format', runWithLock(async () => {\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n\n       await handler1.save(artifactsV0);\n       const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];\n       const loaded = await handler2.load();\n       expect(loaded.modelTopology).toEqual(modelTopology1);\n       expect(loaded.weightSpecs).toEqual(weightSpecs1);\n       expect(loaded.weightData).toEqual(weightData1);\n       expect(loaded.format).toBeUndefined();\n       expect(loaded.generatedBy).toBeUndefined();\n       expect(loaded.convertedBy).toBeUndefined();\n       expect(loaded.userDefinedMetadata).toBeUndefined();\n       expect(loaded.trainingConfig).toBeUndefined();\n       expect(loaded.modelInitializer).toBeUndefined();\n       expect(loaded.initializerSignature).toBeUndefined();\n       expect(loaded.trainingConfig).toBeUndefined();\n     }));\n\n  it('Loading nonexistent model fails.', runWithLock(async () => {\n       const handler =\n           tf.io.getSaveHandlers('localstorage://NonexistentModel')[0];\n       try {\n         await handler.load();\n       } catch (err) {\n         expect(err.message)\n             .toEqual(\n                 'In local storage, there is no model with name ' +\n                 '\\'NonexistentModel\\'');\n         return;  // Success\n       }\n       fail('Loading nonexistent model succeeded unexpectedly.');\n     }));\n\n  it('Loading model with missing topology fails.', runWithLock(async () => {\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n       await handler1.save(artifacts1);\n       // Manually remove the topology item from local storage.\n       window.localStorage.removeItem(\n           'tensorflowjs_models/FooModel/model_topology');\n\n       const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];\n       try {\n         await handler2.load();\n       } catch (err) {\n         expect(err.message)\n             .toEqual(\n                 'In local storage, the topology of model ' +\n                 '\\'FooModel\\' is missing.');\n         return;  // Success\n       }\n       fail('Loading of model with missing topology succeeded unexpectedly.');\n     }));\n\n  it('Loading model with missing weight specs fails.', runWithLock(async () => {\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n       await handler1.save(artifacts1);\n       // Manually remove the weight specs item from local storage.\n       window.localStorage.removeItem(\n           'tensorflowjs_models/FooModel/weight_specs');\n\n       const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];\n       try {\n         await handler2.load();\n       } catch (err) {\n         expect(err.message)\n             .toEqual(\n                 'In local storage, the weight specs of model ' +\n                 '\\'FooModel\\' are missing.');\n         return;  // Success\n       }\n       fail(\n           'Loading of model with missing weight specs ' +\n           'succeeded unexpectedly.');\n     }));\n\n  it('Loading model with missing weight data fails.', runWithLock(async () => {\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n       await handler1.save(artifacts1);\n\n       // Manually remove the weight data item from local storage.\n       window.localStorage.removeItem(\n           'tensorflowjs_models/FooModel/weight_data');\n\n       const handler2 = tf.io.getLoadHandlers('localstorage://FooModel')[0];\n       try {\n         await handler2.load();\n         fail(\n             'Loading of model with missing weight data ' +\n             'succeeded unexpectedly.');\n       } catch (err) {\n         expect(err.message)\n             .toEqual(\n                 'In local storage, the binary weight values of model ' +\n                 '\\'FooModel\\' are missing.');\n       }\n     }));\n\n  it('Data size too large leads to error thrown', runWithLock(async () => {\n       const overflowByteSize = findOverflowingByteSize();\n       const overflowArtifacts: tf.io.ModelArtifacts = {\n         modelTopology: modelTopology1,\n         weightSpecs: weightSpecs1,\n         weightData: new ArrayBuffer(overflowByteSize),\n       };\n       const handler1 = tf.io.getSaveHandlers('localstorage://FooModel')[0];\n       try {\n         await handler1.save(overflowArtifacts);\n         fail(\n             'Saving of model of overflowing-size weight data succeeded ' +\n             'unexpectedly.');\n       } catch (err) {\n         expect(\n             (err.message as string)\n                 .indexOf('Failed to save model \\'FooModel\\' to local storage'))\n             .toEqual(0);\n       }\n     }));\n\n  it('Null, undefined or empty modelPath throws Error', () => {\n    expect(() => browserLocalStorage(null))\n        .toThrowError(\n            /local storage, modelPath must not be null, undefined or empty/);\n    expect(() => browserLocalStorage(undefined))\n        .toThrowError(\n            /local storage, modelPath must not be null, undefined or empty/);\n    expect(() => browserLocalStorage(''))\n        .toThrowError(\n            /local storage, modelPath must not be null, undefined or empty./);\n  });\n\n  it('router', () => {\n    expect(\n        localStorageRouter('localstorage://bar') instanceof BrowserLocalStorage)\n        .toEqual(true);\n    expect(localStorageRouter('indexeddb://bar')).toBeNull();\n    expect(localStorageRouter('qux')).toBeNull();\n  });\n\n  it('Manager: List models: 0 result', runWithLock(async () => {\n       // Before any model is saved, listModels should return empty result.\n       const out = await new BrowserLocalStorageManager().listModels();\n       expect(out).toEqual({});\n     }));\n\n  it('Manager: List models: 1 result', runWithLock(async () => {\n       const handler = tf.io.getSaveHandlers('localstorage://baz/QuxModel')[0];\n       const saveResult = await handler.save(artifacts1);\n\n       // After successful saving, there should be one model.\n       const out = await new BrowserLocalStorageManager().listModels();\n       if (Object.keys(out).length !== 1) {\n         console.log(JSON.stringify(out, null, 2));\n       }\n\n       expect(Object.keys(out).length).toEqual(1);\n       expect(out['baz/QuxModel'].modelTopologyType)\n           .toEqual(saveResult.modelArtifactsInfo.modelTopologyType);\n       expect(out['baz/QuxModel'].modelTopologyBytes)\n           .toEqual(saveResult.modelArtifactsInfo.modelTopologyBytes);\n       expect(out['baz/QuxModel'].weightSpecsBytes)\n           .toEqual(saveResult.modelArtifactsInfo.weightSpecsBytes);\n       expect(out['baz/QuxModel'].weightDataBytes)\n           .toEqual(saveResult.modelArtifactsInfo.weightDataBytes);\n     }));\n\n  it('Manager: List models: 2 results', runWithLock(async () => {\n       // First, save a model.\n       const handler1 = tf.io.getSaveHandlers('localstorage://QuxModel')[0];\n       const saveResult1 = await handler1.save(artifacts1);\n\n       // Then, save the model under another path.\n       const handler2 =\n           tf.io.getSaveHandlers('localstorage://repeat/QuxModel')[0];\n       const saveResult2 = await handler2.save(artifacts1);\n\n       // After successful saving, there should be two models.\n       const out = await new BrowserLocalStorageManager().listModels();\n       if (Object.keys(out).length !== 2) {\n         console.log(JSON.stringify(out, null, 2));\n       }\n       expect(Object.keys(out).length).toEqual(2);\n       expect(out['QuxModel'].modelTopologyType)\n           .toEqual(saveResult1.modelArtifactsInfo.modelTopologyType);\n       expect(out['QuxModel'].modelTopologyBytes)\n           .toEqual(saveResult1.modelArtifactsInfo.modelTopologyBytes);\n       expect(out['QuxModel'].weightSpecsBytes)\n           .toEqual(saveResult1.modelArtifactsInfo.weightSpecsBytes);\n       expect(out['QuxModel'].weightDataBytes)\n           .toEqual(saveResult1.modelArtifactsInfo.weightDataBytes);\n       expect(out['repeat/QuxModel'].modelTopologyType)\n           .toEqual(saveResult2.modelArtifactsInfo.modelTopologyType);\n       expect(out['repeat/QuxModel'].modelTopologyBytes)\n           .toEqual(saveResult2.modelArtifactsInfo.modelTopologyBytes);\n       expect(out['repeat/QuxModel'].weightSpecsBytes)\n           .toEqual(saveResult2.modelArtifactsInfo.weightSpecsBytes);\n       expect(out['repeat/QuxModel'].weightDataBytes)\n           .toEqual(saveResult2.modelArtifactsInfo.weightDataBytes);\n     }));\n\n  it('Manager: Successful deleteModel', runWithLock(async () => {\n       // First, save a model.\n       const handler1 = tf.io.getSaveHandlers('localstorage://QuxModel')[0];\n       await handler1.save(artifacts1);\n\n       // Then, save the model under another path.\n       const handler2 =\n           tf.io.getSaveHandlers('localstorage://repeat/QuxModel')[0];\n       await handler2.save(artifacts1);\n\n       // After successful saving, delete the first save, and then\n       // `listModel` should give only one result.\n       const manager = new BrowserLocalStorageManager();\n       await manager.removeModel('QuxModel');\n       const out = await manager.listModels();\n       expect(Object.keys(out)).toEqual(['repeat/QuxModel']);\n     }));\n});\n"]}
|