UNPKG

47.4 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('nonMaxSuppression', ALL_ENVS, () => {
21 describe('NonMaxSuppression Basic', () => {
22 it('select from three clusters', async () => {
23 const boxes = tf.tensor2d([
24 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
25 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
26 ], [6, 4]);
27 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
28 const maxOutputSize = 3;
29 const iouThreshold = 0.5;
30 const scoreThreshold = 0;
31 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
32 expect(indices.shape).toEqual([3]);
33 expectArraysEqual(await indices.data(), [3, 0, 5]);
34 });
35 it('select from three clusters flipped coordinates', async () => {
36 const boxes = tf.tensor2d([
37 1, 1, 0, 0, 0, 0.1, 1, 1.1, 0, .9, 1, -0.1,
38 0, 10, 1, 11, 1, 10.1, 0, 11.1, 1, 101, 0, 100
39 ], [6, 4]);
40 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
41 const maxOutputSize = 3;
42 const iouThreshold = 0.5;
43 const scoreThreshold = 0;
44 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
45 expect(indices.shape).toEqual([3]);
46 expectArraysEqual(await indices.data(), [3, 0, 5]);
47 });
48 it('select at most two boxes from three clusters', async () => {
49 const boxes = tf.tensor2d([
50 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
51 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
52 ], [6, 4]);
53 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
54 const maxOutputSize = 2;
55 const iouThreshold = 0.5;
56 const scoreThreshold = 0;
57 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
58 expect(indices.shape).toEqual([2]);
59 expectArraysEqual(await indices.data(), [3, 0]);
60 });
61 it('select at most thirty boxes from three clusters', async () => {
62 const boxes = tf.tensor2d([
63 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
64 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
65 ], [6, 4]);
66 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
67 const maxOutputSize = 30;
68 const iouThreshold = 0.5;
69 const scoreThreshold = 0;
70 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
71 expect(indices.shape).toEqual([3]);
72 expectArraysEqual(await indices.data(), [3, 0, 5]);
73 });
74 it('select single box', async () => {
75 const boxes = tf.tensor2d([0, 0, 1, 1], [1, 4]);
76 const scores = tf.tensor1d([0.9]);
77 const maxOutputSize = 3;
78 const iouThreshold = 0.5;
79 const scoreThreshold = 0;
80 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
81 expect(indices.shape).toEqual([1]);
82 expectArraysEqual(await indices.data(), [0]);
83 });
84 it('select from ten identical boxes', async () => {
85 const numBoxes = 10;
86 const corners = new Array(numBoxes)
87 .fill(0)
88 .map(_ => [0, 0, 1, 1])
89 .reduce((arr, curr) => arr.concat(curr));
90 const boxes = tf.tensor2d(corners, [numBoxes, 4]);
91 const scores = tf.tensor1d(Array(numBoxes).fill(0.9));
92 const maxOutputSize = 3;
93 const iouThreshold = 0.5;
94 const scoreThreshold = 0;
95 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
96 expect(indices.shape).toEqual([1]);
97 expectArraysEqual(await indices.data(), [0]);
98 });
99 it('inconsistent box and score shapes', () => {
100 const boxes = tf.tensor2d([
101 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
102 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
103 ], [6, 4]);
104 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5]);
105 const maxOutputSize = 30;
106 const iouThreshold = 0.5;
107 const scoreThreshold = 0;
108 expect(() => tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold))
109 .toThrowError(/scores has incompatible shape with boxes/);
110 });
111 it('invalid iou threshold', () => {
112 const boxes = tf.tensor2d([0, 0, 1, 1], [1, 4]);
113 const scores = tf.tensor1d([0.9]);
114 const maxOutputSize = 3;
115 const iouThreshold = 1.2;
116 const scoreThreshold = 0;
117 expect(() => tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold))
118 .toThrowError(/iouThreshold must be in \[0, 1\]/);
119 });
120 it('empty input', async () => {
121 const boxes = tf.tensor2d([], [0, 4]);
122 const scores = tf.tensor1d([]);
123 const maxOutputSize = 3;
124 const iouThreshold = 0.5;
125 const scoreThreshold = 0;
126 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
127 expect(indices.shape).toEqual([0]);
128 expectArraysEqual(await indices.data(), []);
129 });
130 it('accepts a tensor-like object', async () => {
131 const boxes = [[0, 0, 1, 1], [0, 1, 1, 2]];
132 const scores = [1, 2];
133 const indices = tf.image.nonMaxSuppression(boxes, scores, 10);
134 expect(indices.shape).toEqual([2]);
135 expect(indices.dtype).toEqual('int32');
136 expectArraysEqual(await indices.data(), [1, 0]);
137 });
138 it('throws when boxes is int32', async () => {
139 const boxes = tf.tensor2d([[0, 0, 1, 1], [0, 1, 1, 2]], [2, 4], 'int32');
140 const scores = [1, 2];
141 expect(() => tf.image.nonMaxSuppression(boxes, scores, 10))
142 .toThrowError(/Argument 'boxes' passed to 'nonMaxSuppression' must be float32/);
143 });
144 it('throws when scores is int32', async () => {
145 const boxes = [[0, 0, 1, 1], [0, 1, 1, 2]];
146 const scores = tf.tensor1d([1, 2], 'int32');
147 const errRegex = /Argument 'scores' passed to 'nonMaxSuppression' must be float32/;
148 expect(() => tf.image.nonMaxSuppression(boxes, scores, 10))
149 .toThrowError(errRegex);
150 });
151 it('works when inputs are not explicitly initialized on the CPU', async () => {
152 // This test ensures that asynchronous backends work with NMS, which
153 // requires inputs to reside on the CPU.
154 const boxes = tf.tensor2d([
155 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
156 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
157 ], [6, 4]);
158 const a = tf.tensor1d([0, 1, -2, -4, 4, -4]);
159 const b = tf.tensor1d([0.15, 0.2, 0.25, 0.5, 0.7, 1.2]);
160 const scores = a.div(b);
161 const maxOutputSize = 2;
162 const iouThreshold = 0.5;
163 const scoreThreshold = 0;
164 await scores.data();
165 const indices = tf.image.nonMaxSuppression(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);
166 expect(indices.shape).toEqual([2]);
167 expectArraysEqual(await indices.data(), [4, 1]);
168 });
169 });
170 describe('NonMaxSuppressionWithScore', () => {
171 it('select from three clusters with SoftNMS', async () => {
172 const boxes = tf.tensor2d([
173 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
174 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
175 ], [6, 4]);
176 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
177 const maxOutputSize = 6;
178 const iouThreshold = 1.0;
179 const scoreThreshold = 0;
180 const softNmsSigma = 0.5;
181 const { selectedIndices, selectedScores } = tf.image.nonMaxSuppressionWithScore(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold, softNmsSigma);
182 expectArraysEqual(await selectedIndices.data(), [3, 0, 1, 5, 4, 2]);
183 expectArraysClose(await selectedScores.data(), [0.95, 0.9, 0.384, 0.3, 0.256, 0.197]);
184 });
185 });
186 describe('NonMaxSuppressionPadded', () => {
187 it('select from three clusters with pad five.', async () => {
188 const boxes = tf.tensor2d([
189 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
190 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
191 ], [6, 4]);
192 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
193 const maxOutputSize = 5;
194 const iouThreshold = 0.5;
195 const scoreThreshold = 0;
196 const before = tf.memory().numTensors;
197 const { selectedIndices, validOutputs } = tf.image.nonMaxSuppressionPadded(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold, true);
198 const after = tf.memory().numTensors;
199 expectArraysEqual(await selectedIndices.data(), [3, 0, 5, 0, 0]);
200 expectArraysEqual(await validOutputs.data(), 3);
201 expect(after).toEqual(before + 2);
202 });
203 it('select from three clusters with pad five and score threshold.', async () => {
204 const boxes = tf.tensor2d([
205 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
206 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
207 ], [6, 4]);
208 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
209 const maxOutputSize = 6;
210 const iouThreshold = 0.5;
211 const scoreThreshold = 0.4;
212 const before = tf.memory().numTensors;
213 const { selectedIndices, validOutputs } = tf.image.nonMaxSuppressionPadded(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold, true);
214 const after = tf.memory().numTensors;
215 expectArraysEqual(await selectedIndices.data(), [3, 0, 0, 0, 0, 0]);
216 expectArraysEqual(await validOutputs.data(), 2);
217 expect(after).toEqual(before + 2);
218 });
219 it('select from three clusters with no padding when pad option is false.', async () => {
220 const boxes = tf.tensor2d([
221 0, 0, 1, 1, 0, 0.1, 1, 1.1, 0, -0.1, 1, 0.9,
222 0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100, 1, 101
223 ], [6, 4]);
224 const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);
225 const maxOutputSize = 5;
226 const iouThreshold = 0.5;
227 const scoreThreshold = 0.0;
228 const { selectedIndices, validOutputs } = tf.image.nonMaxSuppressionPadded(boxes, scores, maxOutputSize, iouThreshold, scoreThreshold, false);
229 expectArraysEqual(await selectedIndices.data(), [3, 0, 5]);
230 expectArraysEqual(await validOutputs.data(), 3);
231 });
232 });
233});
234//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"non_max_suppression_test.js","sourceRoot":"","sources":["../../../../../../../tfjs-core/src/ops/image/non_max_suppression_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('nonMaxSuppression', ALL_ENVS, () => {\n  describe('NonMaxSuppression Basic', () => {\n    it('select from three clusters', async () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 3;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([3]);\n      expectArraysEqual(await indices.data(), [3, 0, 5]);\n    });\n\n    it('select from three clusters flipped coordinates', async () => {\n      const boxes = tf.tensor2d(\n          [\n            1, 1,  0, 0,  0, 0.1,  1, 1.1,  0, .9,  1, -0.1,\n            0, 10, 1, 11, 1, 10.1, 0, 11.1, 1, 101, 0, 100\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 3;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([3]);\n      expectArraysEqual(await indices.data(), [3, 0, 5]);\n    });\n\n    it('select at most two boxes from three clusters', async () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 2;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([2]);\n      expectArraysEqual(await indices.data(), [3, 0]);\n    });\n\n    it('select at most thirty boxes from three clusters', async () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 30;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([3]);\n      expectArraysEqual(await indices.data(), [3, 0, 5]);\n    });\n\n    it('select single box', async () => {\n      const boxes = tf.tensor2d([0, 0, 1, 1], [1, 4]);\n      const scores = tf.tensor1d([0.9]);\n      const maxOutputSize = 3;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([1]);\n      expectArraysEqual(await indices.data(), [0]);\n    });\n\n    it('select from ten identical boxes', async () => {\n      const numBoxes = 10;\n      const corners = new Array(numBoxes)\n                          .fill(0)\n                          .map(_ => [0, 0, 1, 1])\n                          .reduce((arr, curr) => arr.concat(curr));\n      const boxes = tf.tensor2d(corners, [numBoxes, 4]);\n      const scores = tf.tensor1d(Array(numBoxes).fill(0.9));\n      const maxOutputSize = 3;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([1]);\n      expectArraysEqual(await indices.data(), [0]);\n    });\n\n    it('inconsistent box and score shapes', () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5]);\n      const maxOutputSize = 30;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      expect(\n          () => tf.image.nonMaxSuppression(\n              boxes, scores, maxOutputSize, iouThreshold, scoreThreshold))\n          .toThrowError(/scores has incompatible shape with boxes/);\n    });\n\n    it('invalid iou threshold', () => {\n      const boxes = tf.tensor2d([0, 0, 1, 1], [1, 4]);\n      const scores = tf.tensor1d([0.9]);\n      const maxOutputSize = 3;\n      const iouThreshold = 1.2;\n      const scoreThreshold = 0;\n      expect(\n          () => tf.image.nonMaxSuppression(\n              boxes, scores, maxOutputSize, iouThreshold, scoreThreshold))\n          .toThrowError(/iouThreshold must be in \\[0, 1\\]/);\n    });\n\n    it('empty input', async () => {\n      const boxes = tf.tensor2d([], [0, 4]);\n      const scores = tf.tensor1d([]);\n      const maxOutputSize = 3;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n      const indices = tf.image.nonMaxSuppression(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold);\n\n      expect(indices.shape).toEqual([0]);\n      expectArraysEqual(await indices.data(), []);\n    });\n\n    it('accepts a tensor-like object', async () => {\n      const boxes = [[0, 0, 1, 1], [0, 1, 1, 2]];\n      const scores = [1, 2];\n      const indices = tf.image.nonMaxSuppression(boxes, scores, 10);\n      expect(indices.shape).toEqual([2]);\n      expect(indices.dtype).toEqual('int32');\n      expectArraysEqual(await indices.data(), [1, 0]);\n    });\n\n    it('throws when boxes is int32', async () => {\n      const boxes = tf.tensor2d([[0, 0, 1, 1], [0, 1, 1, 2]], [2, 4], 'int32');\n      const scores = [1, 2];\n      expect(() => tf.image.nonMaxSuppression(boxes, scores, 10))\n          .toThrowError(\n              /Argument 'boxes' passed to 'nonMaxSuppression' must be float32/);\n    });\n\n    it('throws when scores is int32', async () => {\n      const boxes = [[0, 0, 1, 1], [0, 1, 1, 2]];\n      const scores = tf.tensor1d([1, 2], 'int32');\n      const errRegex =\n          /Argument 'scores' passed to 'nonMaxSuppression' must be float32/;\n      expect(() => tf.image.nonMaxSuppression(boxes, scores, 10))\n          .toThrowError(errRegex);\n    });\n\n    it('works when inputs are not explicitly initialized on the CPU',\n       async () => {\n         // This test ensures that asynchronous backends work with NMS, which\n         // requires inputs to reside on the CPU.\n         const boxes = tf.tensor2d(\n             [\n               0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n               0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n             ],\n             [6, 4]);\n         const a = tf.tensor1d([0, 1, -2, -4, 4, -4]);\n         const b = tf.tensor1d([0.15, 0.2, 0.25, 0.5, 0.7, 1.2]);\n         const scores = a.div(b);\n         const maxOutputSize = 2;\n         const iouThreshold = 0.5;\n         const scoreThreshold = 0;\n         await scores.data();\n         const indices = tf.image.nonMaxSuppression(\n             boxes, scores as tf.Tensor1D, maxOutputSize, iouThreshold,\n             scoreThreshold);\n\n         expect(indices.shape).toEqual([2]);\n         expectArraysEqual(await indices.data(), [4, 1]);\n       });\n  });\n\n  describe('NonMaxSuppressionWithScore', () => {\n    it('select from three clusters with SoftNMS', async () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 6;\n      const iouThreshold = 1.0;\n      const scoreThreshold = 0;\n      const softNmsSigma = 0.5;\n\n      const {selectedIndices, selectedScores} =\n          tf.image.nonMaxSuppressionWithScore(\n              boxes, scores, maxOutputSize, iouThreshold, scoreThreshold,\n              softNmsSigma);\n\n      expectArraysEqual(await selectedIndices.data(), [3, 0, 1, 5, 4, 2]);\n\n      expectArraysClose(\n          await selectedScores.data(), [0.95, 0.9, 0.384, 0.3, 0.256, 0.197]);\n    });\n  });\n\n  describe('NonMaxSuppressionPadded', () => {\n    it('select from three clusters with pad five.', async () => {\n      const boxes = tf.tensor2d(\n          [\n            0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n            0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n          ],\n          [6, 4]);\n      const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n      const maxOutputSize = 5;\n      const iouThreshold = 0.5;\n      const scoreThreshold = 0;\n\n      const before = tf.memory().numTensors;\n      const {selectedIndices, validOutputs} = tf.image.nonMaxSuppressionPadded(\n          boxes, scores, maxOutputSize, iouThreshold, scoreThreshold, true);\n      const after = tf.memory().numTensors;\n\n      expectArraysEqual(await selectedIndices.data(), [3, 0, 5, 0, 0]);\n      expectArraysEqual(await validOutputs.data(), 3);\n      expect(after).toEqual(before + 2);\n    });\n\n    it('select from three clusters with pad five and score threshold.',\n       async () => {\n         const boxes = tf.tensor2d(\n             [\n               0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n               0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n             ],\n             [6, 4]);\n         const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n         const maxOutputSize = 6;\n         const iouThreshold = 0.5;\n         const scoreThreshold = 0.4;\n\n         const before = tf.memory().numTensors;\n         const {selectedIndices, validOutputs} =\n             tf.image.nonMaxSuppressionPadded(\n                 boxes, scores, maxOutputSize, iouThreshold, scoreThreshold,\n                 true);\n         const after = tf.memory().numTensors;\n\n         expectArraysEqual(await selectedIndices.data(), [3, 0, 0, 0, 0, 0]);\n         expectArraysEqual(await validOutputs.data(), 2);\n         expect(after).toEqual(before + 2);\n       });\n\n    it('select from three clusters with no padding when pad option is false.',\n       async () => {\n         const boxes = tf.tensor2d(\n             [\n               0, 0,  1, 1,  0, 0.1,  1, 1.1,  0, -0.1, 1, 0.9,\n               0, 10, 1, 11, 0, 10.1, 1, 11.1, 0, 100,  1, 101\n             ],\n             [6, 4]);\n         const scores = tf.tensor1d([0.9, 0.75, 0.6, 0.95, 0.5, 0.3]);\n         const maxOutputSize = 5;\n         const iouThreshold = 0.5;\n         const scoreThreshold = 0.0;\n\n         const {selectedIndices, validOutputs} =\n             tf.image.nonMaxSuppressionPadded(\n                 boxes, scores, maxOutputSize, iouThreshold, scoreThreshold,\n                 false);\n\n         expectArraysEqual(await selectedIndices.data(), [3, 0, 5]);\n         expectArraysEqual(await validOutputs.data(), 3);\n       });\n  });\n});\n"]}
\No newline at end of file