package.c 25.1 KB
Newer Older
1 2 3
/* package.c - Alpine Package Keeper (APK)
 *
 * Copyright (C) 2005-2008 Natanael Copa <n@tanael.org>
4
 * Copyright (C) 2008-2011 Timo Teräs <timo.teras@iki.fi>
5 6
 * All rights reserved.
 *
7
 * This program is free software; you can redistribute it and/or modify it
8 9 10 11
 * under the terms of the GNU General Public License version 2 as published
 * by the Free Software Foundation. See http://www.gnu.org/ for details.
 */

12
#include <errno.h>
13 14 15
#include <fcntl.h>
#include <ctype.h>
#include <stdio.h>
16
#include <limits.h>
17 18 19 20 21
#include <malloc.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
Natanael Copa's avatar
Natanael Copa committed
22
#include <sys/stat.h>
23

Timo Teräs's avatar
Timo Teräs committed
24 25
#include <openssl/pem.h>

26 27 28 29
#include "apk_defines.h"
#include "apk_archive.h"
#include "apk_package.h"
#include "apk_database.h"
Natanael Copa's avatar
Natanael Copa committed
30
#include "apk_print.h"
31

32 33 34 35 36
void apk_pkg_format_plain(struct apk_package *pkg, apk_blob_t to)
{
	/* pkgname-1.0.apk */
	apk_blob_push_blob(&to, APK_BLOB_STR(pkg->name->name));
	apk_blob_push_blob(&to, APK_BLOB_STR("-"));
37
	apk_blob_push_blob(&to, *pkg->version);
38 39 40 41 42 43 44 45 46
	apk_blob_push_blob(&to, APK_BLOB_STR(".apk"));
	apk_blob_push_blob(&to, APK_BLOB_PTR_LEN("", 1));
}

void apk_pkg_format_cache(struct apk_package *pkg, apk_blob_t to)
{
	/* pkgname-1.0_alpha1.12345678.apk */
	apk_blob_push_blob(&to, APK_BLOB_STR(pkg->name->name));
	apk_blob_push_blob(&to, APK_BLOB_STR("-"));
47
	apk_blob_push_blob(&to, *pkg->version);
48 49 50 51 52 53 54
	apk_blob_push_blob(&to, APK_BLOB_STR("."));
	apk_blob_push_hexdump(&to, APK_BLOB_PTR_LEN((char *) pkg->csum.data,
						    APK_CACHE_CSUM_BYTES));
	apk_blob_push_blob(&to, APK_BLOB_STR(".apk"));
	apk_blob_push_blob(&to, APK_BLOB_PTR_LEN("", 1));
}

55
struct apk_package *apk_pkg_new(void)
56
{
Timo Teräs's avatar
Timo Teräs committed
57 58 59
	struct apk_package *pkg;

	pkg = calloc(1, sizeof(struct apk_package));
Timo Teräs's avatar
Timo Teräs committed
60
	if (pkg != NULL) {
Timo Teräs's avatar
Timo Teräs committed
61
		apk_dependency_array_init(&pkg->depends);
Timo Teräs's avatar
Timo Teräs committed
62 63
		apk_dependency_array_init(&pkg->install_if);
	}
Timo Teräs's avatar
Timo Teräs committed
64 65

	return pkg;
66
}
67

68 69 70 71 72 73 74 75 76 77
struct apk_installed_package *apk_pkg_install(struct apk_database *db,
					      struct apk_package *pkg)
{
	struct apk_installed_package *ipkg;

	if (pkg->ipkg != NULL)
		return pkg->ipkg;

	pkg->ipkg = ipkg = calloc(1, sizeof(struct apk_installed_package));
	ipkg->pkg = pkg;
Timo Teräs's avatar
Timo Teräs committed
78 79
	apk_string_array_init(&ipkg->triggers);
	apk_string_array_init(&ipkg->pending_triggers);
80
	apk_dependency_array_init(&ipkg->replaces);
Timo Teräs's avatar
Timo Teräs committed
81 82 83 84 85 86 87

	/* Overlay override information resides in a nameless package */
	if (pkg->name != NULL) {
		db->installed.stats.packages++;
		list_add_tail(&ipkg->installed_pkgs_list,
			      &db->installed.packages);
	}
88 89 90 91 92 93 94 95 96 97 98 99 100 101

	return ipkg;
}

void apk_pkg_uninstall(struct apk_database *db, struct apk_package *pkg)
{
	struct apk_installed_package *ipkg = pkg->ipkg;
	int i;

	if (ipkg == NULL)
		return;

	if (db != NULL)
		db->installed.stats.packages--;
102

103 104
	list_del(&ipkg->installed_pkgs_list);

Timo Teräs's avatar
Timo Teräs committed
105
	if (ipkg->triggers->num) {
106
		list_del(&ipkg->trigger_pkgs_list);
107
		list_init(&ipkg->trigger_pkgs_list);
108 109 110
		for (i = 0; i < ipkg->triggers->num; i++)
			free(ipkg->triggers->item[i]);
	}
Timo Teräs's avatar
Timo Teräs committed
111 112
	apk_string_array_free(&ipkg->triggers);
	apk_string_array_free(&ipkg->pending_triggers);
113
	apk_dependency_array_free(&ipkg->replaces);
114 115 116 117 118 119

	for (i = 0; i < APK_SCRIPT_MAX; i++)
		if (ipkg->script[i].ptr != NULL)
			free(ipkg->script[i].ptr);
	free(ipkg);
	pkg->ipkg = NULL;
120 121
}

