abuild.in 35 KB
Newer Older
Natanael Copa's avatar
Natanael Copa committed
1 2
#!/bin/sh

3
# script to build apk packages (light version of makepkg)
Natanael Copa's avatar
Natanael Copa committed
4 5 6 7 8 9 10
# Copyright (c) 2008 Natanael Copa <natanael.copa@gmail.com>
#
# Distributed under GPL-2
#
# Depends on: busybox utilities, fakeroot, 
#

Natanael Copa's avatar
Natanael Copa committed
11 12
abuild_ver=@VERSION@
sysconfdir=@sysconfdir@
13
abuildrepo_base=@abuildrepo@
Natanael Copa's avatar
Natanael Copa committed
14
datadir=@datadir@
Natanael Copa's avatar
Natanael Copa committed
15

16
program=${0##*/}
17
abuild_path=$(readlink -f $0)
Natanael Copa's avatar
Natanael Copa committed
18 19

# defaults
20
BUILD_BASE="build-base"
21
SUDO=${SUDO:-"sudo"}
22
FAKEROOT=${FAKEROOT:-"fakeroot"}
23
APK=${APK:-apk}
24
apk_opt_wait="--wait 30"
25

Natanael Copa's avatar
Natanael Copa committed
26
# read config
Natanael Copa's avatar
Natanael Copa committed
27
ABUILD_CONF=${ABUILD_CONF:-"$sysconfdir/abuild.conf"}
Natanael Copa's avatar
Natanael Copa committed
28 29
[ -f "$ABUILD_CONF" ] && . "$ABUILD_CONF"

30
default_colors() {
Natanael Copa's avatar
Natanael Copa committed
31 32 33 34 35 36
	NORMAL="\033[1;0m"
	STRONG="\033[1;1m"
	RED="\033[1;31m"
	GREEN="\033[1;32m"
	YELLOW="\033[1;33m"
	BLUE="\033[1;34m"
37 38 39 40 41 42 43 44 45 46 47 48 49 50
}

monochrome() {
	NORMAL=""
	STRONG=""
	RED=""
	GREEN=""
	YELLOW=""
	BLUE=""
}

#colors
if [ -n "$USE_COLORS" ]; then
    default_colors
Natanael Copa's avatar
Natanael Copa committed
51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
fi	
	

# functions
msg() {
	local prompt="$GREEN>>>${NORMAL}"
	local fake="${FAKEROOTKEY:+${BLUE}*${NORMAL}}"
	local name="${STRONG}${subpkgname:-$pkgname}${NORMAL}"
	[ -z "$quiet" ] && printf "${prompt} ${name}${fake}: $@\n" >&2
}

warning() {
	local prompt="${YELLOW}>>> WARNING:${NORMAL}"
	local fake="${FAKEROOTKEY:+${BLUE}*${NORMAL}}"
	local name="${STRONG}${subpkgname:-$pkgname}${NORMAL}"
	printf "${prompt} ${name}${fake}: $@\n" >&2
}

error() {
	local prompt="${RED}>>> ERROR:${NORMAL}"
	local fake="${FAKEROOTKEY:+${BLUE}*${NORMAL}}"
	local name="${STRONG}${subpkgname:-$pkgname}${NORMAL}"
	printf "${prompt} ${name}${fake}: $@\n" >&2
}
Natanael Copa's avatar
Natanael Copa committed
75 76

	
Linux User's avatar
Linux User committed
77
set_xterm_title() {
78
	if [ "$TERM" = xterm ] && [ -n "$USE_COLORS" ]; then
79
		 printf "\033]0;$1\007" >&2
Linux User's avatar
Linux User committed
80 81 82 83 84
	fi
} 

cleanup() {
	set_xterm_title ""
85
	if [ -z "$install_after" ] && [ -n "$uninstall_after" ]; then
86 87
		msg "Uninstalling dependencies..."
		$SUDO $APK del --quiet $apk_opt_wait $uninstall_after
88
	fi
Linux User's avatar
Linux User committed
89 90
}

Natanael Copa's avatar
Natanael Copa committed
91
die() {
Linux User's avatar
Linux User committed
92
	error "$@"
Linux User's avatar
Linux User committed
93
	cleanup
Natanael Copa's avatar
Natanael Copa committed
94 95 96
	exit 1
}

97
# check if apkbuild is basicly sane
98
default_sanitycheck() {
99
	local i= j= suggestion=
100
	msg "Checking sanity of $APKBUILD..."
101 102 103
	[ -z "$pkgname" ] && die "Missing pkgname in APKBUILD"
	[ -z "${pkgname##* *}" ] && die "pkgname contains spaces"
	[ -z "$pkgver" ] && die "Missing pkgver in APKBUILD"
104
	if [ "$pkgver" != "volatile" ] && [ -z "$nodeps" ]; then
105
		$APK version --check -q "$pkgver" ||\
Natanael Copa's avatar
Natanael Copa committed
106 107
			die "$pkgver is not a valid version"
	fi
108
	[ -z "$pkgrel" ] && die "Missing pkgrel in APKBUILD"
109 110 111 112
	[ -z "$pkgdesc" ] && die "Missing pkgdesc in APKBUILD"
	[ -z "$url" ] && die "Missing url in APKBUILD"
	[ -z "$license" ] && die "Missing license in APKBULID"

113 114 115 116 117 118 119
	# check so no package names starts with -
	for i in $pkgname $subpackages; do
		case $i in
		-*) die "${i%:*} is not a valid package name";;
		esac
	done

120 121 122 123 124 125 126 127 128 129
	# check if CARCH, CBUILD, CHOST and CTARGET is set
	if [ -z "$CARCH" ]; then
		case "$(uname -m)" in
		i[0-9]86) suggestion=" (Suggestion: CARCH=x86)";;
		x86_64) suggestion=" (Suggestion: CARCH=x86_64)";;
		esac
		die "Please set CARCH in /etc/abuild.conf$suggestion"
	fi
	[ -z "$CHOST" ] && die "Please set CHOST in /etc/abuild.conf"

