epm-sh-functions 10.1 KB
Newer Older
Vitaly Lipatov's avatar
Vitaly Lipatov committed
1 2
#!/bin/sh
#
3 4
# Copyright (C) 2012, 2014  Etersoft
# Copyright (C) 2012, 2014  Vitaly Lipatov <lav@etersoft.ru>
Vitaly Lipatov's avatar
Vitaly Lipatov committed
5
#
6 7 8
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
Vitaly Lipatov's avatar
Vitaly Lipatov committed
9 10 11 12 13
# (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
14
# GNU Affero General Public License for more details.
Vitaly Lipatov's avatar
Vitaly Lipatov committed
15
#
16 17
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
Vitaly Lipatov's avatar
Vitaly Lipatov committed
18 19
#

20
# copied from /etc/init.d/outformat (ALT Linux)
21

22
# FIXME on Android: FIX ME! implement ttyname_r() bionic/libc/bionic/stubs.c:366
23 24 25
inputisatty()
{
	# check stdin
Vitaly Lipatov's avatar
Vitaly Lipatov committed
26
	tty -s 2>/dev/null
27 28
}

29 30
isatty()
{
31 32 33 34 35 36
	# check stdout
	test -t 1
}

isatty2()
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
37 38
	# check stderr
	test -t 2
39 40
}

41 42
check_tty()
{
43 44 45 46 47 48
	isatty2 || return

	# Set a sane TERM required for tput
	[ -n "$TERM" ] || TERM=dumb
	export TERM

49
	# egrep from busybox may not --color
Vitaly Lipatov's avatar
Vitaly Lipatov committed
50
	# egrep from MacOS print help to stderr
Vitaly Lipatov's avatar
Vitaly Lipatov committed
51
	if grep -E --help 2>&1 | grep -q -- "--color" ; then
52
		export EGREPCOLOR="--color"
53 54
	fi

55 56 57 58
	which tput >/dev/null 2>/dev/null || return
	# FreeBSD does not support tput -S
	echo | tput -S >/dev/null 2>/dev/null || return
	[ -z "$USETTY" ] || return
59 60 61
	export USETTY=1
}

62 63 64 65
: ${BLACK:=0} ${RED:=1} ${GREEN:=2} ${YELLOW:=3} ${BLUE:=4} ${MAGENTA:=5} ${CYAN:=6} ${WHITE:=7}

set_boldcolor()
{
66
	[ "$USETTY" = "1" ] || return
67 68 69 70 71 72 73 74
	{
		echo bold
		echo setaf $1
	} |tput -S
}

restore_color()
{
75
	[ "$USETTY" = "1" ] || return
76 77 78 79 80 81
	{
		echo op; # set Original color Pair.
		echo sgr0; # turn off all special graphics mode (bold in our case).
	} |tput -S
}

Vitaly Lipatov's avatar
Vitaly Lipatov committed
82 83
echover()
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
84
    [ -z "$verbose" ] && return
85
    echo "$*" >&2
Vitaly Lipatov's avatar
Vitaly Lipatov committed
86 87
}

88 89 90
# echo string without EOL
echon()
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
91
	# default /bin/sh on MacOS does not recognize -n
Vitaly Lipatov's avatar
Vitaly Lipatov committed
92
	/bin/echo -n "$*"
93 94 95
}


Vitaly Lipatov's avatar
Vitaly Lipatov committed
96 97 98
# Used DISTRNAME
set_target_pkg_env()
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
99
	[ -n "$DISTRNAME" ] || fatal "Missing DISTRNAME in set_target_pkg_env."
Vitaly Lipatov's avatar
Vitaly Lipatov committed
100 101 102 103 104 105
	PKGFORMAT=$($DISTRVENDOR -p "$DISTRNAME")
	PKGVENDOR=$($DISTRVENDOR -s "$DISTRNAME")
	RPMVENDOR=$($DISTRVENDOR -n "$DISTRNAME")
}

