zapret/common/base.sh

360 lines
6.0 KiB
Bash
Raw Normal View History

which()
{
# on some systems 'which' command is considered deprecated and not installed by default
# 'command -v' replacement does not work exactly the same way. it outputs shell aliases if present
# $1 - executable name
local IFS=:
for p in $PATH; do
[ -x "$p/$1" ] && {
echo "$p/$1"
return 0
}
done
return 1
}
2022-02-15 19:15:36 +05:00
exists()
{
which "$1" >/dev/null 2>/dev/null
}
existf()
{
type "$1" >/dev/null 2>/dev/null
}
whichq()
{
which $1 2>/dev/null
}
exist_all()
{
while [ -n "$1" ]; do
exists "$1" || return 1
shift
done
return 0
}
on_off_function()
{
# $1 : function name on
# $2 : function name off
# $3 : 0 - off, 1 - on
local F="$1"
[ "$3" = "1" ] || F="$2"
shift
shift
shift
"$F" "$@"
}
contains()
{
# check if substring $2 contains in $1
[ "${1#*$2}" != "$1" ]
}
2024-05-11 18:11:06 +05:00
starts_with()
{
# $1 : what
# $2 : starts with
case "$1" in
"$2"*)
return 0
;;
esac
return 1
}
2022-02-15 19:15:36 +05:00
find_str_in_list()
{
[ -n "$1" ] && {
for v in $2; do
[ "$v" = "$1" ] && return 0
done
}
return 1
}
end_with_newline()
{
2024-05-04 18:01:09 +05:00
local c="$(tail -c 1)"
2022-02-15 19:15:36 +05:00
[ "$c" = "" ]
}
append_separator_list()
2022-02-15 19:15:36 +05:00
{
# $1 - var name to receive result
# $2 - separator
# $3 - quoter
# $4,$5,... - elements
local _var="$1" sep="$2" quo="$3" i
2022-02-15 19:15:36 +05:00
eval i="\$$_var"
shift; shift; shift
2022-02-15 19:15:36 +05:00
while [ -n "$1" ]; do
if [ -n "$i" ] ; then
i="$i$sep$quo$1$quo"
2022-02-15 19:15:36 +05:00
else
i="$quo$1$quo"
2022-02-15 19:15:36 +05:00
fi
shift
done
eval $_var="\$i"
}
make_separator_list()
{
eval $1=''
append_separator_list "$@"
2022-02-15 19:15:36 +05:00
}
make_comma_list()
{
# $1 - var name to receive result
# $2,$3,... - elements
local var="$1"
shift
make_separator_list $var , '' "$@"
2022-02-15 19:15:36 +05:00
}
make_quoted_comma_list()
{
# $1 - var name to receive result
# $2,$3,... - elements
local var="$1"
shift
make_separator_list $var , '"' "$@"
}
2022-02-16 01:11:43 +05:00
unique()
{
local i
for i in "$@"; do echo $i; done | sort -u | xargs
}
2022-02-15 19:15:36 +05:00
is_linked_to_busybox()
{
local IFS F P
IFS=:
for path in $PATH; do
F=$path/$1
P="$(readlink $F)"
if [ -z "$P" ] && [ -x $F ] && [ ! -L $F ]; then return 1; fi
[ "${P%busybox*}" != "$P" ] && return
done
}
get_dir_inode()
{
local dir="$1"
[ -L "$dir" ] && dir=$(readlink "$dir")
ls -id "$dir" | awk '{print $1}'
}
linux_min_version()
{
# $1 - major ver
# $2 - minor ver
local V1=$(sed -nre 's/^Linux version ([0-9]+)\.[0-9]+.*$/\1/p' /proc/version)
local V2=$(sed -nre 's/^Linux version [0-9]+\.([0-9]+).*$/\1/p' /proc/version)
[ -n "$V1" -a -n "$V2" ] && [ "$V1" -gt "$1" -o "$V1" -eq "$1" -a "$V2" -ge "$2" ]
}
linux_get_subsys()
{
2024-05-04 18:01:09 +05:00
local INIT="$(sed 's/\x0/\n/g' /proc/1/cmdline | head -n 1)"
2022-02-15 19:15:36 +05:00
[ -L "$INIT" ] && INIT=$(readlink "$INIT")
2024-08-25 20:04:11 +05:00
INIT="$(basename "$INIT")"
2022-02-15 19:15:36 +05:00
if [ -f "/etc/openwrt_release" ] && [ "$INIT" = "procd" ] ; then
SUBSYS=openwrt
2024-03-22 20:26:00 +05:00
elif [ -x "/bin/ndm" ] ; then
SUBSYS=keenetic
2022-02-15 19:15:36 +05:00
else
# generic linux
SUBSYS=
fi
}
openwrt_fw3()
{
[ ! -x /sbin/fw4 -a -x /sbin/fw3 ]
}
openwrt_fw4()
{
[ -x /sbin/fw4 ]
}
openwrt_fw3_integration()
{
[ "$FWTYPE" = iptables ] && openwrt_fw3
}
create_dev_stdin()
{
[ -e /dev/stdin ] || ln -s /proc/self/fd/0 /dev/stdin
}
2022-02-16 01:11:43 +05:00
call_for_multiple_items()
{
# $1 - function to get an item
# $2 - variable name to put result into
# $3 - space separated parameters to function $1
local i item items
for i in $3; do
$1 item $i
[ -n "$item" ] && {
if [ -n "$items" ]; then
items="$items $item"
else
items="$item"
fi
}
done
eval $2=\"$items\"
}
fix_sbin_path()
{
local IFS=':'
printf "%s\n" $PATH | grep -Fxq '/usr/sbin' || PATH="/usr/sbin:$PATH"
printf "%s\n" $PATH | grep -Fxq '/sbin' || PATH="/sbin:$PATH"
export PATH
}
2023-10-31 16:19:41 +05:00
2023-11-03 12:08:21 +05:00
# it can calculate floating point expr
calc()
{
awk "BEGIN { print $*}";
}
2023-10-31 16:19:41 +05:00
fsleep_setup()
{
[ -n "$FSLEEP" ] || {
2023-10-31 18:42:38 +05:00
if sleep 0.001 2>/dev/null; then
2023-10-31 16:19:41 +05:00
FSLEEP=1
elif busybox usleep 1 2>/dev/null; then
FSLEEP=2
else
2024-05-04 18:01:09 +05:00
local errtext="$(read -t 0.001 2>&1)"
2023-10-31 16:19:41 +05:00
if [ -z "$errtext" ]; then
FSLEEP=3
2023-10-31 18:19:49 +05:00
# newer openwrt has ucode with system function that supports timeout in ms
elif ucode -e "system(['sleep','1'], 1)" 2>/dev/null; then
2023-10-31 18:19:49 +05:00
FSLEEP=4
# older openwrt may have lua and nixio lua module
2023-11-03 11:47:15 +05:00
elif lua -e 'require "nixio".nanosleep(0,1)' 2>/dev/null ; then
FSLEEP=5
2023-10-31 16:19:41 +05:00
else
2023-11-02 23:40:45 +05:00
FSLEEP=0
2023-10-31 16:19:41 +05:00
fi
fi
}
}
2023-11-03 12:08:21 +05:00
msleep()
2023-10-31 16:19:41 +05:00
{
2023-11-03 12:08:21 +05:00
# $1 - milliseconds
2023-10-31 16:19:41 +05:00
case "$FSLEEP" in
1)
2023-11-03 12:08:21 +05:00
sleep $(calc $1/1000)
2023-10-31 16:19:41 +05:00
;;
2)
2023-11-03 12:08:21 +05:00
busybox usleep $(calc $1*1000)
2023-10-31 16:19:41 +05:00
;;
3)
2023-11-03 12:08:21 +05:00
read -t $(calc $1/1000)
2023-10-31 16:19:41 +05:00
;;
2023-10-31 18:19:49 +05:00
4)
2023-11-03 12:08:21 +05:00
ucode -e "system(['sleep','2147483647'], $1)"
2023-10-31 18:19:49 +05:00
;;
5)
2023-11-03 12:08:21 +05:00
lua -e "require 'nixio'.nanosleep($(($1/1000)),$(calc $1%1000*1000000))"
;;
2023-10-31 16:19:41 +05:00
*)
2023-11-03 12:45:18 +05:00
sleep $((($1+999)/1000))
2023-10-31 16:19:41 +05:00
esac
}
2023-11-03 12:08:21 +05:00
minsleep()
{
msleep 100
}
2023-12-12 23:00:22 +05:00
replace_char()
{
local a=$1
local b=$2
shift; shift
echo "$@" | tr $a $b
}
2024-06-13 17:42:12 +05:00
setup_md5()
{
2024-06-13 19:55:33 +05:00
[ -n "$MD5" ] && return
2024-06-13 17:42:12 +05:00
MD5=md5sum
exists $MD5 || MD5=md5
}
2024-01-23 15:17:51 +05:00
random()
{
# $1 - min, $2 - max
local r rs
2024-06-13 19:55:33 +05:00
setup_md5
2024-01-23 15:17:51 +05:00
if [ -c /dev/urandom ]; then
read rs </dev/urandom
else
rs="$RANDOM$RANDOM$(date)"
fi
# shells use signed int64
2024-06-20 13:54:03 +05:00
r=1$(echo $rs | $MD5 | sed 's/[^0-9]//g' | cut -c 1-17)
2024-01-23 15:17:51 +05:00
echo $(( ($r % ($2-$1+1)) + $1 ))
}
2024-05-07 22:19:43 +05:00
shell_name()
{
[ -n "$SHELL_NAME" ] || {
[ -n "$UNAME" ] || UNAME="$(uname)"
if [ "$UNAME" = "Linux" ]; then
SHELL_NAME="$(readlink /proc/$$/exe)"
SHELL_NAME="$(basename "$SHELL_NAME")"
else
SHELL_NAME=$(ps -p $$ -o comm=)
fi
2024-05-11 18:11:06 +05:00
[ -n "$SHELL_NAME" ] || SHELL_NAME="$(basename "$SHELL")"
2024-05-07 22:19:43 +05:00
}
}
process_exists()
{
if exists pgrep; then
pgrep ^$1$ >/dev/null
elif exists pidof; then
pidof $1 >/dev/null
else
return 1
fi
}
win_process_exists()
{
tasklist /NH /FI "IMAGENAME eq ${1}.exe" | grep -q "^${1}.exe"
}
alloc_num()
{
# $1 - source var name
# $2 - target var name
# $3 - min
# $4 - max
local v
eval v="\$$2"
# do not replace existing value
[ -n "$v" ] && return
eval v="\$$1"
[ -n "$v" ] || v=$3
eval $2="$v"
v=$((v + 1))
[ $v -gt $4 ] && v=$3
eval $1="$v"
}
2023-12-12 23:00:22 +05:00
std_ports()
{
HTTP_PORTS=${HTTP_PORTS:-80}
HTTPS_PORTS=${HTTPS_PORTS:-443}
QUIC_PORTS=${QUIC_PORTS:-443}
HTTP_PORTS_IPT=$(replace_char - : $HTTP_PORTS)
HTTPS_PORTS_IPT=$(replace_char - : $HTTPS_PORTS)
QUIC_PORTS_IPT=$(replace_char - : $QUIC_PORTS)
}