/* Copyright 2015 The TensorFlow Authors. 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. ==============================================================================*/ #include #include #include #include #include "tensorflow/core/debug/debug_io_utils.h" #include "tensorflow/core/debug/debug_node_key.h" #include "tensorflow/core/framework/fake_input.h" #include "tensorflow/core/framework/node_def_builder.h" #include "tensorflow/core/framework/summary.pb.h" #include "tensorflow/core/framework/tensor.h" #include "tensorflow/core/framework/tensor_testutil.h" #include "tensorflow/core/framework/types.h" #include "tensorflow/core/framework/types.pb.h" #include "tensorflow/core/kernels/ops_testutil.h" #include "tensorflow/core/kernels/ops_util.h" #include "tensorflow/core/lib/io/path.h" #include "tensorflow/core/lib/strings/strcat.h" #include "tensorflow/core/platform/env.h" #include "tensorflow/core/platform/test.h" #include "tensorflow/core/util/event.pb.h" namespace tensorflow { class DebugIdentityOpTest : public OpsTestBase { protected: Status Init(DataType input_type, const std::vector& debug_urls) { env_ = Env::Default(); TF_CHECK_OK(NodeDefBuilder("op", "DebugIdentity") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Attr("debug_urls", debug_urls) .Finalize(node_def())); return InitOp(); } Status Init(DataType input_type) { std::vector empty_debug_urls; return Init(input_type, empty_debug_urls); } Env* env_; }; TEST_F(DebugIdentityOpTest, Int32Success_6) { TF_ASSERT_OK(Init(DT_INT32)); AddInputFromArray(TensorShape({6}), {1, 2, 3, 4, 5, 6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_INT32, TensorShape({6})); test::FillValues(&expected, {1, 2, 3, 4, 5, 6}); // Verify the identity output test::ExpectTensorEqual(expected, *GetOutput(0)); } TEST_F(DebugIdentityOpTest, Int32Success_6_FileURLs) { const int kNumDumpDirs = 3; const string tmp_dir = testing::TmpDir(); std::vector dump_roots; std::vector debug_urls; for (int i = 0; i < kNumDumpDirs; ++i) { const string dump_root = strings::StrCat(tmp_dir, "_", i); dump_roots.push_back(dump_root); debug_urls.push_back(strings::StrCat("file://", dump_root)); } uint64 wall_time = Env::Default()->NowMicros(); TF_ASSERT_OK(Init(DT_INT32, debug_urls)); AddInputFromArray(TensorShape({6}), {1, 2, 3, 4, 5, 6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_INT32, TensorShape({6})); test::FillValues(&expected, {1, 2, 3, 4, 5, 6}); // Verify the identity output test::ExpectTensorEqual(expected, *GetOutput(0)); for (int i = 0; i < kNumDumpDirs; ++i) { ASSERT_TRUE(env_->FileExists(dump_roots[i]).ok()); ASSERT_TRUE(env_->IsDirectory(dump_roots[i]).ok()); std::vector device_roots; DIR* dir0 = opendir(dump_roots[i].c_str()); struct dirent* ent0; const string kDeviceDirPrefix = strings::StrCat( DebugNodeKey::kMetadataFilePrefix, DebugNodeKey::kDeviceTag); while ((ent0 = readdir(dir0)) != nullptr) { if (!strncmp(ent0->d_name, kDeviceDirPrefix.c_str(), kDeviceDirPrefix.size())) { device_roots.push_back(io::JoinPath(dump_roots[i], ent0->d_name)); } } ASSERT_EQ(1, device_roots.size()); closedir(dir0); const string& device_root = device_roots[0]; DIR* dir = opendir(device_root.c_str()); struct dirent* ent; int dump_files_found = 0; while ((ent = readdir(dir)) != nullptr) { if (strcmp(ent->d_name, ".") && strcmp(ent->d_name, "..")) { dump_files_found++; // Try reading the file into a Event proto. const string dump_file_path = io::JoinPath(device_root, ent->d_name); std::fstream ifs(dump_file_path, std::ios::in | std::ios::binary); Event event; event.ParseFromIstream(&ifs); ifs.close(); ASSERT_GE(event.wall_time(), wall_time); ASSERT_EQ(1, event.summary().value().size()); ASSERT_EQ(strings::StrCat("FakeTensor", ":", 0, ":", "DebugIdentity"), event.summary().value(0).node_name()); Tensor tensor_prime(DT_INT32); ASSERT_TRUE(tensor_prime.FromProto(event.summary().value(0).tensor())); // Verify tensor shape and value from the dump file. ASSERT_EQ(TensorShape({6}), tensor_prime.shape()); for (int j = 0; j < 6; ++j) { ASSERT_EQ(j + 1, tensor_prime.flat()(j)); } } } closedir(dir); ASSERT_EQ(1, dump_files_found); // Remove temporary dump directory and file. int64 undeleted_files = 0; int64 undeleted_dirs = 0; ASSERT_TRUE(env_->DeleteRecursively(dump_roots[i], &undeleted_files, &undeleted_dirs) .ok()); ASSERT_EQ(0, undeleted_files); ASSERT_EQ(0, undeleted_dirs); } } TEST_F(DebugIdentityOpTest, Int32Success_2_3) { TF_ASSERT_OK(Init(DT_INT32)); AddInputFromArray(TensorShape({2, 3}), {1, 2, 3, 4, 5, 6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_INT32, TensorShape({2, 3})); test::FillValues(&expected, {1, 2, 3, 4, 5, 6}); test::ExpectTensorEqual(expected, *GetOutput(0)); } TEST_F(DebugIdentityOpTest, StringSuccess) { TF_ASSERT_OK(Init(DT_STRING)); AddInputFromArray(TensorShape({6}), {"A", "b", "C", "d", "E", "f"}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_STRING, TensorShape({6})); test::FillValues(&expected, {"A", "b", "C", "d", "E", "f"}); test::ExpectTensorEqual(expected, *GetOutput(0)); } // Tests for DebugNanCountOp class DebugNanCountOpTest : public OpsTestBase { protected: Status Init(DataType input_type) { TF_CHECK_OK(NodeDefBuilder("op", "DebugNanCount") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Finalize(node_def())); return InitOp(); } }; TEST_F(DebugNanCountOpTest, Float_has_NaNs) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray(TensorShape({6}), {1.1, std::numeric_limits::quiet_NaN(), 3.3, std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), 6.6}); TF_ASSERT_OK(RunOpKernel()); // Verify the NaN-count debug signal Tensor expected_nan_count(allocator(), DT_INT64, TensorShape({1})); test::FillValues(&expected_nan_count, {3}); test::ExpectTensorEqual(expected_nan_count, *GetOutput(0)); } TEST_F(DebugNanCountOpTest, Float_no_NaNs) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray( TensorShape({6}), {1.1, 2.2, 3.3, std::numeric_limits::infinity(), 5.5, 6.6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected_nan_count(allocator(), DT_INT64, TensorShape({1})); test::FillValues(&expected_nan_count, {0}); test::ExpectTensorEqual(expected_nan_count, *GetOutput(0)); } TEST_F(DebugNanCountOpTest, Double_has_NaNs) { TF_ASSERT_OK(Init(DT_DOUBLE)); AddInputFromArray(TensorShape({6}), {1.1, std::numeric_limits::quiet_NaN(), 3.3, std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), 6.6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected_nan_count(allocator(), DT_INT64, TensorShape({1})); test::FillValues(&expected_nan_count, {3}); test::ExpectTensorEqual(expected_nan_count, *GetOutput(0)); } TEST_F(DebugNanCountOpTest, Double_no_NaNs) { TF_ASSERT_OK(Init(DT_DOUBLE)); AddInputFromArray( TensorShape({6}), {1.1, 2.2, 3.3, std::numeric_limits::infinity(), 5.5, 6.6}); TF_ASSERT_OK(RunOpKernel()); Tensor expected_nan_count(allocator(), DT_INT64, TensorShape({1})); test::FillValues(&expected_nan_count, {0}); test::ExpectTensorEqual(expected_nan_count, *GetOutput(0)); } // Tests for DebugNumericSummaryOp class DebugNumericSummaryOpTest : public OpsTestBase { protected: Status Init(DataType input_type) { TF_CHECK_OK(NodeDefBuilder("op", "DebugNumericSummary") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Finalize(node_def())); return InitOp(); } Status InitGated(DataType input_type, const std::vector& debug_urls) { TF_CHECK_OK(NodeDefBuilder("op", "DebugNumericSummary") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Attr("gated_grpc", true) .Attr("debug_urls", debug_urls) .Finalize(node_def())); return InitOp(); } #if defined(PLATFORM_GOOGLE) void ClearEnabledWatchKeys() { DebugGrpcIO::ClearEnabledWatchKeys(); } #endif }; TEST_F(DebugNumericSummaryOpTest, Float_full_house) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray( TensorShape({18}), {std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), 0.0f, 0.0f, 0.0f, -1.0f, -3.0f, 3.0f, 7.0f, -std::numeric_limits::infinity(), -std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN()}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({15})); test::FillValues( &expected, {1.0, // Is initialized. 18.0, // Total element count. 4.0, // nan count. 2.0, // -inf count. 2.0, // negative number count (excluding -inf). 3.0, // zero count. 2.0, // positive number count (excluding +inf). 5.0, // +inf count. -3.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 0.85714285714, // mean of non-inf and non-nan elements. 8.97959183673, // variance of non-inf and non-nan elements. static_cast(DT_FLOAT), // dtype. 1.0, // Number of dimensions. 18.0}); // Dimension size. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Double_full_house) { TF_ASSERT_OK(Init(DT_DOUBLE)); AddInputFromArray( TensorShape({18}), {std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), 0.0, 0.0, 0.0, -1.0, -3.0, 3.0, 7.0, -std::numeric_limits::infinity(), -std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN()}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({15})); test::FillValues( &expected, {1.0, // Is initialized. 18.0, // Total element count. 4.0, // nan count. 2.0, // -inf count. 2.0, // negative count (excluding -inf). 3.0, // zero count. 2.0, // positive count (excluding +inf). 5.0, // +inf count. -3.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 0.85714285714, // mean of non-inf and non-nan elements. 8.97959183673, // variance of non-inf and non-nan elements. static_cast(DT_DOUBLE), // dtype. 1.0, // Number of dimensions. 18.0}); // Dimension size. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Float_only_valid_values) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray(TensorShape({2, 3}), {0.0f, 0.0f, -1.0f, 3.0f, 3.0f, 7.0f}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues( &expected, {1.0, // Is initialized. 6.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 1.0, // negative count (excluding -inf). 2.0, // zero count. 3.0, // positive count (excluding +inf). 0.0, // +inf count. -1.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 2.0, // mean of non-inf and non-nan elements. 7.33333333333, // variance of non-inf and non-nan elements. static_cast(DT_FLOAT), // dtype 2.0, // Number of dimensions. 2.0, 3.0}); // Dimensoin sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Float_all_Inf_or_NaN) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray(TensorShape({3, 3}), {std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), -std::numeric_limits::infinity(), -std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN()}); TF_ASSERT_OK(RunOpKernel()); Tensor output_tensor = *GetOutput(0); const double* output = output_tensor.template flat().data(); // Use ASSERT_NEAR below because test::ExpectTensorNear does not work with // NaNs. ASSERT_NEAR(1.0, output[0], 1e-8); // Is initialized. ASSERT_NEAR(9.0, output[1], 1e-8); // Total element count. ASSERT_NEAR(4.0, output[2], 1e-8); // nan count. ASSERT_NEAR(2.0, output[3], 1e-8); // -inf count. ASSERT_NEAR(0.0, output[4], 1e-8); // negative count (excluding -inf). ASSERT_NEAR(0.0, output[5], 1e-8); // zero count. ASSERT_NEAR(0.0, output[6], 1e-8); // positive count (excluding +inf). ASSERT_NEAR(3.0, output[7], 1e-8); // +inf count. // Due to the absence of any non-inf and non-nan values, the output of min, // max, mean and var are all degenerate. ASSERT_EQ(std::numeric_limits::infinity(), output[8]); ASSERT_EQ(-std::numeric_limits::infinity(), output[9]); ASSERT_TRUE(Eigen::numext::isnan(output[10])); ASSERT_TRUE(Eigen::numext::isnan(output[11])); ASSERT_EQ(static_cast(DT_FLOAT), output[12]); ASSERT_EQ(2.0, output[13]); ASSERT_EQ(3.0, output[14]); ASSERT_EQ(3.0, output[15]); } TEST_F(DebugNumericSummaryOpTest, Many_dimensions_tensor_shape) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray(TensorShape({1, 3, 1, 1, 1, 1, 1}), {std::numeric_limits::quiet_NaN(), -std::numeric_limits::infinity(), -8.0}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({21})); test::FillValues(&expected, {1.0, // Is initialized. 3.0, // Total element count. 1.0, // nan count. 1.0, // -inf count. 1.0, // negative number count (excluding -inf). 0.0, // zero count. 0.0, // positive number count (excluding +inf). 0.0, // +inf count. -8.0, // minimum of non-inf and non-nan elements. -8.0, // maximum of non-inf and non-nan elements. -8.0, // mean of non-inf and non-nan elements. 0.0, // variance of non-inf and non-nan elements. static_cast(DT_FLOAT), // dtype. 7.0, // Number of dimensions. 1.0, 3.0, 1.0, 1.0, 1.0, 1.0, 1.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Scalar_tensor_shape) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray(TensorShape({}), {42.0}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({14})); test::FillValues(&expected, {1.0, // Is initialized. 1.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 0.0, // negative number count (excluding -inf). 0.0, // zero count. 1.0, // positive number count (excluding +inf). 0.0, // +inf count. 42.0, // minimum of non-inf and non-nan elements. 42.0, // maximum of non-inf and non-nan elements. 42.0, // mean of non-inf and non-nan elements. 0.0, // variance of non-inf and non-nan elements. static_cast(DT_FLOAT), // dtype. 0.0}); // Number of dimensions. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Int16Success) { TF_ASSERT_OK(Init(DT_INT16)); AddInputFromArray(TensorShape({4, 1}), {-1, -3, 3, 7}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues(&expected, {1.0, // Is initialized. 4.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 2.0, // negative count (excluding -inf). 0.0, // zero count. 2.0, // positive count (excluding +inf). 0.0, // +inf count. -3.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 1.5, // mean of non-inf and non-nan elements. 14.75, // variance of non-inf and non-nan elements. static_cast(DT_INT16), // dtype. 2.0, // Number of dimensions. 4.0, 1.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Int32Success) { TF_ASSERT_OK(Init(DT_INT32)); AddInputFromArray(TensorShape({2, 3}), {0, 0, -1, 3, 3, 7}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues( &expected, {1.0, // Is initialized. 6.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 1.0, // negative count (excluding -inf). 2.0, // zero count. 3.0, // positive count (excluding +inf). 0.0, // +inf count. -1.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 2.0, // mean of non-inf and non-nan elements. 7.33333333333, // variance of non-inf and non-nan elements. static_cast(DT_INT32), // dtype. 2.0, // Number of dimensions. 2.0, 3.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, Int64Success) { TF_ASSERT_OK(Init(DT_INT64)); AddInputFromArray(TensorShape({2, 2, 2}), {0, 0, -1, 3, 3, 7, 0, 0}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({17})); test::FillValues(&expected, {1.0, // Is initialized. 8.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 1.0, // negative count (excluding -inf). 4.0, // zero count. 3.0, // positive count (excluding +inf). 0.0, // +inf count. -1.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 1.5, // mean of non-inf and non-nan elements. 6.25, // variance of non-inf and non-nan elements. static_cast(DT_INT64), // dtype. 3.0, // Number of dimensions. 2.0, 2.0, 2.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, UInt8Success) { TF_ASSERT_OK(Init(DT_UINT8)); AddInputFromArray(TensorShape({1, 5}), {0, 10, 30, 30, 70}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues(&expected, {1.0, // Is initialized. 5.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 0.0, // negative count (excluding -inf). 1.0, // zero count. 4.0, // positive count (excluding +inf). 0.0, // +inf count. 0.0, // minimum of non-inf and non-nan elements. 70.0, // maximum of non-inf and non-nan elements. 28.0, // mean of non-inf and non-nan elements. 576.0, // variance of non-inf and non-nan elements. static_cast(DT_UINT8), // dtypes. 2.0, // Number of dimensions. 1.0, 5.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, BoolSuccess) { TF_ASSERT_OK(Init(DT_BOOL)); AddInputFromArray(TensorShape({2, 3}), {false, false, true, true, true, false}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues(&expected, {1.0, // Is initialized. 6.0, // Total element count. 0.0, // nan count. 0.0, // -inf count. 0.0, // negative count (excluding -inf). 3.0, // zero count. 3.0, // positive count (excluding +inf). 0.0, // +inf count. 0.0, // minimum of non-inf and non-nan elements. 1.0, // maximum of non-inf and non-nan elements. 0.5, // mean of non-inf and non-nan elements. 0.25, // variance of non-inf and non-nan elements. static_cast(DT_BOOL), // dtype. 2.0, // Number of dimensions. 2.0, 3.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } #if defined(PLATFORM_GOOGLE) TEST_F(DebugNumericSummaryOpTest, DisabledDueToEmptyEnabledSet) { ClearEnabledWatchKeys(); std::vector debug_urls({"grpc://server:3333"}); TF_ASSERT_OK(InitGated(DT_FLOAT, debug_urls)); AddInputFromArray(TensorShape({2, 2}), {1.0, 3.0, 3.0, 7.0}); TF_ASSERT_OK(RunOpKernel()); Tensor expected_disabled(allocator(), DT_DOUBLE, TensorShape({0})); test::ExpectTensorNear(expected_disabled, *GetOutput(0), 1e-8); } TEST_F(DebugNumericSummaryOpTest, DisabledDueToNonMatchingWatchKey) { ClearEnabledWatchKeys(); DebugGrpcIO::SetDebugNodeKeyGrpcState( "grpc://server:3333", "FakeTensor:1:DebugNumeriSummary", EventReply::DebugOpStateChange::READ_ONLY); std::vector debug_urls({"grpc://server:3333"}); TF_ASSERT_OK(InitGated(DT_FLOAT, debug_urls)); AddInputFromArray(TensorShape({2, 2}), {1.0, 3.0, 3.0, 7.0}); TF_ASSERT_OK(RunOpKernel()); Tensor expected_disabled(allocator(), DT_DOUBLE, TensorShape({0})); test::ExpectTensorNear(expected_disabled, *GetOutput(0), 1e-8); } #endif // Tests for DebugNumericSummaryOp class DebugNumericSummaryOpCustomLowerBoundTest : public OpsTestBase { protected: Status Init(DataType input_type) { TF_CHECK_OK(NodeDefBuilder("op", "DebugNumericSummary") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Attr("lower_bound", -1.2f) .Finalize(node_def())); return InitOp(); } }; TEST_F(DebugNumericSummaryOpCustomLowerBoundTest, Float_full_house) { TF_ASSERT_OK(Init(DT_FLOAT)); AddInputFromArray( TensorShape({18}), {std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN(), 0.0f, 0.0f, 0.0f, -1.0f, -3.0f, 3.0f, 7.0f, -std::numeric_limits::infinity(), -std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::infinity(), std::numeric_limits::quiet_NaN(), std::numeric_limits::quiet_NaN()}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({15})); test::FillValues( &expected, {1.0, // Is initialized. 18.0, // Total element count. 4.0, // nan count. 3.0, // -inf count. 1.0, // negative number count (excluding -inf). 3.0, // zero count. 2.0, // positive number count (excluding +inf). 5.0, // +inf count. -3.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 0.85714285714, // mean of non-inf and non-nan elements. 8.97959183673, // variance of non-inf and non-nan elements. static_cast(DT_FLOAT), // dtype. 1.0, // Number of dimensions. 18.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } // Tests for DebugNumericSummaryOp class DebugNumericSummaryOpCustomLowerUpperBoundsTest : public OpsTestBase { protected: Status Init(DataType input_type) { TF_CHECK_OK(NodeDefBuilder("op", "DebugNumericSummary") .Input(FakeInput(input_type)) .Attr("tensor_name", "FakeTensor:0") .Attr("lower_bound", -0.5f) .Attr("upper_bound", 3.6f) .Finalize(node_def())); return InitOp(); } }; TEST_F(DebugNumericSummaryOpCustomLowerUpperBoundsTest, Int32Success) { TF_ASSERT_OK(Init(DT_INT32)); AddInputFromArray(TensorShape({2, 3}), {0, 0, -1, 3, 3, 7}); TF_ASSERT_OK(RunOpKernel()); Tensor expected(allocator(), DT_DOUBLE, TensorShape({16})); test::FillValues( &expected, {1.0, // Is initialized. 6.0, // Total element count. 0.0, // nan count. 1.0, // -inf count. 0.0, // negative count (excluding -inf). 2.0, // zero count. 2.0, // positive count (excluding +inf). 1.0, // +inf count. -1.0, // minimum of non-inf and non-nan elements. 7.0, // maximum of non-inf and non-nan elements. 2.0, // mean of non-inf and non-nan elements. 7.33333333333, // variance of non-inf and non-nan elements. static_cast(DT_INT32), // dtype. 2.0, // Number of dimensions. 2.0, 3.0}); // Dimension sizes. test::ExpectTensorNear(expected, *GetOutput(0), 1e-8); } } // namespace tensorflow