# Print command line and run command line
106
showcmd()
Vitaly Lipatov's avatar
Vitaly Lipatov committed
107
{
108 109 110
	if [ -z "$quiet" ] ; then
		set_boldcolor $GREEN
		local PROMTSIG="\$"
111
		[ "$EFFUID" = 0 ] && PROMTSIG="#"
Vitaly Lipatov's avatar
Vitaly Lipatov committed
112
		echo " $PROMTSIG $*"
113
		restore_color
114
	fi >&2
115 116 117 118 119
}

# Print command line and run command line
docmd()
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
120
	showcmd "$*$EXTRA_SHOWDOCMD"
121 122
#FIXME
	$@
Vitaly Lipatov's avatar
Vitaly Lipatov committed
123 124
}

125
# Run every arg with docmd
126 127
docmd_foreach()
{
128
	local cmd pkg
129 130 131 132
	cmd="$1"
	#showcmd "$@"
	shift
	for pkg in "$@" ; do
133
		docmd "$cmd" $pkg
134 135 136
	done
}

137
# Print command line and run command line with SUDO
138
sudocmd()
139
{
140
	[ -n "$SUDO" ] && showcmd "$SUDO $*" || showcmd "$*"
141
	$SUDO $@
142 143
}

144
# Run every arg with sudocmd
145
# Returns on any error
146 147
sudocmd_foreach()
{
148
	local cmd pkg
149 150 151 152
	cmd="$1"
	#showcmd "$@"
	shift
	for pkg in "$@" ; do
153
		sudocmd "$cmd" $pkg || return
154 155 156
	done
}

157 158 159 160 161 162 163 164
# add realpath if missed
if ! which realpath 2>/dev/null >/dev/null ; then
realpath()
{
	readlink -f "$@"
}
fi

165 166
get_firstarg()
{
167
	echon "$1"
168 169 170 171 172
}

get_lastarg()
{
	local lastarg
Vitaly Lipatov's avatar
Vitaly Lipatov committed
173
	eval "lastarg=\${$#}"
174
	echon "$lastarg"
175 176
}

177 178 179 180 181 182
# TODO: see etersoft-build-utils/tests/test_isnumber.sh
isnumber()
{
	echo "$*" | filter_strip_spaces | grep -q "^[0-9]\+$"
}

Vitaly Lipatov's avatar
Vitaly Lipatov committed
183 184 185 186 187 188 189 190 191 192 193 194 195
filter_strip_spaces()
{
        # possible use just
        #xargs echo
        sed -e "s| \+| |g" | \
                sed -e "s|^ ||" | sed -e "s| \$||"
}

strip_spaces()
{
        echo "$*" | filter_strip_spaces
}

196 197 198 199 200 201
# param true false
subst_option()
{
	eval "[ -n \"\$$1\" ]" && echo "$2" || echo "$3"
}

202 203 204 205
store_output()
{
    # use make_temp_file from etersoft-build-utils
    RC_STDOUT=$(mktemp)
206 207
    local CMDSTATUS=$RC_STDOUT.pipestatus
    echo 1 >$CMDSTATUS
208
    #RC_STDERR=$(mktemp)
209
    ( $@ 2>&1 ; echo $? >$CMDSTATUS ) | tee $RC_STDOUT
Vitaly Lipatov's avatar
Vitaly Lipatov committed
210
    return "$(cat $CMDSTATUS)"
211
    # bashism
212
    # http://tldp.org/LDP/abs/html/bashver3.html#PIPEFAILREF
213
    #return $PIPESTATUS
214 215
}

216 217 218 219 220 221
showcmd_store_output()
{
    showcmd "$@"
    store_output "$@"
}

222 223
clean_store_output()
{
224
    rm -f $RC_STDOUT $RC_STDOUT.pipestatus
225 226
}

