New fileutils.h method for managing resources on different platforms

Review URL: http://webrtc-codereview.appspot.com/304007

git-svn-id: http://webrtc.googlecode.com/svn/trunk@1105 4adac7df-926f-26a2-2b94-8c16560cd09d
This commit is contained in:
kjellander@webrtc.org 2011-12-05 16:31:12 +00:00
parent 418bce5ccc
commit 4ed4f24074
3 changed files with 220 additions and 56 deletions

View File

@ -25,6 +25,8 @@
#include <cstdio>
#include "typedefs.h" // For architecture defines
namespace webrtc {
namespace test {
@ -36,18 +38,17 @@ static const char* kPathDelimiter = "/";
// The file we're looking for to identify the project root dir.
static const char* kProjectRootFileName = "DEPS";
static const char* kOutputDirName = "out";
static const char* kOutputFallbackPath = "./";
static const char* kFallbackPath = "./";
static const char* kResourcesDirName = "resources";
const char* kCannotFindProjectRootDir = "ERROR_CANNOT_FIND_PROJECT_ROOT_DIR";
std::string ProjectRootPath() {
char path_buffer[FILENAME_MAX];
if (!GET_CURRENT_DIR(path_buffer, sizeof(path_buffer))) {
fprintf(stderr, "Cannot get current directory!\n");
std::string working_dir = WorkingDir();
if (working_dir == kFallbackPath) {
return kCannotFindProjectRootDir;
}
// Check for our file that verifies the root dir.
std::string current_path(path_buffer);
std::string current_path(working_dir);
FILE* file = NULL;
int path_delimiter_index = current_path.find_last_of(kPathDelimiter);
while (path_delimiter_index > -1) {
@ -58,12 +59,10 @@ std::string ProjectRootPath() {
fclose(file);
return current_path + kPathDelimiter;
}
// Move up one directory in the directory tree.
current_path = current_path.substr(0, path_delimiter_index);
path_delimiter_index = current_path.find_last_of(kPathDelimiter);
}
// Reached the root directory.
fprintf(stderr, "Cannot find project root directory!\n");
return kCannotFindProjectRootDir;
@ -72,25 +71,85 @@ std::string ProjectRootPath() {
std::string OutputPath() {
std::string path = ProjectRootPath();
if (path == kCannotFindProjectRootDir) {
return kOutputFallbackPath;
return kFallbackPath;
}
path += kOutputDirName;
struct stat path_info = {0};
// Check if the path exists already:
if (stat(path.c_str(), &path_info) == 0) {
if (!S_ISDIR(path_info.st_mode)) {
fprintf(stderr, "Path %s exists but is not a directory! Remove this file "
"and re-run to create the output folder.\n", path.c_str());
return kOutputFallbackPath;
}
} else {
#ifdef WIN32
_mkdir(path.c_str());
#else
mkdir(path.c_str(), S_IRWXU | S_IRWXG | S_IRWXO);
#endif
if (!CreateDirectory(path)) {
return kFallbackPath;
}
return path + kPathDelimiter;
}
std::string WorkingDir() {
char path_buffer[FILENAME_MAX];
if (!GET_CURRENT_DIR(path_buffer, sizeof(path_buffer))) {
fprintf(stderr, "Cannot get current directory!\n");
return kFallbackPath;
}
else {
return std::string(path_buffer);
}
}
bool CreateDirectory(std::string directory_name) {
struct stat path_info = {0};
// Check if the path exists already:
if (stat(directory_name.c_str(), &path_info) == 0) {
if (!S_ISDIR(path_info.st_mode)) {
fprintf(stderr, "Path %s exists but is not a directory! Remove this "
"file and re-run to create the directory.\n",
directory_name.c_str());
return false;
}
} else {
#ifdef WIN32
return _mkdir(directory_name.c_str()) == 0;
#else
return mkdir(directory_name.c_str(), S_IRWXU | S_IRWXG | S_IRWXO) == 0;
#endif
}
return true;
}
bool FileExists(std::string file_name) {
struct stat file_info = {0};
return stat(file_name.c_str(), &file_info) == 0;
}
std::string ResourcePath(std::string name, std::string extension) {
std::string platform = "win";
#ifdef WEBRTC_LINUX
platform = "linux";
#endif // WEBRTC_LINUX
#ifdef WEBRTC_MAC
platform = "mac";
#endif // WEBRTC_MAC
#ifdef WEBRTC_ARCH_64_BITS
std::string architecture = "64";
#else
std::string architecture = "32";
#endif // WEBRTC_ARCH_64_BITS
std::string resources_path = ProjectRootPath() + kResourcesDirName + kPathDelimiter;
std::string resource_file = resources_path + name + "_" + platform + "_" +
architecture + "." + extension;
if (!FileExists(resource_file)) {
return resource_file;
}
// Try without architecture.
resource_file = resources_path + name + "_" + platform + "." + extension;
if (FileExists(resource_file)) {
return resource_file;
}
// Try without platform.
resource_file = resources_path + name + "_" + architecture + "." + extension;
if (FileExists(resource_file)) {
return resource_file;
}
// Fall back on name without architecture or platform.
return resources_path + name + "." + extension;
}
} // namespace test
} // namespace webrtc

View File

@ -9,6 +9,7 @@
*/
// File utilities for testing purposes.
//
// The ProjectRootPath() method is a convenient way of getting an absolute
// path to the project source tree root directory. Using this, it is easy to
// refer to test resource files in a portable way.
@ -99,6 +100,37 @@ std::string ProjectRootPath();
// found, the current working directory ("./") is returned as a fallback.
std::string OutputPath();
// Returns a path to a resource file for the currently executing platform.
// Adapts to what filenames are currently present in the
// [project-root]/resources/ dir.
// Returns an absolute path according to this priority list (the directory
// part of the path is left out for readability):
// 1. [name]_[platform]_[architecture].[extension]
// 2. [name]_[platform].[extension]
// 3. [name]_[architecture].[extension]
// 4. [name].[extension]
// Where
// * platform is either of "win", "mac" or "linux".
// * architecture is either of "32" or "64".
//
// Arguments:
// name - Name of the resource file. If a plain filename (no directory path)
// is supplied, the file is assumed to be located in resources/
// If a directory path is prepended to the filename, a subdirectory
// hierarchy reflecting that path is assumed to be present.
// extension - File extension, without the dot, i.e. "bmp" or "yuv".
std::string ResourcePath(std::string name, std::string extension);
// Gets the current working directory for the executing program.
// Returns "./" if for some reason it is not possible to find the working
// directory.
std::string WorkingDir();
// Creates a directory if it not already exists.
// Returns true if successful. Will print an error message to stderr and return
// false if a file with the same name already exists.
bool CreateDirectory(std::string directory_name);
} // namespace test
} // namespace webrtc

View File

@ -9,52 +9,99 @@
*/
#include <cstdio>
#include "fileutils.h"
#include <list>
#include <string>
#include "gtest/gtest.h"
#include "testsupport/fileutils.h"
#ifdef WIN32
#include <direct.h>
#define GET_CURRENT_DIR _getcwd
static const char* kPathDelimiter = "\\";
#else
#include <unistd.h>
#define GET_CURRENT_DIR getcwd
static const char* kPathDelimiter = "/";
#endif
static const std::string kDummyDir = "file_utils_unittest_dummy_dir";
static const std::string kResourcesDir = "resources";
static const std::string kTestName = "fileutils_unittest";
static const std::string kExtension = "tmp";
typedef std::list<std::string> FileList;
namespace webrtc {
namespace test {
// Test fixture to restore the working directory between each test, since some
// of them change it with chdir during execution (not restored by the
// gtest framework).
class FileUtilsTest: public testing::Test {
class FileUtilsTest : public testing::Test {
protected:
FileUtilsTest() {
original_working_dir_ = GetWorkingDir();
}
virtual ~FileUtilsTest() {}
// Runs before the first test
static void SetUpTestCase() {
original_working_dir_ = webrtc::test::WorkingDir();
std::string resources_path = original_working_dir_ + kPathDelimiter +
kResourcesDir + kPathDelimiter;
webrtc::test::CreateDirectory(resources_path);
files_.push_back(resources_path + kTestName + "." + kExtension);
files_.push_back(resources_path + kTestName + "_32." + kExtension);
files_.push_back(resources_path + kTestName + "_64." + kExtension);
files_.push_back(resources_path + kTestName + "_linux." + kExtension);
files_.push_back(resources_path + kTestName + "_mac." + kExtension);
files_.push_back(resources_path + kTestName + "_win." + kExtension);
files_.push_back(resources_path + kTestName + "_linux_32." + kExtension);
files_.push_back(resources_path + kTestName + "_mac_32." + kExtension);
files_.push_back(resources_path + kTestName + "_win_32." + kExtension);
files_.push_back(resources_path + kTestName + "_linux_64." + kExtension);
files_.push_back(resources_path + kTestName + "_mac_64." + kExtension);
files_.push_back(resources_path + kTestName + "_win_64." + kExtension);
// Now that the resources dir exists, write some empty test files into it.
for (FileList::iterator file_it = files_.begin();
file_it != files_.end(); ++file_it) {
FILE* file = fopen(file_it->c_str(), "wb");
ASSERT_TRUE(file != NULL) << "Failed to write file: " << file_it->c_str();
ASSERT_TRUE(fprintf(file, "%s", "Dummy data") > 0);
fclose(file);
}
// Create a dummy subdir that can be chdir'ed into for testing purposes.
empty_dummy_dir_ = original_working_dir_ + kPathDelimiter + kDummyDir;
webrtc::test::CreateDirectory(empty_dummy_dir_);
}
static void TearDownTestCase() {
// Clean up all resource files written
for (FileList::iterator file_it = files_.begin();
file_it != files_.end(); ++file_it) {
remove(file_it->c_str());
}
std::remove(empty_dummy_dir_.c_str());
}
void SetUp() {
chdir(original_working_dir_.c_str());
ASSERT_EQ(chdir(original_working_dir_.c_str()), 0);
}
void TearDown() {}
void TearDown() {
ASSERT_EQ(chdir(original_working_dir_.c_str()), 0);
}
protected:
static std::string empty_dummy_dir_;
private:
std::string original_working_dir_;
static std::string GetWorkingDir() {
char path_buffer[FILENAME_MAX];
EXPECT_TRUE(GET_CURRENT_DIR(path_buffer, sizeof(path_buffer)))
<< "Cannot get current working directory!";
return std::string(path_buffer);
}
static FileList files_;
static std::string original_working_dir_;
};
FileList FileUtilsTest::files_;
std::string FileUtilsTest::original_working_dir_ = "";
std::string FileUtilsTest::empty_dummy_dir_ = "";
// Tests that the project root path is returned for the default working
// directory that is automatically set when the test executable is launched.
// The test is not fully testing the implementation, since we cannot be sure
// of where the executable was launched from.
// The test will fail if the top level directory is not named "trunk".
TEST_F(FileUtilsTest, ProjectRootPathFromUnchangedWorkingDir) {
std::string path = ProjectRootPath();
std::string path = webrtc::test::ProjectRootPath();
std::string expected_end = "trunk";
expected_end = kPathDelimiter + expected_end + kPathDelimiter;
ASSERT_EQ(path.length() - expected_end.length(), path.find(expected_end));
@ -62,7 +109,7 @@ TEST_F(FileUtilsTest, ProjectRootPathFromUnchangedWorkingDir) {
// Similar to the above test, but for the output dir
TEST_F(FileUtilsTest, OutputPathFromUnchangedWorkingDir) {
std::string path = OutputPath();
std::string path = webrtc::test::OutputPath();
std::string expected_end = "out";
expected_end = kPathDelimiter + expected_end + kPathDelimiter;
ASSERT_EQ(path.length() - expected_end.length(), path.find(expected_end));
@ -72,21 +119,19 @@ TEST_F(FileUtilsTest, OutputPathFromUnchangedWorkingDir) {
// deeper from the current one. Then testing that the project path returned
// is still the same, when the function under test is called again.
TEST_F(FileUtilsTest, ProjectRootPathFromDeeperWorkingDir) {
std::string path = ProjectRootPath();
std::string path = webrtc::test::ProjectRootPath();
std::string original_working_dir = path; // This is the correct project root
// Change to a subdirectory path (the full path doesn't have to exist).
path += "foo/bar/baz";
chdir(path.c_str());
ASSERT_EQ(original_working_dir, ProjectRootPath());
// Change to a subdirectory path.
ASSERT_EQ(0, chdir(empty_dummy_dir_.c_str()));
ASSERT_EQ(original_working_dir, webrtc::test::ProjectRootPath());
}
// Similar to the above test, but for the output dir
TEST_F(FileUtilsTest, OutputPathFromDeeperWorkingDir) {
std::string path = OutputPath();
std::string path = webrtc::test::OutputPath();
std::string original_working_dir = path;
path += "foo/bar/baz";
chdir(path.c_str());
ASSERT_EQ(original_working_dir, OutputPath());
ASSERT_EQ(0, chdir(empty_dummy_dir_.c_str()));
ASSERT_EQ(original_working_dir, webrtc::test::OutputPath());
}
// Tests with current working directory set to a directory higher up in the
@ -95,15 +140,43 @@ TEST_F(FileUtilsTest, OutputPathFromDeeperWorkingDir) {
TEST_F(FileUtilsTest, ProjectRootPathFromRootWorkingDir) {
// Change current working dir to the root of the current file system
// (this will always be "above" our project root dir).
chdir(kPathDelimiter);
ASSERT_EQ(kCannotFindProjectRootDir, ProjectRootPath());
ASSERT_EQ(0, chdir(kPathDelimiter));
ASSERT_EQ(webrtc::test::kCannotFindProjectRootDir,
webrtc::test::ProjectRootPath());
}
// Similar to the above test, but for the output dir
TEST_F(FileUtilsTest, OutputPathFromRootWorkingDir) {
chdir(kPathDelimiter);
ASSERT_EQ("./", OutputPath());
ASSERT_EQ(0, chdir(kPathDelimiter));
ASSERT_EQ("./", webrtc::test::OutputPath());
}
// Only tests that the code executes
TEST_F(FileUtilsTest, CreateDirectory) {
std::string directory = "fileutils-unittest-empty-dir";
// Make sure it's removed if a previous test has failed:
std::remove(directory.c_str());
ASSERT_TRUE(webrtc::test::CreateDirectory(directory));
std::remove(directory.c_str());
}
TEST_F(FileUtilsTest, WorkingDirReturnsValue) {
// Hard to cover all platforms. Just test that it returns something without
// crashing:
std::string working_dir = webrtc::test::WorkingDir();
ASSERT_TRUE(working_dir.length() > 0);
}
// Due to multiple platforms, it is hard to make a complete test for
// ResourcePath. Manual testing has been performed by removing files and
// verified the result confirms with the specified documentation for the
// function.
TEST_F(FileUtilsTest, ResourcePathReturnsValue) {
std::string resource = webrtc::test::ResourcePath(kTestName, kExtension);
ASSERT_TRUE(resource.find(kTestName) > 0);
ASSERT_TRUE(resource.find(kExtension) > 0);
ASSERT_EQ(0, chdir(kPathDelimiter));
ASSERT_EQ("./", webrtc::test::OutputPath());
}
} // namespace test
} // namespace webrtc