122 123 124 125 126 127
int apk_pkg_parse_name(apk_blob_t apkname,
		       apk_blob_t *name,
		       apk_blob_t *version)
{
	int i, dash = 0;

128 129 130
	if (APK_BLOB_IS_NULL(apkname))
		return -1;

131 132 133 134 135 136 137 138
	for (i = apkname.len - 2; i >= 0; i--) {
		if (apkname.ptr[i] != '-')
			continue;
		if (isdigit(apkname.ptr[i+1]))
			break;
		if (++dash >= 2)
			return -1;
	}
139 140 141
	if (i < 0)
		return -1;

142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
	if (name != NULL)
		*name = APK_BLOB_PTR_LEN(apkname.ptr, i);
	if (version != NULL)
		*version = APK_BLOB_PTR_PTR(&apkname.ptr[i+1],
					    &apkname.ptr[apkname.len-1]);

	return 0;
}

int apk_deps_add(struct apk_dependency_array **depends,
		 struct apk_dependency *dep)
{
	struct apk_dependency_array *deps = *depends;
	int i;

	if (deps != NULL) {
		for (i = 0; i < deps->num; i++) {
Timo Teräs's avatar
Timo Teräs committed
159 160
			if (deps->item[i].name == dep->name) {
				deps->item[i] = *dep;
161
				return 0;
Timo Teräs's avatar
Timo Teräs committed
162
			}
163 164 165 166 167 168
		}
	}

	*apk_dependency_array_add(depends) = *dep;
	return 0;
}
Timo Teräs's avatar
Timo Teräs committed
169

170 171 172 173 174 175 176 177 178 179 180 181 182 183
void apk_deps_del(struct apk_dependency_array **pdeps,
		  struct apk_name *name)
{
	struct apk_dependency_array *deps = *pdeps;
	int i;

	if (deps == NULL)
		return;

	for (i = 0; i < deps->num; i++) {
		if (deps->item[i].name != name)
			continue;

		deps->item[i] = deps->item[deps->num-1];
Timo Teräs's avatar
Timo Teräs committed
184
		apk_dependency_array_resize(pdeps, deps->num-1);
185 186 187 188
		break;
	}
}

189 190 191 192 193
struct parse_depend_ctx {
	struct apk_database *db;
	struct apk_dependency_array **depends;
};

194
void apk_blob_pull_dep(apk_blob_t *b, struct apk_database *db, struct apk_dependency *dep)
195 196
{
	struct apk_name *name;
197
	apk_blob_t bname, bop, bver = APK_BLOB_NULL;
198
	int mask = APK_DEPMASK_REQUIRE;
199
	size_t len;
200

201
	/* [!]name[<,<=,=,>=,>]ver */
202 203 204
	if (APK_BLOB_IS_NULL(*b))
		goto fail;
	if (b->ptr[0] == '!') {
205
		mask = 0;
206 207
		b->ptr++;
		b->len--;
208
	}
209
	if (apk_blob_cspn(*b, "<>= ", &bname, &bop)) {
210 211 212
		int i;

		if (mask == 0)
213
			goto fail;
214
		if (!apk_blob_spn(bop, "<>=", &bop, &bver))
215
			goto fail;
Timo Teräs's avatar
Timo Teräs committed
216 217 218
		mask = 0;
		for (i = 0; i < bop.len; i++) {
			switch (bop.ptr[i]) {
219 220 221 222 223 224 225 226 227 228 229
			case '<':
				mask |= APK_VERSION_LESS;
				break;
			case '>':
				mask |= APK_VERSION_GREATER;
				break;
			case '=':
				mask |= APK_VERSION_EQUAL;
				break;
			}
		}
230 231
		if ((mask & APK_DEPMASK_CHECKSUM) != APK_DEPMASK_CHECKSUM &&
		    !apk_version_validate(bver))
232 233 234 235 236
			goto fail;
	} else {
		bname = *b;
		bop = APK_BLOB_NULL;
		bver = APK_BLOB_NULL;
237
	}
238 239 240
	len = bname.len + bop.len + bver.len;
	b->ptr += len;
	b->len -= len;
241

242
	name = apk_db_get_name(db, bname);
243
	if (name == NULL)
244
		goto fail;
245 246 247

	*dep = (struct apk_dependency){
		.name = name,
248
		.version = apk_blob_atomize_dup(bver),
249
		.result_mask = mask,
250
	};
251 252 253 254
	return;
fail:
	*dep = (struct apk_dependency){ .name = NULL };
	*b = APK_BLOB_NULL;
Timo Teräs's avatar
Timo Teräs committed
255 256 257 258 259
}

void apk_dep_from_pkg(struct apk_dependency *dep, struct apk_database *db,
		      struct apk_package *pkg)
{
260 261 262 263 264 265
	char buf[64];
	apk_blob_t b = APK_BLOB_BUF(buf);

	apk_blob_push_csum(&b, &pkg->csum);
	b = apk_blob_pushed(APK_BLOB_BUF(buf), b);

Timo Teräs's avatar
Timo Teräs committed
266
	*dep = (struct apk_dependency) {
267
		.name = pkg->name,
268 269
		.version = apk_blob_atomize_dup(b),
		.result_mask = APK_DEPMASK_CHECKSUM,
Timo Teräs's avatar
Timo Teräs committed
270 271 272 273 274 275 276 277 278 279 280
	};
}

static int parse_depend(void *ctx, apk_blob_t blob)
{
	struct parse_depend_ctx *pctx = (struct parse_depend_ctx *) ctx;
	struct apk_dependency *dep, p;

	if (blob.len == 0)
		return 0;

281 282
	apk_blob_pull_dep(&blob, pctx->db, &p);
	if (p.name == NULL || blob.len != 0)
Timo Teräs's avatar
Timo Teräs committed
283 284 285 286 287 288
		return -1;

	dep = apk_dependency_array_add(pctx->depends);
	if (dep == NULL)
		return -1;
	*dep = p;
289 290

	return 0;
291 292 293 294
}