227
# run epm, possible from side repo
228 229
epm()
{
230 231
	[ -n "$PROGNAME" ] || fatal "Can't use epm call from the piped script"
	$PROGDIR/$PROGNAME $@
232
}
Vitaly Lipatov's avatar
Vitaly Lipatov committed
233 234 235 236 237

# Print error message and stop the program
fatal()
{
	if [ -z "$TEXTDOMAIN" ] ; then
Vitaly Lipatov's avatar
Vitaly Lipatov committed
238
		echo "Error: $*" >&2
Vitaly Lipatov's avatar
Vitaly Lipatov committed
239 240
#	else
#		echog "Error in $0: $@" >&2
Vitaly Lipatov's avatar
Vitaly Lipatov committed
241 242 243
	fi
	exit 1
}
244

Vitaly Lipatov's avatar
Vitaly Lipatov committed
245 246 247 248
# Print warning message
warning()
{
	if [ -z "$TEXTDOMAIN" ] ; then
Vitaly Lipatov's avatar
Vitaly Lipatov committed
249
		echo "Warning: $*" >&2
Vitaly Lipatov's avatar
Vitaly Lipatov committed
250 251 252 253 254
#	else
#		echog "Error in $0: $@" >&2
	fi
}

255 256 257 258
info()
{
	[ -n "$quiet" ] && return

259 260 261
	# print message to stderr if stderr forwarded to (a file)
	if isatty2 ; then
		isatty || return 0
Vitaly Lipatov's avatar
Vitaly Lipatov committed
262
		echo "$*"
263
	else
Vitaly Lipatov's avatar
Vitaly Lipatov committed
264
		echo "$*" >&2
265
	fi
266 267
}

268 269
set_sudo()
{
270 271 272
	SUDO=""
	# skip SUDO if disabled
	[ -n "$EPMNOSUDO" ] && return
273 274 275 276
	if [ "$DISTRNAME" = "Cygwin" ] || [ "$DISTRNAME" = "Windows" ] ; then
		# skip sudo using on Windows
		return
	fi
277

Vitaly Lipatov's avatar
Vitaly Lipatov committed
278
	EFFUID=$(id -u)
279 280

	# do not need sudo
281
	[ $EFFUID = "0" ] && return
282 283

	# use sudo if possible
284 285
	if which sudo >/dev/null 2>/dev/null ; then
		SUDO="sudo --"
286
		# check for < 1.7 version which do not support -- (and --help possible too)
287
		sudo -h 2>/dev/null | grep -q "  --" || SUDO="sudo"
288 289
		return
	fi
290 291

	SUDO="fatal 'Can't find sudo. Please install sudo or run epm under root.'"
292 293
}

294 295
# wait for n seconds (if possible) during executing command
# args: seconds command
296 297 298
withtimeout()
{
	local TO=$(which timeout 2>/dev/null || which gtimeout 2>/dev/null)
Vitaly Lipatov's avatar
Vitaly Lipatov committed
299
	if [ -x "$TO" ] ; then
300
		$TO "$@"
Vitaly Lipatov's avatar
Vitaly Lipatov committed
301 302 303
		return
	fi
	# fallback: drop time arg and run without timeout
304
	shift
305
	"$@"
306 307
}

308 309 310 311 312 313
set_eatmydata()
{
	# skip if disabled
	[ -n "$EPMNOEATMYDATA" ] && return
	# use if possible
	which eatmydata >/dev/null 2>/dev/null || return
314
	[ -n "$SUDO" ] && SUDO="$SUDO eatmydata" || SUDO="eatmydata"
315
	[ -n "$verbose" ] && info "Uwaga! eatmydata is installed, we will use it for disable all sync operations."
Vitaly Lipatov's avatar
Vitaly Lipatov committed
316
	return 0
317 318
}

319 320
# 
__get_package_for_command()
321
{
322
	case "$1" in
323
		equery|revdep-rebuild)
324
			echo 'gentoolkit'
325
			;;
326
		update-kernel|remove-old-kernels)
