/* * libfud * Copyright 2024 Dominick Allen * * 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 "fud_directory.hpp" #include #include #include namespace fud { using CStat = struct stat; struct Stat : public CStat {}; Result DirectoryEntry::fromStat(const String& name, const Stat& statBuffer) { using RetType = Result; static_assert(std::is_same_v); static_assert(sizeof(decltype(statBuffer.st_ino)) <= sizeof(size_t)); static_assert(std::is_same_v); static_assert(sizeof(decltype(statBuffer.st_size)) <= sizeof(size_t)); static_assert(std::is_same_v); static_assert(sizeof(decltype(statBuffer.st_nlink)) <= sizeof(size_t)); size_t size{0}; if (statBuffer.st_size < 0) { return RetType::error(FudStatus::Failure); } size = static_cast(statBuffer.st_size); DirectoryEntryType entryType{}; switch (statBuffer.st_mode & S_IFMT) { case S_IFBLK: entryType = DirectoryEntryType::Block; break; case S_IFCHR: entryType = DirectoryEntryType::Character; break; case S_IFDIR: entryType = DirectoryEntryType::Directory; break; case S_IFIFO: entryType = DirectoryEntryType::NamedPipe; break; case S_IFLNK: entryType = DirectoryEntryType::SymbolicLink; break; case S_IFREG: entryType = DirectoryEntryType::RegularFile; break; case S_IFSOCK: entryType = DirectoryEntryType::UnixSocket; break; default: entryType = DirectoryEntryType::Unknown; break; } DirectoryEntry entry{}; auto copyStatus = entry.name.copy(name); if (copyStatus != FudStatus::Success) { return RetType::error(copyStatus); } entry.inode = statBuffer.st_ino; entry.size = size; entry.links = static_cast(statBuffer.st_nlink); entry.modificationTime = statBuffer.st_mtime; entry.entryType = entryType; return RetType::okay(std::move(entry)); } Result Directory::make(const String& name) { using RetType = Result; Directory directory{}; if (!name.valid()) { return RetType::error(FudStatus::ArgumentInvalid); } if (!name.utf8Valid()) { return RetType::error(FudStatus::Utf8Invalid); } auto copyResult = directory.m_name.copy(name); if (copyResult != FudStatus::Success) { return RetType::error(copyResult); } directory.m_directory = opendir(name.c_str()); if (directory.m_directory == nullptr) { return RetType::error(FudStatus::Failure); } directory.m_dirFd = dirfd(directory.m_directory); if (directory.m_dirFd == -1) { closedir(directory.m_directory); return RetType::error(FudStatus::Failure); } directory.m_errorCode = 0; return RetType::okay(std::move(directory)); } Directory::Directory(Directory&& rhs) noexcept : m_name{std::move(rhs.m_name)}, m_directory{rhs.m_directory}, m_errorCode{rhs.m_errorCode}, m_dirFd{rhs.m_dirFd} { rhs.m_directory = nullptr; rhs.m_dirFd = -1; } Directory::~Directory() { if (m_directory != nullptr) { closedir(m_directory); m_directory = nullptr; } } Result Directory::info() { using RetType = Result; Stat sBuffer{}; auto fStatus = fstat(m_dirFd, &sBuffer); if (fStatus == -1) { m_errorCode = errno; return RetType::error(FudStatus::Failure); } auto retValue = DirectoryEntry::fromStat(m_name, sBuffer); if (retValue.isOkay()) { m_errorCode = 0; } return retValue; } Result, FudStatus> Directory::getNextEntry() { using RetType = Result, FudStatus>; errno = 0; auto* dirEntry = readdir(m_directory); if (dirEntry == nullptr) { if (errno != 0) { m_errorCode = errno; return RetType::error(FudStatus::Failure); } return RetType::okay(NullOpt); } const char* entryNameCString = dirEntry->d_name; if (entryNameCString == nullptr) { m_errorCode = -1; return RetType::error(FudStatus::NullPointer); } m_errorCode = 0; Stat sBuffer{}; auto flags = 0; auto fStatus = fstatat(m_dirFd, entryNameCString, &sBuffer, flags); if (fStatus == -1) { m_errorCode = errno; return RetType::error(FudStatus::Failure); } auto entryNameResult = String::makeFromCString(entryNameCString); if (entryNameResult.isError()) { return RetType::error(entryNameResult); } auto retValue = DirectoryEntry::fromStat(entryNameResult.getOkay(), sBuffer); if (retValue.isOkay()) { m_errorCode = 0; return RetType::okay(Option::take(retValue.takeOkay())); } return RetType::error(retValue.getError()); } FudStatus Directory::reset() { if (m_directory != nullptr) { rewinddir(m_directory); } else { return FudStatus::ObjectInvalid; } return FudStatus::Success; } } // namespace fud