src/main.c
author James Bowes <jbowes@redhat.com>
Thu Jun 26 19:11:36 2008 -0400 (2008-06-26)
changeset 292 4af421a011b4
parent 291 57d1e05b9427
child 294 d7ecddf38c3d
permissions -rw-r--r--
Add 'search' command
rhughes@241
     1
/*
rhughes@241
     2
 * Copyright (C) 2008  Kristian Høgsberg <krh@redhat.com>
rhughes@241
     3
 * Copyright (C) 2008  Red Hat, Inc
rhughes@241
     4
 *
rhughes@241
     5
 * This program is free software; you can redistribute it and/or modify
rhughes@241
     6
 * it under the terms of the GNU General Public License as published by
rhughes@241
     7
 * the Free Software Foundation; either version 2 of the License, or
rhughes@241
     8
 * (at your option) any later version.
rhughes@241
     9
 *
rhughes@241
    10
 * This program is distributed in the hope that it will be useful,
rhughes@241
    11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
rhughes@241
    12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
rhughes@241
    13
 * GNU General Public License for more details.
rhughes@241
    14
 *
rhughes@241
    15
 * You should have received a copy of the GNU General Public License along
rhughes@241
    16
 * with this program; if not, write to the Free Software Foundation, Inc.,
rhughes@241
    17
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
rhughes@241
    18
 */
rhughes@241
    19
rhughes@241
    20
#include <stdlib.h>
rhughes@241
    21
#include <stddef.h>
rhughes@241
    22
#include <stdio.h>
rhughes@241
    23
#include <stdint.h>
rhughes@241
    24
#include <string.h>
rhughes@241
    25
#include <sys/stat.h>
rhughes@241
    26
#include <unistd.h>
rhughes@241
    27
#include <fcntl.h>
rhughes@241
    28
#include <dirent.h>
rhughes@241
    29
#include <curl/curl.h>
rhughes@241
    30
#include <fnmatch.h>
rhughes@241
    31
#include <errno.h>
rhughes@241
    32
#include "razor.h"
rhughes@241
    33
rhughes@241
    34
static const char system_repo_filename[] = "system.repo";
rhughes@241
    35
static const char next_repo_filename[] = "system-next.repo";
rhughes@241
    36
static const char rawhide_repo_filename[] = "rawhide.repo";
rhughes@241
    37
static const char updated_repo_filename[] = "system-updated.repo";
rhughes@241
    38
static const char install_root[] = "install";
rhughes@241
    39
static const char *repo_filename = system_repo_filename;
rhughes@241
    40
static const char *yum_url;
rhughes@241
    41
krh@271
    42
#define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
krh@271
    43
krh@281
    44
static struct razor_package_iterator *
krh@281
    45
create_iterator_from_argv(struct razor_set *set, int argc, const char *argv[])
rhughes@241
    46
{
krh@281
    47
	struct razor_package_query *query;
krh@281
    48
	struct razor_package_iterator *iter;
rhughes@241
    49
	struct razor_package *package;
krh@281
    50
	const char *name, *version, *arch, *pattern;
krh@281
    51
	int i, count;
rhughes@241
    52
krh@281
    53
	if (argc == 0)
krh@281
    54
		return razor_package_iterator_create(set);
krh@281
    55
krh@281
    56
	query = razor_package_query_create(set);
krh@281
    57
krh@281
    58
	for (i = 0; i < argc; i++) {
krh@281
    59
		iter = razor_package_iterator_create(set);
krh@281
    60
		pattern = argv[i];
krh@281
    61
		count = 0;
krh@281
    62
		while (razor_package_iterator_next(iter, &package,
krh@281
    63
						   &name, &version, &arch)) {
krh@281
    64
			if (fnmatch(pattern, name, 0) != 0)
krh@281
    65
				continue;
krh@281
    66
krh@281
    67
			razor_package_query_add_package(query, package);
krh@281
    68
			count++;
krh@281
    69
		}
krh@281
    70
		razor_package_iterator_destroy(iter);
krh@281
    71
krh@281
    72
		if (count == 0)
krh@281
    73
			fprintf(stderr,
krh@281
    74
				"no package matches \"%s\"\n", pattern);
rhughes@241
    75
	}
rhughes@241
    76
krh@281
    77
	return razor_package_query_finish(query);
krh@281
    78
}
krh@281
    79
krh@281
    80
#define LIST_PACKAGES_ONLY_NAMES 0x01
krh@281
    81
krh@281
    82
static void
krh@281
    83
list_packages(struct razor_package_iterator *iter, uint32_t flags)
krh@281
    84
{
krh@281
    85
	struct razor_package *package;
krh@281
    86
	const char *name, *version, *arch;
krh@281
    87
krh@281
    88
	while (razor_package_iterator_next(iter, &package,
rhughes@241
    89
					   &name, &version, &arch)) {
krh@281
    90
		if (flags & LIST_PACKAGES_ONLY_NAMES)
rhughes@241
    91
			printf("%s\n", name);
rhughes@241
    92
		else
rhughes@241
    93
			printf("%s-%s.%s\n", name, version, arch);
rhughes@241
    94
	}
krh@281
    95
}
krh@281
    96
krh@281
    97
static int
krh@281
    98
command_list(int argc, const char *argv[])
krh@281
    99
{
krh@281
   100
	struct razor_package_iterator *pi;
krh@281
   101
	struct razor_set *set;
krh@281
   102
	uint32_t flags = 0;
krh@281
   103
	int i = 0;
krh@281
   104
krh@281
   105
	if (i < argc && strcmp(argv[i], "--only-names") == 0) {
krh@281
   106
		flags |= LIST_PACKAGES_ONLY_NAMES;
krh@281
   107
		i++;
krh@281
   108
	}
krh@281
   109
krh@281
   110
	set = razor_set_open(repo_filename);
krh@281
   111
	pi = create_iterator_from_argv(set, argc - i, argv + i);
krh@281
   112
	list_packages(pi, flags);
rhughes@241
   113
	razor_package_iterator_destroy(pi);
rhughes@241
   114
	razor_set_destroy(set);
rhughes@241
   115
rhughes@241
   116
	return 0;
rhughes@241
   117
}
rhughes@241
   118
rhughes@241
   119
static int
krh@247
   120
