Skip to content

Commit

Permalink
sensors: Remove uses of spruce
Browse files Browse the repository at this point in the history
  • Loading branch information
jwnimmer-tri committed Oct 11, 2019
1 parent 871d58b commit 43e9197
Show file tree
Hide file tree
Showing 4 changed files with 80 additions and 76 deletions.
4 changes: 2 additions & 2 deletions systems/sensors/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -222,8 +222,8 @@ drake_cc_library(
deps = [
":image",
"//common:essential",
"//common:filesystem",
"//systems/framework",
"@spruce",
"@vtk//:vtkCommonDataModel",
"@vtk//:vtkIOImage",
],
Expand Down Expand Up @@ -280,9 +280,9 @@ drake_cc_googletest(
tags = vtk_test_tags(),
deps = [
":image_writer",
"//common:filesystem",
"//common:temp_directory",
"//common/test_utilities",
"@spruce",
"@vtk//:vtkIOImage",
],
)
Expand Down
33 changes: 22 additions & 11 deletions systems/sensors/image_writer.cc
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,15 @@
#include <utility>
#include <vector>

#include <spruce.hh>
#include <vtkImageData.h>
#include <vtkNew.h>
#include <vtkPNGWriter.h>
#include <vtkSmartPointer.h>
#include <vtkTIFFWriter.h>
#include "fmt/ostream.h"

#include "drake/common/filesystem.h"

namespace drake {
namespace systems {
namespace sensors {
Expand Down Expand Up @@ -107,9 +108,9 @@ const InputPort<double>& ImageWriter::DeclareImageInputPort(
}

// Confirms the implied directory is valid.
spruce::path test_dir{
DirectoryFromFormat(file_name_format, port_name, kPixelType)};
FolderState folder_state = ValidateDirectory(test_dir.getStr());
const std::string test_dir =
DirectoryFromFormat(file_name_format, port_name, kPixelType);
FolderState folder_state = ValidateDirectory(test_dir);
if (folder_state != FolderState::kValid) {
const char* const reason = [folder_state]() {
switch (folder_state) {
Expand All @@ -127,7 +128,7 @@ const InputPort<double>& ImageWriter::DeclareImageInputPort(
throw std::logic_error(
fmt::format("ImageWriter: The format string `{}` implied the invalid "
"directory: '{}'; {}",
file_name_format, test_dir.getStr(), reason));
file_name_format, test_dir, reason));
}

// Confirms file has appropriate extension.
Expand Down Expand Up @@ -185,9 +186,19 @@ std::string ImageWriter::MakeFileName(const std::string& format,
std::string ImageWriter::DirectoryFromFormat(const std::string& format,
const std::string& port_name,
PixelType pixel_type) const {
// Extract the directory.
// Extract the directory. Note that in any error messages to the user, we'll
// report using the argument name from the public method.
if (format.empty()) {
throw std::logic_error(
"ImageWriter: The file_name_format cannot be empty");
}
if (format.back() == '/') {
throw std::logic_error(fmt::format(
"ImageWriter: The file_name_format '{}' cannot end with a '/'",
format));
}
size_t index = format.rfind('/');
std::string dir_format = format.substr(0, index + 1);
std::string dir_format = format.substr(0, index);
// NOTE: [bcdelmosu] are all the characters in: double, msec, and usec.
// Technically, this will also key on '{time_mouse}', but if someone is
// putting that in their file path, they deserve whatever they get.
Expand All @@ -203,10 +214,10 @@ std::string ImageWriter::DirectoryFromFormat(const std::string& format,

ImageWriter::FolderState ImageWriter::ValidateDirectory(
const std::string& file_path_str) {
spruce::path file_path(file_path_str);
if (file_path.exists()) {
if (file_path.isDir()) {
if (access(file_path.getStr().c_str(), W_OK) == 0) {
filesystem::path file_path(file_path_str);
if (filesystem::exists(file_path)) {
if (filesystem::is_directory(file_path)) {
if (::access(file_path.string().c_str(), W_OK) == 0) {
return FolderState::kValid;
} else {
return FolderState::kUnwritable;
Expand Down
3 changes: 2 additions & 1 deletion systems/sensors/image_writer.h
Original file line number Diff line number Diff line change
Expand Up @@ -207,10 +207,11 @@ class ImageWriter : public LeafSystem<double> {

// Given the file format string (and port-specific configuration values),
// extracts, tests, and returns the output folder information.
// The return value will not contain a trailing slash.
// The tests are in support of the statement that the directory path cannot
// depend on time.
// Examples:
// "a/b/c/" --> "a/b/c/"
// "a/b/c/" --> thrown exception.
// "a/b/c" --> "a/b"
// "a/{time_usec}/c" --> thrown exception.
// "a/{port_name}/c" --> "a/my_port" (assuming port_name = "my_port").
Expand Down
116 changes: 54 additions & 62 deletions systems/sensors/test/image_writer_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@
#include <string>

#include <gtest/gtest.h>
#include <spruce.hh>
#include <vtkImageData.h>
#include <vtkImageExport.h>
#include <vtkNew.h>
Expand All @@ -16,6 +15,7 @@
#include <vtkTIFFReader.h>

#include "drake/common/drake_copyable.h"
#include "drake/common/filesystem.h"
#include "drake/common/temp_directory.h"
#include "drake/common/test_utilities/expect_throws_message.h"
#include "drake/systems/framework/event_collection.h"
Expand All @@ -31,11 +31,10 @@ class ImageWriterTester {

explicit ImageWriterTester(const ImageWriter& writer) : writer_(writer) {}

spruce::path DirectoryFromFormat(const std::string& format,
const std::string& port_name,
PixelType pixel_type) const {
return spruce::path(
writer_.DirectoryFromFormat(format, port_name, pixel_type));
std::string DirectoryFromFormat(const std::string& format,
const std::string& port_name,
PixelType pixel_type) const {
return writer_.DirectoryFromFormat(format, port_name, pixel_type);
}

static bool IsDirectoryValid(const std::string& file_path) {
Expand Down Expand Up @@ -146,12 +145,11 @@ class ImageWriterTest : public ::testing::Test {

static void TearDownTestCase() {
for (const auto& file_name : files_) {
spruce::path file_path(file_name);
if (file_path.exists()) {
if (filesystem::exists({file_name})) {
// We'll consider a failure to delete a temporary file as a test
// failure.
unlink(file_path.getStr().c_str());
EXPECT_FALSE(file_path.exists())
unlink(file_name.c_str());
EXPECT_FALSE(filesystem::exists({file_name}))
<< "Failed to delete temporary test file: " << file_name;
}
}
Expand All @@ -165,14 +163,14 @@ class ImageWriterTest : public ::testing::Test {
// examined at tear down for deletion. When it comes to writing images, all
// names should come from here.
static std::string temp_name() {
spruce::path temp_path;
filesystem::path temp_path;
do {
temp_path.setStr(temp_dir());
temp_path = temp_dir();
temp_path.append("image_writer_test_" + std::to_string(++img_count_) +
".png");
} while (temp_path.exists());
files_.insert(temp_path.getStr());
return temp_path.getStr();
} while (filesystem::exists(temp_path));
files_.insert(temp_path.string());
return temp_path.string();
}

// Arbitrary files that are generated can be added to the set of files that
Expand All @@ -185,8 +183,8 @@ class ImageWriterTest : public ::testing::Test {
template <PixelType kPixelType>
static ::testing::AssertionResult ReadImage(const std::string& image_name,
Image<kPixelType>* image) {
spruce::path image_path(image_name);
if (image_path.exists()) {
filesystem::path image_path(image_name);
if (filesystem::exists(image_path)) {
vtkSmartPointer<vtkImageReader2> reader;
switch (kPixelType) {
case PixelType::kRgba8U:
Expand Down Expand Up @@ -257,12 +255,12 @@ class ImageWriterTest : public ::testing::Test {
const double period = 1 / 10.0; // 10 Hz.
const double start_time = 0.25;
const std::string port_name = "port";
spruce::path path(temp_dir());
filesystem::path path(temp_dir());
path.append("{image_type}_{time_usec}");

Image<kPixelType> image = test_image<kPixelType>();
const auto& port = writer.DeclareImageInputPort<kPixelType>(
port_name, path.getStr(), period, start_time);
port_name, path.string(), period, start_time);
auto events = writer.AllocateCompositeEventCollection();
auto context = writer.AllocateContext();
context->FixInputPort(port.get_index(),
Expand All @@ -273,12 +271,12 @@ class ImageWriterTest : public ::testing::Test {
const std::string expected_name = tester.MakeFileName(
tester.port_format(port.get_index()), kPixelType, context->get_time(),
port_name, tester.port_count(port.get_index()));
spruce::path expected_file(expected_name);
EXPECT_FALSE(expected_file.exists());
filesystem::path expected_file(expected_name);
EXPECT_FALSE(filesystem::exists(expected_file));
writer.Publish(*context, events->get_publish_events());
EXPECT_TRUE(expected_file.exists());
EXPECT_TRUE(filesystem::exists(expected_file));
EXPECT_EQ(1, tester.port_count(port.get_index()));
add_file_for_cleanup(expected_file.getStr());
add_file_for_cleanup(expected_file.string());

EXPECT_TRUE(MatchesFileOnDisk(expected_name, image));
}
Expand Down Expand Up @@ -322,28 +320,26 @@ TEST_F(ImageWriterTest, DirectoryFromFormat) {
ImageWriterTester tester{writer};

EXPECT_EQ("",
tester.DirectoryFromFormat("/root", "port_name", PixelType::kRgba8U)
.getStr());
EXPECT_EQ(
"/root",
tester.DirectoryFromFormat("/root/", "port_name", PixelType::kRgba8U)
.getStr());
tester.DirectoryFromFormat("/root", "port_name",
PixelType::kRgba8U));
DRAKE_EXPECT_THROWS_MESSAGE(
tester.DirectoryFromFormat("/root/", "port_name", PixelType::kRgba8U),
std::logic_error,
".*cannot end with a '/'");
EXPECT_EQ(
"/root",
tester.DirectoryFromFormat("/root/file", "port_name", PixelType::kRgba8U)
.getStr());
tester.DirectoryFromFormat("/root/file", "port_name",
PixelType::kRgba8U));
// Don't use all three image types; the FileNameFormatting test already
// tests those permutations. We just want to make sure it's engaged here.
EXPECT_EQ("/root/color",
tester
.DirectoryFromFormat("/root/{image_type}/file", "port_name",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));
EXPECT_EQ("/root/my_port",
tester
.DirectoryFromFormat("/root/{port_name}/file", "my_port",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));

// Test against invalid formatting arguments.
DRAKE_EXPECT_THROWS_MESSAGE(
Expand Down Expand Up @@ -371,23 +367,19 @@ TEST_F(ImageWriterTest, DirectoryFromFormat) {
EXPECT_EQ("/root/time_double",
tester
.DirectoryFromFormat("/root/time_double/file", "my_port",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));
EXPECT_EQ("/root/time_usec",
tester
.DirectoryFromFormat("/root/time_usec/file", "my_port",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));
EXPECT_EQ("/root/time_msec",
tester
.DirectoryFromFormat("/root/time_msec/file", "my_port",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));
EXPECT_EQ("/root/count",
tester
.DirectoryFromFormat("/root/count/file", "my_port",
PixelType::kRgba8U)
.getStr());
PixelType::kRgba8U));
}

// Tests the logic for formatting images.
Expand Down Expand Up @@ -487,10 +479,10 @@ TEST_F(ImageWriterTest, ConfigureInputPortErrors) {

// Now test a port with the same name -- can only happen if one port has
// been successfully declared.
spruce::path path(temp_dir());
filesystem::path path(temp_dir());
path.append("file_{count:3}");
const auto& port = writer.DeclareImageInputPort<PixelType::kRgba8U>(
"port", path.getStr(), 0.1, 0);
"port", path.string(), 0.1, 0);
EXPECT_EQ(0, port.get_index());
{
auto events = writer.AllocateCompositeEventCollection();
Expand All @@ -499,10 +491,10 @@ TEST_F(ImageWriterTest, ConfigureInputPortErrors) {
EXPECT_TRUE(events->HasEvents());
}

spruce::path path2(temp_dir());
filesystem::path path2(temp_dir());
path2.append("alt_file_{count:3}");
DRAKE_EXPECT_THROWS_MESSAGE(writer.DeclareImageInputPort<PixelType::kRgba8U>(
"port", path2.getStr(), 0.1, 0),
"port", path2.string(), 0.1, 0),
std::logic_error,
"System .* already has an input port named .*");
}
Expand All @@ -515,12 +507,12 @@ void TestPixelExtension(const std::string& folder, ImageWriter* writer,

ImageWriterTester tester(*writer);

spruce::path format(folder);
filesystem::path format(folder);
format.append("file");
const auto& port = writer->DeclareImageInputPort<kPixelType>(
"port" + to_string(++(*count)), format.getStr(), 1, 1);
"port" + to_string(++(*count)), format.string(), 1, 1);
const std::string& final_format = tester.port_format(port.get_index());
EXPECT_NE(format.getStr(), final_format);
EXPECT_NE(format.string(), final_format);
const std::string& ext = tester.extension(kPixelType);
EXPECT_EQ(ext, final_format.substr(final_format.size() - ext.size()));
}
Expand All @@ -542,23 +534,23 @@ TEST_F(ImageWriterTest, FileExtension) {
ImageWriterTester tester(writer);
// Case: Format string with correct extension remains unchanged.
{
spruce::path format(temp_dir());
filesystem::path format(temp_dir());
format.append("file.png");
const auto& port = writer.DeclareImageInputPort<PixelType::kRgba8U>(
"port" + to_string(++count), format.getStr(), 1, 1);
"port" + to_string(++count), format.string(), 1, 1);
const std::string& final_format = tester.port_format(port.get_index());
EXPECT_EQ(format.getStr(), final_format);
EXPECT_EQ(format.string(), final_format);
}

// Case: wrong extension gets correct extension appended.
{
spruce::path format(temp_dir());
filesystem::path format(temp_dir());
format.append("file.txt");
const auto& port = writer.DeclareImageInputPort<PixelType::kRgba8U>(
"port" + to_string(++count), format.getStr(), 1, 1);
"port" + to_string(++count), format.string(), 1, 1);
const std::string& final_format = tester.port_format(port.get_index());
const std::string& ext = tester.extension(PixelType::kRgba8U);
EXPECT_EQ(format.getStr() + ext, final_format);
EXPECT_EQ(format.string() + ext, final_format);
}
}

Expand All @@ -580,11 +572,11 @@ TEST_F(ImageWriterTest, SingleConfiguredPort) {
const double start_time = 0.25;
const std::string port_name{"single_color_port"};
const PixelType pixel_type = PixelType::kRgba8U;
spruce::path path(temp_dir());
filesystem::path path(temp_dir());
path.append("single_port_{time_usec}");

const auto& port = writer.DeclareImageInputPort<PixelType::kRgba8U>(
port_name, path.getStr(), period, start_time);
port_name, path.string(), period, start_time);

// Count gets properly initialized to zero (no images written from this port).
EXPECT_EQ(0, tester.port_count(port.get_index()));
Expand Down Expand Up @@ -628,12 +620,12 @@ TEST_F(ImageWriterTest, SingleConfiguredPort) {
const std::string expected_name = tester.MakeFileName(
tester.port_format(port.get_index()), pixel_type, context->get_time(),
port_name, tester.port_count(port.get_index()));
spruce::path expected_file(expected_name);
EXPECT_FALSE(expected_file.exists());
filesystem::path expected_file(expected_name);
EXPECT_FALSE(filesystem::exists(expected_file));
writer.Publish(*context, events->get_publish_events());
EXPECT_TRUE(expected_file.exists());
EXPECT_TRUE(filesystem::exists(expected_file));
EXPECT_EQ(1, tester.port_count(port.get_index()));
add_file_for_cleanup(expected_file.getStr());
add_file_for_cleanup(expected_file.string());
}

// Confirm period is correct.
Expand Down

0 comments on commit 43e9197

Please sign in to comment.