/** * @license * Copyright 2020 Google LLC. All Rights Reserved. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ============================================================================= */ import * as tf from '../index'; import { ALL_ENVS, describeWithFlags } from '../jasmine_util'; import { expectArraysClose } from '../test_util'; describeWithFlags('sub', ALL_ENVS, () => { it('c - A', async () => { const c = tf.scalar(5); const a = tf.tensor1d([7, 2, 3]); const result = tf.sub(c, a); expectArraysClose(await result.data(), [-2, 3, 2]); }); it('A - c', async () => { const a = tf.tensor1d([1, 2, -3]); const c = tf.scalar(5); const result = tf.sub(a, c); expectArraysClose(await result.data(), [-4, -3, -8]); }); it('A - c propagates NaNs', async () => { const a = tf.tensor1d([1, NaN, 3]); const c = tf.scalar(5); const res = tf.sub(a, c); expectArraysClose(await res.data(), [-4, NaN, -2]); }); it('A - B', async () => { const a = tf.tensor1d([2, 5, 1]); const b = tf.tensor1d([4, 2, -1]); const result = tf.sub(a, b); const expected = [-2, 3, 2]; expectArraysClose(await result.data(), expected); }); it('TensorLike', async () => { const a = [2, 5, 1]; const b = [4, 2, -1]; const result = tf.sub(a, b); const expected = [-2, 3, 2]; expectArraysClose(await result.data(), expected); }); it('TensorLike chained', async () => { const a = tf.tensor1d([2, 5, 1]); const b = [4, 2, -1]; const result = a.sub(b); const expected = [-2, 3, 2]; expectArraysClose(await result.data(), expected); }); it('A - B propagates NaNs', async () => { const a = tf.tensor1d([2, 5, 1]); const b = tf.tensor1d([4, NaN, -1]); const res = tf.sub(a, b); expectArraysClose(await res.data(), [-2, NaN, 2]); }); it('A - B throws when passed tensors with different shape', () => { const a = tf.tensor1d([2, 5, 1, 5]); const b = tf.tensor1d([4, 2, -1]); expect(() => tf.sub(a, b)).toThrowError(); expect(() => tf.sub(b, a)).toThrowError(); }); it('A - B broadcasting same rank Tensors different shape', async () => { const a = tf.tensor2d([1, 2, -3, -4], [2, 2]); const b = tf.tensor2d([2, 3], [2, 1]); const result = tf.sub(a, b); expect(result.shape).toEqual([2, 2]); const expected = [-1, 0, -6, -7]; expectArraysClose(await result.data(), expected); }); it('A - B broadcast 2D + 1D', async () => { const a = tf.tensor2d([1, 2, -3, -4], [2, 2]); const b = tf.tensor1d([1, 2]); const result = tf.sub(a, b); expect(result.shape).toEqual([2, 2]); const expected = [0, 0, -4, -6]; expectArraysClose(await result.data(), expected); }); it('2D-scalar broadcast', async () => { const a = tf.tensor2d([1, 2, 3, 4, 5, 6], [2, 3]); const b = tf.scalar(2); const res = tf.sub(a, b); expect(res.shape).toEqual([2, 3]); expectArraysClose(await res.data(), [-1, 0, 1, 2, 3, 4]); }); it('scalar-1D broadcast', async () => { const a = tf.scalar(2); const b = tf.tensor1d([1, 2, 3, 4, 5, 6]); const res = tf.sub(a, b); expect(res.shape).toEqual([6]); expectArraysClose(await res.data(), [1, 0, -1, -2, -3, -4]); }); it('2D-2D broadcast each with 1 dim', async () => { const a = tf.tensor2d([1, 2, 5], [1, 3]); const b = tf.tensor2d([7, 3], [2, 1]); const res = tf.sub(a, b); expect(res.shape).toEqual([2, 3]); expectArraysClose(await res.data(), [-6, -5, -2, -2, -1, 2]); }); it('2D-2D broadcast inner dim of b', async () => { const a = tf.tensor2d([1, 2, 5, 4, 5, 6], [2, 3]); const b = tf.tensor2d([7, 3], [2, 1]); const res = tf.sub(a, b); expect(res.shape).toEqual([2, 3]); expectArraysClose(await res.data(), [-6, -5, -2, 1, 2, 3]); }); it('3D-scalar', async () => { const a = tf.tensor3d([1, 2, 3, 4, 5, 6], [2, 3, 1]); const b = tf.scalar(-1); const res = tf.sub(a, b); expect(res.shape).toEqual([2, 3, 1]); expectArraysClose(await res.data(), [2, 3, 4, 5, 6, 7]); }); it('vec4 same shape', async () => { const a = tf.tensor2d([1, 2, -3, -4], [2, 2]); const b = tf.tensor2d([5, 3, 4, -7], [2, 2]); const expected = [-4, -1, -7, 3]; const result = tf.sub(a, b); expect(result.shape).toEqual([2, 2]); expectArraysClose(await result.data(), expected); }); it('gradients: basic 1D arrays', async () => { const a = tf.tensor1d([1, 2, 3]); const b = tf.tensor1d([3, 2, 1]); const dy = tf.tensor1d([1, 10, 20]); const grads = tf.grads((a, b) => tf.sub(a, b)); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [1, 10, 20]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-1, -10, -20]); }); it('gradient with clones', async () => { const a = tf.tensor1d([1, 2, 3]); const b = tf.tensor1d([3, 2, 1]); const dy = tf.tensor1d([1, 10, 20]); const grads = tf.grads((a, b) => tf.sub(a.clone(), b.clone()).clone()); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [1, 10, 20]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-1, -10, -20]); }); it('gradients: basic 2D arrays', async () => { const a = tf.tensor2d([0, 1, 2, 3], [2, 2]); const b = tf.tensor2d([3, 2, 1, 0], [2, 2]); const dy = tf.tensor2d([1, 10, 15, 20], [2, 2]); const grads = tf.grads((a, b) => tf.sub(a, b)); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [1, 10, 15, 20]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-1, -10, -15, -20]); }); it('gradient: 1D - scalar broadcast', async () => { const a = tf.tensor1d([3, 4, 5]); const b = tf.scalar(2); const dy = tf.tensor1d([7, 8, 9]); const grads = tf.grads((a, b) => tf.sub(a, b)); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [7, 8, 9]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-7 - 8 - 9]); }); it('gradient: scalar - 1D broadcast', async () => { const a = tf.scalar(2); const b = tf.tensor1d([3, 4, 5]); const dy = tf.tensor1d([7, 8, 9]); const grads = tf.grads((a, b) => tf.sub(a, b)); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [7 + 8 + 9]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-7, -8, -9]); }); it('gradient: 2D - 2D broadcast', async () => { const a = tf.tensor2d([4, 5, 6, 7], [2, 2]); const b = tf.tensor2d([2, 3], [2, 1]); const dy = tf.tensor2d([5, 4, 3, 2], [2, 2]); const grads = tf.grads((a, b) => tf.sub(a, b)); const [da, db] = grads([a, b], dy); expect(da.shape).toEqual(a.shape); expect(da.dtype).toEqual('float32'); expectArraysClose(await da.data(), [5, 4, 3, 2]); expect(db.shape).toEqual(b.shape); expect(db.dtype).toEqual('float32'); expectArraysClose(await db.data(), [-5 - 4, -3 - 2]); }); it('complex number subtraction', async () => { const real1 = tf.tensor1d([3]); const imag1 = tf.tensor1d([5]); const complex1 = tf.complex(real1, imag1); const real2 = tf.tensor1d([1]); const imag2 = tf.tensor1d([0]); const complex2 = tf.complex(real2, imag2); const result = complex1.sub(complex2); expect(result.dtype).toBe('complex64'); expect(result.shape).toEqual([1]); expectArraysClose(await result.data(), [2, 5]); }); it('complex number broadcasting subtraction', async () => { const real1 = tf.tensor2d([1, 2, -3, -4], [2, 2]); const imag1 = tf.tensor2d([10, 20, -30, -40], [2, 2]); const complex1 = tf.complex(real1, imag1); const real2 = tf.tensor1d([4]); const imag2 = tf.tensor1d([5]); const complex2 = tf.complex(real2, imag2); const result = tf.sub(complex1, complex2); expect(result.dtype).toEqual('complex64'); expect(result.shape).toEqual([2, 2]); expectArraysClose(await result.data(), [1 - 4, 10 - 5, 2 - 4, 20 - 5, -3 - 4, -30 - 5, -4 - 4, -40 - 5]); }); it('throws when passed a as a non-tensor', () => { expect(() => tf.sub({}, tf.scalar(1))) .toThrowError(/Argument 'a' passed to 'sub' must be a Tensor/); }); it('throws when passed b as a non-tensor', () => { expect(() => tf.sub(tf.scalar(1), {})) .toThrowError(/Argument 'b' passed to 'sub' must be a Tensor/); }); it('upcasts when dtypes dont match', async () => { let res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(1, 'float32')); expect(res.dtype).toBe('float32'); expectArraysClose(await res.data(), [0]); res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(true, 'bool')); expect(res.dtype).toBe('int32'); expectArraysClose(await res.data(), [0]); res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(false, 'bool')); expect(res.dtype).toBe('int32'); expectArraysClose(await res.data(), [1]); res = tf.sub(tf.complex(4, 7), tf.scalar(1, 'float32')); expect(res.dtype).toBe('complex64'); expectArraysClose(await res.data(), [3, 7]); res = tf.sub(tf.complex(4, 7), tf.scalar(1, 'int32')); expect(res.dtype).toBe('complex64'); expectArraysClose(await res.data(), [3, 7]); }); it('accepts a tensor-like object', async () => { const result = tf.sub(5, [7, 2, 3]); expectArraysClose(await result.data(), [-2, 3, 2]); }); }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sub_test.js","sourceRoot":"","sources":["../../../../../../tfjs-core/src/ops/sub_test.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2020 Google LLC. All Rights Reserved.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n * =============================================================================\n */\nimport * as tf from '../index';\nimport {ALL_ENVS, describeWithFlags} from '../jasmine_util';\nimport {expectArraysClose} from '../test_util';\n\ndescribeWithFlags('sub', ALL_ENVS, () => {\n  it('c - A', async () => {\n    const c = tf.scalar(5);\n    const a = tf.tensor1d([7, 2, 3]);\n\n    const result = tf.sub(c, a);\n\n    expectArraysClose(await result.data(), [-2, 3, 2]);\n  });\n\n  it('A - c', async () => {\n    const a = tf.tensor1d([1, 2, -3]);\n    const c = tf.scalar(5);\n\n    const result = tf.sub(a, c);\n\n    expectArraysClose(await result.data(), [-4, -3, -8]);\n  });\n\n  it('A - c propagates NaNs', async () => {\n    const a = tf.tensor1d([1, NaN, 3]);\n    const c = tf.scalar(5);\n\n    const res = tf.sub(a, c);\n\n    expectArraysClose(await res.data(), [-4, NaN, -2]);\n  });\n\n  it('A - B', async () => {\n    const a = tf.tensor1d([2, 5, 1]);\n    const b = tf.tensor1d([4, 2, -1]);\n\n    const result = tf.sub(a, b);\n\n    const expected = [-2, 3, 2];\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('TensorLike', async () => {\n    const a = [2, 5, 1];\n    const b = [4, 2, -1];\n\n    const result = tf.sub(a, b);\n\n    const expected = [-2, 3, 2];\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('TensorLike chained', async () => {\n    const a = tf.tensor1d([2, 5, 1]);\n    const b = [4, 2, -1];\n\n    const result = a.sub(b);\n\n    const expected = [-2, 3, 2];\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('A - B propagates NaNs', async () => {\n    const a = tf.tensor1d([2, 5, 1]);\n    const b = tf.tensor1d([4, NaN, -1]);\n\n    const res = tf.sub(a, b);\n\n    expectArraysClose(await res.data(), [-2, NaN, 2]);\n  });\n\n  it('A - B throws when passed tensors with different shape', () => {\n    const a = tf.tensor1d([2, 5, 1, 5]);\n    const b = tf.tensor1d([4, 2, -1]);\n\n    expect(() => tf.sub(a, b)).toThrowError();\n    expect(() => tf.sub(b, a)).toThrowError();\n  });\n\n  it('A - B broadcasting same rank Tensors different shape', async () => {\n    const a = tf.tensor2d([1, 2, -3, -4], [2, 2]);\n    const b = tf.tensor2d([2, 3], [2, 1]);\n\n    const result = tf.sub(a, b);\n\n    expect(result.shape).toEqual([2, 2]);\n    const expected = [-1, 0, -6, -7];\n\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('A - B broadcast 2D + 1D', async () => {\n    const a = tf.tensor2d([1, 2, -3, -4], [2, 2]);\n    const b = tf.tensor1d([1, 2]);\n\n    const result = tf.sub(a, b);\n\n    expect(result.shape).toEqual([2, 2]);\n    const expected = [0, 0, -4, -6];\n\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('2D-scalar broadcast', async () => {\n    const a = tf.tensor2d([1, 2, 3, 4, 5, 6], [2, 3]);\n    const b = tf.scalar(2);\n    const res = tf.sub(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [-1, 0, 1, 2, 3, 4]);\n  });\n\n  it('scalar-1D broadcast', async () => {\n    const a = tf.scalar(2);\n    const b = tf.tensor1d([1, 2, 3, 4, 5, 6]);\n    const res = tf.sub(a, b);\n    expect(res.shape).toEqual([6]);\n    expectArraysClose(await res.data(), [1, 0, -1, -2, -3, -4]);\n  });\n\n  it('2D-2D broadcast each with 1 dim', async () => {\n    const a = tf.tensor2d([1, 2, 5], [1, 3]);\n    const b = tf.tensor2d([7, 3], [2, 1]);\n    const res = tf.sub(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [-6, -5, -2, -2, -1, 2]);\n  });\n\n  it('2D-2D broadcast inner dim of b', async () => {\n    const a = tf.tensor2d([1, 2, 5, 4, 5, 6], [2, 3]);\n    const b = tf.tensor2d([7, 3], [2, 1]);\n    const res = tf.sub(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [-6, -5, -2, 1, 2, 3]);\n  });\n\n  it('3D-scalar', async () => {\n    const a = tf.tensor3d([1, 2, 3, 4, 5, 6], [2, 3, 1]);\n    const b = tf.scalar(-1);\n    const res = tf.sub(a, b);\n    expect(res.shape).toEqual([2, 3, 1]);\n    expectArraysClose(await res.data(), [2, 3, 4, 5, 6, 7]);\n  });\n\n  it('vec4 same shape', async () => {\n    const a = tf.tensor2d([1, 2, -3, -4], [2, 2]);\n    const b = tf.tensor2d([5, 3, 4, -7], [2, 2]);\n    const expected = [-4, -1, -7, 3];\n    const result = tf.sub(a, b);\n\n    expect(result.shape).toEqual([2, 2]);\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('gradients: basic 1D arrays', async () => {\n    const a = tf.tensor1d([1, 2, 3]);\n    const b = tf.tensor1d([3, 2, 1]);\n    const dy = tf.tensor1d([1, 10, 20]);\n\n    const grads = tf.grads((a, b) => tf.sub(a, b));\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [1, 10, 20]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-1, -10, -20]);\n  });\n\n  it('gradient with clones', async () => {\n    const a = tf.tensor1d([1, 2, 3]);\n    const b = tf.tensor1d([3, 2, 1]);\n    const dy = tf.tensor1d([1, 10, 20]);\n\n    const grads = tf.grads((a, b) => tf.sub(a.clone(), b.clone()).clone());\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [1, 10, 20]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-1, -10, -20]);\n  });\n\n  it('gradients: basic 2D arrays', async () => {\n    const a = tf.tensor2d([0, 1, 2, 3], [2, 2]);\n    const b = tf.tensor2d([3, 2, 1, 0], [2, 2]);\n    const dy = tf.tensor2d([1, 10, 15, 20], [2, 2]);\n\n    const grads = tf.grads((a, b) => tf.sub(a, b));\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [1, 10, 15, 20]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-1, -10, -15, -20]);\n  });\n\n  it('gradient: 1D - scalar broadcast', async () => {\n    const a = tf.tensor1d([3, 4, 5]);\n    const b = tf.scalar(2);\n    const dy = tf.tensor1d([7, 8, 9]);\n\n    const grads = tf.grads((a, b) => tf.sub(a, b));\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [7, 8, 9]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-7 - 8 - 9]);\n  });\n\n  it('gradient: scalar - 1D broadcast', async () => {\n    const a = tf.scalar(2);\n    const b = tf.tensor1d([3, 4, 5]);\n    const dy = tf.tensor1d([7, 8, 9]);\n\n    const grads = tf.grads((a, b) => tf.sub(a, b));\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [7 + 8 + 9]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-7, -8, -9]);\n  });\n\n  it('gradient: 2D - 2D broadcast', async () => {\n    const a = tf.tensor2d([4, 5, 6, 7], [2, 2]);\n    const b = tf.tensor2d([2, 3], [2, 1]);\n    const dy = tf.tensor2d([5, 4, 3, 2], [2, 2]);\n\n    const grads = tf.grads((a, b) => tf.sub(a, b));\n    const [da, db] = grads([a, b], dy);\n\n    expect(da.shape).toEqual(a.shape);\n    expect(da.dtype).toEqual('float32');\n    expectArraysClose(await da.data(), [5, 4, 3, 2]);\n\n    expect(db.shape).toEqual(b.shape);\n    expect(db.dtype).toEqual('float32');\n    expectArraysClose(await db.data(), [-5 - 4, -3 - 2]);\n  });\n\n  it('complex number subtraction', async () => {\n    const real1 = tf.tensor1d([3]);\n    const imag1 = tf.tensor1d([5]);\n    const complex1 = tf.complex(real1, imag1);\n\n    const real2 = tf.tensor1d([1]);\n    const imag2 = tf.tensor1d([0]);\n    const complex2 = tf.complex(real2, imag2);\n\n    const result = complex1.sub(complex2);\n\n    expect(result.dtype).toBe('complex64');\n    expect(result.shape).toEqual([1]);\n    expectArraysClose(await result.data(), [2, 5]);\n  });\n\n  it('complex number broadcasting subtraction', async () => {\n    const real1 = tf.tensor2d([1, 2, -3, -4], [2, 2]);\n    const imag1 = tf.tensor2d([10, 20, -30, -40], [2, 2]);\n    const complex1 = tf.complex(real1, imag1);\n\n    const real2 = tf.tensor1d([4]);\n    const imag2 = tf.tensor1d([5]);\n    const complex2 = tf.complex(real2, imag2);\n\n    const result = tf.sub(complex1, complex2);\n\n    expect(result.dtype).toEqual('complex64');\n    expect(result.shape).toEqual([2, 2]);\n    expectArraysClose(\n        await result.data(),\n        [1 - 4, 10 - 5, 2 - 4, 20 - 5, -3 - 4, -30 - 5, -4 - 4, -40 - 5]);\n  });\n\n  it('throws when passed a as a non-tensor', () => {\n    expect(() => tf.sub({} as tf.Tensor, tf.scalar(1)))\n        .toThrowError(/Argument 'a' passed to 'sub' must be a Tensor/);\n  });\n  it('throws when passed b as a non-tensor', () => {\n    expect(() => tf.sub(tf.scalar(1), {} as tf.Tensor))\n        .toThrowError(/Argument 'b' passed to 'sub' must be a Tensor/);\n  });\n  it('upcasts when dtypes dont match', async () => {\n    let res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(1, 'float32'));\n    expect(res.dtype).toBe('float32');\n    expectArraysClose(await res.data(), [0]);\n\n    res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(true, 'bool'));\n    expect(res.dtype).toBe('int32');\n    expectArraysClose(await res.data(), [0]);\n\n    res = tf.sub(tf.scalar(1, 'int32'), tf.scalar(false, 'bool'));\n    expect(res.dtype).toBe('int32');\n    expectArraysClose(await res.data(), [1]);\n\n    res = tf.sub(tf.complex(4, 7), tf.scalar(1, 'float32'));\n    expect(res.dtype).toBe('complex64');\n    expectArraysClose(await res.data(), [3, 7]);\n\n    res = tf.sub(tf.complex(4, 7), tf.scalar(1, 'int32'));\n    expect(res.dtype).toBe('complex64');\n    expectArraysClose(await res.data(), [3, 7]);\n  });\n\n  it('accepts a tensor-like object', async () => {\n    const result = tf.sub(5, [7, 2, 3]);\n    expectArraysClose(await result.data(), [-2, 3, 2]);\n  });\n});\n"]}