/* apk.c - Alpine Package Keeper (APK)
*
* Copyright (C) 2005-2008 Natanael Copa <n@tanael.org>
* Copyright (C) 2008-2011 Timo Teräs <timo.teras@iki.fi>
* All rights reserved.
*
* SPDX-License-Identifier: GPL-2.0-only
*/
#include <stdio.h>
#include <fcntl.h>
#include <ctype.h>
#include <errno.h>
#include <assert.h>
#include <signal.h>
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
#include <getopt.h>
#include <unistd.h>
#include <sys/stat.h>
#include <openssl/crypto.h>
#ifndef OPENSSL_NO_ENGINE
#include <openssl/engine.h>
#endif
#include <fetch.h>
#include "apk_defines.h"
#include "apk_database.h"
#include "apk_applet.h"
#include "apk_blob.h"
#include "apk_print.h"
#include "apk_io.h"
static struct list_head apk_applet_list;
#define foreach_applet(iter) list_for_each_entry(iter, &apk_applet_list, node)
#ifdef TEST_MODE
static const char *test_installed_db = NULL;
static const char *test_world = NULL;
static struct apk_string_array *test_repos;
#endif
char **apk_argv;
#ifdef TEST_MODE
time_t time(time_t *tloc)
{
const time_t val = 1559567666;
if (tloc) *tloc = val;
return val;
}
#endif
static void version(void)
{
printf("apk-tools " APK_VERSION ", compiled for " APK_DEFAULT_ARCH ".\n"
#ifdef TEST_MODE
"TEST MODE BUILD. NOT FOR PRODUCTION USE.\n"
#endif
);
}
static struct apk_repository_list *apk_repository_new(const char *url)
{
struct apk_repository_list *r = calloc(1, sizeof(struct apk_repository_list));
if (r) {
r->url = url;
list_init(&r->list);
}
return r;
}
enum {
OPT_GLOBAL_allow_untrusted,
OPT_GLOBAL_arch,
OPT_GLOBAL_cache_dir,
OPT_GLOBAL_cache_max_age,
OPT_GLOBAL_force,
OPT_GLOBAL_force_binary_stdout,
OPT_GLOBAL_force_broken_world,
OPT_GLOBAL_force_non_repository,
OPT_GLOBAL_force_old_apk,
OPT_GLOBAL_force_overwrite,
OPT_GLOBAL_force_refresh,
OPT_GLOBAL_help,
OPT_GLOBAL_interactive,
OPT_GLOBAL_keys_dir,
OPT_GLOBAL_no_cache,
OPT_GLOBAL_no_network,
OPT_GLOBAL_no_progress,
OPT_GLOBAL_print_arch,
OPT_GLOBAL_progress,
OPT_GLOBAL_progress_fd,
OPT_GLOBAL_purge,
OPT_GLOBAL_quiet,
OPT_GLOBAL_repositories_file,
OPT_GLOBAL_repository,
OPT_GLOBAL_root,
OPT_GLOBAL_update_cache,
OPT_GLOBAL_verbose,
OPT_GLOBAL_version,
OPT_GLOBAL_wait,
#ifdef TEST_MODE
OPT_GLOBAL_test_instdb,
OPT_GLOBAL_test_repo,
OPT_GLOBAL_test_world,
#endif
};
static const char optiondesc_global[] =
APK_OPTGROUP("Global")
APK_OPT1n("allow-untrusted")
APK_OPT1R("arch")
APK_OPT1R("cache-dir")
APK_OPT1R("cache-max-age")
APK_OPT2n("force", "f")
APK_OPT1n("force-binary-stdout")
APK_OPT1n("force-broken-world")
APK_OPT1n("force-non-repository")
APK_OPT1n("force-old-apk")
APK_OPT1n("force-overwrite")
APK_OPT1n("force-refresh")
APK_OPT2n("help", "h")
APK_OPT2n("interactive", "i")
APK_OPT1R("keys-dir")
APK_OPT1n("no-cache")
APK_OPT1n("no-network")
APK_OPT1n("no-progress")
APK_OPT1n("print-arch")
APK_OPT1n("progress")
APK_OPT1R("progress-fd")
APK_OPT1n("purge")
APK_OPT2n("quiet", "q")
APK_OPT1R("repositories-file")
APK_OPT2R("repository", "X")
APK_OPT2R("root", "p")
APK_OPT2n("update-cache", "U")
APK_OPT2n("verbose", "v")
APK_OPT2n("version", "V")
APK_OPT1R("wait")
#ifdef TEST_MODE
APK_OPT1R("test-instdb")
APK_OPT1R("test-repo")
APK_OPT1R("test-world")
#endif
;
static int option_parse_global(void *ctx, struct apk_db_options *dbopts, int opt, const char *optarg)
{
struct apk_repository_list *repo;
switch (opt) {
case OPT_GLOBAL_help:
return -EINVAL;
case OPT_GLOBAL_root:
dbopts->root = optarg;
break;
case OPT_GLOBAL_keys_dir:
dbopts->keys_dir = optarg;
break;
case OPT_GLOBAL_repositories_file:
dbopts->repositories_file = optarg;
break;
case OPT_GLOBAL_repository:
repo = apk_repository_new(optarg);
if (repo) list_add(&repo->list, &dbopts->repository_list);
break;
case OPT_GLOBAL_quiet:
apk_verbosity--;
break;
case OPT_GLOBAL_verbose:
apk_verbosity++;
break;
case OPT_GLOBAL_version:
version();
return -ESHUTDOWN;
case OPT_GLOBAL_force:
apk_force |= APK_FORCE_OVERWRITE | APK_FORCE_OLD_APK
| APK_FORCE_BROKEN_WORLD | APK_FORCE_NON_REPOSITORY
| APK_FORCE_BINARY_STDOUT;
break;
case OPT_GLOBAL_force_overwrite:
apk_force |= APK_FORCE_OVERWRITE;
break;
case OPT_GLOBAL_force_old_apk:
apk_force |= APK_FORCE_OLD_APK;
break;
case OPT_GLOBAL_force_broken_world:
apk_force |= APK_FORCE_BROKEN_WORLD;
break;
case OPT_GLOBAL_force_refresh:
apk_force |= APK_FORCE_REFRESH;
break;
case OPT_GLOBAL_force_non_repository:
apk_force |= APK_FORCE_NON_REPOSITORY;
break;
case OPT_GLOBAL_force_binary_stdout:
apk_force |= APK_FORCE_BINARY_STDOUT;
break;
case OPT_GLOBAL_interactive:
apk_flags |= APK_INTERACTIVE;
break;
case OPT_GLOBAL_progress:
apk_flags |= APK_PROGRESS;
break;
case OPT_GLOBAL_no_progress:
apk_flags &= ~APK_PROGRESS;
break;
case OPT_GLOBAL_progress_fd:
apk_progress_fd = atoi(optarg);
break;
case OPT_GLOBAL_allow_untrusted:
apk_flags |= APK_ALLOW_UNTRUSTED;
break;
case OPT_GLOBAL_purge:
apk_flags |= APK_PURGE;
break;
case OPT_GLOBAL_wait:
dbopts->lock_wait = atoi(optarg);
break;
case OPT_GLOBAL_no_network:
apk_flags |= APK_NO_NETWORK;
break;
case OPT_GLOBAL_no_cache:
apk_flags |= APK_NO_CACHE;
break;
case OPT_GLOBAL_cache_dir:
dbopts->cache_dir = optarg;
break;
case OPT_GLOBAL_update_cache:
/* Make it one minute, to avoid updating indexes twice
* when doing self-upgrade's re-exec */
dbopts->cache_max_age = 60;
break;
case OPT_GLOBAL_cache_max_age:
dbopts->cache_max_age = atoi(optarg) * 60;
break;
case OPT_GLOBAL_arch:
dbopts->arch = optarg;
break;
case OPT_GLOBAL_print_arch:
puts(APK_DEFAULT_ARCH);
return -ESHUTDOWN;
#ifdef TEST_MODE
case OPT_GLOBAL_test_repo:
*apk_string_array_add(&test_repos) = (char*) optarg;
break;
case OPT_GLOBAL_test_instdb:
test_installed_db = optarg;
break;
case OPT_GLOBAL_test_world:
test_world = optarg;
break;
#endif
default:
return -ENOTSUP;
}
return 0;
}
const struct apk_option_group optgroup_global = {
.desc = optiondesc_global,
.parse = option_parse_global,
};
enum {
OPT_COMMIT_clean_protected,
OPT_COMMIT_initramfs_diskless_boot,
OPT_COMMIT_no_commit_hooks,
OPT_COMMIT_no_scripts,
OPT_COMMIT_overlay_from_stdin,
OPT_COMMIT_simulate,
};
static const char optiondesc_commit[] =
APK_OPTGROUP("commit")
APK_OPT1n("clean-protected")
APK_OPT1n("initramfs-diskless-boot")
APK_OPT1n("no-commit-hooks")
APK_OPT1n("no-scripts")
APK_OPT1n("overlay-from-stdin")
APK_OPT2n("simulate", "s");
static int option_parse_commit(void *ctx, struct apk_db_options *dbopts, int opt, const char *optarg)
{
switch (opt) {
case OPT_COMMIT_simulate:
apk_flags |= APK_SIMULATE;
break;
case OPT_COMMIT_clean_protected:
apk_flags |= APK_CLEAN_PROTECTED;
break;
case OPT_COMMIT_overlay_from_stdin:
apk_flags |= APK_OVERLAY_FROM_STDIN;
break;
case OPT_COMMIT_no_scripts:
apk_flags |= APK_NO_SCRIPTS;
break;
case OPT_COMMIT_no_commit_hooks:
apk_flags |= APK_NO_COMMIT_HOOKS;
break;
case OPT_COMMIT_initramfs_diskless_boot:
dbopts->open_flags |= APK_OPENF_CREATE;
apk_flags |= APK_NO_COMMIT_HOOKS;
apk_force |= APK_FORCE_OVERWRITE | APK_FORCE_OLD_APK
| APK_FORCE_BROKEN_WORLD | APK_FORCE_NON_REPOSITORY;
break;
default:
return -ENOTSUP;
}
return 0;
}
const struct apk_option_group optgroup_commit = {
.desc = optiondesc_commit,
.parse = option_parse_commit,
};
static int usage(struct apk_applet *applet)
{
version();
apk_help(applet);
return 1;
}
static struct apk_applet *find_applet(const char *name)
{
struct apk_applet *a;
foreach_applet(a) {
if (strcmp(name, a->name) == 0)
return a;
}
return NULL;
}
static struct apk_applet *deduce_applet(int argc, char **argv)
{
struct apk_applet *a;
const char *prog;
int i;
prog = strrchr(argv[0], '/');
if (prog == NULL)
prog = argv[0];
else
prog++;
if (strncmp(prog, "apk_", 4) == 0)
return find_applet(prog + 4);
for (i = 1; i < argc; i++) {
if (argv[i][0] == '-') continue;
a = find_applet(argv[i]);
if (a) return a;
}
return NULL;
}
static int parse_options(int argc, char **argv, struct apk_applet *applet, void *ctx, struct apk_db_options *dbopts)
{
const struct apk_option_group *default_optgroups[] = { &optgroup_global, NULL };
const struct apk_option_group *og, **optgroups = default_optgroups;
struct option all_options[80], *opt;
char short_options[256], *sopt;
unsigned short short_option_val[64];
int r, p, help_requested = 0, num_short;
memset(short_option_val, 0, sizeof short_option_val);
if (applet && applet->optgroups[0]) optgroups = applet->optgroups;
for (p = 0, opt = &all_options[0], sopt = short_options; (og = optgroups[p]) != 0; p++) {
assert(opt < &all_options[ARRAY_SIZE(all_options)]);
assert(sopt < &short_options[sizeof short_options]);
const char *d = og->desc + strlen(og->desc) + 1;
for (r = 0; *d; r++) {
opt->val = (p << 10) + r;
opt->flag = 0;
opt->has_arg = no_argument;
if ((unsigned char)*d == 0xaf) {
opt->has_arg = required_argument;
d++;
}
num_short = 1;
if ((unsigned char)*d >= 0xf0)
num_short = *d++ & 0x0f;
for (; num_short > 0; num_short--) {
assert(*d >= 64 && *d < 128);
short_option_val[*d - 64] = opt->val;
*sopt++ = *d++;
if (opt->has_arg != no_argument)
*sopt++ = ':';
}
opt->name = d;
opt++;
d += strlen(d) + 1;
}
}
opt->name = 0;
*sopt = 0;
r = 0;
while ((p = getopt_long(argc, argv, short_options, all_options, NULL)) != -1) {
if (p >= 64 && p < 128) p = short_option_val[p - 64];
og = optgroups[p >> 10];
r = og->parse(ctx, dbopts, p & 0x3ff, optarg);
if (r == 0) continue;
if (r == -EINVAL) {
help_requested = 1;
continue;
}
if (r != -ENOTSUP) return r;
}
if (help_requested || r == -ENOTSUP)
return usage(applet);
if (applet == NULL) {
if (argc > 1) {
apk_error("'%s' is not an apk command. See 'apk --help'.", argv[1]);
return 1;
}
return usage(NULL);
}
return 0;
}
static void fini_openssl(void)
{
EVP_cleanup();
#ifndef OPENSSL_NO_ENGINE
ENGINE_cleanup();
#endif
CRYPTO_cleanup_all_ex_data();
}
static void init_openssl(void)
{
atexit(fini_openssl);
OpenSSL_add_all_algorithms();
#ifndef OPENSSL_NO_ENGINE
ENGINE_load_builtin_engines();
ENGINE_register_all_complete();
#endif
}
static void on_sigwinch(int s)
{
apk_reset_screen_width();
}
static void setup_terminal(void)
{
signal(SIGWINCH, on_sigwinch);
signal(SIGPIPE, SIG_IGN);
}
static void setup_automatic_flags(void)
{
if (!isatty(STDOUT_FILENO) || !isatty(STDERR_FILENO) ||
!isatty(STDIN_FILENO))
return;
apk_flags |= APK_PROGRESS;
if (!(apk_flags & APK_SIMULATE) &&
access("/etc/apk/interactive", F_OK) == 0)
apk_flags |= APK_INTERACTIVE;
}
void apk_applet_register(struct apk_applet *applet)
{
list_init(&applet->node);
list_add_tail(&applet->node, &apk_applet_list);
}
static void apk_applet_register_builtin(void)
{
extern apk_init_func_t __start_initapplets[], __stop_initapplets[];
apk_init_func_t *p;
list_init(&apk_applet_list);
for (p = __start_initapplets; p < __stop_initapplets; p++)
(*p)();
}
static struct apk_database db;
static void on_sigint(int s)
{
apk_db_close(&db);
exit(128 + s);
}
int main(int argc, char **argv)
{
void *ctx = NULL;
struct apk_db_options dbopts;
struct apk_string_array *args;
struct apk_applet *applet;
int r;
apk_string_array_init(&args);
#ifdef TEST_MODE
apk_string_array_init(&test_repos);
#endif
apk_applet_register_builtin();
apk_argv = malloc(sizeof(char*[argc+2]));
memcpy(apk_argv, argv, sizeof(char*[argc]));
apk_argv[argc] = NULL;
apk_argv[argc+1] = NULL;
memset(&dbopts, 0, sizeof(dbopts));
list_init(&dbopts.repository_list);
umask(0);
setup_terminal();
applet = deduce_applet(argc, argv);
if (applet != NULL) {
if (applet->context_size != 0)
ctx = calloc(1, applet->context_size);
dbopts.open_flags = applet->open_flags;
apk_flags |= applet->forced_flags;
apk_force |= applet->forced_force;
}
init_openssl();
setup_automatic_flags();
fetchConnectionCacheInit(32, 4);
r = parse_options(argc, argv, applet, ctx, &dbopts);
if (r != 0) goto err;
argc -= optind;
argv += optind;
if (argc >= 1 && strcmp(argv[0], applet->name) == 0) {
argc--;
argv++;
}
apk_db_init(&db);
signal(SIGINT, on_sigint);
#ifdef TEST_MODE
dbopts.open_flags &= ~(APK_OPENF_WRITE | APK_OPENF_CACHE_WRITE | APK_OPENF_CREATE);
dbopts.open_flags |= APK_OPENF_READ | APK_OPENF_NO_STATE | APK_OPENF_NO_REPOS;
apk_flags |= APK_SIMULATE;
apk_flags &= ~APK_INTERACTIVE;
#endif
r = apk_db_open(&db, &dbopts);
if (r != 0) {
apk_error("Failed to open apk database: %s",
apk_error_str(r));
goto err;
}
#ifdef TEST_MODE
if (test_world != NULL) {
apk_blob_t b = APK_BLOB_STR(test_world);
apk_blob_pull_deps(&b, &db, &db.world);
}
if (test_installed_db != NULL) {
apk_db_index_read(&db, apk_istream_from_file(AT_FDCWD, test_installed_db), -1);
}
for (int i = 0; i < test_repos->num; i++) {
apk_blob_t spec = APK_BLOB_STR(test_repos->item[i]), name, tag;
int repo_tag = 0, repo = APK_REPOSITORY_FIRST_CONFIGURED + i;
if (spec.ptr[0] == '!') {
/* cache's installed repository */
spec.ptr++;
spec.len--;
repo = -2;
}
if (apk_blob_split(spec, APK_BLOB_STR(":"), &tag, &name)) {
repo_tag = apk_db_get_tag_id(&db, tag);
} else {
name = spec;
}
if (apk_db_index_read(&db, apk_istream_from_file(AT_FDCWD, name.ptr), repo) != 0) {
apk_error("Failed to open repository: " BLOB_FMT, BLOB_PRINTF(name));
goto err;
}
if (repo != -2) {
if (!(apk_flags & APK_NO_NETWORK))
db.available_repos |= BIT(repo);
db.repo_tags[repo_tag].allowed_repos |= BIT(repo);
}
}
#endif
apk_string_array_resize(&args, argc);
memcpy(args->item, argv, argc * sizeof(*argv));
r = applet->main(ctx, &db, args);
apk_db_close(&db);
#ifdef TEST_MODE
/* in test mode, we need to always exit 0 since xargs dies otherwise */
r = 0;
#endif
err:
if (r == -ESHUTDOWN) r = 0;
if (ctx) free(ctx);
fetchConnectionCacheClose();
apk_string_array_free(&args);
free(apk_argv);
if (r < 0) r = 250;
if (r > 99) r = 99;
return r;
}