list_properties(const char *package_name, uint32_t type)
rhughes@241
   121
{
rhughes@241
   122
	struct razor_set *set;
rhughes@241
   123
	struct razor_property *property;
rhughes@241
   124
	struct razor_package *package;
rhughes@241
   125
	struct razor_property_iterator *pi;
rhughes@241
   126
	const char *name, *version;
krh@247
   127
	uint32_t flags;
rhughes@241
   128
rhughes@241
   129
	set = razor_set_open(repo_filename);
richard@280
   130
	if (package_name) {
rhughes@241
   131
		package = razor_set_get_package(set, package_name);
richard@280
   132
		if (!package) {
richard@280
   133
			fprintf(stderr, "no package named \"%s\"\n", package_name);
richard@280
   134
			return 1;
richard@280
   135
		}
richard@280
   136
	} else
rhughes@241
   137
		package = NULL;
rhughes@241
   138
rhughes@241
   139
	pi = razor_property_iterator_create(set, package);
rhughes@241
   140
	while (razor_property_iterator_next(pi, &property,
krh@247
   141
					    &name, &flags, &version)) {
krh@247
   142
		if ((flags & RAZOR_PROPERTY_TYPE_MASK) != type)
rhughes@241
   143
			continue;
krh@247
   144
		printf("%s", name);
krh@247
   145
		if (version[0] != '\0')
krh@247
   146
			printf(" %s %s",
krh@247
   147
			       razor_property_relation_to_string(property),
krh@247
   148
			       version);
krh@247
   149
krh@247
   150
		if (flags & ~(RAZOR_PROPERTY_RELATION_MASK | RAZOR_PROPERTY_TYPE_MASK)) {
krh@247
   151
			printf(" [");
krh@247
   152
			if (flags & RAZOR_PROPERTY_PRE)
krh@247
   153
				printf(" pre");
krh@247
   154
			if (flags & RAZOR_PROPERTY_POST)
krh@247
   155
				printf(" post");
krh@247
   156
			if (flags & RAZOR_PROPERTY_PREUN)
krh@247
   157
				printf(" preun");
krh@247
   158
			if (flags & RAZOR_PROPERTY_POSTUN)
krh@247
   159
				printf(" postun");
krh@247
   160
			printf(" ]");
krh@247
   161
		}
krh@247
   162
		printf("\n");
rhughes@241
   163
	}
rhughes@241
   164
	razor_property_iterator_destroy(pi);
rhughes@241
   165
rhughes@241
   166
	razor_set_destroy(set);
rhughes@241
   167
rhughes@241
   168
	return 0;
rhughes@241
   169
}
rhughes@241
   170
rhughes@241
   171
static int
rhughes@241
   172
command_list_requires(int argc, const char *argv[])
rhughes@241
   173
{
rhughes@241
   174
	return list_properties(argv[0], RAZOR_PROPERTY_REQUIRES);
rhughes@241
   175
}
rhughes@241
   176
rhughes@241
   177
static int
rhughes@241
   178
command_list_provides(int argc, const char *argv[])
rhughes@241
   179
{
rhughes@241
   180
	return list_properties(argv[0], RAZOR_PROPERTY_PROVIDES);
rhughes@241
   181
}
rhughes@241
   182
rhughes@241
   183
static int
rhughes@241
   184
command_list_obsoletes(int argc, const char *argv[])
rhughes@241
   185
{
rhughes@241
   186
	return list_properties(argv[0], RAZOR_PROPERTY_OBSOLETES);
rhughes@241
   187
}
rhughes@241
   188
rhughes@241
   189
static int
rhughes@241
   190
command_list_conflicts(int argc, const char *argv[])
rhughes@241
   191
{
rhughes@241
   192
	return list_properties(argv[0], RAZOR_PROPERTY_CONFLICTS);
rhughes@241
   193
}
rhughes@241
   194
rhughes@241
   195
static int
rhughes@241
   196
command_list_files(int argc, const char *argv[])
rhughes@241
   197
{
rhughes@241
   198
	struct razor_set *set;
rhughes@241
   199
rhughes@241
   200
	set = razor_set_open(repo_filename);
rhughes@241
   201
	if (set == NULL)
rhughes@241
   202
		return 1;
jbowes@288
   203
	if (razor_set_open_files(set, "system-files.repo"))
jbowes@288
   204
		return 1;
jbowes@288
   205
rhughes@241
   206
	razor_set_list_files(set, argv[0]);
rhughes@241
   207
	razor_set_destroy(set);
rhughes@241
   208
rhughes@241
   209
	return 0;
rhughes@241
   210
}
rhughes@241
   211
rhughes@241
   212
static int
rhughes@241
   213
command_list_file_packages(int argc, const char *argv[])
rhughes@241
   214
{
rhughes@241
   215
	struct razor_set *set;
rhughes@241
   216
	struct razor_package_iterator *pi;
rhughes@241
   217
rhughes@241
   218
	set = razor_set_open(repo_filename);
rhughes@241
   219
	if (set == NULL)
rhughes@241
   220
		return 1;
jbowes@288
   221
	if (razor_set_open_files(set, "system-files.repo"))
jbowes@288
   222
		return 1;
rhughes@241
   223
rhughes@241
   224
	pi = razor_package_iterator_create_for_file(set, argv[0]);
krh@281
   225
	list_packages(pi, 0);
rhughes@241
   226
	razor_package_iterator_destroy(pi);
rhughes@241
   227
rhughes@241
   228
	razor_set_destroy(set);
rhughes@241
   229
rhughes@241
   230
	return 0;
rhughes@241
   231
}
rhughes@241
   232
rhughes@241
   233
static int
rhughes@241
   234
command_list_package_files(int argc, const char *argv[])
rhughes@241
   235
{
rhughes@241
   236
	struct razor_set *set;
rhughes@241
   237
rhughes@241
   238
	set = razor_set_open(repo_filename);
rhughes@241
   239
	if (set == NULL)
rhughes@241
   240
		return 1;
jbowes@288
   241
	if (razor_set_open_files(set, "system-files.repo"))
jbowes@288
   242
		return 1;
jbowes@288
   243
rhughes@241
   244
	razor_set_list_package_files(set, argv[0]);
rhughes@241
   245
	razor_set_destroy(set);
rhughes@241
   246
rhughes@241
   247
	return 0;
rhughes@241
   248
}
rhughes@241
   249
rhughes@241
   250
static int
rhughes@241
   251