Natanael Copa's avatar
Natanael Copa committed
130
	for i in $install; do
131
		local n=${i%.*}
132
		local suff=${i##*.}
133 134 135 136 137
		case "$suff" in
		pre-install|post-install|pre-upgrade|post-upgrade|pre-deinstall|post-deinstall);;
		*) die "$i: unknown install script suffix"
		esac
		if ! subpackages_has "$n" && [ "$n" != "$pkgname" ]; then
138
			die "$i: install script does not match pkgname or any subpackage"
139
		fi
140
		[ -e "$startdir/$i" ] || die "install script $i is missing"
Natanael Copa's avatar
Natanael Copa committed
141 142
	done
	
143 144 145 146 147 148 149 150
	for i in $triggers; do
		local f=${i%=*}
		local p=${f%.trigger}
		[ "$f" = "$i" ] && die "$f: triggers must contain '='" 
		[ "$p" = "$f" ] && die "$f: triggers scripts must have .trigger suffix"
		if ! subpackages_has "$p" && [ "$p" != "$pkgname" ]; then
			die "$p: trigger script does not match pkgname or any subpackage"
		fi
Natanael Copa's avatar
Natanael Copa committed
151

152 153
		[ -e "$startdir"/$f ] || die "trigger script $f is missing"
	done
154 155
	if [ -n "$source" ]; then
		for i in $source; do
Natanael Copa's avatar
Natanael Copa committed
156 157 158 159
			if install_has "$i"; then
				warning "You should not have \$install in source"
				continue
			fi
160
			md5sums_has ${i##*/} || die "${i##*/} is missing in md5sums"
Natanael Copa's avatar
Natanael Copa committed
161 162 163
			case "$i" in
				https://*) makedepends_has wget || die "wget must be in makedepends when source has https://" ;;
			esac
164 165
		done
	fi
Natanael Copa's avatar
Natanael Copa committed
166

167 168
	if [ -n "$md5sums" ]; then
		for i in $(echo "$md5sums" | awk '{ print $2 }'); do
Natanael Copa's avatar
Natanael Copa committed
169
			source_has $i || die "$i exists in md5sums but is missing in source"
170 171
		done
	fi
172

173 174 175
	# common spelling errors
	[ -n "$depend" ] && die "APKBUILD contains 'depend'. It should be depends"
	[ -n "$makedepend" ] && die "APKBUILD contains 'makedepend'. It should be makedepends"
Natanael Copa's avatar
Natanael Copa committed
176 177

	grep '^# Maintainer:' $APKBUILD >/dev/null || warning "No maintainer"
178 179

	makedepends_has 'g++' && warning "g++ should not be in makedepends"
180 181 182
	return 0
}

183 184 185 186
sanitycheck() {
	default_sanitycheck
}

187
md5check() {
188
	local dummy f endreturnval originalparams origin file
189 190 191
	if [ -z "$source" ]; then
		return 0
	fi
192
	if [ -z "$md5sums" ]; then
193
		die "Use 'abuild checksum' to generate/update the checksum(s)"
194
	fi
195 196 197
	if [ "$(echo $source | wc -l)" -ne "$(echo $md5sums | wc -l)" ]; then
		die "Number of md5sums does not correspond to number of sources"
	fi
198
	fetch || return 1
199
	msg "Checking md5sums..."
Natanael Copa's avatar
Natanael Copa committed
200
	cd "$srcdir" || return 1
201 202 203 204 205 206 207 208
	IFS=$'\n'
	endreturnval=0
	originalparams=$@
	set -- $source
	for src in $md5sums; do
		origin=$1; shift
		echo "$src" | md5sum -c
		if [ $? -ne 0 ]; then
209
			endreturnval=1
210 211 212 213 214 215 216 217 218 219 220
			is_remote $origin || continue
			echo "Because the remote file above failed the md5sum check it will be deleted."
			echo "Rebuilding will cause it to re-download which in some cases may fix the problem."
			file=`echo "$src" | sed 's/.*[ \t\n]\(.*\)/\1/'`
			echo "Deleting: $file"
			rm $file
		fi
	done
	unset IFS
	set -- $originalparams
	return $endreturnval
Natanael Copa's avatar
Natanael Copa committed
221 222
}

