mirror of
https://github.com/moparisthebest/pacman
synced 2024-11-13 21:05:05 -05:00
344ced22c3
This happens for example if you install a new package, and one of its backup config file is already on the file system. If the local file was different, it was saved to .pacorig which is fine. However if the local file and pkg file were the same, the pkg file (temporarily extracted as .paccheck) was left on the system. Signed-off-by: Xavier Chantry <shiningxc@gmail.com> Signed-off-by: Dan McGee <dan@archlinux.org>
896 lines
28 KiB
C
896 lines
28 KiB
C
/*
|
|
* add.c
|
|
*
|
|
* Copyright (c) 2006-2009 Pacman Development Team <pacman-dev@archlinux.org>
|
|
* Copyright (c) 2002-2006 by Judd Vinet <jvinet@zeroflux.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include <time.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include <inttypes.h> /* int64_t */
|
|
#include <stdint.h> /* intmax_t */
|
|
|
|
/* libarchive */
|
|
#include <archive.h>
|
|
#include <archive_entry.h>
|
|
|
|
/* libalpm */
|
|
#include "add.h"
|
|
#include "alpm_list.h"
|
|
#include "trans.h"
|
|
#include "util.h"
|
|
#include "cache.h"
|
|
#include "log.h"
|
|
#include "backup.h"
|
|
#include "package.h"
|
|
#include "db.h"
|
|
#include "conflict.h"
|
|
#include "deps.h"
|
|
#include "remove.h"
|
|
#include "handle.h"
|
|
|
|
int _alpm_add_loadtarget(pmtrans_t *trans, pmdb_t *db, char *name)
|
|
{
|
|
pmpkg_t *pkg = NULL;
|
|
const char *pkgname, *pkgver;
|
|
alpm_list_t *i;
|
|
|
|
ALPM_LOG_FUNC;
|
|
|
|
ASSERT(trans != NULL, RET_ERR(PM_ERR_TRANS_NULL, -1));
|
|
ASSERT(db != NULL, RET_ERR(PM_ERR_DB_NULL, -1));
|
|
ASSERT(name != NULL && strlen(name) != 0, RET_ERR(PM_ERR_WRONG_ARGS, -1));
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "loading target '%s'\n", name);
|
|
|
|
if(alpm_pkg_load(name, 1, &pkg) != 0) {
|
|
goto error;
|
|
}
|
|
pkgname = alpm_pkg_get_name(pkg);
|
|
pkgver = alpm_pkg_get_version(pkg);
|
|
|
|
/* check if an older version of said package is already in transaction
|
|
* packages. if so, replace it in the list */
|
|
for(i = trans->packages; i; i = i->next) {
|
|
pmpkg_t *transpkg = i->data;
|
|
if(strcmp(transpkg->name, pkgname) == 0) {
|
|
if(alpm_pkg_vercmp(transpkg->version, pkgver) < 0) {
|
|
_alpm_log(PM_LOG_WARNING, _("replacing older version %s-%s by %s in target list\n"),
|
|
transpkg->name, transpkg->version, pkgver);
|
|
_alpm_pkg_free(i->data);
|
|
i->data = pkg;
|
|
} else {
|
|
_alpm_log(PM_LOG_WARNING, _("skipping %s-%s because newer version %s is in the target list\n"),
|
|
pkgname, pkgver, transpkg->version);
|
|
_alpm_pkg_free(pkg);
|
|
}
|
|
return(0);
|
|
}
|
|
}
|
|
|
|
/* add the package to the transaction */
|
|
trans->packages = alpm_list_add(trans->packages, pkg);
|
|
|
|
return(0);
|
|
|
|
error:
|
|
_alpm_pkg_free(pkg);
|
|
return(-1);
|
|
}
|
|
|
|
int _alpm_add_prepare(pmtrans_t *trans, pmdb_t *db, alpm_list_t **data)
|
|
{
|
|
alpm_list_t *lp = NULL;
|
|
|
|
ALPM_LOG_FUNC;
|
|
|
|
ASSERT(trans != NULL, RET_ERR(PM_ERR_TRANS_NULL, -1));
|
|
ASSERT(db != NULL, RET_ERR(PM_ERR_DB_NULL, -1));
|
|
|
|
/* Check dependencies
|
|
*/
|
|
if(!(trans->flags & PM_TRANS_FLAG_NODEPS)) {
|
|
EVENT(trans, PM_TRANS_EVT_CHECKDEPS_START, NULL, NULL);
|
|
|
|
/* look for unsatisfied dependencies */
|
|
_alpm_log(PM_LOG_DEBUG, "looking for unsatisfied dependencies\n");
|
|
lp = alpm_checkdeps(_alpm_db_get_pkgcache(db), 1, NULL, trans->packages);
|
|
if(lp != NULL) {
|
|
if(data) {
|
|
*data = lp;
|
|
} else {
|
|
alpm_list_free_inner(lp, (alpm_list_fn_free)_alpm_depmiss_free);
|
|
alpm_list_free(lp);
|
|
}
|
|
RET_ERR(PM_ERR_UNSATISFIED_DEPS, -1);
|
|
}
|
|
|
|
/* no unsatisfied deps, so look for conflicts */
|
|
_alpm_log(PM_LOG_DEBUG, "looking for conflicts\n");
|
|
alpm_list_t *inner = _alpm_innerconflicts(trans->packages);
|
|
alpm_list_t *outer = _alpm_outerconflicts(db, trans->packages);
|
|
lp = alpm_list_join(inner, outer);
|
|
|
|
/* TODO : factorize the conflict resolving code from sync.c to use it here (FS#3492) */
|
|
|
|
if(lp != NULL) {
|
|
if(data) {
|
|
*data = lp;
|
|
} else {
|
|
alpm_list_free_inner(lp, (alpm_list_fn_free)_alpm_conflict_free);
|
|
alpm_list_free(lp);
|
|
}
|
|
if(inner) {
|
|
_alpm_log(PM_LOG_ERROR, _("conflicting packages were found in the target list\n"));
|
|
_alpm_log(PM_LOG_ERROR, _("you cannot install two conflicting packages at the same time\n"));
|
|
}
|
|
if(outer) {
|
|
_alpm_log(PM_LOG_ERROR, _("replacing packages with -U is not supported yet\n"));
|
|
_alpm_log(PM_LOG_ERROR, _("you can replace packages manually using -Rd and -U\n"));
|
|
}
|
|
RET_ERR(PM_ERR_CONFLICTING_DEPS, -1);
|
|
}
|
|
|
|
/* re-order w.r.t. dependencies */
|
|
_alpm_log(PM_LOG_DEBUG, "sorting by dependencies\n");
|
|
lp = _alpm_sortbydeps(trans->packages, 0);
|
|
/* free the old alltargs */
|
|
alpm_list_free(trans->packages);
|
|
trans->packages = lp;
|
|
|
|
EVENT(trans, PM_TRANS_EVT_CHECKDEPS_DONE, NULL, NULL);
|
|
}
|
|
|
|
/* Check for file conflicts */
|
|
if(!(trans->flags & PM_TRANS_FLAG_FORCE)) {
|
|
EVENT(trans, PM_TRANS_EVT_FILECONFLICTS_START, NULL, NULL);
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "looking for file conflicts\n");
|
|
lp = _alpm_db_find_fileconflicts(db, trans, trans->packages, NULL);
|
|
if(lp != NULL) {
|
|
if(data) {
|
|
*data = lp;
|
|
} else {
|
|
alpm_list_free_inner(lp, (alpm_list_fn_free)_alpm_fileconflict_free);
|
|
alpm_list_free(lp);
|
|
}
|
|
RET_ERR(PM_ERR_FILE_CONFLICTS, -1);
|
|
}
|
|
|
|
EVENT(trans, PM_TRANS_EVT_FILECONFLICTS_DONE, NULL, NULL);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int upgrade_remove(pmpkg_t *oldpkg, pmpkg_t *newpkg, pmtrans_t *trans, pmdb_t *db) {
|
|
/* this is kinda odd. If the old package exists, at this point we make a
|
|
* NEW transaction, unrelated to handle->trans, and instantiate a "remove"
|
|
* with the type PM_TRANS_TYPE_REMOVEUPGRADE. TODO: kill this weird
|
|
* behavior. */
|
|
pmtrans_t *tr = _alpm_trans_new();
|
|
|
|
ALPM_LOG_FUNC;
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "removing old package first (%s-%s)\n",
|
|
oldpkg->name, oldpkg->version);
|
|
|
|
if(!tr) {
|
|
RET_ERR(PM_ERR_TRANS_ABORT, -1);
|
|
}
|
|
|
|
if(_alpm_trans_init(tr, PM_TRANS_TYPE_REMOVEUPGRADE, trans->flags,
|
|
NULL, NULL, NULL) == -1) {
|
|
_alpm_trans_free(tr);
|
|
tr = NULL;
|
|
RET_ERR(PM_ERR_TRANS_ABORT, -1);
|
|
}
|
|
|
|
if(_alpm_remove_loadtarget(tr, db, newpkg->name) == -1) {
|
|
_alpm_trans_free(tr);
|
|
tr = NULL;
|
|
RET_ERR(PM_ERR_TRANS_ABORT, -1);
|
|
}
|
|
|
|
/* copy the remove skiplist over */
|
|
tr->skip_remove = alpm_list_strdup(trans->skip_remove);
|
|
const alpm_list_t *b;
|
|
|
|
/* Add files in the NEW backup array to the NoUpgrade array
|
|
* so this removal operation doesn't kill them */
|
|
alpm_list_t *old_noupgrade = alpm_list_strdup(handle->noupgrade);
|
|
/* old package backup list */
|
|
alpm_list_t *filelist = alpm_pkg_get_files(newpkg);
|
|
for(b = alpm_pkg_get_backup(newpkg); b; b = b->next) {
|
|
char *backup = _alpm_backup_file(b->data);
|
|
/* safety check (fix the upgrade026 pactest) */
|
|
if(!alpm_list_find_str(filelist, backup)) {
|
|
FREE(backup);
|
|
continue;
|
|
}
|
|
_alpm_log(PM_LOG_DEBUG, "adding %s to the NoUpgrade array temporarily\n",
|
|
backup);
|
|
handle->noupgrade = alpm_list_add(handle->noupgrade,
|
|
backup);
|
|
}
|
|
|
|
/* TODO: we could also add files in the OLD backup array, but this would
|
|
* change the backup handling behavior, and break several pactests, and we
|
|
* can't do this just before 3.1 release.
|
|
* The unlink_file function in remove.c would also need to be reviewed. */
|
|
#if 0
|
|
/* new package backup list */
|
|
for(b = alpm_pkg_get_backup(oldpkg); b; b = b->next) {
|
|
char *backup = _alpm_backup_file(b->data);
|
|
/* make sure we don't add duplicate entries */
|
|
if(!alpm_list_find_ptr(handle->noupgrade, backup)) {
|
|
_alpm_log(PM_LOG_DEBUG, "adding %s to the NoUpgrade array temporarily\n",
|
|
backup);
|
|
handle->noupgrade = alpm_list_add(handle->noupgrade,
|
|
backup);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
int ret = _alpm_remove_commit(tr, db);
|
|
|
|
_alpm_trans_free(tr);
|
|
tr = NULL;
|
|
|
|
/* restore our "NoUpgrade" list to previous state */
|
|
FREELIST(handle->noupgrade);
|
|
handle->noupgrade = old_noupgrade;
|
|
|
|
if(ret == -1) {
|
|
RET_ERR(PM_ERR_TRANS_ABORT, -1);
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int extract_single_file(struct archive *archive,
|
|
struct archive_entry *entry, pmpkg_t *newpkg, pmpkg_t *oldpkg,
|
|
pmtrans_t *trans, pmdb_t *db)
|
|
{
|
|
const char *entryname;
|
|
mode_t entrymode;
|
|
char filename[PATH_MAX]; /* the actual file we're extracting */
|
|
int needbackup = 0, notouch = 0;
|
|
char *hash_orig = NULL;
|
|
char *entryname_orig = NULL;
|
|
const int archive_flags = ARCHIVE_EXTRACT_OWNER |
|
|
ARCHIVE_EXTRACT_PERM |
|
|
ARCHIVE_EXTRACT_TIME;
|
|
int errors = 0;
|
|
|
|
entryname = archive_entry_pathname(entry);
|
|
entrymode = archive_entry_mode(entry);
|
|
|
|
memset(filename, 0, PATH_MAX); /* just to be sure */
|
|
|
|
if(strcmp(entryname, ".INSTALL") == 0) {
|
|
/* the install script goes inside the db */
|
|
snprintf(filename, PATH_MAX, "%s%s-%s/install", db->path,
|
|
newpkg->name, newpkg->version);
|
|
archive_entry_set_perm(entry, 0644);
|
|
} else if(strcmp(entryname, ".CHANGELOG") == 0) {
|
|
/* the changelog goes inside the db */
|
|
snprintf(filename, PATH_MAX, "%s%s-%s/changelog", db->path,
|
|
newpkg->name, newpkg->version);
|
|
archive_entry_set_perm(entry, 0644);
|
|
} else if(*entryname == '.') {
|
|
/* for now, ignore all files starting with '.' that haven't
|
|
* already been handled (for future possibilities) */
|
|
_alpm_log(PM_LOG_DEBUG, "skipping extraction of '%s'\n", entryname);
|
|
archive_read_data_skip(archive);
|
|
return(0);
|
|
} else {
|
|
/* build the new entryname relative to handle->root */
|
|
snprintf(filename, PATH_MAX, "%s%s", handle->root, entryname);
|
|
}
|
|
|
|
/* if a file is in NoExtract then we never extract it */
|
|
if(alpm_list_find_str(handle->noextract, entryname)) {
|
|
_alpm_log(PM_LOG_DEBUG, "%s is in NoExtract, skipping extraction\n",
|
|
entryname);
|
|
alpm_logaction("note: %s is in NoExtract, skipping extraction\n",
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(0);
|
|
}
|
|
|
|
/* if a file is in the add skiplist we never extract it */
|
|
if(alpm_list_find_str(trans->skip_add, filename)) {
|
|
_alpm_log(PM_LOG_DEBUG, "%s is in trans->skip_add, skipping extraction\n",
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(0);
|
|
}
|
|
|
|
/* Check for file existence. This is one of the more crucial parts
|
|
* to get 'right'. Here are the possibilities, with the filesystem
|
|
* on the left and the package on the top:
|
|
* (F=file, N=node, S=symlink, D=dir)
|
|
* | F/N | S | D
|
|
* non-existent | 1 | 2 | 3
|
|
* F/N | 4 | 5 | 6
|
|
* S | 7 | 8 | 9
|
|
* D | 10 | 11 | 12
|
|
*
|
|
* 1,2,3- extract, no magic necessary. lstat (_alpm_lstat) will fail here.
|
|
* 4,5,6,7,8- conflict checks should have caught this. either overwrite
|
|
* or backup the file.
|
|
* 9- follow the symlink, hopefully it is a directory, check it.
|
|
* 10- file replacing directory- don't allow it.
|
|
* 11- don't extract symlink- a dir exists here. we don't want links to
|
|
* links, etc.
|
|
* 12- skip extraction, dir already exists.
|
|
*/
|
|
|
|
/* do both a lstat and a stat, so we can see what symlinks point to */
|
|
struct stat lsbuf, sbuf;
|
|
if(_alpm_lstat(filename, &lsbuf) != 0 || stat(filename, &sbuf) != 0) {
|
|
/* cases 1,2,3: couldn't stat an existing file, skip all backup checks */
|
|
} else {
|
|
if(S_ISDIR(lsbuf.st_mode)) {
|
|
if(S_ISDIR(entrymode)) {
|
|
/* case 12: existing dir, ignore it */
|
|
if(lsbuf.st_mode != entrymode) {
|
|
/* if filesystem perms are different than pkg perms, warn user */
|
|
int mask = 07777;
|
|
_alpm_log(PM_LOG_WARNING, _("directory permissions differ on %s\n"
|
|
"filesystem: %o package: %o\n"), entryname, lsbuf.st_mode & mask,
|
|
entrymode & mask);
|
|
alpm_logaction("warning: directory permissions differ on %s\n"
|
|
"filesystem: %o package: %o\n", entryname, lsbuf.st_mode & mask,
|
|
entrymode & mask);
|
|
}
|
|
_alpm_log(PM_LOG_DEBUG, "extract: skipping dir extraction of %s\n",
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(0);
|
|
} else {
|
|
/* case 10/11: trying to overwrite dir with file/symlink, don't allow it */
|
|
_alpm_log(PM_LOG_ERROR, _("extract: not overwriting dir with file %s\n"),
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(1);
|
|
}
|
|
} else if(S_ISLNK(lsbuf.st_mode) && S_ISDIR(entrymode)) {
|
|
/* case 9: existing symlink, dir in package */
|
|
if(S_ISDIR(sbuf.st_mode)) {
|
|
/* the symlink on FS is to a directory, so we'll use it */
|
|
_alpm_log(PM_LOG_DEBUG, "extract: skipping symlink overwrite of %s\n",
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(0);
|
|
} else {
|
|
/* this is BAD. symlink was not to a directory */
|
|
_alpm_log(PM_LOG_ERROR, _("extract: symlink %s does not point to dir\n"),
|
|
entryname);
|
|
archive_read_data_skip(archive);
|
|
return(1);
|
|
}
|
|
} else if(S_ISREG(lsbuf.st_mode) && S_ISDIR(entrymode)) {
|
|
/* case 6: trying to overwrite file with dir */
|
|
_alpm_log(PM_LOG_DEBUG, "extract: overwriting file with dir %s\n",
|
|
entryname);
|
|
} else if(S_ISREG(entrymode)) {
|
|
/* case 4,7: */
|
|
/* if file is in NoUpgrade, don't touch it */
|
|
if(alpm_list_find_str(handle->noupgrade, entryname)) {
|
|
notouch = 1;
|
|
} else {
|
|
/* go to the backup array and see if our conflict is there */
|
|
/* check newpkg first, so that adding backup files is retroactive */
|
|
if(alpm_list_find_str(alpm_pkg_get_backup(newpkg), entryname) != NULL) {
|
|
needbackup = 1;
|
|
}
|
|
|
|
/* check oldpkg for a backup entry, store the hash if available */
|
|
if(oldpkg) {
|
|
hash_orig = _alpm_needbackup(entryname, alpm_pkg_get_backup(oldpkg));
|
|
if(hash_orig) {
|
|
needbackup = 1;
|
|
}
|
|
}
|
|
|
|
/* if we force hash_orig to be non-NULL retroactive backup works */
|
|
if(needbackup && !hash_orig) {
|
|
STRDUP(hash_orig, "", RET_ERR(PM_ERR_MEMORY, -1));
|
|
}
|
|
}
|
|
}
|
|
/* else if(S_ISLNK(entrymode)) */
|
|
/* case 5,8: don't need to do anything special */
|
|
}
|
|
|
|
/* we need access to the original entryname later after calls to
|
|
* archive_entry_set_pathname(), so we need to dupe it and free() later */
|
|
STRDUP(entryname_orig, entryname, RET_ERR(PM_ERR_MEMORY, -1));
|
|
|
|
if(needbackup) {
|
|
char checkfile[PATH_MAX];
|
|
char *hash_local = NULL, *hash_pkg = NULL;
|
|
int ret;
|
|
|
|
snprintf(checkfile, PATH_MAX, "%s.paccheck", filename);
|
|
archive_entry_set_pathname(entry, checkfile);
|
|
|
|
ret = archive_read_extract(archive, entry, archive_flags);
|
|
if(ret == ARCHIVE_WARN) {
|
|
/* operation succeeded but a non-critical error was encountered */
|
|
_alpm_log(PM_LOG_DEBUG, "warning extracting %s (%s)\n",
|
|
entryname_orig, archive_error_string(archive));
|
|
} else if(ret != ARCHIVE_OK) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not extract %s (%s)\n"),
|
|
entryname_orig, archive_error_string(archive));
|
|
alpm_logaction("error: could not extract %s (%s)\n",
|
|
entryname_orig, archive_error_string(archive));
|
|
FREE(hash_orig);
|
|
FREE(entryname_orig);
|
|
return(1);
|
|
}
|
|
|
|
hash_local = alpm_compute_md5sum(filename);
|
|
hash_pkg = alpm_compute_md5sum(checkfile);
|
|
|
|
/* append the new md5 hash to it's respective entry
|
|
* in newpkg's backup (it will be the new orginal) */
|
|
alpm_list_t *backups;
|
|
for(backups = alpm_pkg_get_backup(newpkg); backups;
|
|
backups = alpm_list_next(backups)) {
|
|
char *oldbackup = alpm_list_getdata(backups);
|
|
if(!oldbackup || strcmp(oldbackup, entryname_orig) != 0) {
|
|
continue;
|
|
}
|
|
char *backup = NULL;
|
|
/* length is tab char, null byte and MD5 (32 char) */
|
|
size_t backup_len = strlen(oldbackup) + 34;
|
|
MALLOC(backup, backup_len, RET_ERR(PM_ERR_MEMORY, -1));
|
|
|
|
sprintf(backup, "%s\t%s", oldbackup, hash_pkg);
|
|
backup[backup_len-1] = '\0';
|
|
FREE(oldbackup);
|
|
backups->data = backup;
|
|
}
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "checking hashes for %s\n", entryname_orig);
|
|
_alpm_log(PM_LOG_DEBUG, "current: %s\n", hash_local);
|
|
_alpm_log(PM_LOG_DEBUG, "new: %s\n", hash_pkg);
|
|
_alpm_log(PM_LOG_DEBUG, "original: %s\n", hash_orig);
|
|
|
|
if(!oldpkg) {
|
|
if(strcmp(hash_local, hash_pkg) != 0) {
|
|
/* looks like we have a local file that has a different hash as the
|
|
* file in the package, move it to a .pacorig */
|
|
char newpath[PATH_MAX];
|
|
snprintf(newpath, PATH_MAX, "%s.pacorig", filename);
|
|
|
|
/* move the existing file to the "pacorig" */
|
|
if(rename(filename, newpath)) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not rename %s to %s (%s)\n"),
|
|
filename, newpath, strerror(errno));
|
|
alpm_logaction("error: could not rename %s to %s (%s)\n",
|
|
filename, newpath, strerror(errno));
|
|
errors++;
|
|
} else {
|
|
/* rename the file we extracted to the real name */
|
|
if(rename(checkfile, filename)) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not rename %s to %s (%s)\n"),
|
|
checkfile, filename, strerror(errno));
|
|
alpm_logaction("error: could not rename %s to %s (%s)\n",
|
|
checkfile, filename, strerror(errno));
|
|
errors++;
|
|
} else {
|
|
_alpm_log(PM_LOG_WARNING, _("%s saved as %s\n"), filename, newpath);
|
|
alpm_logaction("warning: %s saved as %s\n", filename, newpath);
|
|
}
|
|
}
|
|
} else {
|
|
/* local file is identical to pkg one, so just remove pkg one */
|
|
unlink(checkfile);
|
|
}
|
|
} else if(hash_orig) {
|
|
/* the fun part */
|
|
|
|
if(strcmp(hash_orig, hash_local) == 0) {
|
|
/* installed file has NOT been changed by user */
|
|
if(strcmp(hash_orig, hash_pkg) != 0) {
|
|
_alpm_log(PM_LOG_DEBUG, "action: installing new file: %s\n",
|
|
entryname_orig);
|
|
|
|
if(rename(checkfile, filename)) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not rename %s to %s (%s)\n"),
|
|
checkfile, filename, strerror(errno));
|
|
alpm_logaction("error: could not rename %s to %s (%s)\n",
|
|
checkfile, filename, strerror(errno));
|
|
errors++;
|
|
}
|
|
} else {
|
|
/* there's no sense in installing the same file twice, install
|
|
* ONLY is the original and package hashes differ */
|
|
_alpm_log(PM_LOG_DEBUG, "action: leaving existing file in place\n");
|
|
unlink(checkfile);
|
|
}
|
|
} else if(strcmp(hash_orig, hash_pkg) == 0) {
|
|
/* originally installed file and new file are the same - this
|
|
* implies the case above failed - i.e. the file was changed by a
|
|
* user */
|
|
_alpm_log(PM_LOG_DEBUG, "action: leaving existing file in place\n");
|
|
unlink(checkfile);
|
|
} else if(strcmp(hash_local, hash_pkg) == 0) {
|
|
/* this would be magical. The above two cases failed, but the
|
|
* user changes just so happened to make the new file exactly the
|
|
* same as the one in the package... skip it */
|
|
_alpm_log(PM_LOG_DEBUG, "action: leaving existing file in place\n");
|
|
unlink(checkfile);
|
|
} else {
|
|
char newpath[PATH_MAX];
|
|
_alpm_log(PM_LOG_DEBUG, "action: keeping current file and installing"
|
|
" new one with .pacnew ending\n");
|
|
snprintf(newpath, PATH_MAX, "%s.pacnew", filename);
|
|
if(rename(checkfile, newpath)) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not install %s as %s (%s)\n"),
|
|
filename, newpath, strerror(errno));
|
|
alpm_logaction("error: could not install %s as %s (%s)\n",
|
|
filename, newpath, strerror(errno));
|
|
} else {
|
|
_alpm_log(PM_LOG_WARNING, _("%s installed as %s\n"),
|
|
filename, newpath);
|
|
alpm_logaction("warning: %s installed as %s\n",
|
|
filename, newpath);
|
|
}
|
|
}
|
|
}
|
|
|
|
FREE(hash_local);
|
|
FREE(hash_pkg);
|
|
FREE(hash_orig);
|
|
} else {
|
|
int ret;
|
|
|
|
/* we didn't need a backup */
|
|
if(notouch) {
|
|
/* change the path to a .pacnew extension */
|
|
_alpm_log(PM_LOG_DEBUG, "%s is in NoUpgrade -- skipping\n", filename);
|
|
_alpm_log(PM_LOG_WARNING, _("extracting %s as %s.pacnew\n"), filename, filename);
|
|
alpm_logaction("warning: extracting %s as %s.pacnew\n", filename, filename);
|
|
strncat(filename, ".pacnew", PATH_MAX - strlen(filename));
|
|
} else {
|
|
_alpm_log(PM_LOG_DEBUG, "extracting %s\n", filename);
|
|
}
|
|
|
|
if(trans->flags & PM_TRANS_FLAG_FORCE) {
|
|
/* if FORCE was used, unlink() each file (whether it's there
|
|
* or not) before extracting. This prevents the old "Text file busy"
|
|
* error that crops up if forcing a glibc or pacman upgrade. */
|
|
unlink(filename);
|
|
}
|
|
|
|
archive_entry_set_pathname(entry, filename);
|
|
|
|
ret = archive_read_extract(archive, entry, archive_flags);
|
|
if(ret == ARCHIVE_WARN) {
|
|
/* operation succeeded but a non-critical error was encountered */
|
|
_alpm_log(PM_LOG_DEBUG, "warning extracting %s (%s)\n",
|
|
entryname_orig, archive_error_string(archive));
|
|
} else if(ret != ARCHIVE_OK) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not extract %s (%s)\n"),
|
|
entryname_orig, archive_error_string(archive));
|
|
alpm_logaction("error: could not extract %s (%s)\n",
|
|
entryname_orig, archive_error_string(archive));
|
|
FREE(entryname_orig);
|
|
return(1);
|
|
}
|
|
|
|
/* calculate an hash if this is in newpkg's backup */
|
|
alpm_list_t *b;
|
|
for(b = alpm_pkg_get_backup(newpkg); b; b = b->next) {
|
|
char *backup = NULL, *hash = NULL;
|
|
char *oldbackup = alpm_list_getdata(b);
|
|
/* length is tab char, null byte and MD5 (32 char) */
|
|
size_t backup_len = strlen(oldbackup) + 34;
|
|
|
|
if(!oldbackup || strcmp(oldbackup, entryname_orig) != 0) {
|
|
continue;
|
|
}
|
|
_alpm_log(PM_LOG_DEBUG, "appending backup entry for %s\n", filename);
|
|
|
|
hash = alpm_compute_md5sum(filename);
|
|
MALLOC(backup, backup_len, RET_ERR(PM_ERR_MEMORY, -1));
|
|
|
|
sprintf(backup, "%s\t%s", oldbackup, hash);
|
|
backup[backup_len-1] = '\0';
|
|
FREE(hash);
|
|
FREE(oldbackup);
|
|
b->data = backup;
|
|
}
|
|
}
|
|
FREE(entryname_orig);
|
|
return(errors);
|
|
}
|
|
|
|
static int commit_single_pkg(pmpkg_t *newpkg, int pkg_current, int pkg_count,
|
|
pmtrans_t *trans, pmdb_t *db)
|
|
{
|
|
int i, ret = 0, errors = 0;
|
|
char scriptlet[PATH_MAX+1];
|
|
int is_upgrade = 0;
|
|
pmpkg_t *oldpkg = NULL;
|
|
|
|
ALPM_LOG_FUNC;
|
|
|
|
snprintf(scriptlet, PATH_MAX, "%s%s-%s/install", db->path,
|
|
alpm_pkg_get_name(newpkg), alpm_pkg_get_version(newpkg));
|
|
|
|
/* see if this is an upgrade. if so, remove the old package first */
|
|
pmpkg_t *local = _alpm_db_get_pkgfromcache(db, newpkg->name);
|
|
if(local) {
|
|
is_upgrade = 1;
|
|
|
|
/* we'll need to save some record for backup checks later */
|
|
oldpkg = _alpm_pkg_dup(local);
|
|
/* make sure all infos are loaded because the database entry
|
|
* will be removed soon */
|
|
_alpm_db_read(oldpkg->origin_data.db, oldpkg, INFRQ_ALL);
|
|
|
|
EVENT(trans, PM_TRANS_EVT_UPGRADE_START, newpkg, oldpkg);
|
|
_alpm_log(PM_LOG_DEBUG, "upgrading package %s-%s\n",
|
|
newpkg->name, newpkg->version);
|
|
|
|
/* copy over the install reason */
|
|
newpkg->reason = alpm_pkg_get_reason(oldpkg);
|
|
|
|
/* pre_upgrade scriptlet */
|
|
if(alpm_pkg_has_scriptlet(newpkg) && !(trans->flags & PM_TRANS_FLAG_NOSCRIPTLET)) {
|
|
_alpm_runscriptlet(handle->root, newpkg->origin_data.file,
|
|
"pre_upgrade", newpkg->version, oldpkg->version, trans);
|
|
}
|
|
} else {
|
|
is_upgrade = 0;
|
|
|
|
EVENT(trans, PM_TRANS_EVT_ADD_START, newpkg, NULL);
|
|
_alpm_log(PM_LOG_DEBUG, "adding package %s-%s\n",
|
|
newpkg->name, newpkg->version);
|
|
|
|
/* pre_install scriptlet */
|
|
if(alpm_pkg_has_scriptlet(newpkg) && !(trans->flags & PM_TRANS_FLAG_NOSCRIPTLET)) {
|
|
_alpm_runscriptlet(handle->root, newpkg->origin_data.file,
|
|
"pre_install", newpkg->version, NULL, trans);
|
|
}
|
|
}
|
|
|
|
/* we override any pre-set reason if we have alldeps or allexplicit set */
|
|
if(trans->flags & PM_TRANS_FLAG_ALLDEPS) {
|
|
newpkg->reason = PM_PKG_REASON_DEPEND;
|
|
} else if(trans->flags & PM_TRANS_FLAG_ALLEXPLICIT) {
|
|
newpkg->reason = PM_PKG_REASON_EXPLICIT;
|
|
}
|
|
|
|
if(oldpkg) {
|
|
/* set up fake remove transaction */
|
|
int ret = upgrade_remove(oldpkg, newpkg, trans, db);
|
|
if(ret != 0) {
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
/* prepare directory for database entries so permission are correct after
|
|
changelog/install script installation (FS#12263) */
|
|
if(_alpm_db_prepare(db, newpkg)) {
|
|
alpm_logaction("error: could not create database entry %s-%s\n",
|
|
alpm_pkg_get_name(newpkg), alpm_pkg_get_version(newpkg));
|
|
pm_errno = PM_ERR_DB_WRITE;
|
|
ret = -1;
|
|
goto cleanup;
|
|
}
|
|
|
|
if(!(trans->flags & PM_TRANS_FLAG_DBONLY)) {
|
|
struct archive *archive;
|
|
struct archive_entry *entry;
|
|
char cwd[PATH_MAX] = "";
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "extracting files\n");
|
|
|
|
if ((archive = archive_read_new()) == NULL) {
|
|
pm_errno = PM_ERR_LIBARCHIVE;
|
|
ret = -1;
|
|
goto cleanup;
|
|
}
|
|
|
|
archive_read_support_compression_all(archive);
|
|
archive_read_support_format_all(archive);
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "archive: %s\n", newpkg->origin_data.file);
|
|
if(archive_read_open_filename(archive, newpkg->origin_data.file,
|
|
ARCHIVE_DEFAULT_BYTES_PER_BLOCK) != ARCHIVE_OK) {
|
|
pm_errno = PM_ERR_PKG_OPEN;
|
|
ret = -1;
|
|
goto cleanup;
|
|
}
|
|
|
|
/* save the cwd so we can restore it later */
|
|
if(getcwd(cwd, PATH_MAX) == NULL) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not get current working directory\n"));
|
|
cwd[0] = 0;
|
|
}
|
|
|
|
/* libarchive requires this for extracting hard links */
|
|
chdir(handle->root);
|
|
|
|
/* call PROGRESS once with 0 percent, as we sort-of skip that here */
|
|
if(is_upgrade) {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_UPGRADE_START,
|
|
alpm_pkg_get_name(newpkg), 0, pkg_count, pkg_current);
|
|
} else {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_ADD_START,
|
|
alpm_pkg_get_name(newpkg), 0, pkg_count, pkg_current);
|
|
}
|
|
|
|
for(i = 0; archive_read_next_header(archive, &entry) == ARCHIVE_OK; i++) {
|
|
double percent;
|
|
|
|
if(newpkg->size != 0) {
|
|
/* Using compressed size for calculations here, as newpkg->isize is not
|
|
* exact when it comes to comparing to the ACTUAL uncompressed size
|
|
* (missing metadata sizes) */
|
|
int64_t pos = archive_position_compressed(archive);
|
|
percent = (double)pos / (double)newpkg->size;
|
|
_alpm_log(PM_LOG_DEBUG, "decompression progress: "
|
|
"%f%% (%"PRId64" / %jd)\n",
|
|
percent*100.0, pos, (intmax_t)newpkg->size);
|
|
if(percent >= 1.0) {
|
|
percent = 1.0;
|
|
}
|
|
} else {
|
|
percent = 0.0;
|
|
}
|
|
|
|
if(is_upgrade) {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_UPGRADE_START,
|
|
alpm_pkg_get_name(newpkg), (int)(percent * 100), pkg_count,
|
|
pkg_current);
|
|
} else {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_ADD_START,
|
|
alpm_pkg_get_name(newpkg), (int)(percent * 100), pkg_count,
|
|
pkg_current);
|
|
}
|
|
|
|
/* extract the next file from the archive */
|
|
errors += extract_single_file(archive, entry, newpkg, oldpkg,
|
|
trans, db);
|
|
}
|
|
archive_read_finish(archive);
|
|
|
|
/* restore the old cwd is we have it */
|
|
if(strlen(cwd)) {
|
|
chdir(cwd);
|
|
}
|
|
|
|
if(errors) {
|
|
ret = -1;
|
|
if(is_upgrade) {
|
|
_alpm_log(PM_LOG_ERROR, _("problem occurred while upgrading %s\n"),
|
|
newpkg->name);
|
|
alpm_logaction("error: problem occurred while upgrading %s\n",
|
|
newpkg->name);
|
|
} else {
|
|
_alpm_log(PM_LOG_ERROR, _("problem occurred while installing %s\n"),
|
|
newpkg->name);
|
|
alpm_logaction("error: problem occurred while installing %s\n",
|
|
newpkg->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* make an install date (in UTC) */
|
|
newpkg->installdate = time(NULL);
|
|
|
|
_alpm_log(PM_LOG_DEBUG, "updating database\n");
|
|
_alpm_log(PM_LOG_DEBUG, "adding database entry '%s'\n", newpkg->name);
|
|
|
|
if(_alpm_db_write(db, newpkg, INFRQ_ALL)) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not update database entry %s-%s\n"),
|
|
alpm_pkg_get_name(newpkg), alpm_pkg_get_version(newpkg));
|
|
alpm_logaction("error: could not update database entry %s-%s\n",
|
|
alpm_pkg_get_name(newpkg), alpm_pkg_get_version(newpkg));
|
|
pm_errno = PM_ERR_DB_WRITE;
|
|
ret = -1;
|
|
goto cleanup;
|
|
}
|
|
|
|
if(_alpm_db_add_pkgincache(db, newpkg) == -1) {
|
|
_alpm_log(PM_LOG_ERROR, _("could not add entry '%s' in cache\n"),
|
|
alpm_pkg_get_name(newpkg));
|
|
}
|
|
|
|
if(is_upgrade) {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_UPGRADE_START,
|
|
alpm_pkg_get_name(newpkg), 100, pkg_count, pkg_current);
|
|
} else {
|
|
PROGRESS(trans, PM_TRANS_PROGRESS_ADD_START,
|
|
alpm_pkg_get_name(newpkg), 100, pkg_count, pkg_current);
|
|
}
|
|
|
|
/* run the post-install script if it exists */
|
|
if(alpm_pkg_has_scriptlet(newpkg)
|
|
&& !(trans->flags & PM_TRANS_FLAG_NOSCRIPTLET)) {
|
|
if(is_upgrade) {
|
|
_alpm_runscriptlet(handle->root, scriptlet, "post_upgrade",
|
|
alpm_pkg_get_version(newpkg),
|
|
oldpkg ? alpm_pkg_get_version(oldpkg) : NULL, trans);
|
|
} else {
|
|
_alpm_runscriptlet(handle->root, scriptlet, "post_install",
|
|
alpm_pkg_get_version(newpkg), NULL, trans);
|
|
}
|
|
}
|
|
|
|
if(is_upgrade) {
|
|
EVENT(trans, PM_TRANS_EVT_UPGRADE_DONE, newpkg, oldpkg);
|
|
} else {
|
|
EVENT(trans, PM_TRANS_EVT_ADD_DONE, newpkg, oldpkg);
|
|
}
|
|
|
|
cleanup:
|
|
_alpm_pkg_free(oldpkg);
|
|
return(ret);
|
|
}
|
|
|
|
int _alpm_add_commit(pmtrans_t *trans, pmdb_t *db)
|
|
{
|
|
int pkg_count, pkg_current;
|
|
alpm_list_t *targ;
|
|
|
|
ALPM_LOG_FUNC;
|
|
|
|
ASSERT(trans != NULL, RET_ERR(PM_ERR_TRANS_NULL, -1));
|
|
ASSERT(db != NULL, RET_ERR(PM_ERR_DB_NULL, -1));
|
|
|
|
if(trans->packages == NULL) {
|
|
return(0);
|
|
}
|
|
|
|
pkg_count = alpm_list_count(trans->packages);
|
|
pkg_current = 1;
|
|
|
|
/* loop through our package list adding/upgrading one at a time */
|
|
for(targ = trans->packages; targ; targ = targ->next) {
|
|
if(handle->trans->state == STATE_INTERRUPTED) {
|
|
return(0);
|
|
}
|
|
|
|
pmpkg_t *newpkg = (pmpkg_t *)targ->data;
|
|
commit_single_pkg(newpkg, pkg_current, pkg_count, trans, db);
|
|
pkg_current++;
|
|
}
|
|
|
|
/* run ldconfig if it exists */
|
|
_alpm_ldconfig(handle->root);
|
|
|
|
return(0);
|
|
}
|
|
|
|
/* vim: set ts=2 sw=2 noet: */
|