Branch data Line data Source code
1 : : /*-
2 : : * Copyright (c) 2011-2012 Baptiste Daroussin <bapt@FreeBSD.org>
3 : : * Copyright (c) 2011-2012 Julien Laffaye <jlaffaye@FreeBSD.org>
4 : : * Copyright (c) 2011-2012 Marin Atanasov Nikolov <dnaeon@gmail.com>
5 : : * Copyright (c) 2013-2014 Matthew Seaman <matthew@FreeBSD.org>
6 : : * Copyright (c) 2016 Vsevolod Stakhov <vsevolod@FreeBSD.org>
7 : : * All rights reserved.
8 : : *
9 : : * Redistribution and use in source and binary forms, with or without
10 : : * modification, are permitted provided that the following conditions
11 : : * are met:
12 : : * 1. Redistributions of source code must retain the above copyright
13 : : * notice, this list of conditions and the following disclaimer
14 : : * in this position and unchanged.
15 : : * 2. Redistributions in binary form must reproduce the above copyright
16 : : * notice, this list of conditions and the following disclaimer in the
17 : : * documentation and/or other materials provided with the distribution.
18 : : *
19 : : * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
20 : : * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 : : * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 : : * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
23 : : * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24 : : * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 : : * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 : : * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 : : * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 : : * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 : : */
30 : :
31 : : #include <err.h>
32 : : #include <getopt.h>
33 : : #include <stdio.h>
34 : : #include <string.h>
35 : : #include <unistd.h>
36 : :
37 : : #include <pkg.h>
38 : :
39 : : #include "pkgcli.h"
40 : :
41 : : void
42 : 0 : usage_delete(void)
43 : : {
44 : 0 : fprintf(stderr, "Usage: pkg delete [-DfnqRy] [-Cgix] <pkg-name> ...\n");
45 : 0 : fprintf(stderr, " pkg delete [-Dnqy] -a\n\n");
46 : 0 : fprintf(stderr, "For more information see 'pkg help delete'.\n");
47 : 0 : }
48 : :
49 : : int
50 : 25 : exec_delete(int argc, char **argv)
51 : : {
52 : 25 : struct pkg_jobs *jobs = NULL;
53 : 25 : struct pkgdb *db = NULL;
54 : 25 : match_t match = MATCH_EXACT;
55 : 25 : pkg_flags f = PKG_FLAG_NONE;
56 : 25 : bool recursive_flag = false, rc = false;
57 : 25 : int retcode = EXIT_FAILURE;
58 : : int ch;
59 : : int i;
60 : 25 : int lock_type = PKGDB_LOCK_ADVISORY;
61 : 25 : int locked_pkgs = 0;
62 : :
63 : 25 : struct option longopts[] = {
64 : : { "all", no_argument, NULL, 'a' },
65 : : { "case-sensitive", no_argument, NULL, 'C' },
66 : : { "no-scripts", no_argument, NULL, 'D' },
67 : : { "force", no_argument, NULL, 'f' },
68 : : { "glob", no_argument, NULL, 'g' },
69 : : { "case-insensitive", no_argument, NULL, 'i' },
70 : : { "dry-run", no_argument, NULL, 'n' },
71 : : { "quiet", no_argument, NULL, 'q' },
72 : : { "recursive", no_argument, NULL, 'R' },
73 : : { "regex", no_argument, NULL, 'x' },
74 : : { "yes", no_argument, NULL, 'y' },
75 : : { NULL, 0, NULL, 0 },
76 : : };
77 : :
78 : 25 : nbactions = nbdone = 0;
79 : :
80 [ + + ]: 62 : while ((ch = getopt_long(argc, argv, "+aCDfginqRxy", longopts, NULL)) != -1) {
81 [ + - - + : 37 : switch (ch) {
- - + + -
- + - ]
82 : : case 'a':
83 : 1 : match = MATCH_ALL;
84 : 1 : break;
85 : : case 'C':
86 : 0 : pkgdb_set_case_sensitivity(true);
87 : 0 : break;
88 : : case 'D':
89 : 0 : f |= PKG_FLAG_NOSCRIPT;
90 : 0 : break;
91 : : case 'f':
92 : 5 : f |= PKG_FLAG_FORCE;
93 : 5 : force = true;
94 : 5 : break;
95 : : case 'g':
96 : 0 : match = MATCH_GLOB;
97 : 0 : break;
98 : : case 'i':
99 : 0 : pkgdb_set_case_sensitivity(false);
100 : 0 : break;
101 : : case 'n':
102 : 1 : f |= PKG_FLAG_DRY_RUN;
103 : 1 : lock_type = PKGDB_LOCK_READONLY;
104 : 1 : dry_run = true;
105 : 1 : break;
106 : : case 'q':
107 : 6 : quiet = true;
108 : 6 : break;
109 : : case 'R':
110 : 0 : recursive_flag = true;
111 : 0 : break;
112 : : case 'x':
113 : 0 : match = MATCH_REGEX;
114 : 0 : break;
115 : : case 'y':
116 : 24 : yes = true;
117 : 24 : break;
118 : : default:
119 : 0 : usage_delete();
120 : 0 : return (EXIT_FAILURE);
121 : : }
122 : : }
123 : :
124 : 25 : argc -= optind;
125 : 25 : argv += optind;
126 : :
127 [ + + + - ]: 25 : if (argc < 1 && match != MATCH_ALL) {
128 : 0 : usage_delete();
129 : 0 : return (EXIT_FAILURE);
130 : : }
131 : :
132 [ + + ]: 25 : if (dry_run)
133 : 1 : retcode = pkgdb_access(PKGDB_MODE_READ, PKGDB_DB_LOCAL);
134 : : else
135 : 24 : retcode = pkgdb_access(PKGDB_MODE_READ|PKGDB_MODE_WRITE,
136 : : PKGDB_DB_LOCAL);
137 : :
138 [ - + ]: 25 : if (retcode == EPKG_ENODB) {
139 : 0 : warnx("No packages installed. Nothing to do!");
140 : 0 : return (EXIT_SUCCESS);
141 [ - + ]: 25 : } else if (retcode == EPKG_ENOACCESS) {
142 : 0 : warnx("Insufficient privileges to delete packages");
143 : 0 : return (EXIT_FAILURE);
144 [ - + ]: 25 : } else if (retcode != EPKG_OK) {
145 : 0 : warnx("Error accessing the package database");
146 : 0 : return (EXIT_FAILURE);
147 : : }
148 : :
149 [ - + ]: 25 : if (pkgdb_open(&db, PKGDB_DEFAULT) != EPKG_OK)
150 : 0 : return (EXIT_FAILURE);
151 : :
152 [ - + ]: 25 : if (pkgdb_obtain_lock(db, lock_type) != EPKG_OK) {
153 : 0 : pkgdb_close(db);
154 : 0 : warnx("Cannot get an advisory lock on a database, it is locked by another process");
155 : 0 : return (EXIT_FAILURE);
156 : : }
157 : :
158 : :
159 [ - + ]: 25 : if (pkg_jobs_new(&jobs, PKG_JOBS_DEINSTALL, db) != EPKG_OK) {
160 : 0 : pkgdb_close(db);
161 : 0 : return (EXIT_FAILURE);
162 : : }
163 : :
164 : : /*
165 : : * By default delete packages recursively.
166 : : * If force mode is enabled then we try to remove packages non-recursively.
167 : : * However, if -f and -R flags are both enabled then we return to
168 : : * recursive deletion.
169 : : */
170 [ + + - + ]: 25 : if (!force || recursive_flag)
171 : 20 : f |= PKG_FLAG_RECURSIVE;
172 : :
173 : 25 : pkg_jobs_set_flags(jobs, f);
174 : :
175 [ + + ]: 25 : if (match == MATCH_EXACT) {
176 [ + + ]: 48 : for (i = 0; i < argc; i++) {
177 [ - + ]: 24 : if (strchr(argv[i], '*') != NULL) {
178 : 0 : match = MATCH_GLOB;
179 : 0 : break;
180 : : }
181 : 24 : }
182 : 24 : }
183 : :
184 [ - + ]: 25 : if (pkg_jobs_add(jobs, match, argv, argc) == EPKG_FATAL)
185 : 0 : goto cleanup;
186 : :
187 [ - + ]: 25 : if (pkg_jobs_solve(jobs) != EPKG_OK) {
188 : 0 : fprintf(stderr, "Cannot perform request\n");
189 : 0 : retcode = EXIT_FAILURE;
190 : 0 : goto cleanup;
191 : : }
192 : :
193 [ + + ]: 25 : if (pkg_jobs_has_lockedpkgs(jobs)) {
194 : 2 : printf("The following package(s) are locked or vital and may not ");
195 : 2 : printf("be removed:\n\n");
196 : 2 : pkg_jobs_iter_lockedpkgs(jobs, print_pkg, &locked_pkgs);
197 [ - + ]: 2 : putchar('\n');
198 : 2 : }
199 : :
200 : : /* check if we have something to deinstall */
201 [ + + ]: 25 : if ((nbactions = pkg_jobs_count(jobs)) == 0) {
202 [ + - ]: 2 : if (argc == 0) {
203 [ # # ]: 0 : if (!quiet)
204 : 0 : printf("Nothing to do.\n");
205 : :
206 : 0 : retcode = EXIT_SUCCESS;
207 : 0 : goto cleanup;
208 : : }
209 [ + + ]: 2 : if (!quiet) {
210 : 1 : printf("%d packages requested for removal: "
211 : : "%d locked, %d missing\n",
212 : 1 : argc, locked_pkgs, argc - locked_pkgs);
213 : 1 : }
214 [ + - ]: 2 : if (locked_pkgs > 0) {
215 : 2 : retcode = EPKG_LOCKED;
216 : 2 : } else {
217 : 0 : retcode = EXIT_FAILURE;
218 : : }
219 : 2 : goto cleanup;
220 : : }
221 : :
222 [ + + - + ]: 23 : if (!quiet || dry_run) {
223 [ - + ]: 18 : if (!quiet) {
224 : 36 : print_jobs_summary(jobs,
225 : : "Deinstallation has been requested for the following %d packages "
226 : 18 : "(of %d packages in the universe):\n\n", nbactions,
227 : 18 : pkg_jobs_total(jobs));
228 : 18 : }
229 [ + + ]: 18 : if (dry_run) {
230 : 1 : retcode = EXIT_SUCCESS;
231 : 1 : goto cleanup;
232 : : }
233 : 17 : rc = query_yesno(false,
234 : : "\nProceed with deinstalling packages? ");
235 : 17 : }
236 : : else
237 : 5 : rc = yes;
238 : :
239 [ + - + + ]: 22 : if (!rc || (retcode = pkg_jobs_apply(jobs)) != EPKG_OK)
240 : 1 : goto cleanup;
241 : :
242 [ + + ]: 21 : if (messages != NULL) {
243 : 3 : fflush(messages->fp);
244 : 3 : printf("%s", messages->buf);
245 : 3 : }
246 : 21 : pkgdb_compact(db);
247 : :
248 : 21 : retcode = EXIT_SUCCESS;
249 : :
250 : : cleanup:
251 : 25 : pkgdb_release_lock(db, lock_type);
252 : 25 : pkg_jobs_free(jobs);
253 : 25 : pkgdb_close(db);
254 : :
255 : 25 : return (retcode);
256 : 25 : }
|