list_property_packages(const char *ref_name,
rhughes@241
   252
		       const char *ref_version,
krh@247
   253
		       uint32_t type)
rhughes@241
   254
{
rhughes@241
   255
	struct razor_set *set;
rhughes@241
   256
	struct razor_property *property;
krh@281
   257
	struct razor_property_iterator *prop_iter;
krh@281
   258
	struct razor_package_iterator *pkg_iter;
rhughes@241
   259
	const char *name, *version;
krh@247
   260
	uint32_t flags;
rhughes@241
   261
rhughes@241
   262
	if (ref_name == NULL)
rhughes@241
   263
		return 0;
rhughes@241
   264
rhughes@241
   265
	set = razor_set_open(repo_filename);
rhughes@241
   266
	if (set == NULL)
rhughes@241
   267
		return 1;
rhughes@241
   268
krh@281
   269
	prop_iter = razor_property_iterator_create(set, NULL);
krh@281
   270
	while (razor_property_iterator_next(prop_iter, &property,
krh@247
   271
					    &name, &flags, &version)) {
rhughes@241
   272
		if (strcmp(ref_name, name) != 0)
rhughes@241
   273
			continue;
krh@247
   274
		if (ref_version &&
krh@247
   275
		    (flags & RAZOR_PROPERTY_RELATION_MASK) == RAZOR_PROPERTY_EQUAL &&
rhughes@241
   276
		    strcmp(ref_version, version) != 0)
rhughes@241
   277
			continue;
krh@247
   278
		if ((flags & RAZOR_PROPERTY_TYPE_MASK) != type)
rhughes@241
   279
			continue;
rhughes@241
   280
krh@281
   281
		pkg_iter =
krh@281
   282
			razor_package_iterator_create_for_property(set,
krh@281
   283
								   property);
krh@281
   284
		list_packages(pkg_iter, 0);
krh@281
   285
		razor_package_iterator_destroy(pkg_iter);
rhughes@241
   286
	}
krh@281
   287
	razor_property_iterator_destroy(prop_iter);
rhughes@241
   288
rhughes@241
   289
	return 0;
rhughes@241
   290
}
rhughes@241
   291
rhughes@241
   292
static int
rhughes@241
   293
command_what_requires(int argc, const char *argv[])
rhughes@241
   294
{
rhughes@241
   295
	return list_property_packages(argv[0], argv[1],
rhughes@241
   296
				      RAZOR_PROPERTY_REQUIRES);
rhughes@241
   297
}
rhughes@241
   298
rhughes@241
   299
static int
rhughes@241
   300
command_what_provides(int argc, const char *argv[])
rhughes@241
   301
{
rhughes@241
   302
	return list_property_packages(argv[0], argv[1],
rhughes@241
   303
				      RAZOR_PROPERTY_PROVIDES);
rhughes@241
   304
}
rhughes@241
   305
rhughes@241
   306
static int
rhughes@241
   307
show_progress(void *clientp,
rhughes@241
   308
	      double dltotal, double dlnow, double ultotal, double ulnow)
rhughes@241
   309
{
rhughes@241
   310
	const char *file = clientp;
rhughes@241
   311
rhughes@241
   312
	if (!dlnow < dltotal)
rhughes@241
   313
		fprintf(stderr, "\rdownloading %s, %dkB/%dkB",
rhughes@241
   314
			file, (int) dlnow / 1024, (int) dltotal / 1024);
rhughes@241
   315
rhughes@241
   316
	return 0;
rhughes@241
   317
}
rhughes@241
   318
rhughes@241
   319
static int
rhughes@241
   320
download_if_missing(const char *url, const char *file)
rhughes@241
   321
{
rhughes@241
   322
	CURL *curl;
rhughes@241
   323
	struct stat buf;
rhughes@241
   324
	char error[256];
rhughes@241
   325
	FILE *fp;
rhughes@241
   326
	CURLcode res;
rhughes@241
   327
	long response;
rhughes@241
   328
rhughes@241
   329
	curl = curl_easy_init();
rhughes@241
   330
	if (curl == NULL)
rhughes@241
   331
		return 1;
rhughes@241
   332
rhughes@241
   333
	curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, error);
rhughes@241
   334
	curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0);
rhughes@241
   335
	curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, show_progress);
rhughes@241
   336
	curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, file);
rhughes@241
   337
rhughes@241
   338
	if (stat(file, &buf) < 0) {
rhughes@241
   339
		fp = fopen(file, "w");
rhughes@241
   340
		if (fp == NULL) {
rhughes@241
   341
			fprintf(stderr,
rhughes@241
   342
				"failed to open %s for writing\n", file);
rhughes@241
   343
			return -1;
rhughes@241
   344
		}
rhughes@241
   345
		curl_easy_setopt(curl, CURLOPT_WRITEDATA, fp);
rhughes@241
   346
		curl_easy_setopt(curl, CURLOPT_URL, url);
rhughes@241
   347
		res = curl_easy_perform(curl);
rhughes@241
   348
		fclose(fp);
rhughes@241
   349
		if (res != CURLE_OK) {
rhughes@241
   350
			fprintf(stderr, "curl error: %s\n", error);
rhughes@241
   351
			unlink(file);
rhughes@241
   352
			return -1;
rhughes@241
   353
		}
rhughes@241
   354
		res = curl_easy_getinfo(curl,
rhughes@241
   355
					CURLINFO_RESPONSE_CODE, &response);
rhughes@241
   356
		if (res != CURLE_OK) {
rhughes@241
   357
			fprintf(stderr, "curl error: %s\n", error);
rhughes@241
   358
                        unlink(file);
rhughes@241
   359
                        return -1;
rhughes@241
   360
		}
rhughes@241
   361
		if (response != 200) {
rhughes@241
   362
			fprintf(stderr, " - failed %ld\n", response);
rhughes@241
   363
                        unlink(file);
rhughes@241
   364
                        return -1;
rhughes@241
   365
		}
rhughes@241
   366
		fprintf(stderr, "\n");
rhughes@241
   367
	}
rhughes@241
   368
rhughes@241
   369
	curl_easy_cleanup(curl);
rhughes@241
   370
rhughes@241
   371
	return 0;
rhughes@241
   372
}
rhughes@241
   373
rhughes@241
   374
#define YUM_URL "http://download.fedora.redhat.com" \
rhughes@241
   375
	"/pub/fedora/linux/development/i386/os"
