2023-05-12 21:47:20 +00:00
|
|
|
#include <bits/modes.h>
|
2023-04-19 16:20:44 +00:00
|
|
|
#include <dirent.h>
|
2023-04-13 16:33:04 +00:00
|
|
|
#include <errno.h>
|
2023-04-18 17:36:29 +00:00
|
|
|
#include <fcntl.h>
|
2023-05-02 08:50:39 +00:00
|
|
|
#include <luna/String.h>
|
|
|
|
#include <os/Directory.h>
|
|
|
|
#include <os/FileSystem.h>
|
2023-04-13 16:33:04 +00:00
|
|
|
#include <pwd.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
namespace os::FileSystem
|
|
|
|
{
|
2023-05-03 15:37:26 +00:00
|
|
|
bool exists(const Path& path)
|
2023-04-13 16:33:04 +00:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
|
2023-05-12 21:47:20 +00:00
|
|
|
if (stat(path, st).has_error()) return false;
|
2023-04-13 16:33:04 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
bool is_directory(const Path& path)
|
2023-04-13 16:33:04 +00:00
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
|
2023-05-12 21:47:20 +00:00
|
|
|
if (stat(path, st).has_error()) return false;
|
2023-04-13 16:33:04 +00:00
|
|
|
|
|
|
|
return S_ISDIR(st.st_mode);
|
|
|
|
}
|
|
|
|
|
2023-05-12 21:47:20 +00:00
|
|
|
Result<void> stat(const Path& path, struct stat& st)
|
|
|
|
{
|
|
|
|
long rc =
|
|
|
|
syscall(SYS_fstatat, path.dirfd(), path.name().chars(), &st, path.is_empty_path() ? AT_EMPTY_PATH : 0);
|
|
|
|
|
|
|
|
return Result<void>::from_syscall(rc);
|
|
|
|
}
|
|
|
|
|
2023-04-13 16:33:04 +00:00
|
|
|
Result<void> create_directory(StringView path, mode_t mode)
|
|
|
|
{
|
|
|
|
long rc = syscall(SYS_mkdir, path.chars(), mode);
|
|
|
|
|
2023-04-18 16:49:24 +00:00
|
|
|
return Result<void>::from_syscall(rc);
|
2023-04-13 16:33:04 +00:00
|
|
|
}
|
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
Result<void> remove(const Path& path)
|
2023-04-13 16:33:04 +00:00
|
|
|
{
|
2023-05-03 15:37:26 +00:00
|
|
|
long rc = syscall(SYS_unlinkat, path.dirfd(), path.name().chars(), 0);
|
2023-04-13 16:33:04 +00:00
|
|
|
|
2023-04-18 16:49:24 +00:00
|
|
|
return Result<void>::from_syscall(rc);
|
2023-04-13 16:33:04 +00:00
|
|
|
}
|
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
Result<void> remove_tree(const Path& path)
|
2023-04-19 16:20:44 +00:00
|
|
|
{
|
2023-05-03 15:37:26 +00:00
|
|
|
auto rc = remove(path);
|
2023-04-19 16:20:44 +00:00
|
|
|
if (!rc.has_error()) return {};
|
|
|
|
if (rc.error() != ENOTEMPTY) return rc.release_error();
|
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
auto dir = TRY(os::Directory::open(path));
|
2023-04-19 16:20:44 +00:00
|
|
|
|
2023-04-28 20:41:44 +00:00
|
|
|
Vector<String> entries = TRY(dir->list(os::Directory::Filter::ParentAndBase));
|
2023-04-19 16:20:44 +00:00
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
for (const auto& entry : entries) { TRY(remove_tree({ dir->fd(), entry.view() })); }
|
2023-04-19 16:20:44 +00:00
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
return remove(path);
|
2023-04-19 16:20:44 +00:00
|
|
|
}
|
|
|
|
|
2023-04-13 16:33:04 +00:00
|
|
|
Result<String> working_directory()
|
|
|
|
{
|
|
|
|
char* ptr = getcwd(NULL, 0);
|
|
|
|
if (!ptr) return err(errno);
|
|
|
|
return String { ptr };
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<String> home_directory()
|
|
|
|
{
|
|
|
|
char* home = getenv("HOME");
|
|
|
|
if (home) return String::from_cstring(home);
|
|
|
|
|
|
|
|
struct passwd* pw = getpwuid(getuid());
|
|
|
|
if (!pw) return err(ENOENT);
|
|
|
|
|
|
|
|
return String::from_cstring(pw->pw_dir);
|
|
|
|
}
|
2023-04-18 16:46:19 +00:00
|
|
|
|
|
|
|
Result<void> change_directory(StringView path)
|
|
|
|
{
|
|
|
|
long rc = syscall(SYS_chdir, path.chars());
|
|
|
|
|
2023-04-18 16:49:24 +00:00
|
|
|
return Result<void>::from_syscall(rc);
|
2023-04-18 16:46:19 +00:00
|
|
|
}
|
2023-04-13 16:33:04 +00:00
|
|
|
}
|