int apk_dep_is_satisfied(struct apk_dependency *dep, struct apk_package *pkg)
{
295 296 297 298 299
	if (pkg == NULL) {
		if (dep->result_mask == APK_DEPMASK_CONFLICT)
			return 1;
		return 0;
	}
300 301 302 303 304 305 306 307 308 309 310 311 312 313 314
	if (dep->name != pkg->name)
		return 0;
	if (dep->result_mask == APK_DEPMASK_CHECKSUM) {
		struct apk_checksum csum;
		apk_blob_t b = *dep->version;

		apk_blob_pull_csum(&b, &csum);
		if (apk_checksum_compare(&csum, &pkg->csum) == 0)
			return 1;
	} else {
		if (apk_version_compare_blob(*pkg->version, *dep->version)
		    & dep->result_mask)
			return 1;
	}
	return 0;
315 316
}

317
void apk_blob_pull_deps(apk_blob_t *b, struct apk_database *db, struct apk_dependency_array **deps)
318
{
319
	struct parse_depend_ctx ctx = { db, deps };
320

321 322
	if (b->len > 0 && b->ptr[b->len-1] == '\n')
		b->len--;
323

324
	apk_blob_for_each_segment(*b, " ", parse_depend, &ctx);
325 326
}

327 328 329 330 331 332 333 334 335 336
void apk_blob_push_dep(apk_blob_t *to, struct apk_dependency *dep)
{
	if (dep->result_mask == APK_DEPMASK_CONFLICT)
		apk_blob_push_blob(to, APK_BLOB_PTR_LEN("!", 1));

	apk_blob_push_blob(to, APK_BLOB_STR(dep->name->name));

	if (dep->result_mask != APK_DEPMASK_CONFLICT &&
	    dep->result_mask != APK_DEPMASK_REQUIRE) {
		apk_blob_push_blob(to, APK_BLOB_STR(apk_version_op_string(dep->result_mask)));
337
		apk_blob_push_blob(to, *dep->version);
338 339 340
	}
}

341 342 343 344 345 346 347 348 349 350 351 352 353 354
void apk_blob_push_deps(apk_blob_t *to, struct apk_dependency_array *deps)
{
	int i;

	if (deps == NULL)
		return;

	for (i = 0; i < deps->num; i++) {
		if (i)
			apk_blob_push_blob(to, APK_BLOB_PTR_LEN(" ", 1));
		apk_blob_push_dep(to, &deps->item[i]);
	}
}

355 356
int apk_deps_write(struct apk_dependency_array *deps, struct apk_ostream *os)
{
357 358 359
	apk_blob_t blob;
	char tmp[256];
	int i, n = 0;
360 361 362 363 364

	if (deps == NULL)
		return 0;

	for (i = 0; i < deps->num; i++) {
365 366 367 368 369 370 371 372 373
		blob = APK_BLOB_BUF(tmp);
		if (i)
			apk_blob_push_blob(&blob, APK_BLOB_PTR_LEN(" ", 1));
		apk_blob_push_dep(&blob, &deps->item[i]);

		blob = apk_blob_pushed(APK_BLOB_BUF(tmp), blob);
		if (APK_BLOB_IS_NULL(blob) || 
		    os->write(os, blob.ptr, blob.len) != blob.len)
			return -1;
374

375
		n += blob.len;
376 377 378 379 380
	}

	return n;
}

Timo Teräs's avatar
Timo Teräs committed
381
const char *apk_script_types[] = {
382 383 384 385 386 387
	[APK_SCRIPT_PRE_INSTALL]	= "pre-install",
	[APK_SCRIPT_POST_INSTALL]	= "post-install",
	[APK_SCRIPT_PRE_DEINSTALL]	= "pre-deinstall",
	[APK_SCRIPT_POST_DEINSTALL]	= "post-deinstall",
	[APK_SCRIPT_PRE_UPGRADE]	= "pre-upgrade",
	[APK_SCRIPT_POST_UPGRADE]	= "post-upgrade",
388
	[APK_SCRIPT_TRIGGER]		= "trigger",
389 390 391 392 393 394
};

int apk_script_type(const char *name)
{
	int i;

Timo Teräs's avatar
Timo Teräs committed
395 396 397
	for (i = 0; i < ARRAY_SIZE(apk_script_types); i++)
		if (apk_script_types[i] &&
		    strcmp(apk_script_types[i], name) == 0)
398 399
			return i;

400
	return APK_SCRIPT_INVALID;
401 402
}

403
void apk_sign_ctx_init(struct apk_sign_ctx *ctx, int action,
404
		       struct apk_checksum *identity, int keys_fd)