rhughes@241
   376
rhughes@241
   377
static int
rhughes@241
   378
command_import_yum(int argc, const char *argv[])
rhughes@241
   379
{
rhughes@241
   380
	struct razor_set *set;
rhughes@241
   381
	char buffer[512];
rhughes@241
   382
rhughes@241
   383
	printf("downloading from %s.\n", yum_url);
rhughes@241
   384
	snprintf(buffer, sizeof buffer,
rhughes@241
   385
		 "%s/repodata/primary.xml.gz", yum_url);
rhughes@241
   386
	if (download_if_missing(buffer, "primary.xml.gz") < 0)
rhughes@241
   387
		return -1;
rhughes@241
   388
	snprintf(buffer, sizeof buffer,
rhughes@241
   389
		 "%s/repodata/filelists.xml.gz", yum_url);
rhughes@241
   390
	if (download_if_missing(buffer, "filelists.xml.gz") < 0)
rhughes@241
   391
		return -1;
rhughes@241
   392
rhughes@241
   393
	set = razor_set_create_from_yum();
rhughes@241
   394
	if (set == NULL)
rhughes@241
   395
		return 1;
jbowes@258
   396
	razor_set_write(set, rawhide_repo_filename, RAZOR_REPO_FILE_MAIN);
jbowes@258
   397
	razor_set_write(set, "rawhide-details.repo", RAZOR_REPO_FILE_DETAILS);
jbowes@258
   398
	razor_set_write(set, "rawhide-files.repo", RAZOR_REPO_FILE_FILES);
rhughes@241
   399
	razor_set_destroy(set);
rhughes@241
   400
	printf("wrote %s\n", rawhide_repo_filename);
rhughes@241
   401
rhughes@241
   402
	return 0;
rhughes@241
   403
}
rhughes@241
   404
rhughes@241
   405
static int
rhughes@241
   406
command_import_rpmdb(int argc, const char *argv[])
rhughes@241
   407
{
rhughes@241
   408
	struct razor_set *set;
rhughes@241
   409
rhughes@241
   410
	set = razor_set_create_from_rpmdb();
rhughes@241
   411
	if (set == NULL)
rhughes@241
   412
		return 1;
jbowes@258
   413
	razor_set_write(set, repo_filename, RAZOR_REPO_FILE_MAIN);
jbowes@258
   414
	razor_set_write(set, "system-details.repo", RAZOR_REPO_FILE_DETAILS);
jbowes@258
   415
	razor_set_write(set, "system-files.repo", RAZOR_REPO_FILE_FILES);
rhughes@241
   416
	razor_set_destroy(set);
rhughes@241
   417
	printf("wrote %s\n", repo_filename);
rhughes@241
   418
rhughes@241
   419
	return 0;
rhughes@241
   420
}
rhughes@241
   421
rhughes@241
   422
static int
rhughes@241
   423
mark_packages_for_update(struct razor_transaction *trans,
rhughes@241
   424
			 struct razor_set *set, const char *pattern)
rhughes@241
   425
{
rhughes@241
   426
	struct razor_package_iterator *pi;
rhughes@241
   427
	struct razor_package *package;
rhughes@241
   428
	const char *name, *version, *arch;
rhughes@241
   429
	int matches = 0;
rhughes@241
   430
rhughes@241
   431
	pi = razor_package_iterator_create(set);
rhughes@241
   432
	while (razor_package_iterator_next(pi, &package,
rhughes@241
   433
					   &name, &version, &arch)) {
rhughes@241
   434
		if (pattern && fnmatch(pattern, name, 0) == 0) {
rhughes@241
   435
			razor_transaction_update_package(trans, package);
rhughes@241
   436
			matches++;
rhughes@241
   437
		}
rhughes@241
   438
	}
rhughes@241
   439
	razor_package_iterator_destroy(pi);
rhughes@241
   440
rhughes@241
   441
	return matches;
rhughes@241
   442
}
rhughes@241
   443
rhughes@241
   444
static int
rhughes@241
   445
mark_packages_for_removal(struct razor_transaction *trans,
rhughes@241
   446
			  struct razor_set *set, const char *pattern)
rhughes@241
   447
{
rhughes@241
   448
	struct razor_package_iterator *pi;
rhughes@241
   449
	struct razor_package *package;
rhughes@241
   450
	const char *name, *version, *arch;
rhughes@241
   451
	int matches = 0;
rhughes@241
   452
rhughes@241
   453
	pi = razor_package_iterator_create(set);
rhughes@241
   454
	while (razor_package_iterator_next(pi, &package,
rhughes@241
   455
					   &name, &version, &arch)) {
rhughes@241
   456
		if (pattern && fnmatch(pattern, name, 0) == 0) {
rhughes@241
   457
			razor_transaction_remove_package(trans, package);
rhughes@241
   458
			matches++;
rhughes@241
   459
		}
rhughes@241
   460
	}
rhughes@241
   461
	razor_package_iterator_destroy(pi);
rhughes@241
   462
rhughes@241
   463
	return matches;
rhughes@241
   464
}
rhughes@241
   465
rhughes@241
   466
static int
rhughes@241
   467
command_update(int argc, const char *argv[])
rhughes@241
   468
{
rhughes@241
   469
	struct razor_set *set, *upstream;
rhughes@241
   470
	struct razor_transaction *trans;
rhughes@241
   471
	int i, errors;
rhughes@241
   472
rhughes@241
   473
	set = razor_set_open(repo_filename);
rhughes@241
   474
	upstream = razor_set_open(rawhide_repo_filename);
rhughes@241
   475
	if (set == NULL || upstream == NULL)
rhughes@241
   476
		return 1;
rhughes@241
   477
rhughes@241
   478
	trans = razor_transaction_create(set, upstream);
rhughes@241
   479
	if (argc == 0)
rhughes@241
   480
		razor_transaction_update_all(trans);
rhughes@241
   481
	for (i = 0; i < argc; i++) {
rhughes@241
   482
		if (mark_packages_for_update(trans, set, argv[i]) == 0) {
rhughes@241
   483
			fprintf(stderr, "no match for %s\n", argv[i]);
rhughes@241
   484
			return 1;
rhughes@241
   485
		}
rhughes@241
   486
	}
rhughes@241
   487
krh@252
   488
	razor_transaction_resolve(trans);
krh@252
   489
	errors = razor_transaction_describe(trans);
krh@252
   490
	if (errors) {
krh@252
   491
		fprintf(stderr, "unresolved dependencies\n");
rhughes@241
   492
		return 1;
krh@252
   493
	}
rhughes@241
   494
rhughes@241
   495
	set = razor_transaction_finish(trans);
jbowes@258
   496
	razor_set_write(set, updated_repo_filename, RAZOR_REPO_FILE_MAIN);
rhughes@241
   497
	razor_set_destroy(set);
rhughes@241
   498
	razor_set_destroy(upstream);
rhughes@241
   499
	printf("wrote system-updated.repo\n");
rhughes@241
   500
rhughes@241
   501
	return 0;
rhughes@241
   502
}
rhughes@241
   503