Natanael Copa's avatar
Natanael Copa committed
223 224
# verify upstream sources
sourcecheck() {
225 226 227 228 229 230 231 232 233 234
	local uri
	for uri in $source; do
		is_remote $uri || continue
		case "$uri" in
		saveas-*://*)
			uri=${uri#saveas-}
			uri=${uri%/*}
			;;
		esac
		wget -q -s "$uri" || return 1
Natanael Copa's avatar
Natanael Copa committed
235
	done
236
	return 0
Natanael Copa's avatar
Natanael Copa committed
237
}
238

Natanael Copa's avatar
Natanael Copa committed
239 240
uri_fetch() {
	local uri="$1"
241
	local d="${uri##*/}"	# $(basename $uri)
242 243 244 245
	local opts
	[ -n "$quiet" ] && opts="-q"
	[ -f "$SRCDEST/$d" ] && return 0

246 247 248 249 250
	# fix saveas-*://* URIs
	case "$uri" in
		# remove 'saveas-' from beginning and
		# '/filename' from end of URI
		saveas-*://*) uri="${uri:7:$(expr ${#uri} - 7 - ${#d} - 1)}";;
Natanael Copa's avatar
Natanael Copa committed
251
	esac
252 253 254 255 256

	# we need GNU wget for this
	case "$uri" in
		https://*) opts="--no-check-certificate";;
	esac
Natanael Copa's avatar
Natanael Copa committed
257
	
Natanael Copa's avatar
Natanael Copa committed
258
	mkdir -p "$SRCDEST"
259 260 261
	if [ -f "$SRCDEST/$d.part" ]; then
		msg "Partial download found. Trying to resume"
		opts="$opts -c"
Natanael Copa's avatar
Natanael Copa committed
262
	fi
Linux User's avatar
Linux User committed
263
	msg "Fetching $uri"
264 265
	wget $opts -O "$SRCDEST/$d.part" "$uri" \
		&& mv "$SRCDEST/$d.part" "$SRCDEST/$d"
Natanael Copa's avatar
Natanael Copa committed
266 267
}

268 269
is_remote() {
	case "$1" in
270
		http://*|ftp://*|https://*|saveas-*://*)
271 272 273 274 275
			return 0;;
	esac
	return 1
}

276 277 278 279 280
# try download from file from mirror first
uri_fetch_mirror() {
	local uri="$1"
	local d="${uri##*/}"	# $(basename $uri)
	if [ -n "$DISTFILES_MIRROR" ]; then
281 282 283 284 285
		if is_remote "$DISTFILES_MIRROR"; then
			uri_fetch "$DISTFILES_MIRROR"/$d && return 0
		else
			cp "$DISTFILES_MIRROR"/$d "$SRCDEST" && return 0
		fi
286 287 288 289
	fi
	uri_fetch "$uri"
}

290
default_fetch() {
Natanael Copa's avatar
Natanael Copa committed
291 292
	local s
	mkdir -p "$srcdir"
293 294
	for s in $source; do
		if is_remote "$s"; then
295
			uri_fetch_mirror "$s" || return 1
296 297 298 299
			ln -sf "$SRCDEST/${s##*/}" "$srcdir"/
		else		
			ln -sf "$startdir/$s" "$srcdir/"
		fi
Natanael Copa's avatar
Natanael Copa committed
300 301 302
	done
}

303 304 305 306
fetch() {
	default_fetch
}

Natanael Copa's avatar
Natanael Copa committed
307 308 309 310 311 312 313 314 315 316 317 318 319 320 321
# verify that all init.d scripts are openrc runscripts
initdcheck() {
	local i
	for i in $source; do
		case $i in
		*.initd)
			head -n 1 "$srcdir"/$i | grep -q '/sbin/runscript' \
				&& continue
			error "$i is not an openrc #!/sbin/runscript"
			return 1
			;;
		esac
	done			
}

Natanael Copa's avatar
Natanael Copa committed
322
# unpack the sources
323
default_unpack() {
Natanael Copa's avatar
Natanael Copa committed
324
	local u
325 326
	if [ -z "$force" ]; then
		md5check || return 1
Natanael Copa's avatar
Natanael Copa committed
327
		initdcheck || return 1
328
	fi
Natanael Copa's avatar
Natanael Copa committed
329 330 331 332
	mkdir -p "$srcdir"
	for u in $source; do
		local s="$SRCDEST/${u##*/}"	# $(basename $s)
		case "$s" in
Natanael Copa's avatar
Natanael Copa committed
333
			*.tar.gz|*.tgz)
334
				msg "Unpacking $s..."
Natanael Copa's avatar
Natanael Copa committed
335 336
				tar -C "$srcdir" -zxf "$s" || return 1;;
			*.tar.bz2)
337
				msg "Unpacking $s..."
Natanael Copa's avatar
Natanael Copa committed
338
				tar -C "$srcdir" -jxf "$s" || return 1;;
339 340 341 342
			*.tar.lzma)
				msg "Unpacking $s..."
				unlzma -c "$s" | tar -C "$srcdir" -x  \
					|| return 1;;
343 344 345
			*.tar.xz)
				msg "Unpacking $s..."
				unxz -c "$s" | tar -C "$srcdir" -x || return 1;;
Natanael Copa's avatar
Natanael Copa committed
346 347 348
			*.zip)
				msg "Unpacking $s..."
				unzip "$s" -d "$srcdir" || return 1;;
Natanael Copa's avatar
Natanael Copa committed
349 350 351 352
		esac
	done
}

353 354 355 356
unpack() {
	default_unpack
}

Natanael Copa's avatar
Natanael Copa committed
357 358
# cleanup source and package dir
clean() {
359
	msg "Cleaning temporary build dirs..."
Natanael Copa's avatar
Natanael Copa committed
360
	rm -rf "$srcdir"
361
	rm -rf "$pkgbasedir"
Natanael Copa's avatar
Natanael Copa committed
362 363 364 365 366 367
}

# cleanup fetched sources
cleancache() {
	local s
	for s in $source; do
368 369 370 371
		if is_remote "$s"; then
			msg "Cleaning downloaded ${s##*/}..."
			rm -f "$SRCDEST/${s##*/}"
		fi
Natanael Copa's avatar
Natanael Copa committed
372 373 374
	done
}

375 376 377 378 379 380 381 382 383 384
listpkgnames() {
	local i
	for i in $pkgname $subpackages; do
		echo ${i%:*}
	done
	for i in $linguas; do
		echo $pkgname-lang-$i
	done
}

