LCOV - code coverage report
Current view: top level - src - set.c (source / functions) Hit Total Coverage
Test: plop Lines: 138 219 63.0 %
Date: 2024-12-28 18:40:32 Functions: 2 3 66.7 %
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed Branches: 71 119 59.7 %

           Branch data     Line data    Source code
       1                 :            : /*-
       2                 :            :  * Copyright (c) 2012 Baptiste Daroussin <bapt@FreeBSD.org>
       3                 :            :  * Copyright (c) 2014 Matthew Seaman <matthew@FreeBSD.org>
       4                 :            :  * Copyright (c) 2014 Vsevolod Stakhov <vsevolod@FreeBSD.org>
       5                 :            :  * All rights reserved.
       6                 :            :  *
       7                 :            :  * Redistribution and use in source and binary forms, with or without
       8                 :            :  * modification, are permitted provided that the following conditions
       9                 :            :  * are met:
      10                 :            :  * 1. Redistributions of source code must retain the above copyright
      11                 :            :  *    notice, this list of conditions and the following disclaimer
      12                 :            :  *    in this position and unchanged.
      13                 :            :  * 2. Redistributions in binary form must reproduce the above copyright
      14                 :            :  *    notice, this list of conditions and the following disclaimer in the
      15                 :            :  *    documentation and/or other materials provided with the distribution.
      16                 :            :  *
      17                 :            :  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR(S) ``AS IS'' AND ANY EXPRESS OR
      18                 :            :  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
      19                 :            :  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
      20                 :            :  * IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY DIRECT, INDIRECT,
      21                 :            :  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
      22                 :            :  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
      23                 :            :  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
      24                 :            :  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
      25                 :            :  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
      26                 :            :  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
      27                 :            :  */
      28                 :            : 
      29                 :            : #include <err.h>
      30                 :            : #include <getopt.h>
      31                 :            : #include <stdio.h>
      32                 :            : #include <stdbool.h>
      33                 :            : #include <stdlib.h>
      34                 :            : #include <string.h>
      35                 :            : #include <unistd.h>
      36                 :            : 
      37                 :            : #include <pkg.h>
      38                 :            : 
      39                 :            : #include <xmalloc.h>
      40                 :            : 
      41                 :            : #include <bsd_compat.h>
      42                 :            : 
      43                 :            : #include "pkgcli.h"
      44                 :            : 
      45                 :            : #define AUTOMATIC 1U<<0
      46                 :            : #define ORIGIN 1U<<1
      47                 :            : #define NAME 1U<<2
      48                 :            : #define VITAL 1U<<3
      49                 :            : 
      50                 :            : void
      51                 :          0 : usage_set(void)
      52                 :            : {
      53                 :          0 :         fprintf(stderr, "Usage: pkg set [-a] [-A 0|1] [-o <oldorigin>:<neworigin>] [-n <oldname>:<newname>] [-y] [-Cgix] [-v 0|1] <pkg-name>\n\n");
      54                 :          0 :         fprintf(stderr, "For more information see 'pkg help set'. \n");
      55                 :          0 : }
      56                 :            : 
      57                 :            : static bool
      58                 :          2 : check_change_values(const char *opt, char **oldv, char **newv, char guard)
      59                 :            : {
      60                 :            :         const char *semicolon;
      61                 :            : 
      62         [ +  - ]:          2 :         if (opt == NULL)
      63                 :          0 :                 return (false);
      64                 :            : 
      65                 :          2 :         semicolon = strrchr(opt, ':');
      66                 :            : 
      67         [ +  - ]:          2 :         if (semicolon == NULL)
      68                 :          0 :                 return (false);
      69                 :            : 
      70                 :          2 :         *oldv = xmalloc(semicolon - opt + 1);
      71                 :          2 :         strlcpy(*oldv, opt, semicolon - opt + 1);
      72                 :          2 :         *newv = xstrdup(semicolon + 1);
      73                 :            : 
      74         [ +  + ]:          2 :         if (guard != '\0') {
      75                 :            :                 /* Check guard symbol in both new and old values */
      76   [ +  -  -  + ]:          1 :                 if (strrchr(*oldv, guard) == NULL ||
      77                 :          1 :                         strrchr(*newv, guard) == NULL) {
      78                 :          0 :                         free(*oldv);
      79                 :          0 :                         free(*newv);
      80                 :          0 :                         *oldv = NULL;
      81                 :          0 :                         *newv = NULL;
      82                 :            : 
      83                 :          0 :                         return (false);
      84                 :            :                 }
      85                 :          1 :         }
      86                 :            : 
      87                 :          2 :         return (true);
      88                 :          2 : }
      89                 :            : 
      90                 :            : int
      91                 :          6 : exec_set(int argc, char **argv)
      92                 :            : {
      93                 :          6 :         struct pkgdb    *db = NULL;
      94                 :          6 :         struct pkgdb_it *it = NULL;
      95                 :          6 :         struct pkg      *pkg = NULL;
      96                 :            :         int              ch;
      97                 :            :         int              i;
      98                 :          6 :         match_t          match = MATCH_EXACT;
      99                 :          6 :         int64_t          newautomatic = -1;
     100                 :          6 :         int64_t          newvital = -1;
     101                 :          6 :         bool             automatic = false;
     102                 :          6 :         bool             rc = false;
     103                 :          6 :         bool             vital = false;
     104                 :          6 :         const char      *changed = NULL;
     105                 :          6 :         char            *newvalue = NULL;
     106                 :          6 :         char            *oldvalue = NULL;
     107                 :          6 :         unsigned int     loads = PKG_LOAD_BASIC;
     108                 :          6 :         unsigned int     sets = 0;
     109                 :          6 :         unsigned int     field = 0, depfield = 0;
     110                 :            :         int              retcode;
     111                 :            : 
     112                 :          6 :         struct option longopts[] = {
     113                 :            :                 { "automatic",                required_argument,      NULL,   'A' },
     114                 :            :                 { "all",              no_argument,            NULL,   'a' },
     115                 :            :                 { "case-sensitive",   no_argument,            NULL,   'C' },
     116                 :            :                 { "glob",             no_argument,            NULL,   'g' },
     117                 :            :                 { "case-insensitive", no_argument,            NULL,   'i' },
     118                 :            :                 { "change-origin",    required_argument,      NULL,   'o' },
     119                 :            :                 { "change-name",      required_argument,      NULL,   'n' },
     120                 :            :                 { "regex",            no_argument,            NULL,   'x' },
     121                 :            :                 { "vital",            required_argument,      NULL,   'v' },
     122                 :            :                 { "yes",              no_argument,            NULL,   'y' },
     123                 :            :                 { NULL,                 0,                      NULL,   0   },
     124                 :            :         };
     125                 :            : 
     126         [ +  + ]:         18 :         while ((ch = getopt_long(argc, argv, "+A:aCgio:xyn:v:", longopts, NULL)) != -1) {
     127   [ -  +  -  -  :         12 :                 switch (ch) {
          -  -  +  +  -  
                   +  + ]
     128                 :            :                 case 'A':
     129                 :          2 :                         sets |= AUTOMATIC;
     130                 :          2 :                         newautomatic = optarg[0] - '0';
     131   [ +  +  -  + ]:          2 :                         if (newautomatic != 0 && newautomatic != 1)
     132                 :          0 :                                 errx(EXIT_FAILURE, "Wrong value for -A. "
     133                 :            :                                     "Expecting 0 or 1, got: %s",
     134                 :          0 :                                     optarg);
     135                 :          2 :                         break;
     136                 :            :                 case 'a':
     137                 :          0 :                         match = MATCH_ALL;
     138                 :          0 :                         break;
     139                 :            :                 case 'C':
     140                 :          0 :                         pkgdb_set_case_sensitivity(true);
     141                 :          0 :                         break;
     142                 :            :                 case 'g':
     143                 :          0 :                         match = MATCH_GLOB;
     144                 :          0 :                         break;
     145                 :            :                 case 'i':
     146                 :          0 :                         pkgdb_set_case_sensitivity(false);
     147                 :          0 :                         break;
     148                 :            :                 case 'o':
     149                 :          1 :                         sets |= ORIGIN;
     150                 :          1 :                         loads |= PKG_LOAD_DEPS;
     151                 :          1 :                         match = MATCH_ALL;
     152                 :          1 :                         changed = "origin";
     153         [ +  - ]:          1 :                         if (!check_change_values(optarg, &oldvalue, &newvalue, '/')) {
     154                 :          0 :                                  errx(EXIT_FAILURE, "Wrong format for -o. "
     155                 :            :                                          "Expecting oldorigin:neworigin, got: %s",
     156                 :          0 :                                          optarg);
     157                 :            :                         }
     158                 :          1 :                         break;
     159                 :            :                 case 'n':
     160                 :          1 :                         sets |= NAME;
     161                 :          1 :                         loads |= PKG_LOAD_DEPS;
     162                 :          1 :                         match = MATCH_ALL;
     163                 :          1 :                         changed = "name";
     164         [ +  - ]:          1 :                         if (!check_change_values(optarg, &oldvalue, &newvalue, '\0')) {
     165                 :          0 :                                  errx(EXIT_FAILURE, "Wrong format for -n. "
     166                 :            :                                          "Expecting oldname:newname, got: %s",
     167                 :          0 :                                          optarg);
     168                 :            :                         }
     169                 :          1 :                         break;
     170                 :            :                 case 'x':
     171                 :          0 :                         match = MATCH_REGEX;
     172                 :          0 :                         break;
     173                 :            :                 case 'v':
     174                 :          2 :                         sets |= VITAL;
     175                 :          2 :                         newvital = optarg[0] - '0';
     176   [ +  +  -  + ]:          2 :                         if (newvital != 0 && newvital != 1)
     177                 :          0 :                                 errx(EXIT_FAILURE, "Wrong value for -v. "
     178                 :            :                                     "Expecting 0 or 1, got: %s",
     179                 :          0 :                                     optarg);
     180                 :          2 :                         break;
     181                 :            :                 case 'y':
     182                 :          6 :                         yes = true;
     183                 :          6 :                         break;
     184                 :            :                 default:
     185                 :          0 :                         free(oldvalue);
     186                 :          0 :                         free(newvalue);
     187                 :          0 :                         usage_set();
     188                 :          0 :                         return (EXIT_FAILURE);
     189                 :            :                 }
     190                 :            :         }
     191                 :            : 
     192                 :          6 :         argc -= optind;
     193                 :          6 :         argv += optind;
     194                 :            : 
     195   [ +  +  +  -  :         12 :         if ((argc < 1 && match != MATCH_ALL) ||
                   -  + ]
     196   [ +  +  +  +  :          6 :                 (newautomatic == -1 && newvital == -1 && newvalue == NULL) ||
                   +  - ]
     197                 :          6 :                 (sets & (NAME|ORIGIN)) == (NAME|ORIGIN)) {
     198                 :          0 :                 free(newvalue);
     199                 :          0 :                 usage_set();
     200                 :          0 :                 return (EXIT_FAILURE);
     201                 :            :         }
     202                 :            : 
     203         [ +  + ]:          6 :         if (sets & NAME) {
     204                 :          1 :                 field = PKG_SET_NAME;
     205                 :          1 :                 depfield = PKG_SET_DEPNAME;
     206                 :          1 :         }
     207         [ +  + ]:          5 :         else if (sets & ORIGIN) {
     208                 :          1 :                 field = PKG_SET_ORIGIN;
     209                 :          1 :                 depfield = PKG_SET_DEPORIGIN;
     210                 :          1 :         }
     211                 :            : 
     212                 :          6 :         retcode = pkgdb_access(PKGDB_MODE_READ|PKGDB_MODE_WRITE,
     213                 :            :                                PKGDB_DB_LOCAL);
     214         [ -  + ]:          6 :         if (retcode == EPKG_ENODB) {
     215                 :          0 :                 free(newvalue);
     216         [ #  # ]:          0 :                 if (match == MATCH_ALL)
     217                 :          0 :                         return (EXIT_SUCCESS);
     218         [ #  # ]:          0 :                 if (!quiet)
     219                 :          0 :                         warnx("No packages installed.  Nothing to do!");
     220                 :          0 :                 return (EXIT_SUCCESS);
     221         [ -  + ]:          6 :         } else if (retcode == EPKG_ENOACCESS) {
     222                 :          0 :                 free(newvalue);
     223                 :          0 :                 warnx("Insufficient privileges to modify the package database");
     224                 :          0 :                 return (EXIT_FAILURE);
     225         [ -  + ]:          6 :         } else if (retcode != EPKG_OK) {
     226                 :          0 :                 warnx("Error accessing package database");
     227                 :          0 :                 free(newvalue);
     228                 :          0 :                 return (EXIT_FAILURE);
     229                 :            :         }
     230                 :            : 
     231         [ -  + ]:          6 :         if (pkgdb_open(&db, PKGDB_DEFAULT) != EPKG_OK) {
     232                 :          0 :                 free(newvalue);
     233                 :          0 :                 return (EXIT_FAILURE);
     234                 :            :         }
     235                 :            : 
     236         [ -  + ]:          6 :         if (pkgdb_obtain_lock(db, PKGDB_LOCK_EXCLUSIVE) != EPKG_OK) {
     237                 :          0 :                 pkgdb_close(db);
     238                 :          0 :                 free(newvalue);
     239                 :          0 :                 warnx("Cannot get an exclusive lock on a database, it is locked by another process");
     240                 :          0 :                 return (EXIT_FAILURE);
     241                 :            :         }
     242                 :            : 
     243         [ -  + ]:          6 :         if (pkgdb_transaction_begin(db, NULL) != EPKG_OK) {
     244                 :          0 :                 pkgdb_close(db);
     245                 :          0 :                 free(newvalue);
     246                 :          0 :                 warnx("Cannot start transaction for update");
     247                 :          0 :                 return (EXIT_FAILURE);
     248                 :            :         }
     249                 :            : 
     250                 :            : 
     251         [ +  + ]:          6 :         if (oldvalue != NULL) {
     252                 :          2 :                 match = MATCH_ALL;
     253         [ +  - ]:          2 :                 if ((it = pkgdb_query(db, oldvalue, MATCH_EXACT)) == NULL) {
     254                 :          0 :                         retcode = EXIT_FAILURE;
     255                 :          0 :                         goto cleanup;
     256                 :            :                 }
     257                 :            : 
     258         [ +  - ]:          2 :                 if (pkgdb_it_next(it, &pkg, PKG_LOAD_BASIC) != EPKG_OK) {
     259                 :          0 :                         pkg = NULL;
     260                 :            : /*                      fprintf(stderr, "%s not installed\n", oldorigin);
     261                 :            :                         free(oldorigin);
     262                 :            :                         pkgdb_it_free(it);
     263                 :            :                         pkgdb_close(db);
     264                 :            :                         return (EXIT_FAILURE);*/
     265                 :          0 :                 }
     266                 :            : 
     267                 :          2 :                 rc = yes;
     268         [ +  - ]:          2 :                 if (!yes) {
     269         [ #  # ]:          0 :                         if (pkg != NULL)
     270                 :          0 :                                 rc = query_yesno(false, "Change %S from %S to %S for %n-%v? ",
     271                 :          0 :                                                 changed, oldvalue, newvalue, pkg, pkg);
     272                 :            :                         else
     273                 :          0 :                                 rc = query_yesno(false, "Change %S from %S to %S for all dependencies? ",
     274                 :          0 :                                                 changed, oldvalue, newvalue);
     275                 :          0 :                 }
     276   [ +  -  -  + ]:          2 :                 if (pkg != NULL && rc) {
     277         [ -  + ]:          2 :                         if (pkgdb_set(db, pkg, field, newvalue) != EPKG_OK) {
     278                 :          0 :                                 retcode = EXIT_FAILURE;
     279                 :          0 :                                 goto cleanup;
     280                 :            :                         }
     281                 :          2 :                 }
     282                 :          2 :                 pkgdb_it_free(it);
     283                 :          2 :         }
     284                 :          6 :         i = 0;
     285                 :          6 :         do {
     286                 :          6 :                 bool saved_rc = rc;
     287                 :          6 :                 bool gotone = false;
     288                 :            : 
     289         [ -  + ]:          6 :                 if ((it = pkgdb_query(db, argv[i], match)) == NULL) {
     290                 :          0 :                         retcode = EXIT_FAILURE;
     291                 :          0 :                         goto cleanup;
     292                 :            :                 }
     293                 :            : 
     294         [ +  + ]:         12 :                 while (pkgdb_it_next(it, &pkg, loads) == EPKG_OK) {
     295                 :          6 :                         gotone = true;
     296         [ +  + ]:          6 :                         if ((sets & AUTOMATIC) == AUTOMATIC) {
     297                 :          2 :                                 pkg_get(pkg, PKG_ATTR_AUTOMATIC, &automatic);
     298         [ -  + ]:          2 :                                 if (automatic == newautomatic)
     299                 :          0 :                                         continue;
     300         [ -  + ]:          2 :                                 if (!rc) {
     301         [ +  + ]:          2 :                                         if (newautomatic)
     302                 :          1 :                                                 rc = query_yesno(false,
     303                 :            :                                                                 "Mark %n-%v as automatically installed? ",
     304                 :          1 :                                                                 pkg, pkg);
     305                 :            :                                         else
     306                 :          1 :                                                 rc = query_yesno(false,
     307                 :            :                                                                 "Mark %n-%v as not automatically installed? ",
     308                 :          1 :                                                                 pkg, pkg);
     309                 :          2 :                                 }
     310         [ -  + ]:          2 :                                 if (rc)
     311                 :          2 :                                         pkgdb_set(db, pkg, PKG_SET_AUTOMATIC, (int)newautomatic);
     312                 :          2 :                                 rc = saved_rc;
     313                 :          2 :                         }
     314         [ +  + ]:          6 :                         if ((sets & VITAL) == VITAL) {
     315                 :          2 :                                 pkg_get(pkg, PKG_ATTR_VITAL, &vital);
     316         [ -  + ]:          2 :                                 if (vital == newvital)
     317                 :          0 :                                         continue;
     318         [ -  + ]:          2 :                                 if (!rc) {
     319         [ +  + ]:          2 :                                         if (newvital)
     320                 :          1 :                                                 rc = query_yesno(false,
     321                 :            :                                                                 "Mark %n-%v as vital? ",
     322                 :          1 :                                                                 pkg, pkg);
     323                 :            :                                         else
     324                 :          1 :                                                 rc = query_yesno(false,
     325                 :            :                                                                 "Mark %n-%v as not vital? ",
     326                 :          1 :                                                                 pkg, pkg);
     327                 :          2 :                                 }
     328         [ -  + ]:          2 :                                 if (rc)
     329                 :          2 :                                         pkgdb_set(db, pkg, PKG_SET_VITAL, (int)newvital);
     330                 :          2 :                                 rc = saved_rc;
     331                 :          2 :                         }
     332         [ +  + ]:          6 :                         if (sets & (ORIGIN|NAME)) {
     333                 :          2 :                                 struct pkg_dep *d = NULL;
     334         [ -  + ]:          2 :                                 while (pkg_deps(pkg, &d) == EPKG_OK) {
     335                 :            :                                         /*
     336                 :            :                                          * Do not query user when he has already
     337                 :            :                                          * been queried.
     338                 :            :                                          */
     339         [ #  # ]:          0 :                                         if (pkgdb_set(db, pkg, depfield, oldvalue, newvalue) != EPKG_OK) {
     340                 :          0 :                                                 retcode = EXIT_FAILURE;
     341                 :          0 :                                                 goto cleanup;
     342                 :            :                                         }
     343                 :            :                                 }
     344                 :          2 :                         }
     345                 :            :                 }
     346         [ +  - ]:          6 :                 if (!gotone) {
     347                 :          0 :                         warnx("No package(s) matching %s", argv[i]);
     348                 :          0 :                         retcode = EXIT_FAILURE;
     349                 :          0 :                 }
     350                 :          6 :                 pkgdb_it_free(it);
     351                 :          6 :                 i++;
     352         [ -  + ]:          6 :         } while (i < argc);
     353                 :            : 
     354                 :            : cleanup:
     355                 :          6 :         free(oldvalue);
     356                 :          6 :         free(newvalue);
     357                 :          6 :         pkg_free(pkg);
     358                 :            : 
     359         [ +  - ]:          6 :         if (retcode == 0) {
     360                 :          6 :                 pkgdb_transaction_commit(db, NULL);
     361                 :          6 :         }
     362                 :            :         else {
     363                 :          0 :                 pkgdb_transaction_rollback(db, NULL);
     364                 :            :         }
     365                 :            : 
     366                 :          6 :         pkgdb_release_lock(db, PKGDB_LOCK_EXCLUSIVE);
     367                 :          6 :         pkgdb_close(db);
     368                 :            : 
     369                 :          6 :         return (retcode);
     370                 :          6 : }

Generated by: LCOV version 1.15