rhughes@241
   504
static int
rhughes@241
   505
command_remove(int argc, const char *argv[])
rhughes@241
   506
{
rhughes@241
   507
	struct razor_set *set, *upstream;
rhughes@241
   508
	struct razor_transaction *trans;
rhughes@241
   509
	int i, errors;
rhughes@241
   510
rhughes@241
   511
	set = razor_set_open(repo_filename);
rhughes@241
   512
	if (set == NULL)
rhughes@241
   513
		return 1;
rhughes@241
   514
rhughes@241
   515
	upstream = razor_set_create();
rhughes@241
   516
	trans = razor_transaction_create(set, upstream);
rhughes@241
   517
	for (i = 0; i < argc; i++) {
rhughes@241
   518
		if (mark_packages_for_removal(trans, set, argv[i]) == 0) {
rhughes@241
   519
			fprintf(stderr, "no match for %s\n", argv[i]);
rhughes@241
   520
			return 1;
rhughes@241
   521
		}
rhughes@241
   522
	}
rhughes@241
   523
rhughes@241
   524
	errors = razor_transaction_resolve(trans);
rhughes@241
   525
	if (errors)
rhughes@241
   526
		return 1;
rhughes@241
   527
rhughes@241
   528
	set = razor_transaction_finish(trans);
jbowes@258
   529
	razor_set_write(set, updated_repo_filename, RAZOR_REPO_FILE_MAIN);
rhughes@241
   530
	razor_set_destroy(set);
rhughes@241
   531
	razor_set_destroy(upstream);
rhughes@241
   532
	printf("wrote system-updated.repo\n");
rhughes@241
   533
rhughes@241
   534
	return 0;
rhughes@241
   535
}
rhughes@241
   536
rhughes@241
   537
static void
krh@253
   538
print_diff(enum razor_diff_action action,
krh@253
   539
	   struct razor_package *package,
krh@253
   540
	   const char *name,
krh@253
   541
	   const char *version,
krh@253
   542
	   const char *arch,
rhughes@241
   543
	   void *data)
rhughes@241
   544
{
krh@253
   545
	if (action == RAZOR_DIFF_ACTION_ADD)
krh@253
   546
		printf("install %s-%s.%s\n", name, version, arch);
krh@253
   547
	if (action == RAZOR_DIFF_ACTION_REMOVE)
krh@253
   548
		printf("remove %s-%s.%s\n", name, version, arch);
rhughes@241
   549
}
rhughes@241
   550
rhughes@241
   551
static int
rhughes@241
   552
command_diff(int argc, const char *argv[])
rhughes@241
   553
{
rhughes@241
   554
	struct razor_set *set, *updated;
rhughes@241
   555
rhughes@241
   556
	set = razor_set_open(repo_filename);
rhughes@241
   557
	updated = razor_set_open(updated_repo_filename);
rhughes@241
   558
	if (set == NULL || updated == NULL)
rhughes@241
   559
		return 1;
rhughes@241
   560
rhughes@241
   561
	razor_set_diff(set, updated, print_diff, NULL);
rhughes@241
   562
rhughes@241
   563
	razor_set_destroy(set);
rhughes@241
   564
	razor_set_destroy(updated);
rhughes@241
   565
rhughes@241
   566
	return 0;
rhughes@241
   567
}
rhughes@241
   568
rhughes@241
   569
static int
rhughes@241
   570
command_import_rpms(int argc, const char *argv[])
rhughes@241
   571
{
rhughes@241
   572
	DIR *dir;
rhughes@241
   573
	struct dirent *de;
rhughes@241
   574
	struct razor_importer *importer;
rhughes@241
   575
	struct razor_set *set;
rhughes@241
   576
	struct razor_rpm *rpm;
jbowes@263
   577
	int len, imported_count = 0;
rhughes@241
   578
	char filename[256];
rhughes@241
   579
	const char *dirname = argv[0];
rhughes@241
   580
rhughes@241
   581
	if (dirname == NULL) {
rhughes@241
   582
		fprintf(stderr, "usage: razor import-rpms DIR\n");
rhughes@241
   583
		return -1;
rhughes@241
   584
	}
rhughes@241
   585
rhughes@241
   586
	dir = opendir(dirname);
rhughes@241
   587
	if (dir == NULL) {
rhughes@241
   588
		fprintf(stderr, "couldn't read dir %s\n", dirname);
rhughes@241
   589
		return -1;
rhughes@241
   590
	}
rhughes@241
   591
krh@249
   592
	importer = razor_importer_create();
rhughes@241
   593
rhughes@241
   594
	while (de = readdir(dir), de != NULL) {
rhughes@241
   595
		len = strlen(de->d_name);
rhughes@241
   596
		if (len < 5 || strcmp(de->d_name + len - 4, ".rpm") != 0)
rhughes@241
   597
		    continue;
rhughes@241
   598
		snprintf(filename, sizeof filename,
rhughes@241
   599
			 "%s/%s", dirname, de->d_name);
rhughes@241
   600
		rpm = razor_rpm_open(filename);
rhughes@241
   601
		if (rpm == NULL) {
rhughes@241
   602
			fprintf(stderr,
rhughes@241
   603
				"failed to open rpm \"%s\"\n", filename);
rhughes@241
   604
			continue;
rhughes@241
   605
		}
rhughes@241
   606
		if (razor_importer_add_rpm(importer, rpm)) {
rhughes@241
   607
			fprintf(stderr, "couldn't import %s\n", filename);
rhughes@241
   608
			break;
rhughes@241
   609
		}
rhughes@241
   610
		razor_rpm_close(rpm);
jbowes@263
   611
jbowes@263
   612
		printf("\rimporting %d", ++imported_count);
jbowes@263
   613
		fflush(stdout);
rhughes@241
   614
	}
rhughes@241
   615
rhughes@241
   616
	if (de != NULL) {
rhughes@241
   617
		razor_importer_destroy(importer);
rhughes@241
   618
		return -1;
rhughes@241
   619
	}
rhughes@241
   620
jbowes@263
   621
	printf("\nsaving\n");
rhughes@241
   622
	set = razor_importer_finish(importer);
rhughes@241
   623
jbowes@258
   624
	razor_set_write(set, repo_filename, RAZOR_REPO_FILE_MAIN);
jbowes@291
   625
	razor_set_write(set, "system-details.repo", RAZOR_REPO_FILE_DETAILS);
jbowes@291
   626
	razor_set_write(set, "system-files.repo", RAZOR_REPO_FILE_FILES);
rhughes@241
   627
	razor_set_destroy(set);
rhughes@241
   628
	printf("wrote %s\n", repo_filename);
rhughes@241
   629
rhughes@241
   630
	return 0;
rhughes@241
   631
}
rhughes@241
   632