327
			echo 'update-kernel'
328 329 330 331
			;;
	esac
}

332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358
# TODO:
confirm() {
    local response
    # call with a prompt string or use a default
    read -r -p "${1:-Are you sure? [y/N]} " response
    case $response in
        [yY][eE][sS]|[yY])
            true
            ;;
        *)
            false
            ;;
    esac
}

assure_root()
{
	[ "$EFFUID" = 0 ] || fatal "run me only under root"
}

regexp_subst()
{
	local expression="$1"
	shift
	sed -i -r -e "$expression" "$@"
}

Vitaly Lipatov's avatar
Vitaly Lipatov committed
359
# TODO: we we can't use epm directly?
360 361 362
assure_exists()
{
	load_helper epm-assure
Vitaly Lipatov's avatar
Vitaly Lipatov committed
363
	local package="$2"
364
	local textpackage=
Vitaly Lipatov's avatar
Vitaly Lipatov committed
365
	[ -n "$package" ] || package="$(__get_package_for_command "$1")"
366 367
	[ -n "$3" ] && textpackage=" >= $3"
	__epm_assure "$1" $package $3 || fatal "Can't assure in '$1' command from $package$textpackage package"
368 369
}

370
# will replaced within disabled_eget in packaged version
Vitaly Lipatov's avatar
Vitaly Lipatov committed
371
eget()
372 373 374 375 376 377 378 379 380 381 382 383 384 385 386
{
	# use internal eget only if exists
	if [ -s $SHAREDIR/tools_eget ] ; then
		$SHAREDIR/tools_eget "$@"
		return
	fi

	assure_exists eget
	# run external command, not the function
	EGET=$(which eget) || fatal "Missed command eget from installed package eget"
	$EGET "$@"
}

# will replaced within eget() in packed version
onefile_eget()
Vitaly Lipatov's avatar
Vitaly Lipatov committed
387
{
Vitaly Lipatov's avatar
Vitaly Lipatov committed
388
	assure_exists wget
389
	internal_tools_eget "$@"
Vitaly Lipatov's avatar
Vitaly Lipatov committed
390 391
}

392
# TODO: improve and drop!
393 394 395
get_package_type()
{
	local i
396 397 398 399 400 401 402 403 404
	case $1 in
		*.deb)
			echo "deb"
			return
			;;
		*.rpm)
			echo "rpm"
			return
			;;
405 406 407 408 409 410 411 412
		*.txz)
			echo "txz"
			return
			;;
		*.tbz)
			echo "tbz"
			return
			;;
413 414 415 416
		*.exe)
			echo "exe"
			return
			;;
417 418 419 420
		*.msi)
			echo "msi"
			return
			;;
421 422 423 424 425 426 427
		*)
			#fatal "Don't know type of $1"
			# return package name for info
			echo "$1"
			return 1
			;;
	esac
428 429 430
}


431 432 433
# print options description from HELPCMD/HELPOPT lines in the code
get_help()
{
434 435 436 437
    if [ "$0" = "/dev/stdin" ] || [ "$0" = "sh" ] ; then
        return
    fi

Vitaly Lipatov's avatar
Vitaly Lipatov committed
438
    grep -v -- "^#" $0 | grep -- "# $1" | while read -r n ; do
439 440 441 442 443 444
        opt=$(echo $n | sed -e "s|) # $1:.*||g")
        desc=$(echo $n | sed -e "s|.*) # $1:||g")
        printf "    %-20s %s\n" $opt "$desc"
    done
}

445