Natanael Copa's avatar
Natanael Copa committed
385 386
cleanpkg() {
	local i
387
	getpkgver || return 1
388
	msg "Cleaning built packages..."
389
	for i in $(listpkgnames); do
390 391
		local p="${i%:*}-$pkgver-r$pkgrel"
		rm -f "$PKGDEST/$p.apk" "$PKGDEST/$p.src.tar.gz" \
392
			"$abuildrepo"/$p.apk "$abuildrepo"/*/$p.apk
Natanael Copa's avatar
Natanael Copa committed
393
	done
394
	# remove given packages from index
Natanael Copa's avatar
Natanael Copa committed
395 396
}

397 398 399
# clean all packages except current
cleanoldpkg() {
	local i j
400
	getpkgver || return 1
401
	msg "Cleaning all packages except $pkgver-r$pkgrel..."
402
	for i in $(listpkgnames); do
403 404 405 406 407
		local pn=${i%:*}
		for j in "$PKGDEST"/$pn-[0-9]*.apk ; do
			[ "$j" = "$PKGDEST/$pn-$pkgver-r$pkgrel.apk" ] \
				&& continue
			rm -f "$j" "$abuildrepo"/*/${j##*/}
408 409 410 411
		done
	done
	return 0
}
412 413 414 415 416

mkusers() {
	local i
	for i in $pkgusers; do
		if ! getent passwd $i >/dev/null; then
417
			local gopt=
418
			msg "Creating user $i"
419 420 421 422
			if getent group $i >/dev/null; then
				gopt="-G $i"
			fi
			$SUDO adduser -D -H $gopt $i || return 1
423 424 425 426 427 428 429 430 431 432 433
		fi
	done
	for i in $pkggroups; do
		if ! getent group $i >/dev/null; then
			msg "Creating group $i"
			$SUDO addgroup $i || return 1
		fi
	done
}


Natanael Copa's avatar
Natanael Copa committed
434 435
runpart() {
	local part=$1
436
	[ -n "$DEBUG" ] && msg "$part"
Natanael Copa's avatar
Natanael Copa committed
437 438 439 440
	$part || die "$part failed"
}

# override those in your build script
441 442 443 444 445 446 447 448
getpkgver() {
	# this func is supposed to be overridden by volatile packages
	if [ "$pkgver" = "volatile" ]; then
		error "Please provide a getpkgver() function in your APKBUILD"
		return 1
	fi
}

Natanael Copa's avatar
Natanael Copa committed
449 450 451 452
prepare() {
	:
}

Natanael Copa's avatar
Natanael Copa committed
453
build() {
454
	:
Natanael Copa's avatar
Natanael Copa committed
455 456
}

457 458 459
# generate a simple tar.gz package of pkgdir
targz() {
	cd "$pkgdir" || return 1
Natanael Copa's avatar
Natanael Copa committed
460
	mkdir -p "$PKGDEST"
461 462 463
	tar -czf "$PKGDEST"/$pkgname-$pkgver-r$pkgrel.tar.gz *
}

Natanael Copa's avatar
Natanael Copa committed
464 465 466 467 468 469 470 471 472
get_split_func() {
	# get the 'func' from "sub-pkg:func"
	local func=${1##*:}

	# get 'func' from "sub-pkg-func" if there was no :func
	[ "$func" = "$1" ] && func=${func##*-}
	echo $func
}

473
prepare_subpackages() {
Natanael Copa's avatar
Natanael Copa committed
474 475
	local i
	cd "$startdir"
476
	for i in $subpackages; do
Natanael Copa's avatar
Natanael Copa committed
477 478
		local func=$(get_split_func $i)
		# call abuild recursively, setting subpkg{dir,name}
479
		msg "Running split function $func..."
480
		subpkgdir="$pkgbasedir/${i%:*}" subpkgname="${i%:*}" \
481
			$0 $func prepare_package || return 1
Natanael Copa's avatar
Natanael Copa committed
482
	done
483 484 485 486 487 488 489
	# post check
	i=$(find "$pkgdir" -name '*.la')
	if [ -n "$i" ] && ! options_has "libtool"; then
		error "Libtool archives (*.la) files found and \$options has no 'libtool' flag:"
		echo "$i" | sed 's/^/  /'
		return 1
	fi
490 491
	# post check for /usr/share/locale
	if [ -d "$pkgdir"/usr/share/locale ]; then
492 493
		warning "Found /usr/share/locale"
		warning "Maybe add \$pkgname-lang to subpackages?"
494
	fi
Natanael Copa's avatar
Natanael Copa committed
495 496
}

497
default_lang() {
498
	pkgdesc="Languages for package $pkgname"
499 500 501 502 503 504 505 506 507 508 509 510 511
	install_if="$pkgname=$pkgver-r$pkgrel lang"
	arch="noarch"
	local dir
	for dir in ${langdir:-/usr/share/locale}; do
		mkdir -p "$subpkgdir"/${dir%/*}
		mv "$pkgdir"/"$dir" "$subpkgdir"/"$dir" || return 1
	done
}

lang() {
	default_lang
}

512
default_lang_subpkg() {
513 514 515 516
	if [ -z "$lang" ]; then
		error "lang is not set"
		return 1
	fi
517
	pkgdesc="$pkgname language pack for $lang"
518 519
	arch="noarch"
	install_if="$pkgname=$pkgver-r$pkgrel lang-$lang"
520 521 522 523 524
	local dir
	for dir in ${langdir:-/usr/share/locale}; do
		mkdir -p "$subpkgdir"/$dir
		mv "$pkgdir"/$dir/$lang* \
		"$subpkgdir"/$dir/ \
525
		|| return 1
526 527 528 529 530
	done
}

lang_subpkg() {
	default_lang_subpkg
531 532 533 534 535 536 537 538 539 540 541
}

prepare_language_packs() {
	for lang in $linguas; do
		lang="$lang" \
		subpkgname="$pkgname-lang-$lang" \
		subpkgdir="$pkgbasedir"/$subpkgname \
			$0 lang_subpkg prepare_package || return 1
	done
}

542 543 544 545 546 547 548 549 550 551 552 553
# echo '-dirty' if git is not clean
git_dirty() {
	if [ $(git status -s "$startdir" | wc -l) -ne 0 ]; then
		echo "-dirty"
	fi
}

# echo last commit hash id
git_last_commit() {
	git log --format=oneline -n 1 "$startdir" | awk '{print $1}'
}

554 555 556 557 558 559
get_maintainer() {
	if [ -z "$maintainer" ]; then
		maintainer=$(awk -F': ' '/\# *Maintainer/ {print $2}' "$APKBUILD")
	fi
}

560
prepare_metafiles() {
561
	getpkgver || return 1
562 563
	local name=${subpkgname:-$pkgname}
	[ -z "${name##* *}" ] && die "package name contains spaces"
564
	local dir=${subpkgdir:-$pkgdir}
565
	local pkg="$name-$pkgver-r$pkgrel.apk"
566
	local pkginfo="$controldir"/.PKGINFO
567
	local sub
Natanael Copa's avatar
Natanael Copa committed
568 569 570
	
	[ ! -d "$dir" ] && die "Missing $dir"
	cd "$dir"
571
	mkdir -p "$controldir"
Natanael Copa's avatar
Natanael Copa committed
572 573
	local builddate=$(date -u "+%s")
	local size=$(du -sk | awk '{print $1 * 1024}')
Natanael Copa's avatar
Natanael Copa committed
574
	local parch="$CARCH"
575 576 577 578 579

	# we need to wait with setting noarch til our build infra can handle it
#	if [ "$arch" = "noarch" ]; then
#		parch="noarch"
#	fi
Natanael Copa's avatar
Natanael Copa committed
580

581
	echo "# Generated by $(basename $0) $abuild_ver" >"$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
582
	if [ -n "$FAKEROOTKEY" ]; then
583
		echo "# using $($FAKEROOT -v)" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
584
	fi
585 586
	echo "# $(date -u)" >> "$pkginfo"
	cat >> "$pkginfo" <<EOF
587
pkgname = $name
588
pkgver = $pkgver-r$pkgrel
Natanael Copa's avatar
Natanael Copa committed
589 590 591 592 593
pkgdesc = $pkgdesc
url = $url
builddate = $builddate
packager = ${PACKAGER:-"Unknown"}
size = $size
Natanael Copa's avatar
Natanael Copa committed
594
arch = $parch
Natanael Copa's avatar
Natanael Copa committed
595
origin = $pkgname
Natanael Copa's avatar
Natanael Copa committed
596
EOF
597 598
	local i deps
	deps="$depends"
599
	if [ "$pkgname" != "busybox" ] && ! depends_has busbox; then
600 601
		for i in $install $triggers; do
			local s=${i%=*}
602
			[ "$name" != "${s%.*}" ] && continue
603
			if head -n 1 "$startdir/$s" | grep '^#!/bin/sh' >/dev/null ; then
604 605 606 607 608
				msg "Script found. busybox added as a dependency for $pkg"
				deps="$deps busybox"
				break
			fi
		done
609 610
	fi
	
611 612 613 614 615 616 617 618 619 620
	local last_commit="$(git_last_commit)$(git_dirty)"
	if [ -n "$last_commit" ]; then
		echo "commit = $last_commit" >> "$pkginfo"
	fi

	get_maintainer
	if [ -n "$maintainer" ]; then
		echo "maintainer = $maintainer" >> "$pkginfo"
	fi

Natanael Copa's avatar
Natanael Copa committed
621
	for i in $license; do
622
		echo "license = $i" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
623 624
	done
	for i in $replaces; do
625
		echo "replaces = $i" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
626
	done
627
	for i in $deps; do
628
		echo "depend = $i" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
629 630
	done
	for i in $conflicts; do
631
		echo "conflict = $i" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
632 633
	done
	for i in $provides; do
634
		echo "provides = $i" >> "$pkginfo"
Natanael Copa's avatar
Natanael Copa committed
635
	done
636 637 638 639 640 641
	for i in $triggers; do
		local f=${i%=*}
		local dirs=${i#*=}
		[ "${f%.trigger}" != "$name" ] && continue
		echo "triggers = ${dirs//:/ }" >> "$pkginfo"
	done
642 643 644
	if [ -n "$install_if" ]; then
		echo "install_if = $(echo $install_if)" >> "$pkginfo"
	fi
Natanael Copa's avatar
Natanael Copa committed
645 646

	local metafiles=".PKGINFO"
647 648 649
	for i in $install $triggers; do
		local f=${i%=*}
		local n=${f%.*}
650
		if [ "$n" != "$name" ]; then
651 652
			continue
		fi
653
		script=${f#$name}
654
		msg "Adding $script"
655
		cp "$startdir/$f" "$controldir/$script" || return 1
656
		chmod +x "$controldir/$script"
657 658
		metafiles="$metafiles $script"
	done
659
	echo $metafiles | tr ' ' '\n' > "$controldir"/.metafiles
660
}
661

662 663
prepare_tracedeps() {
	local dir=${subpkgdir:-$pkgdir}
664
	[ "$arch" = "noarch" ] && return 0
665
	options_has "!tracedeps" && return 0
666
	# lets tell all the .so files this package provides in .provides-so
667 668
	find -name '*.so' -o -name '*.so.[0-9]*' | sed 's:.*/::' \
		>"$controldir"/.provides-so
669 670 671 672 673
	# lets tell all the places we should look for .so files - all rpaths
	scanelf -q -Rr "$dir" | sed -e 's/[[:space:]].*//' -e 's/:/\n/' \
		| sort | uniq \
		>"$controldir"/.rpaths
	# now find the so dependencies
674
	scanelf -Rn "$dir" | tr ' ' ':' | awk -F ":" '$1 == "ET_DYN" || $1 == "ET_EXEC" {print $2}'  \
675 676 677
		| sed 's:,:\n:g' | sort | uniq \
	| while read i; do
		# only add files that are not self provided
678 679
		grep "^$i$" "$controldir"/.provides-so >/dev/null \
			|| echo $i >> "$controldir"/.needs-so
680
	done
Natanael Copa's avatar
Natanael Copa committed
681 682
}

683 684 685 686 687 688 689 690 691 692 693 694
# check if dir has arch specific binaries
dir_has_arch_binaries() {
	local dir="$1"
	# if scanelf returns something, then we have binaries
	[ -n "$(scanelf -R "$dir" | head -n 1)" ] && return 0

	# look for static *.a
	[ -n "$(find "$dir" -type f -name '*.a' | head -n 1)" ] && return 0

	return 1
}

695 696 697 698 699
# returns true if this is the -dev package
is_dev_pkg() {
	test "${subpkgname%-dev}" != "$subpkgname"
}

700 701 702 703 704 705 706
# check that noarch is set if needed
archcheck() {
	options_has "!archcheck" && return 0
	if dir_has_arch_binaries "${subpkgdir:-$pkgdir}"; then
		[ "$arch" != "noarch" ] && return 0
		error "Arch specific binaries found so arch must not be set to \"noarch\""
		return 1
707 708 709
	elif [ "$arch" != "noarch" ] && ! is_dev_pkg; then
		# we dont want -dev package go to noarch
		warning "No arch specific binaries found so arch should probably be set to \"noarch\""
710 711 712 713
	fi
	return 0
}

714
prepare_package() {
715
	msg "Preparing ${subpkgname:+sub}package ${subpkgname:-$pkgname}..."
716
	stripbin
717 718
	prepare_metafiles && prepare_tracedeps || return 1
	archcheck	
719 720 721 722 723 724 725 726
}

pkginfo_val() {
	local key="$1"
	local file="$2"
	awk -F ' = ' "\$1 == \"$key\" {print \$2}" "$file"
}

727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749
# find real path to so files
real_so_path() {
	local so="$1"
	shift
	while [ $# -gt 0 ]; do
		[ -e "$1"/$so ] && realpath "$1/$so" && return 0
		shift
	done
	error "$so: path not found"
	return 1
}

# search rpaths and /usr/lib /lib for given so files
find_so_files() {
	local rpaths=$(cat "$1")
	shift
	while [ $# -gt 0 ]; do
		real_so_path "$1" /usr/lib /lib $rpaths || return 1
		shift
	done
	return 0
}

750 751 752
trace_apk_deps() {
	local name="$1"
	local dir="$2"
Natanael Copa's avatar
Natanael Copa committed
753
	local i= j= found= autodeps= deppkgs= missing= so_paths= self_provided=
754
	msg "Tracing dependencies for $name..."
755 756 757
	# add pkgconfig if usr/lib/pkgconfig is found
	if [ -d "$pkgbasedir"/$name/usr/lib/pkgconfig ] \
			&& ! grep -q '^depend = pkgconfig' "$dir"/.PKGINFO; then
758
		msg "  added pkgconfig (found /usr/lib/pkgconfig)"
759 760 761 762
		autodeps="$autodeps pkgconfig"
	fi

	# special case for libpthread: we need depend on libgcc
763
	if [ -f "$dir"/.needs-so ] && grep -q -w '^libpthread.so.*' "$dir"/.needs-so \
764 765
			&& ! grep -q -w "^depend = libgcc" "$dir"/.PKGINFO; then
		autodeps="$autodeps libgcc"
766
		msg "  added libgcc (due to libpthread)"
767
	fi
768
	[ -f "$dir"/.needs-so ] && for i in $(cat "$dir"/.needs-so); do
769
		found=
770
		# first check if its provide by same apkbuild
771
		for j in "$dir"/../.control.*/.provides-so; do
772
			grep -w "$i" "$j" >/dev/null || continue
773
			found=${j%/.provides-so}
774
			found=${found##*/.control.}
775
			break
776
		done
777 778 779 780 781 782
		if [ -n "$found" ]; then
			if ! list_has "$found" $self_provided; then
				self_provided="$self_provided $found"
			fi
		else
			missing="$missing $i"
783
		fi
784 785 786
	done

	# find all packages that holds the so files
787 788 789 790 791
	if [ -f "$dir"/.rpaths ]; then
		so_files=$(find_so_files "$dir"/.rpaths $missing) || return 1
		deppkgs=$($APK info -q -W $so_files) || return 1
	fi

792
	for found in $self_provided $deppkgs; do
793
		if grep -w "^depend = ${found}$" "$dir"/.PKGINFO >/dev/null ; then
794 795 796
			warning "You can remove '$found' from depends"
			continue
		fi
797 798
		if [ "$found" != "$name" ] && ! list_has "$found" $autodeps; then
			autodeps="$autodeps $found"
799
			msg "  added $found"
800
		fi
801
	done
802

803 804 805 806 807 808 809
	[ -z "$autodeps" ] && return 0
	echo "# automatically detected:" >> "$dir"/.PKGINFO
	for i in $autodeps; do
		echo "depend = $i" >> "$dir"/.PKGINFO
	done
}

810 811
create_apks() {
	local file
812
	getpkgver || return 1
Natanael Copa's avatar
Natanael Copa committed
813
	mkdir -p "$PKGDEST"
814
	for file in "$pkgbasedir"/.control.*/.PKGINFO; do
815 816 817
		local dir="${file%/.PKGINFO}"
		local name=$(pkginfo_val pkgname $file)
		local ver=$(pkginfo_val pkgver $file)
818
		local apk=$name-$ver.apk
819
		local datadir="$pkgbasedir"/$name
820 821

		trace_apk_deps "$name" "$dir" || return 1
822
		msg "Creating $apk..."
823
		(
824 825
		cd "$datadir"
		# data.tar.gz
826 827 828 829 830 831
		set -- *
		if [ "$1" = '*' ]; then
			touch .dummy
			set -- .dummy
		fi
		tar -c "$@" | abuild-tar --hash | gzip -9 >"$dir"/data.tar.gz
832 833 834 835 836 837

		# append the hash for data.tar.gz
		local sha256=$(sha256sum "$dir"/data.tar.gz | cut -f1 -d' ')
		echo "datahash = $sha256" >> "$dir"/.PKGINFO

		# control.tar.gz
838
		cd "$dir"
839 840
		tar -c $(cat "$dir"/.metafiles) | abuild-tar --cut \
			| gzip -9 > control.tar.gz
841
		abuild-sign -q control.tar.gz || exit 1
Natanael Copa's avatar
Natanael Copa committed
842

843 844 845
		# create the final apk
		cat control.tar.gz data.tar.gz > "$PKGDEST"/$apk
	)
846 847 848
	done
}

849
clean_abuildrepo() {
850
	local apk
851 852 853
	cd "$abuildrepo" || return 1
	
	# remove compat symlink
854
	for d in "$abuildrepo/$CARCH"; do
855 856
		[ -L "$d" ] && rm "$d"
	done
857

858 859
	# remove broken links from abuildrepo
	for apk in *.apk */*.apk; do
860 861 862 863
		if [ -L "$apk" ] && [ ! -f "$apk" ]; then
			rm -f "$apk"
		fi
	done
864
}
865

866
mklinks_abuildrepo() {
867 868
	local apk
	mkdir -p "$abuildrepo"/$CARCH
869
	cd "$abuildrepo" || return 1
870
	# create links for this package
871
	for apk in $(listpkg); do
872
		[ -f "$PKGDEST"/$apk ] || continue
873
		ln -sf "$PKGDEST"/$apk "$abuildrepo"/$CARCH/$apk
874
	done
875 876 877 878 879 880 881 882 883 884 885 886 887 888
}

update_abuildrepo() {
	local d apk
	if ! apk_up2date || [ -n "$force" ]; then
		sanitycheck && builddeps && clean && fetch && unpack \
			&& prepare && mkusers && rootpkg || return 1
	fi

	clean_abuildrepo
	mklinks_abuildrepo

	cd "$abuildrepo"
	local index=$CARCH/APKINDEX.tar.gz
889

890
	msg "Updating the cached abuild repository index..."
891 892
	local sign=".SIGN.RSA.${SIGN_PUBLIC_KEY##*/}"
	local oldindex=
893 894
	if [ -f "$index" ]; then
		oldindex="--index $index"
895
	fi
896
	$APK index --quiet $oldindex --output "$index".unsigned \
897
		--description "$repo $(cd $startdir && git describe)" \
898 899
		--rewrite-arch $CARCH \
		$CARCH/*.apk || exit 1
900
	msg "Signing the index..."
901 902 903
	abuild-sign -q "$index".unsigned || exit 1
	mv "$index".unsigned "$index"
	chmod 644 "$index"
904 905
}

906
# predefined splitfunc doc
907
default_doc() {
908
	depends="$depends_doc"
909
	pkgdesc="$pkgdesc (documentation)"
910
	arch=${arch_doc:-"noarch"}
911

Natanael Copa's avatar
Natanael Copa committed
912
	local i
913
	for i in doc man info html sgml licenses gtk-doc; do
914 915 916
		if [ -d "$pkgdir/usr/share/$i" ]; then
			mkdir -p "$subpkgdir/usr/share"
			mv "$pkgdir/usr/share/$i" "$subpkgdir/usr/share/"
Natanael Copa's avatar
Natanael Copa committed
917 918
		fi
	done
919 920 921
	
	rm -f "$subpkgdir/usr/share/info/dir"

922 923 924
#	# compress info and man pages
#	find "$subpkgdir/usr/share" \( -name '*.info' -o -name '*.info-[1-9]' \
#		-o -name '*.[1-9]' \) -exec gzip {} \;
925 926 927 928

	# remove if empty, ignore error (not empty)
	rmdir "$pkgdir/usr/share" "$pkgdir/usr" 2>/dev/null

Natanael Copa's avatar
Natanael Copa committed
929
#	[ -d "$subpkgdir/usr/share/man" ] && depends="man"
Natanael Copa's avatar
Natanael Copa committed
930 931 932
	return 0
}

933 934 935 936
doc() {
	default_doc
}

937
# predefined splitfunc mod
938
default_mod() {
939
	depends="$kernel $depends_mod"
Natanael Copa's avatar
Natanael Copa committed
940
	for i in firmware modules; do
941 942 943
		if [ -d "$pkgdir/lib/$i" ]; then
			rm -rf "$subpkgdir/lib"
			mkdir -p "$subpkgdir/lib"
944
			mv "$pkgdir/lib/$i" "$subpkgdir/lib"
Natanael Copa's avatar
Natanael Copa committed
945 946 947 948
		fi
	done
}

949 950 951 952
mod() {
	default_mod
}

953
# predefined splitfunc dev
954
default_dev() {
955
	local i= j=
956
	depends="$pkgname $depends_dev"
957 958
	pkgdesc="$pkgdesc (development files)"

959 960 961 962
	for i in $origsubpackages; do
		[ "${i%:*}" = "$subpkgname" ] || depends="$depends ${i%:*}"
	done
		
963
	cd "$pkgdir" || return 0
964 965
	for i in usr/include usr/lib/pkgconfig usr/share/aclocal\
			usr/share/gettext usr/bin/*-config	\
966
			usr/share/vala/vapi usr/share/gir-[0-9]*\
967
			$(find -name include -type d) 		\
968
			$(find usr/ -name '*.[acho]' 2>/dev/null); do
969 970
		if [ -e "$pkgdir/$i" ] || [ -L "$pkgdir/$i" ]; then
			d="$subpkgdir/${i%/*}"	# dirname $i
Natanael Copa's avatar
Natanael Copa committed
971
			mkdir -p "$d"
972
			mv "$pkgdir/$i" "$d"
973
			rmdir "$pkgdir/${i%/*}" 2>/dev/null
Natanael Copa's avatar
Natanael Copa committed
974 975
		fi
	done
976 977 978 979 980 981 982
	# move *.so links needed when linking the apps to -dev packages
	for i in lib/*.so usr/lib/*.so; do
		if [ -L "$i" ]; then
			mkdir -p "$subpkgdir"/"${i%/*}"
			mv "$i" "$subpkgdir/$i" || return 1
		fi
	done
Natanael Copa's avatar
Natanael Copa committed
983
	return 0
Natanael Copa's avatar
Natanael Copa committed
984 985
}

986 987 988 989
dev() {
	default_dev
}

990 991 992 993
is_function() {
	type "$1" 2>&1 | head -n 1 | egrep -q "is a (shell )?function"
}

994 995 996 997 998 999 1000 1001
do_fakeroot() {
	if [ -n "$FAKEROOT" ]; then
		$FAKEROOT -- "$@"
	else
		"$@"
	fi
}

Natanael Copa's avatar
Natanael Copa committed
1002 1003
# build and package in fakeroot
rootpkg() {
1004 1005 1006 1007
	local do_build=build
	cd "$startdir"
	if is_function package; then
		build || return 1
1008
		do_build=package
1009
	fi
Natanael Copa's avatar
Natanael Copa committed
1010
	cd "$startdir"
1011
	[ -n "$FAKEROOT" ] && msg "Entering fakeroot..."
1012
	do_fakeroot "$abuild_path" $color_opt $do_build \
1013
		prepare_subpackages \
1014
		prepare_language_packs \
1015
		prepare_package \
1016
		create_apks
Natanael Copa's avatar
Natanael Copa committed
1017 1018 1019
}

srcpkg() {
1020
	getpkgver || return 1
1021
	local p="$pkgname-$pkgver-$pkgrel"
Natanael Copa's avatar
Natanael Copa committed
1022
	local prefix="${startdir##*/}"
Natanael Copa's avatar
Natanael Copa committed
1023 1024 1025 1026 1027
	local i files="$prefix/APKBUILD"
	for i in $source; do
		files="$files $prefix/${i##*/}"
	done
	mkdir -p "$PKGDEST"
1028
	msg "Creating source package $p.src.tar.gz..."
1029
	(cd .. && tar -zcf "$PKGDEST/$p.src.tar.gz" $files) 
Natanael Copa's avatar
Natanael Copa committed
1030 1031
}

Natanael Copa's avatar
Natanael Copa committed
1032 1033
# return true if arch is supported or noarch
check_arch() {
Natanael Copa's avatar
Natanael Copa committed
1034
	list_has $CARCH $arch || [ "$arch" = "noarch" ] || [ "$arch" = "all" ]
Natanael Copa's avatar
Natanael Copa committed
1035 1036
}

1037
# check if package is up to date
1038
apk_up2date() {
1039
	getpkgver || return 1
1040
	local pkg="$PKGDEST/$pkgname-$pkgver-r$pkgrel.apk"
1041
	local i s
Natanael Copa's avatar
Natanael Copa committed
1042
	cd "$startdir"
1043 1044 1045 1046 1047
	for i in $pkgname $subpackages; do
		[ -f "$PKGDEST/$pkgname-$pkgver-r$pkgrel.apk" ] || return 1
	done
	[ -n "$keep" ] && return 0

1048
	for i in $source APKBUILD; do
1049 1050 1051 1052 1053 1054
		local s
		if is_remote "$i"; then
			s="$SRCDEST/${i##*/}"	# $(basename $i)
		else
			s="$startdir/${i##*/}"
		fi
1055 1056 1057 1058 1059 1060 1061
		if [ "$s" -nt "$pkg" ]; then
			return 1
		fi
	done
	return 0
}

1062
abuildindex_up2date() {
1063
	local i
1064
	getpkgver || return 1
1065 1066 1067 1068 1069 1070 1071
	local dir="$abuildrepo"/$CARCH
	local apk="${pkgname%:*}-$pkgver-r$pkgrel.apk"
	local idx="$dir"/APKINDEX.tar.gz
	local file="$dir"/$apk

	# check if index is missing
	[ -f "$idx" ] || return 1
1072
			
1073 1074 1075 1076 1077
	# if link or file is missing, then we need update abuildrepo index
	[ -f "$file" ] || return 1

	# if file exists and is newer than index, then we need update index
	[ "$file" -nt "$idx" ] && return 1
1078
		
1079 1080 1081 1082
	return 0
}

up2date() {
Natanael Copa's avatar
Natanael Copa committed
1083
	check_arch || return 0
1084 1085 1086 1087 1088 1089 1090
	apk_up2date && abuildindex_up2date
}

# rebuild package and abuildrepo index if needed
abuildindex() {
	up2date && return 0
	update_abuildrepo
1091 1092
}

1093 1094 1095 1096 1097 1098 1099
# source all APKBUILDs and output:
#  1) origin of package
#  2) all dependencies
# the output is i in a format easy parseable for awk
depparse_aports() {
	# lets run this in a subshell since we source all APKBUILD here
	(
1100 1101
	aportsdir=$(realpath ${APKBUILD%/APKBUILD}/..)
	for i in $aportsdir/*/APKBUILD; do
1102 1103 1104 1105
		pkgname=
		subpackages=
		depends=
		makedepends=
1106
		. $i
1107
		dir=${i%/APKBUILD}
1108
		deps=
1109
		# filter out conflicts from deps and version info
1110 1111 1112 1113
		for j in $depends $makedepends; do
			case "$j" in
				!*) continue;;
			esac
1114
			deps="$deps ${j%%[<>=]*}"
1115
		done
1116
		for j in $pkgname $subpackages; do
1117
			echo "o ${j%%:*} $dir"
1118
			set -- $deps
1119 1120 1121 1122 1123 1124 1125
			echo -n "d ${j%%:*} $1"
			shift
			while [ $# -gt 0 ]; do
				echo -n ",$1"
				shift
			done
			echo
1126 1127
		done
	done
1128
	)
1129 1130
}

1131
deptrace() {
1132 1133 1134
	local deps= i=
	# strip versions from deps
	for i in "$@"; do
1135
		deps="$deps ${i%%[<>=]*}"
1136
	done