krh@254
   633
static const char *
krh@254
   634
rpm_filename(const char *name, const char *version, const char *arch)
rhughes@241
   635
{
krh@254
   636
	static char file[PATH_MAX];
krh@254
   637
 	const char *v;
krh@254
   638
 
krh@254
   639
 	/* Skip epoch */
krh@253
   640
	v = strchr(version, ':');
krh@254
   641
 	if (v != NULL)
krh@254
   642
 		v = v + 1;
krh@254
   643
 	else
krh@253
   644
		v = version;
rhughes@241
   645
krh@254
   646
	snprintf(file, sizeof file, "%s-%s.%s.rpm", name, v, arch);
rhughes@241
   647
krh@254
   648
	return file;
rhughes@241
   649
}
rhughes@241
   650
krh@254
   651
static int
krh@254
   652
download_packages(struct razor_set *system, struct razor_set *next)
rhughes@241
   653
{
krh@254
   654
	struct razor_package_iterator *pi;
krh@254
   655
	struct razor_package *package;
krh@254
   656
	const char *name, *version, *arch;
krh@254
   657
	char file[PATH_MAX], url[256];
krh@254
   658
	int errors;
krh@254
   659
 
krh@254
   660
	pi = razor_set_create_install_iterator(system, next);
krh@254
   661
	errors = 0;
krh@254
   662
	while (razor_package_iterator_next(pi, &package,
krh@254
   663
					   &name, &version, &arch)) {
krh@254
   664
		snprintf(url, sizeof url,
krh@254
   665
			 "%s/Packages/%s",
krh@254
   666
			 yum_url, rpm_filename(name, version, arch));
krh@254
   667
		snprintf(file, sizeof file,
krh@254
   668
			 "rpms/%s", rpm_filename(name, version, arch));
krh@254
   669
		if (download_if_missing(url, file) < 0)
krh@254
   670
			errors++;
krh@254
   671
	}
krh@254
   672
	razor_package_iterator_destroy(pi);
krh@254
   673
krh@254
   674
	if (errors > 0) {
krh@254
   675
		fprintf(stderr, "failed to download %d packages\n", errors);
krh@254
   676
                return -1;
krh@254
   677
        }
krh@254
   678
krh@254
   679
	return 0;
krh@254
   680
}
krh@254
   681
krh@254
   682
static int
krh@254
   683
install_packages(struct razor_set *system, struct razor_set *next)
krh@254
   684
{
krh@254
   685
	struct razor_package_iterator *pi;
krh@254
   686
	struct razor_package *package;
krh@254
   687
	struct razor_rpm *rpm;
krh@254
   688
	const char *name, *version, *arch;
rhughes@241
   689
	char file[PATH_MAX];
rhughes@241
   690
krh@254
   691
	pi = razor_set_create_install_iterator(system, next);
krh@254
   692
	while (razor_package_iterator_next(pi, &package,
krh@254
   693
					   &name, &version, &arch)) {
krh@254
   694
		printf("install %s-%s\n", name, version);
krh@254
   695
krh@254
   696
		snprintf(file, sizeof file,
krh@254
   697
			 "rpms/%s", rpm_filename(name, version, arch));
krh@254
   698
		rpm = razor_rpm_open(file);
krh@254
   699
		if (rpm == NULL) {
krh@254
   700
			fprintf(stderr, "failed to open rpm %s\n", file);
krh@254
   701
			return -1;
krh@254
   702
		}
krh@254
   703
		if (razor_rpm_install(rpm, install_root) < 0) {
krh@254
   704
			fprintf(stderr,
krh@254
   705
				"failed to install rpm %s\n", file);
krh@254
   706
			return -1;
krh@254
   707
		}
krh@254
   708
		razor_rpm_close(rpm);
rhughes@241
   709
	}
krh@254
   710
	razor_package_iterator_destroy(pi);
rhughes@241
   711
krh@254
   712
	return 0;
rhughes@241
   713
}
rhughes@241
   714
rhughes@241
   715
static int
rhughes@241
   716
