327 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			327 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*-
 | 
						|
 * Copyright (c) 2009 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 <stdio.h>
 | 
						|
#include <stdbool.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <dirent.h>
 | 
						|
#include <sys/utsname.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
 | 
						|
#include <xbps_api.h>
 | 
						|
#include "index.h"
 | 
						|
 | 
						|
/* Array of valid architectures */
 | 
						|
static const char *archdirs[] = { "i686", "x86_64", "noarch", NULL };
 | 
						|
 | 
						|
static prop_dictionary_t
 | 
						|
repoidx_getdict(const char *pkgdir)
 | 
						|
{
 | 
						|
	prop_dictionary_t dict;
 | 
						|
	prop_array_t array;
 | 
						|
	char *plist;
 | 
						|
 | 
						|
	plist = xbps_get_pkg_index_plist(pkgdir);
 | 
						|
	if (plist == NULL)
 | 
						|
		return NULL;
 | 
						|
 | 
						|
	dict = prop_dictionary_internalize_from_file(plist);
 | 
						|
	if (dict == NULL) {
 | 
						|
		dict = prop_dictionary_create();
 | 
						|
		if (dict == NULL)
 | 
						|
			goto out;
 | 
						|
 | 
						|
		array = prop_array_create();
 | 
						|
		if (array == NULL) {
 | 
						|
			prop_object_release(dict);
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
 | 
						|
		prop_dictionary_set(dict, "packages", array);
 | 
						|
		prop_object_release(array);
 | 
						|
		prop_dictionary_set_cstring_nocopy(dict,
 | 
						|
		    "pkgindex-version", XBPS_PKGINDEX_VERSION);
 | 
						|
	}
 | 
						|
out:
 | 
						|
	free(plist);
 | 
						|
 | 
						|
	return dict;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
xbps_repo_addpkg_index(prop_dictionary_t idxdict, const char *file)
 | 
						|
{
 | 
						|
	prop_dictionary_t newpkgd, curpkgd;
 | 
						|
	prop_array_t pkgar;
 | 
						|
	struct archive *ar;
 | 
						|
	struct archive_entry *entry;
 | 
						|
	struct stat st;
 | 
						|
	const char *pkgname, *version, *regver;
 | 
						|
	char *sha256, *filen = NULL, *tmpfilen = NULL;
 | 
						|
	int rv = 0;
 | 
						|
 | 
						|
	assert(file != NULL);
 | 
						|
	assert(pkgdir != NULL);
 | 
						|
 | 
						|
	tmpfilen = strdup(file);
 | 
						|
	if (tmpfilen == NULL)
 | 
						|
		return errno;
 | 
						|
 | 
						|
	filen = basename(tmpfilen);
 | 
						|
	if (strcmp(tmpfilen, filen) == 0) {
 | 
						|
		free(tmpfilen);
 | 
						|
		return EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	ar = archive_read_new();
 | 
						|
	if (ar == NULL) {
 | 
						|
		rv = errno;
 | 
						|
		goto out;
 | 
						|
	}
 | 
						|
	/* Enable support for tar format and all compression methods */
 | 
						|
	archive_read_support_compression_all(ar);
 | 
						|
	archive_read_support_format_tar(ar);
 | 
						|
 | 
						|
	if ((rv = archive_read_open_filename(ar, file,
 | 
						|
	     ARCHIVE_READ_BLOCKSIZE)) == -1) {
 | 
						|
		rv = errno;
 | 
						|
		goto out1;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Open the binary package and read the props.plist
 | 
						|
	 * into a buffer.
 | 
						|
	 */
 | 
						|
	while (archive_read_next_header(ar, &entry) == ARCHIVE_OK) {
 | 
						|
		if (strstr(archive_entry_pathname(entry), XBPS_PKGPROPS) == 0) {
 | 
						|
			archive_read_data_skip(ar);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		newpkgd = xbps_read_dict_from_archive_entry(ar, entry);
 | 
						|
		if (newpkgd == NULL) {
 | 
						|
			printf("%s: can't read %s metadata file, skipping!\n",
 | 
						|
			    file, XBPS_PKGPROPS);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		prop_dictionary_get_cstring_nocopy(newpkgd, "pkgname",
 | 
						|
		    &pkgname);
 | 
						|
		prop_dictionary_get_cstring_nocopy(newpkgd, "version",
 | 
						|
		    &version);
 | 
						|
		/*
 | 
						|
		 * Check if this package exists already in the index, but first
 | 
						|
		 * checking the version. If current package version is greater
 | 
						|
		 * than current registered package, update the index; otherwise
 | 
						|
		 * pass to the next one.
 | 
						|
		 */
 | 
						|
		curpkgd = xbps_find_pkg_in_dict(idxdict, "packages", pkgname);
 | 
						|
		if (curpkgd) {
 | 
						|
			prop_dictionary_get_cstring_nocopy(curpkgd,
 | 
						|
			    "version", ®ver);
 | 
						|
			if (xbps_cmpver(version, regver) <= 0) {
 | 
						|
				printf("W: skipping %s. %s-%s already "
 | 
						|
				    "registered.\n", filen, pkgname, regver);
 | 
						|
				prop_object_release(newpkgd);
 | 
						|
				archive_read_data_skip(ar);
 | 
						|
				rv = EEXIST;
 | 
						|
				break;
 | 
						|
			}
 | 
						|
			/*
 | 
						|
			 * Current package is newer than the one that is
 | 
						|
			 * registered actually, remove old package from
 | 
						|
			 * the index.
 | 
						|
			 */
 | 
						|
			rv = xbps_remove_pkg_from_dict(idxdict,
 | 
						|
			    "packages", pkgname);
 | 
						|
			if (rv != 0) {
 | 
						|
				prop_object_release(newpkgd);
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		/*
 | 
						|
		 * We have the dictionary now, add the required
 | 
						|
		 * objects for the index.
 | 
						|
		 */
 | 
						|
		prop_dictionary_set_cstring(newpkgd, "filename", filen);
 | 
						|
		sha256 = xbps_get_file_hash(file);
 | 
						|
		if (sha256 == NULL) {
 | 
						|
			prop_object_release(newpkgd);
 | 
						|
			rv = errno;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		prop_dictionary_set_cstring(newpkgd, "filename-sha256", sha256);
 | 
						|
		free(sha256);
 | 
						|
 | 
						|
		if (stat(file, &st) == -1) {
 | 
						|
			prop_object_release(newpkgd);
 | 
						|
			rv = errno;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		prop_dictionary_set_uint64(newpkgd, "filename-size",
 | 
						|
		    (uint64_t)st.st_size);
 | 
						|
		/*
 | 
						|
		 * Add dictionary into the index and update package count.
 | 
						|
		 */
 | 
						|
		/* Get package array in repo index file */
 | 
						|
		pkgar = prop_dictionary_get(idxdict, "packages");
 | 
						|
		if (pkgar == NULL) {
 | 
						|
			prop_object_release(newpkgd);
 | 
						|
			rv = errno;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		if (!xbps_add_obj_to_array(pkgar, newpkgd)) {
 | 
						|
			prop_object_release(newpkgd);
 | 
						|
			rv = EINVAL;
 | 
						|
			break;
 | 
						|
		}
 | 
						|
		prop_dictionary_set_uint64(idxdict, "total-pkgs",
 | 
						|
		    prop_array_count(pkgar));
 | 
						|
		printf("Registered %s-%s in package index.\n",
 | 
						|
		    pkgname, version);
 | 
						|
		printf("\033[1A\033[K");
 | 
						|
		break;
 | 
						|
	}
 | 
						|
 | 
						|
out1:
 | 
						|
	archive_read_finish(ar);
 | 
						|
out:
 | 
						|
	free(tmpfilen);
 | 
						|
 | 
						|
	return rv;
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
xbps_repo_genindex(const char *pkgdir)
 | 
						|
{
 | 
						|
	prop_dictionary_t idxdict = NULL;
 | 
						|
	struct dirent *dp;
 | 
						|
	DIR *dirp;
 | 
						|
	struct utsname un;
 | 
						|
	uint64_t npkgcnt = 0;
 | 
						|
	char *binfile, *path, *plist;
 | 
						|
	size_t i;
 | 
						|
	int rv = 0;
 | 
						|
	bool registered_newpkgs = false, foundpkg = false;
 | 
						|
 | 
						|
	if (uname(&un) == -1)
 | 
						|
		return errno;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Create or read existing package index plist file.
 | 
						|
	 */
 | 
						|
	idxdict = repoidx_getdict(pkgdir);
 | 
						|
	if (idxdict == NULL)
 | 
						|
		return errno;
 | 
						|
 | 
						|
	plist = xbps_get_pkg_index_plist(pkgdir);
 | 
						|
	if (plist == NULL) {
 | 
						|
		prop_object_release(idxdict);
 | 
						|
		return errno;
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Iterate over the known architecture directories to find
 | 
						|
	 * binary packages.
 | 
						|
	 */
 | 
						|
	for (i = 0; archdirs[i] != NULL; i++) {
 | 
						|
		if ((strcmp(archdirs[i], un.machine)) &&
 | 
						|
		    (strcmp(archdirs[i], "noarch")))
 | 
						|
			continue;
 | 
						|
 | 
						|
		path = xbps_xasprintf("%s/%s", pkgdir, archdirs[i]);
 | 
						|
		if (path == NULL) {
 | 
						|
			rv = errno;
 | 
						|
			goto out;
 | 
						|
		}
 | 
						|
 | 
						|
		dirp = opendir(path);
 | 
						|
		if (dirp == NULL) {
 | 
						|
			free(path);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
 | 
						|
		while ((dp = readdir(dirp)) != NULL) {
 | 
						|
			if ((strcmp(dp->d_name, ".") == 0) ||
 | 
						|
			    (strcmp(dp->d_name, "..") == 0))
 | 
						|
				continue;
 | 
						|
 | 
						|
			/* Ignore unknown files */
 | 
						|
			if (strstr(dp->d_name, ".xbps") == NULL)
 | 
						|
				continue;
 | 
						|
 | 
						|
			foundpkg = true;
 | 
						|
			binfile = xbps_xasprintf("%s/%s", path, dp->d_name);
 | 
						|
			if (binfile == NULL) {
 | 
						|
				(void)closedir(dirp);
 | 
						|
				free(path);
 | 
						|
				rv = errno;
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
			rv = xbps_repo_addpkg_index(idxdict, binfile);
 | 
						|
			free(binfile);
 | 
						|
			if (rv == EEXIST)
 | 
						|
				continue;
 | 
						|
			else if (rv != 0) {
 | 
						|
				(void)closedir(dirp);
 | 
						|
				free(path);
 | 
						|
				goto out;
 | 
						|
			}
 | 
						|
			registered_newpkgs = true;
 | 
						|
		}
 | 
						|
		(void)closedir(dirp);
 | 
						|
		free(path);
 | 
						|
	}
 | 
						|
 | 
						|
	if (foundpkg == false) {
 | 
						|
		/* No packages were found in directory */
 | 
						|
		rv = ENOENT;
 | 
						|
	} else {
 | 
						|
		/*
 | 
						|
		 * Show total count registered packages.
 | 
						|
		 */
 | 
						|
		prop_dictionary_get_uint64(idxdict, "total-pkgs", &npkgcnt);
 | 
						|
		printf("%ju packages registered in package index.\n", npkgcnt);
 | 
						|
		/*
 | 
						|
		 * Don't write plist file if no packages were registered.
 | 
						|
		 */
 | 
						|
		if (registered_newpkgs == false)
 | 
						|
			goto out;
 | 
						|
		/*
 | 
						|
		 * If any package was registered in package index, write
 | 
						|
		 * plist file to storage.
 | 
						|
		 */
 | 
						|
		if (!prop_dictionary_externalize_to_file(idxdict, plist))
 | 
						|
			rv = errno;
 | 
						|
	}
 | 
						|
out:
 | 
						|
	free(plist);
 | 
						|
	prop_object_release(idxdict);
 | 
						|
 | 
						|
	return rv;
 | 
						|
}
 |