Luna/apps/ls.cpp

122 lines
3.2 KiB
C++
Raw Normal View History

2023-05-12 21:47:20 +00:00
#include <grp.h>
#include <os/ArgumentParser.h>
2023-04-28 19:15:41 +00:00
#include <os/Directory.h>
#include <os/File.h>
2023-05-01 18:01:05 +00:00
#include <os/FileSystem.h>
2023-05-12 21:47:20 +00:00
#include <pwd.h>
struct UsernameCache
{
uid_t uid;
String name;
};
struct GroupCache
{
gid_t gid;
String name;
};
Vector<UsernameCache> g_user_cache;
Vector<GroupCache> g_group_cache;
Result<void> find_user_and_group(struct stat& st, StringView& owner, StringView& group)
{
for (const auto& user : g_user_cache)
{
if (user.uid == st.st_uid) owner = user.name.view();
}
if (owner.is_empty())
{
auto* pw = getpwuid(st.st_uid);
if (!pw) owner = "???";
else
owner = pw->pw_name;
auto name = TRY(String::from_string_view(owner));
TRY(g_user_cache.try_append({ st.st_uid, move(name) }));
}
for (const auto& grp : g_group_cache)
{
if (grp.gid == st.st_gid) group = grp.name.view();
}
if (group.is_empty())
{
auto* grp = getgrgid(st.st_gid);
if (!grp) group = "???";
else
group = grp->gr_name;
auto name = TRY(String::from_string_view(group));
TRY(g_group_cache.try_append({ st.st_gid, move(name) }));
}
return {};
}
2023-03-28 23:07:58 +00:00
2023-04-13 15:04:59 +00:00
Result<int> luna_main(int argc, char** argv)
2023-03-28 23:07:58 +00:00
{
StringView pathname;
bool show_all { false };
bool show_almost_all { false };
2023-05-01 18:01:05 +00:00
bool long_list { false };
os::ArgumentParser parser;
parser.add_description("List files contained in a directory (defaults to '.', the current directory)"_sv);
parser.add_system_program_info("ls"_sv);
parser.add_positional_argument(pathname, "directory"_sv, "."_sv);
parser.add_switch_argument(show_all, 'a', "all"_sv, "also list hidden files (whose filename begins with a dot)"_sv);
parser.add_switch_argument(show_almost_all, 'A', "almost-all"_sv, "list all files except '.' and '..'"_sv);
2023-05-01 18:01:05 +00:00
parser.add_switch_argument(long_list, 'l', ""_sv, "use a long listing format"_sv);
parser.parse(argc, argv);
2023-03-28 23:07:58 +00:00
2023-05-01 18:01:05 +00:00
Vector<String> files;
int dirfd = AT_FDCWD;
SharedPtr<os::Directory> dir;
if (os::FileSystem::is_directory(pathname))
{
dir = TRY(os::Directory::open(pathname));
dirfd = dir->fd();
auto filter = os::Directory::Filter::Hidden;
if (show_almost_all) filter = os::Directory::Filter::ParentAndBase;
else if (show_all)
filter = os::Directory::Filter::None;
2023-04-28 19:15:41 +00:00
2023-05-01 18:01:05 +00:00
files = TRY(dir->list(filter));
}
else if (os::FileSystem::exists(pathname))
2023-05-01 18:01:05 +00:00
{
auto str = TRY(String::from_string_view(pathname));
2023-05-01 18:01:05 +00:00
TRY(files.try_append(move(str)));
}
else
return err(ENOENT);
2023-03-28 23:07:58 +00:00
2023-05-01 18:01:05 +00:00
if (!long_list)
{
auto list = TRY(String::join(files, " "_sv));
if (!list.is_empty()) os::println("%s", list.chars());
2023-05-01 18:01:05 +00:00
}
else
{
for (const auto& file : files)
{
struct stat st;
2023-05-12 21:47:20 +00:00
TRY(os::FileSystem::stat({ dirfd, file.view() }, st));
StringView owner;
StringView group;
TRY(find_user_and_group(st, owner, group));
2023-04-28 20:41:44 +00:00
2023-05-12 21:47:20 +00:00
os::println("%6o %u %4s %4s %10lu %s", st.st_mode, st.st_nlink, owner.chars(), group.chars(), st.st_size,
2023-05-01 18:01:05 +00:00
file.chars());
}
}
2023-03-28 23:07:58 +00:00
return 0;
}