Fix and improve how missing pkg dependencies are handled from repos.

- Use an array of strings to store the pkgdeps.
- While adding a missing pkgdep, check if it's already in the queue and
  new required version is greater, in that case replace with new one.

--HG--
extra : convert_revision : xtraeme%40gmail.com-20091129021735-5dqfucofny8slks2
This commit is contained in:
Juan RP 2009-11-29 03:17:35 +01:00
parent eb0567bfab
commit 1a5d19dca2
3 changed files with 138 additions and 47 deletions

View File

@ -56,20 +56,17 @@ show_missing_deps(prop_dictionary_t d, const char *pkgname)
static int static int
show_missing_dep_cb(prop_object_t obj, void *arg, bool *loop_done) show_missing_dep_cb(prop_object_t obj, void *arg, bool *loop_done)
{ {
const char *pkgname, *version; const char *reqpkg;
(void)arg; (void)arg;
(void)loop_done; (void)loop_done;
prop_dictionary_get_cstring_nocopy(obj, "pkgname", &pkgname); reqpkg = prop_string_cstring_nocopy(obj);
prop_dictionary_get_cstring_nocopy(obj, "version", &version); if (reqpkg == NULL)
if (pkgname && version) { return EINVAL;
printf(" * Missing binary package for: %s%s\n",
pkgname, version);
return 0;
}
return EINVAL; printf(" * Missing binary package for: %s\n", reqpkg);
return 0;
} }
static bool static bool

View File

