1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/src/rpm.c Mon Jun 16 17:54:29 2008 -0400
1.3 @@ -0,0 +1,897 @@
1.4 +/*
1.5 + * Copyright (C) 2008 Kristian Høgsberg <krh@redhat.com>
1.6 + * Copyright (C) 2008 Red Hat, Inc
1.7 + *
1.8 + * This program is free software; you can redistribute it and/or modify
1.9 + * it under the terms of the GNU General Public License as published by
1.10 + * the Free Software Foundation; either version 2 of the License, or
1.11 + * (at your option) any later version.
1.12 + *
1.13 + * This program is distributed in the hope that it will be useful,
1.14 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
1.15 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1.16 + * GNU General Public License for more details.
1.17 + *
1.18 + * You should have received a copy of the GNU General Public License along
1.19 + * with this program; if not, write to the Free Software Foundation, Inc.,
1.20 + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
1.21 + */
1.22 +
1.23 +#include <stdlib.h>
1.24 +#include <string.h>
1.25 +#include <stdio.h>
1.26 +#include <dirent.h>
1.27 +#include "razor.h"
1.28 +
1.29 +enum option_type {
1.30 + OPTION_LAST,
1.31 + OPTION_GROUP,
1.32 + OPTION_BOOL,
1.33 + OPTION_STRING
1.34 +};
1.35 +
1.36 +struct option {
1.37 + enum option_type type;
1.38 + const char *name;
1.39 + char short_name;
1.40 + const char *arg_name;
1.41 + const char *description;
1.42 + void *data;
1.43 +};
1.44 +
1.45 +/* A note about all these options: rpm allows options to mean
1.46 + * different things depending on what other options are present on the
1.47 + * command line. For example, if -q or --query is present, -i no
1.48 + * longer means install, but info. The way we handle this is by
1.49 + * setting all the options that may match (ie if -i is given we set
1.50 + * install and info), and then look at the relevent one depending on
1.51 + * what else in on the command line. */
1.52 +
1.53 +static int option_all, option_whatrequires, option_whatprovides;
1.54 +static int option_package;
1.55 +
1.56 +static const struct option query_options[] = {
1.57 + { OPTION_BOOL, "configfiles", 'c', NULL, "list all configuration files", NULL },
1.58 + { OPTION_BOOL, "docfiles", 'd', NULL, "list all documentation files", NULL },
1.59 + { OPTION_BOOL, "dump", 0, NULL, "dump basic file information", NULL },
1.60 + { OPTION_BOOL, "list", 0, NULL, "list files in package", NULL },
1.61 + { OPTION_STRING, "queryformat", 0, "QUERYFORMAT", "use the following query format", NULL },
1.62 + { OPTION_BOOL, "state", 's', NULL, "display the states of the listed files", NULL },
1.63 + { OPTION_BOOL, "all", 'a', NULL, "query/verify all packages", &option_all },
1.64 + { OPTION_BOOL, "file", 'f', NULL, "query/verify package(s) owning file", NULL },
1.65 + { OPTION_BOOL, "group", 'g', NULL, "query/verify package(s) in group", NULL },
1.66 + { OPTION_BOOL, "package", 'p', NULL, "query/verify a package file", &option_package },
1.67 + { OPTION_BOOL, "ftswalk", 'W', NULL, "query/verify package(s) from TOP file tree walk", NULL },
1.68 + { OPTION_BOOL, "pkgid", 0, NULL, "query/verify package(s) with package identifier", NULL },
1.69 + { OPTION_BOOL, "hdrid", 0, NULL, "query/verify package(s) with header identifier", NULL },
1.70 + { OPTION_BOOL, "fileid", 0, NULL, "query/verify package(s) with file identifier", NULL },
1.71 + { OPTION_BOOL, "specfile", 0, NULL, "query a spec file", NULL, },
1.72 + { OPTION_BOOL, "triggeredby", 0, NULL, "query the package(s) triggered by the package", NULL },
1.73 + { OPTION_BOOL, "whatrequires", 0, NULL, "query/verify the package(s) which require a dependency", &option_whatrequires },
1.74 + { OPTION_BOOL, "whatprovides", 0, NULL, "query/verify the package(s) which provide a dependency", &option_whatprovides },
1.75 + { OPTION_BOOL, "nomanifest", 0, NULL, "do not process non-package files as manifests", NULL },
1.76 + { }
1.77 +};
1.78 +
1.79 +static int option_nodeps;
1.80 +
1.81 +static const struct option verify_options[] = {
1.82 + { OPTION_BOOL, "nomd5", 0, NULL, "don't verify MD5 digest of files", NULL },
1.83 + { OPTION_BOOL, "nofiles", 0, NULL, "don't verify files in package", NULL },
1.84 + { OPTION_BOOL, "nodeps", 0, NULL, "don't verify package dependencies", &option_nodeps },
1.85 + { OPTION_BOOL, "noscript", 0, NULL, "don't execute verify script(s)", NULL, },
1.86 + { OPTION_BOOL, "all", 'a', NULL, "query/verify all packages", &option_all },
1.87 + { OPTION_BOOL, "file", 'f', NULL, "query/verify package(s) owning file", NULL },
1.88 + { OPTION_BOOL, "group", 'g', NULL, "query/verify package(s) in group", NULL },
1.89 + { OPTION_BOOL, "package", 'p', NULL, "query/verify a package file", &option_package },
1.90 + { OPTION_BOOL, "ftswalk", 'W', NULL, "query/verify package(s) from TOP file tree walk", NULL },
1.91 + { OPTION_BOOL, "pkgid", 0, NULL, "query/verify package(s) with package identifier", NULL },
1.92 + { OPTION_BOOL, "hdrid", 0, NULL, "query/verify package(s) with header identifier", NULL },
1.93 + { OPTION_BOOL, "fileid", 0, NULL, "query/verify package(s) with file identifier", NULL },
1.94 + { OPTION_BOOL, "specfile", 0, NULL, "query a spec file", NULL },
1.95 + { OPTION_BOOL, "triggeredby", 0, NULL, "query the package(s) triggered by the package", NULL },
1.96 + { OPTION_BOOL, "whatrequires", 0, NULL, "query/verify the package(s) which require a dependency", &option_whatrequires },
1.97 + { OPTION_BOOL, "whatprovides", 0, NULL, "query/verify the package(s) which provide a dependency", &option_whatprovides },
1.98 + { OPTION_BOOL, "nomanifest", 0, NULL, "do not process non-package files as manifests", NULL },
1.99 + { }
1.100 +};
1.101 +
1.102 +static const struct option ftw_options[] = {
1.103 + { OPTION_BOOL, "comfollow", 0, NULL, "FTS_COMFOLLOW: follow command line symlinks", NULL },
1.104 + { OPTION_BOOL, "logical", 0, NULL, "FTS_LOGICAL: logical walk", NULL },
1.105 + { OPTION_BOOL, "nochdir", 0, NULL, "FTS_NOCHDIR: don't change directories", NULL },
1.106 + { OPTION_BOOL, "nostat", 0, NULL, "FTS_NOSTAT: don't get stat info", NULL },
1.107 + { OPTION_BOOL, "physical", 0, NULL, "FTS_PHYSICAL: physical walk", NULL },
1.108 + { OPTION_BOOL, "seedot", 0, NULL, "FTS_SEEDOT: return dot and dot-dot", NULL },
1.109 + { OPTION_BOOL, "xdev", 0, NULL, "FTS_XDEV: don't cross devices", NULL },
1.110 + { OPTION_BOOL, "whiteout", 0, NULL, "FTS_WHITEOUT: return whiteout information", NULL },
1.111 + { }
1.112 +};
1.113 +
1.114 +static const struct option signature_options[] = {
1.115 + { OPTION_BOOL, "addsign", 0, NULL, "sign package(s) (identical to --resign)", NULL, },
1.116 + { OPTION_BOOL, "checksig", 'K', NULL, "verify package signature(s)", NULL, },
1.117 + { OPTION_BOOL, "delsign", 0, NULL, "delete package signatures", NULL, },
1.118 + { OPTION_BOOL, "import", 0, NULL, "import an armored public key", NULL, },
1.119 + { OPTION_BOOL, "resign", 0, NULL, "sign package(s) (identical to --addsign)", NULL, },
1.120 + { OPTION_BOOL, "nodigest", 0, NULL, "don't verify package digest(s)", NULL, },
1.121 + { OPTION_BOOL, "nosignature", 0, NULL, "don't verify package signature(s)", NULL },
1.122 + { }
1.123 +};
1.124 +
1.125 +static int option_initdb;
1.126 +
1.127 +static const struct option database_options[] = {
1.128 + { OPTION_BOOL, "initdb", 0, NULL, "initialize database", &option_initdb },
1.129 + { OPTION_BOOL, "rebuilddb", 0, NULL, "rebuild database inverted lists from installed package headers", NULL },
1.130 + { }
1.131 +};
1.132 +
1.133 +static int option_erase, option_install, option_upgrade, option_justdb;
1.134 +static int option_test;
1.135 +
1.136 +static const struct option install_options[] = {
1.137 + { OPTION_BOOL, "aid", 0, NULL, "add suggested packages to transaction", NULL, },
1.138 + { OPTION_BOOL, "allfiles", 0, NULL, "install all files, even configurations which might otherwise be skipped", NULL, },
1.139 + { OPTION_BOOL, "allmatches", 0, NULL, "remove all packages which match <package> (normally an error is generated if <package> specified multiple packages)", NULL, },
1.140 + { OPTION_BOOL, "badreloc", 0, NULL, "relocate files in non-relocatable package", NULL },
1.141 + { OPTION_BOOL, "erase", 'e', "<package>", "erase (uninstall) package", &option_erase },
1.142 + { OPTION_BOOL, "excludedocs", 0, NULL, "do not install documentation", NULL, },
1.143 + { OPTION_BOOL, "excludepath", 0, "<path>", "skip files with leading component <path> ", NULL, },
1.144 + { OPTION_BOOL, "fileconflicts", 0, NULL, "detect file conflicts between packages", NULL, },
1.145 + { OPTION_BOOL, "force", 0, NULL, "short hand for --replacepkgs --replacefiles", NULL },
1.146 + { OPTION_BOOL, "freshen", 'F', "<packagefile>+", "upgrade package(s) if already installed", NULL },
1.147 + { OPTION_BOOL, "hash", 'h', NULL, "print hash marks as package installs (good with -v)", NULL },
1.148 + { OPTION_BOOL, "ignorearch", 0, NULL, "don't verify package architecture", NULL, },
1.149 + { OPTION_BOOL, "ignoreos", 0, NULL, "don't verify package operating system", NULL, },
1.150 + { OPTION_BOOL, "ignoresize", 0, NULL, "don't check disk space before installing", NULL },
1.151 + { OPTION_BOOL, "install", 'i', NULL, "install package(s)", &option_install },
1.152 + { OPTION_BOOL, "justdb", 0, NULL, "update the database, but do not modify the filesystem", &option_justdb, },
1.153 + { OPTION_BOOL, "nodeps", 0, NULL, "do not verify package dependencies", &option_nodeps, },
1.154 + { OPTION_BOOL, "nomd5", 0, NULL, "don't verify MD5 digest of files", NULL, },
1.155 + { OPTION_BOOL, "nocontexts", 0, NULL, "don't install file security contexts", NULL, },
1.156 + { OPTION_BOOL, "noorder", 0, NULL, "do not reorder package installation to satisfy dependencies", NULL, },
1.157 + { OPTION_BOOL, "nosuggest", 0, NULL, "do not suggest missing dependency resolution(s)", NULL, },
1.158 + { OPTION_BOOL, "noscripts", 0, NULL, "do not execute package scriptlet(s)", NULL, },
1.159 + { OPTION_BOOL, "notriggers", 0, NULL, "do not execute any scriptlet(s) triggered by this package", NULL, },
1.160 + { OPTION_BOOL, "oldpackage", 0, NULL, "upgrade to an old version of the package (--force on upgrades does this automatically)", NULL },
1.161 + { OPTION_BOOL, "percent", 0, NULL, "print percentages as package installs", NULL, },
1.162 + { OPTION_STRING, "prefix", 0, "<dir>", "relocate the package to <dir>, if relocatable", NULL, },
1.163 + { OPTION_STRING, "relocate", 0, "<old>=<new>", "relocate files from path <old> to <new>", NULL, },
1.164 + { OPTION_BOOL, "repackage", 0, NULL, "save erased package files by repackaging", NULL, },
1.165 + { OPTION_BOOL, "replacefiles", 0, NULL, "ignore file conflicts between packages", NULL, },
1.166 + { OPTION_BOOL, "replacepkgs", 0, NULL, "reinstall if the package is already present", NULL, },
1.167 + { OPTION_BOOL, "test", 0, NULL, "don't install, but tell if it would work or not", &option_test },
1.168 + { OPTION_BOOL, "upgrade", 'U', "<packagefile>+", "upgrade package(s)", &option_upgrade },
1.169 + { }
1.170 +};
1.171 +
1.172 +static int option_version;
1.173 +static const char *option_root = "install";
1.174 +
1.175 +static const struct option common_options[] = {
1.176 + { OPTION_STRING, "define", 'D', "MACRO EXPR", "define MACRO with value EXPR", NULL, },
1.177 + { OPTION_STRING, "eval", 'E', "EXPR", "print macro expansion of EXPR", NULL },
1.178 + { OPTION_STRING, "macros", 0, "<FILE:...>", "read <FILE:...> instead of default file(s)", NULL },
1.179 + { OPTION_BOOL, "nodigest", 0, NULL, "don't verify package digest(s)", NULL, },
1.180 + { OPTION_BOOL, "nosignature", 0, NULL, "don't verify package signature(s)", NULL, },
1.181 + { OPTION_STRING, "rcfile", 0, "<FILE:...>", "read <FILE:...> instead of default file(s)", NULL },
1.182 + { OPTION_STRING, "root", 'r', "ROOT", "use ROOT as top level directory (default: \"/\")", &option_root },
1.183 + { OPTION_BOOL, "querytags", 0, NULL, "display known query tags", NULL, },
1.184 + { OPTION_BOOL, "showrc", 0, NULL, "display final rpmrc and macro configuration", NULL, },
1.185 + { OPTION_BOOL, "quiet", 0, NULL, "provide less detailed output", NULL },
1.186 + { OPTION_BOOL, "verbose", 'v', NULL, "provide more detailed output", NULL },
1.187 + { OPTION_BOOL, "version", 0, NULL, "print the version of rpm being used", &option_version },
1.188 + { }
1.189 +};
1.190 +
1.191 +static int option_conflicts, option_obsoletes, option_requires;
1.192 +static int option_provides, option_info, option_changelog;
1.193 +
1.194 +static const struct option alias_options[] = {
1.195 + { OPTION_BOOL, "scripts", 0, NULL, "list install/erase scriptlets from package(s)", NULL, },
1.196 + { OPTION_BOOL, "setperms", 0, NULL, "set permissions of files in a package", NULL, },
1.197 + { OPTION_BOOL, "setugids", 0, NULL, "set user/group ownership of files in a package", NULL, },
1.198 + { OPTION_BOOL, "conflicts", 0, NULL, "list capabilities this package conflicts with", &option_conflicts, },
1.199 + { OPTION_BOOL, "obsoletes", 0, NULL, "list other packages removed by installing this package", &option_obsoletes, },
1.200 + { OPTION_BOOL, "provides", 0, NULL, "list capabilities that this package provides", &option_provides, },
1.201 + { OPTION_BOOL, "requires", 0, NULL, "list capabilities required by package(s)", &option_requires, },
1.202 + { OPTION_BOOL, "info", 'i', NULL, "list descriptive information from package(s)", &option_info, },
1.203 + { OPTION_BOOL, "changelog", 0, NULL, "list change logs for this package", &option_changelog, },
1.204 + { OPTION_BOOL, "xml", 0, NULL, "list metadata in xml", NULL, },
1.205 + { OPTION_BOOL, "triggers", 0, NULL, "list trigger scriptlets from package(s)", NULL, },
1.206 + { OPTION_BOOL, "last", 0, NULL, "list package(s) by install time, most recent first", NULL, },
1.207 + { OPTION_BOOL, "dupes", 0, NULL, "list duplicated packages", NULL, },
1.208 + { OPTION_BOOL, "filesbypkg", 0, NULL, "list all files from each package", NULL, },
1.209 + { OPTION_BOOL, "fileclass", 0, NULL, "list file names with classes", NULL, },
1.210 + { OPTION_BOOL, "filecolor", 0, NULL, "list file names with colors", NULL, },
1.211 + { OPTION_BOOL, "filecontext", 0, NULL, "list file names with security context from header", NULL, },
1.212 + { OPTION_BOOL, "fscontext", 0, NULL, "list file names with security context from file system", NULL, },
1.213 + { OPTION_BOOL, "recontext", 0, NULL, "list file names with security context from policy RE", NULL, },
1.214 + { OPTION_BOOL, "fileprovide", 0, NULL, "list file names with provides", NULL, },
1.215 + { OPTION_BOOL, "filerequire", 0, NULL, "list file names with requires", NULL, },
1.216 + { OPTION_BOOL, "redhatprovides", 0, NULL, "find package name that contains a provided capability (needs rpmdb-redhat package installed)", NULL, },
1.217 + { OPTION_BOOL, "redhatrequires", 0, NULL, "find package name that contains a required capability (needs rpmdb-redhat package installed)", NULL, },
1.218 + { OPTION_STRING, "buildpolicy", 0, "<policy>", "set buildroot <policy> (e.g. compress man pages)", NULL, },
1.219 + { OPTION_BOOL, "with", 0, "<option>", "enable configure <option> for build", NULL, },
1.220 + { OPTION_BOOL, "without", 0, "<option>", "disable configure <option> for build", NULL },
1.221 + { }
1.222 +};
1.223 +
1.224 +static int option_help, option_usage;
1.225 +
1.226 +static const struct option help_options[] = {
1.227 + { OPTION_BOOL, "help", '?', NULL, "Show this help message", &option_help },
1.228 + { OPTION_BOOL, "usage", 0, NULL, "Display brief usage message", &option_usage},
1.229 + { }
1.230 +};
1.231 +
1.232 +static int option_query, option_verify;
1.233 +
1.234 +static const struct option rpm_options[] = {
1.235 + { OPTION_BOOL, "query", 'q', NULL, "Query rpm database", &option_query },
1.236 + { OPTION_BOOL, "verify", 'V', NULL, "Verify rpm database", &option_verify },
1.237 + { OPTION_GROUP, NULL, 0, NULL, "Query options (with -q or --query):", &query_options },
1.238 + { OPTION_GROUP, NULL, 0, NULL, "Verify options (with -V or --verify):", &verify_options },
1.239 + { OPTION_GROUP, NULL, 0, NULL, "File tree walk options (with --ftswalk):", &ftw_options },
1.240 + { OPTION_GROUP, NULL, 0, NULL, "Signature options:", &signature_options },
1.241 + { OPTION_GROUP, NULL, 0, NULL, "Database options:", &database_options },
1.242 + { OPTION_GROUP, NULL, 0, NULL, "Install/Upgrade/Erase options:", &install_options },
1.243 + { OPTION_GROUP, NULL, 0, NULL, "Common options for all rpm modes and executables:", &common_options },
1.244 + { OPTION_GROUP, NULL, 0, NULL, "Options implemented via popt alias/exec:", &alias_options },
1.245 + { OPTION_GROUP, NULL, 0, NULL, "Help options", &help_options },
1.246 + { }
1.247 +};
1.248 +
1.249 +static const char system_repo_filename[] = "system.repo";
1.250 +static const char *repo_filename = system_repo_filename;
1.251 +
1.252 +static void
1.253 +command_initdb(int argc, const char *argv[])
1.254 +{
1.255 + razor_root_create(option_root);
1.256 +}
1.257 +
1.258 +static struct razor_property *
1.259 +add_property_packages(struct razor_set *set,
1.260 + struct razor_package_query *query,
1.261 + const char *ref_name,
1.262 + const char *ref_version,
1.263 + enum razor_property_type ref_type)
1.264 +{
1.265 + struct razor_property *property;
1.266 + struct razor_property_iterator *pi;
1.267 + struct razor_package_iterator *pkgi;
1.268 + const char *name, *version;
1.269 + enum razor_property_type type;
1.270 + enum razor_version_relation relation;
1.271 +
1.272 + pi = razor_property_iterator_create(set, NULL);
1.273 + while (razor_property_iterator_next(pi, &property, &name,
1.274 + &relation, &version, &type)) {
1.275 + if (strcmp(ref_name, name) != 0)
1.276 + continue;
1.277 + if (ref_version && relation == RAZOR_VERSION_EQUAL &&
1.278 + strcmp(ref_version, version) != 0)
1.279 + continue;
1.280 + if (ref_type != type)
1.281 + continue;
1.282 +
1.283 + pkgi = razor_package_iterator_create_for_property(set,
1.284 + property);
1.285 + razor_package_query_add_iterator(query, pkgi);
1.286 + razor_package_iterator_destroy(pkgi);
1.287 + }
1.288 + razor_property_iterator_destroy(pi);
1.289 +
1.290 + return property;
1.291 +}
1.292 +
1.293 +static int
1.294 +strcmpp(const void *p1, const void *p2)
1.295 +{
1.296 + return strcmp(*(char * const *) p1, *(char * const *) p2);
1.297 +}
1.298 +
1.299 +static void
1.300 +add_command_line_packages(struct razor_set *set,
1.301 + struct razor_package_query *query,
1.302 + int argc, const char **argv)
1.303 +{
1.304 + struct razor_package *package;
1.305 + struct razor_package_iterator *pi;
1.306 + const char *name, *version, *arch;
1.307 + int i, cmp, errors;
1.308 +
1.309 + qsort(argv, argc, sizeof(*argv), strcmpp);
1.310 + i = 0;
1.311 + errors = 0;
1.312 +
1.313 + pi = razor_package_iterator_create(set);
1.314 +
1.315 + while (razor_package_iterator_next(pi, &package,
1.316 + &name, &version, &arch)) {
1.317 + while (cmp = strcmp(argv[i], name), cmp < 0 && i < argc) {
1.318 + fprintf(stderr, "error: package %s is not installed\n",
1.319 + argv[i]);
1.320 + errors++;
1.321 + i++;
1.322 + }
1.323 +
1.324 + if (cmp == 0) {
1.325 + razor_package_query_add_package(query, package);
1.326 + i++;
1.327 + }
1.328 + }
1.329 +
1.330 + razor_package_iterator_destroy(pi);
1.331 +
1.332 + if (errors)
1.333 + exit(1);
1.334 +}
1.335 +
1.336 +static struct razor_package_iterator *
1.337 +get_query_packages(struct razor_set *set, int argc, const char *argv[])
1.338 +{
1.339 + struct razor_package_query *query;
1.340 + struct razor_package_iterator *pi;
1.341 + int i;
1.342 +
1.343 + if (option_all + option_whatprovides + option_whatrequires > 1) {
1.344 + printf("only one type of query/verify "
1.345 + "may be performed at a time\n");
1.346 + exit(1);
1.347 + }
1.348 +
1.349 + query = razor_package_query_create(set);
1.350 +
1.351 + if (option_all) {
1.352 + pi = razor_package_iterator_create(set);
1.353 + razor_package_query_add_iterator(query, pi);
1.354 + razor_package_iterator_destroy(pi);
1.355 + } else if (option_whatrequires) {
1.356 + for (i = 0; i < argc; i++)
1.357 + add_property_packages(set, query,
1.358 + argv[i], NULL,
1.359 + RAZOR_PROPERTY_REQUIRES);
1.360 + } else if (option_whatprovides) {
1.361 + for (i = 0; i < argc; i++)
1.362 + add_property_packages(set, query,
1.363 + argv[i], NULL,
1.364 + RAZOR_PROPERTY_PROVIDES);
1.365 + } else if (argc > 0) {
1.366 + add_command_line_packages(set, query, argc, argv);
1.367 + } else {
1.368 + printf("no arguments given for query/verify\n");
1.369 + exit(1);
1.370 + }
1.371 +
1.372 + return razor_package_query_finish(query);
1.373 +}
1.374 +
1.375 +static const char *relation_string[] = { "<", "<=", "=", ">=", ">" };
1.376 +
1.377 +static void
1.378 +print_package_properties(struct razor_set *set,
1.379 + struct razor_package *package,
1.380 + enum razor_property_type ref_type)
1.381 +{
1.382 + struct razor_property *property;
1.383 + struct razor_property_iterator *pi;
1.384 + const char *name, *version;
1.385 + enum razor_property_type type;
1.386 + enum razor_version_relation relation;
1.387 +
1.388 + pi = razor_property_iterator_create(set, package);
1.389 + while (razor_property_iterator_next(pi, &property,
1.390 + &name, &relation, &version,
1.391 + &type)) {
1.392 + if (type != ref_type)
1.393 + continue;
1.394 + if (version[0] == '\0')
1.395 + printf("%s\n", name);
1.396 + else
1.397 + printf("%s %s %s\n", name,
1.398 + relation_string[relation], version);
1.399 + }
1.400 + razor_property_iterator_destroy(pi);
1.401 +}
1.402 +
1.403 +static void
1.404 +print_package_info(struct razor_set *set, struct razor_package *package)
1.405 +{
1.406 + printf("FIXME: Package info not tracked.\n");
1.407 +}
1.408 +
1.409 +static void
1.410 +print_package_changelog(struct razor_set *set, struct razor_package *package)
1.411 +{
1.412 + printf("FIXME: Package changelog not tracked.\n");
1.413 +}
1.414 +
1.415 +static struct razor_set *
1.416 +create_set_from_command_line(int argc, const char *argv[])
1.417 +{
1.418 + struct razor_importer *importer;
1.419 + struct razor_rpm *rpm;
1.420 + int i;
1.421 +
1.422 + importer = razor_importer_new();
1.423 +
1.424 + for (i = 0; i < argc; i++) {
1.425 + rpm = razor_rpm_open(argv[i]);
1.426 + if (rpm == NULL)
1.427 + continue;
1.428 + if (razor_importer_add_rpm(importer, rpm))
1.429 + printf("couldn't import %s\n", argv[i]);
1.430 +
1.431 + razor_rpm_close(rpm);
1.432 + }
1.433 +
1.434 + return razor_importer_finish(importer);
1.435 +}
1.436 +
1.437 +static void
1.438 +command_query(int argc, const char *argv[])
1.439 +{
1.440 + struct razor_set *set;
1.441 + struct razor_package_iterator *pi;
1.442 + struct razor_package *package;
1.443 + const char *name, *version, *arch;
1.444 +
1.445 + if (option_package) {
1.446 + set = create_set_from_command_line(argc, argv);
1.447 + argc = 0;
1.448 + option_all = 1;
1.449 + } else {
1.450 + set = razor_root_open_read_only(option_root);
1.451 + }
1.452 +
1.453 + pi = get_query_packages(set, argc, argv);
1.454 +
1.455 + while (razor_package_iterator_next(pi, &package,
1.456 + &name, &version, &arch)) {
1.457 + if (option_conflicts)
1.458 + print_package_properties(set, package,
1.459 + RAZOR_PROPERTY_CONFLICTS);
1.460 + if (option_obsoletes)
1.461 + print_package_properties(set, package,
1.462 + RAZOR_PROPERTY_OBSOLETES);
1.463 + if (option_requires)
1.464 + print_package_properties(set, package,
1.465 + RAZOR_PROPERTY_REQUIRES);
1.466 + if (option_provides)
1.467 + print_package_properties(set, package,
1.468 + RAZOR_PROPERTY_PROVIDES);
1.469 + if (option_info)
1.470 + print_package_info(set, package);
1.471 + if (option_changelog)
1.472 + print_package_changelog(set, package);
1.473 +
1.474 + if (option_conflicts + option_obsoletes +
1.475 + option_requires + option_provides +
1.476 + option_info + option_changelog == 0)
1.477 + printf("%s-%s.%s\n", name, version, arch);
1.478 + }
1.479 +
1.480 + razor_package_iterator_destroy(pi);
1.481 +
1.482 + razor_set_destroy(set);
1.483 +
1.484 + return;
1.485 +}
1.486 +
1.487 +static void
1.488 +command_verify(int argc, const char *argv[])
1.489 +{
1.490 + struct razor_set *set;
1.491 + struct razor_package_iterator *pi;
1.492 + struct razor_package *package;
1.493 + const char *name, *version, *arch;
1.494 +
1.495 + if (option_package) {
1.496 + set = create_set_from_command_line(argc, argv);
1.497 + argc = 0;
1.498 + option_all = 1;
1.499 + } else {
1.500 + set = razor_root_open_read_only(option_root);
1.501 + }
1.502 +
1.503 + pi = get_query_packages(set, argc, argv);
1.504 +
1.505 + while (razor_package_iterator_next(pi, &package,
1.506 + &name, &version, &arch)) {
1.507 + printf("verify %s-%s.%s - not implemented\n",
1.508 + name, version, arch);
1.509 + }
1.510 +
1.511 + razor_package_iterator_destroy(pi);
1.512 +}
1.513 +
1.514 +static void
1.515 +remove_package(const char *name,
1.516 + const char *old_version, const char *new_version,
1.517 + const char *arch, void *data)
1.518 +{
1.519 + if (old_version)
1.520 + printf("remove %s-%s.%s\n", name, old_version, arch);
1.521 +}
1.522 +
1.523 +static void
1.524 +command_erase(int argc, const char *argv[])
1.525 +{
1.526 + struct razor_set *set, *upstream, *next;
1.527 + struct razor_transaction *trans;
1.528 + struct razor_package_query *query;
1.529 + struct razor_package_iterator *pi;
1.530 + struct razor_package *package;
1.531 + const char *name, *version, *arch;
1.532 +
1.533 + if (argc == 0) {
1.534 + printf("no packages given for erase\n");
1.535 + exit(1);
1.536 + }
1.537 +
1.538 + set = razor_set_open(repo_filename);
1.539 + upstream = razor_set_create();
1.540 +
1.541 + trans = razor_transaction_create(set, upstream);
1.542 +
1.543 + query = razor_package_query_create(set);
1.544 + add_command_line_packages(set, query, argc, argv);
1.545 +
1.546 + pi = razor_package_query_finish(query);
1.547 + while (razor_package_iterator_next(pi, &package,
1.548 + &name, &version, &arch))
1.549 + razor_transaction_remove_package(trans, package);
1.550 + razor_package_iterator_destroy(pi);
1.551 +
1.552 + if (!option_nodeps && razor_transaction_describe(trans) > 0) {
1.553 + printf("unsatisfied dependencies.\n");
1.554 + exit(1);
1.555 + }
1.556 +
1.557 + if (option_test)
1.558 + exit(0);
1.559 +
1.560 + next = razor_transaction_finish(trans);
1.561 +
1.562 + if (!option_justdb)
1.563 + razor_set_diff(set, next, remove_package, NULL);
1.564 +
1.565 + razor_set_destroy(set);
1.566 + razor_set_destroy(upstream);
1.567 +
1.568 + razor_set_destroy(next);
1.569 +}
1.570 +
1.571 +static void
1.572 +install_package(const char *name,
1.573 + const char *old_version, const char *new_version,
1.574 + const char *arch, void *data)
1.575 +{
1.576 + if (new_version)
1.577 + printf("install %s-%s.%s\n", name, new_version, arch);
1.578 +}
1.579 +
1.580 +static void
1.581 +command_install(int argc, const char *argv[])
1.582 +{
1.583 + struct razor_set *set, *upstream, *next;
1.584 + struct razor_transaction *trans;
1.585 + struct razor_package_iterator *pi;
1.586 + struct razor_package *package;
1.587 + const char *name, *version, *arch;
1.588 +
1.589 + if (argc == 0) {
1.590 + printf("no packages given for install\n");
1.591 + exit(1);
1.592 + }
1.593 +
1.594 + set = razor_set_open(repo_filename);
1.595 + upstream = create_set_from_command_line(argc, argv);
1.596 +
1.597 + trans = razor_transaction_create(set, upstream);
1.598 +
1.599 + pi = razor_package_iterator_create(upstream);
1.600 + while (razor_package_iterator_next(pi, &package,
1.601 + &name, &version, &arch))
1.602 + razor_transaction_install_package(trans, package);
1.603 + razor_package_iterator_destroy(pi);
1.604 +
1.605 + if (!option_nodeps && razor_transaction_describe(trans) > 0) {
1.606 + printf("unsatisfied dependencies.\n");
1.607 + exit(1);
1.608 + }
1.609 +
1.610 + if (option_test)
1.611 + exit(0);
1.612 +
1.613 + next = razor_transaction_finish(trans);
1.614 +
1.615 + if (!option_justdb)
1.616 + razor_set_diff(set, next, install_package, NULL);
1.617 +
1.618 + razor_set_destroy(set);
1.619 + razor_set_destroy(upstream);
1.620 +
1.621 + razor_set_destroy(next);
1.622 +}
1.623 +
1.624 +static void
1.625 +update_package(const char *name,
1.626 + const char *old_version, const char *new_version,
1.627 + const char *arch, void *data)
1.628 +{
1.629 + if (old_version)
1.630 + printf("remove %s-%s.%s\n", name, old_version, arch);
1.631 + if (new_version)
1.632 + printf("install %s-%s.%s\n", name, new_version, arch);
1.633 +}
1.634 +
1.635 +static void
1.636 +command_update(int argc, const char *argv[])
1.637 +{
1.638 + struct razor_set *set, *upstream, *next;
1.639 + struct razor_transaction *trans;
1.640 + struct razor_package_iterator *pi;
1.641 + struct razor_package *package;
1.642 + const char *name, *version, *arch;
1.643 +
1.644 + if (argc == 0) {
1.645 + printf("no packages given for update\n");
1.646 + exit(1);
1.647 + }
1.648 +
1.649 + set = razor_set_open(repo_filename);
1.650 + upstream = create_set_from_command_line(argc, argv);
1.651 +
1.652 + trans = razor_transaction_create(set, upstream);
1.653 +
1.654 + pi = razor_package_iterator_create(upstream);
1.655 + while (razor_package_iterator_next(pi, &package,
1.656 + &name, &version, &arch))
1.657 + razor_transaction_update_package(trans, package);
1.658 + razor_package_iterator_destroy(pi);
1.659 +
1.660 + if (!option_nodeps && razor_transaction_describe(trans) > 0) {
1.661 + printf("unsatisfied dependencies.\n");
1.662 + exit(1);
1.663 + }
1.664 +
1.665 + if (option_test)
1.666 + exit(0);
1.667 +
1.668 + next = razor_transaction_finish(trans);
1.669 +
1.670 + if (!option_justdb)
1.671 + razor_set_diff(set, next, update_package, NULL);
1.672 +
1.673 + razor_set_destroy(set);
1.674 + razor_set_destroy(upstream);
1.675 +
1.676 + razor_set_destroy(next);
1.677 +}
1.678 +
1.679 +static int
1.680 +for_each_option(const struct option *options,
1.681 + const char *name, char short_name,
1.682 + void (*fn)(const struct option *o,
1.683 + const char *name, char short_name,
1.684 + void *data), void *data)
1.685 +{
1.686 + int i, count = 0;
1.687 +
1.688 + for (i = 0; options[i].type != OPTION_LAST; i++) {
1.689 + switch (options[i].type) {
1.690 + case OPTION_GROUP:
1.691 + count += for_each_option(options[i].data,
1.692 + name, short_name, fn, data);
1.693 + break;
1.694 +
1.695 + case OPTION_BOOL:
1.696 + case OPTION_STRING:
1.697 + if (name && strcmp(options[i].name, name) == 0) {
1.698 + fn(&options[i], name, 0, data);
1.699 + count++;
1.700 + break;
1.701 + }
1.702 +
1.703 + if (short_name &&
1.704 + short_name == options[i].short_name) {
1.705 + fn(&options[i], NULL, short_name, data);
1.706 + count++;
1.707 + break;
1.708 + }
1.709 + break;
1.710 +
1.711 + case OPTION_LAST:
1.712 + break;
1.713 + }
1.714 + }
1.715 +
1.716 + return count;
1.717 +}
1.718 +
1.719 +static void
1.720 +handle_option(const struct option *o,
1.721 + const char *name, char short_name, void *data)
1.722 +{
1.723 + if (o->data == NULL) {
1.724 + if (name)
1.725 + printf("option --%s not supported\n", name);
1.726 + else
1.727 + printf("option -%c not supported\n", short_name);
1.728 + return;
1.729 + }
1.730 +
1.731 + switch (o->type) {
1.732 + case OPTION_BOOL:
1.733 + *(int *) o->data = 1;
1.734 + break;
1.735 +
1.736 + case OPTION_STRING:
1.737 + *(const char **) o->data = name + strlen(o->name) + 1;
1.738 + break;
1.739 +
1.740 + case OPTION_LAST:
1.741 + case OPTION_GROUP:
1.742 + /* Shouldn't happen. */
1.743 + break;
1.744 + }
1.745 +}
1.746 +
1.747 +static int
1.748 +parse_options(const struct option *options, int argc, const char **argv)
1.749 +{
1.750 + int i, j, k;
1.751 +
1.752 + for (i = 1, j = 0; i < argc; i++) {
1.753 + if (argv[i][0] != '-') {
1.754 + argv[j++] = argv[i];
1.755 + continue;
1.756 + }
1.757 +
1.758 + if (argv[i][1] == '-') {
1.759 + if (for_each_option(options, &argv[i][2], 0,
1.760 + handle_option, NULL) == 0) {
1.761 + printf("unknown option: %s\n", argv[i]);
1.762 + exit(1);
1.763 + }
1.764 + continue;
1.765 + }
1.766 +
1.767 + for (k = 1; argv[i][k]; k++) {
1.768 + if (for_each_option(options, NULL, argv[i][k],
1.769 + handle_option, NULL) == 0) {
1.770 + printf("unknown option: -%c\n", argv[i][k]);
1.771 + exit(1);
1.772 + }
1.773 + }
1.774 + }
1.775 +
1.776 + return j;
1.777 +}
1.778 +
1.779 +static void
1.780 +print_options_help(const struct option *options)
1.781 +{
1.782 + int i;
1.783 +
1.784 + for (i = 0; options[i].type != OPTION_LAST; i++) {
1.785 + switch (options[i].type) {
1.786 + case OPTION_GROUP:
1.787 + printf("%s\n", options[i].description);
1.788 + print_options_help(options[i].data);
1.789 + printf("\n");
1.790 + break;
1.791 +
1.792 + case OPTION_BOOL:
1.793 + case OPTION_STRING:
1.794 + printf(" ");
1.795 + if (options[i].short_name)
1.796 + printf("-%c", options[i].short_name);
1.797 + if (options[i].short_name && options[i].name)
1.798 + printf(", ");
1.799 + if (options[i].name)
1.800 + printf("--%s", options[i].name);
1.801 + if (options[i].arg_name)
1.802 + printf("=%s", options[i].arg_name);
1.803 + if (options[i].description)
1.804 + printf("\t\t%s", options[i].description);
1.805 + printf("\n");
1.806 + break;
1.807 +
1.808 + case OPTION_LAST:
1.809 + break;
1.810 + }
1.811 + }
1.812 +}
1.813 +
1.814 +static void
1.815 +print_options_usage(const struct option *options)
1.816 +{
1.817 + int i;
1.818 +
1.819 + for (i = 0; options[i].type != OPTION_LAST; i++) {
1.820 + switch (options[i].type) {
1.821 + case OPTION_GROUP:
1.822 + print_options_usage(options[i].data);
1.823 + break;
1.824 +
1.825 + case OPTION_BOOL:
1.826 + printf("[");
1.827 + if (options[i].short_name)
1.828 + printf("-%c", options[i].short_name);
1.829 + if (options[i].short_name && options[i].name)
1.830 + printf("|");
1.831 + if (options[i].name)
1.832 + printf("--%s", options[i].name);
1.833 + printf("] ");
1.834 + break;
1.835 +
1.836 + case OPTION_STRING:
1.837 + printf("[");
1.838 + if (options[i].short_name)
1.839 + printf("-%c", options[i].short_name);
1.840 + if (options[i].short_name && options[i].name)
1.841 + printf("|");
1.842 + if (options[i].name)
1.843 + printf("--%s", options[i].name);
1.844 + if (options[i].arg_name)
1.845 + printf("=%s", options[i].arg_name);
1.846 + printf("] ");
1.847 + break;
1.848 +
1.849 +
1.850 + break;
1.851 +
1.852 + case OPTION_LAST:
1.853 + break;
1.854 + }
1.855 + }
1.856 +}
1.857 +
1.858 +int
1.859 +main(int argc, const char *argv[])
1.860 +{
1.861 + argc = parse_options(rpm_options, argc, argv);
1.862 +
1.863 + if (option_version) {
1.864 + printf("razor rpm version hoopla.\n");
1.865 + exit(0);
1.866 + }
1.867 +
1.868 + if (option_help) {
1.869 + printf("Usage: rpm [OPTION...]\n");
1.870 + print_options_help(rpm_options);
1.871 + exit(0);
1.872 + }
1.873 +
1.874 + if (option_usage) {
1.875 + printf("Usage: rpm [OPTION...]\n");
1.876 + print_options_usage(rpm_options);
1.877 + printf("\n");
1.878 + exit(0);
1.879 + }
1.880 +
1.881 + if (option_initdb) {
1.882 + command_initdb(argc, argv);
1.883 + } else if (option_verify) {
1.884 + command_verify(argc, argv);
1.885 + } else if (option_query) {
1.886 + command_query(argc, argv);
1.887 + } else if (option_install) {
1.888 + command_install(argc, argv);
1.889 + } else if (option_upgrade) {
1.890 + command_update(argc, argv);
1.891 + } else if (option_erase) {
1.892 + command_erase(argc, argv);
1.893 + } else {
1.894 + print_options_usage(rpm_options);
1.895 + printf("\n");
1.896 + exit(0);
1.897 + }
1.898 +
1.899 + return 0;
1.900 +}