ladybird/Userland/Utilities/man.cpp

131 lines
4.5 KiB
C++
Raw Normal View History

/*
* Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/Assertions.h>
#include <AK/ByteBuffer.h>
#include <AK/DeprecatedString.h>
2020-02-18 08:54:29 +00:00
#include <LibCore/ArgsParser.h>
#include <LibCore/File.h>
2022-09-14 15:12:56 +00:00
#include <LibCore/Stream.h>
2021-11-24 21:13:53 +00:00
#include <LibCore/System.h>
#include <LibMain/Main.h>
#include <LibManual/PageNode.h>
#include <LibManual/SectionNode.h>
#include <LibMarkdown/Document.h>
#include <spawn.h>
#include <sys/ioctl.h>
#include <sys/wait.h>
2019-09-20 21:46:47 +00:00
#include <unistd.h>
static ErrorOr<pid_t> pipe_to_pager(DeprecatedString const& command)
{
char const* argv[] = { "sh", "-c", command.characters(), nullptr };
auto stdout_pipe = TRY(Core::System::pipe2(O_CLOEXEC));
posix_spawn_file_actions_t action;
posix_spawn_file_actions_init(&action);
posix_spawn_file_actions_adddup2(&action, stdout_pipe[0], STDIN_FILENO);
pid_t pid = TRY(Core::System::posix_spawnp("sh"sv, &action, nullptr, const_cast<char**>(argv), environ));
posix_spawn_file_actions_destroy(&action);
TRY(Core::System::dup2(stdout_pipe[1], STDOUT_FILENO));
TRY(Core::System::close(stdout_pipe[1]));
TRY(Core::System::close(stdout_pipe[0]));
return pid;
}
2021-11-24 21:13:53 +00:00
ErrorOr<int> serenity_main(Main::Arguments arguments)
2019-09-20 21:46:47 +00:00
{
int view_width = 0;
if (isatty(STDOUT_FILENO) != 0) {
struct winsize ws;
if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) == 0)
view_width = ws.ws_col;
}
if (view_width == 0)
view_width = 80;
TRY(Core::System::pledge("stdio rpath exec proc"));
2021-11-24 21:13:53 +00:00
TRY(Core::System::unveil("/usr/share/man", "r"));
TRY(Core::System::unveil("/bin", "x"));
TRY(Core::System::unveil(nullptr, nullptr));
2020-01-22 10:36:22 +00:00
DeprecatedString section;
DeprecatedString name;
DeprecatedString pager;
2019-09-20 21:46:47 +00:00
2020-02-18 08:54:29 +00:00
Core::ArgsParser args_parser;
args_parser.set_general_help("Read manual pages. Try 'man man' to get started.");
2020-02-18 08:54:29 +00:00
args_parser.add_positional_argument(section, "Section of the man page", "section", Core::ArgsParser::Required::No);
args_parser.add_positional_argument(name, "Name of the man page", "name");
args_parser.add_option(pager, "Pager to pipe the man page to", "pager", 'P', "pager");
2021-11-24 21:13:53 +00:00
args_parser.parse(arguments);
2019-09-20 21:46:47 +00:00
Optional<NonnullRefPtr<Manual::PageNode>> page;
if (section.is_empty()) {
for (auto const& s : Manual::sections) {
auto const maybe_page = make_ref_counted<Manual::PageNode>(s, TRY(String::from_utf8(name)));
if (Core::File::exists(TRY(maybe_page->path()).to_deprecated_string())) {
page = maybe_page;
section = s->section_name().to_deprecated_string();
2019-09-20 21:46:47 +00:00
break;
}
}
} else {
auto number_section = section.to_uint();
if (number_section.has_value())
page = make_ref_counted<Manual::PageNode>(Manual::sections[number_section.value() - 1], TRY(String::from_utf8(name)));
else
warnln("Section name '{}' invalid", section);
2019-09-20 21:46:47 +00:00
}
if (!page.has_value()) {
warnln("No man page for {}", name);
exit(1);
} else if (!Core::File::exists(TRY((*page)->path()))) {
warnln("No man page for {} in section {}", name, section);
exit(1);
}
2019-09-20 21:46:47 +00:00
if (pager.is_empty())
pager = TRY(String::formatted("less -P 'Manual Page {}({}) line %l?e (END):.'", StringView(name).replace("'"sv, "'\\''"sv, ReplaceMode::FirstOnly), StringView(section).replace("'"sv, "'\\''"sv, ReplaceMode::FirstOnly))).to_deprecated_string();
pid_t pager_pid = TRY(pipe_to_pager(pager));
auto file = TRY(Core::Stream::File::open(TRY((*page)->path()), Core::Stream::OpenMode::Read));
2020-01-13 00:32:01 +00:00
TRY(Core::System::pledge("stdio proc"));
2020-01-13 00:32:01 +00:00
dbgln("Loading man page from {}", (*page)->path());
2022-09-14 15:12:56 +00:00
auto buffer = TRY(file->read_all());
auto source = DeprecatedString::copy(buffer);
2019-09-20 21:46:47 +00:00
const DeprecatedString title("SerenityOS manual");
int spaces = view_width / 2 - DeprecatedString(name).length() - DeprecatedString(section).length() - title.length() / 2 - 4;
if (spaces < 0)
spaces = 0;
out("{}({})", name, section);
while (spaces--)
out(" ");
outln(title);
2019-09-20 21:46:47 +00:00
auto document = Markdown::Document::parse(source);
VERIFY(document);
2019-09-20 21:46:47 +00:00
DeprecatedString rendered = document->render_for_terminal(view_width);
outln("{}", rendered);
// FIXME: Remove this wait, it shouldn't be necessary but Shell does not
// resume properly without it. This wait also breaks <C-z> backgrounding
fclose(stdout);
int wstatus;
waitpid(pager_pid, &wstatus, 0);
2021-11-24 21:13:53 +00:00
return 0;
2019-09-20 21:46:47 +00:00
}