/*- * Copyright (c) 2009-2012 Juan Romero Pardines. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include "xbps_api_impl.h" /** * @file lib/transaction_dictionary.c * @brief Transaction handling routines * @defgroup transaction Transaction handling functions * * The following image shows off the full transaction dictionary returned * by xbps_transaction_prepare(). * * @image html images/xbps_transaction_dictionary.png * * Legend: * - Salmon bg box: The transaction dictionary. * - White bg box: mandatory objects. * - Grey bg box: optional objects. * - Green bg box: possible value set in the object, only one of them * will be set. * * Text inside of white boxes are the key associated with the object, its * data type is specified on its edge, i.e string, array, integer, dictionary. */ static int compute_transaction_stats(struct xbps_handle *xhp) { prop_dictionary_t pkg_metad; prop_object_iterator_t iter; prop_object_t obj; uint64_t tsize, dlsize, instsize, rmsize; uint32_t inst_pkgcnt, up_pkgcnt, cf_pkgcnt, rm_pkgcnt; int rv = 0; const char *tract, *pkgname, *repo; inst_pkgcnt = up_pkgcnt = cf_pkgcnt = rm_pkgcnt = 0; tsize = dlsize = instsize = rmsize = 0; iter = xbps_array_iter_from_dict(xhp->transd, "packages"); if (iter == NULL) return EINVAL; while ((obj = prop_object_iterator_next(iter)) != NULL) { /* * Count number of pkgs to be removed, configured, * installed and updated. */ prop_dictionary_get_cstring_nocopy(obj, "pkgname", &pkgname); prop_dictionary_get_cstring_nocopy(obj, "transaction", &tract); prop_dictionary_get_cstring_nocopy(obj, "repository", &repo); if (strcmp(tract, "configure") == 0) { cf_pkgcnt++; continue; } else if (strcmp(tract, "install") == 0) { inst_pkgcnt++; } else if (strcmp(tract, "update") == 0) { up_pkgcnt++; } else if (strcmp(tract, "remove") == 0) { rm_pkgcnt++; } tsize = 0; /* * If removing or updating a package, get installed_size * from pkg's metadata dictionary. */ if ((strcmp(tract, "remove") == 0) || (strcmp(tract, "update") == 0)) { pkg_metad = xbps_metadir_get_pkgd(xhp, pkgname); if (pkg_metad == NULL) continue; prop_dictionary_get_uint64(pkg_metad, "installed_size", &tsize); rmsize += tsize; } if ((strcmp(tract, "install") == 0) || (strcmp(tract, "update") == 0)) { prop_dictionary_get_uint64(obj, "installed_size", &tsize); instsize += tsize; if (xbps_check_is_repository_uri_remote(repo)) { prop_dictionary_get_uint64(obj, "filename-size", &tsize); dlsize += tsize; } } } if (inst_pkgcnt && !prop_dictionary_set_uint32(xhp->transd, "total-install-pkgs", inst_pkgcnt)) { rv = EINVAL; goto out; } if (up_pkgcnt && !prop_dictionary_set_uint32(xhp->transd, "total-update-pkgs", up_pkgcnt)) { rv = EINVAL; goto out; } if (cf_pkgcnt && !prop_dictionary_set_uint32(xhp->transd, "total-configure-pkgs", cf_pkgcnt)) { rv = EINVAL; goto out; } if (rm_pkgcnt && !prop_dictionary_set_uint32(xhp->transd, "total-remove-pkgs", rm_pkgcnt)) { rv = EINVAL; goto out; } if (instsize > rmsize) { instsize -= rmsize; rmsize = 0; } else if (rmsize > instsize) { rmsize -= instsize; instsize = 0; } else instsize = rmsize = 0; /* * Add object in transaction dictionary with total installed * size that it will take. */ if (!prop_dictionary_set_uint64(xhp->transd, "total-installed-size", instsize)) { rv = EINVAL; goto out; } /* * Add object in transaction dictionary with total download * size that needs to be sucked in. */ if (!prop_dictionary_set_uint64(xhp->transd, "total-download-size", dlsize)) { rv = EINVAL; goto out; } /* * Add object in transaction dictionary with total size to be * freed from packages to be removed. */ if (!prop_dictionary_set_uint64(xhp->transd, "total-removed-size", rmsize)) { rv = EINVAL; goto out; } out: prop_object_iterator_release(iter); return rv; } int HIDDEN xbps_transaction_init(struct xbps_handle *xhp) { prop_array_t unsorted, mdeps, conflicts; if (xhp->transd != NULL) return 0; if ((xhp->transd = prop_dictionary_create()) == NULL) return ENOMEM; if ((unsorted = prop_array_create()) == NULL) { prop_object_release(xhp->transd); xhp->transd = NULL; return ENOMEM; } if (!xbps_add_obj_to_dict(xhp->transd, unsorted, "unsorted_deps")) { prop_object_release(xhp->transd); xhp->transd = NULL; return EINVAL; } if ((mdeps = prop_array_create()) == NULL) { prop_object_release(xhp->transd); xhp->transd = NULL; return ENOMEM; } if (!xbps_add_obj_to_dict(xhp->transd, mdeps, "missing_deps")) { prop_object_release(xhp->transd); xhp->transd = NULL; return EINVAL; } if ((conflicts = prop_array_create()) == NULL) { prop_object_release(xhp->transd); xhp->transd = NULL; return ENOMEM; } if (!xbps_add_obj_to_dict(xhp->transd, conflicts, "conflicts")) { prop_object_release(xhp->transd); xhp->transd = NULL; return EINVAL; } return 0; } int xbps_transaction_prepare(struct xbps_handle *xhp) { prop_array_t mdeps, conflicts; int rv = 0; if (xhp->transd == NULL) return ENXIO; /* * If there are missing deps bail out. */ mdeps = prop_dictionary_get(xhp->transd, "missing_deps"); if (prop_array_count(mdeps) > 0) return ENODEV; /* * If there are package conflicts bail out. */ conflicts = prop_dictionary_get(xhp->transd, "conflicts"); if (prop_array_count(conflicts) > 0) return EAGAIN; /* * Check for packages to be replaced. */ if ((rv = xbps_transaction_package_replace(xhp)) != 0) { prop_object_release(xhp->transd); xhp->transd = NULL; return rv; } /* * Sort package dependencies if necessary. */ if ((rv = xbps_transaction_sort_pkg_deps(xhp)) != 0) { prop_object_release(xhp->transd); xhp->transd = NULL; return rv; } /* * Add transaction stats for total download/installed size, * number of packages to be installed, updated, configured * and removed to the transaction dictionary. */ if ((rv = compute_transaction_stats(xhp)) != 0) { prop_object_release(xhp->transd); xhp->transd = NULL; return rv; } /* * The missing deps and conflicts arrays are not necessary anymore. */ prop_dictionary_remove(xhp->transd, "missing_deps"); prop_dictionary_remove(xhp->transd, "conflicts"); prop_dictionary_make_immutable(xhp->transd); return 0; }