Timo Teräs's avatar
Timo Teräs committed
405 406
{
	memset(ctx, 0, sizeof(struct apk_sign_ctx));
407
	ctx->keys_fd = keys_fd;
Timo Teräs's avatar
Timo Teräs committed
408 409
	ctx->action = action;
	switch (action) {
Timo Teräs's avatar
Timo Teräs committed
410 411 412
	case APK_SIGN_VERIFY:
		ctx->md = EVP_md_null();
		break;
413 414 415 416 417 418 419 420 421 422
	case APK_SIGN_VERIFY_IDENTITY:
		if (identity->type == APK_CHECKSUM_MD5) {
			ctx->md = EVP_md5();
			ctx->control_started = 1;
			ctx->data_started = 1;
		} else {
			ctx->md = EVP_sha1();
		}
		memcpy(&ctx->identity, identity, sizeof(ctx->identity));
		break;
Timo Teräs's avatar
Timo Teräs committed
423
	case APK_SIGN_GENERATE:
424
	case APK_SIGN_VERIFY_AND_GENERATE:
Timo Teräs's avatar
Timo Teräs committed
425 426
		ctx->md = EVP_sha1();
		break;
427 428 429 430 431 432
	default:
		action = APK_SIGN_NONE;
		ctx->md = EVP_md_null();
		ctx->control_started = 1;
		ctx->data_started = 1;
		break;
Timo Teräs's avatar
Timo Teräs committed
433
	}
434 435
	EVP_MD_CTX_init(&ctx->mdctx);
	EVP_DigestInit_ex(&ctx->mdctx, ctx->md, NULL);
436
	EVP_MD_CTX_set_flags(&ctx->mdctx, EVP_MD_CTX_FLAG_ONESHOT);
Timo Teräs's avatar
Timo Teräs committed
437 438 439 440 441 442 443 444
}

void apk_sign_ctx_free(struct apk_sign_ctx *ctx)
{
	if (ctx->signature.data.ptr != NULL)
		free(ctx->signature.data.ptr);
	if (ctx->signature.pkey != NULL)
		EVP_PKEY_free(ctx->signature.pkey);
445
	EVP_MD_CTX_cleanup(&ctx->mdctx);
Timo Teräs's avatar
Timo Teräs committed
446 447 448 449 450 451 452 453 454
}

int apk_sign_ctx_process_file(struct apk_sign_ctx *ctx,
			      const struct apk_file_info *fi,
			      struct apk_istream *is)
{
	if (ctx->data_started)
		return 1;

455
	if (fi->name[0] != '.' || strchr(fi->name, '/') != NULL) {
456 457 458 459 460 461
		/* APKv1.0 compatibility - first non-hidden file is
		 * considered to start the data section of the file.
		 * This does not make any sense if the file has v2.0
		 * style .PKGINFO */
		if (ctx->has_data_checksum)
			return -ENOMSG;
Timo Teräs's avatar
Timo Teräs committed
462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478
		ctx->data_started = 1;
		ctx->control_started = 1;
		return 1;
	}

	if (ctx->control_started)
		return 1;

	if (strncmp(fi->name, ".SIGN.", 6) != 0) {
		ctx->control_started = 1;
		return 1;
	}

	/* A signature file */
	ctx->num_signatures++;

	/* Found already a trusted key */
479 480
	if ((ctx->action != APK_SIGN_VERIFY &&
	     ctx->action != APK_SIGN_VERIFY_AND_GENERATE) ||
481
	    ctx->signature.pkey != NULL)
Timo Teräs's avatar
Timo Teräs committed
482 483
		return 0;

484 485 486
	if (ctx->keys_fd < 0)
		return 0;

Timo Teräs's avatar
Timo Teräs committed
487 488
	if (strncmp(&fi->name[6], "RSA.", 4) == 0 ||
	    strncmp(&fi->name[6], "DSA.", 4) == 0) {
489
		int fd = openat(ctx->keys_fd, &fi->name[10], O_RDONLY|O_CLOEXEC);
490 491 492 493 494
	        BIO *bio;

		if (fd < 0)
			return 0;

495
		bio = BIO_new_fp(fdopen(fd, "r"), BIO_CLOSE);
496
		ctx->signature.pkey = PEM_read_bio_PUBKEY(bio, NULL, NULL, NULL);
Timo Teräs's avatar
Timo Teräs committed
497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512
		if (ctx->signature.pkey != NULL) {
			if (fi->name[6] == 'R')
				ctx->md = EVP_sha1();
			else
				ctx->md = EVP_dss1();
		}
		BIO_free(bio);
	} else
		return 0;

	if (ctx->signature.pkey != NULL)
		ctx->signature.data = apk_blob_from_istream(is, fi->size);

	return 0;
}

513
int apk_sign_ctx_parse_pkginfo_line(void *ctx, apk_blob_t line)
514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539
{
	struct apk_sign_ctx *sctx = (struct apk_sign_ctx *) ctx;
	apk_blob_t l, r;

	if (line.ptr == NULL || line.len < 1 || line.ptr[0] == '#')
		return 0;

	if (!apk_blob_split(line, APK_BLOB_STR(" = "), &l, &r))
		return 0;

	if (sctx->data_started == 0 &&
	    apk_blob_compare(APK_BLOB_STR("datahash"), l) == 0) {
		sctx->has_data_checksum = 1;
		sctx->md = EVP_sha256();
		apk_blob_pull_hexdump(
			&r, APK_BLOB_PTR_LEN(sctx->data_checksum,
					     EVP_MD_size(sctx->md)));
	}

	return 0;
}

int apk_sign_ctx_verify_tar(void *sctx, const struct apk_file_info *fi,
			    struct apk_istream *is)
{
	struct apk_sign_ctx *ctx = (struct apk_sign_ctx *) sctx;
540
	int r;
541

542 543 544
	r = apk_sign_ctx_process_file(ctx, fi, is);
	if (r <= 0)
		return r;
545 546 547

	if (strcmp(fi->name, ".PKGINFO") == 0) {
		apk_blob_t blob = apk_blob_from_istream(is, fi->size);
548 549 550
		apk_blob_for_each_segment(
			blob, "\n",
			apk_sign_ctx_parse_pkginfo_line, ctx);
551 552 553 554 555 556
		free(blob.ptr);
	}

	return 0;
}

557
int apk_sign_ctx_mpart_cb(void *ctx, int part, apk_blob_t data)
Timo Teräs's avatar
Timo Teräs committed
558 559 560
{
	struct apk_sign_ctx *sctx = (struct apk_sign_ctx *) ctx;
	unsigned char calculated[EVP_MAX_MD_SIZE];
Timo Teräs's avatar
Timo Teräs committed
561 562 563 564 565 566 567
	int r, end_of_control;

	if ((part == APK_MPART_DATA) ||
	    (part == APK_MPART_BOUNDARY && sctx->data_started))
		goto update_digest;

	/* Still in signature blocks? */
568 569 570
	if (!sctx->control_started) {
		if (part == APK_MPART_END)
			return -EKEYREJECTED;
Timo Teräs's avatar
Timo Teräs committed
571
		goto reset_digest;
572
	}
Timo Teräs's avatar
Timo Teräs committed
573 574 575 576 577 578

	/* Grab state and mark all remaining block as data */
	end_of_control = (sctx->data_started == 0);
	sctx->data_started = 1;

	/* End of control-block and control does not have data checksum? */
579 580
	if (sctx->has_data_checksum == 0 && end_of_control &&
	    part != APK_MPART_END)
Timo Teräs's avatar
Timo Teräs committed
581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601
		goto update_digest;

	/* Drool in the remaining of the digest block now, we will finish
	 * it on all cases */
	EVP_DigestUpdate(&sctx->mdctx, data.ptr, data.len);

	/* End of control-block and checking control hash/signature or
	 * end of data-block and checking its hash/signature */
	if (sctx->has_data_checksum && !end_of_control) {
		/* End of control-block and check it's hash */
		EVP_DigestFinal_ex(&sctx->mdctx, calculated, NULL);
		if (EVP_MD_CTX_size(&sctx->mdctx) == 0 ||
		    memcmp(calculated, sctx->data_checksum,
		           EVP_MD_CTX_size(&sctx->mdctx)) != 0)
			return -EKEYREJECTED;
		sctx->data_verified = 1;
		if (!(apk_flags & APK_ALLOW_UNTRUSTED) &&
		    !sctx->control_verified)
			return -ENOKEY;
		return 0;
	}
Timo Teräs's avatar
Timo Teräs committed
602

Timo Teräs's avatar
Timo Teräs committed
603 604
	switch (sctx->action) {
	case APK_SIGN_VERIFY:
605
	case APK_SIGN_VERIFY_AND_GENERATE:
Timo Teräs's avatar
Timo Teräs committed
606 607 608 609
		if (sctx->signature.pkey == NULL) {
			if (apk_flags & APK_ALLOW_UNTRUSTED)
				break;
			return -ENOKEY;
Timo Teräs's avatar
Timo Teräs committed
610 611
		}

Timo Teräs's avatar
Timo Teräs committed
612 613 614 615 616 617 618 619
		r = EVP_VerifyFinal(&sctx->mdctx,
			(unsigned char *) sctx->signature.data.ptr,
			sctx->signature.data.len,
			sctx->signature.pkey);
		if (r != 1)
			return -EKEYREJECTED;
		sctx->control_verified = 1;
		if (!sctx->has_data_checksum && part == APK_MPART_END)
620
			sctx->data_verified = 1;
Timo Teräs's avatar
Timo Teräs committed
621 622 623 624 625 626 627 628 629
		break;
	case APK_SIGN_VERIFY_IDENTITY:
		/* Reset digest for hashing data */
		EVP_DigestFinal_ex(&sctx->mdctx, calculated, NULL);
		if (memcmp(calculated, sctx->identity.data,
			   sctx->identity.type) != 0)
			return -EKEYREJECTED;
		sctx->control_verified = 1;
		if (!sctx->has_data_checksum && part == APK_MPART_END)
630
			sctx->data_verified = 1;
Timo Teräs's avatar
Timo Teräs committed
631 632 633 634 635 636 637 638
		break;
	case APK_SIGN_GENERATE:
		/* Package identity is the checksum */
		sctx->identity.type = EVP_MD_CTX_size(&sctx->mdctx);
		EVP_DigestFinal_ex(&sctx->mdctx, sctx->identity.data, NULL);
		if (sctx->action == APK_SIGN_GENERATE &&
		    sctx->has_data_checksum)
			return -ECANCELED;
639
		break;
Timo Teräs's avatar
Timo Teräs committed
640
	}
641 642 643 644
	if (sctx->action == APK_SIGN_VERIFY_AND_GENERATE) {
		sctx->identity.type = EVP_MD_CTX_size(&sctx->mdctx);
		EVP_DigestFinal_ex(&sctx->mdctx, sctx->identity.data, NULL);
	}
Timo Teräs's avatar
Timo Teräs committed
645 646 647 648 649 650 651 652
reset_digest:
	EVP_DigestInit_ex(&sctx->mdctx, sctx->md, NULL);
	EVP_MD_CTX_set_flags(&sctx->mdctx, EVP_MD_CTX_FLAG_ONESHOT);
	return 0;

update_digest:
	EVP_MD_CTX_clear_flags(&sctx->mdctx, EVP_MD_CTX_FLAG_ONESHOT);
	EVP_DigestUpdate(&sctx->mdctx, data.ptr, data.len);
Timo Teräs's avatar
Timo Teräs committed
653 654 655
	return 0;
}

656 657 658
struct read_info_ctx {
	struct apk_database *db;
	struct apk_package *pkg;
Timo Teräs's avatar
Timo Teräs committed
659
	struct apk_sign_ctx *sctx;
660 661
};

662 663
int apk_pkg_add_info(struct apk_database *db, struct apk_package *pkg,
		     char field, apk_blob_t value)
Timo Teräs's avatar
Timo Teräs committed
664 665 666
{
	switch (field) {
	case 'P':
667
		pkg->name = apk_db_get_name(db, value);
Timo Teräs's avatar
Timo Teräs committed
668 669
		break;
	case 'V':
670
		pkg->version = apk_blob_atomize_dup(value);
Timo Teräs's avatar
Timo Teräs committed
671 672 673 674 675 676 677 678
		break;
	case 'T':
		pkg->description = apk_blob_cstr(value);
		break;
	case 'U':
		pkg->url = apk_blob_cstr(value);
		break;
	case 'L':
679
		pkg->license = apk_blob_atomize_dup(value);
Timo Teräs's avatar
Timo Teräs committed
680
		break;
681
	case 'A':
682
		pkg->arch = apk_blob_atomize_dup(value);
683
		break;
Timo Teräs's avatar
Timo Teräs committed
684
	case 'D':
685
		apk_blob_pull_deps(&value, db, &pkg->depends);
Timo Teräs's avatar
Timo Teräs committed
686 687
		break;
	case 'C':
Timo Teräs's avatar
Timo Teräs committed
688
		apk_blob_pull_csum(&value, &pkg->csum);
Timo Teräs's avatar
Timo Teräs committed
689 690
		break;
	case 'S':
691
		pkg->size = apk_blob_pull_uint(&value, 10);
Timo Teräs's avatar
Timo Teräs committed
692 693
		break;
	case 'I':
694
		pkg->installed_size = apk_blob_pull_uint(&value, 10);
Timo Teräs's avatar
Timo Teräs committed
695
		break;
Timo Teräs's avatar
Timo Teräs committed
696
	case 'i':
697
		apk_blob_pull_deps(&value, db, &pkg->install_if);
Timo Teräs's avatar
Timo Teräs committed
698
		break;
699 700 701 702 703 704 705 706 707 708 709 710
	case 'o':
		pkg->origin = apk_blob_atomize_dup(value);
		break;
	case 'm':
		pkg->maintainer = apk_blob_atomize_dup(value);
		break;
	case 't':
		pkg->build_time = apk_blob_pull_uint(&value, 10);
		break;
	case 'c':
		pkg->commit = apk_blob_cstr(value);
		break;
711
	case 'F': case 'M': case 'R': case 'Z': case 'r':
712 713
		/* installed db entries which are handled in database.c */
		return 1;
714
	default:
715 716 717 718 719 720 721
		/* lower case index entries are safe to be ignored */
		if (!islower(field)) {
			pkg->filename = APK_PKG_UNINSTALLABLE;
			db->compat_notinstallable = 1;
		}
		db->compat_newfeatures = 1;
		return 1;
Timo Teräs's avatar
Timo Teräs committed
722
	}
723 724
	if (APK_BLOB_IS_NULL(value))
		return -1;
Timo Teräs's avatar
Timo Teräs committed
725 726 727 728 729 730 731 732 733
	return 0;
}

static int read_info_line(void *ctx, apk_blob_t line)
{
	static struct {
		const char *str;
		char field;
	} fields[] = {
Timo Teräs's avatar
Timo Teräs committed
734 735 736 737 738 739 740 741 742
		{ "pkgname",	'P' },
		{ "pkgver", 	'V' },
		{ "pkgdesc",	'T' },
		{ "url",	'U' },
		{ "size",	'I' },
		{ "license",	'L' },
		{ "arch",	'A' },
		{ "depend",	'D' },
		{ "install_if",	'i' },
743 744 745 746
		{ "origin",	'o' },
		{ "maintainer",	'm' },
		{ "builddate",	't' },
		{ "commit",	'c' },
Timo Teräs's avatar
Timo Teräs committed
747 748 749 750 751 752 753 754
	};
	struct read_info_ctx *ri = (struct read_info_ctx *) ctx;
	apk_blob_t l, r;
	int i;

	if (line.ptr == NULL || line.len < 1 || line.ptr[0] == '#')
		return 0;

755
	if (!apk_blob_split(line, APK_BLOB_STR(" = "), &l, &r))
Timo Teräs's avatar
Timo Teräs committed
756 757 758
		return 0;

	for (i = 0; i < ARRAY_SIZE(fields); i++) {
759
		if (apk_blob_compare(APK_BLOB_STR(fields[i].str), l) == 0) {
760
			apk_pkg_add_info(ri->db, ri->pkg, fields[i].field, r);
761
			return 0;
Timo Teräs's avatar
Timo Teräs committed
762 763
		}
	}
764
	apk_sign_ctx_parse_pkginfo_line(ri->sctx, line);
765

Timo Teräs's avatar
Timo Teräs committed
766 767 768
	return 0;
}

769
static int read_info_entry(void *ctx, const struct apk_file_info *ae,
770
			   struct apk_istream *is)
771 772 773
{
	struct read_info_ctx *ri = (struct read_info_ctx *) ctx;
	struct apk_package *pkg = ri->pkg;
774
	int r;
775

Timo Teräs's avatar
Timo Teräs committed
776
	/* Meta info and scripts */
777 778 779
	r = apk_sign_ctx_process_file(ri->sctx, ae, is);
	if (r <= 0)
		return r;
780

Timo Teräs's avatar
Timo Teräs committed
781
	if (ae->name[0] == '.') {
Timo Teräs's avatar
Timo Teräs committed
782 783
		/* APK 2.0 format */
		if (strcmp(ae->name, ".PKGINFO") == 0) {
784
			apk_blob_t blob = apk_blob_from_istream(is, ae->size);
Timo Teräs's avatar
Timo Teräs committed
785 786
			apk_blob_for_each_segment(blob, "\n", read_info_line, ctx);
			free(blob.ptr);
787
		} else if (strcmp(ae->name, ".INSTALL") == 0) {
788 789
			apk_warning("Package '%s-%s' contains deprecated .INSTALL",
				    pkg->name->name, pkg->version);
Timo Teräs's avatar
Timo Teräs committed
790
		}
791 792 793
		return 0;
	}

794 795 796
	return 0;
}

797 798
int apk_pkg_read(struct apk_database *db, const char *file,
	         struct apk_sign_ctx *sctx, struct apk_package **pkg)
799 800
{
	struct read_info_ctx ctx;
801
	struct apk_file_info fi;
802
	struct apk_bstream *bs;
803
	struct apk_istream *tar;
804
	int r;
805

806
	r = apk_file_get_info(AT_FDCWD, file, APK_CHECKSUM_NONE, &fi);
807 808
	if (r != 0)
		return r;
809

810
	memset(&ctx, 0, sizeof(ctx));
Timo Teräs's avatar
Timo Teräs committed
811
	ctx.sctx = sctx;
812
	ctx.pkg = apk_pkg_new();
813
	r = -ENOMEM;
814
	if (ctx.pkg == NULL)
815
		goto err;
816
	bs = apk_bstream_from_file(AT_FDCWD, file);
Timo Teräs's avatar
Timo Teräs committed
817
	if (bs == NULL)
818 819 820
		goto err;

	ctx.db = db;
821
	ctx.pkg->size = fi.size;
822

Timo Teräs's avatar
Timo Teräs committed
823
	tar = apk_bstream_gunzip_mpart(bs, apk_sign_ctx_mpart_cb, sctx);
824
	r = apk_tar_parse(tar, read_info_entry, &ctx, FALSE, &db->id_cache);
825
	tar->close(tar);
Timo Teräs's avatar
Timo Teräs committed
826
	if (r < 0 && r != -ECANCELED)
827
		goto err;
828 829 830
	if (ctx.pkg->name == NULL ||
	    ctx.pkg->filename == APK_PKG_UNINSTALLABLE) {
		r = -ENOTSUP;
831
		goto err;
832
	}
Timo Teräs's avatar
Timo Teräs committed
833 834
	if (sctx->action != APK_SIGN_VERIFY)
		ctx.pkg->csum = sctx->identity;
835
	ctx.pkg->filename = strdup(file);
836

837 838 839
	ctx.pkg = apk_db_pkg_add(db, ctx.pkg);
	if (pkg != NULL)
		*pkg = ctx.pkg;
840
	return 0;
841 842
err:
	apk_pkg_free(ctx.pkg);
843
	return r;
844 845 846 847 848 849 850
}

void apk_pkg_free(struct apk_package *pkg)
{
	if (pkg == NULL)
		return;

851
	apk_pkg_uninstall(NULL, pkg);
Timo Teräs's avatar
Timo Teräs committed
852
	apk_dependency_array_free(&pkg->depends);
Timo Teräs's avatar
Timo Teräs committed
853
	apk_dependency_array_free(&pkg->install_if);
854 855 856 857
	if (pkg->url)
		free(pkg->url);
	if (pkg->description)
		free(pkg->description);
858 859
	if (pkg->commit)
		free(pkg->commit);
860 861 862
	free(pkg);
}

863 864 865
int apk_ipkg_add_script(struct apk_installed_package *ipkg,
			struct apk_istream *is,
			unsigned int type, unsigned int size)
866
{
867
	void *ptr;
868 869
	int r;

870 871 872 873 874
	if (type >= APK_SCRIPT_MAX)
		return -1;

	ptr = malloc(size);
	r = is->read(is, ptr, size);
875
	if (r < 0) {
876
		free(ptr);
877 878 879
		return r;
	}

880 881 882 883 884
	if (ipkg->script[type].ptr)
		free(ipkg->script[type].ptr);
	ipkg->script[type].ptr = ptr;
	ipkg->script[type].len = size;
	return 0;
885 886
}

887 888
int apk_ipkg_run_script(struct apk_installed_package *ipkg,
			struct apk_database *db,
889
			unsigned int type, char **argv)
890
{
891
	static char * const environment[] = {
892 893 894
		"PATH=/usr/sbin:/usr/bin:/sbin:/bin",
		NULL
	};
895 896
	struct apk_package *pkg = ipkg->pkg;
	char fn[PATH_MAX];
897
	int fd, status, root_fd = db->root_fd;
898 899
	pid_t pid;

900 901
	if (type >= APK_SCRIPT_MAX)
		return -1;
902

903 904 905 906
	if (ipkg->script[type].ptr == NULL)
		return 0;

	argv[0] = (char *) apk_script_types[type];
907

908
	/* Avoid /tmp as it can be mounted noexec */
909 910
	snprintf(fn, sizeof(fn), "var/cache/misc/" PKG_VER_FMT ".%s",
		PKG_VER_PRINTF(pkg),
911 912
		apk_script_types[type]);

913 914 915 916
	apk_message("Executing %s", &fn[15]);
	if (apk_flags & APK_SIMULATE)
		return 0;

917
	fd = openat(root_fd, fn, O_CREAT|O_RDWR|O_TRUNC|O_CLOEXEC, 0755);
918 919
	if (fd < 0) {
		mkdirat(root_fd, "var/cache/misc", 0755);
920
		fd = openat(root_fd, fn, O_CREAT|O_RDWR|O_TRUNC|O_CLOEXEC, 0755);
921 922 923
		if (fd < 0)
			return -errno;
	}
924 925 926 927
	if (write(fd, ipkg->script[type].ptr, ipkg->script[type].len) < 0) {
		close(fd);
		return -errno;
	}
928
	close(fd);
929

930 931
	pid = fork();
	if (pid == -1)
932
		return -1;
933
	if (pid == 0) {
934
		if (fchdir(root_fd) < 0 || chroot(".") < 0) {
935 936 937 938 939
			apk_error("chroot: %s", strerror(errno));
		} else {
			execve(fn, argv, environment);
		}
		exit(1);
940
	}
941 942
	waitpid(pid, &status, 0);
	unlinkat(root_fd, fn, 0);
943
	apk_id_cache_reset(&db->id_cache);
944

945 946 947 948 949 950
	if (WIFEXITED(status)) {
		int rc = WEXITSTATUS(status);
		if (rc != 0)
			apk_warning("%s: returned error %d", &fn[15], rc);
		return 0;
	}
951
	return -1;
952 953
}

