532 lines
14 KiB
C
532 lines
14 KiB
C
/**
|
|
* @brief Package Management Utility for ToaruOS
|
|
*
|
|
* This is a not-quite-faithful reconstruction of the original
|
|
* Python msk. The supported package format is a bit different,
|
|
* to avoid the need to implement a full JSON parser.
|
|
*
|
|
* Packages can optionally be uncompressed, which is also
|
|
* important for bootstrapping at the moment.
|
|
*
|
|
* @copyright
|
|
* This file is part of ToaruOS and is released under the terms
|
|
* of the NCSA / University of Illinois License - see LICENSE.md
|
|
* Copyright (C) 2018 K. Lange
|
|
*/
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include <toaru/confreader.h>
|
|
#include <toaru/list.h>
|
|
#include <toaru/hashmap.h>
|
|
|
|
#define MSK_VERSION "1.0.0"
|
|
#define VAR_PATH "/var/msk"
|
|
#define LOCK_PATH "/var/run/msk.lock"
|
|
|
|
static confreader_t * msk_config = NULL;
|
|
static confreader_t * msk_manifest = NULL;
|
|
static hashmap_t * msk_installed = NULL;
|
|
static int lock_fd = -1;
|
|
|
|
static int verbose = 0;
|
|
|
|
static void release_lock(void) {
|
|
if (lock_fd != -1) {
|
|
unlink(LOCK_PATH);
|
|
}
|
|
}
|
|
|
|
static void needs_lock(void) {
|
|
if (lock_fd == -1) {
|
|
lock_fd = open(LOCK_PATH, O_RDWR|O_CREAT|O_EXCL);
|
|
if (lock_fd < 0) {
|
|
fprintf(stderr, "msk: failed to obtain exclusive lock\n");
|
|
exit(1);
|
|
}
|
|
atexit(release_lock);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* checks whether 'candidate' is newer than 'current'.
|
|
*
|
|
* Requires version strings to be of the form x.y.z
|
|
*
|
|
* > 0 candidate is newer
|
|
* = 0 candidate is the same
|
|
* < 0 candidate is older
|
|
*/
|
|
static int compare_version_strings(char * current, char * candidate) {
|
|
int current_x, current_y, current_z;
|
|
int candidate_x, candidate_y, candidate_z;
|
|
|
|
sscanf(current, "%d.%d.%d", ¤t_x, ¤t_y, ¤t_z);
|
|
sscanf(candidate, "%d.%d.%d", &candidate_x, &candidate_y, &candidate_z);
|
|
|
|
if (candidate_x > current_x) {
|
|
return 1;
|
|
} else if (candidate_x == current_x) {
|
|
if (candidate_y >current_y) {
|
|
return 1;
|
|
} else if (candidate_y == current_y) {
|
|
if (candidate_z > current_z) {
|
|
return 1;
|
|
} else if (candidate_z == current_z) {
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
static void read_config(void) {
|
|
confreader_t * conf = confreader_load("/etc/msk.conf");
|
|
if (!conf) {
|
|
fprintf(stderr, "failed to read configuration file\n");
|
|
exit(1);
|
|
}
|
|
|
|
if (!strcmp(confreader_getd(conf, "", "verbose",""), "y")) {
|
|
verbose = 1;
|
|
}
|
|
|
|
msk_config = conf;
|
|
}
|
|
|
|
static void read_manifest(int required) {
|
|
confreader_t * conf = confreader_load(VAR_PATH "/manifest");
|
|
if (!conf) {
|
|
if (required) {
|
|
fprintf(stderr, "no manifest; try `msk update` first\n");
|
|
exit(1);
|
|
} else {
|
|
conf = confreader_create_empty();
|
|
}
|
|
}
|
|
|
|
msk_manifest = conf;
|
|
}
|
|
|
|
static void read_installed(void) {
|
|
msk_installed = hashmap_create(10);
|
|
|
|
FILE * installed = fopen(VAR_PATH "/installed", "r");
|
|
if (!installed) return;
|
|
|
|
while (!feof(installed)) {
|
|
char tmp[128] = {0};
|
|
if (!fgets(tmp, 128, installed)) break;
|
|
char * nl = strstr(tmp, "\n");
|
|
if (nl) *nl = '\0';
|
|
|
|
char * eqeq = strstr(tmp, "==");
|
|
if (!eqeq) {
|
|
fprintf(stderr, "Installation cache is malformed\n");
|
|
fprintf(stderr, "line was: [%s]\n", tmp);
|
|
exit(1);
|
|
}
|
|
|
|
*eqeq = '\0';
|
|
char * version = eqeq+2;
|
|
|
|
hashmap_set(msk_installed, tmp, strdup(version));
|
|
}
|
|
}
|
|
|
|
static void make_var(void) {
|
|
struct stat buf;
|
|
if (stat(VAR_PATH, &buf)) {
|
|
mkdir(VAR_PATH, 0755);
|
|
}
|
|
}
|
|
|
|
static void needs_root(void) {
|
|
if (geteuid() != 0) {
|
|
fprintf(stderr, "only root can install packages; try `sudo`\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
static int usage(int argc, char * argv[]) {
|
|
#define _IT "\033[3m"
|
|
#define _END "\033[0m\n"
|
|
fprintf(stderr,
|
|
"%s - package manager " MSK_VERSION "\n"
|
|
"\n"
|
|
"usage: %s update\n"
|
|
" %s install [PACKAGE...]\n"
|
|
"\n"
|
|
" update " _IT "update local manifest from remote" _END
|
|
" install " _IT "install packages" _END
|
|
"\n", argv[0], argv[0], argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
static int update_stores(int argc, char * argv[]) {
|
|
needs_root();
|
|
if (argc > 2) {
|
|
fprintf(stderr,"%s: %s: unexpected arguments in command\n", argv[0], argv[1]);
|
|
return usage(argc,argv);
|
|
}
|
|
|
|
needs_lock();
|
|
|
|
read_config();
|
|
make_var();
|
|
|
|
confreader_t * manifest_out = confreader_create_empty();
|
|
hashmap_t * remotes = hashmap_get(msk_config->sections, "remotes");
|
|
|
|
int one_success = 0;
|
|
|
|
char * order = strdup(confreader_getd(msk_config, "", "remote_order", ""));
|
|
char * save;
|
|
char * tok = strtok_r(order, ",", &save);
|
|
do {
|
|
char * remote_name = strdup(tok);
|
|
char * remote_path = hashmap_get(remotes, remote_name);
|
|
if (!remote_path) {
|
|
fprintf(stderr, "Undefined remote specified in remote_order: %s\n", remote_name);
|
|
goto _next;
|
|
}
|
|
|
|
confreader_t * manifest;
|
|
|
|
if (remote_path[0] == '/') {
|
|
char source[512];
|
|
sprintf(source, "%s/manifest", remote_path);
|
|
manifest = confreader_load(source);
|
|
if (!manifest) {
|
|
fprintf(stderr, "Skipping unavailable local manifest '%s'.\n", remote_name);
|
|
goto _next;
|
|
}
|
|
} else {
|
|
char cmd[512];
|
|
sprintf(cmd, "fetch -vo /tmp/.msk_remote_%s %s/manifest", remote_name, remote_path);
|
|
fprintf(stderr, "Downloading remote manifest '%s'...\n", remote_name);
|
|
if (system(cmd)) {
|
|
fprintf(stderr, "Skipping unavailable remote manifest '%s' (%s).\n", remote_name, remote_path);
|
|
goto _next;
|
|
}
|
|
sprintf(cmd, "/tmp/.msk_remote_%s", remote_name);
|
|
manifest = confreader_load(cmd);
|
|
}
|
|
|
|
list_t * packages = hashmap_keys(manifest->sections);
|
|
foreach(nnode, packages) {
|
|
char * package_name = (char*)nnode->value;
|
|
hashmap_t * package_data = (hashmap_t*)hashmap_get(manifest->sections, package_name);
|
|
if (!strcmp(package_name,"")) continue; /* skip intro section - remote repo information */
|
|
|
|
hashmap_set(package_data, "remote_path", remote_path);
|
|
hashmap_set(package_data, "remote_name", remote_name);
|
|
|
|
if (!hashmap_has(manifest_out->sections, package_name)) {
|
|
/* Package not yet known */
|
|
hashmap_set(manifest_out->sections, package_name, package_data);
|
|
} else {
|
|
/* Package is known, keep the newer version */
|
|
char * old_version = confreader_get(manifest_out, package_name, "version");
|
|
char * new_version = confreader_get(manifest, package_name, "version");
|
|
|
|
if (compare_version_strings(old_version, new_version) > 0) {
|
|
hashmap_set(manifest_out->sections, package_name, package_data);
|
|
}
|
|
}
|
|
}
|
|
|
|
one_success = 1;
|
|
|
|
_next:
|
|
tok = strtok_r(NULL, ",", &save);
|
|
} while (tok);
|
|
free(order);
|
|
|
|
if (!one_success) {
|
|
fprintf(stderr, "\033[1;31merror\033[0m: no remote succeeded, no packages are available\n");
|
|
return 1;
|
|
}
|
|
|
|
return confreader_write(manifest_out, VAR_PATH "/manifest");
|
|
}
|
|
|
|
static int list_contains(list_t * list, char * key) {
|
|
foreach(node, list) {
|
|
char * v = node->value;
|
|
if (!strcmp(v,key)) return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int process_package(list_t * pkgs, char * name) {
|
|
if (hashmap_has(msk_installed, name)) return 0;
|
|
if (list_contains(pkgs, name)) return 0;
|
|
|
|
if (!hashmap_has(msk_manifest->sections, name)) {
|
|
fprintf(stderr, "don't know how to install '%s'\n", name);
|
|
return 1;
|
|
}
|
|
|
|
/* Gather dependencies */
|
|
char * tmp = confreader_get(msk_manifest, name, "dependencies");
|
|
if (strlen(tmp)) {
|
|
char * deps = strdup(tmp);
|
|
char * save;
|
|
char * tok = strtok_r(deps, " ", &save);
|
|
do {
|
|
process_package(pkgs, tok);
|
|
tok = strtok_r(NULL, " ", &save);
|
|
} while (tok);
|
|
free(deps);
|
|
}
|
|
|
|
/* Insert */
|
|
list_insert(pkgs, strdup(name));
|
|
return 0;
|
|
}
|
|
|
|
static int install_package(char * pkg) {
|
|
|
|
char * type = confreader_getd(msk_manifest, pkg, "type", "");
|
|
char * msk_remote = confreader_get(msk_manifest, pkg, "remote_path");
|
|
|
|
if (strstr(msk_remote, "http:") == msk_remote || strstr(msk_remote, "https:") == msk_remote) {
|
|
char * source = confreader_get(msk_manifest, pkg, "source");
|
|
if (source) {
|
|
fprintf(stderr, "Download %s...\n", pkg);
|
|
char cmd[1024];
|
|
sprintf(cmd, "fetch -o /tmp/msk.file -v %s/%s", msk_remote,
|
|
source);
|
|
int status;
|
|
if ((status = system(cmd))) {
|
|
return status;
|
|
}
|
|
hashmap_set(hashmap_get(msk_manifest->sections, pkg), "source", "/tmp/msk.file");
|
|
}
|
|
} else if (msk_remote[0] == '/') {
|
|
char * source = confreader_get(msk_manifest, pkg, "source");
|
|
if (source) {
|
|
char * pkg_name = malloc(strlen(msk_remote) + strlen(source) + 2);
|
|
sprintf(pkg_name, "%s/%s", msk_remote, source);
|
|
hashmap_set(hashmap_get(msk_manifest->sections, pkg), "source", pkg_name);
|
|
}
|
|
}
|
|
|
|
fprintf(stderr, "Install '%s'...\n", pkg);
|
|
|
|
if (!strcmp(type, "file")) {
|
|
/* Legacy single-file package, has a source and a destination */
|
|
|
|
if (verbose) {
|
|
fprintf(stderr, " - Copy file '%s' to '%s' and set its mask to '%s'\n",
|
|
confreader_get(msk_manifest, pkg, "source"),
|
|
confreader_get(msk_manifest, pkg, "destination"),
|
|
confreader_get(msk_manifest, pkg, "mask"));
|
|
}
|
|
|
|
char cmd[1024];
|
|
sprintf(cmd, "cp %s %s; chmod 0%s %s",
|
|
confreader_get(msk_manifest, pkg, "source"),
|
|
confreader_get(msk_manifest, pkg, "destination"),
|
|
confreader_get(msk_manifest, pkg, "mask"),
|
|
confreader_get(msk_manifest, pkg, "destination"));
|
|
|
|
int status;
|
|
if ((status = system(cmd))) {
|
|
fprintf(stderr, "installation command returned %d\n", status);
|
|
return status;
|
|
}
|
|
|
|
} else if (!strcmp(type, "tar")) {
|
|
/* Uncompressed archive */
|
|
|
|
if (verbose) {
|
|
fprintf(stderr, " - Extract '%s' to '%s'\n",
|
|
confreader_get(msk_manifest, pkg, "source"),
|
|
confreader_get(msk_manifest, pkg, "destination"));
|
|
}
|
|
|
|
char cmd[1024];
|
|
sprintf(cmd, "cd %s; tar -xf %s",
|
|
confreader_get(msk_manifest, pkg, "destination"),
|
|
confreader_get(msk_manifest, pkg, "source"));
|
|
|
|
int status;
|
|
if ((status = system(cmd))) {
|
|
fprintf(stderr, "installation command returned %d\n", status);
|
|
return status;
|
|
}
|
|
|
|
} else if (!strcmp(type, "tgz")) {
|
|
/* Compressed archive */
|
|
|
|
if (verbose) {
|
|
fprintf(stderr, " - Extract (compressed) '%s' to '%s'\n",
|
|
confreader_get(msk_manifest, pkg, "source"),
|
|
confreader_get(msk_manifest, pkg, "destination"));
|
|
}
|
|
|
|
char cmd[1024];
|
|
sprintf(cmd, "cd %s; tar -xzf %s",
|
|
confreader_get(msk_manifest, pkg, "destination"),
|
|
confreader_get(msk_manifest, pkg, "source"));
|
|
|
|
int status;
|
|
if ((status = system(cmd))) {
|
|
fprintf(stderr, "installation command returned %d\n", status);
|
|
return status;
|
|
}
|
|
|
|
} else if (!strcmp(type, "meta")) {
|
|
/* Do nothing */
|
|
|
|
} else {
|
|
fprintf(stderr, "Unknown package type: %s\n", type);
|
|
return 1;
|
|
}
|
|
|
|
char * source = confreader_get(msk_manifest, pkg, "source");
|
|
if (source && !strcmp(source, "/tmp/msk.file")) {
|
|
char cmd[1024];
|
|
sprintf(cmd, "rm %s", source);
|
|
int status;
|
|
if ((status = system(cmd))) {
|
|
fprintf(stderr, "cleanup command returned %d\n", status);
|
|
return status;
|
|
}
|
|
}
|
|
|
|
char * post = confreader_getd(msk_manifest, pkg, "post", "");
|
|
if (strlen(post)) {
|
|
int status;
|
|
if ((status = system(post))) {
|
|
fprintf(stderr, "post-installation command returned %d\n", status);
|
|
return status;
|
|
}
|
|
}
|
|
|
|
/* Mark as installed */
|
|
FILE * installed = fopen(VAR_PATH "/installed", "a");
|
|
fprintf(installed, "%s==%s\n", pkg, confreader_get(msk_manifest, pkg, "version"));
|
|
fclose(installed);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int install_packages(int argc, char * argv[]) {
|
|
needs_root();
|
|
needs_lock();
|
|
read_config();
|
|
read_manifest(1);
|
|
read_installed();
|
|
|
|
/* Go through each package and find its dependencies */
|
|
list_t * ordered = list_create();
|
|
|
|
for (int i = 2; i < argc; ++i) {
|
|
if (process_package(ordered, argv[i])) {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/* Additional packages must be installed, let's ask. */
|
|
if (ordered->length != (unsigned int)(argc - 2) && !getenv("MSK_YES")) {
|
|
fprintf(stderr, "The following packages will be installed:\n");
|
|
fprintf(stderr, " ");
|
|
int notfirst = 0;
|
|
foreach(node, ordered) {
|
|
fprintf(stderr, "%s%s", notfirst ? " " : "", (char*)node->value);
|
|
notfirst = 1;
|
|
}
|
|
fprintf(stderr, "\nContinue? [Y/n] ");
|
|
fflush(stderr);
|
|
char resp[5];
|
|
fgets(resp, 5, stdin);
|
|
if (!(!strcmp(resp,"\n") || !strcmp(resp,"y\n") || !strcmp(resp,"Y\n") || !strcmp(resp,"yes\n"))) {
|
|
fprintf(stderr, "Aborting.\n");
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
foreach(node, ordered) {
|
|
if (install_package(node->value)) {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int list_packages(int argc, char * argv[]) {
|
|
read_config();
|
|
read_manifest(0);
|
|
read_installed();
|
|
|
|
/* Go through sections */
|
|
list_t * packages = hashmap_keys(msk_manifest->sections);
|
|
foreach(node, packages) {
|
|
char * name = node->value;
|
|
if (!strlen(name)) continue; /* skip empty section */
|
|
char * desc = confreader_get(msk_manifest, name, "description");
|
|
fprintf(stderr, " %c %20s %s\n", hashmap_has(msk_installed, name) ? 'I' : ' ', name, desc);
|
|
/* TODO: Installation status */
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int count_packages(int argc, char * argv[]) {
|
|
read_config();
|
|
read_manifest(0);
|
|
read_installed();
|
|
|
|
int installed = 0;
|
|
int available = 0;
|
|
|
|
/* Go through sections */
|
|
list_t * packages = hashmap_keys(msk_manifest->sections);
|
|
foreach(node, packages) {
|
|
char * name = node->value;
|
|
if (!strlen(name)) continue; /* skip empty section */
|
|
available++;
|
|
if (hashmap_has(msk_installed, name)) {
|
|
installed++;
|
|
}
|
|
}
|
|
|
|
fprintf(stdout, "%d installed; %d available\n", installed, available);
|
|
return 0;
|
|
}
|
|
|
|
static int version(void) {
|
|
fprintf(stderr, "msk " MSK_VERSION "\n");
|
|
return 0;
|
|
}
|
|
|
|
int main(int argc, char * argv[]) {
|
|
|
|
if (argc < 2) {
|
|
return usage(argc,argv);
|
|
} else if (!strcmp(argv[1],"--version")) {
|
|
return version();
|
|
} else if (!strcmp(argv[1],"update")) {
|
|
return update_stores(argc,argv);
|
|
} else if (!strcmp(argv[1], "install")) {
|
|
return install_packages(argc,argv);
|
|
} else if (!strcmp(argv[1], "list")) {
|
|
return list_packages(argc, argv);
|
|
} else if (!strcmp(argv[1], "count")) {
|
|
return count_packages(argc, argv);
|
|
} else {
|
|
fprintf(stderr, "%s: unknown command '%s'\n", argv[0], argv[1]);
|
|
return usage(argc,argv);
|
|
}
|
|
|
|
}
|
|
|