help: auto construct help

And add some more verbosity to the help message.
cute-signatures
Timo Teras 2009-06-25 15:14:07 +03:00
parent 3a48856475
commit 7a29678aac
13 changed files with 265 additions and 143 deletions

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -51,49 +51,41 @@ static void md5_str(const char *str, md5sum_t csum)
static int cup(void) static int cup(void)
{ {
/* compressed/uncompressed size is 259/1213 */ /* compressed/uncompressed size is 259/1213 */
static unsigned char z[] = { static unsigned char z[] = {
0x78,0x9c,0x9d,0x94,0x3d,0x8e,0xc4,0x20, 0x78,0x9c,0x9d,0x94,0x3d,0x8e,0xc4,0x20,0x0c,0x85,0xfb,0x9c,
0xc,0x85,0xfb,0x9c,0xc2,0x72,0x43,0x46, 0xc2,0x72,0x43,0x46,0x8a,0x4d,0x3f,0x67,0x89,0x64,0x77,0x2b,
0x8a,0x4d,0x3f,0x67,0x89,0x64,0x77,0x2b, 0x6d,0xbb,0x6d,0x0e,0x3f,0xc6,0x84,0x4d,0x08,0x84,0x55,0xd6,
0x6d,0xbb,0x6d,0xe,0x3f,0xc6,0x84,0x4d, 0xa2,0xe0,0xef,0x7b,0x36,0xe1,0x11,0x80,0x6e,0xcc,0x53,0x7f,
0x8,0x84,0x55,0xd6,0xa2,0xe0,0xef,0x7b, 0x3e,0xc5,0xeb,0xcf,0x1d,0x20,0x22,0xcc,0x3c,0x53,0x8e,0x17,
0x36,0xe1,0x11,0x80,0x6e,0xcc,0x53,0x7f, 0xd9,0x80,0x6d,0xee,0x0e,0x61,0x42,0x3c,0x8b,0xcf,0xc7,0x12,
0x3e,0xc5,0xeb,0xcf,0x1d,0x20,0x22,0xcc, 0x22,0x71,0x8b,0x31,0x05,0xd5,0xb0,0x11,0x4b,0xa7,0x32,0x2f,
0x3c,0x53,0x8e,0x17,0xd9,0x80,0x6d,0xee, 0x80,0x69,0x6b,0xb0,0x98,0x40,0xe2,0xcd,0xba,0x6a,0xba,0xe4,
0xe,0x61,0x42,0x3c,0x8b,0xcf,0xc7,0x12, 0x65,0xed,0x61,0x23,0x44,0xb5,0x95,0x06,0x8b,0xde,0x6c,0x61,
0x22,0x71,0x8b,0x31,0x5,0xd5,0xb0,0x11, 0x70,0xde,0x0e,0xb6,0xed,0xc4,0x43,0x0c,0x56,0x6f,0x8f,0x31,
0x4b,0xa7,0x32,0x2f,0x80,0x69,0x6b,0xb0, 0xd0,0x35,0xb5,0xc7,0x58,0x06,0xff,0x81,0x49,0x84,0xb8,0x0e,
0x98,0x40,0xe2,0xcd,0xba,0x6a,0xba,0xe4, 0xb1,0xd8,0xc1,0x66,0x31,0x0e,0x46,0x5c,0x43,0xc9,0xef,0xe5,
0x65,0xed,0x61,0x23,0x44,0xb5,0x95,0x6, 0xdc,0x63,0xb1,0xdc,0x67,0x6d,0x31,0xb3,0xc9,0x69,0x74,0x87,
0x8b,0xde,0x6c,0x61,0x70,0xde,0xe,0xb6, 0xc7,0xa3,0x1b,0x6a,0xb3,0xbd,0x2f,0x3b,0xd5,0x0c,0x57,0x3b,
0xed,0xc4,0x43,0xc,0x56,0x6f,0x8f,0x31, 0xce,0x7c,0x5e,0xe5,0x48,0xd0,0x48,0x01,0x92,0x49,0x8b,0xf7,
0xd0,0x35,0xb5,0xc7,0x58,0x6,0xff,0x81, 0xfc,0x58,0x67,0xb3,0xf7,0x14,0x20,0x5c,0x4c,0x9e,0xcc,0xeb,
0x49,0x84,0xb8,0xe,0xb1,0xd8,0xc1,0x66, 0x78,0x7e,0x64,0xa6,0xa1,0xf5,0xb2,0x70,0x38,0x09,0x7c,0x7f,
0x31,0xe,0x46,0x5c,0x43,0xc9,0xef,0xe5, 0xfd,0xc0,0x8a,0x4e,0xc8,0x55,0xe8,0x12,0xe2,0x9f,0x1a,0xb1,
0xdc,0x63,0xb1,0xdc,0x67,0x6d,0x31,0xb3, 0xb9,0x82,0x52,0x02,0x7a,0xe5,0xf9,0xd9,0x88,0x47,0x79,0x3b,
0xc9,0x69,0x74,0x87,0xc7,0xa3,0x1b,0x6a, 0x46,0x61,0x27,0xf9,0x51,0xb1,0x17,0xb0,0x2c,0x0e,0xd5,0x39,
0xb3,0xbd,0x2f,0x3b,0xd5,0xc,0x57,0x3b, 0x2d,0x96,0x25,0x27,0xd6,0xd1,0x3f,0xa5,0x08,0xe1,0x9e,0x4e,
0xce,0x7c,0x5e,0xe5,0x48,0xd0,0x48,0x1, 0xa7,0xe9,0x03,0xb1,0x0a,0xb6,0x75
0x92,0x49,0x8b,0xf7,0xfc,0x58,0x67,0xb3, };
0xf7,0x14,0x20,0x5c,0x4c,0x9e,0xcc,0xeb,
0x78,0x7e,0x64,0xa6,0xa1,0xf5,0xb2,0x70,
0x38,0x9,0x7c,0x7f,0xfd,0xc0,0x8a,0x4e,
0xc8,0x55,0xe8,0x12,0xe2,0x9f,0x1a,0xb1,
0xb9,0x82,0x52,0x2,0x7a,0xe5,0xf9,0xd9,
0x88,0x47,0x79,0x3b,0x46,0x61,0x27,0xf9,
0x51,0xb1,0x17,0xb0,0x2c,0xe,0xd5,0x39,
0x2d,0x96,0x25,0x27,0xd6,0xd1,0x3f,0xa5,
0x8,0xe1,0x9e,0x4e,0xa7,0xe9,0x3,0xb1,
0xa,0xb6,0x75,};
unsigned char buf[1213]; unsigned char buf[1213];
unsigned long len = 1213; //sizeof(buf); unsigned long len = sizeof(buf);
uncompress(buf, &len, z, sizeof(z)); uncompress(buf, &len, z, sizeof(z));
write(STDOUT_FILENO, buf, len); write(STDOUT_FILENO, buf, len);
return 0; return 0;
} }
static int add_main(void *ctx, int argc, char **argv) static int add_main(void *ctx, int argc, char **argv)
{ {
struct add_ctx *actx = (struct add_ctx *) ctx; struct add_ctx *actx = (struct add_ctx *) ctx;
@ -104,21 +96,21 @@ static int add_main(void *ctx, int argc, char **argv)
struct apk_dependency virtdep; struct apk_dependency virtdep;
int i, r; int i, r;
if ((argc > 0) && (apk_verbosity > 1) && if ((argc > 0) && (apk_verbosity > 1) &&
(strcmp(argv[0], "coffee") == 0)) (strcmp(argv[0], "coffee") == 0))
return cup(); return cup();
r = apk_db_open(&db, apk_root, actx->open_flags | APK_OPENF_WRITE); r = apk_db_open(&db, apk_root, actx->open_flags | APK_OPENF_WRITE);
if (r != 0) if (r != 0)
return r; return r;
if (actx->virtpkg) { if (actx->virtpkg) {
virtpkg = apk_pkg_new(); virtpkg = apk_pkg_new();
if (virtpkg == NULL) { if (virtpkg == NULL) {
apk_error("Failed to allocate virtual meta package"); apk_error("Failed to allocate virtual meta package");
goto err; goto err;
} }
virtpkg->name = apk_db_get_name(&db, APK_BLOB_STR(actx->virtpkg)); virtpkg->name = apk_db_get_name(&db, APK_BLOB_STR(actx->virtpkg));
md5_str(virtpkg->name->name, virtpkg->csum); md5_str(virtpkg->name->name, virtpkg->csum);
virtpkg->version = strdup("0"); virtpkg->version = strdup("0");
virtpkg->description = strdup("virtual meta package"); virtpkg->description = strdup("virtual meta package");
@ -174,15 +166,21 @@ err:
return r; return r;
} }
static struct option add_options[] = { static struct apk_option add_options[] = {
{ "initdb", no_argument, NULL, 0x10000 }, { 0x10000, "initdb", "Initialize database" },
{ "upgrade", no_argument, NULL, 'u' }, { 'u', "upgrade", "Prefer to upgrade package" },
{ "virtual", required_argument, NULL, 't' }, { 't', "virtual",
"Instead of adding all the packages to 'world', create a new virtual "
"package with the listed dependencies and add that to 'world'. The "
"actions of the command are easily reverted by deleting the virtual "
"package.", required_argument, "NAME" },
}; };
static struct apk_applet apk_add = { static struct apk_applet apk_add = {
.name = "add", .name = "add",
.usage = "[--initdb] [-u|--upgrade] [-t|--virtual NAME] PACKAGE...", .help = "Add (or update) PACKAGEs to main dependencies and install "
"them, while ensuring that all dependencies are met.",
.arguments = "PACKAGE...",
.context_size = sizeof(struct add_ctx), .context_size = sizeof(struct add_ctx),
.num_options = ARRAY_SIZE(add_options), .num_options = ARRAY_SIZE(add_options),
.options = add_options, .options = add_options,

190
src/apk.c
View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -20,12 +20,29 @@
#include "apk_defines.h" #include "apk_defines.h"
#include "apk_applet.h" #include "apk_applet.h"
#include "apk_blob.h"
const char *apk_root; const char *apk_root;
struct apk_repository_url apk_repository_list; struct apk_repository_url apk_repository_list;
int apk_verbosity = 1, apk_cwd_fd; int apk_verbosity = 1, apk_cwd_fd;
unsigned int apk_flags = 0; unsigned int apk_flags = 0;
static struct apk_option generic_options[] = {
{ 'h', "help", "Show generic help or applet specific help" },
{ 'p', "root", "Install packages to DIR",
required_argument, "DIR" },
{ 'X', "repository", "Use packages from REPO",
required_argument, "REPO" },
{ 'q', "quiet", "Print less information" },
{ 'v', "verbose", "Print more information" },
{ 'V', "version", "Print program version and exit" },
{ 'f', "force", "Do what was asked even if it looks dangerous" },
{ 0x101, "progress", "Show a progress bar" },
{ 0x102, "clean-protected",
"Do not create .apk-new files to configuration dirs" },
{ 0x104, "simulate", "Show what would be done without actually doing it" },
};
void apk_log(const char *prefix, const char *format, ...) void apk_log(const char *prefix, const char *format, ...)
{ {
va_list va; va_list va;
@ -38,34 +55,116 @@ void apk_log(const char *prefix, const char *format, ...)
fprintf(stderr, "\n"); fprintf(stderr, "\n");
} }
int version(void) static int version(void)
{ {
printf("apk-tools " APK_VERSION "\n"); printf("apk-tools " APK_VERSION "\n");
return 0; return 0;
} }
int generic_usage(void) struct apk_indent {
{ int x;
struct apk_applet **a; int indent;
printf("usage: apk COMMAND [-h|--help] [-p|--root DIR] [-X|--repository REPO]\n" };
"\t\t [-q|--quiet] [-v|--verbose] [-V|--version] [-f|--force]\n"
"\t\t [--progress] [--clean-protected] [--simulate] [ARGS]...\n\n"
"commands: ");
for (a = &__start_apkapplets; a < &__stop_apkapplets; a++) { static int print_indented(struct apk_indent *i, apk_blob_t blob)
printf("%s ", {
(*a)->name); static const int wrap_length = 80;
if (i->x + blob.len + 1 >= wrap_length) {
i->x = i->indent;
printf("\n%*s", i->indent - 1, "");
} }
printf("\n\n"); i->x += printf(" %.*s", blob.len, blob.ptr);
return 1; return 0;
} }
int usage(struct apk_applet *applet) static void print_indented_words(struct apk_indent *i, const char *text)
{ {
apk_blob_for_each_segment(APK_BLOB_STR(text), " ",
(apk_blob_cb) print_indented, i);
}
static int format_option(char *buf, size_t len, struct apk_option *o,
const char *separator)
{
int i = 0;
if (o->val <= 0xff && isalnum(o->val)) {
i += snprintf(&buf[i], len - i, "-%c", o->val);
if (o->name != NULL)
i += snprintf(&buf[i], len - i, "%s", separator);
}
if (o->name != NULL)
i += snprintf(&buf[i], len - i, "--%s", o->name);
if (o->arg_name != NULL)
i += snprintf(&buf[i], len - i, " %s", o->arg_name);
return i;
}
static void print_usage(const char *cmd, const char *args, int num_opts,
struct apk_option *opts)
{
struct apk_indent indent = { 0, 11 };
char word[128];
int i, j;
indent.x = printf("\nusage: apk %s", cmd) - 1;
for (i = 0; i < num_opts; i++) {
j = 0;
word[j++] = '[';
j += format_option(&word[j], sizeof(word) - j, &opts[i], "|");
word[j++] = ']';
print_indented(&indent, APK_BLOB_PTR_LEN(word, j));
}
if (args != NULL)
print_indented(&indent, APK_BLOB_STR(args));
printf("\n");
}
static void print_options(int num_opts, struct apk_option *opts)
{
struct apk_indent indent = { 0, 26 };
char word[128];
int i;
for (i = 0; i < num_opts; i++) {
format_option(word, sizeof(word), &opts[i], ", ");
indent.x = printf(" %-*s", indent.indent - 3, word);
print_indented_words(&indent, opts[i].help);
printf("\n");
}
}
static int usage(struct apk_applet *applet)
{
struct apk_indent indent = { 0, 2 };
version(); version();
if (applet == NULL) if (applet == NULL) {
return generic_usage(); struct apk_applet **a;
printf("usage: apk %s %s\n\n", applet->name, applet->usage);
print_usage("COMMAND", "[ARGS]...",
ARRAY_SIZE(generic_options), generic_options);
printf("\navailable commands:\n ");
for (a = &__start_apkapplets; a < &__stop_apkapplets; a++)
printf("%s ", (*a)->name);
} else {
print_usage(applet->name, applet->arguments,
applet->num_options, applet->options);
printf("\ndescription:\n%*s", indent.indent - 1, "");
print_indented_words(&indent, applet->help);
}
printf("\n\ngeneric options:\n");
print_options(ARRAY_SIZE(generic_options), generic_options);
if (applet != NULL && applet->num_options > 0) {
printf("\noptions for %s command:\n", applet->name);
print_options(applet->num_options, applet->options);
}
printf("\nThis apk has coffee making abilities.\n\n");
return 1; return 1;
} }
@ -77,7 +176,7 @@ static struct apk_applet *find_applet(const char *name)
if (strcmp(name, (*a)->name) == 0) if (strcmp(name, (*a)->name) == 0)
return *a; return *a;
} }
return NULL; return NULL;
} }
@ -110,7 +209,7 @@ static struct apk_applet *deduce_applet(int argc, char **argv)
static struct apk_repository_url *apk_repository_new(const char *url) static struct apk_repository_url *apk_repository_new(const char *url)
{ {
struct apk_repository_url *r = calloc(1, struct apk_repository_url *r = calloc(1,
sizeof(struct apk_repository_url)); sizeof(struct apk_repository_url));
if (r) { if (r) {
r->url = url; r->url = url;
@ -119,26 +218,25 @@ static struct apk_repository_url *apk_repository_new(const char *url)
return r; return r;
} }
#define NUM_GENERIC_OPTS 10 static void merge_options(struct option *opts, struct apk_option *ao, int num)
static struct option generic_options[32] = { {
{ "help", no_argument, NULL, 'h'}, int i;
{ "root", required_argument, NULL, 'p' },
{ "repository", required_argument, NULL, 'X' }, for (i = 0; i < num; i++, opts++, ao++) {
{ "quiet", no_argument, NULL, 'q' }, opts->name = ao->name;
{ "verbose", no_argument, NULL, 'v' }, opts->has_arg = ao->has_arg;
{ "version", no_argument, NULL, 'V' }, opts->flag = NULL;
{ "force", no_argument, NULL, 'f' }, opts->val = ao->val;
{ "progress", no_argument, NULL, 0x101 }, }
{ "clean-protected", no_argument, NULL, 0x102 }, opts->name = NULL;
{ "simulate", no_argument, NULL, 0x104 }, }
};
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
struct apk_applet *applet; struct apk_applet *applet;
char short_options[256], *sopt; char short_options[256], *sopt;
struct option *opt; struct option *opt, *all_options;
int r, optindex; int r, optindex, num_options;
void *ctx = NULL; void *ctx = NULL;
struct apk_repository_url *repo = NULL; struct apk_repository_url *repo = NULL;
@ -148,19 +246,20 @@ int main(int argc, char **argv)
list_init(&apk_repository_list.list); list_init(&apk_repository_list.list);
applet = deduce_applet(argc, argv); applet = deduce_applet(argc, argv);
num_options = ARRAY_SIZE(generic_options) + 1;
if (applet != NULL)
num_options += applet->num_options;
all_options = alloca(sizeof(struct option) * num_options);
merge_options(&all_options[0], generic_options,
ARRAY_SIZE(generic_options));
if (applet != NULL) { if (applet != NULL) {
if (applet->num_options && applet->options) { merge_options(&all_options[ARRAY_SIZE(generic_options)],
memcpy(&generic_options[NUM_GENERIC_OPTS], applet->options, applet->num_options);
applet->options,
applet->num_options * sizeof(struct option));
}
if (applet->context_size != 0) if (applet->context_size != 0)
ctx = calloc(1, applet->context_size); ctx = calloc(1, applet->context_size);
} }
for (opt = &generic_options[0], sopt = short_options; for (opt = all_options, sopt = short_options; opt->name != NULL; opt++) {
opt->name != NULL; opt++) {
if (opt->flag == NULL && if (opt->flag == NULL &&
opt->val <= 0xff && isalnum(opt->val)) { opt->val <= 0xff && isalnum(opt->val)) {
*(sopt++) = opt->val; *(sopt++) = opt->val;
@ -171,7 +270,7 @@ int main(int argc, char **argv)
optindex = 0; optindex = 0;
while ((r = getopt_long(argc, argv, short_options, while ((r = getopt_long(argc, argv, short_options,
generic_options, &optindex)) != -1) { all_options, &optindex)) != -1) {
switch (r) { switch (r) {
case 0: case 0:
break; break;
@ -208,7 +307,8 @@ int main(int argc, char **argv)
break; break;
default: default:
if (applet == NULL || applet->parse == NULL || if (applet == NULL || applet->parse == NULL ||
applet->parse(ctx, r, optindex - NUM_GENERIC_OPTS, applet->parse(ctx, r,
optindex - ARRAY_SIZE(generic_options),
optarg) != 0) optarg) != 0)
return usage(applet); return usage(applet);
break; break;

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -24,13 +24,22 @@ struct apk_repository_url {
extern struct apk_repository_url apk_repository_list; extern struct apk_repository_url apk_repository_list;
struct apk_option {
int val;
const char *name;
const char *help;
int has_arg;
const char *arg_name;
};
struct apk_applet { struct apk_applet {
const char *name; const char *name;
const char *usage; const char *arguments;
const char *help;
int context_size; int context_size;
int num_options; int num_options;
struct option *options; struct apk_option *options;
int (*parse)(void *ctx, int optch, int optindex, const char *optarg); int (*parse)(void *ctx, int optch, int optindex, const char *optarg);
int (*main)(void *ctx, int argc, char **argv); int (*main)(void *ctx, int argc, char **argv);

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -19,6 +19,7 @@ struct apk_blob {
char *ptr; char *ptr;
}; };
typedef struct apk_blob apk_blob_t; typedef struct apk_blob apk_blob_t;
typedef int (*apk_blob_cb)(void *ctx, apk_blob_t blob);
#define APK_BLOB_IS_NULL(blob) (blob.ptr == NULL) #define APK_BLOB_IS_NULL(blob) (blob.ptr == NULL)
@ -44,7 +45,7 @@ unsigned long apk_blob_hash(apk_blob_t str);
int apk_blob_compare(apk_blob_t a, apk_blob_t b); int apk_blob_compare(apk_blob_t a, apk_blob_t b);
int apk_blob_for_each_segment(apk_blob_t blob, const char *split, int apk_blob_for_each_segment(apk_blob_t blob, const char *split,
int (*cb)(void *ctx, apk_blob_t blob), void *ctx); apk_blob_cb cb, void *ctx);
int apk_hexdump_parse(apk_blob_t to, apk_blob_t from); int apk_hexdump_parse(apk_blob_t to, apk_blob_t from);
int apk_hexdump_format(int tolen, char *to, apk_blob_t from); int apk_hexdump_format(int tolen, char *to, apk_blob_t from);

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -128,13 +128,15 @@ static int audit_main(void *ctx, int argc, char **argv)
return r; return r;
} }
static struct option audit_options[] = { static struct apk_option audit_options[] = {
{ "backup", no_argument, NULL, 0x10000 }, { 0x10000, "backup",
"List all modified configuration files that need to be backed up" },
}; };
static struct apk_applet apk_audit = { static struct apk_applet apk_audit = {
.name = "audit", .name = "audit",
.usage = "--backup", .help = "Audit the filesystem for changes compared to installed "
"database.",
.context_size = sizeof(struct audit_ctx), .context_size = sizeof(struct audit_ctx),
.num_options = ARRAY_SIZE(audit_options), .num_options = ARRAY_SIZE(audit_options),
.options = audit_options, .options = audit_options,

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -60,7 +60,8 @@ out:
static struct apk_applet apk_del = { static struct apk_applet apk_del = {
.name = "del", .name = "del",
.usage = "PACKAGE...", .help = "Remove PACKAGEs from the main dependencies and uninstall them.",
.arguments = "PACKAGE...",
.main = del_main, .main = del_main,
}; };

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -184,17 +184,20 @@ err:
return r; return r;
} }
static struct option fetch_options[] = { static struct apk_option fetch_options[] = {
{ "recursive", no_argument, NULL, 'R' }, { 'l', "link", "Create hard links if possible" },
{ "stdout", no_argument, NULL, 's' }, { 'R', "recursive", "Fetch the PACKAGE and all it's dependencies" },
{ "link", no_argument, NULL, 'L' }, { 's', "stdout",
{ "output", required_argument, NULL, 'o' }, "Dump the .apk to stdout (incompatible with -o and -R)" },
{ 'o', "output", "Directory to place the PACKAGEs to",
required_argument, "DIR" },
}; };
static struct apk_applet apk_fetch = { static struct apk_applet apk_fetch = {
.name = "fetch", .name = "fetch",
.usage = "[-R|--recursive|--stdout] [-L|--link] [-o|--output DIR]\n" .help = "Download PACKAGEs from repositories to a local directory from "
"\t\t PACKAGE...", "which a local mirror repository can be created.",
.arguments = "PACKAGE...",
.context_size = sizeof(struct fetch_ctx), .context_size = sizeof(struct fetch_ctx),
.num_options = ARRAY_SIZE(fetch_options), .num_options = ARRAY_SIZE(fetch_options),
.options = fetch_options, .options = fetch_options,

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -113,13 +113,16 @@ static int index_main(void *ctx, int argc, char **argv)
return 0; return 0;
} }
static struct option index_options[] = { static struct apk_option index_options[] = {
{ "delete", required_argument, NULL, 'd' }, { 'd', "delete",
"Read existing INDEXFILE and delete the listed FILEs from it",
required_argument, "INDEXFILE" },
}; };
static struct apk_applet apk_index = { static struct apk_applet apk_index = {
.name = "index", .name = "index",
.usage = "[-d|--delete INDEXFILE] FILE...", .help = "Create repository index file from FILEs.",
.arguments = "FILE...",
.context_size = sizeof(struct index_ctx), .context_size = sizeof(struct index_ctx),
.num_options = ARRAY_SIZE(index_options), .num_options = ARRAY_SIZE(index_options),
.options = index_options, .options = index_options,

View File

@ -4,7 +4,7 @@
* Copyright (C) 2009 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2009 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -88,7 +88,7 @@ static int info_exists(struct info_ctx *ctx, struct apk_database *db,
for (i = 0; i < argc; i++) { for (i = 0; i < argc; i++) {
name = apk_db_query_name(db, APK_BLOB_STR(argv[i])); name = apk_db_query_name(db, APK_BLOB_STR(argv[i]));
if (name == NULL) { if (name == NULL) {
ret++; ret++;
continue; continue;
} }
@ -199,7 +199,7 @@ static void info_print_depends(struct apk_package *pkg)
{ {
int i; int i;
char *separator = apk_verbosity > 1 ? " " : "\n"; char *separator = apk_verbosity > 1 ? " " : "\n";
if (apk_verbosity == 1) if (apk_verbosity == 1)
printf("%s-%s depends on:\n", pkg->name->name, pkg->version); printf("%s-%s depends on:\n", pkg->name->name, pkg->version);
if (pkg->depends == NULL) if (pkg->depends == NULL)
return; return;
@ -325,21 +325,21 @@ static int info_main(void *ctx, int argc, char **argv)
return r; return r;
} }
static struct option info_options[] = { static struct apk_option info_options[] = {
{ "contents", no_argument, NULL, 'L' }, { 'L', "contents", "List contents of the PACKAGE" },
{ "installed", no_argument, NULL, 'e' }, { 'e', "installed", "Check if PACKAGE is installed" },
{ "who-owns", no_argument, NULL, 'W' }, { 'W', "who-owns", "Print the package owning the specified file" },
{ "depends", no_argument, NULL, 'R' }, { 'R', "depends", "List packages that the PACKAGE depends on" },
{ "rdepends", no_argument, NULL, 'r' }, { 'r', "rdepends", "List all packages depending on PACKAGE" },
{ "webpage", no_argument, NULL, 'w' }, { 'w', "webpage", "Show URL for more information about PACKAGE" },
{ "size", no_argument, NULL, 's' }, { 's', "size", "Show installed size of PACKAGE" },
{ "description", no_argument, NULL, 'd' }, { 'd', "description", "Print description for PACKAGE" },
}; };
static struct apk_applet apk_info = { static struct apk_applet apk_info = {
.name = "info", .name = "info",
.usage = "[-L|--contents] [-e|--installed] [-W|--who-owns] [-R|--depends]\n" .help = "Give detailed information about PACKAGEs.",
"\t\t[-r|--rdepends] [-w|--webage] [-s|--size] PACKAGE...", .arguments = "PACKAGE...",
.context_size = sizeof(struct info_ctx), .context_size = sizeof(struct info_ctx),
.num_options = ARRAY_SIZE(info_options), .num_options = ARRAY_SIZE(info_options),
.options = info_options, .options = info_options,

View File

@ -130,13 +130,14 @@ static int search_main(void *ctx, int argc, char **argv)
return r; return r;
} }
static struct option search_options[] = { static struct apk_option search_options[] = {
{ "description", no_argument, NULL, 'd' }, { 'd', "description", "Search also package descriptions" },
}; };
static struct apk_applet apk_search = { static struct apk_applet apk_search = {
.name = "search", .name = "search",
.usage = "[--description|-d] PATTERN", .help = "Search package names (and descriptions) by wildcard PATTERN.",
.arguments = "PATTERN",
.context_size = sizeof(struct search_ctx), .context_size = sizeof(struct search_ctx),
.num_options = ARRAY_SIZE(search_options), .num_options = ARRAY_SIZE(search_options),
.options = search_options, .options = search_options,

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -30,7 +30,7 @@ static int update_main(void *ctx, int argc, char **argv)
static struct apk_applet apk_update = { static struct apk_applet apk_update = {
.name = "update", .name = "update",
.usage = "", .help = "Update repository indexes from all remote repositories.",
.main = update_main, .main = update_main,
}; };

View File

@ -57,13 +57,16 @@ err:
return r; return r;
} }
static struct option upgrade_options[] = { static struct apk_option upgrade_options[] = {
{ "available", no_argument, NULL, 'a' }, { 'a', "available",
"Re-install or downgrade if currently installed package is not "
"currently available from any repository" },
}; };
static struct apk_applet apk_upgrade = { static struct apk_applet apk_upgrade = {
.name = "upgrade", .name = "upgrade",
.usage = "[-a|--available]", .help = "Upgrade (or downgrade with -a) the currently installed "
"packages to versions available in repositories.",
.num_options = ARRAY_SIZE(upgrade_options), .num_options = ARRAY_SIZE(upgrade_options),
.options = upgrade_options, .options = upgrade_options,
.parse = upgrade_parse, .parse = upgrade_parse,

View File

@ -4,7 +4,7 @@
* Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi> * Copyright (C) 2008 Timo Teräs <timo.teras@iki.fi>
* All rights reserved. * All rights reserved.
* *
* This program is free software; you can redistribute it and/or modify it * This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 as published * under the terms of the GNU General Public License version 2 as published
* by the Free Software Foundation. See http://www.gnu.org/ for details. * by the Free Software Foundation. See http://www.gnu.org/ for details.
*/ */
@ -57,7 +57,7 @@ static int ver_validate(int argc, char **argv)
} }
return r; return r;
} }
static int ver_parse(void *ctx, int opt, int optindex, const char *optarg) static int ver_parse(void *ctx, int opt, int optindex, const char *optarg)
{ {
struct ver_ctx *ictx = (struct ver_ctx *) ctx; struct ver_ctx *ictx = (struct ver_ctx *) ctx;
@ -84,7 +84,7 @@ static void ver_print_package_status(struct apk_package *pkg)
latest = pkg; latest = pkg;
for (i = 0; i < name->pkgs->num; i++) { for (i = 0; i < name->pkgs->num; i++) {
tmp = name->pkgs->item[i]; tmp = name->pkgs->item[i];
if (tmp->name != name || if (tmp->name != name ||
apk_pkg_get_state(tmp) == APK_PKG_INSTALLED) apk_pkg_get_state(tmp) == APK_PKG_INSTALLED)
continue; continue;
r = apk_pkg_version_compare(tmp, latest); r = apk_pkg_version_compare(tmp, latest);
@ -137,14 +137,15 @@ ver_exit:
return ret; return ret;
} }
static struct option ver_options[] = { static struct apk_option ver_options[] = {
{ "test", no_argument, NULL, 't' }, { 't', "test", "Compare two given versions" },
{ "check", no_argument, NULL, 'c' }, { 'c', "check", "Check if the given version string is valid" },
}; };
static struct apk_applet apk_ver = { static struct apk_applet apk_ver = {
.name = "version", .name = "version",
.usage = "[-t|--test version1 version2] [-c|--check]", .help = "Compare package versions (in installed database vs. available)"
" or do tests on version strings given on command line.",
.context_size = sizeof(struct ver_ctx), .context_size = sizeof(struct ver_ctx),
.num_options = ARRAY_SIZE(ver_options), .num_options = ARRAY_SIZE(ver_options),
.options = ver_options, .options = ver_options,