src/rpm.c
author James Bowes <jbowes@redhat.com>
Mon Jun 23 20:09:56 2008 -0400 (2008-06-23)
changeset 275 1862b95f4cd6
parent 266 9b652c3617d9
child 276 4b0a1201342a
permissions -rw-r--r--
Implement rpm --query --list
     1 /*
     2  * Copyright (C) 2008  Kristian Høgsberg <krh@redhat.com>
     3  * Copyright (C) 2008  Red Hat, Inc
     4  *
     5  * This program is free software; you can redistribute it and/or modify
     6  * it under the terms of the GNU General Public License as published by
     7  * the Free Software Foundation; either version 2 of the License, or
     8  * (at your option) any later version.
     9  *
    10  * This program is distributed in the hope that it will be useful,
    11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    13  * GNU General Public License for more details.
    14  *
    15  * You should have received a copy of the GNU General Public License along
    16  * with this program; if not, write to the Free Software Foundation, Inc.,
    17  * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
    18  */
    19 
    20 #include <stdlib.h>
    21 #include <string.h>
    22 #include <stdio.h>
    23 #include <stdint.h>
    24 #include <dirent.h>
    25 #include "razor.h"
    26 
    27 enum option_type {
    28 	OPTION_LAST,
    29 	OPTION_GROUP,
    30 	OPTION_BOOL,
    31 	OPTION_STRING
    32 };
    33 
    34 struct option {
    35 	enum option_type type;
    36 	const char *name;
    37 	char short_name;
    38 	const char *arg_name;
    39 	const char *description;
    40 	void *data;
    41 };
    42 
    43 /* A note about all these options: rpm allows options to mean
    44  * different things depending on what other options are present on the
    45  * command line.  For example, if -q or --query is present, -i no
    46  * longer means install, but info.  The way we handle this is by
    47  * setting all the options that may match (ie if -i is given we set
    48  * install and info), and then look at the relevent one depending on
    49  * what else in on the command line. */
    50 
    51 static int option_all, option_list, option_whatrequires, option_whatprovides;
    52 static int option_package;
    53 
    54 static const struct option query_options[] = {
    55 	{ OPTION_BOOL, "configfiles", 'c', NULL, "list all configuration files", NULL },
    56 	{ OPTION_BOOL, "docfiles", 'd', NULL, "list all documentation files", NULL },
    57 	{ OPTION_BOOL, "dump", 0, NULL, "dump basic file information", NULL },
    58 	{ OPTION_BOOL, "list", 'l', NULL, "list files in package", &option_list },
    59 	{ OPTION_STRING, "queryformat", 0, "QUERYFORMAT", "use the following query format", NULL },
    60 	{ OPTION_BOOL, "state", 's', NULL, "display the states of the listed files", NULL },
    61 	{ OPTION_BOOL, "all", 'a', NULL, "query/verify all packages", &option_all },
    62 	{ OPTION_BOOL, "file", 'f', NULL, "query/verify package(s) owning file", NULL },
    63 	{ OPTION_BOOL, "group", 'g', NULL, "query/verify package(s) in group", NULL },
    64 	{ OPTION_BOOL, "package", 'p', NULL, "query/verify a package file", &option_package },
    65 	{ OPTION_BOOL, "ftswalk", 'W', NULL, "query/verify package(s) from TOP file tree walk", NULL },
    66 	{ OPTION_BOOL, "pkgid", 0, NULL, "query/verify package(s) with package identifier", NULL },
    67 	{ OPTION_BOOL, "hdrid", 0, NULL, "query/verify package(s) with header identifier", NULL },
    68 	{ OPTION_BOOL, "fileid", 0, NULL, "query/verify package(s) with file identifier", NULL },
    69 	{ OPTION_BOOL, "specfile", 0, NULL, "query a spec file", NULL, },
    70 	{ OPTION_BOOL, "triggeredby", 0, NULL, "query the package(s) triggered by the package", NULL },
    71 	{ OPTION_BOOL, "whatrequires", 0, NULL, "query/verify the package(s) which require a dependency", &option_whatrequires },
    72 	{ OPTION_BOOL, "whatprovides", 0, NULL, "query/verify the package(s) which provide a dependency", &option_whatprovides },
    73 	{ OPTION_BOOL, "nomanifest", 0, NULL, "do not process non-package files as manifests", NULL },
    74 	{ }
    75 };
    76 
    77 static int option_nodeps;
    78 
    79 static const struct option verify_options[] = {
    80 	{ OPTION_BOOL, "nomd5", 0, NULL, "don't verify MD5 digest of files", NULL },
    81 	{ OPTION_BOOL, "nofiles", 0, NULL, "don't verify files in package", NULL },
    82 	{ OPTION_BOOL, "nodeps", 0, NULL, "don't verify package dependencies", &option_nodeps },
    83 	{ OPTION_BOOL, "noscript", 0, NULL, "don't execute verify script(s)", NULL, },
    84 	{ OPTION_BOOL, "all", 'a', NULL, "query/verify all packages", &option_all },
    85 	{ OPTION_BOOL, "file", 'f', NULL, "query/verify package(s) owning file", NULL },
    86 	{ OPTION_BOOL, "group", 'g', NULL, "query/verify package(s) in group", NULL },
    87 	{ OPTION_BOOL, "package", 'p', NULL, "query/verify a package file", &option_package },
    88 	{ OPTION_BOOL, "ftswalk", 'W', NULL, "query/verify package(s) from TOP file tree walk", NULL },
    89 	{ OPTION_BOOL, "pkgid", 0, NULL, "query/verify package(s) with package identifier", NULL },
    90 	{ OPTION_BOOL, "hdrid", 0, NULL, "query/verify package(s) with header identifier", NULL },
    91 	{ OPTION_BOOL, "fileid", 0, NULL, "query/verify package(s) with file identifier", NULL },
    92 	{ OPTION_BOOL, "specfile", 0, NULL, "query a spec file", NULL },
    93 	{ OPTION_BOOL, "triggeredby", 0, NULL, "query the package(s) triggered by the package", NULL },
    94 	{ OPTION_BOOL, "whatrequires", 0, NULL, "query/verify the package(s) which require a dependency", &option_whatrequires },
    95 	{ OPTION_BOOL, "whatprovides", 0, NULL, "query/verify the package(s) which provide a dependency", &option_whatprovides },
    96 	{ OPTION_BOOL, "nomanifest", 0, NULL, "do not process non-package files as manifests", NULL },
    97 	{ }
    98 };
    99 
   100 static const struct option ftw_options[] = {
   101 	{ OPTION_BOOL, "comfollow", 0, NULL, "FTS_COMFOLLOW: follow command line symlinks", NULL },
   102 	{ OPTION_BOOL, "logical", 0, NULL, "FTS_LOGICAL: logical walk", NULL },
   103 	{ OPTION_BOOL, "nochdir", 0, NULL, "FTS_NOCHDIR: don't change directories", NULL },
   104 	{ OPTION_BOOL, "nostat", 0, NULL, "FTS_NOSTAT: don't get stat info", NULL },
   105 	{ OPTION_BOOL, "physical", 0, NULL, "FTS_PHYSICAL: physical walk", NULL },
   106 	{ OPTION_BOOL, "seedot", 0, NULL, "FTS_SEEDOT: return dot and dot-dot", NULL },
   107 	{ OPTION_BOOL, "xdev", 0, NULL, "FTS_XDEV: don't cross devices", NULL },
   108 	{ OPTION_BOOL, "whiteout", 0, NULL, "FTS_WHITEOUT: return whiteout information", NULL },
   109 	{ }
   110 };
   111 
   112 static const struct option signature_options[] = {
   113 	{ OPTION_BOOL, "addsign", 0, NULL, "sign package(s) (identical to --resign)", NULL, },
   114 	{ OPTION_BOOL, "checksig", 'K', NULL, "verify package signature(s)", NULL, },
   115 	{ OPTION_BOOL, "delsign", 0, NULL, "delete package signatures", NULL, },
   116 	{ OPTION_BOOL, "import", 0, NULL, "import an armored public key", NULL, },
   117 	{ OPTION_BOOL, "resign", 0, NULL, "sign package(s) (identical to --addsign)", NULL, },
   118 	{ OPTION_BOOL, "nodigest", 0, NULL, "don't verify package digest(s)", NULL, },
   119 	{ OPTION_BOOL, "nosignature", 0, NULL, "don't verify package signature(s)", NULL },
   120 	{ }
   121 };
   122 
   123 static int option_initdb;
   124 
   125 static const struct option database_options[] = {
   126 	{ OPTION_BOOL, "initdb", 0, NULL, "initialize database", &option_initdb },
   127 	{ OPTION_BOOL, "rebuilddb", 0, NULL, "rebuild database inverted lists from installed package headers", NULL },
   128 	{ }
   129 };
   130 
   131 static int option_erase, option_install, option_upgrade, option_justdb;
   132 static int option_test;
   133 
   134 static const struct option install_options[] = {
   135 	{ OPTION_BOOL, "aid", 0, NULL, "add suggested packages to transaction", NULL, },
   136 	{ OPTION_BOOL, "allfiles", 0, NULL, "install all files, even configurations which might otherwise be skipped", NULL, },
   137 	{ OPTION_BOOL, "allmatches", 0, NULL, "remove all packages which match <package> (normally an error is generated if <package> specified multiple packages)", NULL, },
   138 	{ OPTION_BOOL, "badreloc", 0, NULL, "relocate files in non-relocatable package", NULL },
   139 	{ OPTION_BOOL, "erase", 'e', "<package>", "erase (uninstall) package", &option_erase },
   140 	{ OPTION_BOOL, "excludedocs", 0, NULL, "do not install documentation", NULL, },
   141 	{ OPTION_BOOL, "excludepath", 0, "<path>", "skip files with leading component <path> ", NULL, },
   142 	{ OPTION_BOOL, "fileconflicts", 0, NULL, "detect file conflicts between packages", NULL, },
   143 	{ OPTION_BOOL, "force", 0, NULL, "short hand for --replacepkgs --replacefiles", NULL },
   144 	{ OPTION_BOOL, "freshen", 'F', "<packagefile>+", "upgrade package(s) if already installed", NULL },
   145 	{ OPTION_BOOL, "hash", 'h', NULL, "print hash marks as package installs (good with -v)", NULL },
   146 	{ OPTION_BOOL, "ignorearch", 0, NULL, "don't verify package architecture", NULL, },
   147 	{ OPTION_BOOL, "ignoreos", 0, NULL, "don't verify package operating system", NULL, },
   148 	{ OPTION_BOOL, "ignoresize", 0, NULL, "don't check disk space before installing", NULL },
   149 	{ OPTION_BOOL, "install", 'i', NULL, "install package(s)", &option_install },
   150 	{ OPTION_BOOL, "justdb", 0, NULL, "update the database, but do not modify the filesystem", &option_justdb, },
   151 	{ OPTION_BOOL, "nodeps", 0, NULL, "do not verify package dependencies", &option_nodeps, },
   152 	{ OPTION_BOOL, "nomd5", 0, NULL, "don't verify MD5 digest of files", NULL, },
   153 	{ OPTION_BOOL, "nocontexts", 0, NULL, "don't install file security contexts", NULL, },
   154 	{ OPTION_BOOL, "noorder", 0, NULL, "do not reorder package installation to satisfy dependencies", NULL, },
   155 	{ OPTION_BOOL, "nosuggest", 0, NULL, "do not suggest missing dependency resolution(s)", NULL, },
   156 	{ OPTION_BOOL, "noscripts", 0, NULL, "do not execute package scriptlet(s)", NULL, },
   157 	{ OPTION_BOOL, "notriggers", 0, NULL, "do not execute any scriptlet(s) triggered by this package", NULL, },
   158 	{ OPTION_BOOL, "oldpackage", 0, NULL, "upgrade to an old version of the package (--force on upgrades does this automatically)", NULL },
   159 	{ OPTION_BOOL, "percent", 0, NULL, "print percentages as package installs", NULL, },
   160 	{ OPTION_STRING, "prefix", 0, "<dir>", "relocate the package to <dir>, if relocatable", NULL, },
   161 	{ OPTION_STRING, "relocate", 0, "<old>=<new>", "relocate files from path <old> to <new>", NULL, },
   162 	{ OPTION_BOOL, "repackage", 0, NULL, "save erased package files by repackaging", NULL, },
   163 	{ OPTION_BOOL, "replacefiles", 0, NULL, "ignore file conflicts between packages", NULL, },
   164 	{ OPTION_BOOL, "replacepkgs", 0, NULL, "reinstall if the package is already present", NULL, },
   165 	{ OPTION_BOOL, "test", 0, NULL, "don't install, but tell if it would work or not", &option_test },
   166 	{ OPTION_BOOL, "upgrade", 'U', "<packagefile>+", "upgrade package(s)", &option_upgrade },
   167 	{ }
   168 };
   169 
   170 static int option_version;
   171 static const char *option_root = "install";
   172 
   173 static const struct option common_options[] = {
   174 	{ OPTION_STRING, "define", 'D', "MACRO EXPR", "define MACRO with value EXPR", NULL, },
   175 	{ OPTION_STRING, "eval", 'E', "EXPR", "print macro expansion of EXPR", NULL },
   176 	{ OPTION_STRING, "macros", 0, "<FILE:...>", "read <FILE:...> instead of default file(s)", NULL },
   177 	{ OPTION_BOOL, "nodigest", 0, NULL, "don't verify package digest(s)", NULL, },
   178 	{ OPTION_BOOL, "nosignature", 0, NULL, "don't verify package signature(s)", NULL, },
   179 	{ OPTION_STRING, "rcfile", 0, "<FILE:...>", "read <FILE:...> instead of default file(s)", NULL },
   180 	{ OPTION_STRING, "root", 'r', "ROOT", "use ROOT as top level directory (default: \"/\")", &option_root },
   181 	{ OPTION_BOOL, "querytags", 0, NULL, "display known query tags", NULL, },
   182 	{ OPTION_BOOL, "showrc", 0, NULL, "display final rpmrc and macro configuration", NULL, },
   183 	{ OPTION_BOOL, "quiet", 0, NULL, "provide less detailed output", NULL },
   184 	{ OPTION_BOOL, "verbose", 'v', NULL, "provide more detailed output", NULL },
   185 	{ OPTION_BOOL, "version", 0, NULL, "print the version of rpm being used", &option_version },
   186 	{ }
   187 };
   188 
   189 static int option_conflicts, option_obsoletes, option_requires;
   190 static int option_provides, option_info, option_changelog;
   191 
   192 static const struct option alias_options[] = {
   193 	{ OPTION_BOOL, "scripts", 0, NULL, "list install/erase scriptlets from package(s)", NULL, },
   194 	{ OPTION_BOOL, "setperms", 0, NULL, "set permissions of files in a package", NULL, },
   195 	{ OPTION_BOOL, "setugids", 0, NULL, "set user/group ownership of files in a package", NULL, },
   196 	{ OPTION_BOOL, "conflicts", 0, NULL, "list capabilities this package conflicts with", &option_conflicts, },
   197 	{ OPTION_BOOL, "obsoletes", 0, NULL, "list other packages removed by installing this package", &option_obsoletes, },
   198 	{ OPTION_BOOL, "provides", 0, NULL, "list capabilities that this package provides", &option_provides, },
   199 	{ OPTION_BOOL, "requires", 0, NULL, "list capabilities required by package(s)", &option_requires, },
   200 	{ OPTION_BOOL, "info", 'i', NULL, "list descriptive information from package(s)", &option_info, },
   201 	{ OPTION_BOOL, "changelog", 0, NULL, "list change logs for this package", &option_changelog, },
   202 	{ OPTION_BOOL, "xml", 0, NULL, "list metadata in xml", NULL, },
   203 	{ OPTION_BOOL, "triggers", 0, NULL, "list trigger scriptlets from package(s)", NULL, },
   204 	{ OPTION_BOOL, "last", 0, NULL, "list package(s) by install time, most recent first", NULL, },
   205 	{ OPTION_BOOL, "dupes", 0, NULL, "list duplicated packages", NULL, },
   206 	{ OPTION_BOOL, "filesbypkg", 0, NULL, "list all files from each package", NULL, },
   207 	{ OPTION_BOOL, "fileclass", 0, NULL, "list file names with classes", NULL, },
   208 	{ OPTION_BOOL, "filecolor", 0, NULL, "list file names with colors", NULL, },
   209 	{ OPTION_BOOL, "filecontext", 0, NULL, "list file names with security context from header", NULL, },
   210 	{ OPTION_BOOL, "fscontext", 0, NULL, "list file names with security context from file system", NULL, },
   211 	{ OPTION_BOOL, "recontext", 0, NULL, "list file names with security context from policy RE", NULL, },
   212 	{ OPTION_BOOL, "fileprovide", 0, NULL, "list file names with provides", NULL, },
   213 	{ OPTION_BOOL, "filerequire", 0, NULL, "list file names with requires", NULL, },
   214 	{ OPTION_BOOL, "redhatprovides", 0, NULL, "find package name that contains a provided capability (needs rpmdb-redhat package installed)", NULL, },
   215 	{ OPTION_BOOL, "redhatrequires", 0, NULL, "find package name that contains a required capability (needs rpmdb-redhat package installed)", NULL, },
   216 	{ OPTION_STRING, "buildpolicy", 0, "<policy>", "set buildroot <policy> (e.g. compress man pages)", NULL, },
   217 	{ OPTION_BOOL, "with", 0, "<option>", "enable configure <option> for build", NULL, },
   218 	{ OPTION_BOOL, "without", 0, "<option>", "disable configure <option> for build", NULL },
   219 	{ }
   220 };
   221 
   222 static int option_help, option_usage;
   223 
   224 static const struct option help_options[] = {
   225 	{ OPTION_BOOL, "help", '?', NULL, "Show this help message", &option_help },
   226 	{ OPTION_BOOL, "usage", 0, NULL, "Display brief usage message", &option_usage},
   227 	{ }
   228 };
   229 
   230 static int option_query, option_verify;
   231 
   232 static const struct option rpm_options[] = {
   233 	{ OPTION_BOOL, "query", 'q', NULL, "Query rpm database", &option_query },
   234 	{ OPTION_BOOL, "verify", 'V', NULL, "Verify rpm database", &option_verify },
   235 	{ OPTION_GROUP, NULL, 0, NULL, "Query options (with -q or --query):", &query_options },
   236 	{ OPTION_GROUP, NULL, 0, NULL, "Verify options (with -V or --verify):", &verify_options },
   237 	{ OPTION_GROUP, NULL, 0, NULL, "File tree walk options (with --ftswalk):", &ftw_options },
   238 	{ OPTION_GROUP, NULL, 0, NULL, "Signature options:", &signature_options },
   239 	{ OPTION_GROUP, NULL, 0, NULL, "Database options:", &database_options },
   240 	{ OPTION_GROUP, NULL, 0, NULL, "Install/Upgrade/Erase options:", &install_options },
   241 	{ OPTION_GROUP, NULL, 0, NULL, "Common options for all rpm modes and executables:", &common_options },
   242 	{ OPTION_GROUP, NULL, 0, NULL, "Options implemented via popt alias/exec:", &alias_options },
   243 	{ OPTION_GROUP, NULL, 0, NULL, "Help options", &help_options },
   244 	{ }
   245 };
   246 
   247 static const char system_repo_filename[] = "system.repo";
   248 static const char *repo_filename = system_repo_filename;
   249 
   250 static void
   251 command_initdb(int argc, const char *argv[])
   252 {
   253 	razor_root_create(option_root);
   254 }
   255 
   256 static struct razor_property *
   257 add_property_packages(struct razor_set *set,
   258 		      struct razor_package_query *query,
   259 		      const char *ref_name,
   260 		      const char *ref_version,
   261 		      uint32_t ref_type)
   262 {
   263 	struct razor_property *property;
   264 	struct razor_property_iterator *pi;
   265 	struct razor_package_iterator *pkgi;
   266 	const char *name, *version;
   267 	uint32_t flags;
   268 
   269 	pi = razor_property_iterator_create(set, NULL);
   270 	while (razor_property_iterator_next(pi, &property, &name,
   271 					    &flags, &version)) {
   272 		if (strcmp(ref_name, name) != 0)
   273 			continue;
   274 		if (ref_version &&
   275 		    (flags & RAZOR_PROPERTY_RELATION_MASK) == RAZOR_PROPERTY_EQUAL &&
   276 		    strcmp(ref_version, version) != 0)
   277 			continue;
   278 		if ((flags & RAZOR_PROPERTY_TYPE_MASK) != ref_type)
   279 			continue;
   280 
   281 		pkgi = razor_package_iterator_create_for_property(set,
   282 								  property);
   283 		razor_package_query_add_iterator(query, pkgi);
   284 		razor_package_iterator_destroy(pkgi);
   285 	}
   286 	razor_property_iterator_destroy(pi);
   287 
   288 	return property;
   289 }
   290 
   291 static int
   292 strcmpp(const void *p1, const void *p2)
   293 {
   294 	return strcmp(*(char * const *) p1, *(char * const *) p2);
   295 }
   296 
   297 static void
   298 add_command_line_packages(struct razor_set *set,
   299 			  struct razor_package_query *query,
   300 			  int argc, const char **argv)
   301 {
   302 	struct razor_package *package;
   303 	struct razor_package_iterator *pi;
   304 	const char *name, *version, *arch;
   305 	int i, cmp, errors;
   306 
   307 	qsort(argv, argc, sizeof(*argv), strcmpp);
   308 	i = 0;
   309 	errors = 0;
   310 
   311 	pi = razor_package_iterator_create(set);
   312 
   313 	while (razor_package_iterator_next(pi, &package,
   314 					   &name, &version, &arch)) {
   315 		while (cmp = strcmp(argv[i], name), cmp < 0 && i < argc) {
   316 			fprintf(stderr, "error: package %s is not installed\n",
   317 				argv[i]);
   318 			errors++;
   319 			i++;
   320 		}
   321 
   322 		if (cmp == 0) {
   323 			razor_package_query_add_package(query, package);
   324 			i++;
   325 		}
   326 	}
   327 
   328 	razor_package_iterator_destroy(pi);
   329 
   330 	if (errors)
   331 		exit(1);
   332 }
   333 
   334 static struct razor_package_iterator *
   335 get_query_packages(struct razor_set *set, int argc, const char *argv[])
   336 {
   337 	struct razor_package_query *query;
   338 	struct razor_package_iterator *pi;
   339 	int i;
   340 
   341 	if (option_all + option_whatprovides + option_whatrequires > 1) {
   342 		printf("only one type of query/verify "
   343 		       "may be performed at a time\n");
   344 		exit(1);
   345 	}
   346 
   347 	query = razor_package_query_create(set);
   348 
   349 	if (option_all) {
   350 		pi = razor_package_iterator_create(set);
   351 		razor_package_query_add_iterator(query, pi);
   352 		razor_package_iterator_destroy(pi);
   353 	} else if (option_whatrequires) {
   354 		for (i = 0; i < argc; i++)
   355 			add_property_packages(set, query,
   356 					      argv[i], NULL,
   357 					      RAZOR_PROPERTY_REQUIRES);
   358 	} else if (option_whatprovides) {
   359 		for (i = 0; i < argc; i++)
   360 			add_property_packages(set, query,
   361 					      argv[i], NULL,
   362 					      RAZOR_PROPERTY_PROVIDES);
   363 	} else if (argc > 0) {
   364 		add_command_line_packages(set, query, argc, argv);
   365 	} else {
   366 		printf("no arguments given for query/verify\n");
   367 		exit(1);
   368 	}
   369 
   370 	return razor_package_query_finish(query);
   371 }
   372 
   373 static void
   374 print_package_properties(struct razor_set *set,
   375 			 struct razor_package *package,
   376 			 uint32_t ref_type)
   377 {
   378 	struct razor_property *property;
   379 	struct razor_property_iterator *pi;
   380 	const char *name, *version;
   381 	uint32_t flags;
   382 
   383 	pi = razor_property_iterator_create(set, package);
   384 	while (razor_property_iterator_next(pi, &property,
   385 					    &name, &flags, &version)) {
   386 		if ((flags & RAZOR_PROPERTY_TYPE_MASK) != ref_type)
   387 			continue;
   388 		if (version[0] == '\0')
   389 			printf("%s\n", name);
   390 		else
   391 			printf("%s %s %s\n", name,
   392 			       razor_property_relation_to_string(property),
   393 			       version);
   394 	}
   395 	razor_property_iterator_destroy(pi);
   396 }
   397 
   398 static void
   399 print_package_info(struct razor_set *set, struct razor_package *package,
   400 		   const char *name, const char *version, const char *arch)
   401 {
   402 	const char *summary, *description, *url, *license;
   403 
   404 	razor_package_get_details(set, package, &summary, &description,
   405 				  &url, &license);
   406 
   407 	printf("Name:        %s\n", name);
   408 	printf("Arch:        %s\n", arch);
   409 	printf("Version:     %s\n", version);
   410 	printf("URL:         %s\n", url);
   411 	printf("License:     %s\n", license);
   412 	printf("Summary:     %s\n", summary);
   413 	printf("Description:\n");
   414 	printf("%s\n", description);
   415 	printf("\n");
   416 }
   417 
   418 static void
   419 print_package_changelog(struct razor_set *set, struct razor_package *package)
   420 {
   421 	printf("FIXME: Package changelog not tracked.\n");
   422 }
   423 
   424 static struct razor_set *
   425 create_set_from_command_line(int argc, const char *argv[])
   426 {
   427 	struct razor_importer *importer;
   428 	struct razor_rpm *rpm;
   429 	int i;
   430 
   431 	importer = razor_importer_create();
   432 
   433 	for (i = 0; i < argc; i++) {
   434 		rpm = razor_rpm_open(argv[i]);
   435 		if (rpm == NULL)
   436 			continue;
   437 		if (razor_importer_add_rpm(importer, rpm))
   438 			printf("couldn't import %s\n", argv[i]);
   439 
   440 		razor_rpm_close(rpm);
   441 	}
   442 
   443 	return razor_importer_finish(importer);
   444 }
   445 
   446 static void
   447 command_query(int argc, const char *argv[])
   448 {
   449 	struct razor_set *set;
   450 	struct razor_package_iterator *pi;
   451 	struct razor_package *package;
   452 	const char *name, *version, *arch, *details, *files;
   453 
   454 	if (option_package) {
   455 		set = create_set_from_command_line(argc, argv);
   456 		argc = 0;
   457 		option_all = 1;
   458 	} else {
   459 		set = razor_root_open_read_only(option_root);
   460 	}
   461 
   462 	pi = get_query_packages(set, argc, argv);
   463 
   464 	/* FIXME: We need to figure out how to do this right. */
   465 	details = "install/var/lib/razor/system-details.repo";
   466 	if (option_info)
   467 		razor_set_open_details(set, details);
   468 	files = "install/var/lib/razor/system-files.repo";
   469 	if (option_list)
   470 		razor_set_open_files(set, files);
   471 
   472 	while (razor_package_iterator_next(pi, &package,
   473 					   &name, &version, &arch)) {
   474 		if (option_conflicts)
   475 			print_package_properties(set, package,
   476 						 RAZOR_PROPERTY_CONFLICTS);
   477 		if (option_obsoletes)
   478 			print_package_properties(set, package,
   479 						 RAZOR_PROPERTY_OBSOLETES);
   480 		if (option_requires)
   481 			print_package_properties(set, package,
   482 						 RAZOR_PROPERTY_REQUIRES);
   483 		if (option_provides)
   484 			print_package_properties(set, package,
   485 						 RAZOR_PROPERTY_PROVIDES);
   486 		if (option_info)
   487 			print_package_info(set, package, name, version, arch);
   488 		if (option_changelog)
   489 			print_package_changelog(set, package);
   490 		if (option_list)
   491 			razor_set_list_package_files(set, name);
   492 
   493 		if (option_conflicts + option_obsoletes +
   494 		    option_requires + option_provides +
   495 		    option_info + option_changelog + option_list == 0)
   496 			printf("%s-%s.%s\n", name, version, arch);
   497 	}
   498 
   499 	razor_package_iterator_destroy(pi);
   500 
   501 	razor_set_destroy(set);
   502 
   503 	return;
   504 }
   505 
   506 static void
   507 command_verify(int argc, const char *argv[])
   508 {
   509 	struct razor_set *set;
   510 	struct razor_package_iterator *pi;
   511 	struct razor_package *package;
   512 	const char *name, *version, *arch;
   513 
   514 	if (option_package) {
   515 		set = create_set_from_command_line(argc, argv);
   516 		argc = 0;
   517 		option_all = 1;
   518 	} else {
   519 		set = razor_root_open_read_only(option_root);
   520 	}
   521 
   522 	pi = get_query_packages(set, argc, argv);
   523 
   524 	while (razor_package_iterator_next(pi, &package,
   525 					   &name, &version, &arch)) {
   526 		printf("verify %s-%s.%s - not implemented\n",
   527 		       name, version, arch);
   528 	}
   529 
   530 	razor_package_iterator_destroy(pi);
   531 }
   532 
   533 static void
   534 update_package(enum razor_diff_action action,
   535 	       struct razor_package *package,
   536 	       const char *name,
   537 	       const char *version,
   538 	       const char *arch,
   539 	       void *data)
   540 {
   541 	if (action == RAZOR_DIFF_ACTION_ADD)
   542 		printf("install %s-%s.%s\n", name, version, arch);
   543 	if (action == RAZOR_DIFF_ACTION_REMOVE)
   544 		printf("remove %s-%s.%s\n", name, version, arch);
   545 }
   546 
   547 static void
   548 command_erase(int argc, const char *argv[])
   549 {
   550 	struct razor_set *set, *upstream, *next;
   551 	struct razor_transaction *trans;
   552 	struct razor_package_query *query;
   553 	struct razor_package_iterator *pi;
   554 	struct razor_package *package;
   555 	const char *name, *version, *arch;
   556 
   557 	if (argc == 0) {
   558 		printf("no packages given for erase\n");
   559 		exit(1);
   560 	}
   561 
   562 	set = razor_set_open(repo_filename);
   563 	upstream = razor_set_create();
   564 
   565 	trans = razor_transaction_create(set, upstream);
   566 
   567 	query = razor_package_query_create(set);
   568 	add_command_line_packages(set, query, argc, argv);
   569 
   570 	pi = razor_package_query_finish(query);
   571 	while (razor_package_iterator_next(pi, &package,
   572 					   &name, &version, &arch))
   573 		razor_transaction_remove_package(trans, package);
   574 	razor_package_iterator_destroy(pi);
   575 
   576 	if (!option_nodeps && razor_transaction_describe(trans) > 0) {
   577 		printf("unsatisfied dependencies.\n");
   578 		exit(1);
   579 	}
   580 
   581 	if (option_test)
   582 		exit(0);
   583 
   584 	next = razor_transaction_finish(trans);
   585 
   586 	if (!option_justdb)
   587 		razor_set_diff(set, next, update_package, NULL);
   588 
   589 	razor_set_destroy(set);
   590 	razor_set_destroy(upstream);
   591 
   592 	razor_set_destroy(next);
   593 }
   594 
   595 static void
   596 command_install(int argc, const char *argv[])
   597 {
   598 	struct razor_set *set, *upstream, *next;
   599 	struct razor_transaction *trans;
   600 	struct razor_package_iterator *pi;
   601 	struct razor_package *package;
   602 	const char *name, *version, *arch;
   603 
   604 	if (argc == 0) {
   605 		printf("no packages given for install\n");
   606 		exit(1);
   607 	}
   608 
   609 	set = razor_set_open(repo_filename);
   610 	upstream = create_set_from_command_line(argc, argv);
   611 
   612 	trans = razor_transaction_create(set, upstream);
   613 
   614 	pi = razor_package_iterator_create(upstream);
   615 	while (razor_package_iterator_next(pi, &package,
   616 					   &name, &version, &arch))
   617 		razor_transaction_install_package(trans, package);
   618 	razor_package_iterator_destroy(pi);
   619 
   620 	if (!option_nodeps && razor_transaction_describe(trans) > 0) {
   621 		printf("unsatisfied dependencies.\n");
   622 		exit(1);
   623 	}
   624 
   625 	if (option_test)
   626 		exit(0);
   627 
   628 	next = razor_transaction_finish(trans);
   629 
   630 	if (!option_justdb)
   631 		razor_set_diff(set, next, update_package, NULL);
   632 
   633 	razor_set_destroy(set);
   634 	razor_set_destroy(upstream);
   635 
   636 	razor_set_destroy(next);
   637 }
   638 
   639 static void
   640 command_update(int argc, const char *argv[])
   641 {
   642 	struct razor_set *set, *upstream, *next;
   643 	struct razor_transaction *trans;
   644 	struct razor_package_iterator *pi;
   645 	struct razor_package *package;
   646 	const char *name, *version, *arch;
   647 
   648 	if (argc == 0) {
   649 		printf("no packages given for update\n");
   650 		exit(1);
   651 	}
   652 
   653 	set = razor_set_open(repo_filename);
   654 	upstream = create_set_from_command_line(argc, argv);
   655 
   656 	trans = razor_transaction_create(set, upstream);
   657 
   658 	pi = razor_package_iterator_create(upstream);
   659 	while (razor_package_iterator_next(pi, &package,
   660 					   &name, &version, &arch))
   661 		razor_transaction_update_package(trans, package);
   662 	razor_package_iterator_destroy(pi);
   663 
   664 	if (!option_nodeps && razor_transaction_describe(trans) > 0) {
   665 		printf("unsatisfied dependencies.\n");
   666 		exit(1);
   667 	}
   668 
   669 	if (option_test)
   670 		exit(0);
   671 
   672 	next = razor_transaction_finish(trans);
   673 
   674 	if (!option_justdb)
   675 		razor_set_diff(set, next, update_package, NULL);
   676 
   677 	razor_set_destroy(set);
   678 	razor_set_destroy(upstream);
   679 
   680 	razor_set_destroy(next);
   681 }
   682 
   683 static int
   684 for_each_option(const struct option *options,
   685 		const char *name, char short_name,
   686 		void (*fn)(const struct option *o,
   687 			   const char *name, char short_name,
   688 			   void *data), void *data)
   689 {
   690 	int i, count = 0;
   691 
   692 	for (i = 0; options[i].type != OPTION_LAST; i++) {
   693 		switch (options[i].type) {
   694 		case OPTION_GROUP:
   695 			count += for_each_option(options[i].data,
   696 						 name, short_name, fn, data);
   697 			break;
   698 
   699 		case OPTION_BOOL:
   700 		case OPTION_STRING:
   701 			if (name && strcmp(options[i].name, name) == 0) {
   702 				fn(&options[i], name, 0, data);
   703 				count++;
   704 				break;
   705 			}
   706 
   707 			if (short_name &&
   708 			    short_name == options[i].short_name) {
   709 				fn(&options[i], NULL, short_name, data);
   710 				count++;
   711 				break;
   712 			}
   713 			break;
   714 
   715 		case OPTION_LAST:
   716 			break;
   717 		}
   718 	}
   719 
   720 	return count;
   721 }
   722 
   723 static void
   724 handle_option(const struct option *o,
   725 	      const char *name, char short_name, void *data)
   726 {
   727 	if (o->data == NULL) {
   728 		if (name)
   729 			printf("option --%s not supported\n", name);
   730 		else
   731 			printf("option -%c not supported\n", short_name);
   732 		return;
   733 	}
   734 
   735 	switch (o->type) {
   736 	case OPTION_BOOL:
   737 		*(int *) o->data = 1;
   738 		break;
   739 
   740 	case OPTION_STRING:
   741 		*(const char **) o->data = name + strlen(o->name) + 1;
   742 		break;
   743 
   744 	case OPTION_LAST:
   745 	case OPTION_GROUP:
   746 		/* Shouldn't happen. */
   747 		break;
   748 	}
   749 }
   750 
   751 static int
   752 parse_options(const struct option *options, int argc, const char **argv)
   753 {
   754 	int i, j, k;
   755 
   756 	for (i = 1, j = 0; i < argc; i++) {
   757 		if (argv[i][0] != '-') {
   758 			argv[j++] = argv[i];
   759 			continue;
   760 		}
   761 
   762 		if (argv[i][1] == '-') {
   763 			if (for_each_option(options, &argv[i][2], 0,
   764 					    handle_option, NULL) == 0) {
   765 				printf("unknown option: %s\n", argv[i]);
   766 				exit(1);
   767 			}
   768 			continue;
   769 		}
   770 
   771 		for (k = 1; argv[i][k]; k++) {
   772 			if (for_each_option(options, NULL, argv[i][k],
   773 					    handle_option, NULL) == 0) {
   774 				printf("unknown option: -%c\n", argv[i][k]);
   775 				exit(1);
   776 			}
   777 		}
   778 	}
   779 
   780 	return j;
   781 }
   782 
   783 static void
   784 print_options_help(const struct option *options)
   785 {
   786 	int i;
   787 
   788 	for (i = 0; options[i].type != OPTION_LAST; i++) {
   789 		switch (options[i].type) {
   790 		case OPTION_GROUP:
   791 			printf("%s\n", options[i].description);
   792 			print_options_help(options[i].data);
   793 			printf("\n");
   794 			break;
   795 
   796 		case OPTION_BOOL:
   797 		case OPTION_STRING:
   798 			printf("  ");
   799 			if (options[i].short_name)
   800 				printf("-%c", options[i].short_name);
   801 			if (options[i].short_name && options[i].name)
   802 				printf(", ");
   803 			if (options[i].name)
   804 				printf("--%s", options[i].name);
   805 			if (options[i].arg_name)
   806 				printf("=%s", options[i].arg_name);
   807 			if (options[i].description)
   808 				printf("\t\t%s", options[i].description);
   809 			printf("\n");
   810 			break;
   811 
   812 		case OPTION_LAST:
   813 			break;
   814 		}
   815 	}
   816 }
   817 
   818 static void
   819 print_options_usage(const struct option *options)
   820 {
   821 	int i;
   822 
   823 	for (i = 0; options[i].type != OPTION_LAST; i++) {
   824 		switch (options[i].type) {
   825 		case OPTION_GROUP:
   826 			print_options_usage(options[i].data);
   827 			break;
   828 
   829 		case OPTION_BOOL:
   830 			printf("[");
   831 			if (options[i].short_name)
   832 				printf("-%c", options[i].short_name);
   833 			if (options[i].short_name && options[i].name)
   834 				printf("|");
   835 			if (options[i].name)
   836 				printf("--%s", options[i].name);
   837 			printf("] ");
   838 			break;
   839 
   840 		case OPTION_STRING:
   841 			printf("[");
   842 			if (options[i].short_name)
   843 				printf("-%c", options[i].short_name);
   844 			if (options[i].short_name && options[i].name)
   845 				printf("|");
   846 			if (options[i].name)
   847 				printf("--%s", options[i].name);
   848 			if (options[i].arg_name)
   849 				printf("=%s", options[i].arg_name);
   850 			printf("] ");
   851 			break;
   852 
   853 
   854 			break;
   855 
   856 		case OPTION_LAST:
   857 			break;
   858 		}
   859 	}
   860 }
   861 
   862 int
   863 main(int argc, const char *argv[])
   864 {
   865 	argc = parse_options(rpm_options, argc, argv);
   866 
   867 	if (option_version) {
   868 		printf("razor rpm version hoopla.\n");
   869 		exit(0);
   870 	}
   871 
   872 	if (option_help) {
   873 		printf("Usage: rpm [OPTION...]\n");
   874 		print_options_help(rpm_options);
   875 		exit(0);
   876 	}
   877 
   878 	if (option_usage) {
   879 		printf("Usage: rpm [OPTION...]\n");
   880 		print_options_usage(rpm_options);
   881 		printf("\n");
   882 		exit(0);
   883 	}
   884 
   885 	if (option_initdb) {
   886 		command_initdb(argc, argv);
   887 	} else if (option_verify) {
   888 		command_verify(argc, argv);
   889 	} else if (option_query) {
   890 		command_query(argc, argv);
   891 	} else if (option_install) {
   892 		command_install(argc, argv);
   893 	} else if (option_upgrade) {
   894 		command_update(argc, argv);
   895 	} else if (option_erase) {
   896 		command_erase(argc, argv);
   897 	} else {
   898 		print_options_usage(rpm_options);
   899 		printf("\n");
   900 		exit(0);
   901 	}
   902 
   903 	return 0;
   904 }