2020-01-18 08:38:21 +00:00
|
|
|
/*
|
2020-01-24 13:45:29 +00:00
|
|
|
* Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
|
2020-01-18 08:38:21 +00:00
|
|
|
*
|
2021-04-22 08:24:48 +00:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 08:38:21 +00:00
|
|
|
*/
|
|
|
|
|
2021-05-15 10:34:40 +00:00
|
|
|
#include <AK/Assertions.h>
|
2020-02-14 21:29:06 +00:00
|
|
|
#include <AK/ByteBuffer.h>
|
2022-12-04 18:02:33 +00:00
|
|
|
#include <AK/DeprecatedString.h>
|
2020-02-18 08:54:29 +00:00
|
|
|
#include <LibCore/ArgsParser.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>
|
2020-04-28 19:04:25 +00:00
|
|
|
#include <LibMarkdown/Document.h>
|
2021-07-06 07:43:17 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <spawn.h>
|
2019-09-20 21:46:47 +00:00
|
|
|
#include <stdio.h>
|
2020-10-21 14:36:36 +00:00
|
|
|
#include <sys/ioctl.h>
|
2021-07-06 07:43:17 +00:00
|
|
|
#include <sys/wait.h>
|
2019-09-20 21:46:47 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
static ErrorOr<pid_t> pipe_to_pager(DeprecatedString const& command)
|
2021-07-06 07:43:17 +00:00
|
|
|
{
|
|
|
|
char const* argv[] = { "sh", "-c", command.characters(), nullptr };
|
|
|
|
|
2021-11-24 21:36:36 +00:00
|
|
|
auto stdout_pipe = TRY(Core::System::pipe2(O_CLOEXEC));
|
|
|
|
|
2021-07-06 07:43:17 +00:00
|
|
|
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;
|
|
|
|
if ((errno = posix_spawnp(&pid, argv[0], &action, nullptr, const_cast<char**>(argv), environ))) {
|
|
|
|
perror("posix_spawn");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
posix_spawn_file_actions_destroy(&action);
|
|
|
|
|
2021-11-24 21:36:36 +00:00
|
|
|
TRY(Core::System::dup2(stdout_pipe[1], STDOUT_FILENO));
|
|
|
|
TRY(Core::System::close(stdout_pipe[1]));
|
|
|
|
TRY(Core::System::close(stdout_pipe[0]));
|
2021-07-06 07:43:17 +00:00
|
|
|
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
|
|
|
{
|
2020-10-21 14:36:36 +00:00
|
|
|
int view_width = 0;
|
|
|
|
if (isatty(STDOUT_FILENO)) {
|
|
|
|
struct winsize ws;
|
|
|
|
if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) == 0)
|
|
|
|
view_width = ws.ws_col;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (view_width == 0)
|
|
|
|
view_width = 80;
|
|
|
|
|
2021-11-27 22:26:34 +00:00
|
|
|
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
|
|
|
|
2022-07-11 20:42:03 +00:00
|
|
|
StringView section {};
|
|
|
|
StringView name {};
|
|
|
|
StringView pager {};
|
2019-09-20 21:46:47 +00:00
|
|
|
|
2020-02-18 08:54:29 +00:00
|
|
|
Core::ArgsParser args_parser;
|
2020-12-05 15:22:58 +00:00
|
|
|
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");
|
2021-07-06 07:43:17 +00:00
|
|
|
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
|
|
|
|
2022-07-11 20:42:03 +00:00
|
|
|
auto make_path = [name](StringView section) {
|
2022-12-04 18:02:33 +00:00
|
|
|
return DeprecatedString::formatted("/usr/share/man/man{}/{}.md", section, name);
|
2019-09-20 21:46:47 +00:00
|
|
|
};
|
2022-07-11 20:42:03 +00:00
|
|
|
if (section.is_empty()) {
|
|
|
|
constexpr StringView sections[] = {
|
|
|
|
"1"sv,
|
|
|
|
"2"sv,
|
|
|
|
"3"sv,
|
|
|
|
"4"sv,
|
|
|
|
"5"sv,
|
|
|
|
"6"sv,
|
|
|
|
"7"sv,
|
|
|
|
"8"sv
|
2019-09-20 21:46:47 +00:00
|
|
|
};
|
2020-02-18 08:54:29 +00:00
|
|
|
for (auto s : sections) {
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString path = make_path(s);
|
2019-09-20 21:46:47 +00:00
|
|
|
if (access(path.characters(), R_OK) == 0) {
|
|
|
|
section = s;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-11-24 21:13:53 +00:00
|
|
|
auto filename = make_path(section);
|
2022-01-07 18:46:21 +00:00
|
|
|
if (section == nullptr) {
|
|
|
|
warnln("No man page for {}", name);
|
|
|
|
exit(1);
|
|
|
|
} else if (access(filename.characters(), R_OK) != 0) {
|
|
|
|
warnln("No man page for {} in section {}", name, section);
|
|
|
|
exit(1);
|
|
|
|
}
|
2019-09-20 21:46:47 +00:00
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString pager_command;
|
2022-07-11 20:42:03 +00:00
|
|
|
if (!pager.is_empty())
|
2021-09-10 23:15:44 +00:00
|
|
|
pager_command = pager;
|
|
|
|
else
|
2022-12-04 18:02:33 +00:00
|
|
|
pager_command = DeprecatedString::formatted("less -P 'Manual Page {}({}) line %l?e (END):.'", StringView(name).replace("'"sv, "'\\''"sv, ReplaceMode::FirstOnly), StringView(section).replace("'"sv, "'\\''"sv, ReplaceMode::FirstOnly));
|
2021-11-24 21:36:36 +00:00
|
|
|
pid_t pager_pid = TRY(pipe_to_pager(pager_command));
|
2021-07-06 07:43:17 +00:00
|
|
|
|
2022-09-14 15:12:56 +00:00
|
|
|
auto file = TRY(Core::Stream::File::open(filename, Core::Stream::OpenMode::Read));
|
2020-01-13 00:32:01 +00:00
|
|
|
|
2021-11-27 22:26:34 +00:00
|
|
|
TRY(Core::System::pledge("stdio proc"));
|
2020-01-13 00:32:01 +00:00
|
|
|
|
2022-09-14 15:12:56 +00:00
|
|
|
dbgln("Loading man page from {}", filename);
|
|
|
|
auto buffer = TRY(file->read_all());
|
2022-12-04 18:02:33 +00:00
|
|
|
auto source = DeprecatedString::copy(buffer);
|
2019-09-20 21:46:47 +00:00
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
const DeprecatedString title("SerenityOS manual");
|
2022-02-16 21:10:44 +00:00
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
int spaces = view_width / 2 - DeprecatedString(name).length() - DeprecatedString(section).length() - title.length() / 2 - 4;
|
2022-02-16 21:10:44 +00:00
|
|
|
if (spaces < 0)
|
|
|
|
spaces = 0;
|
|
|
|
out("{}({})", name, section);
|
|
|
|
while (spaces--)
|
|
|
|
out(" ");
|
|
|
|
outln(title);
|
2019-09-20 21:46:47 +00:00
|
|
|
|
2020-05-11 17:55:31 +00:00
|
|
|
auto document = Markdown::Document::parse(source);
|
2021-02-23 19:42:32 +00:00
|
|
|
VERIFY(document);
|
2019-09-20 21:46:47 +00:00
|
|
|
|
2022-12-04 18:02:33 +00:00
|
|
|
DeprecatedString rendered = document->render_for_terminal(view_width);
|
2022-02-16 21:10:44 +00:00
|
|
|
outln("{}", rendered);
|
2021-07-06 07:43:17 +00:00
|
|
|
|
|
|
|
// 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
|
|
|
}
|