2 const char *__progname;
7 #include <rpm/rpmcli.h>
8 #include <rpm/rpmlib.h> /* RPMSIGTAG, rpmReadPackageFile .. */
9 #include <rpm/rpmbuild.h>
10 #include <rpm/rpmlog.h>
11 #include <rpm/rpmdb.h>
12 #include <rpm/rpmps.h>
13 #include <rpm/rpmts.h>
16 #include "lib/signature.h"
24 MODE_QUERY = (1 << 0),
25 MODE_VERIFY = (1 << 3),
26 #define MODES_QV (MODE_QUERY | MODE_VERIFY)
28 MODE_INSTALL = (1 << 1),
29 MODE_ERASE = (1 << 2),
30 #define MODES_IE (MODE_INSTALL | MODE_ERASE)
32 MODE_CHECKSIG = (1 << 6),
33 MODE_RESIGN = (1 << 7),
34 #define MODES_K (MODE_CHECKSIG | MODE_RESIGN)
36 MODE_INITDB = (1 << 10),
37 MODE_REBUILDDB = (1 << 12),
38 MODE_VERIFYDB = (1 << 13),
39 #define MODES_DB (MODE_INITDB | MODE_REBUILDDB | MODE_VERIFYDB)
45 #define MODES_FOR_DBPATH (MODES_IE | MODES_QV | MODES_DB)
46 #define MODES_FOR_NODEPS (MODES_IE | MODE_VERIFY)
47 #define MODES_FOR_TEST (MODES_IE)
48 #define MODES_FOR_ROOT (MODES_IE | MODES_QV | MODES_DB | MODES_K)
52 /* the structure describing the options we take and the defaults */
53 static struct poptOption optionsTable[] = {
56 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmQVSourcePoptTable, 0,
57 N_("Query/Verify package selection options:"),
59 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmQueryPoptTable, 0,
60 N_("Query options (with -q or --query):"),
62 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmVerifyPoptTable, 0,
63 N_("Verify options (with -V or --verify):"),
65 #endif /* IAM_RPMQV */
68 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmSignPoptTable, 0,
69 N_("Signature options:"),
74 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmDatabasePoptTable, 0,
75 N_("Database options:"),
77 #endif /* IAM_RPMDB */
80 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmInstallPoptTable, 0,
81 N_("Install/Upgrade/Erase options:"),
83 #endif /* IAM_RPMEIU */
85 { "quiet", '\0', 0, &quiet, 0, NULL, NULL},
87 { NULL, '\0', POPT_ARG_INCLUDE_TABLE, rpmcliAllPoptTable, 0,
88 N_("Common options for all rpm modes and executables:"),
96 int main(int argc, char *argv[])
99 enum modes bigMode = MODE_UNKNOWN;
101 #if defined(IAM_RPMQV)
102 QVA_t qva = &rpmQVKArgs;
106 struct rpmInstallArguments_s * ia = &rpmIArgs;
109 #if defined(IAM_RPMDB)
110 struct rpmDatabaseArguments_s * da = &rpmDBArgs;
113 #if defined(IAM_RPMK)
114 QVA_t ka = &rpmQVKArgs;
117 #if defined(IAM_RPMK)
118 char * passPhrase = "";
130 setprogname(argv[0]); /* Retrofit glibc __progname */
132 /* XXX glibc churn sanity */
133 if (__progname == NULL) {
134 if ((__progname = strrchr(argv[0], '/')) != NULL) __progname++;
135 else __progname = argv[0];
138 optCon = initCli("rpm", optionsTable, argc, argv);
140 /* Set the major mode based on argv[0] */
142 if (rstreq(__progname, "rpmquery")) bigMode = MODE_QUERY;
143 if (rstreq(__progname, "rpmverify")) bigMode = MODE_VERIFY;
146 #if defined(IAM_RPMQV)
147 /* Jumpstart option from argv[0] if necessary. */
149 case MODE_QUERY: qva->qva_mode = 'q'; break;
150 case MODE_VERIFY: qva->qva_mode = 'V'; break;
151 case MODE_CHECKSIG: qva->qva_mode = 'K'; break;
152 case MODE_RESIGN: qva->qva_mode = 'R'; break;
165 if (bigMode == MODE_UNKNOWN || (bigMode & MODES_DB)) {
167 if (bigMode != MODE_UNKNOWN)
168 argerror(_("only one major mode may be specified"));
170 bigMode = MODE_INITDB;
173 if (bigMode != MODE_UNKNOWN)
174 argerror(_("only one major mode may be specified"));
176 bigMode = MODE_REBUILDDB;
179 if (bigMode != MODE_UNKNOWN)
180 argerror(_("only one major mode may be specified"));
182 bigMode = MODE_VERIFYDB;
185 #endif /* IAM_RPMDB */
188 if (bigMode == MODE_UNKNOWN || (bigMode & MODES_QV)) {
189 switch (qva->qva_mode) {
190 case 'q': bigMode = MODE_QUERY; break;
191 case 'V': bigMode = MODE_VERIFY; break;
194 if (qva->qva_sourceCount) {
195 if (qva->qva_sourceCount > 2)
196 argerror(_("one type of query/verify may be performed at a "
199 if (qva->qva_flags && (bigMode & ~MODES_QV))
200 argerror(_("unexpected query flags"));
202 if (qva->qva_queryFormat && (bigMode & ~MODES_QV))
203 argerror(_("unexpected query format"));
205 if (qva->qva_source != RPMQV_PACKAGE && (bigMode & ~MODES_QV))
206 argerror(_("unexpected query source"));
208 #endif /* IAM_RPMQV */
211 if (bigMode == MODE_UNKNOWN || (bigMode & MODES_IE))
212 { int iflags = (ia->installInterfaceFlags &
213 (INSTALL_UPGRADE|INSTALL_FRESHEN|INSTALL_INSTALL));
214 int eflags = (ia->installInterfaceFlags & INSTALL_ERASE);
217 argerror(_("only one major mode may be specified"));
219 bigMode = MODE_INSTALL;
221 bigMode = MODE_ERASE;
223 #endif /* IAM_RPMEIU */
226 if (bigMode == MODE_UNKNOWN || (bigMode & MODES_K)) {
227 switch (ka->qva_mode) {
231 case RPMSIGN_IMPORT_PUBKEY:
232 case RPMSIGN_CHK_SIGNATURE:
233 bigMode = MODE_CHECKSIG;
236 case RPMSIGN_ADD_SIGNATURE:
237 case RPMSIGN_NEW_SIGNATURE:
238 case RPMSIGN_DEL_SIGNATURE:
239 bigMode = MODE_RESIGN;
240 ka->sign = (ka->qva_mode != RPMSIGN_DEL_SIGNATURE);
244 #endif /* IAM_RPMK */
246 #if defined(IAM_RPMEIU)
247 if (!( bigMode == MODE_INSTALL ) &&
248 (ia->probFilter & (RPMPROB_FILTER_REPLACEPKG | RPMPROB_FILTER_OLDPACKAGE)))
249 argerror(_("only installation, upgrading, rmsource and rmspec may be forced"));
250 if (bigMode != MODE_INSTALL && (ia->probFilter & RPMPROB_FILTER_FORCERELOCATE))
251 argerror(_("files may only be relocated during package installation"));
253 if (ia->relocations && ia->prefix)
254 argerror(_("cannot use --prefix with --relocate or --excludepath"));
256 if (bigMode != MODE_INSTALL && ia->relocations)
257 argerror(_("--relocate and --excludepath may only be used when installing new packages"));
259 if (bigMode != MODE_INSTALL && ia->prefix)
260 argerror(_("--prefix may only be used when installing new packages"));
262 if (ia->prefix && ia->prefix[0] != '/')
263 argerror(_("arguments to --prefix must begin with a /"));
265 if (bigMode != MODE_INSTALL && (ia->installInterfaceFlags & INSTALL_HASH))
266 argerror(_("--hash (-h) may only be specified during package "
269 if (bigMode != MODE_INSTALL && (ia->installInterfaceFlags & INSTALL_PERCENT))
270 argerror(_("--percent may only be specified during package "
273 if (bigMode != MODE_INSTALL && (ia->probFilter & RPMPROB_FILTER_REPLACEPKG))
274 argerror(_("--replacepkgs may only be specified during package "
277 if (bigMode != MODE_INSTALL && (ia->transFlags & RPMTRANS_FLAG_NODOCS))
278 argerror(_("--excludedocs may only be specified during package "
281 if (bigMode != MODE_INSTALL && ia->incldocs)
282 argerror(_("--includedocs may only be specified during package "
285 if (ia->incldocs && (ia->transFlags & RPMTRANS_FLAG_NODOCS))
286 argerror(_("only one of --excludedocs and --includedocs may be "
289 if (bigMode != MODE_INSTALL && (ia->probFilter & RPMPROB_FILTER_IGNOREARCH))
290 argerror(_("--ignorearch may only be specified during package "
293 if (bigMode != MODE_INSTALL && (ia->probFilter & RPMPROB_FILTER_IGNOREOS))
294 argerror(_("--ignoreos may only be specified during package "
297 if (bigMode != MODE_INSTALL && bigMode != MODE_ERASE &&
298 (ia->probFilter & (RPMPROB_FILTER_DISKSPACE|RPMPROB_FILTER_DISKNODES)))
299 argerror(_("--ignoresize may only be specified during package "
302 if ((ia->installInterfaceFlags & UNINSTALL_ALLMATCHES) && bigMode != MODE_ERASE)
303 argerror(_("--allmatches may only be specified during package "
306 if ((ia->transFlags & RPMTRANS_FLAG_ALLFILES) && bigMode != MODE_INSTALL)
307 argerror(_("--allfiles may only be specified during package "
310 if ((ia->transFlags & RPMTRANS_FLAG_JUSTDB) &&
311 bigMode != MODE_INSTALL && bigMode != MODE_ERASE)
312 argerror(_("--justdb may only be specified during package "
313 "installation and erasure"));
315 if (bigMode != MODE_INSTALL && bigMode != MODE_ERASE && bigMode != MODE_VERIFY &&
316 (ia->transFlags & (RPMTRANS_FLAG_NOSCRIPTS | _noTransScripts | _noTransTriggers)))
317 argerror(_("script disabling options may only be specified during "
318 "package installation and erasure"));
320 if (bigMode != MODE_INSTALL && bigMode != MODE_ERASE && bigMode != MODE_VERIFY &&
321 (ia->transFlags & (RPMTRANS_FLAG_NOTRIGGERS | _noTransTriggers)))
322 argerror(_("trigger disabling options may only be specified during "
323 "package installation and erasure"));
325 if (ia->noDeps & (bigMode & ~MODES_FOR_NODEPS))
326 argerror(_("--nodeps may only be specified during package "
327 "building, rebuilding, recompilation, installation,"
328 "erasure, and verification"));
330 if ((ia->transFlags & RPMTRANS_FLAG_TEST) && (bigMode & ~MODES_FOR_TEST))
331 argerror(_("--test may only be specified during package installation, "
332 "erasure, and building"));
333 #endif /* IAM_RPMEIU */
335 if (rpmcliRootDir && rpmcliRootDir[1] && (bigMode & ~MODES_FOR_ROOT))
336 argerror(_("--root (-r) may only be specified during "
337 "installation, erasure, querying, and "
338 "database rebuilds"));
340 if (rpmcliRootDir && rpmcliRootDir[0] != '/') {
341 argerror(_("arguments to --root (-r) must begin with a /"));
345 rpmSetVerbosity(RPMLOG_WARNING);
349 if (bigMode == MODE_RESIGN) {
354 if ((av = poptGetArgs(optCon)) == NULL) {
355 fprintf(stderr, _("no files to sign\n"));
359 if (stat(*av, &sb)) {
360 fprintf(stderr, _("cannot access file %s\n"), *av);
371 if (poptPeekArg(optCon)) {
372 int sigTag = rpmLookupSignatureType(RPMLOOKUPSIG_QUERY);
380 passPhrase = rpmGetPassPhrase(_("Enter pass phrase: "), sigTag);
381 if (passPhrase == NULL) {
382 fprintf(stderr, _("Pass phrase check failed\n"));
386 fprintf(stderr, _("Pass phrase is good.\n"));
387 passPhrase = xstrdup(passPhrase);
391 _("Invalid %%_signature spec in macro file.\n"));
398 argerror(_("--sign may only be used during package building"));
401 /* Make rpmLookupSignatureType() return 0 ("none") from now on */
402 (void) rpmLookupSignatureType(RPMLOOKUPSIG_DISABLE);
404 #endif /* IAM_RPMK */
406 if (rpmcliPipeOutput) {
408 fprintf(stderr, _("creating a pipe for --pipe failed: %m\n"));
412 if (!(pipeChild = fork())) {
413 (void) signal(SIGPIPE, SIG_DFL);
415 (void) dup2(p[0], STDIN_FILENO);
417 (void) execl("/bin/sh", "/bin/sh", "-c", rpmcliPipeOutput, NULL);
418 fprintf(stderr, _("exec failed\n"));
422 (void) dup2(p[1], STDOUT_FILENO);
427 (void) rpmtsSetRootDir(ts, rpmcliRootDir);
431 ec = rpmtsInitDB(ts, 0644);
435 { rpmVSFlags vsflags = rpmExpandNumeric("%{_vsflags_rebuilddb}");
436 rpmVSFlags ovsflags = rpmtsSetVSFlags(ts, vsflags);
437 ec = rpmtsRebuildDB(ts);
438 vsflags = rpmtsSetVSFlags(ts, ovsflags);
441 ec = rpmtsVerifyDB(ts);
443 #endif /* IAM_RPMDB */
447 if (ia->noDeps) ia->installInterfaceFlags |= UNINSTALL_NODEPS;
449 if (!poptPeekArg(optCon)) {
450 argerror(_("no packages given for erase"));
452 ec += rpmErase(ts, ia, (ARGV_const_t) poptGetArgs(optCon));
458 /* RPMTRANS_FLAG_KEEPOBSOLETE */
461 if (ia->transFlags & RPMTRANS_FLAG_NODOCS) {
463 } else if (rpmExpandNumeric("%{_excludedocs}"))
464 ia->transFlags |= RPMTRANS_FLAG_NODOCS;
467 if (ia->noDeps) ia->installInterfaceFlags |= INSTALL_NODEPS;
469 /* we've already ensured !(!ia->prefix && !ia->relocations) */
471 ia->relocations = xmalloc(2 * sizeof(*ia->relocations));
472 ia->relocations[0].oldPath = NULL; /* special case magic */
473 ia->relocations[0].newPath = ia->prefix;
474 ia->relocations[1].oldPath = NULL;
475 ia->relocations[1].newPath = NULL;
476 } else if (ia->relocations) {
477 ia->relocations = xrealloc(ia->relocations,
478 sizeof(*ia->relocations) * (ia->numRelocations + 1));
479 ia->relocations[ia->numRelocations].oldPath = NULL;
480 ia->relocations[ia->numRelocations].newPath = NULL;
483 if (!poptPeekArg(optCon)) {
484 argerror(_("no packages given for install"));
486 /* FIX: ia->relocations[0].newPath undefined */
487 ec += rpmInstall(ts, ia, (ARGV_t) poptGetArgs(optCon));
491 #endif /* IAM_RPMEIU */
495 if (!poptPeekArg(optCon) && !(qva->qva_source == RPMQV_ALL))
496 argerror(_("no arguments given for query"));
498 qva->qva_specQuery = rpmspecQuery;
499 ec = rpmcliQuery(ts, qva, (ARGV_const_t) poptGetArgs(optCon));
500 qva->qva_specQuery = NULL;
504 { rpmVerifyFlags verifyFlags = VERIFY_ALL;
506 verifyFlags &= ~qva->qva_flags;
507 qva->qva_flags = (rpmQueryFlags) verifyFlags;
509 if (!poptPeekArg(optCon) && !(qva->qva_source == RPMQV_ALL))
510 argerror(_("no arguments given for verify"));
511 ec = rpmcliVerify(ts, qva, (ARGV_const_t) poptGetArgs(optCon));
513 #endif /* IAM_RPMQV */
517 { rpmVerifyFlags verifyFlags =
518 (VERIFY_FILEDIGEST|VERIFY_DIGEST|VERIFY_SIGNATURE);
520 verifyFlags &= ~ka->qva_flags;
521 ka->qva_flags = (rpmQueryFlags) verifyFlags;
524 if (!poptPeekArg(optCon))
525 argerror(_("no arguments given"));
526 ka->passPhrase = passPhrase;
527 ec = rpmcliSign(ts, ka, (ARGV_const_t) poptGetArgs(optCon));
529 #endif /* IAM_RPMK */
531 #if !defined(IAM_RPMQV)
535 #if !defined(IAM_RPMK)
539 #if !defined(IAM_RPMDB)
544 #if !defined(IAM_RPMEIU)
549 if (poptPeekArg(optCon) != NULL || argc <= 1 || rpmIsVerbose()) {
550 printUsage(optCon, stderr, 0);
561 (void) fclose(stdout);
562 (void) waitpid(pipeChild, &status, 0);
566 qva->qva_queryFormat = _free(qva->qva_queryFormat);
570 if (ia->relocations != NULL)
571 for (i = 0; i < ia->numRelocations; i++)
572 ia->relocations[i].oldPath = _free(ia->relocations[i].oldPath);
573 ia->relocations = _free(ia->relocations);
576 return finishCli(optCon, ec);