446 447 448 449
# FIXME: detect if not recognized
set_pm_type()
{
	local CMD
450 451 452

	# Fill for use: PMTYPE, DISTRNAME, DISTRVERSION, PKGFORMAT, PKGVENDOR, RPMVENDOR
	DISTRVENDOR=$PROGDIR/distr_info
453
	[ -n "$DISTRNAME" ] || DISTRNAME=$($DISTRVENDOR -d) || fatal "Can't get distro name."
454 455 456
	[ -n "$DISTRVERSION" ] || DISTRVERSION=$($DISTRVENDOR -v)
	set_target_pkg_env

457 458 459 460 461 462
# override package manager detection result
if [ -n "$FORCEPM" ] ; then
	PMTYPE=$FORCEPM
	return
fi

463 464
# TODO: move it in distr_vendor?
# FIXME: some problems with multibased distros (Server Edition on CentOS and Desktop Edition on Ubuntu)
465
case $DISTRNAME in
466
	ALTLinux)
467
		CMD="apt-rpm"
468
		#which ds-install 2>/dev/null >/dev/null && CMD=deepsolver-rpm
469 470
		;;
	PCLinux)
471 472
		CMD="apt-rpm"
		;;
473
	Ubuntu|Debian|Mint|AstraLinux|Elbrus)
474
		CMD="apt-dpkg"
Vitaly Lipatov's avatar
Vitaly Lipatov committed
475
		#which aptitude 2>/dev/null >/dev/null && CMD=aptitude-dpkg
476
		which snappy 2>/dev/null >/dev/null && CMD=snappy
477
		;;
Vitaly Lipatov's avatar
Vitaly Lipatov committed
478
	Mandriva|ROSA)
479 480
		CMD="urpm-rpm"
		;;
481 482
	FreeBSD|NetBSD|OpenBSD|Solaris)
		CMD="pkgsrc"
483
		which pkg 2>/dev/null >/dev/null && CMD=pkgng
484
		;;
485 486 487
	Gentoo)
		CMD="emerge"
		;;
488 489 490
	ArchLinux)
		CMD="pacman"
		;;
491
	Fedora|LinuxXP|ASPLinux|CentOS|RHEL|Scientific|GosLinux)
492
		CMD="yum-rpm"
493
		which dnf 2>/dev/null >/dev/null && test -d /var/lib/dnf/yumdb && CMD=dnf-rpm
494
		;;
495
	Slackware)
496
		CMD="slackpkg"
497
		;;
498
	SUSE|SLED|SLES|Tumbleweed)
499 500
		CMD="zypper-rpm"
		;;
501 502 503
	ForesightLinux|rPathLinux)
		CMD="conary"
		;;
504
	Windows)
505
		CMD="chocolatey"
506
		;;
507 508 509 510 511 512
	MacOS)
		CMD="homebrew"
		;;
	OpenWRT)
		CMD="ipkg"
		;;
513 514 515
	GNU/Linux/Guix)
		CMD="guix"
		;;
516 517 518
	Android)
		CMD="android"
		;;
519 520 521
	Cygwin)
		CMD="aptcyg"
		;;
522 523 524
	alpine)
		CMD="apk"
		;;
525 526 527
	TinyCoreLinux)
		CMD="tce"
		;;
528 529 530
	VoidLinux)
		CMD="xbps"
		;;
531
	*)
532
		fatal "Have no suitable DISTRNAME $DISTRNAME"
533 534 535 536
		;;
esac
PMTYPE=$CMD
}
537

538 539 540 541 542 543 544 545

is_active_systemd()
{
	local a
	SYSTEMCTL=/bin/systemctl
	SYSTEMD_CGROUP_DIR=/sys/fs/cgroup/systemd
	[ -x "$SYSTEMCTL" ] || return
	[ -d "$SYSTEMD_CGROUP_DIR" ] || return
Vitaly Lipatov's avatar
Vitaly Lipatov committed
546
	a='' mountpoint -q "$SYSTEMD_CGROUP_DIR" || return
547
	readlink /sbin/init | grep -q 'systemd' || return
548
	# some hack
Vitaly Lipatov's avatar
Vitaly Lipatov committed
549
	# shellcheck disable=SC2009
550
	ps ax | grep '[s]ystemd' | grep -q -v 'systemd-udev'
551
}