This commit is contained in:
parent
98a74e2351
commit
bcaeb1ac8c
|
@ -91,7 +91,8 @@ function(build_libhdfs_test NAME LIBRARY)
|
||||||
list(APPEND FILES ${CMAKE_SOURCE_DIR}/main/native/libhdfs-tests/${FIL})
|
list(APPEND FILES ${CMAKE_SOURCE_DIR}/main/native/libhdfs-tests/${FIL})
|
||||||
endif()
|
endif()
|
||||||
endforeach()
|
endforeach()
|
||||||
add_executable("${NAME}_${LIBRARY}" ${FILES})
|
add_executable("${NAME}_${LIBRARY}" $<TARGET_OBJECTS:x_platform_obj_c_api> $<TARGET_OBJECTS:x_platform_obj> ${FILES})
|
||||||
|
target_include_directories("${NAME}_${LIBRARY}" PRIVATE main/native/libhdfspp/lib)
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
function(add_libhdfs_test NAME LIBRARY)
|
function(add_libhdfs_test NAME LIBRARY)
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include "hdfspp/hdfs_ext.h"
|
#include "hdfspp/hdfs_ext.h"
|
||||||
#include "native_mini_dfs.h"
|
#include "native_mini_dfs.h"
|
||||||
#include "os/thread.h"
|
#include "os/thread.h"
|
||||||
|
#include "x-platform/c-api/syscall.h"
|
||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <inttypes.h>
|
#include <inttypes.h>
|
||||||
|
@ -126,7 +127,8 @@ static int hdfsCurlData(const char *host, const tPort port, const char *dirNm,
|
||||||
EXPECT_NONNULL(pw = getpwuid(uid));
|
EXPECT_NONNULL(pw = getpwuid(uid));
|
||||||
|
|
||||||
int fd = -1;
|
int fd = -1;
|
||||||
EXPECT_NONNEGATIVE(fd = mkstemp(tmpFile));
|
EXPECT_NONNEGATIVE(fd = x_platform_syscall_create_and_open_temp_file(
|
||||||
|
tmpFile, sizeof tmpFile));
|
||||||
|
|
||||||
tSize sz = 0;
|
tSize sz = 0;
|
||||||
while (sz < fileSz) {
|
while (sz < fileSz) {
|
||||||
|
|
|
@ -18,6 +18,26 @@
|
||||||
|
|
||||||
#include "x-platform/syscall.h"
|
#include "x-platform/syscall.h"
|
||||||
|
|
||||||
extern "C" int x_platform_syscall_write_to_stdout(const char* msg) {
|
#include <algorithm>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
int x_platform_syscall_write_to_stdout(const char* msg) {
|
||||||
return XPlatform::Syscall::WriteToStdout(msg) ? 1 : 0;
|
return XPlatform::Syscall::WriteToStdout(msg) ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int x_platform_syscall_create_and_open_temp_file(char* pattern,
|
||||||
|
const size_t pattern_len) {
|
||||||
|
std::vector<char> pattern_vec(pattern, pattern + pattern_len);
|
||||||
|
|
||||||
|
const auto fd = XPlatform::Syscall::CreateAndOpenTempFile(pattern_vec);
|
||||||
|
if (fd != -1) {
|
||||||
|
std::copy_n(pattern_vec.begin(), pattern_len, pattern);
|
||||||
|
}
|
||||||
|
return fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
int x_platform_syscall_close_file(const int fd) {
|
||||||
|
return XPlatform::Syscall::CloseFile(fd);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -24,5 +24,8 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int x_platform_syscall_write_to_stdout(const char* msg);
|
int x_platform_syscall_write_to_stdout(const char* msg);
|
||||||
|
int x_platform_syscall_create_and_open_temp_file(char* pattern,
|
||||||
|
size_t pattern_len);
|
||||||
|
int x_platform_syscall_close_file(int fd);
|
||||||
|
|
||||||
#endif // NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_SYSCALL_H
|
#endif // NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_SYSCALL_H
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_SYSCALL
|
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_SYSCALL
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The {@link XPlatform} namespace contains components that
|
* The {@link XPlatform} namespace contains components that
|
||||||
|
@ -84,6 +85,34 @@ class Syscall {
|
||||||
static bool StringCompareIgnoreCase(const std::string& a,
|
static bool StringCompareIgnoreCase(const std::string& a,
|
||||||
const std::string& b);
|
const std::string& b);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates and opens a temporary file with a given {@link pattern}.
|
||||||
|
* The {@link pattern} must end with a minimum of 6 'X' characters.
|
||||||
|
* This function will first modify the last 6 'X' characters with
|
||||||
|
* random character values, which serve as the temporary file name.
|
||||||
|
* Subsequently opens the file and returns the file descriptor for
|
||||||
|
* the same. The behaviour of this function is the same as that of
|
||||||
|
* POSIX mkstemp function. The file must be later closed by the
|
||||||
|
* application and is not handled by this function.
|
||||||
|
*
|
||||||
|
* @param pattern the pattern to be used for the temporary filename.
|
||||||
|
* @returns an integer representing the file descriptor for the
|
||||||
|
* opened temporary file. Returns -1 in the case of error and sets
|
||||||
|
* the global errno with the appropriate error code.
|
||||||
|
*/
|
||||||
|
static int CreateAndOpenTempFile(std::vector<char>& pattern);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the file corresponding to given {@link file_descriptor}.
|
||||||
|
*
|
||||||
|
* @param file_descriptor the file descriptor of the file to close.
|
||||||
|
* @returns a boolean indicating the status of the call to this
|
||||||
|
* function. true if it's a success, false in the case of an error.
|
||||||
|
* The global errno is set if the call to this function was not
|
||||||
|
* successful.
|
||||||
|
*/
|
||||||
|
static bool CloseFile(int file_descriptor);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static bool WriteToStdoutImpl(const char* message);
|
static bool WriteToStdoutImpl(const char* message);
|
||||||
};
|
};
|
||||||
|
|
|
@ -21,6 +21,7 @@
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "syscall.h"
|
#include "syscall.h"
|
||||||
|
|
||||||
|
@ -59,3 +60,13 @@ bool XPlatform::Syscall::StringCompareIgnoreCase(const std::string& a,
|
||||||
const std::string& b) {
|
const std::string& b) {
|
||||||
return strcasecmp(a.c_str(), b.c_str()) == 0;
|
return strcasecmp(a.c_str(), b.c_str()) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int XPlatform::Syscall::CreateAndOpenTempFile(std::vector<char>& pattern) {
|
||||||
|
// Make space for mkstemp to add NULL character at the end
|
||||||
|
pattern.resize(pattern.size() + 1);
|
||||||
|
return mkstemp(pattern.data());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool XPlatform::Syscall::CloseFile(const int file_descriptor) {
|
||||||
|
return close(file_descriptor) == 0;
|
||||||
|
}
|
||||||
|
|
|
@ -19,7 +19,14 @@
|
||||||
#include <Shlwapi.h>
|
#include <Shlwapi.h>
|
||||||
#include <WinBase.h>
|
#include <WinBase.h>
|
||||||
#include <Windows.h>
|
#include <Windows.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <io.h>
|
||||||
|
#include <share.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cstdlib>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
#include "syscall.h"
|
#include "syscall.h"
|
||||||
|
@ -64,3 +71,26 @@ bool XPlatform::Syscall::StringCompareIgnoreCase(const std::string& a,
|
||||||
const std::string& b) {
|
const std::string& b) {
|
||||||
return _stricmp(a.c_str(), b.c_str()) == 0;
|
return _stricmp(a.c_str(), b.c_str()) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int XPlatform::Syscall::CreateAndOpenTempFile(std::vector<char>& pattern) {
|
||||||
|
if (_set_errno(0) != 0) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make space for _mktemp_s to add NULL character at the end
|
||||||
|
pattern.resize(pattern.size() + 1);
|
||||||
|
if (_mktemp_s(pattern.data(), pattern.size()) != 0) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto fd{-1};
|
||||||
|
if (_sopen_s(&fd, pattern.data(), _O_RDWR | _O_CREAT | _O_EXCL, _SH_DENYNO,
|
||||||
|
_S_IREAD | _S_IWRITE) != 0) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool XPlatform::Syscall::CloseFile(const int file_descriptor) {
|
||||||
|
return _close(file_descriptor) == 0;
|
||||||
|
}
|
||||||
|
|
|
@ -96,11 +96,13 @@ add_executable(node_exclusion_test node_exclusion_test.cc)
|
||||||
target_link_libraries(node_exclusion_test fs gmock_main common ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(node_exclusion_test fs gmock_main common ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(node_exclusion node_exclusion_test)
|
add_memcheck_test(node_exclusion node_exclusion_test)
|
||||||
|
|
||||||
add_executable(configuration_test configuration_test.cc)
|
add_executable(configuration_test $<TARGET_OBJECTS:x_platform_obj> configuration_test.cc)
|
||||||
|
target_include_directories(configuration_test PRIVATE ../lib)
|
||||||
target_link_libraries(configuration_test common gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(configuration_test common gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(configuration configuration_test)
|
add_memcheck_test(configuration configuration_test)
|
||||||
|
|
||||||
add_executable(hdfs_configuration_test hdfs_configuration_test.cc)
|
add_executable(hdfs_configuration_test $<TARGET_OBJECTS:x_platform_obj> hdfs_configuration_test.cc)
|
||||||
|
target_include_directories(hdfs_configuration_test PRIVATE ../lib)
|
||||||
target_link_libraries(hdfs_configuration_test common gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(hdfs_configuration_test common gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(hdfs_configuration hdfs_configuration_test)
|
add_memcheck_test(hdfs_configuration hdfs_configuration_test)
|
||||||
|
|
||||||
|
@ -108,11 +110,13 @@ add_executable(hdfspp_errors_test hdfspp_errors.cc)
|
||||||
target_link_libraries(hdfspp_errors_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(hdfspp_errors_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(hdfspp_errors hdfspp_errors_test)
|
add_memcheck_test(hdfspp_errors hdfspp_errors_test)
|
||||||
|
|
||||||
add_executable(hdfs_builder_test hdfs_builder_test.cc)
|
add_executable(hdfs_builder_test $<TARGET_OBJECTS:x_platform_obj> hdfs_builder_test.cc)
|
||||||
|
target_include_directories(hdfs_builder_test PRIVATE ../lib)
|
||||||
target_link_libraries(hdfs_builder_test test_common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(hdfs_builder_test test_common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(hdfs_builder_test hdfs_builder_test)
|
add_memcheck_test(hdfs_builder_test hdfs_builder_test)
|
||||||
|
|
||||||
add_executable(logging_test logging_test.cc $<TARGET_OBJECTS:x_platform_obj>)
|
add_executable(logging_test logging_test.cc $<TARGET_OBJECTS:x_platform_obj>)
|
||||||
|
target_include_directories(logging_test PRIVATE ../lib)
|
||||||
target_link_libraries(logging_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(logging_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} gmock_main ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(logging_test logging_test)
|
add_memcheck_test(logging_test logging_test)
|
||||||
|
|
||||||
|
@ -124,7 +128,8 @@ add_executable(user_lock_test user_lock_test.cc)
|
||||||
target_link_libraries(user_lock_test fs gmock_main common ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(user_lock_test fs gmock_main common ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(user_lock user_lock_test)
|
add_memcheck_test(user_lock user_lock_test)
|
||||||
|
|
||||||
add_executable(hdfs_config_connect_bugs_test hdfs_config_connect_bugs.cc)
|
add_executable(hdfs_config_connect_bugs_test $<TARGET_OBJECTS:x_platform_obj> hdfs_config_connect_bugs.cc)
|
||||||
|
target_include_directories(hdfs_config_connect_bugs_test PRIVATE ../lib)
|
||||||
target_link_libraries(hdfs_config_connect_bugs_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
target_link_libraries(hdfs_config_connect_bugs_test common gmock_main bindings_c fs rpc proto common reader connection ${PROTOBUF_LIBRARIES} ${OPENSSL_LIBRARIES} ${SASL_LIBRARIES} ${CMAKE_THREAD_LIBS_INIT})
|
||||||
add_memcheck_test(hdfs_config_connect_bugs hdfs_config_connect_bugs_test)
|
add_memcheck_test(hdfs_config_connect_bugs hdfs_config_connect_bugs_test)
|
||||||
|
|
||||||
|
|
|
@ -299,11 +299,12 @@ TEST(ConfigurationTest, TestFileReads)
|
||||||
// Single stream
|
// Single stream
|
||||||
{
|
{
|
||||||
TempFile tempFile;
|
TempFile tempFile;
|
||||||
writeSimpleConfig(tempFile.filename, "key1", "value1");
|
writeSimpleConfig(tempFile.GetFileName(), "key1", "value1");
|
||||||
|
|
||||||
ConfigurationLoader config_loader;
|
ConfigurationLoader config_loader;
|
||||||
config_loader.ClearSearchPath();
|
config_loader.ClearSearchPath();
|
||||||
optional<Configuration> config = config_loader.LoadFromFile<Configuration>(tempFile.filename);
|
optional<Configuration> config =
|
||||||
|
config_loader.LoadFromFile<Configuration>(tempFile.GetFileName());
|
||||||
EXPECT_TRUE(config && "Parse first stream");
|
EXPECT_TRUE(config && "Parse first stream");
|
||||||
EXPECT_EQ("value1", config->GetWithDefault("key1", ""));
|
EXPECT_EQ("value1", config->GetWithDefault("key1", ""));
|
||||||
}
|
}
|
||||||
|
@ -311,16 +312,18 @@ TEST(ConfigurationTest, TestFileReads)
|
||||||
// Multiple files
|
// Multiple files
|
||||||
{
|
{
|
||||||
TempFile tempFile;
|
TempFile tempFile;
|
||||||
writeSimpleConfig(tempFile.filename, "key1", "value1");
|
writeSimpleConfig(tempFile.GetFileName(), "key1", "value1");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
optional<Configuration> config = loader.LoadFromFile<Configuration>(tempFile.filename);
|
optional<Configuration> config =
|
||||||
|
loader.LoadFromFile<Configuration>(tempFile.GetFileName());
|
||||||
ASSERT_TRUE(config && "Parse first stream");
|
ASSERT_TRUE(config && "Parse first stream");
|
||||||
EXPECT_EQ("value1", config->GetWithDefault("key1", ""));
|
EXPECT_EQ("value1", config->GetWithDefault("key1", ""));
|
||||||
|
|
||||||
TempFile tempFile2;
|
TempFile tempFile2;
|
||||||
writeSimpleConfig(tempFile2.filename, "key2", "value2");
|
writeSimpleConfig(tempFile2.GetFileName(), "key2", "value2");
|
||||||
optional<Configuration> config2 = loader.OverlayResourceFile(*config, tempFile2.filename);
|
optional<Configuration> config2 =
|
||||||
|
loader.OverlayResourceFile(*config, tempFile2.GetFileName());
|
||||||
ASSERT_TRUE(config2 && "Parse second stream");
|
ASSERT_TRUE(config2 && "Parse second stream");
|
||||||
EXPECT_EQ("value1", config2->GetWithDefault("key1", ""));
|
EXPECT_EQ("value1", config2->GetWithDefault("key1", ""));
|
||||||
EXPECT_EQ("value2", config2->GetWithDefault("key2", ""));
|
EXPECT_EQ("value2", config2->GetWithDefault("key2", ""));
|
||||||
|
@ -350,13 +353,13 @@ TEST(ConfigurationTest, TestFileReads)
|
||||||
{
|
{
|
||||||
TempDir tempDir1;
|
TempDir tempDir1;
|
||||||
TempFile tempFile1(tempDir1.path + "/file1.xml");
|
TempFile tempFile1(tempDir1.path + "/file1.xml");
|
||||||
writeSimpleConfig(tempFile1.filename, "key1", "value1");
|
writeSimpleConfig(tempFile1.GetFileName(), "key1", "value1");
|
||||||
TempDir tempDir2;
|
TempDir tempDir2;
|
||||||
TempFile tempFile2(tempDir2.path + "/file2.xml");
|
TempFile tempFile2(tempDir2.path + "/file2.xml");
|
||||||
writeSimpleConfig(tempFile2.filename, "key2", "value2");
|
writeSimpleConfig(tempFile2.GetFileName(), "key2", "value2");
|
||||||
TempDir tempDir3;
|
TempDir tempDir3;
|
||||||
TempFile tempFile3(tempDir3.path + "/file3.xml");
|
TempFile tempFile3(tempDir3.path + "/file3.xml");
|
||||||
writeSimpleConfig(tempFile3.filename, "key3", "value3");
|
writeSimpleConfig(tempFile3.GetFileName(), "key3", "value3");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
loader.SetSearchPath(tempDir1.path + ":" + tempDir2.path + ":" + tempDir3.path);
|
loader.SetSearchPath(tempDir1.path + ":" + tempDir2.path + ":" + tempDir3.path);
|
||||||
|
@ -377,7 +380,7 @@ TEST(ConfigurationTest, TestDefaultConfigs) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile coreSite(tempDir.path + "/core-site.xml");
|
TempFile coreSite(tempDir.path + "/core-site.xml");
|
||||||
writeSimpleConfig(coreSite.filename, "key1", "value1");
|
writeSimpleConfig(coreSite.GetFileName(), "key1", "value1");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
loader.SetSearchPath(tempDir.path);
|
loader.SetSearchPath(tempDir.path);
|
||||||
|
|
|
@ -21,11 +21,19 @@
|
||||||
#include "hdfspp/config_parser.h"
|
#include "hdfspp/config_parser.h"
|
||||||
#include "common/configuration.h"
|
#include "common/configuration.h"
|
||||||
#include "common/configuration_loader.h"
|
#include "common/configuration_loader.h"
|
||||||
|
#include "x-platform/syscall.h"
|
||||||
|
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <istream>
|
#include <istream>
|
||||||
|
#include <string>
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include <ftw.h>
|
#include <ftw.h>
|
||||||
|
#include <unistd.h>
|
||||||
#include <gmock/gmock.h>
|
#include <gmock/gmock.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
namespace hdfs {
|
namespace hdfs {
|
||||||
|
|
||||||
|
@ -108,22 +116,50 @@ void writeDamagedConfig(const std::string& filename, Args... args) {
|
||||||
// TempDir: is deleted on destruction
|
// TempDir: is deleted on destruction
|
||||||
class TempFile {
|
class TempFile {
|
||||||
public:
|
public:
|
||||||
std::string filename;
|
TempFile() {
|
||||||
char fn_buffer[128];
|
std::vector<char> tmp_buf(filename_.begin(), filename_.end());
|
||||||
int tempFileHandle;
|
fd_ = XPlatform::Syscall::CreateAndOpenTempFile(tmp_buf);
|
||||||
TempFile() : tempFileHandle(-1) {
|
EXPECT_NE(fd_, -1);
|
||||||
strncpy(fn_buffer, "/tmp/test_XXXXXXXXXX", sizeof(fn_buffer));
|
filename_.assign(tmp_buf.data());
|
||||||
tempFileHandle = mkstemp(fn_buffer);
|
|
||||||
EXPECT_NE(-1, tempFileHandle);
|
|
||||||
filename = fn_buffer;
|
|
||||||
}
|
}
|
||||||
TempFile(const std::string & fn) : filename(fn), tempFileHandle(-1) {
|
|
||||||
strncpy(fn_buffer, fn.c_str(), sizeof(fn_buffer));
|
|
||||||
fn_buffer[sizeof(fn_buffer)-1] = 0;
|
|
||||||
}
|
|
||||||
~TempFile() { if(-1 != tempFileHandle) close(tempFileHandle); unlink(fn_buffer); }
|
|
||||||
};
|
|
||||||
|
|
||||||
|
TempFile(std::string fn) : filename_(std::move(fn)) {}
|
||||||
|
|
||||||
|
TempFile(const TempFile& other) = default;
|
||||||
|
|
||||||
|
TempFile(TempFile&& other) noexcept
|
||||||
|
: filename_{std::move(other.filename_)}, fd_{other.fd_} {}
|
||||||
|
|
||||||
|
TempFile& operator=(const TempFile& other) {
|
||||||
|
if (&other != this) {
|
||||||
|
filename_ = other.filename_;
|
||||||
|
fd_ = other.fd_;
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
TempFile& operator=(TempFile&& other) noexcept {
|
||||||
|
if (&other != this) {
|
||||||
|
filename_ = std::move(other.filename_);
|
||||||
|
fd_ = other.fd_;
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] const std::string& GetFileName() const { return filename_; }
|
||||||
|
|
||||||
|
~TempFile() {
|
||||||
|
if (-1 != fd_) {
|
||||||
|
EXPECT_NE(XPlatform::Syscall::CloseFile(fd_), -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
unlink(filename_.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string filename_{"/tmp/test_XXXXXXXXXX"};
|
||||||
|
int fd_{-1};
|
||||||
|
};
|
||||||
|
|
||||||
// Callback to remove a directory in the nftw visitor
|
// Callback to remove a directory in the nftw visitor
|
||||||
int nftw_remove(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf)
|
int nftw_remove(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf)
|
||||||
|
|
|
@ -45,7 +45,7 @@ TEST(HdfsBuilderTest, TestRead)
|
||||||
{
|
{
|
||||||
TempDir tempDir1;
|
TempDir tempDir1;
|
||||||
TempFile tempFile1(tempDir1.path + "/core-site.xml");
|
TempFile tempFile1(tempDir1.path + "/core-site.xml");
|
||||||
writeSimpleConfig(tempFile1.filename, "key1", "value1");
|
writeSimpleConfig(tempFile1.GetFileName(), "key1", "value1");
|
||||||
|
|
||||||
hdfsBuilder * builder = hdfsNewBuilderFromDirectory(tempDir1.path.c_str());
|
hdfsBuilder * builder = hdfsNewBuilderFromDirectory(tempDir1.path.c_str());
|
||||||
|
|
||||||
|
@ -68,7 +68,7 @@ TEST(HdfsBuilderTest, TestRead)
|
||||||
{
|
{
|
||||||
TempDir tempDir1;
|
TempDir tempDir1;
|
||||||
TempFile tempFile1(tempDir1.path + "/core-site.xml");
|
TempFile tempFile1(tempDir1.path + "/core-site.xml");
|
||||||
writeSimpleConfig(tempFile1.filename, "key1", "100");
|
writeSimpleConfig(tempFile1.GetFileName(), "key1", "100");
|
||||||
|
|
||||||
hdfsBuilder * builder = hdfsNewBuilderFromDirectory(tempDir1.path.c_str());
|
hdfsBuilder * builder = hdfsNewBuilderFromDirectory(tempDir1.path.c_str());
|
||||||
|
|
||||||
|
|
|
@ -72,9 +72,9 @@ TEST(HdfsConfigurationTest, TestDefaultConfigs) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile coreSite(tempDir.path + "/core-site.xml");
|
TempFile coreSite(tempDir.path + "/core-site.xml");
|
||||||
writeSimpleConfig(coreSite.filename, "key1", "value1");
|
writeSimpleConfig(coreSite.GetFileName(), "key1", "value1");
|
||||||
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
||||||
writeSimpleConfig(hdfsSite.filename, "key2", "value2");
|
writeSimpleConfig(hdfsSite.GetFileName(), "key2", "value2");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
loader.SetSearchPath(tempDir.path);
|
loader.SetSearchPath(tempDir.path);
|
||||||
|
@ -89,7 +89,7 @@ TEST(HdfsConfigurationTest, TestDefaultConfigs) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile coreSite(tempDir.path + "/core-site.xml");
|
TempFile coreSite(tempDir.path + "/core-site.xml");
|
||||||
writeSimpleConfig(coreSite.filename, "key1", "value1");
|
writeSimpleConfig(coreSite.GetFileName(), "key1", "value1");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
loader.SetSearchPath(tempDir.path);
|
loader.SetSearchPath(tempDir.path);
|
||||||
|
@ -103,7 +103,7 @@ TEST(HdfsConfigurationTest, TestDefaultConfigs) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
||||||
writeSimpleConfig(hdfsSite.filename, "key2", "value2");
|
writeSimpleConfig(hdfsSite.GetFileName(), "key2", "value2");
|
||||||
|
|
||||||
ConfigurationLoader loader;
|
ConfigurationLoader loader;
|
||||||
loader.SetSearchPath(tempDir.path);
|
loader.SetSearchPath(tempDir.path);
|
||||||
|
@ -121,9 +121,9 @@ TEST(HdfsConfigurationTest, TestConfigParserAPI) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile coreSite(tempDir.path + "/core-site.xml");
|
TempFile coreSite(tempDir.path + "/core-site.xml");
|
||||||
writeSimpleConfig(coreSite.filename, "key1", "value1");
|
writeSimpleConfig(coreSite.GetFileName(), "key1", "value1");
|
||||||
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
||||||
writeSimpleConfig(hdfsSite.filename, "key2", "value2");
|
writeSimpleConfig(hdfsSite.GetFileName(), "key2", "value2");
|
||||||
|
|
||||||
ConfigParser parser(tempDir.path);
|
ConfigParser parser(tempDir.path);
|
||||||
|
|
||||||
|
@ -142,9 +142,9 @@ TEST(HdfsConfigurationTest, TestConfigParserAPI) {
|
||||||
{
|
{
|
||||||
TempDir tempDir;
|
TempDir tempDir;
|
||||||
TempFile coreSite(tempDir.path + "/core-site.xml");
|
TempFile coreSite(tempDir.path + "/core-site.xml");
|
||||||
writeSimpleConfig(coreSite.filename, "key1", "value1");
|
writeSimpleConfig(coreSite.GetFileName(), "key1", "value1");
|
||||||
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
TempFile hdfsSite(tempDir.path + "/hdfs-site.xml");
|
||||||
writeDamagedConfig(hdfsSite.filename, "key2", "value2");
|
writeDamagedConfig(hdfsSite.GetFileName(), "key2", "value2");
|
||||||
|
|
||||||
ConfigParser parser(tempDir.path);
|
ConfigParser parser(tempDir.path);
|
||||||
|
|
||||||
|
|
|
@ -85,3 +85,21 @@ TEST(XPlatformSyscall, StringCompareIgnoreCaseNegative) {
|
||||||
EXPECT_FALSE(XPlatform::Syscall::StringCompareIgnoreCase("abcd", "abcde"));
|
EXPECT_FALSE(XPlatform::Syscall::StringCompareIgnoreCase("abcd", "abcde"));
|
||||||
EXPECT_FALSE(XPlatform::Syscall::StringCompareIgnoreCase("12345", "abcde"));
|
EXPECT_FALSE(XPlatform::Syscall::StringCompareIgnoreCase("12345", "abcde"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(XPlatformSyscall, CreateAndOpenTempFileBasic) {
|
||||||
|
std::string pattern("tmp-XXXXXX");
|
||||||
|
std::vector<char> pattern_vec(pattern.begin(), pattern.end());
|
||||||
|
|
||||||
|
const auto fd = XPlatform::Syscall::CreateAndOpenTempFile(pattern_vec);
|
||||||
|
EXPECT_GT(fd, -1);
|
||||||
|
EXPECT_TRUE(XPlatform::Syscall::CloseFile(fd));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(XPlatformSyscall, CreateAndOpenTempFileNegative) {
|
||||||
|
std::string pattern("does-not-adhere-to-pattern");
|
||||||
|
std::vector<char> pattern_vec(pattern.begin(), pattern.end());
|
||||||
|
|
||||||
|
const auto fd = XPlatform::Syscall::CreateAndOpenTempFile(pattern_vec);
|
||||||
|
EXPECT_EQ(fd, -1);
|
||||||
|
EXPECT_FALSE(XPlatform::Syscall::CloseFile(fd));
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue