1 | /**
|
2 | * @license
|
3 | * Copyright 2018 Google LLC
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style
|
6 | * license that can be found in the LICENSE file or at
|
7 | * https://opensource.org/licenses/MIT.
|
8 | * =============================================================================
|
9 | */
|
10 | /* Original Source: engine/training.py */
|
11 | import * as tfc from '@tensorflow/tfjs-core';
|
12 | import { io, Optimizer, scalar, serialization, Tensor, tensor1d, util } from '@tensorflow/tfjs-core';
|
13 | import * as K from '../backend/tfjs_backend';
|
14 | import { nameScope } from '../common';
|
15 | import { NotImplementedError, RuntimeError, ValueError } from '../errors';
|
16 | import { deserialize } from '../layers/serialization';
|
17 | import * as losses from '../losses';
|
18 | import * as Metrics from '../metrics';
|
19 | import * as optimizers from '../optimizers';
|
20 | import { checkUserDefinedMetadata } from '../user_defined_metadata';
|
21 | import { count, pyListRepeat, singletonOrArray, toCamelCase, toSnakeCase, unique } from '../utils/generic_utils';
|
22 | import { printSummary } from '../utils/layer_utils';
|
23 | import { range } from '../utils/math_utils';
|
24 | import { convertPythonicToTs } from '../utils/serialization_utils';
|
25 | import { version } from '../version';
|
26 | import { Container } from './container';
|
27 | import { execute, FeedDict } from './executor';
|
28 | import { evaluateDataset, fitDataset } from './training_dataset';
|
29 | import { checkBatchSize, disposeNewTensors, ensureTensorsRank2OrHigher, fitTensors, makeBatches, sliceArrays, sliceArraysByIndices } from './training_tensors';
|
30 | import { computeWeightedLoss, standardizeClassWeights, standardizeWeights } from './training_utils';
|
31 | /**
|
32 | * Helper function for polymorphic input data: 1. singleton Tensor.
|
33 | */
|
34 | export function isDataTensor(x) {
|
35 | return x instanceof Tensor;
|
36 | }
|
37 | /**
|
38 | * Helper function for polymorphic input data: 2. Array of Tensor.
|
39 | */
|
40 | export function isDataArray(x) {
|
41 | return Array.isArray(x);
|
42 | }
|
43 | /**
|
44 | * Helper function for polymorphic input data: 3. "dict" of Tensor.
|
45 | */
|
46 | export function isDataDict(x) {
|
47 | return !isDataTensor(x) && !isDataArray(x);
|
48 | }
|
49 | /**
|
50 | * Normalizes inputs and targets provided by users.
|
51 | * @param data User-provided input data (polymorphic).
|
52 | * @param names An Array of expected Tensor names.
|
53 | * @param shapes Optional Array of expected Tensor shapes.
|
54 | * @param checkBatchAxis Whether to check that the batch axis of the arrays
|
55 | * match the expected value found in `shapes`.
|
56 | * @param exceptionPrefix String prefix used for exception formatting.
|
57 | * @returns List of standardized input Tensors (one Tensor per model input).
|
58 | * @throws ValueError: in case of improperly formatted user data.
|
59 | */
|
60 | export function standardizeInputData(data, names, shapes, checkBatchAxis = true, exceptionPrefix = '') {
|
61 | if (names == null || names.length === 0) {
|
62 | // Check for the case where the model expected no data, but some data got
|
63 | // sent.
|
64 | if (data != null) {
|
65 | let gotUnexpectedData = false;
|
66 | if (isDataArray(data) && data.length > 0) {
|
67 | gotUnexpectedData = true;
|
68 | }
|
69 | else if (isDataDict(data)) {
|
70 | for (const key in data) {
|
71 | if (data.hasOwnProperty(key)) {
|
72 | gotUnexpectedData = true;
|
73 | break;
|
74 | }
|
75 | }
|
76 | }
|
77 | else {
|
78 | // `data` is a singleton Tensor in this case.
|
79 | gotUnexpectedData = true;
|
80 | }
|
81 | if (gotUnexpectedData) {
|
82 | throw new ValueError(`Error when checking model ${exceptionPrefix} expected no data, ` +
|
83 | `but got ${data}`);
|
84 | }
|
85 | }
|
86 | return [];
|
87 | }
|
88 | if (data == null) {
|
89 | return names.map(name => null);
|
90 | }
|
91 | let arrays;
|
92 | if (isDataDict(data)) {
|
93 | data = data;
|
94 | arrays = [];
|
95 | for (const name of names) {
|
96 | if (data[name] == null) {
|
97 | throw new ValueError(`No data provided for "${name}". Need data for each key in: ` +
|
98 | `${names}`);
|
99 | }
|
100 | arrays.push(data[name]);
|
101 | }
|
102 | }
|
103 | else if (isDataArray(data)) {
|
104 | data = data;
|
105 | if (data.length !== names.length) {
|
106 | throw new ValueError(`Error when checking model ${exceptionPrefix}: the Array of ` +
|
107 | `Tensors that you are passing to your model is not the size the ` +
|
108 | `model expected. Expected to see ${names.length} Tensor(s), but ` +
|
109 | `instead got the following list of Tensor(s): ${data}`);
|
110 | }
|
111 | arrays = data;
|
112 | }
|
113 | else {
|
114 | data = data;
|
115 | if (names.length > 1) {
|
116 | throw new ValueError(`The model ${exceptionPrefix} expects ${names.length} Tensor(s), ` +
|
117 | `but only received one Tensor. Found: Tensor with shape ${data.shape}`);
|
118 | }
|
119 | arrays = [data];
|
120 | }
|
121 | arrays = ensureTensorsRank2OrHigher(arrays);
|
122 | // Check shape compatibility.
|
123 | if (shapes != null) {
|
124 | for (let i = 0; i < names.length; ++i) {
|
125 | if (shapes[i] == null) {
|
126 | continue;
|
127 | }
|
128 | const array = arrays[i];
|
129 | if (array.shape.length !== shapes[i].length) {
|
130 | throw new ValueError(`Error when checking ${exceptionPrefix}: expected ${names[i]} ` +
|
131 | `to have ${shapes[i].length} dimension(s). but got array with ` +
|
132 | `shape ${array.shape}`);
|
133 | }
|
134 | for (let j = 0; j < shapes[i].length; ++j) {
|
135 | if (j === 0 && !checkBatchAxis) {
|
136 | // Skip the first (batch) axis.
|
137 | continue;
|
138 | }
|
139 | const dim = array.shape[j];
|
140 | const refDim = shapes[i][j];
|
141 | if (refDim != null && refDim >= 0 && dim !== refDim) {
|
142 | throw new ValueError(`${exceptionPrefix} expected a batch of elements where each ` +
|
143 | `example has shape [${shapes[i].slice(1, shapes[i].length)}] ` +
|
144 | `(i.e.,tensor shape [*,${shapes[i].slice(1, shapes[i].length)}])` +
|
145 | ` but the ${exceptionPrefix} received an input with ${array.shape[0]}` +
|
146 | ` examples, each with shape [${array.shape.slice(1, array.shape.length)}]` +
|
147 | ` (tensor shape [${array.shape}])`);
|
148 | }
|
149 | }
|
150 | }
|
151 | }
|
152 | return arrays;
|
153 | }
|
154 | /**
|
155 | * User input validation for Tensors.
|
156 | * @param inputs `Array` of `tf.Tensor`s for inputs.
|
157 | * @param targets `Array` of `tf.Tensor`s for targets.
|
158 | * @param weights Optional `Array` of `tf.Tensor`s for sample weights.
|
159 | * @throws ValueError: in case of incorrectly formatted data.
|
160 | */
|
161 | export function checkArrayLengths(inputs, targets, weights) {
|
162 | const setX = unique(inputs.map(input => input.shape[0]));
|
163 | setX.sort();
|
164 | const setY = unique(targets.map(target => target.shape[0]));
|
165 | setY.sort();
|
166 | // TODO(cais): Check `weights` as well.
|
167 | if (setX.length > 1) {
|
168 | throw new ValueError(`All input Tensors (x) should have the same number of samples. ` +
|
169 | `Got array shapes: ` +
|
170 | `${JSON.stringify(inputs.map(input => input.shape))}`);
|
171 | }
|
172 | if (setY.length > 1) {
|
173 | throw new ValueError(`All target Tensors (y) should have the same number of samples. ` +
|
174 | `Got array shapes: ` +
|
175 | `${JSON.stringify(targets.map(target => target.shape))}`);
|
176 | }
|
177 | if (setX.length > 0 && setY.length > 0 && !util.arraysEqual(setX, setY)) {
|
178 | throw new ValueError(`Input Tensors should have the same number of samples as target ` +
|
179 | `Tensors. Found ${setX[0]} input sample(s) and ${setY[0]} target ` +
|
180 | `sample(s).`);
|
181 | }
|
182 | }
|
183 | /**
|
184 | * Validation on the compatibility of targes and loss functions.
|
185 | *
|
186 | * This helps prevent users from using loss functions incorrectly.
|
187 | *
|
188 | * @param targets `Array` of `tf.Tensor`s of targets.
|
189 | * @param lossFns `Array` of loss functions.
|
190 | * @param outputShapes `Array` of shapes of model outputs.
|
191 | */
|
192 | function checkLossAndTargetCompatibility(targets, lossFns, outputShapes) {
|
193 | // TODO(cais): Dedicated test coverage?
|
194 | const keyLosses = [
|
195 | losses.meanSquaredError, losses.binaryCrossentropy,
|
196 | losses.categoricalCrossentropy
|
197 | ];
|
198 | for (let i = 0; i < targets.length; ++i) {
|
199 | const y = targets[i];
|
200 | const loss = lossFns[i];
|
201 | const shape = outputShapes[i];
|
202 | if (loss == null) {
|
203 | continue;
|
204 | }
|
205 | if (loss === losses.categoricalCrossentropy) {
|
206 | if (y.shape[y.shape.length - 1] === 1) {
|
207 | throw new ValueError(`You are passing a target array of shape ${y.shape} while using ` +
|
208 | `a loss 'categorical_crossentropy'. 'categorical_crossentropy'` +
|
209 | `expects targets to be binary matrices (1s and 0s) of shape ` +
|
210 | `[samples, classes].`);
|
211 | // TODO(cais): Example code in error message.
|
212 | }
|
213 | }
|
214 | if (keyLosses.indexOf(loss) !== -1) {
|
215 | const slicedYShape = y.shape.slice(1);
|
216 | const slicedShape = shape.slice(1);
|
217 | for (let j = 0; j < slicedYShape.length; ++j) {
|
218 | const targetDim = slicedYShape[j];
|
219 | const outDim = slicedShape[j];
|
220 | if (outDim != null && targetDim !== outDim) {
|
221 | throw new ValueError(`A target Tensor with shape ${y.shape} was passed for an ` +
|
222 | `output of shape ${shape}, while using a loss function that ` +
|
223 | `expects targets to have the same shape as the output.`);
|
224 | }
|
225 | }
|
226 | }
|
227 | }
|
228 | }
|
229 | /**
|
230 | * Check inputs provided by the user.
|
231 | *
|
232 | * Porting Note: This corresponds to _standardize_input_data() in Python
|
233 | * Keras. Because of the strong typing in TF.js, we do not need to convert
|
234 | * the data. Specifically:
|
235 | * 1) in PyKeras, `data` can be `DataFrame` instances from pandas, for
|
236 | * example. We don't need to worry about that here because there is no
|
237 | * widely popular javascript/typesdcript equivalent of pandas (so far).
|
238 | * If one becomes available in the future, we can add support.
|
239 | * 2) in PyKeras, inputs can be Python dict. But here we are stipulating
|
240 | * that the data is either a single `tf.Tensor` or an Array of `tf.Tensor`s. We
|
241 | * may add support for `Object` data inputs in the future when the need
|
242 | * arises.
|
243 | *
|
244 | * Instead, we perform basic checks for number of parameters and shapes.
|
245 | *
|
246 | * @param data: The input data.
|
247 | * @param names: Name for the inputs, from the model.
|
248 | * @param shapes: Expected shapes for the input data, from the model.
|
249 | * @param checkBatchAxis: Whether the size along the batch axis (i.e., the
|
250 | * first dimension) will be checked for matching.
|
251 | * @param exceptionPrefix: Execption prefix message, used in generating error
|
252 | * messages.
|
253 | * @throws ValueError: on incorrect number of inputs or mismatches in shapes.
|
254 | */
|
255 | function checkInputData(data, names, shapes, checkBatchAxis = true, exceptionPrefix = '') {
|
256 | let arrays;
|
257 | if (Array.isArray(data)) {
|
258 | if (data.length !== names.length) {
|
259 | throw new ValueError(`Error when checking model ${exceptionPrefix}: the Array of ` +
|
260 | `Tensors that you are passing to your model is not the size the ` +
|
261 | `the model expected. Expected to see ${names.length} Tensor(s),` +
|
262 | ` but instead got ${data.length} Tensors(s).`);
|
263 | }
|
264 | arrays = data;
|
265 | }
|
266 | else {
|
267 | if (names.length > 1) {
|
268 | throw new ValueError(`The model expects ${names.length} ${exceptionPrefix} Tensors, ` +
|
269 | `but only received one Tensor. Found: array with shape ` +
|
270 | `${JSON.stringify(data.shape)}.`);
|
271 | }
|
272 | arrays = [data];
|
273 | }
|
274 | if (shapes != null) {
|
275 | for (let i = 0; i < names.length; ++i) {
|
276 | if (shapes[i] == null) {
|
277 | continue;
|
278 | }
|
279 | const array = arrays[i];
|
280 | if (array.shape.length !== shapes[i].length) {
|
281 | throw new ValueError(`Error when checking ${exceptionPrefix}: expected ${names[i]} ` +
|
282 | `to have ${shapes[i].length} dimension(s), but got array with ` +
|
283 | `shape ${JSON.stringify(array.shape)}`);
|
284 | }
|
285 | for (let j = 0; j < shapes[i].length; ++j) {
|
286 | if (j === 0 && !checkBatchAxis) {
|
287 | continue;
|
288 | }
|
289 | const dim = array.shape[j];
|
290 | const refDim = shapes[i][j];
|
291 | if (refDim != null) {
|
292 | if (refDim !== dim) {
|
293 | throw new ValueError(`Error when checking ${exceptionPrefix}: expected ` +
|
294 | `${names[i]} to have shape ${JSON.stringify(shapes[i])} but ` +
|
295 | `got array with shape ${JSON.stringify(array.shape)}.`);
|
296 | }
|
297 | }
|
298 | }
|
299 | }
|
300 | }
|
301 | }
|
302 | /**
|
303 | * Maps metric functions to model outputs.
|
304 | * @param metrics An shortcut strings name, metric function, `Array` or dict
|
305 | * (`Object`) of metric functions.
|
306 | * @param outputNames An `Array` of the names of model outputs.
|
307 | * @returns An `Array` (one entry per model output) of `Array` of metric
|
308 | * functions. For instance, if the model has 2 outputs, and for the first
|
309 | * output we want to compute `binaryAccuracy` and `binaryCrossentropy`,
|
310 | * and just `binaryAccuracy` for the second output, the `Array` would look
|
311 | * like:
|
312 | * `[[binaryAccuracy, binaryCrossentropy], [binaryAccuracy]]`
|
313 | * @throws TypeError: incompatible metrics format.
|
314 | */
|
315 | export function collectMetrics(metrics, outputNames) {
|
316 | if (metrics == null || Array.isArray(metrics) && metrics.length === 0) {
|
317 | return outputNames.map(name => []);
|
318 | }
|
319 | let wrappedMetrics;
|
320 | if (typeof metrics === 'string' || typeof metrics === 'function') {
|
321 | wrappedMetrics = [metrics];
|
322 | }
|
323 | else if (Array.isArray(metrics) || typeof metrics === 'object') {
|
324 | wrappedMetrics = metrics;
|
325 | }
|
326 | else {
|
327 | throw new TypeError('Type of metrics argument not understood. Expected an string,' +
|
328 | `function, Array, or Object, found: ${metrics}`);
|
329 | }
|
330 | if (Array.isArray(wrappedMetrics)) {
|
331 | // We then apply all metrics to all outputs.
|
332 | return outputNames.map(name => wrappedMetrics);
|
333 | }
|
334 | else {
|
335 | // In this case, metrics is a dict.
|
336 | const nestedMetrics = [];
|
337 | for (const name of outputNames) {
|
338 | let outputMetrics = wrappedMetrics.hasOwnProperty(name) ? wrappedMetrics[name] : [];
|
339 | if (!Array.isArray(outputMetrics)) {
|
340 | outputMetrics = [outputMetrics];
|
341 | }
|
342 | nestedMetrics.push(outputMetrics);
|
343 | }
|
344 | return nestedMetrics;
|
345 | }
|
346 | }
|
347 | const LAYERS_MODEL_FORMAT_NAME = 'layers-model';
|
348 | /**
|
349 | * A `tf.LayersModel` is a directed, acyclic graph of `tf.Layer`s plus methods
|
350 | * for training, evaluation, prediction and saving.
|
351 | *
|
352 | * `tf.LayersModel` is the basic unit of training, inference and evaluation in
|
353 | * TensorFlow.js. To create a `tf.LayersModel`, use `tf.LayersModel`.
|
354 | *
|
355 | * See also:
|
356 | * `tf.Sequential`, `tf.loadLayersModel`.
|
357 | *
|
358 | * @doc {heading: 'Models', subheading: 'Classes'}
|
359 | */
|
360 | export class LayersModel extends Container {
|
361 | constructor(args) {
|
362 | super(args);
|
363 | this.isTraining = false;
|
364 | }
|
365 | /**
|
366 | * Print a text summary of the model's layers.
|
367 | *
|
368 | * The summary includes
|
369 | * - Name and type of all layers that comprise the model.
|
370 | * - Output shape(s) of the layers
|
371 | * - Number of weight parameters of each layer
|
372 | * - If the model has non-sequential-like topology, the inputs each layer
|
373 | * receives
|
374 | * - The total number of trainable and non-trainable parameters of the model.
|
375 | *
|
376 | * ```js
|
377 | * const input1 = tf.input({shape: [10]});
|
378 | * const input2 = tf.input({shape: [20]});
|
379 | * const dense1 = tf.layers.dense({units: 4}).apply(input1);
|
380 | * const dense2 = tf.layers.dense({units: 8}).apply(input2);
|
381 | * const concat = tf.layers.concatenate().apply([dense1, dense2]);
|
382 | * const output =
|
383 | * tf.layers.dense({units: 3, activation: 'softmax'}).apply(concat);
|
384 | *
|
385 | * const model = tf.model({inputs: [input1, input2], outputs: output});
|
386 | * model.summary();
|
387 | * ```
|
388 | *
|
389 | * @param lineLength Custom line length, in number of characters.
|
390 | * @param positions Custom widths of each of the columns, as either
|
391 | * fractions of `lineLength` (e.g., `[0.5, 0.75, 1]`) or absolute number
|
392 | * of characters (e.g., `[30, 50, 65]`). Each number corresponds to
|
393 | * right-most (i.e., ending) position of a column.
|
394 | * @param printFn Custom print function. Can be used to replace the default
|
395 | * `console.log`. For example, you can use `x => {}` to mute the printed
|
396 | * messages in the console.
|
397 | *
|
398 | * @doc {heading: 'Models', subheading: 'Classes'}
|
399 | */
|
400 | summary(lineLength, positions, printFn = console.log) {
|
401 | if (!this.built) {
|
402 | throw new ValueError(`This model has never been called, thus its weights have not been ` +
|
403 | `created yet. So no summary can be displayed. Build the model ` +
|
404 | `first (e.g., by calling it on some test data).`);
|
405 | }
|
406 | printSummary(this, lineLength, positions, printFn);
|
407 | }
|
408 | /**
|
409 | * Configures and prepares the model for training and evaluation. Compiling
|
410 | * outfits the model with an optimizer, loss, and/or metrics. Calling `fit`
|
411 | * or `evaluate` on an un-compiled model will throw an error.
|
412 | *
|
413 | * @param args a `ModelCompileArgs` specifying the loss, optimizer, and
|
414 | * metrics to be used for fitting and evaluating this model.
|
415 | *
|
416 | * @doc {heading: 'Models', subheading: 'Classes'}
|
417 | */
|
418 | compile(args) {
|
419 | if (args.loss == null) {
|
420 | args.loss = [];
|
421 | }
|
422 | this.loss = args.loss;
|
423 | if (typeof args.optimizer === 'string') {
|
424 | this.optimizer_ = optimizers.getOptimizer(args.optimizer);
|
425 | this.isOptimizerOwned = true;
|
426 | }
|
427 | else {
|
428 | if (!(args.optimizer instanceof Optimizer)) {
|
429 | throw new ValueError(`User-defined optimizer must be an instance of tf.Optimizer.`);
|
430 | }
|
431 | this.optimizer_ = args.optimizer;
|
432 | this.isOptimizerOwned = false;
|
433 | }
|
434 | // TODO(cais): Add lossWeights.
|
435 | // TODO(cais): Add sampleWeightMode.
|
436 | // Prepare loss functions.
|
437 | let lossFunctions = [];
|
438 | if (!Array.isArray(args.loss) && typeof args.loss !== 'string' &&
|
439 | typeof args.loss !== 'function') {
|
440 | args.loss = args.loss;
|
441 | for (const name in args.loss) {
|
442 | if (this.outputNames.indexOf(name) === -1) {
|
443 | throw new ValueError(`Unknown entry in loss dictionary: "${name}". ` +
|
444 | `Only expected the following keys: ${this.outputNames}`);
|
445 | }
|
446 | }
|
447 | for (const name of this.outputNames) {
|
448 | if (args.loss[name] == null) {
|
449 | console.warn(`Output "${name}" is missing from loss dictionary. We assume ` +
|
450 | `this was done on purpose, and we will not be expecting data ` +
|
451 | `to be passed to ${name} during training`);
|
452 | }
|
453 | lossFunctions.push(losses.get(args.loss[name]));
|
454 | }
|
455 | }
|
456 | else if (Array.isArray(args.loss)) {
|
457 | if (args.loss.length !== this.outputs.length) {
|
458 | throw new ValueError(`When passing an Array as loss, it should have one entry per ` +
|
459 | `model output. The model has ${this.outputs.length} output(s), ` +
|
460 | `but you passed loss=${args.loss}.`);
|
461 | }
|
462 | const theLosses = args.loss;
|
463 | lossFunctions = theLosses.map(l => losses.get(l));
|
464 | }
|
465 | else {
|
466 | const lossFunction = losses.get(args.loss);
|
467 | this.outputs.forEach(_ => {
|
468 | lossFunctions.push(lossFunction);
|
469 | });
|
470 | }
|
471 | this.lossFunctions = lossFunctions;
|
472 | this.feedOutputNames = [];
|
473 | this.feedOutputShapes = [];
|
474 | this.feedLossFns = [];
|
475 | for (let i = 0; i < this.outputs.length; ++i) {
|
476 | // TODO(cais): Logic for skipping target(s).
|
477 | const shape = this.internalOutputShapes[i];
|
478 | const name = this.outputNames[i];
|
479 | this.feedOutputNames.push(name);
|
480 | this.feedOutputShapes.push(shape);
|
481 | this.feedLossFns.push(this.lossFunctions[i]);
|
482 | }
|
483 | // TODO(cais): Add logic for output masks.
|
484 | // TODO(cais): Add logic for sample weights.
|
485 | const skipTargetIndices = [];
|
486 | // Prepare metrics.
|
487 | this.metrics = args.metrics;
|
488 | // TODO(cais): Add weightedMetrics.
|
489 | this.metricsNames = ['loss'];
|
490 | this.metricsTensors = [];
|
491 | // Compute total loss.
|
492 | // Porting Note: In PyKeras, metrics_tensors are symbolic tensor objects.
|
493 | // Here, metricsTensors are TypeScript functions. This difference is due
|
494 | // to the difference in symbolic/imperative property of the backends.
|
495 | nameScope('loss', () => {
|
496 | for (let i = 0; i < this.outputs.length; ++i) {
|
497 | if (skipTargetIndices.indexOf(i) !== -1) {
|
498 | continue;
|
499 | }
|
500 | // TODO(cais): Add weightedLoss, sampleWeight and mask.
|
501 | // The following line should be weightedLoss
|
502 | const weightedLoss = this.lossFunctions[i];
|
503 | if (this.outputs.length > 1) {
|
504 | this.metricsTensors.push([weightedLoss, i]);
|
505 | this.metricsNames.push(this.outputNames[i] + '_loss');
|
506 | }
|
507 | }
|
508 | // Porting Note: Due to the imperative nature of the backend, we calculate
|
509 | // the regularizer penalties in the totalLossFunction, instead of here.
|
510 | });
|
511 | const nestedMetrics = collectMetrics(args.metrics, this.outputNames);
|
512 | // TODO(cais): Add nestedWeightedMetrics.
|
513 | /**
|
514 | * Helper function used in loop below.
|
515 | */
|
516 | const appendMetric = (outputIndex, metricName, metricTensor) => {
|
517 | if (this.outputNames.length > 1) {
|
518 | metricName = this.outputNames[outputIndex] + '_' + metricName;
|
519 | }
|
520 | this.metricsNames.push(metricName);
|
521 | this.metricsTensors.push([metricTensor, outputIndex]);
|
522 | };
|
523 | nameScope('metric', () => {
|
524 | for (let i = 0; i < this.outputs.length; ++i) {
|
525 | if (skipTargetIndices.indexOf(i) !== -1) {
|
526 | continue;
|
527 | }
|
528 | const outputMetrics = nestedMetrics[i];
|
529 | // TODO(cais): Add weights and outputWeightedMetrics.
|
530 | // TODO(cais): Add optional arg `weights` to the following function.
|
531 | const handleMetrics = (metrics) => {
|
532 | const metricNamePrefix = '';
|
533 | let metricName;
|
534 | let accFn;
|
535 | let weightedMetricFn;
|
536 | // TODO(cais): Use 'weights_' for weighted metrics.
|
537 | for (const metric of metrics) {
|
538 | if (typeof metric === 'string' &&
|
539 | ['accuracy', 'acc', 'crossentropy', 'ce'].indexOf(metric) !==
|
540 | -1) {
|
541 | const outputShape = this.internalOutputShapes[i];
|
542 | if (outputShape[outputShape.length - 1] === 1 ||
|
543 | this.lossFunctions[i] === losses.binaryCrossentropy) {
|
544 | // case: binary accuracy/crossentropy.
|
545 | if (['accuracy', 'acc'].indexOf(metric) !== -1) {
|
546 | accFn = Metrics.binaryAccuracy;
|
547 | }
|
548 | else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {
|
549 | accFn = Metrics.binaryCrossentropy;
|
550 | }
|
551 | }
|
552 | else if (this.lossFunctions[i] ===
|
553 | losses.sparseCategoricalCrossentropy) {
|
554 | // case: categorical accuracy / crossentropy with sparse
|
555 | // targets.
|
556 | if (['accuracy', 'acc'].indexOf(metric) !== -1) {
|
557 | accFn = Metrics.sparseCategoricalAccuracy;
|
558 | }
|
559 | else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {
|
560 | accFn = Metrics.sparseCategoricalCrossentropy;
|
561 | }
|
562 | }
|
563 | else {
|
564 | // case: categorical accuracy / crossentropy.
|
565 | if (['accuracy', 'acc'].indexOf(metric) !== -1) {
|
566 | accFn = Metrics.categoricalAccuracy;
|
567 | }
|
568 | else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {
|
569 | accFn = Metrics.categoricalCrossentropy;
|
570 | }
|
571 | }
|
572 | let suffix;
|
573 | if (['accuracy', 'acc'].indexOf(metric) !== -1) {
|
574 | suffix = 'acc';
|
575 | }
|
576 | else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {
|
577 | suffix = 'ce';
|
578 | }
|
579 | // TODO(cais): Add weighting actually.
|
580 | weightedMetricFn = accFn;
|
581 | metricName = metricNamePrefix + suffix;
|
582 | }
|
583 | else {
|
584 | const metricFn = Metrics.get(metric);
|
585 | // TODO(cais): Add weighting actually.
|
586 | weightedMetricFn = metricFn;
|
587 | metricName =
|
588 | metricNamePrefix + Metrics.getLossOrMetricName(metric);
|
589 | }
|
590 | // TODO(cais): Add weighting and masking to metricResult.
|
591 | let metricResult;
|
592 | nameScope(metricName, () => {
|
593 | metricResult = weightedMetricFn;
|
594 | });
|
595 | appendMetric(i, metricName, metricResult);
|
596 | }
|
597 | };
|
598 | handleMetrics(outputMetrics);
|
599 | // TODO(cais): Call handleMetrics with weights.
|
600 | }
|
601 | });
|
602 | // Porting Notes: Given the imperative backend of tfjs-core,
|
603 | // there is no need for constructing the symbolic graph and placeholders.
|
604 | this.collectedTrainableWeights = this.trainableWeights;
|
605 | }
|
606 | /**
|
607 | * Check trainable weights count consistency.
|
608 | *
|
609 | * This will raise a warning if `this.trainableWeights` and
|
610 | * `this.collectedTrainableWeights` are inconsistent (i.e., have different
|
611 | * numbers of parameters).
|
612 | * Inconsistency will typically arise when one modifies `model.trainable`
|
613 | * without calling `model.compile()` again.
|
614 | */
|
615 | checkTrainableWeightsConsistency() {
|
616 | if (this.collectedTrainableWeights == null) {
|
617 | return;
|
618 | }
|
619 | if (this.trainableWeights.length !==
|
620 | this.collectedTrainableWeights.length) {
|
621 | console.warn('Discrepancy between trainableweights and collected trainable ' +
|
622 | 'weights. Did you set `model.trainable` without calling ' +
|
623 | '`model.compile()` afterwards?');
|
624 | }
|
625 | }
|
626 | /**
|
627 | * Returns the loss value & metrics values for the model in test mode.
|
628 | *
|
629 | * Loss and metrics are specified during `compile()`, which needs to happen
|
630 | * before calls to `evaluate()`.
|
631 | *
|
632 | * Computation is done in batches.
|
633 | *
|
634 | * ```js
|
635 | * const model = tf.sequential({
|
636 | * layers: [tf.layers.dense({units: 1, inputShape: [10]})]
|
637 | * });
|
638 | * model.compile({optimizer: 'sgd', loss: 'meanSquaredError'});
|
639 | * const result = model.evaluate(
|
640 | * tf.ones([8, 10]), tf.ones([8, 1]), {batchSize: 4});
|
641 | * result.print();
|
642 | * ```
|
643 | *
|
644 | * @param x `tf.Tensor` of test data, or an `Array` of `tf.Tensor`s if the
|
645 | * model has multiple inputs.
|
646 | * @param y `tf.Tensor` of target data, or an `Array` of `tf.Tensor`s if the
|
647 | * model has multiple outputs.
|
648 | * @param args A `ModelEvaluateArgs`, containing optional fields.
|
649 | *
|
650 | * @return `Scalar` test loss (if the model has a single output and no
|
651 | * metrics) or `Array` of `Scalar`s (if the model has multiple outputs
|
652 | * and/or metrics). The attribute `model.metricsNames`
|
653 | * will give you the display labels for the scalar outputs.
|
654 | *
|
655 | * @doc {heading: 'Models', subheading: 'Classes'}
|
656 | */
|
657 | evaluate(x, y, args = {}) {
|
658 | const batchSize = args.batchSize == null ? 32 : args.batchSize;
|
659 | checkBatchSize(batchSize);
|
660 | // TODO(cais): Standardize `config.sampleWeights` as well.
|
661 | // Validate user data.
|
662 | const checkBatchAxis = true;
|
663 | const standardizedOuts = this.standardizeUserDataXY(x, y, checkBatchAxis, batchSize);
|
664 | try {
|
665 | // TODO(cais): If uses `useLearningPhase`, set the corresponding element
|
666 | // of the input to 0.
|
667 | const ins = standardizedOuts[0].concat(standardizedOuts[1]);
|
668 | this.makeTestFunction();
|
669 | const f = this.testFunction;
|
670 | const testOuts = this.testLoop(f, ins, batchSize, args.verbose, args.steps);
|
671 | return singletonOrArray(testOuts);
|
672 | }
|
673 | finally {
|
674 | disposeNewTensors(standardizedOuts[0], x);
|
675 | disposeNewTensors(standardizedOuts[1], y);
|
676 | }
|
677 | }
|
678 | // TODO(cais): Add code snippet below once real dataset objects are
|
679 | // available.
|
680 | /**
|
681 | * Evaluate model using a dataset object.
|
682 | *
|
683 | * Note: Unlike `evaluate()`, this method is asynchronous (`async`);
|
684 | *
|
685 | * @param dataset A dataset object. Its `iterator()` method is expected
|
686 | * to generate a dataset iterator object, the `next()` method of which
|
687 | * is expected to produce data batches for evaluation. The return value
|
688 | * of the `next()` call ought to contain a boolean `done` field and a
|
689 | * `value` field. The `value` field is expected to be an array of two
|
690 | * `tf.Tensor`s or an array of two nested `tf.Tensor` structures. The former
|
691 | * case is for models with exactly one input and one output (e.g..
|
692 | * a sequential model). The latter case is for models with multiple
|
693 | * inputs and/or multiple outputs. Of the two items in the array, the
|
694 | * first is the input feature(s) and the second is the output target(s).
|
695 | * @param args A configuration object for the dataset-based evaluation.
|
696 | * @returns Loss and metric values as an Array of `Scalar` objects.
|
697 | *
|
698 | * @doc {heading: 'Models', subheading: 'Classes'}
|
699 | */
|
700 | async evaluateDataset(dataset, args) {
|
701 | this.makeTestFunction();
|
702 | return evaluateDataset(this, dataset, args);
|
703 | }
|
704 | /**
|
705 | * Get number of samples provided for training, evaluation or prediction.
|
706 | *
|
707 | * @param ins Input `tf.Tensor`.
|
708 | * @param batchSize Integer batch size, optional.
|
709 | * @param steps Total number of steps (batches of samples) before
|
710 | * declaring loop finished. Optional.
|
711 | * @param stepsName The public API's parameter name for `steps`.
|
712 | * @returns Number of samples provided.
|
713 | */
|
714 | checkNumSamples(ins, batchSize, steps, stepsName = 'steps') {
|
715 | let numSamples;
|
716 | if (steps != null) {
|
717 | numSamples = null;
|
718 | if (batchSize != null) {
|
719 | throw new ValueError(`If ${stepsName} is set, batchSize must be null or undefined.` +
|
720 | `Got batchSize = ${batchSize}`);
|
721 | }
|
722 | }
|
723 | else if (ins != null) {
|
724 | if (Array.isArray(ins)) {
|
725 | numSamples = ins[0].shape[0];
|
726 | }
|
727 | else {
|
728 | numSamples = ins.shape[0];
|
729 | }
|
730 | }
|
731 | else {
|
732 | throw new ValueError(`Either the input data should have a defined shape, or ` +
|
733 | `${stepsName} shoud be specified.`);
|
734 | }
|
735 | return numSamples;
|
736 | }
|
737 | /**
|
738 | * Execute internal tensors of the model with input data feed.
|
739 | * @param inputs Input data feed. Must match the inputs of the model.
|
740 | * @param outputs Names of the output tensors to be fetched. Must match
|
741 | * names of the SymbolicTensors that belong to the graph.
|
742 | * @returns Fetched values for `outputs`.
|
743 | */
|
744 | execute(inputs, outputs) {
|
745 | if (Array.isArray(outputs) && outputs.length === 0) {
|
746 | throw new ValueError('`outputs` is an empty Array, which is not allowed.');
|
747 | }
|
748 | const outputsIsArray = Array.isArray(outputs);
|
749 | const outputNames = (outputsIsArray ? outputs : [outputs]);
|
750 | const outputSymbolicTensors = this.retrieveSymbolicTensors(outputNames);
|
751 | // Format the input into a FeedDict.
|
752 | const feedDict = new FeedDict();
|
753 | if (inputs instanceof Tensor) {
|
754 | inputs = [inputs];
|
755 | }
|
756 | if (Array.isArray(inputs)) {
|
757 | if (inputs.length !== this.inputs.length) {
|
758 | throw new ValueError(`The number of inputs provided (${inputs.length}) ` +
|
759 | `does not match the number of inputs of this model ` +
|
760 | `(${this.inputs.length}).`);
|
761 | }
|
762 | for (let i = 0; i < this.inputs.length; ++i) {
|
763 | feedDict.add(this.inputs[i], inputs[i]);
|
764 | }
|
765 | }
|
766 | else {
|
767 | for (const input of this.inputs) {
|
768 | const tensorValue = inputs[input.name];
|
769 | if (tensorValue == null) {
|
770 | throw new ValueError(`No value is provided for the model's input ${input.name}`);
|
771 | }
|
772 | feedDict.add(input, tensorValue);
|
773 | }
|
774 | }
|
775 | // Run execution.
|
776 | const executeOutputs = execute(outputSymbolicTensors, feedDict);
|
777 | return outputsIsArray ? executeOutputs : executeOutputs[0];
|
778 | }
|
779 | /**
|
780 | * Retrieve the model's internal symbolic tensors from symbolic-tensor names.
|
781 | */
|
782 | retrieveSymbolicTensors(symbolicTensorNames) {
|
783 | const outputSymbolicTensors = pyListRepeat(null, symbolicTensorNames.length);
|
784 | let outputsRemaining = symbolicTensorNames.length;
|
785 | for (const layer of this.layers) {
|
786 | const layerOutputs = Array.isArray(layer.output) ? layer.output : [layer.output];
|
787 | const layerOutputNames = layerOutputs.map(output => output.name);
|
788 | for (let i = 0; i < symbolicTensorNames.length; ++i) {
|
789 | const index = layerOutputNames.indexOf(symbolicTensorNames[i]);
|
790 | if (index !== -1) {
|
791 | outputSymbolicTensors[i] = layerOutputs[index];
|
792 | outputsRemaining--;
|
793 | }
|
794 | if (outputsRemaining === 0) {
|
795 | break;
|
796 | }
|
797 | }
|
798 | if (outputsRemaining === 0) {
|
799 | break;
|
800 | }
|
801 | }
|
802 | if (outputsRemaining > 0) {
|
803 | const remainingNames = [];
|
804 | outputSymbolicTensors.forEach((tensor, i) => {
|
805 | if (tensor == null) {
|
806 | remainingNames.push(symbolicTensorNames[i]);
|
807 | }
|
808 | });
|
809 | throw new ValueError(`Cannot find SymbolicTensors for output name(s): ` +
|
810 | `${JSON.stringify(remainingNames)}`);
|
811 | }
|
812 | return outputSymbolicTensors;
|
813 | }
|
814 | /**
|
815 | * Helper method to loop over some data in batches.
|
816 | *
|
817 | * Porting Note: Not using the functional approach in the Python equivalent
|
818 | * due to the imperative backend.
|
819 | * Porting Note: Does not support step mode currently.
|
820 | *
|
821 | * @param ins: input data
|
822 | * @param batchSize: integer batch size.
|
823 | * @param verbose: verbosity model
|
824 | * @returns: Predictions as `tf.Tensor` (if a single output) or an `Array` of
|
825 | * `tf.Tensor` (if multipe outputs).
|
826 | */
|
827 | predictLoop(ins, batchSize = 32, verbose = false) {
|
828 | return tfc.tidy(() => {
|
829 | const numSamples = this.checkNumSamples(ins);
|
830 | if (verbose) {
|
831 | throw new NotImplementedError('Verbose predictLoop() is not implemented yet.');
|
832 | }
|
833 | // Sample-based predictions.
|
834 | // Porting Note: Tensor currently does not support sliced assignments as
|
835 | // in numpy, e.g., x[1:3] = y. Therefore we use concatenation while
|
836 | // iterating over the batches.
|
837 | const batches = makeBatches(numSamples, batchSize);
|
838 | const outsBatches = this.outputs.map(output => []);
|
839 | // TODO(cais): Can the scope() be pushed down inside the for loop?
|
840 | for (let batchIndex = 0; batchIndex < batches.length; ++batchIndex) {
|
841 | const batchOuts = tfc.tidy(() => {
|
842 | const batchStart = batches[batchIndex][0];
|
843 | const batchEnd = batches[batchIndex][1];
|
844 | // TODO(cais): Take care of the case of the last element is a flag for
|
845 | // training/test.
|
846 | const insBatch = sliceArrays(ins, batchStart, batchEnd);
|
847 | // Construct the feeds for execute();
|
848 | const feeds = [];
|
849 | if (Array.isArray(insBatch)) {
|
850 | for (let i = 0; i < insBatch.length; ++i) {
|
851 | feeds.push({ key: this.inputs[i], value: insBatch[i] });
|
852 | }
|
853 | }
|
854 | else {
|
855 | feeds.push({ key: this.inputs[0], value: insBatch });
|
856 | }
|
857 | const feedDict = new FeedDict(feeds);
|
858 | return execute(this.outputs, feedDict);
|
859 | });
|
860 | batchOuts.forEach((batchOut, i) => outsBatches[i].push(batchOut));
|
861 | }
|
862 | return singletonOrArray(outsBatches.map(batches => tfc.concat(batches, 0)));
|
863 | });
|
864 | }
|
865 | /**
|
866 | * Generates output predictions for the input samples.
|
867 | *
|
868 | * Computation is done in batches.
|
869 | *
|
870 | * Note: the "step" mode of predict() is currently not supported.
|
871 | * This is because the TensorFlow.js core backend is imperative only.
|
872 | *
|
873 | * ```js
|
874 | * const model = tf.sequential({
|
875 | * layers: [tf.layers.dense({units: 1, inputShape: [10]})]
|
876 | * });
|
877 | * model.predict(tf.ones([8, 10]), {batchSize: 4}).print();
|
878 | * ```
|
879 | *
|
880 | * @param x The input data, as a Tensor, or an `Array` of `tf.Tensor`s if
|
881 | * the model has multiple inputs.
|
882 | * @param args A `ModelPredictArgs` object containing optional fields.
|
883 | *
|
884 | * @return Prediction results as a `tf.Tensor`(s).
|
885 | *
|
886 | * @exception ValueError In case of mismatch between the provided input data
|
887 | * and the model's expectations, or in case a stateful model receives a
|
888 | * number of samples that is not a multiple of the batch size.
|
889 | *
|
890 | * @doc {heading: 'Models', subheading: 'Classes'}
|
891 | */
|
892 | predict(x, args = {}) {
|
893 | const xsRank2OrHigher = ensureTensorsRank2OrHigher(x);
|
894 | checkInputData(xsRank2OrHigher, this.inputNames, this.feedInputShapes, false);
|
895 | try {
|
896 | // TODO(cais): Take care of stateful models.
|
897 | // if (this.stateful) ...
|
898 | // TODO(cais): Take care of the learning_phase boolean flag.
|
899 | // if (this.useLearningPhase) ...
|
900 | const batchSize = args.batchSize == null ? 32 : args.batchSize;
|
901 | checkBatchSize(batchSize);
|
902 | return this.predictLoop(xsRank2OrHigher, batchSize);
|
903 | }
|
904 | finally {
|
905 | disposeNewTensors(xsRank2OrHigher, x);
|
906 | }
|
907 | }
|
908 | /**
|
909 | * Returns predictions for a single batch of samples.
|
910 | *
|
911 | * ```js
|
912 | * const model = tf.sequential({
|
913 | * layers: [tf.layers.dense({units: 1, inputShape: [10]})]
|
914 | * });
|
915 | * model.predictOnBatch(tf.ones([8, 10])).print();
|
916 | * ```
|
917 | * @param x: Input samples, as a Tensor (for models with exactly one
|
918 | * input) or an array of Tensors (for models with more than one input).
|
919 | * @return Tensor(s) of predictions
|
920 | *
|
921 | * @doc {heading: 'Models', subheading: 'Classes'}
|
922 | */
|
923 | predictOnBatch(x) {
|
924 | checkInputData(x, this.inputNames, this.feedInputShapes, true);
|
925 | // TODO(cais): Take care of the learning_phase boolean flag.
|
926 | // if (this.useLearningPhase) ...
|
927 | const batchSize = (Array.isArray(x) ? x[0] : x).shape[0];
|
928 | return this.predictLoop(x, batchSize);
|
929 | }
|
930 | standardizeUserDataXY(x, y, checkBatchAxis = true, batchSize) {
|
931 | // TODO(cais): Add sampleWeight, classWeight
|
932 | if (this.optimizer_ == null) {
|
933 | throw new RuntimeError('You must compile a model before training/testing. Use ' +
|
934 | 'LayersModel.compile(modelCompileArgs).');
|
935 | }
|
936 | const outputShapes = [];
|
937 | for (let i = 0; i < this.feedOutputShapes.length; ++i) {
|
938 | const outputShape = this.feedOutputShapes[i];
|
939 | const lossFn = this.feedLossFns[i];
|
940 | if (lossFn === losses.sparseCategoricalCrossentropy) {
|
941 | outputShapes.push(outputShape.slice(0, outputShape.length - 1).concat([1]));
|
942 | }
|
943 | else {
|
944 | // Porting Note: Because of strong typing `lossFn` must be a function.
|
945 | outputShapes.push(outputShape);
|
946 | }
|
947 | }
|
948 | x = standardizeInputData(x, this.feedInputNames, this.feedInputShapes, false, 'input');
|
949 | y = standardizeInputData(y, this.feedOutputNames, outputShapes, false, 'target');
|
950 | // TODO(cais): Standardize sampleWeights & classWeights.
|
951 | checkArrayLengths(x, y, null);
|
952 | // TODO(cais): Check sampleWeights as well.
|
953 | checkLossAndTargetCompatibility(y, this.feedLossFns, this.feedOutputShapes);
|
954 | if (this.stateful && batchSize != null && batchSize > 0) {
|
955 | if (x[0].shape[0] % batchSize !== 0) {
|
956 | throw new ValueError(`In a stateful network, you should only pass inputs with a ` +
|
957 | `number of samples that is divisible by the batch size ` +
|
958 | `${batchSize}. Found: ${x[0].shape[0]} sample(s).`);
|
959 | }
|
960 | }
|
961 | return [x, y];
|
962 | }
|
963 | async standardizeUserData(x, y, sampleWeight, classWeight, checkBatchAxis = true, batchSize) {
|
964 | const [standardXs, standardYs] = this.standardizeUserDataXY(x, y, checkBatchAxis, batchSize);
|
965 | // TODO(cais): Handle sampleWeights.
|
966 | if (sampleWeight != null) {
|
967 | throw new Error('sample weight is not supported yet.');
|
968 | }
|
969 | let standardSampleWeights = null;
|
970 | if (classWeight != null) {
|
971 | const classWeights = standardizeClassWeights(classWeight, this.outputNames);
|
972 | standardSampleWeights = [];
|
973 | for (let i = 0; i < classWeights.length; ++i) {
|
974 | standardSampleWeights.push(await standardizeWeights(standardYs[i], null, classWeights[i]));
|
975 | }
|
976 | }
|
977 | // TODO(cais): Deal with the case of model.stateful == true.
|
978 | return [standardXs, standardYs, standardSampleWeights];
|
979 | }
|
980 | /**
|
981 | * Loop over some test data in batches.
|
982 | * @param f A Function returning a list of tensors.
|
983 | * @param ins Array of tensors to be fed to `f`.
|
984 | * @param batchSize Integer batch size or `null` / `undefined`.
|
985 | * @param verbose verbosity mode.
|
986 | * @param steps Total number of steps (batches of samples) before
|
987 | * declaring test finished. Ignored with the default value of `null` /
|
988 | * `undefined`.
|
989 | * @returns Array of Scalars.
|
990 | */
|
991 | testLoop(f, ins, batchSize, verbose = 0, steps) {
|
992 | return tfc.tidy(() => {
|
993 | const numSamples = this.checkNumSamples(ins, batchSize, steps, 'steps');
|
994 | const outs = [];
|
995 | if (verbose > 0) {
|
996 | throw new NotImplementedError('Verbose mode is not implemented yet.');
|
997 | }
|
998 | // TODO(cais): Use `indicesForConversionToDense' to prevent slow down.
|
999 | if (steps != null) {
|
1000 | throw new NotImplementedError('steps mode in testLoop() is not implemented yet');
|
1001 | }
|
1002 | else {
|
1003 | const batches = makeBatches(numSamples, batchSize);
|
1004 | const indexArray = tensor1d(range(0, numSamples));
|
1005 | for (let batchIndex = 0; batchIndex < batches.length; ++batchIndex) {
|
1006 | const batchStart = batches[batchIndex][0];
|
1007 | const batchEnd = batches[batchIndex][1];
|
1008 | const batchIds = K.sliceAlongFirstAxis(indexArray, batchStart, batchEnd - batchStart);
|
1009 | // TODO(cais): In ins, train flag can be a number, instead of an
|
1010 | // Tensor? Do we need to handle this in tfjs-layers?
|
1011 | const insBatch = sliceArraysByIndices(ins, batchIds);
|
1012 | const batchOuts = f(insBatch);
|
1013 | if (batchIndex === 0) {
|
1014 | for (let i = 0; i < batchOuts.length; ++i) {
|
1015 | outs.push(scalar(0));
|
1016 | }
|
1017 | }
|
1018 | for (let i = 0; i < batchOuts.length; ++i) {
|
1019 | const batchOut = batchOuts[i];
|
1020 | outs[i] =
|
1021 | tfc.add(outs[i], tfc.mul(batchEnd - batchStart, batchOut));
|
1022 | }
|
1023 | }
|
1024 | for (let i = 0; i < outs.length; ++i) {
|
1025 | outs[i] = tfc.div(outs[i], numSamples);
|
1026 | }
|
1027 | }
|
1028 | return outs;
|
1029 | });
|
1030 | }
|
1031 | getDedupedMetricsNames() {
|
1032 | const outLabels = this.metricsNames;
|
1033 | // Rename duplicated metrics names (can happen with an output layer
|
1034 | // shared among multiple dataflows).
|
1035 | const dedupedOutLabels = [];
|
1036 | for (let i = 0; i < outLabels.length; ++i) {
|
1037 | const label = outLabels[i];
|
1038 | let newLabel = label;
|
1039 | if (count(outLabels, label) > 1) {
|
1040 | const dupIndex = count(outLabels.slice(0, i), label);
|
1041 | newLabel += `_${dupIndex}`;
|
1042 | }
|
1043 | dedupedOutLabels.push(newLabel);
|
1044 | }
|
1045 | return dedupedOutLabels;
|
1046 | }
|
1047 | /**
|
1048 | * Creates a function that performs the following actions:
|
1049 | *
|
1050 | * 1. computes the losses
|
1051 | * 2. sums them to get the total loss
|
1052 | * 3. call the optimizer computes the gradients of the LayersModel's
|
1053 | * trainable weights w.r.t. the total loss and update the variables
|
1054 | * 4. calculates the metrics
|
1055 | * 5. returns the values of the losses and metrics.
|
1056 | */
|
1057 | makeTrainFunction() {
|
1058 | return (data) => {
|
1059 | const lossValues = [];
|
1060 | const inputs = data.slice(0, this.inputs.length);
|
1061 | const targets = data.slice(this.inputs.length, this.inputs.length + this.outputs.length);
|
1062 | const sampleWeights = data.slice(this.inputs.length + this.outputs.length, this.inputs.length + this.outputs.length * 2);
|
1063 | const metricsValues = [];
|
1064 | // Create a function that computes the total loss based on the
|
1065 | // inputs. This function is used for obtaining gradients through
|
1066 | // backprop.
|
1067 | const totalLossFunction = () => {
|
1068 | const feeds = [];
|
1069 | for (let i = 0; i < this.inputs.length; ++i) {
|
1070 | feeds.push({ key: this.inputs[i], value: inputs[i] });
|
1071 | }
|
1072 | const feedDict = new FeedDict(feeds);
|
1073 | const outputs = execute(this.outputs, feedDict, { 'training': true });
|
1074 | // TODO(cais): Take care of the case of multiple outputs from a
|
1075 | // single layer?
|
1076 | let totalLoss;
|
1077 | for (let i = 0; i < this.lossFunctions.length; ++i) {
|
1078 | const lossFunction = this.lossFunctions[i];
|
1079 | let loss = lossFunction(targets[i], outputs[i]);
|
1080 | if (sampleWeights[i] != null) {
|
1081 | loss = computeWeightedLoss(loss, sampleWeights[i]);
|
1082 | }
|
1083 | // TODO(cais): push Scalar instead.
|
1084 | const meanLoss = tfc.mean(loss);
|
1085 | // TODO(cais): Use a scope() instead, to avoid ownership.
|
1086 | lossValues.push(meanLoss);
|
1087 | if (i === 0) {
|
1088 | totalLoss = loss;
|
1089 | }
|
1090 | else {
|
1091 | totalLoss = tfc.add(totalLoss, loss);
|
1092 | }
|
1093 | }
|
1094 | // Compute the metrics.
|
1095 | // TODO(cais): These should probably be calculated outside
|
1096 | // totalLossFunction to benefit speed?
|
1097 | for (let i = 0; i < this.metricsTensors.length; ++i) {
|
1098 | let weightedMetric;
|
1099 | if (this.outputs.length > 1 && i < this.outputs.length) {
|
1100 | weightedMetric = lossValues[i];
|
1101 | }
|
1102 | else {
|
1103 | const metric = this.metricsTensors[i][0];
|
1104 | const outputIndex = this.metricsTensors[i][1];
|
1105 | weightedMetric =
|
1106 | tfc.mean(metric(targets[outputIndex], outputs[outputIndex]));
|
1107 | }
|
1108 | tfc.keep(weightedMetric);
|
1109 | // TODO(cais): Use a scope() instead, to avoid ownership.
|
1110 | metricsValues.push(weightedMetric);
|
1111 | }
|
1112 | totalLoss = tfc.mean(totalLoss);
|
1113 | // Add regularizer penalties.
|
1114 | this.calculateLosses().forEach(regularizerLoss => {
|
1115 | totalLoss = tfc.add(totalLoss, regularizerLoss);
|
1116 | });
|
1117 | return totalLoss;
|
1118 | };
|
1119 | const variables = this.collectedTrainableWeights.map(param => param.read());
|
1120 | const returnCost = true;
|
1121 | const totalLossValue = this.optimizer_.minimize(totalLossFunction, returnCost, variables);
|
1122 | return [totalLossValue].concat(metricsValues);
|
1123 | };
|
1124 | }
|
1125 | /**
|
1126 | * Create a function which, when invoked with an array of `tf.Tensor`s as a
|
1127 | * batch of inputs, returns the prespecified loss and metrics of the model
|
1128 | * under the batch of input data.
|
1129 | */
|
1130 | makeTestFunction() {
|
1131 | this.testFunction = (data) => {
|
1132 | return tfc.tidy(() => {
|
1133 | const valOutputs = [];
|
1134 | let totalLoss;
|
1135 | const inputs = data.slice(0, this.inputs.length);
|
1136 | const targets = data.slice(this.inputs.length, this.inputs.length + this.outputs.length);
|
1137 | const feeds = [];
|
1138 | for (let i = 0; i < this.inputs.length; ++i) {
|
1139 | feeds.push({ key: this.inputs[i], value: inputs[i] });
|
1140 | }
|
1141 | const feedDict = new FeedDict(feeds);
|
1142 | const outputs = execute(this.outputs, feedDict);
|
1143 | // Compute total loss.
|
1144 | for (let i = 0; i < this.lossFunctions.length; ++i) {
|
1145 | const lossFunction = this.lossFunctions[i];
|
1146 | // TODO(cais): Add sample weighting and replace the simple
|
1147 | // averaging.
|
1148 | const loss = tfc.mean(lossFunction(targets[i], outputs[i]));
|
1149 | if (i === 0) {
|
1150 | totalLoss = loss;
|
1151 | }
|
1152 | else {
|
1153 | totalLoss = tfc.add(totalLoss, loss);
|
1154 | }
|
1155 | valOutputs.push(totalLoss);
|
1156 | }
|
1157 | // Compute the metrics.
|
1158 | for (let i = 0; i < this.metricsTensors.length; ++i) {
|
1159 | const metric = this.metricsTensors[i][0];
|
1160 | const outputIndex = this.metricsTensors[i][1];
|
1161 | // TODO(cais): Replace K.mean() with a proper weighting function.
|
1162 | const meanMetric = tfc.mean(metric(targets[outputIndex], outputs[outputIndex]));
|
1163 | valOutputs.push(meanMetric);
|
1164 | }
|
1165 | return valOutputs;
|
1166 | });
|
1167 | };
|
1168 | }
|
1169 | /**
|
1170 | * Trains the model for a fixed number of epochs (iterations on a
|
1171 | * dataset).
|
1172 | *
|
1173 | * ```js
|
1174 | * const model = tf.sequential({
|
1175 | * layers: [tf.layers.dense({units: 1, inputShape: [10]})]
|
1176 | * });
|
1177 | * model.compile({optimizer: 'sgd', loss: 'meanSquaredError'});
|
1178 | * for (let i = 1; i < 5 ; ++i) {
|
1179 | * const h = await model.fit(tf.ones([8, 10]), tf.ones([8, 1]), {
|
1180 | * batchSize: 4,
|
1181 | * epochs: 3
|
1182 | * });
|
1183 | * console.log("Loss after Epoch " + i + " : " + h.history.loss[0]);
|
1184 | * }
|
1185 | * ```
|
1186 | *
|
1187 | * @param x `tf.Tensor` of training data, or an array of `tf.Tensor`s if the
|
1188 | * model has multiple inputs. If all inputs in the model are named, you
|
1189 | * can also pass a dictionary mapping input names to `tf.Tensor`s.
|
1190 | * @param y `tf.Tensor` of target (label) data, or an array of `tf.Tensor`s if
|
1191 | * the model has multiple outputs. If all outputs in the model are named,
|
1192 | * you can also pass a dictionary mapping output names to `tf.Tensor`s.
|
1193 | * @param args A `ModelFitArgs`, containing optional fields.
|
1194 | *
|
1195 | * @return A `History` instance. Its `history` attribute contains all
|
1196 | * information collected during training.
|
1197 | *
|
1198 | * @exception ValueError In case of mismatch between the provided input
|
1199 | * data and what the model expects.
|
1200 | *
|
1201 | * @doc {heading: 'Models', subheading: 'Classes'}
|
1202 | */
|
1203 | async fit(x, y, args = {}) {
|
1204 | return fitTensors(this, x, y, args);
|
1205 | }
|
1206 | // TODO(cais): Add code snippet below when it's possible to instantiate
|
1207 | // actual dataset objects.
|
1208 | /**
|
1209 | * Trains the model using a dataset object.
|
1210 | *
|
1211 | * @param dataset A dataset object. Its `iterator()` method is expected
|
1212 | * to generate a dataset iterator object, the `next()` method of which
|
1213 | * is expected to produce data batches for training. The return value
|
1214 | * of the `next()` call ought to contain a boolean `done` field and a
|
1215 | * `value` field. The `value` field is expected to be an array of two
|
1216 | * `tf.Tensor`s or an array of two nested `tf.Tensor` structures. The former
|
1217 | * case is for models with exactly one input and one output (e.g..
|
1218 | * a sequential model). The latter case is for models with multiple
|
1219 | * inputs and/or multiple outputs.
|
1220 | * Of the two items in the array, the first is the input feature(s) and
|
1221 | * the second is the output target(s).
|
1222 | * @param args A `ModelFitDatasetArgs`, containing optional fields.
|
1223 | *
|
1224 | * @return A `History` instance. Its `history` attribute contains all
|
1225 | * information collected during training.
|
1226 | *
|
1227 | * @doc {heading: 'Models', subheading: 'Classes'}
|
1228 | */
|
1229 | async fitDataset(dataset, args) {
|
1230 | return fitDataset(this, dataset, args);
|
1231 | }
|
1232 | /**
|
1233 | * Runs a single gradient update on a single batch of data.
|
1234 | *
|
1235 | * This method differs from `fit()` and `fitDataset()` in the following
|
1236 | * regards:
|
1237 | * - It operates on exactly one batch of data.
|
1238 | * - It returns only the loss and matric values, instead of
|
1239 | * returning the batch-by-batch loss and metric values.
|
1240 | * - It doesn't support fine-grained options such as verbosity and
|
1241 | * callbacks.
|
1242 | *
|
1243 | * @param x Input data. It could be one of the following:
|
1244 | * - A `tf.Tensor`, or an Array of `tf.Tensor`s (in case the model has
|
1245 | * multiple inputs).
|
1246 | * - An Object mapping input names to corresponding `tf.Tensor` (if the
|
1247 | * model has named inputs).
|
1248 | * @param y Target darta. It could be either a `tf.Tensor` a multiple
|
1249 | * `tf.Tensor`s. It should be consistent with `x`.
|
1250 | * @returns Training loss or losses (in case the model has
|
1251 | * multiple outputs), along with metrics (if any), as numbers.
|
1252 | *
|
1253 | * @doc {heading: 'Models', subheading: 'Classes'}
|
1254 | */
|
1255 | async trainOnBatch(x, y) {
|
1256 | // TODO(cais): Support sampleWeight and classWeight.
|
1257 | // TODO(cais): Support Dataset objects.
|
1258 | const standardizeOut = await this.standardizeUserData(x, y);
|
1259 | const inputs = standardizeOut[0];
|
1260 | const targets = standardizeOut[1];
|
1261 | const trainFunction = this.makeTrainFunction();
|
1262 | const losses = trainFunction(inputs.concat(targets));
|
1263 | const lossValues = [];
|
1264 | for (const loss of losses) {
|
1265 | const v = await loss.data();
|
1266 | lossValues.push(v[0]);
|
1267 | }
|
1268 | tfc.dispose(losses);
|
1269 | disposeNewTensors(standardizeOut[0], x);
|
1270 | disposeNewTensors(standardizeOut[1], y);
|
1271 | return singletonOrArray(lossValues);
|
1272 | }
|
1273 | /**
|
1274 | * Extract weight values of the model.
|
1275 | *
|
1276 | * @param config: An instance of `io.SaveConfig`, which specifies
|
1277 | * model-saving options such as whether only trainable weights are to be
|
1278 | * saved.
|
1279 | * @returns A `NamedTensorMap` mapping original weight names (i.e.,
|
1280 | * non-uniqueified weight names) to their values.
|
1281 | */
|
1282 | getNamedWeights(config) {
|
1283 | const namedWeights = [];
|
1284 | const trainableOnly = config != null && config.trainableOnly;
|
1285 | const weights = trainableOnly ? this.trainableWeights : this.weights;
|
1286 | const weightValues = this.getWeights(trainableOnly);
|
1287 | for (let i = 0; i < weights.length; ++i) {
|
1288 | if (trainableOnly && !weights[i].trainable) {
|
1289 | // Optionally skip non-trainable weights.
|
1290 | continue;
|
1291 | }
|
1292 | namedWeights.push({ name: weights[i].originalName, tensor: weightValues[i] });
|
1293 | }
|
1294 | return namedWeights;
|
1295 | }
|
1296 | /**
|
1297 | * Setter used for force stopping of LayersModel.fit() (i.e., training).
|
1298 | *
|
1299 | * Example:
|
1300 | *
|
1301 | * ```js
|
1302 | * const input = tf.input({shape: [10]});
|
1303 | * const output = tf.layers.dense({units: 1}).apply(input);
|
1304 | * const model = tf.model({inputs: [input], outputs: [output]});
|
1305 | * model.compile({loss: 'meanSquaredError', optimizer: 'sgd'});
|
1306 | * const xs = tf.ones([8, 10]);
|
1307 | * const ys = tf.zeros([8, 1]);
|
1308 | *
|
1309 | * const history = await model.fit(xs, ys, {
|
1310 | * epochs: 10,
|
1311 | * callbacks: {
|
1312 | * onEpochEnd: async (epoch, logs) => {
|
1313 | * if (epoch === 2) {
|
1314 | * model.stopTraining = true;
|
1315 | * }
|
1316 | * }
|
1317 | * }
|
1318 | * });
|
1319 | *
|
1320 | * // There should be only 3 values in the loss array, instead of 10
|
1321 | * values,
|
1322 | * // due to the stopping after 3 epochs.
|
1323 | * console.log(history.history.loss);
|
1324 | * ```
|
1325 | */
|
1326 | set stopTraining(stop) {
|
1327 | this.stopTraining_ = stop;
|
1328 | }
|
1329 | get stopTraining() {
|
1330 | return this.stopTraining_;
|
1331 | }
|
1332 | get optimizer() {
|
1333 | return this.optimizer_;
|
1334 | }
|
1335 | set optimizer(optimizer) {
|
1336 | if (this.optimizer_ !== optimizer) {
|
1337 | this.optimizer_ = optimizer;
|
1338 | this.isOptimizerOwned = false;
|
1339 | }
|
1340 | }
|
1341 | dispose() {
|
1342 | const result = super.dispose();
|
1343 | if (result.refCountAfterDispose === 0 && this.optimizer != null &&
|
1344 | this.isOptimizerOwned) {
|
1345 | const numTensorsBeforeOptmizerDisposal = tfc.memory().numTensors;
|
1346 | this.optimizer_.dispose();
|
1347 | result.numDisposedVariables +=
|
1348 | numTensorsBeforeOptmizerDisposal - tfc.memory().numTensors;
|
1349 | }
|
1350 | return result;
|
1351 | }
|
1352 | getLossIdentifiers() {
|
1353 | let lossNames;
|
1354 | if (typeof this.loss === 'string') {
|
1355 | lossNames = toSnakeCase(this.loss);
|
1356 | }
|
1357 | else if (Array.isArray(this.loss)) {
|
1358 | for (const loss of this.loss) {
|
1359 | if (typeof loss !== 'string') {
|
1360 | throw new Error('Serialization of non-string loss is not supported.');
|
1361 | }
|
1362 | }
|
1363 | lossNames = this.loss.map(name => toSnakeCase(name));
|
1364 | }
|
1365 | else {
|
1366 | const outputNames = Object.keys(this.loss);
|
1367 | lossNames = {};
|
1368 | const losses = this.loss;
|
1369 | for (const outputName of outputNames) {
|
1370 | if (typeof losses[outputName] === 'string') {
|
1371 | lossNames[outputName] =
|
1372 | toSnakeCase(losses[outputName]);
|
1373 | }
|
1374 | else {
|
1375 | throw new Error('Serialization of non-string loss is not supported.');
|
1376 | }
|
1377 | }
|
1378 | }
|
1379 | return lossNames;
|
1380 | }
|
1381 | getMetricIdentifiers() {
|
1382 | if (typeof this.metrics === 'string' ||
|
1383 | typeof this.metrics === 'function') {
|
1384 | return [toSnakeCase(Metrics.getLossOrMetricName(this.metrics))];
|
1385 | }
|
1386 | else if (Array.isArray(this.metrics)) {
|
1387 | return this.metrics.map(metric => toSnakeCase(Metrics.getLossOrMetricName(metric)));
|
1388 | }
|
1389 | else {
|
1390 | const metricsIdentifiers = {};
|
1391 | for (const key in this.metrics) {
|
1392 | metricsIdentifiers[key] =
|
1393 | toSnakeCase(Metrics.getLossOrMetricName(this.metrics[key]));
|
1394 | }
|
1395 | return metricsIdentifiers;
|
1396 | }
|
1397 | }
|
1398 | getTrainingConfig() {
|
1399 | return {
|
1400 | loss: this.getLossIdentifiers(),
|
1401 | metrics: this.getMetricIdentifiers(),
|
1402 | optimizer_config: {
|
1403 | class_name: this.optimizer.getClassName(),
|
1404 | config: this.optimizer.getConfig()
|
1405 | }
|
1406 | };
|
1407 | // TODO(cais): Add weight_metrics when they are supported.
|
1408 | // TODO(cais): Add sample_weight_mode when it's supported.
|
1409 | // TODO(cais): Add loss_weights when it's supported.
|
1410 | }
|
1411 | loadTrainingConfig(trainingConfig) {
|
1412 | if (trainingConfig.weighted_metrics != null) {
|
1413 | throw new Error('Loading weight_metrics is not supported yet.');
|
1414 | }
|
1415 | if (trainingConfig.loss_weights != null) {
|
1416 | throw new Error('Loading loss_weights is not supported yet.');
|
1417 | }
|
1418 | if (trainingConfig.sample_weight_mode != null) {
|
1419 | throw new Error('Loading sample_weight_mode is not supported yet.');
|
1420 | }
|
1421 | const tsConfig = convertPythonicToTs(trainingConfig.optimizer_config);
|
1422 | const optimizer = deserialize(tsConfig);
|
1423 | let loss;
|
1424 | if (typeof trainingConfig.loss === 'string') {
|
1425 | loss = toCamelCase(trainingConfig.loss);
|
1426 | }
|
1427 | else if (Array.isArray(trainingConfig.loss)) {
|
1428 | loss = trainingConfig.loss.map(lossEntry => toCamelCase(lossEntry));
|
1429 | }
|
1430 | else if (trainingConfig.loss != null) {
|
1431 | loss = {};
|
1432 | for (const key in trainingConfig.loss) {
|
1433 | loss[key] = toCamelCase(trainingConfig.loss[key]);
|
1434 | }
|
1435 | }
|
1436 | let metrics;
|
1437 | if (Array.isArray(trainingConfig.metrics)) {
|
1438 | metrics = trainingConfig.metrics.map(metric => toCamelCase(metric));
|
1439 | }
|
1440 | else if (trainingConfig.metrics != null) {
|
1441 | metrics = {};
|
1442 | for (const key in trainingConfig.metrics) {
|
1443 | metrics[key] = toCamelCase(trainingConfig.metrics[key]);
|
1444 | }
|
1445 | }
|
1446 | this.compile({ loss, metrics, optimizer });
|
1447 | }
|
1448 | /**
|
1449 | * Save the configuration and/or weights of the LayersModel.
|
1450 | *
|
1451 | * An `IOHandler` is an object that has a `save` method of the proper
|
1452 | * signature defined. The `save` method manages the storing or
|
1453 | * transmission of serialized data ("artifacts") that represent the
|
1454 | * model's topology and weights onto or via a specific medium, such as
|
1455 | * file downloads, local storage, IndexedDB in the web browser and HTTP
|
1456 | * requests to a server. TensorFlow.js provides `IOHandler`
|
1457 | * implementations for a number of frequently used saving mediums, such as
|
1458 | * `tf.io.browserDownloads` and `tf.io.browserLocalStorage`. See `tf.io`
|
1459 | * for more details.
|
1460 | *
|
1461 | * This method also allows you to refer to certain types of `IOHandler`s
|
1462 | * as URL-like string shortcuts, such as 'localstorage://' and
|
1463 | * 'indexeddb://'.
|
1464 | *
|
1465 | * Example 1: Save `model`'s topology and weights to browser [local
|
1466 | * storage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage);
|
1467 | * then load it back.
|
1468 | *
|
1469 | * ```js
|
1470 | * const model = tf.sequential(
|
1471 | * {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});
|
1472 | * console.log('Prediction from original model:');
|
1473 | * model.predict(tf.ones([1, 3])).print();
|
1474 | *
|
1475 | * const saveResults = await model.save('localstorage://my-model-1');
|
1476 | *
|
1477 | * const loadedModel = await tf.loadLayersModel('localstorage://my-model-1');
|
1478 | * console.log('Prediction from loaded model:');
|
1479 | * loadedModel.predict(tf.ones([1, 3])).print();
|
1480 | * ```
|
1481 | *
|
1482 | * Example 2. Saving `model`'s topology and weights to browser
|
1483 | * [IndexedDB](https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API);
|
1484 | * then load it back.
|
1485 | *
|
1486 | * ```js
|
1487 | * const model = tf.sequential(
|
1488 | * {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});
|
1489 | * console.log('Prediction from original model:');
|
1490 | * model.predict(tf.ones([1, 3])).print();
|
1491 | *
|
1492 | * const saveResults = await model.save('indexeddb://my-model-1');
|
1493 | *
|
1494 | * const loadedModel = await tf.loadLayersModel('indexeddb://my-model-1');
|
1495 | * console.log('Prediction from loaded model:');
|
1496 | * loadedModel.predict(tf.ones([1, 3])).print();
|
1497 | * ```
|
1498 | *
|
1499 | * Example 3. Saving `model`'s topology and weights as two files
|
1500 | * (`my-model-1.json` and `my-model-1.weights.bin`) downloaded from
|
1501 | * browser.
|
1502 | *
|
1503 | * ```js
|
1504 | * const model = tf.sequential(
|
1505 | * {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});
|
1506 | * const saveResults = await model.save('downloads://my-model-1');
|
1507 | * ```
|
1508 | *
|
1509 | * Example 4. Send `model`'s topology and weights to an HTTP server.
|
1510 | * See the documentation of `tf.io.http` for more details
|
1511 | * including specifying request parameters and implementation of the
|
1512 | * server.
|
1513 | *
|
1514 | * ```js
|
1515 | * const model = tf.sequential(
|
1516 | * {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});
|
1517 | * const saveResults = await model.save('http://my-server/model/upload');
|
1518 | * ```
|
1519 | *
|
1520 | * @param handlerOrURL An instance of `IOHandler` or a URL-like,
|
1521 | * scheme-based string shortcut for `IOHandler`.
|
1522 | * @param config Options for saving the model.
|
1523 | * @returns A `Promise` of `SaveResult`, which summarizes the result of
|
1524 | * the saving, such as byte sizes of the saved artifacts for the model's
|
1525 | * topology and weight values.
|
1526 | *
|
1527 | * @doc {heading: 'Models', subheading: 'Classes', ignoreCI: true}
|
1528 | */
|
1529 | async save(handlerOrURL, config) {
|
1530 | if (typeof handlerOrURL === 'string') {
|
1531 | const handlers = io.getSaveHandlers(handlerOrURL);
|
1532 | if (handlers.length === 0) {
|
1533 | throw new ValueError(`Cannot find any save handlers for URL '${handlerOrURL}'`);
|
1534 | }
|
1535 | else if (handlers.length > 1) {
|
1536 | throw new ValueError(`Found more than one (${handlers.length}) save handlers for ` +
|
1537 | `URL '${handlerOrURL}'`);
|
1538 | }
|
1539 | handlerOrURL = handlers[0];
|
1540 | }
|
1541 | if (handlerOrURL.save == null) {
|
1542 | throw new ValueError('LayersModel.save() cannot proceed because the IOHandler ' +
|
1543 | 'provided does not have the `save` attribute defined.');
|
1544 | }
|
1545 | const weightDataAndSpecs = await io.encodeWeights(this.getNamedWeights(config));
|
1546 | const returnString = false;
|
1547 | const unusedArg = null;
|
1548 | const modelConfig = this.toJSON(unusedArg, returnString);
|
1549 | const modelArtifacts = {
|
1550 | modelTopology: modelConfig,
|
1551 | format: LAYERS_MODEL_FORMAT_NAME,
|
1552 | generatedBy: `TensorFlow.js tfjs-layers v${version}`,
|
1553 | convertedBy: null,
|
1554 | };
|
1555 | const includeOptimizer = config == null ? false : config.includeOptimizer;
|
1556 | if (includeOptimizer && this.optimizer != null) {
|
1557 | modelArtifacts.trainingConfig = this.getTrainingConfig();
|
1558 | const weightType = 'optimizer';
|
1559 | const { data: optimizerWeightData, specs: optimizerWeightSpecs } = await io.encodeWeights(await this.optimizer.getWeights(), weightType);
|
1560 | weightDataAndSpecs.specs.push(...optimizerWeightSpecs);
|
1561 | weightDataAndSpecs.data = io.concatenateArrayBuffers([weightDataAndSpecs.data, optimizerWeightData]);
|
1562 | }
|
1563 | if (this.userDefinedMetadata != null) {
|
1564 | // Check serialized size of user-defined metadata.
|
1565 | const checkSize = true;
|
1566 | checkUserDefinedMetadata(this.userDefinedMetadata, this.name, checkSize);
|
1567 | modelArtifacts.userDefinedMetadata = this.userDefinedMetadata;
|
1568 | }
|
1569 | modelArtifacts.weightData = weightDataAndSpecs.data;
|
1570 | modelArtifacts.weightSpecs = weightDataAndSpecs.specs;
|
1571 | return handlerOrURL.save(modelArtifacts);
|
1572 | }
|
1573 | /**
|
1574 | * Set user-defined metadata.
|
1575 | *
|
1576 | * The set metadata will be serialized together with the topology
|
1577 | * and weights of the model during `save()` calls.
|
1578 | *
|
1579 | * @param setUserDefinedMetadata
|
1580 | */
|
1581 | setUserDefinedMetadata(userDefinedMetadata) {
|
1582 | checkUserDefinedMetadata(userDefinedMetadata, this.name);
|
1583 | this.userDefinedMetadata = userDefinedMetadata;
|
1584 | }
|
1585 | /**
|
1586 | * Get user-defined metadata.
|
1587 | *
|
1588 | * The metadata is supplied via one of the two routes:
|
1589 | * 1. By calling `setUserDefinedMetadata()`.
|
1590 | * 2. Loaded during model loading (if the model is constructed
|
1591 | * via `tf.loadLayersModel()`.)
|
1592 | *
|
1593 | * If no user-defined metadata is available from either of the
|
1594 | * two routes, this function will return `undefined`.
|
1595 | */
|
1596 | getUserDefinedMetadata() {
|
1597 | return this.userDefinedMetadata;
|
1598 | }
|
1599 | }
|
1600 | // The class name is 'Model' rather than 'LayersModel' for backwards
|
1601 | // compatibility since this class name shows up in the serialization format.
|
1602 | /** @nocollapse */
|
1603 | LayersModel.className = 'Model';
|
1604 | serialization.registerClass(LayersModel);
|
1605 | /**
|
1606 | * A `tf.Functional` is an alias to `tf.LayersModel`.
|
1607 | *
|
1608 | * See also:
|
1609 | * `tf.LayersModel`, `tf.Sequential`, `tf.loadLayersModel`.
|
1610 | */
|
1611 | /** @doc {heading: 'Models', subheading: 'Classes'} */
|
1612 | export class Functional extends LayersModel {
|
1613 | }
|
1614 | Functional.className = 'Functional';
|
1615 | serialization.registerClass(Functional);
|
1616 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"training.js","sourceRoot":"","sources":["../../../../../../tfjs-layers/src/engine/training.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2018 Google LLC\n *\n * Use of this source code is governed by an MIT-style\n * license that can be found in the LICENSE file or at\n * https://opensource.org/licenses/MIT.\n * =============================================================================\n */\n\n/* Original Source: engine/training.py */\n\nimport * as tfc from '@tensorflow/tfjs-core';\nimport {io, ModelPredictConfig as ModelPredictArgs, NamedTensorMap, Optimizer, Scalar, scalar, serialization, Tensor, Tensor1D, tensor1d, util} from '@tensorflow/tfjs-core';\n\nimport * as K from '../backend/tfjs_backend';\nimport {History, ModelLoggingVerbosity} from '../base_callbacks';\nimport {nameScope} from '../common';\nimport {NotImplementedError, RuntimeError, ValueError} from '../errors';\nimport {Shape} from '../keras_format/common';\nimport {LossIdentifier} from '../keras_format/loss_config';\nimport {OptimizerSerialization} from '../keras_format/optimizer_config';\nimport {MetricsIdentifier, TrainingConfig} from '../keras_format/training_config';\nimport {deserialize} from '../layers/serialization';\nimport * as losses from '../losses';\nimport * as Metrics from '../metrics';\nimport * as optimizers from '../optimizers';\nimport {LossOrMetricFn, NamedTensor} from '../types';\nimport {checkUserDefinedMetadata} from '../user_defined_metadata';\nimport {count, pyListRepeat, singletonOrArray, toCamelCase, toSnakeCase, unique} from '../utils/generic_utils';\nimport {printSummary} from '../utils/layer_utils';\nimport {range} from '../utils/math_utils';\nimport {convertPythonicToTs} from '../utils/serialization_utils';\nimport {LayerVariable} from '../variables';\nimport {version} from '../version';\n\nimport {Container, ContainerArgs} from './container';\nimport {Dataset} from './dataset_stub';\nimport {execute, FeedDict} from './executor';\nimport {DisposeResult, SymbolicTensor} from './topology';\nimport {evaluateDataset, fitDataset, ModelEvaluateDatasetArgs, ModelFitDatasetArgs} from './training_dataset';\nimport {checkBatchSize, disposeNewTensors, ensureTensorsRank2OrHigher, fitTensors, makeBatches, ModelFitArgs, sliceArrays, sliceArraysByIndices} from './training_tensors';\nimport {ClassWeight, ClassWeightMap, computeWeightedLoss, standardizeClassWeights, standardizeWeights} from './training_utils';\n\n/**\n * Helper function for polymorphic input data: 1. singleton Tensor.\n */\nexport function isDataTensor(x: Tensor|Tensor[]|{[inputName: string]: Tensor}|\n                             {[inputName: string]: Tensor[]}): boolean {\n  return x instanceof Tensor;\n}\n\n/**\n * Helper function for polymorphic input data: 2. Array of Tensor.\n */\nexport function isDataArray(x: Tensor|Tensor[]|\n                            {[inputName: string]: Tensor}): boolean {\n  return Array.isArray(x);\n}\n\n/**\n * Helper function for polymorphic input data: 3. \"dict\" of Tensor.\n */\nexport function isDataDict(x: Tensor|Tensor[]|\n                           {[inputName: string]: Tensor}): boolean {\n  return !isDataTensor(x) && !isDataArray(x);\n}\n\n/**\n * Normalizes inputs and targets provided by users.\n * @param data User-provided input data (polymorphic).\n * @param names An Array of expected Tensor names.\n * @param shapes Optional Array of expected Tensor shapes.\n * @param checkBatchAxis Whether to check that the batch axis of the arrays\n *   match  the expected value found in `shapes`.\n * @param exceptionPrefix String prefix used for exception formatting.\n * @returns List of standardized input Tensors (one Tensor per model input).\n * @throws ValueError: in case of improperly formatted user data.\n */\nexport function standardizeInputData(\n    data: Tensor|Tensor[]|{[inputName: string]: Tensor}, names: string[],\n    shapes?: Shape[], checkBatchAxis = true, exceptionPrefix = ''): Tensor[] {\n  if (names == null || names.length === 0) {\n    // Check for the case where the model expected no data, but some data got\n    // sent.\n    if (data != null) {\n      let gotUnexpectedData = false;\n      if (isDataArray(data) && (data as Tensor[]).length > 0) {\n        gotUnexpectedData = true;\n      } else if (isDataDict(data)) {\n        for (const key in data) {\n          if (data.hasOwnProperty(key)) {\n            gotUnexpectedData = true;\n            break;\n          }\n        }\n      } else {\n        // `data` is a singleton Tensor in this case.\n        gotUnexpectedData = true;\n      }\n      if (gotUnexpectedData) {\n        throw new ValueError(\n            `Error when checking model ${exceptionPrefix} expected no data, ` +\n            `but got ${data}`);\n      }\n    }\n    return [];\n  }\n  if (data == null) {\n    return names.map(name => null);\n  }\n\n  let arrays: Tensor[];\n  if (isDataDict(data)) {\n    data = data as {[inputName: string]: Tensor};\n    arrays = [];\n    for (const name of names) {\n      if (data[name] == null) {\n        throw new ValueError(\n            `No data provided for \"${name}\". Need data for each key in: ` +\n            `${names}`);\n      }\n      arrays.push(data[name]);\n    }\n  } else if (isDataArray(data)) {\n    data = data as Tensor[];\n    if (data.length !== names.length) {\n      throw new ValueError(\n          `Error when checking model ${exceptionPrefix}: the Array of ` +\n          `Tensors that you are passing to your model is not the size the ` +\n          `model expected. Expected to see ${names.length} Tensor(s), but ` +\n          `instead got the following list of Tensor(s): ${data}`);\n    }\n    arrays = data;\n  } else {\n    data = data as Tensor;\n    if (names.length > 1) {\n      throw new ValueError(\n          `The model ${exceptionPrefix} expects ${names.length} Tensor(s), ` +\n          `but only received one Tensor. Found: Tensor with shape ${\n              data.shape}`);\n    }\n    arrays = [data];\n  }\n\n  arrays = ensureTensorsRank2OrHigher(arrays);\n\n  // Check shape compatibility.\n  if (shapes != null) {\n    for (let i = 0; i < names.length; ++i) {\n      if (shapes[i] == null) {\n        continue;\n      }\n      const array = arrays[i];\n      if (array.shape.length !== shapes[i].length) {\n        throw new ValueError(\n            `Error when checking ${exceptionPrefix}: expected ${names[i]} ` +\n            `to have ${shapes[i].length} dimension(s). but got array with ` +\n            `shape ${array.shape}`);\n      }\n      for (let j = 0; j < shapes[i].length; ++j) {\n        if (j === 0 && !checkBatchAxis) {\n          // Skip the first (batch) axis.\n          continue;\n        }\n        const dim = array.shape[j];\n        const refDim = shapes[i][j];\n        if (refDim != null && refDim >= 0 && dim !== refDim) {\n          throw new ValueError(\n              `${exceptionPrefix} expected a batch of elements where each ` +\n              `example has shape [${shapes[i].slice(1, shapes[i].length)}] ` +\n              `(i.e.,tensor shape [*,${\n                  shapes[i].slice(1, shapes[i].length)}])` +\n              ` but the ${exceptionPrefix} received an input with ${\n                  array.shape[0]}` +\n              ` examples, each with shape [${\n                  array.shape.slice(1, array.shape.length)}]` +\n              ` (tensor shape [${array.shape}])`);\n        }\n      }\n    }\n  }\n  return arrays;\n}\n\n/**\n * User input validation for Tensors.\n * @param inputs `Array` of `tf.Tensor`s for inputs.\n * @param targets `Array` of `tf.Tensor`s for targets.\n * @param weights Optional `Array` of `tf.Tensor`s for sample weights.\n * @throws ValueError: in case of incorrectly formatted data.\n */\nexport function checkArrayLengths(\n    inputs: Tensor[], targets: Tensor[], weights?: Tensor[]) {\n  const setX = unique(inputs.map(input => input.shape[0]));\n  setX.sort();\n  const setY = unique(targets.map(target => target.shape[0]));\n  setY.sort();\n  // TODO(cais): Check `weights` as well.\n  if (setX.length > 1) {\n    throw new ValueError(\n        `All input Tensors (x) should have the same number of samples. ` +\n        `Got array shapes: ` +\n        `${JSON.stringify(inputs.map(input => input.shape))}`);\n  }\n  if (setY.length > 1) {\n    throw new ValueError(\n        `All target Tensors (y) should have the same number of samples. ` +\n        `Got array shapes: ` +\n        `${JSON.stringify(targets.map(target => target.shape))}`);\n  }\n  if (setX.length > 0 && setY.length > 0 && !util.arraysEqual(setX, setY)) {\n    throw new ValueError(\n        `Input Tensors should have the same number of samples as target ` +\n        `Tensors. Found ${setX[0]} input sample(s) and ${setY[0]} target ` +\n        `sample(s).`);\n  }\n}\n\n/**\n * Validation on the compatibility of targes and loss functions.\n *\n * This helps prevent users from using loss functions incorrectly.\n *\n * @param targets `Array` of `tf.Tensor`s of targets.\n * @param lossFns `Array` of loss functions.\n * @param outputShapes `Array` of shapes of model outputs.\n */\nfunction checkLossAndTargetCompatibility(\n    targets: Tensor[], lossFns: LossOrMetricFn[], outputShapes: Shape[]) {\n  // TODO(cais): Dedicated test coverage?\n  const keyLosses = [\n    losses.meanSquaredError, losses.binaryCrossentropy,\n    losses.categoricalCrossentropy\n  ];\n  for (let i = 0; i < targets.length; ++i) {\n    const y = targets[i];\n    const loss = lossFns[i];\n    const shape = outputShapes[i];\n    if (loss == null) {\n      continue;\n    }\n    if (loss === losses.categoricalCrossentropy) {\n      if (y.shape[y.shape.length - 1] === 1) {\n        throw new ValueError(\n            `You are passing a target array of shape ${y.shape} while using ` +\n            `a loss 'categorical_crossentropy'. 'categorical_crossentropy'` +\n            `expects targets to be binary matrices (1s and 0s) of shape ` +\n            `[samples, classes].`);\n        // TODO(cais): Example code in error message.\n      }\n    }\n    if (keyLosses.indexOf(loss) !== -1) {\n      const slicedYShape = y.shape.slice(1);\n      const slicedShape = shape.slice(1);\n      for (let j = 0; j < slicedYShape.length; ++j) {\n        const targetDim = slicedYShape[j];\n        const outDim = slicedShape[j];\n        if (outDim != null && targetDim !== outDim) {\n          throw new ValueError(\n              `A target Tensor with shape ${y.shape} was passed for an ` +\n              `output of shape ${shape}, while using a loss function that ` +\n              `expects targets to have the same shape as the output.`);\n        }\n      }\n    }\n  }\n}\n\n/**\n * Check inputs provided by the user.\n *\n * Porting Note: This corresponds to _standardize_input_data() in Python\n *   Keras. Because of the strong typing in TF.js, we do not need to convert\n *   the data. Specifically:\n *   1) in PyKeras, `data` can be `DataFrame` instances from pandas, for\n *      example. We don't need to worry about that here because there is no\n *      widely popular javascript/typesdcript equivalent of pandas (so far).\n *      If one becomes available in the future, we can add support.\n *   2) in PyKeras, inputs can be Python dict. But here we are stipulating\n * that the data is either a single `tf.Tensor` or an Array of `tf.Tensor`s. We\n * may add support for `Object` data inputs in the future when the need\n * arises.\n *\n * Instead, we perform basic checks for number of parameters and shapes.\n *\n * @param data: The input data.\n * @param names: Name for the inputs, from the model.\n * @param shapes: Expected shapes for the input data, from the model.\n * @param checkBatchAxis: Whether the size along the batch axis (i.e., the\n *   first dimension) will be checked for matching.\n * @param exceptionPrefix: Execption prefix message, used in generating error\n *   messages.\n * @throws ValueError: on incorrect number of inputs or mismatches in shapes.\n */\nfunction checkInputData(\n    data: Tensor|Tensor[], names: string[], shapes?: Shape[],\n    checkBatchAxis = true, exceptionPrefix = '') {\n  let arrays: Tensor[];\n  if (Array.isArray(data)) {\n    if (data.length !== names.length) {\n      throw new ValueError(\n          `Error when checking model ${exceptionPrefix}: the Array of ` +\n          `Tensors that you are passing to your model is not the size the ` +\n          `the model expected. Expected to see ${names.length} Tensor(s),` +\n          ` but instead got ${data.length} Tensors(s).`);\n    }\n    arrays = data;\n  } else {\n    if (names.length > 1) {\n      throw new ValueError(\n          `The model expects ${names.length} ${exceptionPrefix} Tensors, ` +\n          `but only received one Tensor. Found: array with shape ` +\n          `${JSON.stringify(data.shape)}.`);\n    }\n    arrays = [data];\n  }\n\n  if (shapes != null) {\n    for (let i = 0; i < names.length; ++i) {\n      if (shapes[i] == null) {\n        continue;\n      }\n      const array = arrays[i];\n      if (array.shape.length !== shapes[i].length) {\n        throw new ValueError(\n            `Error when checking ${exceptionPrefix}: expected ${names[i]} ` +\n            `to have ${shapes[i].length} dimension(s), but got array with ` +\n            `shape ${JSON.stringify(array.shape)}`);\n      }\n      for (let j = 0; j < shapes[i].length; ++j) {\n        if (j === 0 && !checkBatchAxis) {\n          continue;\n        }\n        const dim = array.shape[j];\n        const refDim = shapes[i][j];\n        if (refDim != null) {\n          if (refDim !== dim) {\n            throw new ValueError(\n                `Error when checking ${exceptionPrefix}: expected ` +\n                `${names[i]} to have shape ${JSON.stringify(shapes[i])} but ` +\n                `got array with shape ${JSON.stringify(array.shape)}.`);\n          }\n        }\n      }\n    }\n  }\n}\n\n/**\n * Maps metric functions to model outputs.\n * @param metrics An shortcut strings name, metric function, `Array` or dict\n *   (`Object`) of metric functions.\n * @param outputNames An `Array` of the names of model outputs.\n * @returns An `Array` (one entry per model output) of `Array` of metric\n *   functions. For instance, if the model has 2 outputs, and for the first\n *   output we want to compute `binaryAccuracy` and `binaryCrossentropy`,\n *   and just `binaryAccuracy` for the second output, the `Array` would look\n *   like:\n *     `[[binaryAccuracy, binaryCrossentropy],  [binaryAccuracy]]`\n * @throws TypeError: incompatible metrics format.\n */\nexport function collectMetrics(\n    metrics: string|LossOrMetricFn|Array<string|LossOrMetricFn>|\n    {[outputName: string]: string | LossOrMetricFn},\n    outputNames: string[]): Array<Array<string|LossOrMetricFn>> {\n  if (metrics == null || Array.isArray(metrics) && metrics.length === 0) {\n    return outputNames.map(name => []);\n  }\n\n  let wrappedMetrics: Array<string|LossOrMetricFn>|\n      {[outputName: string]: string | LossOrMetricFn};\n  if (typeof metrics === 'string' || typeof metrics === 'function') {\n    wrappedMetrics = [metrics];\n  } else if (Array.isArray(metrics) || typeof metrics === 'object') {\n    wrappedMetrics = metrics as Array<string|LossOrMetricFn>|\n        {[outputName: string]: string} | {[outputName: string]: LossOrMetricFn};\n  } else {\n    throw new TypeError(\n        'Type of metrics argument not understood. Expected an string,' +\n        `function, Array, or Object, found: ${metrics}`);\n  }\n\n  if (Array.isArray(wrappedMetrics)) {\n    // We then apply all metrics to all outputs.\n    return outputNames.map(\n        name => wrappedMetrics as Array<string|LossOrMetricFn>);\n  } else {\n    // In this case, metrics is a dict.\n    const nestedMetrics: Array<Array<string|LossOrMetricFn>> = [];\n    for (const name of outputNames) {\n      let outputMetrics: string|LossOrMetricFn|Array<string|LossOrMetricFn> =\n          wrappedMetrics.hasOwnProperty(name) ? wrappedMetrics[name] : [];\n      if (!Array.isArray(outputMetrics)) {\n        outputMetrics = [outputMetrics];\n      }\n      nestedMetrics.push(outputMetrics);\n    }\n    return nestedMetrics;\n  }\n}\n\nexport interface ModelEvaluateArgs {\n  /**\n   * Batch size (Integer). If unspecified, it will default to 32.\n   */\n  batchSize?: number;\n\n  /**\n   * Verbosity mode.\n   */\n  verbose?: ModelLoggingVerbosity;\n\n  /**\n   * Tensor of weights to weight the contribution of different samples to the\n   * loss and metrics.\n   */\n  sampleWeight?: Tensor;\n\n  /**\n   * integer: total number of steps (batches of samples)\n   * before declaring the evaluation round finished. Ignored with the default\n   * value of `undefined`.\n   */\n  steps?: number;\n}\n\n/**\n * Configuration for calls to `LayersModel.compile()`.\n */\nexport interface ModelCompileArgs {\n  /**\n   * An instance of `tf.train.Optimizer` or a string name for an Optimizer.\n   */\n  optimizer: string|Optimizer;\n\n  /**\n   * Object function(s) or name(s) of object function(s).\n   * If the model has multiple outputs, you can use a different loss\n   * on each output by passing a dictionary or an Array of losses.\n   * The loss value that will be minimized by the model will then be the sum\n   * of all individual losses.\n   */\n  loss: string|string[]|{[outputName: string]: string}|LossOrMetricFn|\n      LossOrMetricFn[]|{[outputName: string]: LossOrMetricFn};\n\n  /**\n   * List of metrics to be evaluated by the model during training and testing.\n   * Typically you will use `metrics=['accuracy']`.\n   * To specify different metrics for different outputs of a multi-output\n   * model, you could also pass a dictionary.\n   */\n  metrics?: string|LossOrMetricFn|Array<string|LossOrMetricFn>|\n      {[outputName: string]: string | LossOrMetricFn};\n\n  // TODO(cais): Add lossWeights, sampleWeightMode, weightedMetrics, and\n  //   targetTensors.\n}\n\nconst LAYERS_MODEL_FORMAT_NAME = 'layers-model';\n\n/**\n * A `tf.LayersModel` is a directed, acyclic graph of `tf.Layer`s plus methods\n * for training, evaluation, prediction and saving.\n *\n * `tf.LayersModel` is the basic unit of training, inference and evaluation in\n * TensorFlow.js. To create a `tf.LayersModel`, use `tf.LayersModel`.\n *\n * See also:\n *   `tf.Sequential`, `tf.loadLayersModel`.\n *\n * @doc {heading: 'Models', subheading: 'Classes'}\n */\nexport class LayersModel extends Container implements tfc.InferenceModel {\n  // The class name is 'Model' rather than 'LayersModel' for backwards\n  // compatibility since this class name shows up in the serialization format.\n  /** @nocollapse */\n  static className = 'Model';\n  protected optimizer_: Optimizer;\n  // Whether the model instance owns the optimizer: `true` if and only if\n  // `optimizer` is created from a string parameter during `compile()` call.\n  protected isOptimizerOwned: boolean;\n\n  loss: string|string[]|{[outputName: string]: string}|LossOrMetricFn|\n      LossOrMetricFn[]|{[outputName: string]: LossOrMetricFn};\n  lossFunctions: LossOrMetricFn[];\n\n  // TODO(cais): These private variables should probably not have the string\n  //   'feed' in their names, because we are not dealing with a symbolic\n  //   backend.\n  private feedOutputShapes: Shape[];\n  private feedLossFns: LossOrMetricFn[];\n  private collectedTrainableWeights: LayerVariable[];\n  private testFunction: (data: Tensor[]) => Scalar[];\n  history: History;\n\n  // A public property that can be set by Callbacks to order early stopping\n  // during `fit()` calls.\n  protected stopTraining_: boolean;\n  protected isTraining: boolean;\n\n  metrics: string|LossOrMetricFn|Array<string|LossOrMetricFn>|\n      {[outputName: string]: string | LossOrMetricFn};\n  metricsNames: string[];\n  // Porting Note: `metrics_tensors` in PyKeras is a symbolic tensor. But given\n  //   the imperative nature of tfjs-core, `metricsTensors` is a\n  //   TypeScript function here.\n  //   Also note that due to the imperative nature of tfjs-core, `metricsTensor`\n  //   here needs an output index to keep track of which output of the\n  //   LayersModel a metric belongs to. This is unlike `metrics_tensors` in\n  //   PyKeras, which is a `list` of symbolic tensors, each of which has\n  //   implicit \"knowledge\" of the outputs it depends on.\n  metricsTensors: Array<[LossOrMetricFn, number]>;\n\n  // User defind metadata (if any).\n  private userDefinedMetadata: {};\n\n  constructor(args: ContainerArgs) {\n    super(args);\n    this.isTraining = false;\n  }\n\n  /**\n   * Print a text summary of the model's layers.\n   *\n   * The summary includes\n   * - Name and type of all layers that comprise the model.\n   * - Output shape(s) of the layers\n   * - Number of weight parameters of each layer\n   * - If the model has non-sequential-like topology, the inputs each layer\n   *   receives\n   * - The total number of trainable and non-trainable parameters of the model.\n   *\n   * ```js\n   * const input1 = tf.input({shape: [10]});\n   * const input2 = tf.input({shape: [20]});\n   * const dense1 = tf.layers.dense({units: 4}).apply(input1);\n   * const dense2 = tf.layers.dense({units: 8}).apply(input2);\n   * const concat = tf.layers.concatenate().apply([dense1, dense2]);\n   * const output =\n   *     tf.layers.dense({units: 3, activation: 'softmax'}).apply(concat);\n   *\n   * const model = tf.model({inputs: [input1, input2], outputs: output});\n   * model.summary();\n   * ```\n   *\n   * @param lineLength Custom line length, in number of characters.\n   * @param positions Custom widths of each of the columns, as either\n   *   fractions of `lineLength` (e.g., `[0.5, 0.75, 1]`) or absolute number\n   *   of characters (e.g., `[30, 50, 65]`). Each number corresponds to\n   *   right-most (i.e., ending) position of a column.\n   * @param printFn Custom print function. Can be used to replace the default\n   *   `console.log`. For example, you can use `x => {}` to mute the printed\n   *   messages in the console.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  summary(\n      lineLength?: number, positions?: number[],\n      printFn:\n          // tslint:disable-next-line:no-any\n      (message?: any, ...optionalParams: any[]) => void = console.log) {\n    if (!this.built) {\n      throw new ValueError(\n          `This model has never been called, thus its weights have not been ` +\n          `created yet. So no summary can be displayed. Build the model ` +\n          `first (e.g., by calling it on some test data).`);\n    }\n    printSummary(this, lineLength, positions, printFn);\n  }\n\n  /**\n   * Configures and prepares the model for training and evaluation.  Compiling\n   * outfits the model with an optimizer, loss, and/or metrics.  Calling `fit`\n   * or `evaluate` on an un-compiled model will throw an error.\n   *\n   * @param args a `ModelCompileArgs` specifying the loss, optimizer, and\n   * metrics to be used for fitting and evaluating this model.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  compile(args: ModelCompileArgs): void {\n    if (args.loss == null) {\n      args.loss = [];\n    }\n    this.loss = args.loss;\n\n    if (typeof args.optimizer === 'string') {\n      this.optimizer_ = optimizers.getOptimizer(args.optimizer);\n      this.isOptimizerOwned = true;\n    } else {\n      if (!(args.optimizer instanceof Optimizer)) {\n        throw new ValueError(\n            `User-defined optimizer must be an instance of tf.Optimizer.`);\n      }\n      this.optimizer_ = args.optimizer;\n      this.isOptimizerOwned = false;\n    }\n\n    // TODO(cais): Add lossWeights.\n    // TODO(cais): Add sampleWeightMode.\n\n    // Prepare loss functions.\n    let lossFunctions: LossOrMetricFn[] = [];\n    if (!Array.isArray(args.loss) && typeof args.loss !== 'string' &&\n        typeof args.loss !== 'function') {\n      args.loss = args.loss as {[outputName: string]: string};\n      for (const name in args.loss) {\n        if (this.outputNames.indexOf(name) === -1) {\n          throw new ValueError(\n              `Unknown entry in loss dictionary: \"${name}\". ` +\n              `Only expected the following keys: ${this.outputNames}`);\n        }\n      }\n      for (const name of this.outputNames) {\n        if (args.loss[name] == null) {\n          console.warn(\n              `Output \"${name}\" is missing from loss dictionary. We assume ` +\n              `this was done on purpose, and we will not be expecting data ` +\n              `to be passed to ${name} during training`);\n        }\n        lossFunctions.push(losses.get(args.loss[name]));\n      }\n    } else if (Array.isArray(args.loss)) {\n      if (args.loss.length !== this.outputs.length) {\n        throw new ValueError(\n            `When passing an Array as loss, it should have one entry per ` +\n            `model output. The model has ${this.outputs.length} output(s), ` +\n            `but you passed loss=${args.loss}.`);\n      }\n      const theLosses = args.loss as Array<string|LossOrMetricFn>;\n      lossFunctions = theLosses.map(l => losses.get(l));\n    } else {\n      const lossFunction = losses.get(args.loss);\n      this.outputs.forEach(_ => {\n        lossFunctions.push(lossFunction);\n      });\n    }\n\n    this.lossFunctions = lossFunctions;\n\n    this.feedOutputNames = [];\n    this.feedOutputShapes = [];\n    this.feedLossFns = [];\n    for (let i = 0; i < this.outputs.length; ++i) {\n      // TODO(cais): Logic for skipping target(s).\n      const shape = this.internalOutputShapes[i];\n      const name = this.outputNames[i];\n      this.feedOutputNames.push(name);\n      this.feedOutputShapes.push(shape);\n      this.feedLossFns.push(this.lossFunctions[i]);\n    }\n\n    // TODO(cais): Add logic for output masks.\n    // TODO(cais): Add logic for sample weights.\n    const skipTargetIndices: number[] = [];\n\n    // Prepare metrics.\n    this.metrics = args.metrics;\n    // TODO(cais): Add weightedMetrics.\n    this.metricsNames = ['loss'];\n    this.metricsTensors = [];\n\n    // Compute total loss.\n    // Porting Note: In PyKeras, metrics_tensors are symbolic tensor objects.\n    //   Here, metricsTensors are TypeScript functions. This difference is due\n    //   to the difference in symbolic/imperative property of the backends.\n    nameScope('loss', () => {\n      for (let i = 0; i < this.outputs.length; ++i) {\n        if (skipTargetIndices.indexOf(i) !== -1) {\n          continue;\n        }\n        // TODO(cais): Add weightedLoss, sampleWeight and mask.\n        //   The following line should be weightedLoss\n        const weightedLoss = this.lossFunctions[i];\n        if (this.outputs.length > 1) {\n          this.metricsTensors.push([weightedLoss, i]);\n          this.metricsNames.push(this.outputNames[i] + '_loss');\n        }\n      }\n\n      // Porting Note: Due to the imperative nature of the backend, we calculate\n      //   the regularizer penalties in the totalLossFunction, instead of here.\n    });\n\n    const nestedMetrics = collectMetrics(args.metrics, this.outputNames);\n    // TODO(cais): Add nestedWeightedMetrics.\n\n    /**\n     * Helper function used in loop below.\n     */\n    const appendMetric =\n        (outputIndex: number, metricName: string,\n         metricTensor: LossOrMetricFn) => {\n          if (this.outputNames.length > 1) {\n            metricName = this.outputNames[outputIndex] + '_' + metricName;\n          }\n          this.metricsNames.push(metricName);\n          this.metricsTensors.push([metricTensor, outputIndex]);\n        };\n\n    nameScope('metric', () => {\n      for (let i = 0; i < this.outputs.length; ++i) {\n        if (skipTargetIndices.indexOf(i) !== -1) {\n          continue;\n        }\n        const outputMetrics = nestedMetrics[i];\n        // TODO(cais): Add weights and outputWeightedMetrics.\n\n        // TODO(cais): Add optional arg `weights` to the following function.\n        const handleMetrics = (metrics: Array<string|LossOrMetricFn>) => {\n          const metricNamePrefix = '';\n          let metricName: string;\n          let accFn: LossOrMetricFn;\n          let weightedMetricFn: LossOrMetricFn;\n          //  TODO(cais): Use 'weights_' for weighted metrics.\n\n          for (const metric of metrics) {\n            if (typeof metric === 'string' &&\n                ['accuracy', 'acc', 'crossentropy', 'ce'].indexOf(metric) !==\n                    -1) {\n              const outputShape = this.internalOutputShapes[i];\n\n              if (outputShape[outputShape.length - 1] === 1 ||\n                  this.lossFunctions[i] === losses.binaryCrossentropy) {\n                // case: binary accuracy/crossentropy.\n                if (['accuracy', 'acc'].indexOf(metric) !== -1) {\n                  accFn = Metrics.binaryAccuracy;\n                } else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {\n                  accFn = Metrics.binaryCrossentropy;\n                }\n              } else if (\n                  this.lossFunctions[i] ===\n                  losses.sparseCategoricalCrossentropy) {\n                // case: categorical accuracy / crossentropy with sparse\n                // targets.\n                if (['accuracy', 'acc'].indexOf(metric) !== -1) {\n                  accFn = Metrics.sparseCategoricalAccuracy;\n                } else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {\n                  accFn = Metrics.sparseCategoricalCrossentropy;\n                }\n              } else {\n                // case: categorical accuracy / crossentropy.\n                if (['accuracy', 'acc'].indexOf(metric) !== -1) {\n                  accFn = Metrics.categoricalAccuracy;\n                } else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {\n                  accFn = Metrics.categoricalCrossentropy;\n                }\n              }\n              let suffix: string;\n              if (['accuracy', 'acc'].indexOf(metric) !== -1) {\n                suffix = 'acc';\n              } else if (['crossentropy', 'ce'].indexOf(metric) !== -1) {\n                suffix = 'ce';\n              }\n              // TODO(cais): Add weighting actually.\n              weightedMetricFn = accFn;\n              metricName = metricNamePrefix + suffix;\n            } else {\n              const metricFn = Metrics.get(metric);\n              // TODO(cais): Add weighting actually.\n              weightedMetricFn = metricFn;\n              metricName =\n                  metricNamePrefix + Metrics.getLossOrMetricName(metric);\n            }\n\n            // TODO(cais): Add weighting and masking to metricResult.\n            let metricResult: LossOrMetricFn;\n            nameScope(metricName, () => {\n              metricResult = weightedMetricFn;\n            });\n            appendMetric(i, metricName, metricResult);\n          }\n        };\n\n        handleMetrics(outputMetrics);\n        // TODO(cais): Call handleMetrics with weights.\n      }\n    });\n\n    // Porting Notes: Given the imperative backend of tfjs-core,\n    //   there is no need for constructing the symbolic graph and placeholders.\n    this.collectedTrainableWeights = this.trainableWeights;\n  }\n\n  /**\n   * Check trainable weights count consistency.\n   *\n   * This will raise a warning if `this.trainableWeights` and\n   * `this.collectedTrainableWeights` are inconsistent (i.e., have different\n   * numbers of parameters).\n   * Inconsistency will typically arise when one modifies `model.trainable`\n   * without calling `model.compile()` again.\n   */\n  protected checkTrainableWeightsConsistency(): void {\n    if (this.collectedTrainableWeights == null) {\n      return;\n    }\n    if (this.trainableWeights.length !==\n        this.collectedTrainableWeights.length) {\n      console.warn(\n          'Discrepancy between trainableweights and collected trainable ' +\n          'weights. Did you set `model.trainable` without calling ' +\n          '`model.compile()` afterwards?');\n    }\n  }\n\n  /**\n   * Returns the loss value & metrics values for the model in test mode.\n   *\n   * Loss and metrics are specified during `compile()`, which needs to happen\n   * before calls to `evaluate()`.\n   *\n   * Computation is done in batches.\n   *\n   * ```js\n   * const model = tf.sequential({\n   *   layers: [tf.layers.dense({units: 1, inputShape: [10]})]\n   * });\n   * model.compile({optimizer: 'sgd', loss: 'meanSquaredError'});\n   * const result = model.evaluate(\n   *     tf.ones([8, 10]), tf.ones([8, 1]), {batchSize: 4});\n   * result.print();\n   * ```\n   *\n   * @param x `tf.Tensor` of test data, or an `Array` of `tf.Tensor`s if the\n   * model has multiple inputs.\n   * @param y `tf.Tensor` of target data, or an `Array` of `tf.Tensor`s if the\n   * model has multiple outputs.\n   * @param args A `ModelEvaluateArgs`, containing optional fields.\n   *\n   * @return `Scalar` test loss (if the model has a single output and no\n   *   metrics) or `Array` of `Scalar`s (if the model has multiple outputs\n   *   and/or metrics). The attribute `model.metricsNames`\n   *   will give you the display labels for the scalar outputs.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  evaluate(\n      x: Tensor|Tensor[], y: Tensor|Tensor[],\n      args: ModelEvaluateArgs = {}): Scalar|Scalar[] {\n    const batchSize = args.batchSize == null ? 32 : args.batchSize;\n    checkBatchSize(batchSize);\n\n    // TODO(cais): Standardize `config.sampleWeights` as well.\n    // Validate user data.\n    const checkBatchAxis = true;\n    const standardizedOuts =\n        this.standardizeUserDataXY(x, y, checkBatchAxis, batchSize);\n    try {\n      // TODO(cais): If uses `useLearningPhase`, set the corresponding element\n      // of the input to 0.\n      const ins = standardizedOuts[0].concat(standardizedOuts[1]);\n      this.makeTestFunction();\n      const f = this.testFunction;\n      const testOuts =\n          this.testLoop(f, ins, batchSize, args.verbose, args.steps);\n      return singletonOrArray(testOuts);\n    } finally {\n      disposeNewTensors(standardizedOuts[0], x);\n      disposeNewTensors(standardizedOuts[1], y);\n    }\n  }\n\n  // TODO(cais): Add code snippet below once real dataset objects are\n  //   available.\n  /**\n   * Evaluate model using a dataset object.\n   *\n   * Note: Unlike `evaluate()`, this method is asynchronous (`async`);\n   *\n   * @param dataset A dataset object. Its `iterator()` method is expected\n   *   to generate a dataset iterator object, the `next()` method of which\n   *   is expected to produce data batches for evaluation. The return value\n   *   of the `next()` call ought to contain a boolean `done` field and a\n   *   `value` field. The `value` field is expected to be an array of two\n   *   `tf.Tensor`s or an array of two nested `tf.Tensor` structures. The former\n   *   case is for models with exactly one input and one output (e.g..\n   *   a sequential model). The latter case is for models with multiple\n   *   inputs and/or multiple outputs. Of the two items in the array, the\n   *   first is the input feature(s) and the second is the output target(s).\n   * @param args A configuration object for the dataset-based evaluation.\n   * @returns Loss and metric values as an Array of `Scalar` objects.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  async evaluateDataset(dataset: Dataset<{}>, args?: ModelEvaluateDatasetArgs):\n      Promise<Scalar|Scalar[]> {\n    this.makeTestFunction();\n    return evaluateDataset(this, dataset, args);\n  }\n\n  /**\n   * Get number of samples provided for training, evaluation or prediction.\n   *\n   * @param ins Input `tf.Tensor`.\n   * @param batchSize Integer batch size, optional.\n   * @param steps Total number of steps (batches of samples) before\n   * declaring loop finished. Optional.\n   * @param stepsName The public API's parameter name for `steps`.\n   * @returns Number of samples provided.\n   */\n  private checkNumSamples(\n      ins: Tensor|Tensor[], batchSize?: number, steps?: number,\n      stepsName = 'steps'): number {\n    let numSamples: number;\n    if (steps != null) {\n      numSamples = null;\n      if (batchSize != null) {\n        throw new ValueError(\n            `If ${stepsName} is set, batchSize must be null or undefined.` +\n            `Got batchSize = ${batchSize}`);\n      }\n    } else if (ins != null) {\n      if (Array.isArray(ins)) {\n        numSamples = ins[0].shape[0];\n      } else {\n        numSamples = ins.shape[0];\n      }\n    } else {\n      throw new ValueError(\n          `Either the input data should have a defined shape, or ` +\n          `${stepsName} shoud be specified.`);\n    }\n    return numSamples;\n  }\n\n  /**\n   * Execute internal tensors of the model with input data feed.\n   * @param inputs Input data feed. Must match the inputs of the model.\n   * @param outputs Names of the output tensors to be fetched. Must match\n   *   names of the SymbolicTensors that belong to the graph.\n   * @returns Fetched values for `outputs`.\n   */\n  execute(inputs: Tensor|Tensor[]|NamedTensorMap, outputs: string|string[]):\n      Tensor|Tensor[] {\n    if (Array.isArray(outputs) && outputs.length === 0) {\n      throw new ValueError(\n          '`outputs` is an empty Array, which is not allowed.');\n    }\n\n    const outputsIsArray = Array.isArray(outputs);\n    const outputNames =\n        (outputsIsArray ? outputs as string[] : [outputs as string]);\n    const outputSymbolicTensors = this.retrieveSymbolicTensors(outputNames);\n\n    // Format the input into a FeedDict.\n    const feedDict = new FeedDict();\n    if (inputs instanceof Tensor) {\n      inputs = [inputs];\n    }\n    if (Array.isArray(inputs)) {\n      if (inputs.length !== this.inputs.length) {\n        throw new ValueError(\n            `The number of inputs provided (${inputs.length}) ` +\n            `does not match the number of inputs of this model ` +\n            `(${this.inputs.length}).`);\n      }\n      for (let i = 0; i < this.inputs.length; ++i) {\n        feedDict.add(this.inputs[i], inputs[i]);\n      }\n    } else {\n      for (const input of this.inputs) {\n        const tensorValue = inputs[input.name];\n        if (tensorValue == null) {\n          throw new ValueError(\n              `No value is provided for the model's input ${input.name}`);\n        }\n        feedDict.add(input, tensorValue);\n      }\n    }\n\n    // Run execution.\n    const executeOutputs = execute(outputSymbolicTensors, feedDict) as Tensor[];\n    return outputsIsArray ? executeOutputs : executeOutputs[0];\n  }\n\n  /**\n   * Retrieve the model's internal symbolic tensors from symbolic-tensor names.\n   */\n  private retrieveSymbolicTensors(symbolicTensorNames: string[]):\n      SymbolicTensor[] {\n    const outputSymbolicTensors: SymbolicTensor[] =\n        pyListRepeat(null, symbolicTensorNames.length);\n    let outputsRemaining = symbolicTensorNames.length;\n    for (const layer of this.layers) {\n      const layerOutputs: SymbolicTensor[] =\n          Array.isArray(layer.output) ? layer.output : [layer.output];\n      const layerOutputNames = layerOutputs.map(output => output.name);\n      for (let i = 0; i < symbolicTensorNames.length; ++i) {\n        const index = layerOutputNames.indexOf(symbolicTensorNames[i]);\n        if (index !== -1) {\n          outputSymbolicTensors[i] = layerOutputs[index];\n          outputsRemaining--;\n        }\n        if (outputsRemaining === 0) {\n          break;\n        }\n      }\n      if (outputsRemaining === 0) {\n        break;\n      }\n    }\n\n    if (outputsRemaining > 0) {\n      const remainingNames: string[] = [];\n      outputSymbolicTensors.forEach((tensor, i) => {\n        if (tensor == null) {\n          remainingNames.push(symbolicTensorNames[i]);\n        }\n      });\n      throw new ValueError(\n          `Cannot find SymbolicTensors for output name(s): ` +\n          `${JSON.stringify(remainingNames)}`);\n    }\n    return outputSymbolicTensors;\n  }\n\n  /**\n   * Helper method to loop over some data in batches.\n   *\n   * Porting Note: Not using the functional approach in the Python equivalent\n   *   due to the imperative backend.\n   * Porting Note: Does not support step mode currently.\n   *\n   * @param ins: input data\n   * @param batchSize: integer batch size.\n   * @param verbose: verbosity model\n   * @returns: Predictions as `tf.Tensor` (if a single output) or an `Array` of\n   *   `tf.Tensor` (if multipe outputs).\n   */\n  private predictLoop(ins: Tensor|Tensor[], batchSize = 32, verbose = false):\n      Tensor|Tensor[] {\n    return tfc.tidy(() => {\n      const numSamples = this.checkNumSamples(ins);\n      if (verbose) {\n        throw new NotImplementedError(\n            'Verbose predictLoop() is not implemented yet.');\n      }\n\n      // Sample-based predictions.\n      // Porting Note: Tensor currently does not support sliced assignments as\n      //   in numpy, e.g., x[1:3] = y. Therefore we use concatenation while\n      //   iterating over the batches.\n\n      const batches = makeBatches(numSamples, batchSize);\n      const outsBatches: Tensor[][] = this.outputs.map(output => []);\n\n      // TODO(cais): Can the scope() be pushed down inside the for loop?\n      for (let batchIndex = 0; batchIndex < batches.length; ++batchIndex) {\n        const batchOuts = tfc.tidy(() => {\n          const batchStart = batches[batchIndex][0];\n          const batchEnd = batches[batchIndex][1];\n          // TODO(cais): Take care of the case of the last element is a flag for\n          //   training/test.\n          const insBatch = sliceArrays(ins, batchStart, batchEnd);\n\n          // Construct the feeds for execute();\n          const feeds = [];\n          if (Array.isArray(insBatch)) {\n            for (let i = 0; i < insBatch.length; ++i) {\n              feeds.push({key: this.inputs[i], value: insBatch[i]});\n            }\n          } else {\n            feeds.push({key: this.inputs[0], value: insBatch});\n          }\n          const feedDict = new FeedDict(feeds);\n          return execute(this.outputs, feedDict) as Tensor[];\n        });\n        batchOuts.forEach((batchOut, i) => outsBatches[i].push(batchOut));\n      }\n      return singletonOrArray(\n          outsBatches.map(batches => tfc.concat(batches, 0)));\n    });\n  }\n\n  /**\n   * Generates output predictions for the input samples.\n   *\n   * Computation is done in batches.\n   *\n   * Note: the \"step\" mode of predict() is currently not supported.\n   *   This is because the TensorFlow.js core backend is imperative only.\n   *\n   * ```js\n   * const model = tf.sequential({\n   *   layers: [tf.layers.dense({units: 1, inputShape: [10]})]\n   * });\n   * model.predict(tf.ones([8, 10]), {batchSize: 4}).print();\n   * ```\n   *\n   * @param x The input data, as a Tensor, or an `Array` of `tf.Tensor`s if\n   *   the model has multiple inputs.\n   * @param args A `ModelPredictArgs` object containing optional fields.\n   *\n   * @return Prediction results as a `tf.Tensor`(s).\n   *\n   * @exception ValueError In case of mismatch between the provided input data\n   *   and the model's expectations, or in case a stateful model receives a\n   *   number of samples that is not a multiple of the batch size.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  predict(x: Tensor|Tensor[], args: ModelPredictArgs = {}): Tensor|Tensor[] {\n    const xsRank2OrHigher = ensureTensorsRank2OrHigher(x);\n    checkInputData(\n        xsRank2OrHigher, this.inputNames, this.feedInputShapes, false);\n    try {\n      // TODO(cais): Take care of stateful models.\n      //   if (this.stateful) ...\n      // TODO(cais): Take care of the learning_phase boolean flag.\n      //   if (this.useLearningPhase) ...\n      const batchSize = args.batchSize == null ? 32 : args.batchSize;\n      checkBatchSize(batchSize);\n      return this.predictLoop(xsRank2OrHigher, batchSize);\n    } finally {\n      disposeNewTensors(xsRank2OrHigher, x);\n    }\n  }\n\n  /**\n   * Returns predictions for a single batch of samples.\n   *\n   * ```js\n   * const model = tf.sequential({\n   *   layers: [tf.layers.dense({units: 1, inputShape: [10]})]\n   * });\n   * model.predictOnBatch(tf.ones([8, 10])).print();\n   * ```\n   * @param x: Input samples, as a Tensor (for models with exactly one\n   *   input) or an array of Tensors (for models with more than one input).\n   * @return Tensor(s) of predictions\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  predictOnBatch(x: Tensor|Tensor[]): Tensor|Tensor[] {\n    checkInputData(x, this.inputNames, this.feedInputShapes, true);\n    // TODO(cais): Take care of the learning_phase boolean flag.\n    //   if (this.useLearningPhase) ...\n    const batchSize = (Array.isArray(x) ? x[0] : x).shape[0];\n    return this.predictLoop(x, batchSize);\n  }\n\n  protected standardizeUserDataXY(\n      x: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      y: Tensor|Tensor[]|{[inputName: string]: Tensor}, checkBatchAxis = true,\n      batchSize?: number): [Tensor[], Tensor[]] {\n    // TODO(cais): Add sampleWeight, classWeight\n    if (this.optimizer_ == null) {\n      throw new RuntimeError(\n          'You must compile a model before training/testing. Use ' +\n          'LayersModel.compile(modelCompileArgs).');\n    }\n    const outputShapes: Shape[] = [];\n    for (let i = 0; i < this.feedOutputShapes.length; ++i) {\n      const outputShape = this.feedOutputShapes[i];\n      const lossFn = this.feedLossFns[i];\n      if (lossFn === losses.sparseCategoricalCrossentropy) {\n        outputShapes.push(\n            outputShape.slice(0, outputShape.length - 1).concat([1]));\n      } else {\n        // Porting Note: Because of strong typing `lossFn` must be a function.\n        outputShapes.push(outputShape);\n      }\n    }\n    x = standardizeInputData(\n        x, this.feedInputNames, this.feedInputShapes, false, 'input');\n    y = standardizeInputData(\n        y, this.feedOutputNames, outputShapes, false, 'target');\n    // TODO(cais): Standardize sampleWeights & classWeights.\n    checkArrayLengths(x, y, null);\n    // TODO(cais): Check sampleWeights as well.\n    checkLossAndTargetCompatibility(y, this.feedLossFns, this.feedOutputShapes);\n    if (this.stateful && batchSize != null && batchSize > 0) {\n      if (x[0].shape[0] % batchSize !== 0) {\n        throw new ValueError(\n            `In a stateful network, you should only pass inputs with a ` +\n            `number of samples that is divisible by the batch size ` +\n            `${batchSize}. Found: ${x[0].shape[0]} sample(s).`);\n      }\n    }\n    return [x, y];\n  }\n\n  protected async standardizeUserData(\n      x: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      y: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      sampleWeight?: Tensor|Tensor[]|{[outputName: string]: Tensor},\n      classWeight?: ClassWeight|ClassWeight[]|ClassWeightMap,\n      checkBatchAxis = true,\n      batchSize?: number): Promise<[Tensor[], Tensor[], Tensor[]]> {\n    const [standardXs, standardYs] =\n        this.standardizeUserDataXY(x, y, checkBatchAxis, batchSize);\n    // TODO(cais): Handle sampleWeights.\n    if (sampleWeight != null) {\n      throw new Error('sample weight is not supported yet.');\n    }\n\n    let standardSampleWeights: Tensor[] = null;\n    if (classWeight != null) {\n      const classWeights =\n          standardizeClassWeights(classWeight, this.outputNames);\n      standardSampleWeights = [];\n      for (let i = 0; i < classWeights.length; ++i) {\n        standardSampleWeights.push(\n            await standardizeWeights(standardYs[i], null, classWeights[i]));\n      }\n    }\n\n    // TODO(cais): Deal with the case of model.stateful == true.\n    return [standardXs, standardYs, standardSampleWeights];\n  }\n\n  /**\n   * Loop over some test data in batches.\n   * @param f A Function returning a list of tensors.\n   * @param ins Array of tensors to be fed to `f`.\n   * @param batchSize Integer batch size or `null` / `undefined`.\n   * @param verbose verbosity mode.\n   * @param steps Total number of steps (batches of samples) before\n   * declaring test finished. Ignored with the default value of `null` /\n   * `undefined`.\n   * @returns Array of Scalars.\n   */\n  private testLoop(\n      f: (data: Tensor[]) => Scalar[], ins: Tensor[], batchSize?: number,\n      verbose = 0, steps?: number): Scalar[] {\n    return tfc.tidy(() => {\n      const numSamples = this.checkNumSamples(ins, batchSize, steps, 'steps');\n      const outs: Scalar[] = [];\n      if (verbose > 0) {\n        throw new NotImplementedError('Verbose mode is not implemented yet.');\n      }\n      // TODO(cais): Use `indicesForConversionToDense' to prevent slow down.\n      if (steps != null) {\n        throw new NotImplementedError(\n            'steps mode in testLoop() is not implemented yet');\n      } else {\n        const batches = makeBatches(numSamples, batchSize);\n        const indexArray = tensor1d(range(0, numSamples));\n        for (let batchIndex = 0; batchIndex < batches.length; ++batchIndex) {\n          const batchStart = batches[batchIndex][0];\n          const batchEnd = batches[batchIndex][1];\n          const batchIds =\n              K.sliceAlongFirstAxis(\n                  indexArray, batchStart, batchEnd - batchStart) as Tensor1D;\n          // TODO(cais): In ins, train flag can be a number, instead of an\n          //   Tensor? Do we need to handle this in tfjs-layers?\n          const insBatch = sliceArraysByIndices(ins, batchIds) as Scalar[];\n          const batchOuts = f(insBatch);\n          if (batchIndex === 0) {\n            for (let i = 0; i < batchOuts.length; ++i) {\n              outs.push(scalar(0));\n            }\n          }\n          for (let i = 0; i < batchOuts.length; ++i) {\n            const batchOut = batchOuts[i];\n            outs[i] =\n                tfc.add(outs[i], tfc.mul(batchEnd - batchStart, batchOut));\n          }\n        }\n        for (let i = 0; i < outs.length; ++i) {\n          outs[i] = tfc.div(outs[i], numSamples);\n        }\n      }\n      return outs;\n    });\n  }\n\n  protected getDedupedMetricsNames(): string[] {\n    const outLabels = this.metricsNames;\n    // Rename duplicated metrics names (can happen with an output layer\n    // shared among multiple dataflows).\n    const dedupedOutLabels = [];\n    for (let i = 0; i < outLabels.length; ++i) {\n      const label = outLabels[i];\n      let newLabel = label;\n      if (count(outLabels, label) > 1) {\n        const dupIndex = count(outLabels.slice(0, i), label);\n        newLabel += `_${dupIndex}`;\n      }\n      dedupedOutLabels.push(newLabel);\n    }\n    return dedupedOutLabels;\n  }\n\n  /**\n   * Creates a function that performs the following actions:\n   *\n   * 1. computes the losses\n   * 2. sums them to get the total loss\n   * 3. call the optimizer computes the gradients of the LayersModel's\n   *    trainable weights w.r.t. the total loss and update the variables\n   * 4. calculates the metrics\n   * 5. returns the values of the losses and metrics.\n   */\n  protected makeTrainFunction(): (data: Tensor[]) => Scalar[] {\n    return (data: Tensor[]) => {\n      const lossValues: Scalar[] = [];\n\n      const inputs = data.slice(0, this.inputs.length);\n      const targets = data.slice(\n          this.inputs.length, this.inputs.length + this.outputs.length);\n      const sampleWeights = data.slice(\n          this.inputs.length + this.outputs.length,\n          this.inputs.length + this.outputs.length * 2);\n\n      const metricsValues: Scalar[] = [];\n\n      // Create a function that computes the total loss based on the\n      // inputs. This function is used for obtaining gradients through\n      // backprop.\n      const totalLossFunction = () => {\n        const feeds = [];\n        for (let i = 0; i < this.inputs.length; ++i) {\n          feeds.push({key: this.inputs[i], value: inputs[i]});\n        }\n        const feedDict = new FeedDict(feeds);\n        const outputs =\n            execute(this.outputs, feedDict, {'training': true}) as Tensor[];\n        // TODO(cais): Take care of the case of multiple outputs from a\n        //   single layer?\n\n        let totalLoss: Tensor;\n        for (let i = 0; i < this.lossFunctions.length; ++i) {\n          const lossFunction = this.lossFunctions[i];\n          let loss = lossFunction(targets[i], outputs[i]);\n          if (sampleWeights[i] != null) {\n            loss = computeWeightedLoss(loss, sampleWeights[i]);\n          }\n\n          // TODO(cais): push Scalar instead.\n          const meanLoss: Scalar = tfc.mean(loss);\n          // TODO(cais): Use a scope() instead, to avoid ownership.\n          lossValues.push(meanLoss);\n          if (i === 0) {\n            totalLoss = loss;\n          } else {\n            totalLoss = tfc.add(totalLoss, loss);\n          }\n        }\n\n        // Compute the metrics.\n        // TODO(cais): These should probably be calculated outside\n        //   totalLossFunction to benefit speed?\n        for (let i = 0; i < this.metricsTensors.length; ++i) {\n          let weightedMetric: Scalar;\n\n          if (this.outputs.length > 1 && i < this.outputs.length) {\n            weightedMetric = lossValues[i];\n          } else {\n            const metric = this.metricsTensors[i][0];\n            const outputIndex = this.metricsTensors[i][1];\n            weightedMetric =\n                tfc.mean(metric(targets[outputIndex], outputs[outputIndex]));\n          }\n\n          tfc.keep(weightedMetric);\n          // TODO(cais): Use a scope() instead, to avoid ownership.\n          metricsValues.push(weightedMetric);\n        }\n\n        totalLoss = tfc.mean(totalLoss);\n\n        // Add regularizer penalties.\n        this.calculateLosses().forEach(regularizerLoss => {\n          totalLoss = tfc.add(totalLoss, regularizerLoss);\n        });\n\n        return totalLoss as Scalar;\n      };\n\n      const variables = this.collectedTrainableWeights.map(\n          param => param.read() as tfc.Variable);\n      const returnCost = true;\n      const totalLossValue =\n          this.optimizer_.minimize(totalLossFunction, returnCost, variables);\n\n      return [totalLossValue].concat(metricsValues);\n    };\n  }\n\n  /**\n   * Create a function which, when invoked with an array of `tf.Tensor`s as a\n   * batch of inputs, returns the prespecified loss and metrics of the model\n   * under the batch of input data.\n   */\n  private makeTestFunction() {\n    this.testFunction = (data: Tensor[]) => {\n      return tfc.tidy(() => {\n        const valOutputs: Scalar[] = [];\n        let totalLoss: Scalar;\n        const inputs = data.slice(0, this.inputs.length);\n        const targets = data.slice(\n            this.inputs.length, this.inputs.length + this.outputs.length);\n        const feeds = [];\n        for (let i = 0; i < this.inputs.length; ++i) {\n          feeds.push({key: this.inputs[i], value: inputs[i]});\n        }\n        const feedDict = new FeedDict(feeds);\n        const outputs = execute(this.outputs, feedDict) as Tensor[];\n        // Compute total loss.\n        for (let i = 0; i < this.lossFunctions.length; ++i) {\n          const lossFunction = this.lossFunctions[i];\n          // TODO(cais): Add sample weighting and replace the simple\n          // averaging.\n          const loss: Scalar = tfc.mean(lossFunction(targets[i], outputs[i]));\n          if (i === 0) {\n            totalLoss = loss;\n          } else {\n            totalLoss = tfc.add(totalLoss, loss);\n          }\n          valOutputs.push(totalLoss);\n        }\n        // Compute the metrics.\n        for (let i = 0; i < this.metricsTensors.length; ++i) {\n          const metric = this.metricsTensors[i][0];\n          const outputIndex = this.metricsTensors[i][1];\n          // TODO(cais): Replace K.mean() with a proper weighting function.\n          const meanMetric =\n              tfc.mean(metric(targets[outputIndex], outputs[outputIndex]));\n          valOutputs.push(meanMetric as Scalar);\n        }\n        return valOutputs;\n      });\n    };\n  }\n\n  /**\n   * Trains the model for a fixed number of epochs (iterations on a\n   * dataset).\n   *\n   * ```js\n   * const model = tf.sequential({\n   *     layers: [tf.layers.dense({units: 1, inputShape: [10]})]\n   * });\n   * model.compile({optimizer: 'sgd', loss: 'meanSquaredError'});\n   * for (let i = 1; i < 5 ; ++i) {\n   *   const h = await model.fit(tf.ones([8, 10]), tf.ones([8, 1]), {\n   *       batchSize: 4,\n   *       epochs: 3\n   *   });\n   *   console.log(\"Loss after Epoch \" + i + \" : \" + h.history.loss[0]);\n   * }\n   * ```\n   *\n   * @param x `tf.Tensor` of training data, or an array of `tf.Tensor`s if the\n   * model has multiple inputs. If all inputs in the model are named, you\n   * can also pass a dictionary mapping input names to `tf.Tensor`s.\n   * @param y `tf.Tensor` of target (label) data, or an array of `tf.Tensor`s if\n   * the model has multiple outputs. If all outputs in the model are named,\n   * you can also pass a dictionary mapping output names to `tf.Tensor`s.\n   * @param args A `ModelFitArgs`, containing optional fields.\n   *\n   * @return A `History` instance. Its `history` attribute contains all\n   *   information collected during training.\n   *\n   * @exception ValueError In case of mismatch between the provided input\n   * data and what the model expects.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  async fit(\n      x: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      y: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      args: ModelFitArgs = {}): Promise<History> {\n    return fitTensors(this, x, y, args);\n  }\n\n  // TODO(cais): Add code snippet below when it's possible to instantiate\n  //   actual dataset objects.\n  /**\n   * Trains the model using a dataset object.\n   *\n   * @param dataset A dataset object. Its `iterator()` method is expected\n   *   to generate a dataset iterator object, the `next()` method of which\n   *   is expected to produce data batches for training. The return value\n   *   of the `next()` call ought to contain a boolean `done` field and a\n   *   `value` field. The `value` field is expected to be an array of two\n   *   `tf.Tensor`s or an array of two nested `tf.Tensor` structures. The former\n   *   case is for models with exactly one input and one output (e.g..\n   *   a sequential model). The latter case is for models with multiple\n   *   inputs and/or multiple outputs.\n   *   Of the two items in the array, the first is the input feature(s) and\n   *   the second is the output target(s).\n   * @param args A `ModelFitDatasetArgs`, containing optional fields.\n   *\n   * @return A `History` instance. Its `history` attribute contains all\n   *   information collected during training.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  async fitDataset<T>(dataset: Dataset<T>, args: ModelFitDatasetArgs<T>):\n      Promise<History> {\n    return fitDataset(this, dataset, args);\n  }\n\n  /**\n   * Runs a single gradient update on a single batch of data.\n   *\n   * This method differs from `fit()` and `fitDataset()` in the following\n   * regards:\n   *   - It operates on exactly one batch of data.\n   *   - It returns only the loss and matric values, instead of\n   *     returning the batch-by-batch loss and metric values.\n   *   - It doesn't support fine-grained options such as verbosity and\n   *     callbacks.\n   *\n   * @param x Input data. It could be one of the following:\n   *   - A `tf.Tensor`, or an Array of `tf.Tensor`s (in case the model has\n   *     multiple inputs).\n   *   - An Object mapping input names to corresponding `tf.Tensor` (if the\n   *     model has named inputs).\n   * @param y Target darta. It could be either a `tf.Tensor` a multiple\n   *   `tf.Tensor`s. It should be consistent with `x`.\n   * @returns Training loss or losses (in case the model has\n   *   multiple outputs), along with metrics (if any), as numbers.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes'}\n   */\n  async trainOnBatch(\n      x: Tensor|Tensor[]|{[inputName: string]: Tensor},\n      y: Tensor|Tensor[]|\n      {[inputName: string]: Tensor}): Promise<number|number[]> {\n    // TODO(cais): Support sampleWeight and classWeight.\n    // TODO(cais): Support Dataset objects.\n    const standardizeOut = await this.standardizeUserData(x, y);\n    const inputs = standardizeOut[0];\n    const targets = standardizeOut[1];\n    const trainFunction = this.makeTrainFunction();\n    const losses = trainFunction(inputs.concat(targets));\n    const lossValues: number[] = [];\n    for (const loss of losses) {\n      const v = await loss.data();\n      lossValues.push(v[0]);\n    }\n    tfc.dispose(losses);\n    disposeNewTensors(standardizeOut[0], x);\n    disposeNewTensors(standardizeOut[1], y);\n    return singletonOrArray(lossValues);\n  }\n\n  /**\n   * Extract weight values of the model.\n   *\n   * @param config: An instance of `io.SaveConfig`, which specifies\n   * model-saving options such as whether only trainable weights are to be\n   * saved.\n   * @returns A `NamedTensorMap` mapping original weight names (i.e.,\n   *   non-uniqueified weight names) to their values.\n   */\n  protected getNamedWeights(config?: io.SaveConfig): NamedTensor[] {\n    const namedWeights: NamedTensor[] = [];\n\n    const trainableOnly = config != null && config.trainableOnly;\n    const weights = trainableOnly ? this.trainableWeights : this.weights;\n    const weightValues = this.getWeights(trainableOnly);\n    for (let i = 0; i < weights.length; ++i) {\n      if (trainableOnly && !weights[i].trainable) {\n        // Optionally skip non-trainable weights.\n        continue;\n      }\n      namedWeights.push(\n          {name: weights[i].originalName, tensor: weightValues[i]});\n    }\n    return namedWeights;\n  }\n\n  /**\n   * Setter used for force stopping of LayersModel.fit() (i.e., training).\n   *\n   * Example:\n   *\n   * ```js\n   * const input = tf.input({shape: [10]});\n   * const output = tf.layers.dense({units: 1}).apply(input);\n   * const model = tf.model({inputs: [input], outputs: [output]});\n   * model.compile({loss: 'meanSquaredError', optimizer: 'sgd'});\n   * const xs = tf.ones([8, 10]);\n   * const ys = tf.zeros([8, 1]);\n   *\n   * const history = await model.fit(xs, ys, {\n   *   epochs: 10,\n   *   callbacks: {\n   *     onEpochEnd: async (epoch, logs) => {\n   *       if (epoch === 2) {\n   *         model.stopTraining = true;\n   *       }\n   *     }\n   *   }\n   * });\n   *\n   * // There should be only 3 values in the loss array, instead of 10\n   * values,\n   * // due to the stopping after 3 epochs.\n   * console.log(history.history.loss);\n   * ```\n   */\n  set stopTraining(stop: boolean) {\n    this.stopTraining_ = stop;\n  }\n\n  get stopTraining(): boolean {\n    return this.stopTraining_;\n  }\n\n  get optimizer(): Optimizer {\n    return this.optimizer_;\n  }\n\n  set optimizer(optimizer: Optimizer) {\n    if (this.optimizer_ !== optimizer) {\n      this.optimizer_ = optimizer;\n      this.isOptimizerOwned = false;\n    }\n  }\n\n  dispose(): DisposeResult {\n    const result = super.dispose();\n    if (result.refCountAfterDispose === 0 && this.optimizer != null &&\n        this.isOptimizerOwned) {\n      const numTensorsBeforeOptmizerDisposal = tfc.memory().numTensors;\n      this.optimizer_.dispose();\n      result.numDisposedVariables +=\n          numTensorsBeforeOptmizerDisposal - tfc.memory().numTensors;\n    }\n    return result;\n  }\n\n  private getLossIdentifiers(): LossIdentifier|LossIdentifier[]|\n      {[outputName: string]: LossIdentifier} {\n    let lossNames: LossIdentifier|LossIdentifier[]|\n        {[outputName: string]: LossIdentifier};\n    if (typeof this.loss === 'string') {\n      lossNames = toSnakeCase(this.loss) as LossIdentifier;\n    } else if (Array.isArray(this.loss)) {\n      for (const loss of this.loss) {\n        if (typeof loss !== 'string') {\n          throw new Error('Serialization of non-string loss is not supported.');\n        }\n      }\n      lossNames = (this.loss as string[]).map(name => toSnakeCase(name)) as\n          LossIdentifier[];\n    } else {\n      const outputNames = Object.keys(this.loss);\n      lossNames = {} as {[outputName: string]: LossIdentifier};\n      const losses =\n          this.loss as {[outputName: string]: LossOrMetricFn | string};\n      for (const outputName of outputNames) {\n        if (typeof losses[outputName] === 'string') {\n          lossNames[outputName] =\n              toSnakeCase(losses[outputName] as string) as LossIdentifier;\n        } else {\n          throw new Error('Serialization of non-string loss is not supported.');\n        }\n      }\n    }\n    return lossNames;\n  }\n\n  private getMetricIdentifiers(): MetricsIdentifier[]|\n      {[key: string]: MetricsIdentifier} {\n    if (typeof this.metrics === 'string' ||\n        typeof this.metrics === 'function') {\n      return [toSnakeCase(Metrics.getLossOrMetricName(this.metrics))];\n    } else if (Array.isArray(this.metrics)) {\n      return this.metrics.map(\n          metric => toSnakeCase(Metrics.getLossOrMetricName(metric)));\n    } else {\n      const metricsIdentifiers: {[key: string]: MetricsIdentifier} = {};\n      for (const key in this.metrics) {\n        metricsIdentifiers[key] =\n            toSnakeCase(Metrics.getLossOrMetricName(this.metrics[key]));\n      }\n      return metricsIdentifiers;\n    }\n  }\n\n  protected getTrainingConfig(): TrainingConfig {\n    return {\n      loss: this.getLossIdentifiers(),\n      metrics: this.getMetricIdentifiers(),\n      optimizer_config: {\n        class_name: this.optimizer.getClassName(),\n        config: this.optimizer.getConfig()\n      } as OptimizerSerialization\n    };\n    // TODO(cais): Add weight_metrics when they are supported.\n    // TODO(cais): Add sample_weight_mode when it's supported.\n    // TODO(cais): Add loss_weights when it's supported.\n  }\n\n  loadTrainingConfig(trainingConfig: TrainingConfig) {\n    if (trainingConfig.weighted_metrics != null) {\n      throw new Error('Loading weight_metrics is not supported yet.');\n    }\n    if (trainingConfig.loss_weights != null) {\n      throw new Error('Loading loss_weights is not supported yet.');\n    }\n    if (trainingConfig.sample_weight_mode != null) {\n      throw new Error('Loading sample_weight_mode is not supported yet.');\n    }\n\n    const tsConfig = convertPythonicToTs(trainingConfig.optimizer_config) as\n        serialization.ConfigDict;\n    const optimizer = deserialize(tsConfig) as Optimizer;\n\n    let loss;\n    if (typeof trainingConfig.loss === 'string') {\n      loss = toCamelCase(trainingConfig.loss);\n    } else if (Array.isArray(trainingConfig.loss)) {\n      loss = trainingConfig.loss.map(lossEntry => toCamelCase(lossEntry));\n    } else if (trainingConfig.loss != null) {\n      loss = {} as {[outputName: string]: LossIdentifier};\n      for (const key in trainingConfig.loss) {\n        loss[key] = toCamelCase(trainingConfig.loss[key]) as LossIdentifier;\n      }\n    }\n\n    let metrics;\n    if (Array.isArray(trainingConfig.metrics)) {\n      metrics = trainingConfig.metrics.map(metric => toCamelCase(metric));\n    } else if (trainingConfig.metrics != null) {\n      metrics = {} as {[outputName: string]: MetricsIdentifier};\n      for (const key in trainingConfig.metrics) {\n        metrics[key] = toCamelCase(trainingConfig.metrics[key]);\n      }\n    }\n\n    this.compile({loss, metrics, optimizer});\n  }\n\n  /**\n   * Save the configuration and/or weights of the LayersModel.\n   *\n   * An `IOHandler` is an object that has a `save` method of the proper\n   * signature defined. The `save` method manages the storing or\n   * transmission of serialized data (\"artifacts\") that represent the\n   * model's topology and weights onto or via a specific medium, such as\n   * file downloads, local storage, IndexedDB in the web browser and HTTP\n   * requests to a server. TensorFlow.js provides `IOHandler`\n   * implementations for a number of frequently used saving mediums, such as\n   * `tf.io.browserDownloads` and `tf.io.browserLocalStorage`. See `tf.io`\n   * for more details.\n   *\n   * This method also allows you to refer to certain types of `IOHandler`s\n   * as URL-like string shortcuts, such as 'localstorage://' and\n   * 'indexeddb://'.\n   *\n   * Example 1: Save `model`'s topology and weights to browser [local\n   * storage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage);\n   * then load it back.\n   *\n   * ```js\n   * const model = tf.sequential(\n   *     {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});\n   * console.log('Prediction from original model:');\n   * model.predict(tf.ones([1, 3])).print();\n   *\n   * const saveResults = await model.save('localstorage://my-model-1');\n   *\n   * const loadedModel = await tf.loadLayersModel('localstorage://my-model-1');\n   * console.log('Prediction from loaded model:');\n   * loadedModel.predict(tf.ones([1, 3])).print();\n   * ```\n   *\n   * Example 2. Saving `model`'s topology and weights to browser\n   * [IndexedDB](https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API);\n   * then load it back.\n   *\n   * ```js\n   * const model = tf.sequential(\n   *     {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});\n   * console.log('Prediction from original model:');\n   * model.predict(tf.ones([1, 3])).print();\n   *\n   * const saveResults = await model.save('indexeddb://my-model-1');\n   *\n   * const loadedModel = await tf.loadLayersModel('indexeddb://my-model-1');\n   * console.log('Prediction from loaded model:');\n   * loadedModel.predict(tf.ones([1, 3])).print();\n   * ```\n   *\n   * Example 3. Saving `model`'s topology and weights as two files\n   * (`my-model-1.json` and `my-model-1.weights.bin`) downloaded from\n   * browser.\n   *\n   * ```js\n   * const model = tf.sequential(\n   *     {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});\n   * const saveResults = await model.save('downloads://my-model-1');\n   * ```\n   *\n   * Example 4. Send  `model`'s topology and weights to an HTTP server.\n   * See the documentation of `tf.io.http` for more details\n   * including specifying request parameters and implementation of the\n   * server.\n   *\n   * ```js\n   * const model = tf.sequential(\n   *     {layers: [tf.layers.dense({units: 1, inputShape: [3]})]});\n   * const saveResults = await model.save('http://my-server/model/upload');\n   * ```\n   *\n   * @param handlerOrURL An instance of `IOHandler` or a URL-like,\n   * scheme-based string shortcut for `IOHandler`.\n   * @param config Options for saving the model.\n   * @returns A `Promise` of `SaveResult`, which summarizes the result of\n   * the saving, such as byte sizes of the saved artifacts for the model's\n   *   topology and weight values.\n   *\n   * @doc {heading: 'Models', subheading: 'Classes', ignoreCI: true}\n   */\n  async save(handlerOrURL: io.IOHandler|string, config?: io.SaveConfig):\n      Promise<io.SaveResult> {\n    if (typeof handlerOrURL === 'string') {\n      const handlers = io.getSaveHandlers(handlerOrURL);\n      if (handlers.length === 0) {\n        throw new ValueError(\n            `Cannot find any save handlers for URL '${handlerOrURL}'`);\n      } else if (handlers.length > 1) {\n        throw new ValueError(\n            `Found more than one (${handlers.length}) save handlers for ` +\n            `URL '${handlerOrURL}'`);\n      }\n      handlerOrURL = handlers[0];\n    }\n    if (handlerOrURL.save == null) {\n      throw new ValueError(\n          'LayersModel.save() cannot proceed because the IOHandler ' +\n          'provided does not have the `save` attribute defined.');\n    }\n\n    const weightDataAndSpecs =\n        await io.encodeWeights(this.getNamedWeights(config));\n\n    const returnString = false;\n    const unusedArg: {} = null;\n    const modelConfig = this.toJSON(unusedArg, returnString);\n    const modelArtifacts: io.ModelArtifacts = {\n      modelTopology: modelConfig,\n      format: LAYERS_MODEL_FORMAT_NAME,\n      generatedBy: `TensorFlow.js tfjs-layers v${version}`,\n      convertedBy: null,\n    };\n\n    const includeOptimizer = config == null ? false : config.includeOptimizer;\n    if (includeOptimizer && this.optimizer != null) {\n      modelArtifacts.trainingConfig = this.getTrainingConfig();\n      const weightType = 'optimizer';\n      const {data: optimizerWeightData, specs: optimizerWeightSpecs} =\n          await io.encodeWeights(await this.optimizer.getWeights(), weightType);\n      weightDataAndSpecs.specs.push(...optimizerWeightSpecs);\n      weightDataAndSpecs.data = io.concatenateArrayBuffers(\n          [weightDataAndSpecs.data, optimizerWeightData]);\n    }\n\n    if (this.userDefinedMetadata != null) {\n      // Check serialized size of user-defined metadata.\n      const checkSize = true;\n      checkUserDefinedMetadata(this.userDefinedMetadata, this.name, checkSize);\n      modelArtifacts.userDefinedMetadata = this.userDefinedMetadata;\n    }\n\n    modelArtifacts.weightData = weightDataAndSpecs.data;\n    modelArtifacts.weightSpecs = weightDataAndSpecs.specs;\n    return handlerOrURL.save(modelArtifacts);\n  }\n\n  /**\n   * Set user-defined metadata.\n   *\n   * The set metadata will be serialized together with the topology\n   * and weights of the model during `save()` calls.\n   *\n   * @param setUserDefinedMetadata\n   */\n  setUserDefinedMetadata(userDefinedMetadata: {}): void {\n    checkUserDefinedMetadata(userDefinedMetadata, this.name);\n    this.userDefinedMetadata = userDefinedMetadata;\n  }\n\n  /**\n   * Get user-defined metadata.\n   *\n   * The metadata is supplied via one of the two routes:\n   *   1. By calling `setUserDefinedMetadata()`.\n   *   2. Loaded during model loading (if the model is constructed\n   *      via `tf.loadLayersModel()`.)\n   *\n   * If no user-defined metadata is available from either of the\n   * two routes, this function will return `undefined`.\n   */\n  getUserDefinedMetadata(): {} {\n    return this.userDefinedMetadata;\n  }\n}\nserialization.registerClass(LayersModel);\n\n/**\n * A `tf.Functional` is an alias to `tf.LayersModel`.\n *\n * See also:\n *   `tf.LayersModel`, `tf.Sequential`, `tf.loadLayersModel`.\n */\n/** @doc {heading: 'Models', subheading: 'Classes'} */\nexport class Functional extends LayersModel {\n  static className = 'Functional';\n}\nserialization.registerClass(Functional);\n"]} |
\ | No newline at end of file |