Timo Teräs's avatar
Timo Teräs committed
954
static int parse_index_line(void *ctx, apk_blob_t line)
955
{
Timo Teräs's avatar
Timo Teräs committed
956
	struct read_info_ctx *ri = (struct read_info_ctx *) ctx;
957

Timo Teräs's avatar
Timo Teräs committed
958 959
	if (line.len < 3 || line.ptr[1] != ':')
		return 0;
960

961
	apk_pkg_add_info(ri->db, ri->pkg, line.ptr[0], APK_BLOB_PTR_LEN(line.ptr+2, line.len-2));
962 963 964 965 966
	return 0;
}

struct apk_package *apk_pkg_parse_index_entry(struct apk_database *db, apk_blob_t blob)
{
Timo Teräs's avatar
Timo Teräs committed
967
	struct read_info_ctx ctx;
968

969
	ctx.pkg = apk_pkg_new();
Timo Teräs's avatar
Timo Teräs committed
970
	if (ctx.pkg == NULL)
971 972
		return NULL;

Timo Teräs's avatar
Timo Teräs committed
973
	ctx.db = db;
974

Timo Teräs's avatar
Timo Teräs committed
975 976 977 978
	apk_blob_for_each_segment(blob, "\n", parse_index_line, &ctx);

	if (ctx.pkg->name == NULL) {
		apk_pkg_free(ctx.pkg);
979 980
		apk_error("Failed to parse index entry: " BLOB_FMT,
			  BLOB_PRINTF(blob));
Timo Teräs's avatar
Timo Teräs committed
981
		ctx.pkg = NULL;
982 983
	}

Timo Teräs's avatar
Timo Teräs committed
984
	return ctx.pkg;
985 986
}

Timo Teräs's avatar
Timo Teräs committed
987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005
static int write_depends(struct apk_ostream *os, const char *field,
			 struct apk_dependency_array *deps)
{
	int r;

	if (deps->num == 0)
		return 0;

	if (os->write(os, field, 2) != 2)
		return -1;
	r = apk_deps_write(deps, os);
	if (r < 0)
		return r;
	if (os->write(os, "\n", 1) != 1)
		return -1;

	return 0;
}

1006 1007
int apk_pkg_write_index_entry(struct apk_package *info,
			      struct apk_ostream *os)
1008
{
1009
	char buf[512];
1010 1011 1012
	apk_blob_t bbuf = APK_BLOB_BUF(buf);

	apk_blob_push_blob(&bbuf, APK_BLOB_STR("C:"));
Timo Teräs's avatar
Timo Teräs committed
1013
	apk_blob_push_csum(&bbuf, &info->csum);
1014 1015 1016
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nP:"));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR(info->name->name));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nV:"));
1017
	apk_blob_push_blob(&bbuf, *info->version);
1018 1019
	if (info->arch != NULL) {
		apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nA:"));
1020
		apk_blob_push_blob(&bbuf, *info->arch);
1021
	}
1022 1023 1024 1025 1026 1027 1028 1029 1030
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nS:"));
	apk_blob_push_uint(&bbuf, info->size, 10);
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nI:"));
	apk_blob_push_uint(&bbuf, info->installed_size, 10);
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nT:"));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR(info->description));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nU:"));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR(info->url));
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nL:"));
1031
	apk_blob_push_blob(&bbuf, *info->license);
1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047
	if (info->origin) {
		apk_blob_push_blob(&bbuf, APK_BLOB_STR("\no:"));
		apk_blob_push_blob(&bbuf, *info->origin);
	}
	if (info->maintainer) {
		apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nm:"));
		apk_blob_push_blob(&bbuf, *info->maintainer);
	}
	if (info->build_time) {
		apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nt:"));
		apk_blob_push_uint(&bbuf, info->build_time, 10);
	}
	if (info->commit) {
		apk_blob_push_blob(&bbuf, APK_BLOB_STR("\nc:"));
		apk_blob_push_blob(&bbuf, APK_BLOB_STR(info->commit));
	}
1048 1049
	apk_blob_push_blob(&bbuf, APK_BLOB_STR("\n"));

Timo Teräs's avatar
Timo Teräs committed
1050 1051 1052
	if (APK_BLOB_IS_NULL(bbuf))
		return -1;

1053
	bbuf = apk_blob_pushed(APK_BLOB_BUF(buf), bbuf);
Timo Teräs's avatar
Timo Teräs committed
1054 1055 1056
	if (os->write(os, bbuf.ptr, bbuf.len) != bbuf.len ||
	    write_depends(os, "D:", info->depends) ||
	    write_depends(os, "i:", info->install_if))
1057
		return -1;
1058

1059
	return 0;
1060
}
1061

1062 1063
int apk_pkg_version_compare(struct apk_package *a, struct apk_package *b)
{
1064 1065 1066 1067
	if (a->version == b->version)
		return APK_VERSION_EQUAL;

	return apk_version_compare_blob(*a->version, *b->version);
1068
}