Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[RFC] lib: don't allow xbps self-update to bypass integrity checks #597

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions bin/xbps-install/state_cb.c
Original file line number Diff line number Diff line change
Expand Up @@ -158,6 +158,18 @@ state_cb(const struct xbps_state_cb_data *xscd, void *cbdata UNUSED)
case XBPS_STATE_UNPACK_FILE_PRESERVED:
printf("%s\n", xscd->desc);
break;
case XBPS_STATE_OUTOFDATE:
if (!xbps_pkg_name(pkgname, sizeof(pkgname), xscd->arg)) {
abort();
}
newver = xscd->arg;
pkgd = xbps_pkgdb_get_pkg(xscd->xhp, pkgname);
xbps_dictionary_get_cstring_nocopy(pkgd, "pkgver", &instver);
fprintf(stderr, "WARNING: The '%s' package is out of date, '%s' is available.\n", instver, newver);
if (xscd->desc)
fprintf(stderr, "%s\n", xscd->desc);
fprintf(stderr, "NOTE: This can be resolved by running `xbps-install -u`\n");
break;
/* errors */
case XBPS_STATE_TRANS_FAIL:
case XBPS_STATE_UNPACK_FAIL:
Expand Down
11 changes: 0 additions & 11 deletions bin/xbps-install/transaction.c
Original file line number Diff line number Diff line change
Expand Up @@ -316,13 +316,6 @@ dist_upgrade(struct xbps_handle *xhp, unsigned int cols, bool yes, bool drun)
if (rv == ENOENT) {
xbps_error_printf("No packages currently registered.\n");
return 0;
} else if (rv == EBUSY) {
if (drun) {
rv = 0;
} else {
xbps_error_printf("The 'xbps' package must be updated, please run `xbps-install -u xbps`\n");
return rv;
}
} else if (rv == EEXIST) {
return 0;
} else if (rv == ENOTSUP) {
Expand Down Expand Up @@ -350,8 +343,6 @@ install_new_pkg(struct xbps_handle *xhp, const char *pkg, bool force)
xbps_error_printf("No repositories currently registered!\n");
else if (rv == ENXIO)
xbps_error_printf("Package `%s' contains invalid dependencies, exiting.\n", pkg);
else if (rv == EBUSY)
xbps_error_printf("The 'xbps' package must be updated, please run `xbps-install -u xbps`\n");
else if (rv != 0) {
xbps_error_printf("Unexpected error: %s\n", strerror(rv));
rv = -1;
Expand All @@ -375,8 +366,6 @@ update_pkg(struct xbps_handle *xhp, const char *pkg, bool force)
xbps_error_printf("No repositories currently registered!\n");
else if (rv == ENXIO)
xbps_error_printf("Package `%s' contains invalid dependencies, exiting.\n", pkg);
else if (rv == EBUSY)
xbps_error_printf("The 'xbps' package must be updated, please run `xbps-install -u xbps`\n");
else if (rv != 0) {
xbps_error_printf("Unexpected error: %s\n", strerror(rv));
return -1;
Expand Down
3 changes: 2 additions & 1 deletion include/xbps.h.in
Original file line number Diff line number Diff line change
Expand Up @@ -374,7 +374,8 @@ typedef enum xbps_state {
XBPS_STATE_ALTGROUP_REMOVED,
XBPS_STATE_ALTGROUP_SWITCHED,
XBPS_STATE_ALTGROUP_LINK_ADDED,
XBPS_STATE_ALTGROUP_LINK_REMOVED
XBPS_STATE_ALTGROUP_LINK_REMOVED,
XBPS_STATE_OUTOFDATE
} xbps_state_t;

/**
Expand Down
1 change: 1 addition & 0 deletions lib/cb_util.c
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ xbps_set_cb_state(struct xbps_handle *xhp,
xscd.state = state;
xscd.err = err;
xscd.arg = arg;
xscd.desc = NULL;
if (fmt != NULL) {
va_start(va, fmt);
retval = vasprintf(&buf, fmt, va);
Expand Down
163 changes: 76 additions & 87 deletions lib/transaction_ops.c
Original file line number Diff line number Diff line change
Expand Up @@ -226,19 +226,18 @@ trans_find_pkg(struct xbps_handle *xhp, const char *pkg, bool force)
* Returns 1 if there's an update, 0 if none or -1 on error.
*/
static int
xbps_autoupdate(struct xbps_handle *xhp)
trans_check_uptodate(struct xbps_handle *xhp, const char *checkpkg, const char **new_pkgver)
{
xbps_array_t rdeps;
xbps_dictionary_t pkgd;
const char *pkgver = NULL, *pkgname = NULL;
int rv;
xbps_array_t pkgs;
xbps_dictionary_t pkgd = NULL, pkg_repod = NULL;
const char *pkgver = NULL, *pkgname = NULL, *repopkgver = NULL, *repoloc = NULL;

/*
* Check if there's a new update for XBPS before starting
* another transaction.
*/
if (((pkgd = xbps_pkgdb_get_pkg(xhp, "xbps")) == NULL) &&
((pkgd = xbps_pkgdb_get_virtualpkg(xhp, "xbps")) == NULL))
if (((pkgd = xbps_pkgdb_get_pkg(xhp, checkpkg)) == NULL) &&
((pkgd = xbps_pkgdb_get_virtualpkg(xhp, checkpkg)) == NULL))
return 0;

if (!xbps_dictionary_get_cstring_nocopy(pkgd, "pkgver", &pkgver)) {
Expand All @@ -248,52 +247,59 @@ xbps_autoupdate(struct xbps_handle *xhp)
return EINVAL;
}

rv = trans_find_pkg(xhp, pkgname, false);

xbps_dbg_printf("%s: trans_find_pkg xbps: %d\n", __func__, rv);

if (rv == 0) {
if (xhp->flags & XBPS_FLAG_DOWNLOAD_ONLY) {
if (xbps_dictionary_get(pkgd, "repolock")) {
struct xbps_repo *repo;
/* find update from repo */
xbps_dictionary_get_cstring_nocopy(pkgd, "repository", &repoloc);
assert(repoloc);
if ((repo = xbps_regget_repo(xhp, repoloc)) == NULL) {
/* not found */
return 0;
}
/* a new xbps version is available, check its revdeps */
rdeps = xbps_pkgdb_get_pkg_revdeps(xhp, "xbps");
for (unsigned int i = 0; i < xbps_array_count(rdeps); i++) {
const char *curpkgver = NULL;
char curpkgn[XBPS_NAME_SIZE] = {0};
pkg_repod = xbps_repo_get_pkg(repo, pkgname);
} else {
/* find update from rpool */
pkg_repod = xbps_rpool_get_pkg(xhp, pkgname);
}
if (pkg_repod == NULL) {
/* not found */
return 0;
}

xbps_array_get_cstring_nocopy(rdeps, i, &curpkgver);
xbps_dbg_printf("%s: processing revdep %s\n", __func__, curpkgver);
xbps_dictionary_get_cstring_nocopy(pkg_repod, "pkgver", &repopkgver);

/*
* Compare installed version vs best pkg available in repos
* for pkg updates.
*/
xbps_dictionary_get_cstring_nocopy(pkgd,
"pkgver", &pkgver);
if (xbps_cmpver(repopkgver, pkgver) <= 0 &&
!xbps_pkg_reverts(pkg_repod, pkgver)) {
xbps_dictionary_get_cstring_nocopy(pkg_repod,
"repository", &repoloc);
xbps_dbg_printf("%s: [rpool] Skipping `%s' "
"(installed: %s) from repository `%s'\n",
__func__, repopkgver, pkgver, repoloc);
return 0;
}

if (!xbps_pkg_name(curpkgn, sizeof(curpkgn), curpkgver)) {
abort();
}
rv = trans_find_pkg(xhp, curpkgn, false);
xbps_dbg_printf("%s: trans_find_pkg revdep %s: %d\n", __func__, curpkgver, rv);
if (rv && rv != ENOENT && rv != EEXIST && rv != ENODEV)
return -1;
}
/*
* Set XBPS_FLAG_FORCE_REMOVE_REVDEPS to ignore broken
* reverse dependencies in xbps_transaction_prepare().
*
* This won't skip revdeps of the xbps pkg, rather other
* packages in rootdir that could be broken indirectly.
*
* A sysup transaction after updating xbps should fix them
* again.
*/
xhp->flags |= XBPS_FLAG_FORCE_REMOVE_REVDEPS;
return 1;
} else if (rv == ENOENT || rv == EEXIST || rv == ENODEV) {
/* no update */
pkgs = xbps_dictionary_get(xhp->transd, "packages");
/*
* Find out if package being checked already has a queued
* transaction, in that case ignore it.
*/
if (xbps_find_pkg_in_array(pkgs, repopkgver, 0)) {
xbps_dbg_printf("%s: [update] `%s' already queued in "
"transaction.\n", __func__, repopkgver);
return 0;
} else {
/* error */
return -1;
}

return 0;
if (xhp->flags & XBPS_FLAG_DOWNLOAD_ONLY) {
return 0;
}
if (new_pkgver) *new_pkgver = repopkgver;
return 1;
}

int
Expand All @@ -305,17 +311,8 @@ xbps_transaction_update_packages(struct xbps_handle *xhp)
bool newpkg_found = false;
int rv = 0;

rv = xbps_autoupdate(xhp);
switch (rv) {
case 1:
/* xbps needs to be updated, don't allow any other update */
return EBUSY;
case -1:
/* error */
if (xbps_pkgdb_init(xhp) != 0)
return EINVAL;
default:
break;
}

iter = xbps_dictionary_iterator(xhp->pkgdb);
assert(iter);
Expand Down Expand Up @@ -346,6 +343,13 @@ xbps_transaction_update_packages(struct xbps_handle *xhp)
}
xbps_object_iterator_release(iter);

/* Notify consumer if xbps is out of date */
{
const char *pkgver = NULL;
if (trans_check_uptodate(xhp, "xbps", &pkgver) == 1)
xbps_set_cb_state(xhp, XBPS_STATE_OUTOFDATE, 0, pkgver, NULL);
}

return newpkg_found ? rv : EEXIST;
}

Expand All @@ -355,22 +359,6 @@ xbps_transaction_update_pkg(struct xbps_handle *xhp, const char *pkg, bool force
xbps_array_t rdeps;
int rv;

rv = xbps_autoupdate(xhp);
xbps_dbg_printf("%s: xbps_autoupdate %d\n", __func__, rv);
switch (rv) {
case 1:
/* xbps needs to be updated, only allow xbps to be updated */
if (strcmp(pkg, "xbps"))
return EBUSY;
return 0;
case -1:
/* error */
return EINVAL;
default:
/* no update */
break;
}

/* update its reverse dependencies */
rdeps = xbps_pkgdb_get_pkg_revdeps(xhp, pkg);
if (xhp->flags & XBPS_FLAG_DOWNLOAD_ONLY) {
Expand All @@ -397,6 +385,14 @@ xbps_transaction_update_pkg(struct xbps_handle *xhp, const char *pkg, bool force
/* add pkg repod */
rv = trans_find_pkg(xhp, pkg, force);
xbps_dbg_printf("%s: trans_find_pkg %s: %d\n", __func__, pkg, rv);

/* Notify consumer if xbps is out of date */
{
const char *pkgver = NULL;
if (trans_check_uptodate(xhp, "xbps", &pkgver) == 1)
xbps_set_cb_state(xhp, XBPS_STATE_OUTOFDATE, 0, pkgver, NULL);
}

return rv;
}

Expand All @@ -406,21 +402,6 @@ xbps_transaction_install_pkg(struct xbps_handle *xhp, const char *pkg, bool forc
xbps_array_t rdeps;
int rv;

rv = xbps_autoupdate(xhp);
switch (rv) {
case 1:
/* xbps needs to be updated, only allow xbps to be updated */
if (strcmp(pkg, "xbps"))
return EBUSY;
return 0;
case -1:
/* error */
return EINVAL;
default:
/* no update */
break;
}

/* update its reverse dependencies */
rdeps = xbps_pkgdb_get_pkg_revdeps(xhp, pkg);
if (xhp->flags & XBPS_FLAG_DOWNLOAD_ONLY) {
Expand All @@ -446,6 +427,14 @@ xbps_transaction_install_pkg(struct xbps_handle *xhp, const char *pkg, bool forc
}
rv = trans_find_pkg(xhp, pkg, force);
xbps_dbg_printf("%s: trans_find_pkg %s: %d\n", __func__, pkg, rv);

/* Notify consumer if xbps is out of date */
{
const char *pkgver = NULL;
if (trans_check_uptodate(xhp, "xbps", &pkgver) == 1)
xbps_set_cb_state(xhp, XBPS_STATE_OUTOFDATE, 0, pkgver, NULL);
}

return rv;
}

Expand Down
10 changes: 2 additions & 8 deletions tests/xbps/libxbps/shell/install_test.sh
Original file line number Diff line number Diff line change
Expand Up @@ -484,10 +484,7 @@ update_xbps_body() {
cd ..
xbps-rindex -d -a repo/*.xbps
atf_check_equal $? 0
out=$(xbps-install -r root --repository=repo -yun)
set -- $out
exp="$1 $2 $3 $4"
atf_check_equal "$exp" "xbps-1.1_1 update noarch $(readlink -f repo)"
atf_check -s exit:0 -o ignore -e match:"^WARNING: The 'xbps-1\.0_1' package is out of date, 'xbps-1\.1_1' is available\.$" -- xbps-install -r root --repository=repo -yn B

xbps-install -r root --repository=repo -yu xbps
atf_check_equal $? 0
Expand Down Expand Up @@ -537,10 +534,7 @@ update_xbps_virtual_body() {
cd ..
xbps-rindex -d -a repo/*.xbps
atf_check_equal $? 0
out=$(xbps-install -r root --repository=repo -yun)
set -- $out
exp="$1 $2 $3 $4"
atf_check_equal "$exp" "xbps-git-1.1_1 update noarch $(readlink -f repo)"
atf_check -s exit:0 -o ignore -e match:"^WARNING: The 'xbps-git-1\.0_1' package is out of date, 'xbps-git-1\.1_1' is available\.$" -- xbps-install -r root --repository=repo -yn B

xbps-install -r root --repository=repo -yu xbps
atf_check_equal $? 0
Expand Down
Loading