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