2023-05-12 21:47:20 +00:00
|
|
|
#include <grp.h>
|
2023-05-28 19:51:25 +00:00
|
|
|
#include <luna/Sort.h>
|
|
|
|
#include <luna/StringBuilder.h>
|
2023-05-13 09:15:28 +00:00
|
|
|
#include <luna/Units.h>
|
2023-03-29 16:27:02 +00:00
|
|
|
#include <os/ArgumentParser.h>
|
2023-04-28 19:15:41 +00:00
|
|
|
#include <os/Directory.h>
|
2023-05-01 17:32:00 +00:00
|
|
|
#include <os/File.h>
|
2023-05-01 18:01:05 +00:00
|
|
|
#include <os/FileSystem.h>
|
2023-05-26 16:22:50 +00:00
|
|
|
#include <os/Mode.h>
|
2023-05-12 21:47:20 +00:00
|
|
|
#include <pwd.h>
|
|
|
|
|
2023-05-20 20:10:01 +00:00
|
|
|
void find_user_and_group(struct stat& st, StringView& owner, StringView& group)
|
2023-05-12 21:47:20 +00:00
|
|
|
{
|
2023-05-20 20:10:01 +00:00
|
|
|
auto* pw = getpwuid(st.st_uid);
|
|
|
|
if (!pw) owner = "???";
|
|
|
|
else
|
|
|
|
owner = pw->pw_name;
|
2023-05-12 21:47:20 +00:00
|
|
|
|
2023-05-20 20:10:01 +00:00
|
|
|
auto* grp = getgrgid(st.st_gid);
|
|
|
|
if (!grp) group = "???";
|
|
|
|
else
|
|
|
|
group = grp->gr_name;
|
2023-05-12 21:47:20 +00:00
|
|
|
}
|
2023-03-28 23:07:58 +00:00
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
int sort_name(const os::Directory::Entry* a, const os::Directory::Entry* b)
|
|
|
|
{
|
|
|
|
return String::compare(&a->name, &b->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int sort_size(const os::Directory::Entry* a, const os::Directory::Entry* b)
|
|
|
|
{
|
|
|
|
return (a->size < b->size) ? -1 : ((a->size == b->size) ? 0 : 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int sort_time(const os::Directory::Entry* a, const os::Directory::Entry* b)
|
|
|
|
{
|
|
|
|
return (a->mtime < b->mtime) ? -1 : ((a->mtime == b->mtime) ? 0 : 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int (*sort_function)(const os::Directory::Entry*, const os::Directory::Entry*) = sort_name;
|
|
|
|
|
|
|
|
int sort_reverse(const os::Directory::Entry* a, const os::Directory::Entry* b)
|
|
|
|
{
|
|
|
|
int rc = sort_function(a, b);
|
|
|
|
if (rc < 0) return 1;
|
|
|
|
if (rc > 0) return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Result<String> entry_join(const Vector<os::Directory::Entry>& vec, StringView delim)
|
|
|
|
{
|
|
|
|
if (vec.size() == 0) return String {};
|
|
|
|
if (vec.size() == 1) return vec[0].name.clone();
|
|
|
|
|
|
|
|
StringBuilder sb;
|
|
|
|
TRY(sb.add(vec[0].name));
|
|
|
|
|
|
|
|
for (usize i = 1; i < vec.size(); i++)
|
|
|
|
{
|
|
|
|
TRY(sb.add(delim));
|
|
|
|
TRY(sb.add(vec[i].name));
|
|
|
|
}
|
|
|
|
|
|
|
|
return sb.string();
|
|
|
|
}
|
|
|
|
|
2023-04-13 15:04:59 +00:00
|
|
|
Result<int> luna_main(int argc, char** argv)
|
2023-03-28 23:07:58 +00:00
|
|
|
{
|
2023-03-29 16:27:02 +00:00
|
|
|
StringView pathname;
|
2023-03-29 20:19:53 +00:00
|
|
|
bool show_all { false };
|
|
|
|
bool show_almost_all { false };
|
2023-05-01 18:01:05 +00:00
|
|
|
bool long_list { false };
|
2023-05-13 09:15:28 +00:00
|
|
|
bool human_readable { false };
|
2023-05-13 10:01:09 +00:00
|
|
|
bool si { false };
|
2023-05-20 19:48:46 +00:00
|
|
|
bool follow_symlink_args { false };
|
2023-05-23 12:53:38 +00:00
|
|
|
bool one_per_line { false };
|
|
|
|
bool list_directories { false };
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
StringView sort_type { "name" };
|
|
|
|
|
|
|
|
bool reverse_sort { false };
|
|
|
|
|
|
|
|
bool should_sort { true };
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
os::ArgumentParser parser;
|
2023-04-19 17:16:45 +00:00
|
|
|
parser.add_description("List files contained in a directory (defaults to '.', the current directory)"_sv);
|
2023-04-28 14:33:05 +00:00
|
|
|
parser.add_system_program_info("ls"_sv);
|
2023-04-11 20:13:54 +00:00
|
|
|
parser.add_positional_argument(pathname, "directory"_sv, "."_sv);
|
2023-04-19 17:16:45 +00:00
|
|
|
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);
|
2023-05-13 09:15:28 +00:00
|
|
|
parser.add_switch_argument(human_readable, 'h', "human-readable"_sv,
|
|
|
|
"with -l, show human-readable sizes e.g. 2KiB, 6GiB"_sv);
|
2023-05-13 10:01:09 +00:00
|
|
|
parser.add_switch_argument(si, ' ', "si"_sv, "same as -h, but show sizes in powers of 10"_sv);
|
2023-05-20 19:48:46 +00:00
|
|
|
parser.add_switch_argument(follow_symlink_args, 'H', "dereference-args"_sv,
|
|
|
|
"follow symbolic links listed as arguments"_sv);
|
2023-05-23 12:53:38 +00:00
|
|
|
parser.add_switch_argument(one_per_line, '1', ""_sv, "list one file per line"_sv);
|
|
|
|
parser.add_switch_argument(list_directories, 'd', "directory"_sv, "list directories instead of their contents"_sv);
|
2023-05-28 19:51:25 +00:00
|
|
|
parser.add_value_argument(sort_type, ' ', "sort"_sv, true, "sort by name, size or time"_sv);
|
|
|
|
parser.add_switch_argument(reverse_sort, 'r', "reverse"_sv, "reverse order while sorting"_sv);
|
2023-03-29 16:27:02 +00:00
|
|
|
parser.parse(argc, argv);
|
2023-03-28 23:07:58 +00:00
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
Vector<os::Directory::Entry> files;
|
2023-05-01 18:01:05 +00:00
|
|
|
int dirfd = AT_FDCWD;
|
|
|
|
SharedPtr<os::Directory> dir;
|
|
|
|
|
2023-05-23 12:53:38 +00:00
|
|
|
if (!long_list) follow_symlink_args = true;
|
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
if (sort_type == "none"_sv) { should_sort = false; }
|
|
|
|
else if (sort_type == "name"_sv) { sort_function = sort_name; }
|
|
|
|
else if (sort_type == "size"_sv) { sort_function = sort_size; }
|
|
|
|
else if (sort_type == "time"_sv) { sort_function = sort_time; }
|
|
|
|
else
|
|
|
|
{
|
|
|
|
os::eprintln("%s: unknown sort type: %s", argv[0], sort_type.chars());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-05-23 12:53:38 +00:00
|
|
|
if (os::FileSystem::is_directory(pathname, follow_symlink_args) && !list_directories)
|
2023-05-01 18:01:05 +00:00
|
|
|
{
|
|
|
|
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));
|
|
|
|
}
|
2023-05-20 19:48:46 +00:00
|
|
|
else if (os::FileSystem::exists(pathname, follow_symlink_args))
|
2023-05-01 18:01:05 +00:00
|
|
|
{
|
2023-05-28 19:51:25 +00:00
|
|
|
struct stat st;
|
|
|
|
TRY(os::FileSystem::stat(pathname, st, false));
|
|
|
|
|
|
|
|
os::Directory::Entry entry = {
|
|
|
|
.name = TRY(String::from_string_view(pathname)),
|
|
|
|
.mode = st.st_mode,
|
|
|
|
.size = (usize)st.st_size,
|
|
|
|
.mtime = st.st_mtime,
|
|
|
|
};
|
|
|
|
|
|
|
|
TRY(files.try_append(move(entry)));
|
2023-05-01 18:01:05 +00:00
|
|
|
}
|
2023-05-02 08:50:39 +00:00
|
|
|
else
|
|
|
|
return err(ENOENT);
|
2023-03-28 23:07:58 +00:00
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
if (should_sort)
|
|
|
|
{
|
|
|
|
if (reverse_sort) sort(files.begin(), files.end(), sort_reverse);
|
|
|
|
else
|
|
|
|
sort(files.begin(), files.end(), sort_function);
|
|
|
|
}
|
|
|
|
|
2023-05-01 18:01:05 +00:00
|
|
|
if (!long_list)
|
|
|
|
{
|
2023-05-28 19:51:25 +00:00
|
|
|
auto list = TRY(entry_join(files, one_per_line ? "\n"_sv : " "_sv));
|
2023-05-02 09:00:28 +00:00
|
|
|
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-28 19:51:25 +00:00
|
|
|
TRY(os::FileSystem::stat({ dirfd, file.name.view() }, st, false));
|
2023-05-12 21:47:20 +00:00
|
|
|
|
2023-05-28 19:51:25 +00:00
|
|
|
auto link = TRY(os::FileSystem::readlink({ dirfd, file.name.view() }));
|
2023-05-23 13:42:38 +00:00
|
|
|
|
2023-05-12 21:47:20 +00:00
|
|
|
StringView owner;
|
|
|
|
StringView group;
|
|
|
|
|
2023-05-20 20:10:01 +00:00
|
|
|
find_user_and_group(st, owner, group);
|
2023-04-28 20:41:44 +00:00
|
|
|
|
2023-05-26 16:22:50 +00:00
|
|
|
char formatted_mode[11];
|
|
|
|
os::format_mode(st.st_mode, formatted_mode);
|
|
|
|
|
2023-05-13 10:01:09 +00:00
|
|
|
if (!human_readable && !si)
|
2023-05-13 09:15:28 +00:00
|
|
|
{
|
2023-05-26 16:22:50 +00:00
|
|
|
os::println("%s %u %4s %4s %10lu %s%s%s", formatted_mode, st.st_nlink, owner.chars(), group.chars(),
|
2023-05-28 19:51:25 +00:00
|
|
|
st.st_size, file.name.chars(), link.is_empty() ? "" : " -> ", link.chars());
|
2023-05-13 09:15:28 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-05-13 10:01:09 +00:00
|
|
|
auto size = TRY(to_dynamic_unit(st.st_size, 10, false, si ? Unit::SI : Unit::Binary, false));
|
2023-05-26 16:22:50 +00:00
|
|
|
os::println("%s %u %4s %4s %6s %s%s%s", formatted_mode, st.st_nlink, owner.chars(), group.chars(),
|
2023-05-28 19:51:25 +00:00
|
|
|
size.chars(), file.name.chars(), link.is_empty() ? "" : " -> ", link.chars());
|
2023-05-13 09:15:28 +00:00
|
|
|
}
|
2023-05-01 18:01:05 +00:00
|
|
|
}
|
|
|
|
}
|
2023-03-28 23:07:58 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|