HDFS-16463. Make dirent cross platform compatible (#4370)
* jnihelper.c in HDFS native client uses dirent.h. This header file isn't available on Windows. * This PR provides a cross platform compatible implementation for dirent under the XPlatform library.
This commit is contained in:
parent
6e11c94170
commit
d557c44d94
@ -35,6 +35,8 @@ add_library(native_mini_dfs
|
|||||||
../libhdfs/jni_helper.c
|
../libhdfs/jni_helper.c
|
||||||
${OS_DIR}/mutexes.c
|
${OS_DIR}/mutexes.c
|
||||||
${OS_DIR}/thread_local_storage.c
|
${OS_DIR}/thread_local_storage.c
|
||||||
|
$<TARGET_OBJECTS:x_platform_obj>
|
||||||
|
$<TARGET_OBJECTS:x_platform_obj_c_api>
|
||||||
)
|
)
|
||||||
|
|
||||||
add_executable(test_native_mini_dfs test_native_mini_dfs.c)
|
add_executable(test_native_mini_dfs test_native_mini_dfs.c)
|
||||||
|
@ -39,6 +39,8 @@ hadoop_add_dual_library(hdfs
|
|||||||
jclasses.c
|
jclasses.c
|
||||||
${OS_DIR}/mutexes.c
|
${OS_DIR}/mutexes.c
|
||||||
${OS_DIR}/thread_local_storage.c
|
${OS_DIR}/thread_local_storage.c
|
||||||
|
$<TARGET_OBJECTS:x_platform_obj>
|
||||||
|
$<TARGET_OBJECTS:x_platform_obj_c_api>
|
||||||
)
|
)
|
||||||
if(NEED_LINK_DL)
|
if(NEED_LINK_DL)
|
||||||
set(LIB_DL dl)
|
set(LIB_DL dl)
|
||||||
|
@ -23,10 +23,10 @@
|
|||||||
#include "platform.h"
|
#include "platform.h"
|
||||||
#include "os/mutexes.h"
|
#include "os/mutexes.h"
|
||||||
#include "os/thread_local_storage.h"
|
#include "os/thread_local_storage.h"
|
||||||
|
#include "x-platform/c-api/dirent.h"
|
||||||
#include "x-platform/types.h"
|
#include "x-platform/types.h"
|
||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <dirent.h>
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
@ -22,5 +22,6 @@ else()
|
|||||||
set(SYSCALL_SRC syscall_linux.cc)
|
set(SYSCALL_SRC syscall_linux.cc)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_library(x_platform_obj OBJECT ${SYSCALL_SRC} utils.cc)
|
add_library(x_platform_obj OBJECT ${SYSCALL_SRC} utils.cc dirent.cc)
|
||||||
add_library(x_platform_obj_c_api OBJECT $<TARGET_OBJECTS:x_platform_obj> c-api/syscall.cc)
|
add_library(x_platform_obj_c_api OBJECT $<TARGET_OBJECTS:x_platform_obj> c-api/syscall.cc c-api/dirent.cc)
|
||||||
|
target_compile_definitions(x_platform_obj_c_api PRIVATE USE_X_PLATFORM_DIRENT)
|
||||||
|
@ -0,0 +1,68 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_CORE_DIRENT_H
|
||||||
|
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_CORE_DIRENT_H
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DIR struct holds the pointer to XPlatform::Dirent instance. Since this will
|
||||||
|
* be used in C, we can't hold the pointer to XPlatform::Dirent. We're working
|
||||||
|
* around this by using a void pointer and casting it to XPlatform::Dirent when
|
||||||
|
* needed in C++.
|
||||||
|
*/
|
||||||
|
typedef struct DIR {
|
||||||
|
void *x_platform_dirent_ptr;
|
||||||
|
} DIR;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* dirent struct contains the name of the file/folder while iterating through
|
||||||
|
* the directory's children.
|
||||||
|
*/
|
||||||
|
struct dirent {
|
||||||
|
char d_name[256];
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a directory for iteration. Internally, it instantiates DIR struct for
|
||||||
|
* the given path. closedir must be called on the returned pointer to DIR struct
|
||||||
|
* when done.
|
||||||
|
*
|
||||||
|
* @param dir_path The path to the directory to iterate through.
|
||||||
|
* @return A pointer to the DIR struct.
|
||||||
|
*/
|
||||||
|
DIR *opendir(const char *dir_path);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* For iterating through the children of the directory pointed to by the DIR
|
||||||
|
* struct pointer.
|
||||||
|
*
|
||||||
|
* @param dir The pointer to the DIR struct.
|
||||||
|
* @return A pointer to dirent struct containing the name of the current child
|
||||||
|
* file/folder.
|
||||||
|
*/
|
||||||
|
struct dirent *readdir(DIR *dir);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* De-allocates the XPlatform::Dirent instance pointed to by the DIR pointer.
|
||||||
|
*
|
||||||
|
* @param dir The pointer to DIR struct to close.
|
||||||
|
* @return 0 if successful.
|
||||||
|
*/
|
||||||
|
int closedir(DIR *dir);
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,92 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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 <algorithm>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <iostream>
|
||||||
|
#include <iterator>
|
||||||
|
#include <system_error>
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
|
#include "x-platform/c-api/dirent.h"
|
||||||
|
#include "x-platform/dirent.h"
|
||||||
|
|
||||||
|
DIR *opendir(const char *dir_path) {
|
||||||
|
const auto dir = new DIR;
|
||||||
|
dir->x_platform_dirent_ptr = new XPlatform::Dirent(dir_path);
|
||||||
|
return dir;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct dirent *readdir(DIR *dir) {
|
||||||
|
/*
|
||||||
|
* We will use a static variable to hold the dirent, so that we align with the
|
||||||
|
* readdir's implementation in dirent.h header file in Linux.
|
||||||
|
*/
|
||||||
|
static struct dirent static_dir_entry;
|
||||||
|
|
||||||
|
// Get the XPlatform::Dirent instance and move the iterator.
|
||||||
|
const auto x_platform_dirent =
|
||||||
|
static_cast<XPlatform::Dirent *>(dir->x_platform_dirent_ptr);
|
||||||
|
const auto dir_entry = x_platform_dirent->NextFile();
|
||||||
|
|
||||||
|
// End of iteration.
|
||||||
|
if (std::holds_alternative<std::monostate>(dir_entry)) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error in iteration.
|
||||||
|
if (std::holds_alternative<std::error_code>(dir_entry)) {
|
||||||
|
const auto err = std::get<std::error_code>(dir_entry);
|
||||||
|
errno = err.value();
|
||||||
|
|
||||||
|
#ifdef X_PLATFORM_C_API_DIRENT_DEBUG
|
||||||
|
std::cerr << "Error in listing directory: " << err.message() << std::endl;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the current child file/folder's name.
|
||||||
|
if (std::holds_alternative<std::filesystem::directory_entry>(dir_entry)) {
|
||||||
|
const auto entry = std::get<std::filesystem::directory_entry>(dir_entry);
|
||||||
|
const auto filename = entry.path().filename().string();
|
||||||
|
|
||||||
|
// The file name's length shouldn't exceed 256.
|
||||||
|
if (filename.length() >= 256) {
|
||||||
|
errno = 1;
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::fill(std::begin(static_dir_entry.d_name),
|
||||||
|
std::end(static_dir_entry.d_name), '\0');
|
||||||
|
std::copy(filename.begin(), filename.end(),
|
||||||
|
std::begin(static_dir_entry.d_name));
|
||||||
|
}
|
||||||
|
return &static_dir_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
int closedir(DIR *dir) {
|
||||||
|
const auto x_platform_dirent =
|
||||||
|
static_cast<XPlatform::Dirent *>(dir->x_platform_dirent_ptr);
|
||||||
|
delete x_platform_dirent;
|
||||||
|
delete dir;
|
||||||
|
|
||||||
|
// We can't use the void return type for closedir since we want to align the
|
||||||
|
// closedir method's signature in dirent.h header file in Linux.
|
||||||
|
return 0;
|
||||||
|
}
|
@ -0,0 +1,39 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_DIRENT_H
|
||||||
|
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_DIRENT_H
|
||||||
|
|
||||||
|
#if !(defined(WIN32) || defined(USE_X_PLATFORM_DIRENT))
|
||||||
|
|
||||||
|
/*
|
||||||
|
* For non-Windows environments, we use the dirent.h header itself.
|
||||||
|
*/
|
||||||
|
#include <dirent.h>
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If it's a Windows environment or if the macro USE_X_PLATFORM_DIRENT is
|
||||||
|
* defined, we switch to using dirent from the XPlatform library.
|
||||||
|
*/
|
||||||
|
#include "x-platform/c-api/extern/dirent.h"
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,35 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_EXTERN_DIRENT_H
|
||||||
|
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_C_API_EXTERN_DIRENT_H
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We will use extern "C" only on Windows.
|
||||||
|
*/
|
||||||
|
#if defined(WIN32) && defined(__cplusplus)
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "x-platform/c-api/core/dirent.h"
|
||||||
|
|
||||||
|
#if defined(WIN32) && defined(__cplusplus)
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,38 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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 <filesystem>
|
||||||
|
#include <system_error>
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
|
#include "dirent.h"
|
||||||
|
|
||||||
|
std::variant<std::monostate, std::filesystem::directory_entry, std::error_code>
|
||||||
|
XPlatform::Dirent::NextFile() {
|
||||||
|
if (dir_it_err_) {
|
||||||
|
return dir_it_err_;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dir_it_ == std::filesystem::end(dir_it_)) {
|
||||||
|
return std::monostate();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::filesystem::directory_entry dir_entry = *dir_it_;
|
||||||
|
dir_it_ = dir_it_.increment(dir_it_err_);
|
||||||
|
return dir_entry;
|
||||||
|
}
|
@ -0,0 +1,77 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_DIRENT
|
||||||
|
#define NATIVE_LIBHDFSPP_LIB_CROSS_PLATFORM_DIRENT
|
||||||
|
|
||||||
|
#include <filesystem>
|
||||||
|
#include <string>
|
||||||
|
#include <system_error>
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
|
namespace XPlatform {
|
||||||
|
/**
|
||||||
|
* {@class XPlatform::Dirent} provides the functionality to perform a one-time
|
||||||
|
* iteration per {@link XPlatform::Dirent} through the child files or folders
|
||||||
|
* under a given path.
|
||||||
|
*/
|
||||||
|
class Dirent {
|
||||||
|
public:
|
||||||
|
Dirent(const std::string &path)
|
||||||
|
: dir_it_{std::filesystem::path{path}, dir_it_err_} {}
|
||||||
|
|
||||||
|
// Abiding to the Rule of 5
|
||||||
|
Dirent(const Dirent &) = default;
|
||||||
|
Dirent(Dirent &&) = default;
|
||||||
|
Dirent &operator=(const Dirent &) = default;
|
||||||
|
Dirent &operator=(Dirent &&) = default;
|
||||||
|
~Dirent() = default;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Advances the iterator {@link XPlatform::Dirent#dir_it_} to the next file in
|
||||||
|
* the given path.
|
||||||
|
*
|
||||||
|
* @return An {@link std::variant} comprising of any one of the following
|
||||||
|
* types:
|
||||||
|
* 1. {@link std::monostate} which indicates the end of iteration of all the
|
||||||
|
* files in the given path.
|
||||||
|
* 2. {@link std::filesystem::directory_entry} which is the directory entry of
|
||||||
|
* the current file.
|
||||||
|
* 3. {@link std::error_code} which corresponds to the error in retrieving the
|
||||||
|
* file.
|
||||||
|
*/
|
||||||
|
std::variant<std::monostate, std::filesystem::directory_entry,
|
||||||
|
std::error_code>
|
||||||
|
NextFile();
|
||||||
|
|
||||||
|
private:
|
||||||
|
/**
|
||||||
|
* Indicates the error corresponding to the most recent invocation of
|
||||||
|
* directory iteration by {@link XPlatform::Dirent#dir_it_}.
|
||||||
|
*/
|
||||||
|
std::error_code dir_it_err_{};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The iterator used for iterating through the files or folders under the
|
||||||
|
* given path.
|
||||||
|
*/
|
||||||
|
std::filesystem::directory_iterator dir_it_;
|
||||||
|
};
|
||||||
|
} // namespace XPlatform
|
||||||
|
|
||||||
|
#endif
|
@ -36,3 +36,15 @@ add_executable(x_platform_types_test types_test.cc)
|
|||||||
target_include_directories(x_platform_types_test PRIVATE ${LIBHDFSPP_LIB_DIR})
|
target_include_directories(x_platform_types_test PRIVATE ${LIBHDFSPP_LIB_DIR})
|
||||||
target_link_libraries(x_platform_types_test gtest_main)
|
target_link_libraries(x_platform_types_test gtest_main)
|
||||||
add_test(x_platform_types_test x_platform_types_test)
|
add_test(x_platform_types_test x_platform_types_test)
|
||||||
|
|
||||||
|
add_library(x_platform_dirent_test_obj OBJECT $<TARGET_OBJECTS:x_platform_obj> dirent_test.cc)
|
||||||
|
add_executable(x_platform_dirent_test $<TARGET_OBJECTS:x_platform_dirent_test_obj> $<TARGET_OBJECTS:x_platform_obj>)
|
||||||
|
target_include_directories(x_platform_dirent_test PRIVATE ${LIBHDFSPP_LIB_DIR})
|
||||||
|
target_link_libraries(x_platform_dirent_test PRIVATE gtest_main)
|
||||||
|
add_test(x_platform_dirent_test x_platform_dirent_test)
|
||||||
|
|
||||||
|
add_executable(x_platform_dirent_c_test $<TARGET_OBJECTS:x_platform_dirent_test_obj> $<TARGET_OBJECTS:x_platform_obj> $<TARGET_OBJECTS:x_platform_obj_c_api> c-api/dirent_test.cc)
|
||||||
|
target_compile_definitions(x_platform_dirent_c_test PRIVATE USE_X_PLATFORM_DIRENT)
|
||||||
|
target_include_directories(x_platform_dirent_c_test PRIVATE ${LIBHDFSPP_LIB_DIR} ../)
|
||||||
|
target_link_libraries(x_platform_dirent_c_test PRIVATE gtest_main)
|
||||||
|
add_test(x_platform_dirent_c_test x_platform_dirent_c_test)
|
||||||
|
@ -0,0 +1,100 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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 <cassert>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <unordered_set>
|
||||||
|
|
||||||
|
#include "x-platform/c-api/dirent.h"
|
||||||
|
#include "x-platform/c-api/dirent_test.h"
|
||||||
|
|
||||||
|
std::unordered_set<std::string>
|
||||||
|
DirentCApiTest::ListDirAndFiles(const std::string &path) const {
|
||||||
|
std::stringstream err_msg;
|
||||||
|
std::unordered_set<std::string> paths;
|
||||||
|
|
||||||
|
DIR *dir = opendir(path.c_str());
|
||||||
|
if (dir == nullptr) {
|
||||||
|
err_msg << "Unable to open directory " << path;
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
for (struct dirent *file; (file = readdir(dir)) != nullptr; errno = 0) {
|
||||||
|
std::filesystem::path absolute_path{path};
|
||||||
|
absolute_path = absolute_path / file->d_name;
|
||||||
|
paths.emplace(absolute_path.string());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errno != 0) {
|
||||||
|
err_msg << "Expected errno to be 0, instead it is " << errno;
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (const auto result = closedir(dir); result != 0) {
|
||||||
|
err_msg << "Expected the return value of closedir() to be 0, instead it is "
|
||||||
|
<< result;
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
return paths;
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestEmptyFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestOneFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(1, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestOneFile) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 1);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestMultipleFolders) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(10, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestMultipleFiles) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 10);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestOneFileAndFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(1, 1);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentCApiTest, TestMultipleFilesAndFolders) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(10, 10);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
@ -0,0 +1,37 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef LIBHDFSPP_CROSS_PLATFORM_C_API_DIRENT_TEST
|
||||||
|
#define LIBHDFSPP_CROSS_PLATFORM_C_API_DIRENT_TEST
|
||||||
|
|
||||||
|
#include "x-platform/dirent.h"
|
||||||
|
#include "x-platform/dirent_test.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test fixture for testing {@link XPlatform::Dirent}'s C APIs.
|
||||||
|
*/
|
||||||
|
class DirentCApiTest : public DirentTest {
|
||||||
|
protected:
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
|
[[nodiscard]] std::unordered_set<std::string>
|
||||||
|
ListDirAndFiles(const std::string &path) const override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -0,0 +1,151 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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 <algorithm>
|
||||||
|
#include <filesystem>
|
||||||
|
#include <fstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <sstream>
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <string>
|
||||||
|
#include <system_error>
|
||||||
|
#include <unordered_set>
|
||||||
|
#include <variant>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
#include "dirent_test.h"
|
||||||
|
#include "x-platform/dirent.h"
|
||||||
|
#include "x-platform/syscall.h"
|
||||||
|
|
||||||
|
void DirentTest::SetUp() {
|
||||||
|
tmp_root_ = std::filesystem::temp_directory_path() / GetTempName();
|
||||||
|
if (!std::filesystem::create_directories(tmp_root_)) {
|
||||||
|
std::stringstream err_msg;
|
||||||
|
err_msg << "Unable to create temp directory " << tmp_root_.string();
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void DirentTest::TearDown() { std::filesystem::remove_all(tmp_root_); }
|
||||||
|
|
||||||
|
std::string DirentTest::GetTempName(const std::string &pattern) const {
|
||||||
|
std::vector pattern_raw(pattern.begin(), pattern.end());
|
||||||
|
if (!XPlatform::Syscall::CreateTempDir(pattern_raw)) {
|
||||||
|
std::stringstream err_msg;
|
||||||
|
err_msg << "Creating temp dir failed" << std::endl;
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string tmp_dir_path(pattern_raw.data());
|
||||||
|
std::filesystem::remove(tmp_dir_path);
|
||||||
|
return tmp_dir_path;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unordered_set<std::string>
|
||||||
|
DirentTest::CreateTempDirAndFiles(std::size_t num_dirs,
|
||||||
|
std::size_t num_files) const {
|
||||||
|
std::unordered_set<std::string> paths;
|
||||||
|
for (std::size_t i = 0; i < num_dirs; ++i) {
|
||||||
|
const auto tmp_dir_absolute_path = tmp_root_ / std::to_string(i);
|
||||||
|
if (std::error_code err;
|
||||||
|
!std::filesystem::create_directories(tmp_dir_absolute_path, err)) {
|
||||||
|
std::stringstream err_msg;
|
||||||
|
err_msg << "Unable to create the temp dir "
|
||||||
|
<< tmp_dir_absolute_path.string() << " reason: " << err.message();
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
paths.emplace(tmp_dir_absolute_path.string());
|
||||||
|
}
|
||||||
|
|
||||||
|
for (std::size_t i = 0; i < num_files; ++i) {
|
||||||
|
std::stringstream tmp_filename;
|
||||||
|
tmp_filename << i << ".txt";
|
||||||
|
const auto tmp_file_absolute_path = tmp_root_ / tmp_filename.str();
|
||||||
|
std::ofstream{tmp_file_absolute_path};
|
||||||
|
paths.emplace(tmp_file_absolute_path.string());
|
||||||
|
}
|
||||||
|
return paths;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unordered_set<std::string>
|
||||||
|
DirentTest::ListDirAndFiles(const std::string &path) const {
|
||||||
|
std::unordered_set<std::string> paths;
|
||||||
|
|
||||||
|
XPlatform::Dirent dirent(path);
|
||||||
|
for (auto dir_entry = dirent.NextFile();
|
||||||
|
!std::holds_alternative<std::monostate>(dir_entry);
|
||||||
|
dir_entry = dirent.NextFile()) {
|
||||||
|
if (std::holds_alternative<std::error_code>(dir_entry)) {
|
||||||
|
std::stringstream err_msg;
|
||||||
|
const auto err = std::get<std::error_code>(dir_entry);
|
||||||
|
err_msg << "Error in listing directory " << path
|
||||||
|
<< " cause: " << err.message();
|
||||||
|
throw std::runtime_error(err_msg.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (std::holds_alternative<std::filesystem::directory_entry>(dir_entry)) {
|
||||||
|
const auto entry = std::get<std::filesystem::directory_entry>(dir_entry);
|
||||||
|
paths.emplace(entry.path().string());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return paths;
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestEmptyFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestOneFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(1, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestOneFile) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 1);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestMultipleFolders) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(10, 0);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestMultipleFiles) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(0, 10);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestOneFileAndFolder) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(1, 1);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(DirentTest, TestMultipleFilesAndFolders) {
|
||||||
|
const auto expected = CreateTempDirAndFiles(10, 10);
|
||||||
|
const auto actual = ListDirAndFiles(tmp_root_.string());
|
||||||
|
EXPECT_EQ(expected, actual);
|
||||||
|
}
|
@ -0,0 +1,78 @@
|
|||||||
|
/**
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef LIBHDFSPP_CROSS_PLATFORM_DIRENT_TEST
|
||||||
|
#define LIBHDFSPP_CROSS_PLATFORM_DIRENT_TEST
|
||||||
|
|
||||||
|
#include <filesystem>
|
||||||
|
#include <string>
|
||||||
|
#include <unordered_set>
|
||||||
|
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
#include "x-platform/dirent.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test fixture for testing {@link XPlatform::Dirent}.
|
||||||
|
*/
|
||||||
|
class DirentTest : public ::testing::Test {
|
||||||
|
protected:
|
||||||
|
void SetUp() override;
|
||||||
|
void TearDown() override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets a name for creating temporary file or folder. This also ensures that
|
||||||
|
* the temporary file or folder does not exist.
|
||||||
|
*
|
||||||
|
* @param pattern The pattern to use for naming the temporary directory.
|
||||||
|
* @return The temporary file or folder name that can be used for creating the
|
||||||
|
* same.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] std::string
|
||||||
|
GetTempName(const std::string &pattern = "test_XXXXXX") const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates the given number of temporary files and directories under the
|
||||||
|
* {@link DirentTest#tmp_root_}.
|
||||||
|
*
|
||||||
|
* @param num_dirs The number of temporary directories to create.
|
||||||
|
* @param num_files The number of temporary files to create.
|
||||||
|
* @return An {@link std::unordered_set> of the absolute paths of all the
|
||||||
|
* temporary files and folders that were created.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] std::unordered_set<std::string>
|
||||||
|
CreateTempDirAndFiles(std::size_t num_dirs, std::size_t num_files) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lists all the children of the given path.
|
||||||
|
*
|
||||||
|
* @param path The path whose children must be listed.
|
||||||
|
* @return An {@link std::unordered_set} containing the absolute paths of all
|
||||||
|
* the children of the given path.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] virtual std::unordered_set<std::string>
|
||||||
|
ListDirAndFiles(const std::string &path) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The root in temp folder under which the rest of all the temporary files and
|
||||||
|
* folders will be created for the purpose of testing.
|
||||||
|
*/
|
||||||
|
std::filesystem::path tmp_root_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue
Block a user