command_install(int argc, const char *argv[])
rhughes@241
   717
{
rhughes@241
   718
	struct razor_root *root;
krh@250
   719
	struct razor_set *system, *upstream, *next;
rhughes@241
   720
	struct razor_transaction *trans;
krh@281
   721
	int i = 0, dependencies = 1;
rhughes@241
   722
rhughes@241
   723
	if (i < argc && strcmp(argv[i], "--no-dependencies") == 0) {
rhughes@241
   724
		dependencies = 0;
rhughes@241
   725
		i++;
rhughes@241
   726
	}
rhughes@241
   727
krh@250
   728
	root = razor_root_open(install_root);
krh@255
   729
	if (root == NULL)
krh@255
   730
		return 1;
krh@255
   731
krh@250
   732
	system = razor_root_get_system_set(root);
rhughes@241
   733
	upstream = razor_set_open(rawhide_repo_filename);
krh@250
   734
	trans = razor_transaction_create(system, upstream);
rhughes@241
   735
rhughes@241
   736
	for (; i < argc; i++) {
rhughes@241
   737
		if (mark_packages_for_update(trans, upstream, argv[i]) == 0) {
rhughes@241
   738
			fprintf(stderr, "no package matched %s\n", argv[i]);
rhughes@241
   739
			razor_root_close(root);
rhughes@241
   740
			return 1;
rhughes@241
   741
		}
rhughes@241
   742
	}
rhughes@241
   743
rhughes@241
   744
	if (dependencies) {
krh@245
   745
		razor_transaction_resolve(trans);
krh@245
   746
		if (razor_transaction_describe(trans) > 0) {
rhughes@241
   747
			razor_root_close(root);
rhughes@241
   748
			return 1;
rhughes@241
   749
		}
rhughes@241
   750
	}
rhughes@241
   751
rhughes@241
   752
	next = razor_transaction_finish(trans);
rhughes@241
   753
rhughes@241
   754
	razor_root_update(root, next);
rhughes@241
   755
rhughes@241
   756
	if (mkdir("rpms", 0777) && errno != EEXIST) {
rhughes@241
   757
		fprintf(stderr, "failed to create rpms directory.\n");
rhughes@241
   758
		razor_root_close(root);
rhughes@241
   759
		return 1;
rhughes@241
   760
	}
rhughes@241
   761
krh@254
   762
	if (download_packages(system, next) < 0) {
rhughes@241
   763
		razor_root_close(root);
rhughes@241
   764
                return 1;
rhughes@241
   765
        }
rhughes@241
   766
krh@254
   767
	install_packages(system, next);
rhughes@241
   768
rhughes@241
   769
	razor_set_destroy(next);
rhughes@241
   770
	razor_set_destroy(upstream);
rhughes@241
   771
rhughes@241
   772
	return razor_root_commit(root);
rhughes@241
   773
}
rhughes@241
   774
rhughes@241
   775
static int
rhughes@241
   776
command_init(int argc, const char *argv[])
rhughes@241
   777
{
rhughes@241
   778
	return razor_root_create(install_root);
rhughes@241
   779
}
rhughes@241
   780
rhughes@241
   781
static int
rhughes@241
   782
command_download(int argc, const char *argv[])
rhughes@241
   783
{
rhughes@241
   784
	struct razor_set *set;
rhughes@241
   785
	struct razor_package_iterator *pi;
rhughes@241
   786
	struct razor_package *package;
rhughes@241
   787
	const char *pattern = argv[0], *name, *version, *arch;
rhughes@241
   788
	char url[256], file[256];
rhughes@241
   789
	int matches = 0;
rhughes@241
   790
rhughes@241
   791
	if (mkdir("rpms", 0777) && errno != EEXIST) {
rhughes@241
   792
		fprintf(stderr, "failed to create rpms directory.\n");
rhughes@241
   793
		return 1;
rhughes@241
   794
	}
rhughes@241
   795
rhughes@241
   796
	set = razor_set_open(rawhide_repo_filename);
rhughes@241
   797
	pi = razor_package_iterator_create(set);
rhughes@241
   798
	while (razor_package_iterator_next(pi, &package,
rhughes@241
   799
					   &name, &version, &arch)) {
rhughes@241
   800
		if (pattern && fnmatch(pattern, name, 0) != 0)
rhughes@241
   801
			continue;
rhughes@241
   802
rhughes@241
   803
		matches++;
rhughes@241
   804
		snprintf(url, sizeof url,
rhughes@241
   805
			 "%s/Packages/%s-%s.%s.rpm",
rhughes@241
   806
			 yum_url, name, version, arch);
rhughes@241
   807
		snprintf(file, sizeof file,
rhughes@241
   808
			 "rpms/%s-%s.%s.rpm", name, version, arch);
rhughes@241
   809
		download_if_missing(url, file);
rhughes@241
   810
	}
rhughes@241
   811
	razor_package_iterator_destroy(pi);
rhughes@241
   812
	razor_set_destroy(set);
rhughes@241
   813
rhughes@241
   814
	if (matches == 0)
rhughes@241
   815
		fprintf(stderr, "no packages matched \"%s\"\n", pattern);
rhughes@241
   816
	else if (matches == 1)
rhughes@241
   817
		fprintf(stderr, "downloaded 1 package\n");
rhughes@241
   818
	else
rhughes@241
   819
		fprintf(stderr, "downloaded %d packages\n", matches);
rhughes@241
   820
rhughes@241
   821
	return 0;
rhughes@241
   822
}
rhughes@241
   823
jbowes@258
   824
static int
jbowes@258
   825
command_info(int argc, const char *argv[])
jbowes@258
   826
{
jbowes@258
   827
	struct razor_set *set;
jbowes@258
   828
	struct razor_package_iterator *pi;
jbowes@258
   829
	struct razor_package *package;
jbowes@258
   830
	const char *pattern = argv[0], *name, *version, *arch;
jbowes@258
   831
	const char *summary, *description, *url, *license;
jbowes@258
   832
jbowes@258
   833
	set = razor_set_open(repo_filename);
jbowes@288
   834
	if (set == NULL)
jbowes@288
   835
		return 1;
jbowes@288
   836
	if (razor_set_open_details(set, "system-details.repo"))
jbowes@288
   837
		return 1;
jbowes@258
   838
	pi = razor_package_iterator_create(set);
jbowes@258
   839
	while (razor_package_iterator_next(pi, &package,
jbowes@258
   840
					   &name, &version, &arch)) {
jbowes@258
   841
		if (pattern && fnmatch(pattern, name, 0) != 0)
jbowes@258
   842
			continue;
jbowes@258
   843
jbowes@258
   844
		razor_package_get_details (set, package, &summary, &description,
jbowes@258
   845
					   &url, &license);
jbowes@258
   846
jbowes@258
   847
		printf ("Name:        %s\n", name);
jbowes@258
   848
		printf ("Arch:        %s\n", arch);
jbowes@258
   849
		printf ("Version:     %s\n", version);
jbowes@258
   850
		printf ("URL:         %s\n", url);
jbowes@258
   851
		printf ("License:     %s\n", license);
jbowes@258
   852
		printf ("Summary:     %s\n", summary);
jbowes@258
   853
		printf ("Description:\n");
jbowes@258
   854
		printf ("%s\n", description);
jbowes@258
   855
		printf ("\n");
jbowes@258
   856
	}
jbowes@258
   857
	razor_package_iterator_destroy(pi);
jbowes@258
   858
	razor_set_destroy(set);
jbowes@258
   859
jbowes@258
   860
	return 0;
jbowes@258
   861
}
jbowes@258
   862
