UNPKG

49.8 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright 2020 Google LLC. All Rights Reserved.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 * =============================================================================
16 */
17import * as tf from '../index';
18import { ALL_ENVS, describeWithFlags } from '../jasmine_util';
19import { expectArraysClose, expectArraysEqual } from '../test_util';
20describeWithFlags('add', ALL_ENVS, () => {
21 it('c + A', async () => {
22 const c = tf.scalar(5);
23 const a = tf.tensor1d([1, 2, 3]);
24 const result = tf.add(c, a);
25 expectArraysClose(await result.data(), [6, 7, 8]);
26 });
27 it('c + A propagates NaNs', async () => {
28 const c = tf.scalar(NaN);
29 const a = tf.tensor1d([1, 2, 3]);
30 const res = tf.add(c, a);
31 expectArraysEqual(await res.data(), [NaN, NaN, NaN]);
32 });
33 it('A + B broadcasting same rank Tensors different shape', async () => {
34 const a = tf.tensor2d([1, 2, -3, -4], [2, 2]);
35 const b = tf.tensor2d([2, 3], [2, 1]);
36 const result = tf.add(a, b);
37 expect(result.shape).toEqual([2, 2]);
38 const expected = [3, 4, 0, -1];
39 expectArraysClose(await result.data(), expected);
40 });
41 it('A + B broadcast 2D + 1D', async () => {
42 const a = tf.tensor2d([1, 2, -3, -4], [2, 2]);
43 const b = tf.tensor1d([1, 2]);
44 const result = tf.add(a, b);
45 expect(result.shape).toEqual([2, 2]);
46 const expected = [2, 4, -2, -2];
47 expectArraysClose(await result.data(), expected);
48 });
49 it('A + B', async () => {
50 const a = tf.tensor1d([2, 5, 1]);
51 const b = tf.tensor1d([4, 2, -1]);
52 const result = tf.add(a, b);
53 const expected = [6, 7, 0];
54 expectArraysClose(await result.data(), expected);
55 });
56 it('TensorLike', async () => {
57 const a = [2, 5, 1];
58 const b = [4, 2, -1];
59 const result = tf.add(a, b);
60 const expected = [6, 7, 0];
61 expectArraysClose(await result.data(), expected);
62 });
63 it('TensorLike chained', async () => {
64 const a = tf.tensor1d([2, 5, 1]);
65 const b = [4, 2, -1];
66 const result = a.add(b);
67 const expected = [6, 7, 0];
68 expectArraysClose(await result.data(), expected);
69 });
70 it('A + B propagates NaNs', async () => {
71 const a = tf.tensor1d([2, 5, NaN]);
72 const b = tf.tensor1d([4, 2, -1]);
73 const res = tf.add(a, b);
74 expectArraysClose(await res.data(), [6, 7, NaN]);
75 });
76 it('A + B throws when passed tensors with different shape', () => {
77 const a = tf.tensor1d([2, 5, 1, 5]);
78 const b = tf.tensor1d([4, 2, -1]);
79 expect(() => tf.add(a, b)).toThrowError();
80 expect(() => tf.add(b, a)).toThrowError();
81 });
82 it('2D+scalar broadcast', async () => {
83 const a = tf.tensor2d([1, 2, 3, 4, 5, 6], [2, 3]);
84 const b = tf.scalar(2);
85 const res = tf.add(a, b);
86 expect(res.shape).toEqual([2, 3]);
87 expectArraysClose(await res.data(), [3, 4, 5, 6, 7, 8]);
88 });
89 it('scalar+1D broadcast', async () => {
90 const a = tf.scalar(2);
91 const b = tf.tensor1d([1, 2, 3, 4, 5, 6]);
92 const res = tf.add(a, b);
93 expect(res.shape).toEqual([6]);
94 expectArraysClose(await res.data(), [3, 4, 5, 6, 7, 8]);
95 });
96 it('2D+2D broadcast each with 1 dim', async () => {
97 const a = tf.tensor2d([1, 2, 5], [1, 3]);
98 const b = tf.tensor2d([7, 3], [2, 1]);
99 const res = tf.add(a, b);
100 expect(res.shape).toEqual([2, 3]);
101 expectArraysClose(await res.data(), [8, 9, 12, 4, 5, 8]);
102 });
103 it('2D+2D broadcast inner dim of b', async () => {
104 const a = tf.tensor2d([1, 2, 5, 4, 5, 6], [2, 3]);
105 const b = tf.tensor2d([7, 3], [2, 1]);
106 const res = tf.add(a, b);
107 expect(res.shape).toEqual([2, 3]);
108 expectArraysClose(await res.data(), [8, 9, 12, 7, 8, 9]);
109 });
110 it('3D+scalar', async () => {
111 const a = tf.tensor3d([1, 2, 3, 4, 5, 6], [2, 3, 1]);
112 const b = tf.scalar(-1);
113 const res = tf.add(a, b);
114 expect(res.shape).toEqual([2, 3, 1]);
115 expectArraysClose(await res.data(), [0, 1, 2, 3, 4, 5]);
116 });
117 it('6D+scalar', async () => {
118 const a = tf.range(0, 64).reshape([2, 2, 2, 2, 2, 2]);
119 const b = tf.scalar(-1);
120 const res = tf.add(a, b);
121 expect(res.shape).toEqual([2, 2, 2, 2, 2, 2]);
122 const expectedResult = [
123 -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
124 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30,
125 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46,
126 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62
127 ];
128 expectArraysClose(await res.data(), expectedResult);
129 });
130 it('6D+2D', async () => {
131 const a = tf.range(0, 64).reshape([2, 2, 2, 2, 2, 2]);
132 const b = tf.tensor2d([11, 13, 17, 19], [2, 2]);
133 const res = tf.add(a, b);
134 expect(res.shape).toEqual([2, 2, 2, 2, 2, 2]);
135 const expectedResult = [
136 11, 14, 19, 22, 15, 18, 23, 26, 19, 22, 27, 30, 23, 26, 31, 34,
137 27, 30, 35, 38, 31, 34, 39, 42, 35, 38, 43, 46, 39, 42, 47, 50,
138 43, 46, 51, 54, 47, 50, 55, 58, 51, 54, 59, 62, 55, 58, 63, 66,
139 59, 62, 67, 70, 63, 66, 71, 74, 67, 70, 75, 78, 71, 74, 79, 82
140 ];
141 expectArraysClose(await res.data(), expectedResult);
142 });
143 it('add tensors with 0 in shape', async () => {
144 const a = tf.tensor1d([1]);
145 const b = tf.tensor3d([], [0, 0, 5]);
146 const res = tf.add(a, b);
147 expect(res.shape).toEqual([0, 0, 5]);
148 expectArraysEqual(await res.data(), []);
149 });
150 it('gradient: scalar + 1D broadcast', async () => {
151 const a = tf.scalar(2);
152 const b = tf.tensor1d([3, 4, 5]);
153 const dy = tf.tensor1d([7, 8, 9]);
154 const grads = tf.grads((a, b) => tf.add(a, b));
155 const [da, db] = grads([a, b], dy);
156 expect(da.shape).toEqual(a.shape);
157 expect(da.dtype).toEqual('float32');
158 expectArraysClose(await da.data(), [7 + 8 + 9]);
159 expect(db.shape).toEqual(b.shape);
160 expect(db.dtype).toEqual('float32');
161 expectArraysClose(await db.data(), [7, 8, 9]);
162 });
163 it('gradient with clones', async () => {
164 const a = tf.scalar(2);
165 const b = tf.tensor1d([3, 4, 5]);
166 const dy = tf.tensor1d([7, 8, 9]);
167 const grads = tf.grads((a, b) => tf.add(a.clone(), b.clone()).clone());
168 const [da, db] = grads([a, b], dy);
169 expect(da.shape).toEqual(a.shape);
170 expect(da.dtype).toEqual('float32');
171 expectArraysClose(await da.data(), [7 + 8 + 9]);
172 expect(db.shape).toEqual(b.shape);
173 expect(db.dtype).toEqual('float32');
174 expectArraysClose(await db.data(), [7, 8, 9]);
175 });
176 it('gradient: 2D + 2D broadcast', async () => {
177 const a = tf.tensor2d([2, 3], [2, 1]);
178 const b = tf.tensor2d([4, 5, 6, 7], [2, 2]);
179 const dy = tf.tensor2d([5, 4, 3, 2], [2, 2]);
180 const grads = tf.grads((a, b) => tf.add(a, b));
181 const [da, db] = grads([a, b], dy);
182 expect(da.shape).toEqual(a.shape);
183 expect(da.dtype).toEqual('float32');
184 expectArraysClose(await da.data(), [5 + 4, 3 + 2]);
185 expect(db.shape).toEqual(b.shape);
186 expect(db.dtype).toEqual('float32');
187 expectArraysClose(await db.data(), [5, 4, 3, 2]);
188 });
189 it('complex number addition', async () => {
190 const real1 = tf.tensor1d([1]);
191 const imag1 = tf.tensor1d([2]);
192 const complex1 = tf.complex(real1, imag1);
193 const real2 = tf.tensor1d([3]);
194 const imag2 = tf.tensor1d([4]);
195 const complex2 = tf.complex(real2, imag2);
196 const result = complex1.add(complex2);
197 expect(result.dtype).toBe('complex64');
198 expect(result.shape).toEqual([1]);
199 expectArraysClose(await result.data(), [4, 6]);
200 });
201 it('complex number reshape and then addition', async () => {
202 const real1 = tf.tensor1d([1]);
203 const imag1 = tf.tensor1d([2]);
204 const complex1 = tf.complex(real1, imag1);
205 const real2 = tf.tensor1d([3]);
206 const imag2 = tf.tensor1d([4]);
207 const complex2 = tf.complex(real2, imag2);
208 const complex1Reshaped = complex1.reshape([1, 1, 1]);
209 const complex2Reshaped = complex2.reshape([1, 1, 1]);
210 const result = complex1Reshaped.add(complex2Reshaped);
211 expect(result.dtype).toBe('complex64');
212 expect(result.shape).toEqual([1, 1, 1]);
213 expectArraysClose(await result.data(), [4, 6]);
214 });
215 it('complex number broadcasting addition', async () => {
216 const real1 = tf.tensor2d([1, 2, -3, -4], [2, 2]);
217 const imag1 = tf.tensor2d([10, 20, -30, -40], [2, 2]);
218 const complex1 = tf.complex(real1, imag1);
219 const real2 = tf.tensor1d([4]);
220 const imag2 = tf.tensor1d([5]);
221 const complex2 = tf.complex(real2, imag2);
222 const result = tf.add(complex1, complex2);
223 expect(result.dtype).toEqual('complex64');
224 expect(result.shape).toEqual([2, 2]);
225 expectArraysClose(await result.data(), [1 + 4, 10 + 5, 2 + 4, 20 + 5, -3 + 4, -30 + 5, -4 + 4, -40 + 5]);
226 });
227 it('throws when passed a as a non-tensor', () => {
228 expect(() => tf.add({}, tf.scalar(1)))
229 .toThrowError(/Argument 'a' passed to 'add' must be a Tensor/);
230 });
231 it('throws when passed b as a non-tensor', () => {
232 expect(() => tf.add(tf.scalar(1), {}))
233 .toThrowError(/Argument 'b' passed to 'add' must be a Tensor/);
234 });
235 it('upcasts when dtypes dont match', async () => {
236 let res = tf.add(tf.scalar(1, 'int32'), tf.scalar(1, 'float32'));
237 expect(res.dtype).toBe('float32');
238 expectArraysClose(await res.data(), [2]);
239 res = tf.add(tf.scalar(1, 'int32'), tf.scalar(true, 'bool'));
240 expect(res.dtype).toBe('int32');
241 expectArraysClose(await res.data(), [2]);
242 res = tf.add(tf.scalar(1, 'int32'), tf.scalar(false, 'bool'));
243 expect(res.dtype).toBe('int32');
244 expectArraysClose(await res.data(), [1]);
245 res = tf.add(tf.complex(4, 7), tf.scalar(1, 'float32'));
246 expect(res.dtype).toBe('complex64');
247 expectArraysClose(await res.data(), [5, 7]);
248 res = tf.add(tf.complex(4, 7), tf.scalar(1, 'int32'));
249 expect(res.dtype).toBe('complex64');
250 expectArraysClose(await res.data(), [5, 7]);
251 });
252 it('accepts a tensor-like object', async () => {
253 const result = tf.add(5, [1, 2, 3]);
254 expectArraysClose(await result.data(), [6, 7, 8]);
255 });
256});
257//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"add_test.js","sourceRoot":"","sources":["../../../../../../tfjs-core/src/ops/add_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, expectArraysEqual} from '../test_util';\n\ndescribeWithFlags('add', ALL_ENVS, () => {\n  it('c + A', async () => {\n    const c = tf.scalar(5);\n    const a = tf.tensor1d([1, 2, 3]);\n\n    const result = tf.add(c, a);\n\n    expectArraysClose(await result.data(), [6, 7, 8]);\n  });\n\n  it('c + A propagates NaNs', async () => {\n    const c = tf.scalar(NaN);\n    const a = tf.tensor1d([1, 2, 3]);\n\n    const res = tf.add(c, a);\n\n    expectArraysEqual(await res.data(), [NaN, NaN, NaN]);\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.add(a, b);\n\n    expect(result.shape).toEqual([2, 2]);\n    const expected = [3, 4, 0, -1];\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.add(a, b);\n\n    expect(result.shape).toEqual([2, 2]);\n    const expected = [2, 4, -2, -2];\n\n    expectArraysClose(await result.data(), expected);\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.add(a, b);\n\n    const expected = [6, 7, 0];\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.add(a, b);\n\n    const expected = [6, 7, 0];\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.add(b);\n\n    const expected = [6, 7, 0];\n    expectArraysClose(await result.data(), expected);\n  });\n\n  it('A + B propagates NaNs', async () => {\n    const a = tf.tensor1d([2, 5, NaN]);\n    const b = tf.tensor1d([4, 2, -1]);\n\n    const res = tf.add(a, b);\n    expectArraysClose(await res.data(), [6, 7, NaN]);\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.add(a, b)).toThrowError();\n    expect(() => tf.add(b, a)).toThrowError();\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.add(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [3, 4, 5, 6, 7, 8]);\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.add(a, b);\n    expect(res.shape).toEqual([6]);\n    expectArraysClose(await res.data(), [3, 4, 5, 6, 7, 8]);\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.add(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [8, 9, 12, 4, 5, 8]);\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.add(a, b);\n    expect(res.shape).toEqual([2, 3]);\n    expectArraysClose(await res.data(), [8, 9, 12, 7, 8, 9]);\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.add(a, b);\n    expect(res.shape).toEqual([2, 3, 1]);\n    expectArraysClose(await res.data(), [0, 1, 2, 3, 4, 5]);\n  });\n\n  it('6D+scalar', async () => {\n    const a = tf.range(0, 64).reshape([2, 2, 2, 2, 2, 2]);\n    const b = tf.scalar(-1);\n    const res = tf.add(a, b);\n    expect(res.shape).toEqual([2, 2, 2, 2, 2, 2]);\n    const expectedResult = [\n      -1, 0,  1,  2,  3,  4,  5,  6,  7,  8,  9,  10, 11, 12, 13, 14,\n      15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30,\n      31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46,\n      47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62\n    ];\n    expectArraysClose(await res.data(), expectedResult);\n  });\n\n  it('6D+2D', async () => {\n    const a = tf.range(0, 64).reshape([2, 2, 2, 2, 2, 2]);\n    const b = tf.tensor2d([11, 13, 17, 19], [2, 2]);\n    const res = tf.add(a, b);\n    expect(res.shape).toEqual([2, 2, 2, 2, 2, 2]);\n    const expectedResult = [\n      11, 14, 19, 22, 15, 18, 23, 26, 19, 22, 27, 30, 23, 26, 31, 34,\n      27, 30, 35, 38, 31, 34, 39, 42, 35, 38, 43, 46, 39, 42, 47, 50,\n      43, 46, 51, 54, 47, 50, 55, 58, 51, 54, 59, 62, 55, 58, 63, 66,\n      59, 62, 67, 70, 63, 66, 71, 74, 67, 70, 75, 78, 71, 74, 79, 82\n    ];\n    expectArraysClose(await res.data(), expectedResult);\n  });\n\n  it('add tensors with 0 in shape', async () => {\n    const a = tf.tensor1d([1]);\n    const b = tf.tensor3d([], [0, 0, 5]);\n    const res = tf.add(a, b);\n    expect(res.shape).toEqual([0, 0, 5]);\n    expectArraysEqual(await res.data(), []);\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.add(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 with clones', 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.add(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(), [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([2, 3], [2, 1]);\n    const b = tf.tensor2d([4, 5, 6, 7], [2, 2]);\n    const dy = tf.tensor2d([5, 4, 3, 2], [2, 2]);\n\n    const grads = tf.grads((a, b) => tf.add(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 addition', async () => {\n    const real1 = tf.tensor1d([1]);\n    const imag1 = tf.tensor1d([2]);\n    const complex1 = tf.complex(real1, imag1);\n\n    const real2 = tf.tensor1d([3]);\n    const imag2 = tf.tensor1d([4]);\n    const complex2 = tf.complex(real2, imag2);\n\n    const result = complex1.add(complex2);\n\n    expect(result.dtype).toBe('complex64');\n    expect(result.shape).toEqual([1]);\n    expectArraysClose(await result.data(), [4, 6]);\n  });\n\n  it('complex number reshape and then addition', async () => {\n    const real1 = tf.tensor1d([1]);\n    const imag1 = tf.tensor1d([2]);\n    const complex1 = tf.complex(real1, imag1);\n\n    const real2 = tf.tensor1d([3]);\n    const imag2 = tf.tensor1d([4]);\n    const complex2 = tf.complex(real2, imag2);\n\n    const complex1Reshaped = complex1.reshape([1, 1, 1]);\n    const complex2Reshaped = complex2.reshape([1, 1, 1]);\n\n    const result = complex1Reshaped.add(complex2Reshaped);\n\n    expect(result.dtype).toBe('complex64');\n    expect(result.shape).toEqual([1, 1, 1]);\n    expectArraysClose(await result.data(), [4, 6]);\n  });\n\n  it('complex number broadcasting addition', 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.add(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.add({} as tf.Tensor, tf.scalar(1)))\n        .toThrowError(/Argument 'a' passed to 'add' must be a Tensor/);\n  });\n  it('throws when passed b as a non-tensor', () => {\n    expect(() => tf.add(tf.scalar(1), {} as tf.Tensor))\n        .toThrowError(/Argument 'b' passed to 'add' must be a Tensor/);\n  });\n\n  it('upcasts when dtypes dont match', async () => {\n    let res = tf.add(tf.scalar(1, 'int32'), tf.scalar(1, 'float32'));\n    expect(res.dtype).toBe('float32');\n    expectArraysClose(await res.data(), [2]);\n\n    res = tf.add(tf.scalar(1, 'int32'), tf.scalar(true, 'bool'));\n    expect(res.dtype).toBe('int32');\n    expectArraysClose(await res.data(), [2]);\n\n    res = tf.add(tf.scalar(1, 'int32'), tf.scalar(false, 'bool'));\n    expect(res.dtype).toBe('int32');\n    expectArraysClose(await res.data(), [1]);\n\n    res = tf.add(tf.complex(4, 7), tf.scalar(1, 'float32'));\n    expect(res.dtype).toBe('complex64');\n    expectArraysClose(await res.data(), [5, 7]);\n\n    res = tf.add(tf.complex(4, 7), tf.scalar(1, 'int32'));\n    expect(res.dtype).toBe('complex64');\n    expectArraysClose(await res.data(), [5, 7]);\n  });\n\n  it('accepts a tensor-like object', async () => {\n    const result = tf.add(5, [1, 2, 3]);\n    expectArraysClose(await result.data(), [6, 7, 8]);\n  });\n});\n"]}
\No newline at end of file