2023-08-08 10:47:01 +03:00
|
|
|
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 17:15:36 +03: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 16:11:06 +03:00
|
|
|
starts_with()
|
|
|
|
{
|
|
|
|
# $1 : what
|
|
|
|
# $2 : starts with
|
|
|
|
case "$1" in
|
|
|
|
"$2"*)
|
|
|
|
return 0
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
return 1
|
|
|
|
}
|
2022-02-15 17:15:36 +03: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 16:01:09 +03:00
|
|
|
local c="$(tail -c 1)"
|
2022-02-15 17:15:36 +03:00
|
|
|
[ "$c" = "" ]
|
|
|
|
}
|
2022-02-16 21:48:02 +03:00
|
|
|
|
|
|
|
append_separator_list()
|
2022-02-15 17:15:36 +03:00
|
|
|
{
|
|
|
|
# $1 - var name to receive result
|
|
|
|
# $2 - separator
|
2022-02-16 21:48:02 +03:00
|
|
|
# $3 - quoter
|
|
|
|
# $4,$5,... - elements
|
|
|
|
local _var="$1" sep="$2" quo="$3" i
|
2022-02-15 17:15:36 +03:00
|
|
|
|
2022-02-16 21:48:02 +03:00
|
|
|
eval i="\$$_var"
|
|
|
|
shift; shift; shift
|
2022-02-15 17:15:36 +03:00
|
|
|
while [ -n "$1" ]; do
|
|
|
|
if [ -n "$i" ] ; then
|
2022-02-16 21:48:02 +03:00
|
|
|
i="$i$sep$quo$1$quo"
|
2022-02-15 17:15:36 +03:00
|
|
|
else
|
2022-02-16 21:48:02 +03:00
|
|
|
i="$quo$1$quo"
|
2022-02-15 17:15:36 +03:00
|
|
|
fi
|
|
|
|
shift
|
|
|
|
done
|
2022-02-16 21:48:02 +03:00
|
|
|
eval $_var="\$i"
|
|
|
|
}
|
|
|
|
make_separator_list()
|
|
|
|
{
|
|
|
|
eval $1=''
|
|
|
|
append_separator_list "$@"
|
2022-02-15 17:15:36 +03:00
|
|
|
}
|
|
|
|
make_comma_list()
|
|
|
|
{
|
|
|
|
# $1 - var name to receive result
|
|
|
|
# $2,$3,... - elements
|
|
|
|
local var="$1"
|
|
|
|
shift
|
2022-02-16 21:48:02 +03:00
|
|
|
make_separator_list $var , '' "$@"
|
2022-02-15 17:15:36 +03:00
|
|
|
}
|
2022-02-16 22:08:01 +03:00
|
|
|
make_quoted_comma_list()
|
|
|
|
{
|
|
|
|
# $1 - var name to receive result
|
|
|
|
# $2,$3,... - elements
|
|
|
|
local var="$1"
|
|
|
|
shift
|
|
|
|
make_separator_list $var , '"' "$@"
|
|
|
|
}
|
2022-02-15 23:11:43 +03:00
|
|
|
unique()
|
|
|
|
{
|
|
|
|
local i
|
|
|
|
for i in "$@"; do echo $i; done | sort -u | xargs
|
|
|
|
}
|
2022-02-15 17:15:36 +03: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 16:01:09 +03:00
|
|
|
local INIT="$(sed 's/\x0/\n/g' /proc/1/cmdline | head -n 1)"
|
2022-02-15 17:15:36 +03:00
|
|
|
|
|
|
|
[ -L "$INIT" ] && INIT=$(readlink "$INIT")
|
2024-08-25 18:04:11 +03:00
|
|
|
INIT="$(basename "$INIT")"
|
2022-02-15 17:15:36 +03:00
|
|
|
if [ -f "/etc/openwrt_release" ] && [ "$INIT" = "procd" ] ; then
|
|
|
|
SUBSYS=openwrt
|
2024-03-22 18:26:00 +03:00
|
|
|
elif [ -x "/bin/ndm" ] ; then
|
|
|
|
SUBSYS=keenetic
|
2022-02-15 17:15:36 +03: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-15 23:11:43 +03: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\"
|
|
|
|
}
|
2022-03-27 18:58:47 +03:00
|
|
|
|
|
|
|
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 14:19:41 +03:00
|
|
|
|
2023-11-03 10:08:21 +03:00
|
|
|
# it can calculate floating point expr
|
|
|
|
calc()
|
|
|
|
{
|
|
|
|
awk "BEGIN { print $*}";
|
|
|
|
}
|
|
|
|
|
2023-10-31 14:19:41 +03:00
|
|
|
fsleep_setup()
|
|
|
|
{
|
|
|
|
[ -n "$FSLEEP" ] || {
|
2023-10-31 16:42:38 +03:00
|
|
|
if sleep 0.001 2>/dev/null; then
|
2023-10-31 14:19:41 +03:00
|
|
|
FSLEEP=1
|
|
|
|
elif busybox usleep 1 2>/dev/null; then
|
|
|
|
FSLEEP=2
|
|
|
|
else
|
2024-05-04 16:01:09 +03:00
|
|
|
local errtext="$(read -t 0.001 2>&1)"
|
2023-10-31 14:19:41 +03:00
|
|
|
if [ -z "$errtext" ]; then
|
|
|
|
FSLEEP=3
|
2023-10-31 16:19:49 +03:00
|
|
|
# newer openwrt has ucode with system function that supports timeout in ms
|
2023-11-02 21:29:26 +03:00
|
|
|
elif ucode -e "system(['sleep','1'], 1)" 2>/dev/null; then
|
2023-10-31 16:19:49 +03:00
|
|
|
FSLEEP=4
|
2023-11-02 21:29:26 +03:00
|
|
|
# older openwrt may have lua and nixio lua module
|
2023-11-03 09:47:15 +03:00
|
|
|
elif lua -e 'require "nixio".nanosleep(0,1)' 2>/dev/null ; then
|
2023-11-02 21:29:26 +03:00
|
|
|
FSLEEP=5
|
2023-10-31 14:19:41 +03:00
|
|
|
else
|
2023-11-02 21:40:45 +03:00
|
|
|
FSLEEP=0
|
2023-10-31 14:19:41 +03:00
|
|
|
fi
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
}
|
2023-11-03 10:08:21 +03:00
|
|
|
msleep()
|
2023-10-31 14:19:41 +03:00
|
|
|
{
|
2023-11-03 10:08:21 +03:00
|
|
|
# $1 - milliseconds
|
2023-10-31 14:19:41 +03:00
|
|
|
case "$FSLEEP" in
|
|
|
|
1)
|
2023-11-03 10:08:21 +03:00
|
|
|
sleep $(calc $1/1000)
|
2023-10-31 14:19:41 +03:00
|
|
|
;;
|
|
|
|
2)
|
2023-11-03 10:08:21 +03:00
|
|
|
busybox usleep $(calc $1*1000)
|
2023-10-31 14:19:41 +03:00
|
|
|
;;
|
|
|
|
3)
|
2023-11-03 10:08:21 +03:00
|
|
|
read -t $(calc $1/1000)
|
2023-10-31 14:19:41 +03:00
|
|
|
;;
|
2023-10-31 16:19:49 +03:00
|
|
|
4)
|
2023-11-03 10:08:21 +03:00
|
|
|
ucode -e "system(['sleep','2147483647'], $1)"
|
2023-10-31 16:19:49 +03:00
|
|
|
;;
|
2023-11-02 21:29:26 +03:00
|
|
|
5)
|
2023-11-03 10:08:21 +03:00
|
|
|
lua -e "require 'nixio'.nanosleep($(($1/1000)),$(calc $1%1000*1000000))"
|
2023-11-02 21:29:26 +03:00
|
|
|
;;
|
2023-10-31 14:19:41 +03:00
|
|
|
*)
|
2023-11-03 10:45:18 +03:00
|
|
|
sleep $((($1+999)/1000))
|
2023-10-31 14:19:41 +03:00
|
|
|
esac
|
|
|
|
}
|
2023-11-03 10:08:21 +03:00
|
|
|
minsleep()
|
|
|
|
{
|
|
|
|
msleep 100
|
|
|
|
}
|
2023-12-12 21:00:22 +03:00
|
|
|
|
|
|
|
replace_char()
|
|
|
|
{
|
|
|
|
local a=$1
|
|
|
|
local b=$2
|
|
|
|
shift; shift
|
|
|
|
echo "$@" | tr $a $b
|
|
|
|
}
|
|
|
|
|
2024-06-13 15:42:12 +03:00
|
|
|
setup_md5()
|
|
|
|
{
|
2024-06-13 17:55:33 +03:00
|
|
|
[ -n "$MD5" ] && return
|
2024-06-13 15:42:12 +03:00
|
|
|
MD5=md5sum
|
|
|
|
exists $MD5 || MD5=md5
|
|
|
|
}
|
|
|
|
|
2024-01-23 13:17:51 +03:00
|
|
|
random()
|
|
|
|
{
|
|
|
|
# $1 - min, $2 - max
|
|
|
|
local r rs
|
2024-06-13 17:55:33 +03:00
|
|
|
setup_md5
|
2024-01-23 13:17:51 +03:00
|
|
|
if [ -c /dev/urandom ]; then
|
|
|
|
read rs </dev/urandom
|
|
|
|
else
|
|
|
|
rs="$RANDOM$RANDOM$(date)"
|
|
|
|
fi
|
|
|
|
# shells use signed int64
|
2024-06-20 11:54:03 +03:00
|
|
|
r=1$(echo $rs | $MD5 | sed 's/[^0-9]//g' | cut -c 1-17)
|
2024-01-23 13:17:51 +03:00
|
|
|
echo $(( ($r % ($2-$1+1)) + $1 ))
|
|
|
|
}
|
|
|
|
|
2024-05-07 20:19:43 +03: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 16:11:06 +03:00
|
|
|
[ -n "$SHELL_NAME" ] || SHELL_NAME="$(basename "$SHELL")"
|
2024-05-07 20:19:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-12 21:00:22 +03: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)
|
|
|
|
}
|