2017-06-15 12:19:59 +08:00
|
|
|
// Implementation of the realpath builtin.
|
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2019-10-14 06:50:48 +08:00
|
|
|
#include "builtin_realpath.h"
|
|
|
|
|
2017-06-15 12:19:59 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2019-10-14 06:50:48 +08:00
|
|
|
|
2019-03-13 06:07:07 +08:00
|
|
|
#include <cstring>
|
2019-03-13 05:06:01 +08:00
|
|
|
#include <cwchar>
|
2017-06-15 12:19:59 +08:00
|
|
|
|
|
|
|
#include "builtin.h"
|
|
|
|
#include "common.h"
|
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
|
|
|
#include "io.h"
|
|
|
|
#include "wutil.h" // IWYU pragma: keep
|
|
|
|
|
2019-10-22 08:21:40 +08:00
|
|
|
/// An implementation of the external realpath command. Doesn't support any options.
|
2018-11-19 04:23:36 +08:00
|
|
|
/// In general scripts shouldn't invoke this directly. They should just use `realpath` which
|
|
|
|
/// will fallback to this builtin if an external command cannot be found.
|
2017-06-15 12:19:59 +08:00
|
|
|
int builtin_realpath(parser_t &parser, io_streams_t &streams, wchar_t **argv) {
|
|
|
|
const wchar_t *cmd = argv[0];
|
2017-06-17 11:42:33 +08:00
|
|
|
help_only_cmd_opts_t opts;
|
2018-11-19 04:23:36 +08:00
|
|
|
int argc = builtin_count_args(argv);
|
2017-06-15 12:19:59 +08:00
|
|
|
int optind;
|
2017-06-17 11:42:33 +08:00
|
|
|
int retval = parse_help_only_cmd_opts(opts, &optind, argc, argv, parser, streams);
|
2018-11-19 04:23:36 +08:00
|
|
|
|
2017-06-15 12:19:59 +08:00
|
|
|
if (retval != STATUS_CMD_OK) return retval;
|
|
|
|
|
|
|
|
if (opts.print_help) {
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.out);
|
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|
|
|
|
|
2019-05-05 18:09:25 +08:00
|
|
|
if (optind + 1 != argc) { // TODO: allow arbitrary args. `realpath *` should print many paths
|
2017-06-15 12:19:59 +08:00
|
|
|
streams.err.append_format(BUILTIN_ERR_ARG_COUNT1, cmd, 1, argc - optind);
|
|
|
|
builtin_print_help(parser, streams, cmd, streams.out);
|
|
|
|
return STATUS_INVALID_ARGS;
|
|
|
|
}
|
|
|
|
|
2017-10-11 15:08:26 +08:00
|
|
|
if (auto real_path = wrealpath(argv[optind])) {
|
|
|
|
streams.out.append(*real_path);
|
2017-06-15 12:19:59 +08:00
|
|
|
} else {
|
2018-11-19 04:23:36 +08:00
|
|
|
if (errno) {
|
2018-11-19 00:56:24 +08:00
|
|
|
// realpath() just couldn't do it. Report the error and make it clear
|
|
|
|
// this is an error from our builtin, not the system's realpath.
|
|
|
|
streams.err.append_format(L"builtin %ls: %ls: %s\n", cmd, argv[optind],
|
2019-03-13 06:07:07 +08:00
|
|
|
std::strerror(errno));
|
2018-11-19 04:23:36 +08:00
|
|
|
} else {
|
2018-11-19 00:56:24 +08:00
|
|
|
// Who knows. Probably a bug in our wrealpath() implementation.
|
|
|
|
streams.err.append_format(_(L"builtin %ls: Invalid path: %ls\n"), cmd, argv[optind]);
|
2018-11-19 04:23:36 +08:00
|
|
|
}
|
2018-11-19 00:56:24 +08:00
|
|
|
|
2017-06-15 12:19:59 +08:00
|
|
|
return STATUS_CMD_ERROR;
|
|
|
|
}
|
2018-11-19 00:56:24 +08:00
|
|
|
|
2017-06-15 12:19:59 +08:00
|
|
|
streams.out.append(L"\n");
|
2018-11-19 00:56:24 +08:00
|
|
|
|
2017-06-15 12:19:59 +08:00
|
|
|
return STATUS_CMD_OK;
|
|
|
|
}
|