jbowes@292
   863
#define SEARCH_MAX 256
jbowes@292
   864
jbowes@292
   865
static int
jbowes@292
   866
command_search(int argc, const char *argv[])
jbowes@292
   867
{
jbowes@292
   868
	struct razor_set *set;
jbowes@292
   869
	struct razor_package_iterator *pi;
jbowes@292
   870
	struct razor_package *package;
jbowes@292
   871
	char pattern[SEARCH_MAX];
jbowes@292
   872
	const char *name, *version, *arch;
jbowes@292
   873
	const char *summary, *description, *url, *license;
jbowes@292
   874
jbowes@292
   875
	if (!argv[0]) {
jbowes@292
   876
		fprintf(stderr, "must specify a search term\n");
jbowes@292
   877
		return 1;
jbowes@292
   878
	}
jbowes@292
   879
jbowes@292
   880
	sprintf(pattern, "*%s*", argv[0]);
jbowes@292
   881
jbowes@292
   882
	set = razor_set_open(repo_filename);
jbowes@292
   883
	if (set == NULL)
jbowes@292
   884
		return 1;
jbowes@292
   885
	if (razor_set_open_details(set, "system-details.repo"))
jbowes@292
   886
		return 1;
jbowes@292
   887
jbowes@292
   888
	pi = razor_package_iterator_create(set);
jbowes@292
   889
	while (razor_package_iterator_next(pi, &package,
jbowes@292
   890
					   &name, &version, &arch)) {
jbowes@292
   891
		if (!fnmatch(pattern, name, 0))
jbowes@292
   892
			printf("%s-%s.%s\n", name, version, arch);
jbowes@292
   893
		else {
jbowes@292
   894
			razor_package_get_details (set, package, &summary,
jbowes@292
   895
						   &description, &url,
jbowes@292
   896
						   &license);
jbowes@292
   897
			if (!fnmatch(pattern, url, 0) ||
jbowes@292
   898
			    !fnmatch(pattern, summary, 0) ||
jbowes@292
   899
			    !fnmatch(pattern, description, 0))
jbowes@292
   900
				printf("%s-%s.%s\n", name, version, arch);
jbowes@292
   901
		}
jbowes@292
   902
	}
jbowes@292
   903
	razor_package_iterator_destroy(pi);
jbowes@292
   904
	razor_set_destroy(set);
jbowes@292
   905
jbowes@292
   906
	return 0;
jbowes@292
   907
}
jbowes@292
   908
rhughes@241
   909
static struct {
rhughes@241
   910
	const char *name;
rhughes@241
   911
	const char *description;
rhughes@241
   912
	int (*func)(int argc, const char *argv[]);
rhughes@241
   913
} razor_commands[] = {
rhughes@241
   914
	{ "list", "list all packages", command_list },
rhughes@241
   915
	{ "list-requires", "list all requires for the given package", command_list_requires },
rhughes@241
   916
	{ "list-provides", "list all provides for the given package", command_list_provides },
rhughes@241
   917
	{ "list-obsoletes", "list all obsoletes for the given package", command_list_obsoletes },
rhughes@241
   918
	{ "list-conflicts", "list all conflicts for the given package", command_list_conflicts },
rhughes@241
   919
	{ "list-files", "list files for package set", command_list_files },
rhughes@241
   920
	{ "list-file-packages", "list packages owning file", command_list_file_packages },
rhughes@241
   921
	{ "list-package-files", "list files in package", command_list_package_files },
rhughes@241
   922
	{ "what-requires", "list the packages that have the given requires", command_what_requires },
rhughes@241
   923
	{ "what-provides", "list the packages that have the given provides", command_what_provides },
rhughes@241
   924
	{ "import-yum", "import yum metadata files", command_import_yum },
rhughes@241
   925
	{ "import-rpmdb", "import the system rpm database", command_import_rpmdb },
rhughes@241
   926
	{ "import-rpms", "import rpms from the given directory", command_import_rpms },
rhughes@241
   927
	{ "update", "update all or specified packages", command_update },
rhughes@241
   928
	{ "remove", "remove specified packages", command_remove },
rhughes@241
   929
	{ "diff", "show diff between two package sets", command_diff },
rhughes@241
   930
	{ "install", "install rpm", command_install },
rhughes@241
   931
	{ "init", "init razor root", command_init },
jbowes@258
   932
	{ "download", "download packages", command_download },
jbowes@292
   933
	{ "info", "display package details", command_info },
jbowes@292
   934
	{ "search", "search package details", command_search }
rhughes@241
   935
};
rhughes@241
   936
rhughes@241
   937
static int
rhughes@241
   938
usage(void)
rhughes@241
   939
{
rhughes@241
   940
	int i;
rhughes@241
   941
rhughes@241
   942
	printf("usage:\n");
rhughes@241
   943
	for (i = 0; i < ARRAY_SIZE(razor_commands); i++)
rhughes@241
   944
		printf("  %-20s%s\n",
rhughes@241
   945
		       razor_commands[i].name, razor_commands[i].description);
rhughes@241
   946
rhughes@241
   947
	return 1;
rhughes@241
   948
}
rhughes@241
   949
rhughes@241
   950
int
rhughes@241
   951
main(int argc, const char *argv[])
rhughes@241
   952
{
rhughes@241
   953
	char *repo;
rhughes@241
   954
	int i;
rhughes@241
   955
rhughes@241
   956
	repo = getenv("RAZOR_REPO");
rhughes@241
   957
	if (repo != NULL)
rhughes@241
   958
		repo_filename = repo;
rhughes@241
   959
rhughes@241
   960
	yum_url = getenv("YUM_URL");
rhughes@241
   961
	if (yum_url == NULL)
rhughes@241
   962
		yum_url = YUM_URL;
rhughes@241
   963
rhughes@241
   964
	if (argc < 2)
rhughes@241
   965
		return usage();
rhughes@241
   966
rhughes@241
   967
	for (i = 0; i < ARRAY_SIZE(razor_commands); i++)
rhughes@241
   968
		if (strcmp(razor_commands[i].name, argv[1]) == 0)
rhughes@241
   969
			return razor_commands[i].func(argc - 2, argv + 2);
rhughes@241
   970
rhughes@241
   971
	return usage();
rhughes@241
   972
}