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