sortix-mirror/tix/tix.c

307 lines
8.1 KiB
C
Raw Permalink Normal View History

/*
* Copyright (c) 2013, 2015, 2016, 2023 Jonas 'Sortie' Termansen.
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*
* tix.c
* Front end to the Tix package management system.
*/
2013-03-26 11:40:43 +00:00
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <assert.h>
#include <ctype.h>
#include <dirent.h>
2016-08-09 23:29:04 +00:00
#include <err.h>
2013-03-26 11:40:43 +00:00
#include <errno.h>
#include <fcntl.h>
#include <libgen.h>
2013-03-26 11:40:43 +00:00
#include <limits.h>
#include <signal.h>
2013-03-26 11:40:43 +00:00
#include <stdarg.h>
2016-02-28 23:10:59 +00:00
#include <stdbool.h>
2013-03-26 11:40:43 +00:00
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "util.h"
typedef struct
{
char* collection;
char* tixdb_path;
string_array_t coll_conf;
string_array_t repo_list;
2015-08-29 20:10:50 +00:00
bool reinstall;
2013-03-26 11:40:43 +00:00
} params_t;
2015-08-29 20:10:50 +00:00
char* FindPackageInRepository(params_t* params,
const char* repo,
const char* pkg_name)
2013-03-26 11:40:43 +00:00
{
2015-08-29 20:10:50 +00:00
char* repo_location;
if ( repo[0] == '/' )
repo_location = strdup(repo);
else
repo_location = join_paths(params->tixdb_path, repo);
char* tix_name = print_string("%s.tix.tar.xz", pkg_name);
char* tix_path = join_paths(repo_location, tix_name);
free(repo_location);
free(tix_name);
if ( !IsFile(tix_path) )
{
free(tix_path);
return NULL;
}
return tix_path;
2013-03-26 11:40:43 +00:00
}
2015-08-29 20:10:50 +00:00
char* FindPackage(params_t* params,
string_array_t* repositories,
const char* pkg_name)
2013-03-26 11:40:43 +00:00
{
for ( size_t i = 0; i < repositories->length; i++ )
{
const char* repo = repositories->strings[i];
2015-08-29 20:10:50 +00:00
char* ret = FindPackageInRepository(params, repo, pkg_name);
2013-03-26 11:40:43 +00:00
if ( ret )
return ret;
}
return NULL;
}
string_array_t GetPackageDependencies(params_t* params, const char* pkg_name)
{
string_array_t ret = string_array_make();
2015-08-29 20:10:50 +00:00
char* pkg_path = FindPackage(params, &params->repo_list, pkg_name);
2013-03-26 11:40:43 +00:00
if ( !pkg_path )
2016-08-09 23:29:04 +00:00
err(1, "unable to locate package `%s'", pkg_name);
2013-03-26 11:40:43 +00:00
// TODO: After releasing Sortix 1.1, delete generation 2 compatibility.
bool modern = false;
const char* tixinfo_path = "tix/tixinfo/";
if ( !TarContainsFile(pkg_path, tixinfo_path) )
{
const char* tixinfo_path_old = "tix/tixinfo";
if ( !TarContainsFile(pkg_path, tixinfo_path_old) )
errx(1, "`%s' doesn't contain a `%s' directory", pkg_path,
tixinfo_path);
tixinfo_path = tixinfo_path_old;
modern = false;
}
2013-03-26 11:40:43 +00:00
string_array_t tixinfo = string_array_make();
FILE* tixinfo_fp = TarOpenFile(pkg_path, tixinfo_path);
switch ( variables_append_file(&tixinfo, tixinfo_fp) )
{
case -1: err(1, "%s: %s", pkg_path, tixinfo_path);
case -2: errx(1, "%s: %s: Syntax error", pkg_path, tixinfo_path);
}
2013-03-26 11:40:43 +00:00
fclose(tixinfo_fp);
VerifyTixInformation(&tixinfo, pkg_path);
const char* deps = dictionary_get_def(&tixinfo, modern ? "RUNTIME_DEPS" :
"pkg.runtime-deps", "");
2013-03-26 11:40:43 +00:00
string_array_append_token_string(&ret, deps);
const char* alias = dictionary_get_def(&tixinfo, modern ? "ALIAS_OF" :
"pkg.alias-of", "");
Add split packages and cross-bootstrapping support to tix-build(8). pkg.use-bootstrap can now be set to true to add a bootstrap phase to cross-builds. I.e. the package is built for the native platform and installed to a temporary location, which is in the PATH during the actual cross-compilation. This feature is useful for some misbehaving ports that can cross-compile, but require the exact same version of the software installed locally. The bootstrap build is controlled with the bootstrap.foo variables rather than the normal pkg.foo variables. pkg.source-package can now be set to the name of another package, whose source code is built using the current tixbuildinfo. This feature allows providing multiple packages using the same source code package. By default, the source code of the source package is assumed to be in ../${pkg.source-package}, but this can be overridden with the option --source-directory. pkg.alias-of can now be set to the name of another package to specify that this package is an alias of the other package, creating an empty binary package depending on the real package. pkg.subdir support has been fixed in the clean and post-install phases. pkg-config support has been improved and PKG_CONFIG is now set to $HOST-pkg-config and PKG_CONFIG_FOR_BUILD is set to pkg-config. tix-build has been refactored as needed and generally cleaned up. Error handling, such as on allocations, have been added in a lot of cases. The support for FOO_FOR_BUILD variables have been unified and simplified. Appending to PATH now correctly handles the empty PATH.
2016-12-26 12:59:51 +00:00
string_array_append_token_string(&ret, alias);
2013-03-26 11:40:43 +00:00
string_array_reset(&tixinfo);
free(pkg_path);
return ret;
}
void GetPackageRecursiveDependencies(params_t* params, string_array_t* sofar,
const char* pkg_name)
{
if ( string_array_contains(sofar, pkg_name) )
return;
// Avoid endless recursion by adding our package before the recursive call,
// in case we need to satisfy cyclic dependencies.
string_array_append(sofar, pkg_name);
string_array_t pkg_deps = GetPackageDependencies(params, pkg_name);
for ( size_t i = 0; i < pkg_deps.length; i++ )
if ( !string_array_contains(sofar, pkg_deps.strings[i]) )
GetPackageRecursiveDependencies(params, sofar, pkg_deps.strings[i]);
string_array_reset(&pkg_deps);
}
void InstallPackageOfName(params_t* params, const char* pkg_name)
{
2015-08-29 20:10:50 +00:00
char* pkg_path = FindPackage(params, &params->repo_list, pkg_name);
2013-03-26 11:40:43 +00:00
if ( !pkg_path )
2016-08-09 23:29:04 +00:00
err(1, "unable to locate package `%s'", pkg_name);
2013-03-26 11:40:43 +00:00
if ( fork_and_wait_or_death() )
{
const char* cmd_argv[] =
{
"tix-install",
2013-03-26 11:40:43 +00:00
"--collection", params->collection,
"--", pkg_path,
NULL
};
execvp(cmd_argv[0], (char* const*) cmd_argv);
2016-08-09 23:29:04 +00:00
err(127, "`%s'", cmd_argv[0]);
2013-03-26 11:40:43 +00:00
}
free(pkg_path);
}
static void help(FILE* fp, const char* argv0)
2013-03-26 11:40:43 +00:00
{
2015-08-29 20:10:50 +00:00
fprintf(fp, "Usage: %s [PREFIX] COMMAND ...\n", argv0);
2013-03-26 11:40:43 +00:00
fprintf(fp, "Front end to the Tix package management system.\n");
2015-08-29 20:10:50 +00:00
fprintf(fp, "\n");
fprintf(fp, "Commands:\n");
fprintf(fp, " install [--reinstall] PACKAGE...\n");
2013-03-26 11:40:43 +00:00
}
static void version(FILE* fp, const char* argv0)
2013-03-26 11:40:43 +00:00
{
fprintf(fp, "%s (Sortix) %s\n", argv0, VERSIONSTR);
2013-03-26 11:40:43 +00:00
}
int main(int argc, char* argv[])
{
params_t params;
2015-08-29 20:10:50 +00:00
memset(&params, 0, sizeof(params));
params.collection = NULL;
2013-03-26 11:40:43 +00:00
const char* argv0 = argv[0];
for ( int i = 0; i < argc; i++ )
{
const char* arg = argv[i];
if ( arg[0] != '-' || !arg[1] )
2013-03-26 11:40:43 +00:00
continue;
argv[i] = NULL;
if ( !strcmp(arg, "--") )
break;
if ( arg[1] != '-' )
{
2016-02-28 23:10:59 +00:00
char c;
while ( (c = *++arg) ) switch ( c )
2013-03-26 11:40:43 +00:00
{
default:
fprintf(stderr, "%s: unknown option -- '%c'\n", argv0, c);
help(stderr, argv0);
2013-03-26 11:40:43 +00:00
exit(1);
}
}
else if ( !strcmp(arg, "--help") )
help(stdout, argv0), exit(0);
else if ( !strcmp(arg, "--version") )
version(stdout, argv0), exit(0);
2013-03-26 11:40:43 +00:00
else if ( GET_OPTION_VARIABLE("--collection", &params.collection) ) { }
2015-08-29 20:10:50 +00:00
else if ( !strcmp(arg, "--reinstall") )
params.reinstall = true;
2013-03-26 11:40:43 +00:00
else
{
fprintf(stderr, "%s: unknown option: %s\n", argv0, arg);
help(stderr, argv0);
2013-03-26 11:40:43 +00:00
exit(1);
}
}
if ( argc == 1 )
{
help(stdout, argv0);
2013-03-26 11:40:43 +00:00
exit(0);
}
compact_arguments(&argc, &argv);
2013-03-26 11:40:43 +00:00
ParseOptionalCommandLineCollectionPrefix(&params.collection, &argc, &argv);
VerifyCommandLineCollection(&params.collection);
params.tixdb_path = join_paths(params.collection, "tix");
char* coll_conf_path = join_paths(params.tixdb_path, "collection.conf");
params.coll_conf = string_array_make();
switch ( variables_append_file_path(&params.coll_conf, coll_conf_path) )
{
case -1: err(1, "%s", coll_conf_path);
case -2: errx(2, "%s: Syntax error", coll_conf_path);
}
2013-03-26 11:40:43 +00:00
VerifyTixCollectionConfiguration(&params.coll_conf, coll_conf_path);
free(coll_conf_path);
// TODO: Decide the fate of repository.list.
2013-03-26 11:40:43 +00:00
char* repo_list_path = join_paths(params.tixdb_path, "repository.list");
params.repo_list = string_array_make();
if ( !string_array_append_file_path(&params.repo_list, repo_list_path) &&
errno != ENOENT )
2016-08-09 23:29:04 +00:00
err(1, "`%s'", repo_list_path);
2013-03-26 11:40:43 +00:00
free(repo_list_path);
if ( argc == 1 )
2016-08-09 23:29:04 +00:00
errx(1, "error: no command specified.");
2013-03-26 11:40:43 +00:00
const char* cmd = argv[1];
if ( !strcmp(cmd, "install") )
{
if ( argc == 2 )
2016-08-09 23:29:04 +00:00
errx(1, "expected list of packages to install after `install'");
2013-03-26 11:40:43 +00:00
string_array_t work = string_array_make();
char* tixinfo_dir = join_paths(params.tixdb_path, "tixinfo");
if ( !tixinfo_dir )
err(1, "malloc");
2013-03-26 11:40:43 +00:00
for ( int i = 2; i < argc; i++ )
{
const char* pkg_name = argv[i];
char* tixinfo = join_paths(tixinfo_dir, pkg_name);
bool installed = !access(tixinfo, F_OK);
free(tixinfo);
if ( installed )
2013-03-26 11:40:43 +00:00
{
2015-08-29 20:10:50 +00:00
if ( params.reinstall )
{
string_array_append(&work, pkg_name);
continue;
}
2013-03-26 11:40:43 +00:00
printf("Package `%s' is already installed.\n", pkg_name);
continue;
}
GetPackageRecursiveDependencies(&params, &work, pkg_name);
}
free(tixinfo_dir);
2013-03-26 11:40:43 +00:00
for ( size_t i = 0; i < work.length; i++ )
InstallPackageOfName(&params, work.strings[i]);
string_array_reset(&work);
return 0;
}
else
{
fprintf(stderr, "%s: unknown command: `%s'\n", argv0, cmd);
exit(1);
}
}