@ -30,7 +30,8 @@
#include <xbps_api.h> #include <xbps_api.h>
static int add_missing_reqdep(prop_dictionary_t, const char *, const char *); static int add_missing_reqdep(prop_dictionary_t, const char *);
static int remove_missing_reqdep(prop_dictionary_t, const char *);
static int find_repo_deps(prop_dictionary_t, prop_dictionary_t, static int find_repo_deps(prop_dictionary_t, prop_dictionary_t,
const char *, prop_array_t); const char *, prop_array_t);
@ -102,43 +103,139 @@ store_dependency(prop_dictionary_t master, prop_dictionary_t depd,
} }
static int static int
add_missing_reqdep(prop_dictionary_t master, const char *pkgname, add_missing_reqdep(prop_dictionary_t master, const char *reqpkg)
const char *version)
{ {
prop_array_t missing_rdeps; prop_array_t missing_rdeps;
prop_dictionary_t mdepd, tmpd; prop_string_t reqpkg_str;
prop_object_iterator_t iter = NULL;
prop_object_t obj;
size_t idx = 0;
bool add_pkgdep, pkgfound, update_pkgdep;
assert(array != NULL); assert(master != NULL);
assert(reqdep != NULL); assert(reqpkg != NULL);
tmpd = xbps_find_pkg_in_dict(master, "missing_deps", pkgname); add_pkgdep = update_pkgdep = pkgfound = false;
if (tmpd == NULL) {
if (errno && errno != ENOENT)
return errno;
} else if (tmpd)
return EEXIST;
mdepd = prop_dictionary_create(); reqpkg_str = prop_string_create_cstring_nocopy(reqpkg);
if (mdepd == NULL) if (reqpkg_str == NULL)
return errno; return errno;
missing_rdeps = prop_dictionary_get(master, "missing_deps"); missing_rdeps = prop_dictionary_get(master, "missing_deps");
if (!prop_dictionary_set_cstring(mdepd, "pkgname", pkgname)) { iter = prop_array_iterator(missing_rdeps);
prop_object_release(mdepd); if (iter == NULL)
return errno; goto out;
while ((obj = prop_object_iterator_next(iter)) != NULL) {
const char *curdep, *curver, *pkgver;
char *curpkgnamedep = NULL, *pkgnamedep = NULL;
assert(prop_object_type(obj) == PROP_TYPE_STRING);
curdep = prop_string_cstring_nocopy(obj);
curver = xbps_get_pkgdep_version(curdep);
pkgver = xbps_get_pkgdep_version(reqpkg);
if (curver == NULL || pkgver == NULL)
goto out;
curpkgnamedep = xbps_get_pkgdep_name(curdep);
if (curpkgnamedep == NULL)
goto out;
pkgnamedep = xbps_get_pkgdep_name(reqpkg);
if (pkgnamedep == NULL) {
free(curpkgnamedep);
goto out;
}
if (strcmp(pkgnamedep, curpkgnamedep) == 0) {
pkgfound = true;
/*
* if new dependency version is greater than current
* one, store it.
*/
DPRINTF(("Missing pkgdep name matched, curver: %s "
"newver: %s\n", curver, pkgver));
if (xbps_cmpver(curver, pkgver) <= 0) {
add_pkgdep = false;
free(curpkgnamedep);
free(pkgnamedep);
goto out;
}
update_pkgdep = true;
}
free(curpkgnamedep);
free(pkgnamedep);
if (pkgfound)
break;
idx++;
} }
if (!prop_dictionary_set_cstring(mdepd, "version", version)) { add_pkgdep = true;
prop_object_release(mdepd); out:
if (iter)
prop_object_iterator_release(iter);
if (update_pkgdep)
prop_array_remove(missing_rdeps, idx);
if (add_pkgdep && !xbps_add_obj_to_array(missing_rdeps, reqpkg_str)) {
prop_object_release(reqpkg_str);
return errno; return errno;
} }
if (!xbps_add_obj_to_array(missing_rdeps, mdepd)) {
prop_object_release(mdepd);
return EINVAL;
}
return 0; return 0;
} }
static int
remove_missing_reqdep(prop_dictionary_t master, const char *reqpkg)
{
prop_array_t missing_rdeps;
prop_object_iterator_t iter = NULL;
prop_object_t obj;
size_t idx = 0;
int rv = 0;
bool found = false;
assert(master != NULL);
assert(reqpkg != NULL);
missing_rdeps = prop_dictionary_get(master, "missing_deps");
iter = prop_array_iterator(missing_rdeps);
if (iter == NULL)
return errno;
while ((obj = prop_object_iterator_next(iter)) != NULL) {
const char *curdep;
char *curpkgnamedep, *reqpkgname;
curdep = prop_string_cstring_nocopy(obj);
curpkgnamedep = xbps_get_pkgdep_name(curdep);
if (curpkgnamedep == NULL) {
rv = errno;
goto out;
}
reqpkgname = xbps_get_pkgdep_name(reqpkg);
if (reqpkgname == NULL) {
free(curpkgnamedep);
rv = errno;
goto out;
}
if (strcmp(reqpkgname, curpkgnamedep) == 0)
found = true;
free(curpkgnamedep);
free(reqpkgname);
if (found)
break;
idx++;
}
out:
prop_object_iterator_release(iter);
if (found) {
prop_array_remove(missing_rdeps, idx);
return 0;
}
if (rv == 0)
rv = ENOENT;
return rv;
}
int SYMEXPORT int SYMEXPORT
xbps_find_deps_in_pkg(prop_dictionary_t master, prop_dictionary_t pkg) xbps_find_deps_in_pkg(prop_dictionary_t master, prop_dictionary_t pkg)
{ {
@ -147,8 +244,8 @@ xbps_find_deps_in_pkg(prop_dictionary_t master, prop_dictionary_t pkg)
const char *pkgname; const char *pkgname;
int rv = 0; int rv = 0;
assert(master != NULL);
assert(pkg != NULL); assert(pkg != NULL);
assert(iter != NULL);
pkg_rdeps = prop_dictionary_get(pkg, "run_depends"); pkg_rdeps = prop_dictionary_get(pkg, "run_depends");
if (pkg_rdeps == NULL) if (pkg_rdeps == NULL)
@ -174,7 +271,7 @@ xbps_find_deps_in_pkg(prop_dictionary_t master, prop_dictionary_t pkg)
if ((rv = find_repo_deps(master, rdata->rd_repod, if ((rv = find_repo_deps(master, rdata->rd_repod,
rdata->rd_uri, pkg_rdeps)) != 0) { rdata->rd_uri, pkg_rdeps)) != 0) {
DPRINTF(("Error '%s' while checking rundeps!\n", DPRINTF(("Error '%s' while checking rundeps!\n",
strerror(errno))); strerror(rv)));
goto out; goto out;
} }
} }
@ -194,8 +291,8 @@ xbps_find_deps_in_pkg(prop_dictionary_t master, prop_dictionary_t pkg)
SIMPLEQ_FOREACH(rdata, &repodata_queue, chain) { SIMPLEQ_FOREACH(rdata, &repodata_queue, chain) {
if ((rv = find_repo_deps(master, rdata->rd_repod, if ((rv = find_repo_deps(master, rdata->rd_repod,
rdata->rd_uri, missing_rdeps)) != 0) { rdata->rd_uri, missing_rdeps)) != 0) {
DPRINTF(("Error '%s' while checking for" DPRINTF(("Error '%s' while checking for "
"missing rundeps!\n", strerror(errno))); "missing rundeps!\n", strerror(rv)));
goto out; goto out;
} }
} }
@ -299,7 +396,7 @@ find_repo_deps(prop_dictionary_t master, prop_dictionary_t repo,
break; break;
} }
rv = add_missing_reqdep(master, pkgname, reqvers); rv = add_missing_reqdep(master, reqpkg);
if (rv != 0 && rv != EEXIST) { if (rv != 0 && rv != EEXIST) {
DPRINTF(("add missing reqdep failed %s\n", DPRINTF(("add missing reqdep failed %s\n",
reqpkg)); reqpkg));
@ -313,7 +410,7 @@ find_repo_deps(prop_dictionary_t master, prop_dictionary_t repo,
continue; continue;
} else { } else {
DPRINTF(("Added missing dep %s (repo: %s).\n", DPRINTF(("Added missing dep %s (repo: %s).\n",
pkgname, repoloc)); reqpkg, repoloc));
free(pkgname); free(pkgname);
continue; continue;
} }
@ -336,6 +433,7 @@ find_repo_deps(prop_dictionary_t master, prop_dictionary_t repo,
} else if (tmpd) { } else if (tmpd) {
rv = xbps_get_pkg_state_installed(pkgname, &state); rv = xbps_get_pkg_state_installed(pkgname, &state);
if (rv != 0) { if (rv != 0) {
free(pkgname);
prop_object_release(tmpd); prop_object_release(tmpd);
break; break;
} }
@ -348,32 +446,30 @@ find_repo_deps(prop_dictionary_t master, prop_dictionary_t repo,
prop_object_release(tmpd); prop_object_release(tmpd);
} }
free(pkgname);
/* /*
* Package is on repo, add it into the dictionary. * Package is on repo, add it into the dictionary.
*/ */
if ((rv = store_dependency(master, curpkgd, repoloc)) != 0) { if ((rv = store_dependency(master, curpkgd, repoloc)) != 0) {
DPRINTF(("store_dependency failed %s\n", reqpkg)); DPRINTF(("store_dependency failed %s\n", reqpkg));
free(pkgname);
break; break;
} }
DPRINTF(("Added reqdep %s (repo: %s)\n", pkgname, repoloc)); DPRINTF(("Added reqdep %s (repo: %s)\n", reqpkg, repoloc));
/* /*
* If package was added in the missing_deps array, we * If package was added in the missing_deps array, we
* can remove it now it has been found in current repository. * can remove it now it has been found in current repository.
*/ */
rv = xbps_remove_pkg_from_dict(master, "missing_deps", pkgname); rv = remove_missing_reqdep(master, reqpkg);
if (rv == ENOENT) { if (rv == ENOENT) {
rv = 0; rv = 0;
} else if (rv == 0) { } else if (rv == 0) {
DPRINTF(("Removed missing dep %s.\n", pkgname)); DPRINTF(("Removed missing dep %s.\n", reqpkg));
} else { } else {
DPRINTF(("Removing missing dep %s returned %s\n", DPRINTF(("Removing missing dep %s returned %s\n",
pkgname, strerror(rv))); reqpkg, strerror(rv)));
free(pkgname);
break; break;
} }
free(pkgname);
/* /*
* If package doesn't have rundeps, pass to the next one. * If package doesn't have rundeps, pass to the next one.

View File

@ -265,8 +265,6 @@ xbps_remove_string_from_array(prop_array_t array, const char *str)
return errno; return errno;
while ((obj = prop_object_iterator_next(iter)) != NULL) { while ((obj = prop_object_iterator_next(iter)) != NULL) {
if (prop_object_type(obj) != PROP_TYPE_STRING)
continue;
if (prop_string_equals_cstring(obj, str)) { if (prop_string_equals_cstring(obj, str)) {
found = true; found = true;
break; break;