telegram-bot-bash/bashbot.sh

948 lines
32 KiB
Bash
Raw Permalink Normal View History

#!/bin/bash
2020-12-27 13:53:00 +00:00
##################################################################
#
# File: bashbot.sh
# Note: DO NOT EDIT! this file will be overwritten on update
2021-01-01 20:36:48 +00:00
# shellcheck disable=SC2140,SC2031,SC2120,SC1091,SC1117,SC2059
#
# Description: bashbot, the Telegram bot written in bash.
#
2020-12-27 13:53:00 +00:00
# Written by Drew (@topkecleon) KayM (@gnadelwartz).
2021-01-02 09:18:58 +00:00
# Also contributed: Daniil Gentili (@danog), JuanPotato, BigNerd95,
# TiagoDanin, iicc1, dcoomber
2020-12-27 13:53:00 +00:00
# https://github.com/topkecleon/telegram-bot-bash
#
# This file is public domain in the USA and all free countries.
# Elsewhere, consider it to be WTFPLv2. (wtfpl.net/txt/copying)
2019-04-01 10:52:25 +00:00
#
2021-01-15 18:05:53 +00:00
# Usage: bashbot.sh BOTCOMMAND
BOTCOMMANDS="-h help init start stop status suspendback resumeback killback"
2021-01-01 20:36:48 +00:00
#
2019-04-01 10:52:25 +00:00
# Exit Codes:
2020-12-27 13:53:00 +00:00
# 0 - success (hopefully)
# 1 - can't change to dir
# 2 - can't write to tmp, count or token
# 3 - user / command / file not found
# 4 - unknown command
2021-01-01 18:56:19 +00:00
# 5 - cannot start, stop or get status
2020-12-27 13:53:00 +00:00
# 6 - mandatory module not found
# 7 - can't get bottoken
# 8 - curl/wget missing
# 10 - not bash!
#
2022-06-27 18:18:58 +00:00
#### $$VERSION$$ v1.52-1-g0dae2db
2020-12-27 13:53:00 +00:00
##################################################################
2019-03-22 16:47:36 +00:00
2020-06-23 14:35:50 +00:00
# are we running in a terminal?
2020-12-26 20:18:18 +00:00
NN="\n"
2021-01-04 22:08:09 +00:00
if [ -t 1 ] && [ -n "${TERM}" ]; then
2020-12-27 10:26:38 +00:00
INTERACTIVE='yes'
RED='\e[31m'
GREEN='\e[32m'
ORANGE='\e[35m'
2020-05-18 12:57:53 +00:00
GREY='\e[1;30m'
NC='\e[0m'
2020-12-27 11:38:52 +00:00
NN="${NC}\n"
fi
declare -r INTERACTIVE RED GREEN ORANGE GREY NC NN
2020-12-24 08:19:54 +00:00
# telegram uses utf-8 characters, check if we have an utf-8 charset
if [ "${LANG}" = "${LANG%[Uu][Tt][Ff]*}" ]; then
2020-12-26 20:18:18 +00:00
printf "${ORANGE}Warning: Telegram uses utf-8, but looks like you are using non utf-8 locale:${NC} ${LANG}\n"
2020-12-24 08:19:54 +00:00
fi
2020-09-27 18:24:15 +00:00
# we need some bash 4+ features, check for old bash by feature
2021-01-01 16:14:59 +00:00
if [ "$({ LC_ALL=C.utf-8 printf "%b" "\u1111"; } 2>/dev/null)" = "\u1111" ]; then
2020-12-26 20:18:18 +00:00
printf "${ORANGE}Warning: Missing unicode '\uxxxx' support, missing C.utf-8 locale or to old bash version.${NN}"
2020-09-27 18:24:15 +00:00
fi
2020-12-07 14:50:56 +00:00
2021-01-04 15:03:53 +00:00
# in UTF-8 äöü etc. are part of [:alnum:] and ranges (e.g. a-z), but we want ASCII a-z ranges!
# for more information see doc/4_expert.md#Character_classes
azazaz='abcdefghijklmnopqrstuvwxyz' # a-z :lower:
AZAZAZ='ABCDEFGHIJKLMNOPQRSTUVWXYZ' # A-Z :upper:
o9o9o9='0123456789' # 0-9 :digit:
2021-01-04 15:03:53 +00:00
azAZaz="${azazaz}${AZAZAZ}" # a-zA-Z :alpha:
azAZo9="${azAZaz}${o9o9o9}" # a-zA-z0-9 :alnum:
2021-01-04 15:03:53 +00:00
2020-05-14 19:31:52 +00:00
# some important helper functions
# returns true if command exist
_exists() {
[ "$(type -t "$1")" = "file" ]
2020-05-14 19:31:52 +00:00
}
# execute function if exists
_exec_if_function() {
[ "$(type -t "$1")" != "function" ] && return 1
2020-06-15 07:09:08 +00:00
"$@"
2020-05-14 19:31:52 +00:00
}
# returns true if function exist
_is_function() {
[ "$(type -t "$1")" = "function" ]
}
# round $1 in international notation! , returns float with $2 decimal digits
2020-08-01 09:37:32 +00:00
# if $2 is not given or is not a positive number zero is assumed
_round_float() {
local digit="$2"; [[ "$2" =~ ^[${o9o9o9}]+$ ]] || digit="0"
: "$(LC_ALL=C printf "%.${digit}f" "$1" 2>/dev/null)"
2021-01-20 18:50:19 +00:00
printf "%s" "${_//,/.}" # make more LANG independent
2020-05-14 19:31:52 +00:00
}
# date is external, printf is much faster
_date(){
printf "%(%c)T\n" -1
}
setConfigKey() {
[[ "$1" =~ ^[-${azAZo9},._]+$ ]] || return 3
2020-06-20 18:12:36 +00:00
[ -z "${BOTCONFIG}" ] && return 1
2020-06-19 10:49:18 +00:00
printf '["%s"]\t"%s"\n' "${1//,/\",\"}" "${2//\"/\\\"}" >>"${BOTCONFIG}.jssh"
}
2020-06-12 19:18:32 +00:00
getConfigKey() {
[[ "$1" =~ ^[-${azAZo9},._]+$ ]] || return 3
2021-01-16 10:20:29 +00:00
[ -r "${BOTCONFIG}.jssh" ] && sed -n 's/\["'"$1"'"\]\t*"\(.*\)"/\1/p' "${BOTCONFIG}.jssh" | tail -n 1
}
2021-02-05 13:25:21 +00:00
# escape characters in json strings for telegram
# $1 string, output escaped string
JsonEscape(){
2021-02-05 13:46:11 +00:00
sed -E -e 's/\r//g' -e 's/([-"`´,§$%&/(){}#@!?*.\t])/\\\1/g' <<< "${1//$'\n'/\\n}"
}
2021-02-05 13:25:21 +00:00
# clean \ from escaped json string
# $1 string, output cleaned string
2021-02-21 22:34:34 +00:00
cleanEscape(){ # remove " all \ but \n\u \n or \r
2021-02-10 20:30:42 +00:00
sed -E -e 's/\\"/+/g' -e 's/\\([^nu])/\1/g' -e 's/(\r|\n)//g' <<<"$1"
2021-02-05 13:25:21 +00:00
}
2020-08-01 09:37:32 +00:00
# check if $1 seems a valid token
2020-06-27 08:43:08 +00:00
# return true if token seems to be valid
check_token(){
[[ "$1" =~ ^[${o9o9o9}]{8,10}:[${azAZo9}_-]{35}$ ]] && return 0
return 1
}
# log $1 with date
2021-01-06 14:33:07 +00:00
log_error(){ printf "%(%c)T: %s\n" -1 "$*" >>"${ERRORLOG}"; }
log_debug(){ printf "%(%c)T: %s\n" -1 "$*" >>"${DEBUGLOG}"; }
log_update(){ printf "%(%c)T: %s\n" -1 "$*" >>"${UPDATELOG}"; }
# log $1 with date, special first \n
2021-01-06 14:33:07 +00:00
log_message(){ printf "\n%(%c)T: %s\n" -1 "${1/\\n/$'\n'}" >>"${MESSAGELOG}"; }
2021-01-14 21:15:32 +00:00
# curl is preferred, try detect curl even not in PATH
# sets BASHBOT_CURL to point to curl
DETECTED_CURL="curl"
detect_curl() {
local file warn="Warning: Curl not detected, try fallback to wget! pls install curl or adjust BASHBOT_CURL/BASHBOT_WGET environment variables."
# custom curl command
[ -n "${BASHBOT_CURL}" ] && return 0
# use wget
[ -n "${BASHBOT_WGET}" ] && DETECTED_CURL="wget" && return 1
# default use curl in PATH
BASHBOT_CURL="curl"
_exists curl && return 0
# search in usual locations
for file in /usr/bin /bin /usr/local/bin; do
[ -x "${file}/curl" ] && BASHBOT_CURL="${file}/curl" && return 0
done
# curl not in PATH and not in usual locations
DETECTED_CURL="wget"
log_update "${warn}"; [ -n "${BASHBOTDEBUG}" ] && log_debug "${warn}"
return 1
}
2020-12-27 13:53:00 +00:00
# additional tests if we run in debug mode
2020-06-30 05:26:16 +00:00
export BASHBOTDEBUG
2020-07-10 06:39:33 +00:00
[[ "${BASH_ARGV[0]}" == *"debug"* ]] && BASHBOTDEBUG="yes"
2020-12-27 13:53:00 +00:00
2020-07-10 06:39:33 +00:00
# $1 where $2 command $3 may debug
2020-07-11 06:34:57 +00:00
# shellcheck disable=SC2094
2020-12-15 16:19:09 +00:00
debug_checks(){ {
2020-07-09 20:31:58 +00:00
[ -z "${BASHBOTDEBUG}" ] && return
2021-01-15 12:09:32 +00:00
local token where="$1"; shift
2021-01-06 14:33:07 +00:00
printf "%(%c)T: debug_checks: %s: bashbot.sh %s\n" -1 "${where}" "${1##*/}"
2020-07-11 06:34:57 +00:00
# shellcheck disable=SC2094
2021-01-06 14:33:07 +00:00
[ -z "${DEBUGLOG}" ] && printf "%(%c)T: %s\n" -1 "DEBUGLOG not set! =========="
token="$(getConfigKey "bottoken")"
2021-01-06 14:33:07 +00:00
[ -z "${token}" ] && printf "%(%c)T: %s\n" -1 "Bot token is missing! =========="
check_token "${token}" || printf "%(%c)T: %s\n%s\n" -1 "Invalid bot token! ==========" "${token}"
[ -z "$(getConfigKey "botadmin")" ] && printf "%(%c)T: %s\n" -1 "Bot admin is missing! =========="
# call user defined debug_checks if exists
_exec_if_function my_debug_checks "$(_date)" "${where}" "$*"
} 2>/dev/null >>"${DEBUGLOG}"
2020-12-15 16:19:09 +00:00
}
2020-05-14 19:31:52 +00:00
2021-01-02 05:17:02 +00:00
# some Linux distributions (e.g. Manjaro) doesn't seem to have C locale activated by default
2021-01-03 21:39:36 +00:00
if _exists locale && [ "$(locale -a | grep -c -e "^C$" -e "^C.[uU][tT][fF]")" -lt 2 ]; then
2020-12-26 20:18:18 +00:00
printf "${ORANGE}Warning: locale ${NC}${GREY}C${NC}${ORANGE} and/or ${NC}${GREY}C.utf8${NC}${ORANGE} seems missing, use \"${NC}${GREY}locale -a${NC}${ORANGE}\" to show what locales are installed on your system.${NN}"
2020-12-07 14:50:56 +00:00
fi
2019-04-23 11:07:20 +00:00
# get location and name of bashbot.sh
SCRIPT="$0"
REALME="${BASH_SOURCE[0]}"
SCRIPTDIR="$(dirname "${REALME}")"
RUNDIR="$(dirname "$0")"
MODULEDIR="${SCRIPTDIR}/modules"
2021-02-01 09:00:08 +00:00
# adjust stuff for source, use return from source without source
exit_source() { exit "$1"; }
2021-02-01 09:00:08 +00:00
if [[ "${SCRIPT}" != "${REALME}" || "$1" == "source" ]]; then
SOURCE="yes"
2021-06-03 11:00:37 +00:00
SCRIPT="${REALME}"
[ -z "$1" ] && exit_source() { printf "Exit from source ...\n"; return "$1"; }
2021-02-01 09:00:08 +00:00
fi
# emmbeded system may claim bash but it is not
# check for bash like ARRAY handlung
if ! (unset a; set -A a a; eval "a=(a b)"; eval '[ -n "${a[1]}" ]'; ) > /dev/null 2>&1; then
printf "Error: Current shell does not support ARRAY's, may be busybox ash shell. pls install a real bash!\n"
exit_source 10
fi
2019-05-23 10:26:53 +00:00
2021-02-01 09:00:08 +00:00
# adjust path variables
2021-01-04 22:08:09 +00:00
if [ -n "${BASHBOT_HOME}" ]; then
SCRIPTDIR="${BASHBOT_HOME}"
else
BASHBOT_HOME="${SCRIPTDIR}"
2019-05-23 10:26:53 +00:00
fi
2021-01-04 22:08:09 +00:00
[ -z "${BASHBOT_ETC}" ] && BASHBOT_ETC="${BASHBOT_HOME}"
[ -z "${BASHBOT_VAR}" ] && BASHBOT_VAR="${BASHBOT_HOME}"
2019-05-23 10:26:53 +00:00
2020-06-06 07:21:56 +00:00
ADDONDIR="${BASHBOT_ETC:-.}/addons"
2021-02-01 09:00:08 +00:00
RUNUSER="${USER}" # save original USER
2019-03-31 10:52:11 +00:00
2021-01-01 20:36:48 +00:00
# provide help
case "$1" in
2021-01-08 21:32:05 +00:00
"") [ -z "${SOURCE}" ] && printf "${ORANGE}Available commands: ${GREY}${BOTCOMMANDS}${NN}" && exit
2021-01-01 23:40:42 +00:00
;;
2021-01-09 21:25:15 +00:00
"-h"*) LOGO="${BASHBOT_HOME:-.}/doc/bashbot.ascii"
2021-01-15 18:05:53 +00:00
{ [ -r "${LOGO}" ] && cat "${LOGO}"
2021-01-16 10:20:29 +00:00
sed -nE -e '/(NOT EDIT)|(shellcheck)/d' -e '3,/###/p' "$0"; } | more
2021-01-01 20:36:48 +00:00
exit;;
"help") HELP="${BASHBOT_HOME:-.}/README"
if [ -n "${INTERACTIVE}" ];then
2021-01-04 22:08:09 +00:00
_exists w3m && w3m "${HELP}.html" && exit
_exists lynx && lynx "${HELP}.html" && exit
_exists less && less "${HELP}.txt" && exit
2021-01-01 20:36:48 +00:00
fi
2021-01-04 22:08:09 +00:00
cat "${HELP}.txt"
2021-01-01 20:36:48 +00:00
exit;;
esac
2020-12-27 13:53:00 +00:00
# OK, ENVIRONMENT is set up, let's do some additional tests
2021-01-04 22:08:09 +00:00
if [[ -z "${SOURCE}" && -z "${BASHBOT_HOME}" ]] && ! cd "${RUNDIR}" ; then
2020-12-26 20:18:18 +00:00
printf "${RED}ERROR: Can't change to ${RUNDIR} ...${NN}"
2021-02-01 09:00:08 +00:00
exit_source 1
2019-03-22 16:47:36 +00:00
fi
2021-01-01 23:11:23 +00:00
RUNDIR="."
[ ! -w "." ] && printf "${ORANGE}WARNING: ${RUNDIR} is not writeable!${NN}"
2020-12-27 13:53:00 +00:00
# check if JSON.sh is available
JSONSHFILE="${BASHBOT_JSONSH:-${SCRIPTDIR}/JSON.sh/JSON.sh}"
if [ ! -x "${JSONSHFILE}" ]; then
printf "${RED}ERROR:${NC} ${JSONSHFILE} ${RED}does not exist, are we in dev environment?${NN}${GREY}%s${NN}\n"\
"\$JSONSHFILE is set wrong or bashbot is not installed correctly, see doc/0_install.md"
exit_source 3
fi
2020-12-27 13:53:00 +00:00
# file locations based on ENVIRONMENT
2020-06-19 10:49:18 +00:00
BOTCONFIG="${BASHBOT_ETC:-.}/botconfig"
BOTACL="${BASHBOT_ETC:-.}/botacl"
DATADIR="${BASHBOT_VAR:-.}/data-bot-bash"
BLOCKEDFILE="${BASHBOT_VAR:-.}/blocked"
COUNTFILE="${BASHBOT_VAR:-.}/count"
LOGDIR="${RUNDIR:-.}/logs"
2020-12-27 13:53:00 +00:00
# CREATE botconfig if not exist
2020-08-01 09:37:32 +00:00
# assume everything already set up correctly if TOKEN is set
if [ -z "${BOTTOKEN}" ]; then
# BOTCONFIG does not exist, create
[ ! -f "${BOTCONFIG}.jssh" ] && printf '["bot_config_key"]\t"config_key_value"\n' >>"${BOTCONFIG}.jssh"
if [ -z "$(getConfigKey "bottoken")" ]; then
# ask user for bot token
if [ -z "${INTERACTIVE}" ] && [ "$1" != "init" ]; then
2020-12-26 20:18:18 +00:00
printf "Running headless, set BOTTOKEN or run ${SCRIPT} init first!\n"
2019-04-12 09:27:20 +00:00
exit 2
else
2021-01-01 20:48:59 +00:00
printf "${RED}ENTER BOT TOKEN...${NN}${ORANGE}PLEASE WRITE YOUR TOKEN HERE OR PRESS CTRL+C TO ABORT${NN}"
read -r token
printf "\n"
fi
[ -n "${token}" ] && printf '["bottoken"]\t"%s"\n' "${token}" >> "${BOTCONFIG}.jssh"
fi
# no botadmin, setup botadmin
if [ -z "$(getConfigKey "botadmin")" ]; then
2020-12-27 13:58:34 +00:00
# ask user for bot admin
if [ -z "${INTERACTIVE}" ]; then
2020-12-26 20:18:18 +00:00
printf "Running headless, set botadmin to AUTO MODE!\n"
else
2021-01-01 20:48:59 +00:00
printf "${RED}ENTER BOT ADMIN...${NN}${ORANGE}PLEASE WRITE YOUR TELEGRAM ID HERE OR PRESS ENTER\nTO MAKE FIRST USER TYPING '/start' BOT ADMIN${NN}?\b"
read -r admin
fi
[ -z "${admin}" ] && admin='?'
2020-06-19 10:49:18 +00:00
printf '["botadmin"]\t"%s"\n' "${admin}" >> "${BOTCONFIG}.jssh"
fi
2021-02-07 17:01:34 +00:00
# setup botacl file
if [ ! -f "${BOTACL}" ]; then
printf "${GREY}Create initial ${BOTACL} file.${NN}"
printf '\n' >"${BOTACL}"
fi
# check data dir file
if [ ! -w "${DATADIR}" ]; then
printf "${RED}ERROR: ${DATADIR} does not exist or is not writeable!.${NN}"
[ "$1" != "init" ] && exit_source 2 # skip on init
fi
# setup count file
if [ ! -f "${COUNTFILE}.jssh" ]; then
printf '["counted_user_chat_id"]\t"num_messages_seen"\n' >> "${COUNTFILE}.jssh"
elif [ ! -w "${COUNTFILE}.jssh" ]; then
printf "${RED}WARNING: Can't write to ${COUNTFILE}!.${NN}"
ls -l "${COUNTFILE}.jssh"
fi
# setup blocked file
if [ ! -f "${BLOCKEDFILE}.jssh" ]; then
printf '["blocked_user_or_chat_id"]\t"name and reason"\n' >>"${BLOCKEDFILE}.jssh"
fi
2019-03-22 16:47:36 +00:00
fi
2020-12-15 16:19:09 +00:00
if [[ ! -d "${LOGDIR}" || ! -w "${LOGDIR}" ]]; then
2020-07-09 11:41:09 +00:00
LOGDIR="${RUNDIR:-.}"
fi
DEBUGLOG="${LOGDIR}/DEBUG.log"
ERRORLOG="${LOGDIR}/ERROR.log"
UPDATELOG="${LOGDIR}/BASHBOT.log"
2020-12-15 09:12:18 +00:00
MESSAGELOG="${LOGDIR}/MESSAGE.log"
2020-07-09 11:41:09 +00:00
2020-06-12 19:18:32 +00:00
# read BOTTOKEN from bot database if not set
if [ -z "${BOTTOKEN}" ]; then
BOTTOKEN="$(getConfigKey "bottoken")"
2020-06-19 11:15:50 +00:00
if [ -z "${BOTTOKEN}" ]; then
2020-12-18 14:24:21 +00:00
BOTERROR="Warning: can't get bot token, try to recover working config..."
2020-12-26 20:18:18 +00:00
printf "${ORANGE}${BOTERROR}${NC} "
if [ -r "${BOTCONFIG}.jssh.ok" ]; then
log_error "${BOTERROR}"
2020-12-27 12:12:38 +00:00
mv "${BOTCONFIG}.jssh" "${BOTCONFIG}.jssh.bad"
2020-12-26 20:18:18 +00:00
cp "${BOTCONFIG}.jssh.ok" "${BOTCONFIG}.jssh"; printf "OK\n"
BOTTOKEN="$(getConfigKey "bottoken")"
else
2020-12-27 12:12:38 +00:00
printf "\n${RED}Error: Can't recover from missing bot token! Remove ${BOTCONFIG}.jssh and run${NC} bashbot.sh init\n"
2021-02-01 09:00:08 +00:00
exit_source 7
fi
fi
2020-05-18 12:57:53 +00:00
fi
2020-05-19 12:58:29 +00:00
# BOTTOKEN format checks
2020-06-27 10:43:46 +00:00
if ! check_token "${BOTTOKEN}"; then
2021-01-01 20:55:13 +00:00
printf "\n${ORANGE}Warning: Your bot token is incorrect, it should have the following format:${NC}\n%b%b"\
"<your_bot_id>${RED}:${NC}<35_alphanumeric_characters-hash> ${RED}e.g. =>${NC} 123456789${RED}:${NC}Aa-Zz_0Aa-Zz_1Aa-Zz_2Aa-Zz_3Aa-Zz_4\n\n"\
"${GREY}Your bot token: '${NC}${BOTTOKEN//:/${RED}:${NC}}'\n"
2020-12-27 11:34:39 +00:00
if [[ ! "${BOTTOKEN}" =~ ^[${o9o9o9}]{8,10}: ]]; then
2021-01-01 20:55:13 +00:00
printf "${GREY}\tHint: Bot id not a number or wrong len: ${NC}$(($(wc -c <<<"${BOTTOKEN%:*}")-1)) ${GREY}but should be${NC} 8-10\n"
2020-12-27 11:34:39 +00:00
[ -n "$(getConfigKey "botid")" ] && printf "\t${GREEN}Did you mean: \"${NC}$(getConfigKey "botid")${GREEN}\" ?${NN}"
2020-12-05 13:11:21 +00:00
fi
[[ ! "${BOTTOKEN}" =~ :[${azAZo9}_-]{35}$ ]] &&\
2021-01-01 20:55:13 +00:00
printf "${GREY}\tHint: Hash contains invalid character or has not len${NC} 35 ${GREY}, hash len is ${NC}$(($(wc -c <<<"${BOTTOKEN#*:}")-1))\n"
2020-12-27 11:34:39 +00:00
printf "\n"
2020-05-18 12:57:53 +00:00
fi
2019-03-22 16:47:36 +00:00
2020-05-20 14:38:56 +00:00
##################
# here we start with the real stuff
2021-01-20 18:50:19 +00:00
BASHBOT_RETRY="" # retry by default
2019-04-12 18:49:58 +00:00
2020-06-10 06:20:48 +00:00
URL="${BASHBOT_URL:-https://api.telegram.org/bot}${BOTTOKEN}"
2021-02-06 15:05:18 +00:00
FILEURL="${URL%%/bot*}/file/bot${BOTTOKEN}"
2021-01-04 22:08:09 +00:00
ME_URL=${URL}'/getMe'
2019-04-12 18:49:58 +00:00
2020-06-12 19:18:32 +00:00
#################
# BASHBOT COMMON functions
declare -rx SCRIPT SCRIPTDIR MODULEDIR RUNDIR ADDONDIR BOTACL DATADIR COUNTFILE
2021-01-14 16:47:06 +00:00
declare -rx BOTTOKEN URL ME_URL
2019-05-28 19:12:02 +00:00
2019-05-28 18:50:19 +00:00
declare -ax CMD
declare -Ax UPD BOTSENT USER MESSAGE URLS CONTACT LOCATION CHAT FORWARD REPLYTO VENUE iQUERY iBUTTON
2020-12-13 11:00:18 +00:00
declare -Ax SERVICE NEWMEMBER LEFTMEMBER PINNED MIGRATE
2021-03-03 12:27:07 +00:00
export res CAPTION ME BOTADMIN
2019-04-12 18:49:58 +00:00
2019-05-26 19:25:01 +00:00
2021-01-20 17:20:03 +00:00
###############
# load modules
for module in "${MODULEDIR:-.}"/*.sh ; do
# shellcheck source=./modules/aliases.sh
if ! _is_function "$(basename "${module}")" && [ -r "${module}" ]; then source "${module}" "source"; fi
done
##################
# read commands file if we are not sourced
2019-05-10 09:33:41 +00:00
COMMANDS="${BASHBOT_ETC:-.}/commands.sh"
2021-01-01 23:11:23 +00:00
if [ -r "${COMMANDS}" ]; then
# shellcheck source=./commands.sh
source "${COMMANDS}" "source"
else
[ -z "${SOURCE}" ] && printf "${RED}Warning: ${COMMANDS} does not exist or is not readable!.${NN}"
2019-05-10 09:33:41 +00:00
fi
2021-02-01 17:30:34 +00:00
# no debug checks on source
[ -z "${SOURCE}" ] && debug_checks "start" "$@"
2019-05-10 09:33:41 +00:00
2020-06-10 06:20:48 +00:00
#####################
# BASHBOT INTERNAL functions
#
2020-06-12 19:18:32 +00:00
# do we have BSD sed
2021-01-01 23:11:23 +00:00
sed '1ia' </dev/null 2>/dev/null || printf "${ORANGE}Warning: You may run on a BSD style system without gnu utils ...${NN}"
2020-06-10 06:20:48 +00:00
#jsonDB is now mandatory
2021-01-01 23:32:32 +00:00
if ! _is_function jssh_newDB; then
printf "${RED}ERROR: Mandatory module jsonDB is missing or not readable!${NN}"
exit_source 6
2021-01-01 23:32:32 +00:00
fi
2020-06-12 19:18:32 +00:00
# $1 postfix, e.g. chatid
# $2 prefix, back- or startbot-
procname(){
printf '%s\n' "$2${ME}_$1"
}
# $1 string to search for programme incl. parameters
# returns a list of PIDs of all current bot processes matching $1
proclist() {
# shellcheck disable=SC2009
ps -fu "${UID}" | grep -F "$1" | grep -v ' grep'| grep -F "${ME}" | sed 's/\s\+/\t/g' | cut -f 2
}
# $1 string to search for programme to kill
killallproc() {
local procid; procid="$(proclist "$1")"
if [ -n "${procid}" ] ; then
# shellcheck disable=SC2046
kill $(proclist "$1")
sleep 1
procid="$(proclist "$1")"
# shellcheck disable=SC2046
[ -n "${procid}" ] && kill $(proclist -9 "$1")
fi
debug_checks "end killallproc" "$1"
}
# URL path for file id, $1 file_id
# use download_file "path" to download file
get_file() {
[ -z "$1" ] && return
2021-01-14 18:49:01 +00:00
sendJson "" '"file_id": "'"$1"'"' "${URL}/getFile"
printf "%s\n" "${UPD["result,file_path"]}"
}
# download file to DATADIR
# $1 URL path, $2 proposed filename (may modified/ignored)
# outputs final filename
# keep old function name for backward compatibility
alias download="download_file"
download_file() {
local url="$1" file="${2:-$1}"
# old mode if full URL is given
if [[ "${1}" =~ ^https*:// ]]; then
# random filename if not given for http
if [ -z "$2" ]; then
: "$(mktemp -u -p . "XXXXXXXXXX" 2>/dev/null)"
file="download-${_#./}"
fi
else
# prefix https://api.telegram...
2021-02-06 15:05:18 +00:00
url="${FILEURL}/${url}"
fi
# filename: replace "/" with "-", use mktemp if exist
file="${DATADIR:-.}/${file//\//-}"
[ -f "${file}" ] && file="$(mktemp -p "${DATADIR:-.}" "XXXXX-${file##*/}" )"
getJson "${url}" >"${file}" || return
# output absolute file path
printf "%s\n" "$(cd "${file%/*}" >/dev/null 2>&1 && pwd)/${file##*/}"
}
2021-02-08 16:55:43 +00:00
# notify mycommands about errors while sending
# $1 calling function $2 error $3 chat $4 user $5 error message $6 ... remaining args to calling function
# calls function based on error: bashbotError{function} basbotError{error}
# if no specific function exist try to call bashbotProcessError
processError(){
local func="$1" err="$2"
2021-02-08 18:56:01 +00:00
[[ "${err}" != "4"* ]] && return 1
2021-02-08 16:55:43 +00:00
# check for bashbotError${func} provided in mycommands
# shellcheck disable=SC2082
if _is_function "bashbotError_${func}"; then
"bashbotError_${func}" "$@"
2021-02-08 16:55:43 +00:00
# check for bashbotError${err} provided in mycommands
elif _is_function "bashbotError_${err}"; then
"bashbotError_${err}" "$@"
2021-02-08 16:55:43 +00:00
# noting found, try bashbotProcessError
else
_exec_if_function bashbotProcessError "$@"
2021-02-08 16:55:43 +00:00
fi
}
2020-11-28 16:36:34 +00:00
# iconv used to filter out broken utf characters, if not installed fake it
if ! _exists iconv; then
2020-12-15 19:53:42 +00:00
log_update "Warning: iconv not installed, pls imstall iconv!"
2020-11-28 16:36:34 +00:00
function iconv() { cat; }
fi
2021-01-04 22:08:09 +00:00
TIMEOUT="${BASHBOT_TIMEOUT:-20}"
[[ "${TIMEOUT}" =~ ^[${o9o9o9}]+$ ]] || TIMEOUT="20"
# usage: sendJson "chat" "JSON" "URL"
sendJson(){
2020-12-15 09:07:43 +00:00
local json chat=""
if [ -n "$1" ]; then
chat='"chat_id":'"$1"','
2021-01-20 18:50:19 +00:00
[[ "$1" == *[!${o9o9o9}-]* ]] && chat='"chat_id":"'"$1"' NAN",' # chat id not a number!
2020-12-15 09:12:18 +00:00
fi
# compose final json
2020-12-15 09:07:43 +00:00
json='{'"${chat} $(iconv -f utf-8 -t utf-8 -c <<<"$2")"'}'
if [ -n "${BASHBOTDEBUG}" ] ; then
2021-02-21 22:34:34 +00:00
log_update "sendJson (${DETECTED_CURL}) CHAT=${chat#*:} JSON=$(cleanEscape "${json:0:100}") URL=${3##*/}"
log_message "DEBUG sendJson ==========\n$("${JSONSHFILE}" -b -n <<<"$(cleanEscape "${json}")" 2>&1)"
2020-12-15 09:07:43 +00:00
fi
# chat id not a number
if [[ "${chat}" == *"NAN\"," ]]; then
sendJsonResult "$(printf '["ok"]\tfalse\n["error_code"]\t400\n["description"]\t"Bad Request: chat id not a number"\n')"\
"sendJson (NAN)" "$@"
return
fi
# OK here we go ...
# route to curl/wget specific function
res="$(sendJson_do "${json}" "$3")"
# check telegram response
sendJsonResult "${res}" "sendJson (${DETECTED_CURL})" "$@"
2020-06-26 07:07:00 +00:00
[ -n "${BASHBOT_EVENT_SEND[*]}" ] && event_send "send" "${@}" &
}
2021-02-10 20:11:15 +00:00
UPLOADDIR="${BASHBOT_UPLOAD:-${DATADIR}/upload}"
# $1 chat $2 file, $3 calling function
# return final file name or empty string on error
checkUploadFile() {
local err file="$2"
2021-02-17 09:20:45 +00:00
[[ "${file}" == *'..'* || "${file}" == '.'* ]] && err=1 # no directory traversal
if [[ "${file}" == '/'* ]] ; then
2021-02-10 20:11:15 +00:00
[[ ! "${file}" =~ ${FILE_REGEX} ]] && err=2 # absolute must match REGEX
else
file="${UPLOADDIR:-NOUPLOADDIR}/${file}" # others must be in UPLOADDIR
fi
[ ! -r "${file}" ] && err=3 # and file must exits of course
# file path error, generate error response
if [ -n "${err}" ]; then
BOTSENT=(); BOTSENT[OK]="false"
case "${err}" in
1) BOTSENT[ERROR]="Path to file $2 contains to much '../' or starts with '.'";;
2) BOTSENT[ERROR]="Path to file $2 does not match regex: ${FILE_REGEX} ";;
3) if [[ "$2" == "/"* ]];then
BOTSENT[ERROR]="File not found: $2"
else
BOTSENT[ERROR]="File not found: ${UPLOADDIR}/$2"
fi;;
esac
[ -n "${BASHBOTDEBUG}" ] && log_debug "$3: CHAT=$1 FILE=$2 MSG=${BOTSENT[DESCRIPTION]}"
return 1
fi
2021-02-17 09:20:45 +00:00
printf "%s\n" "${file}"
2021-02-10 20:11:15 +00:00
}
2021-01-29 19:48:31 +00:00
#
# curl / wget specific functions
#
if detect_curl ; then
# here we have curl ----
[ -z "${BASHBOT_CURL}" ] && BASHBOT_CURL="curl"
2021-01-14 16:47:06 +00:00
# $1 URL, $2 hack: log getJson if not ""
getJson(){
# shellcheck disable=SC2086
"${BASHBOT_CURL}" -sL -k ${BASHBOT_CURL_ARGS} -m "${TIMEOUT}" -K - <<<"url=$1"
}
# curl variant for sendJson
# usage: "JSON" "URL"
sendJson_do(){
# shellcheck disable=SC2086
"${BASHBOT_CURL}" -s -k ${BASHBOT_CURL_ARGS} -m "${TIMEOUT}"\
-d "$1" -X POST -K - <<<"url=$2" -H "Content-Type: application/json" | "${JSONSHFILE}" -b -n 2>/dev/null
}
#$1 Chat, $2 what, $3 file, $4 URL, $5 caption
2019-05-20 15:26:21 +00:00
sendUpload() {
[ "$#" -lt 4 ] && return
if [ -n "$5" ]; then
2021-01-05 21:01:32 +00:00
[ -n "${BASHBOTDEBUG}" ] &&\
log_update "sendUpload CHAT=$1 WHAT=$2 FILE=$3 CAPT=$5"
# shellcheck disable=SC2086
res="$("${BASHBOT_CURL}" -s -k ${BASHBOT_CURL_ARGS} -K - <<<"url=$4" -F "chat_id=$1"\
-F "$2=@$3;${3##*/}" -F "caption=$5" | "${JSONSHFILE}" -b -n 2>/dev/null )"
2019-05-20 15:26:21 +00:00
else
2021-01-05 21:01:32 +00:00
# shellcheck disable=SC2086
res="$("${BASHBOT_CURL}" -s -k ${BASHBOT_CURL_ARGS} -K - <<<"url=$4" -F "chat_id=$1"\
-F "$2=@$3;${3##*/}" | "${JSONSHFILE}" -b -n 2>/dev/null )"
2019-05-20 15:26:21 +00:00
fi
sendJsonResult "${res}" "sendUpload (curl)" "$@"
2020-06-26 07:07:00 +00:00
[ -n "${BASHBOT_EVENT_SEND[*]}" ] && event_send "upload" "$@" &
2019-05-20 15:26:21 +00:00
}
else
2020-11-29 16:20:57 +00:00
# NO curl, try wget
if _exists wget; then
getJson(){
# shellcheck disable=SC2086
wget --no-check-certificate -t 2 -T "${TIMEOUT}" ${BASHBOT_WGET_ARGS} -qO - -i <<<"$1"
2020-11-29 16:20:57 +00:00
}
# curl variant for sendJson
# usage: "JSON" "URL"
sendJson_do(){
# shellcheck disable=SC2086
wget --no-check-certificate -t 2 -T "${TIMEOUT}" ${BASHBOT_WGET_ARGS} -qO - --post-data="$1" \
--header='Content-Type:application/json' -i <<<"$2" | "${JSONSHFILE}" -b -n 2>/dev/null
2020-11-29 16:20:57 +00:00
}
sendUpload() {
2020-06-27 08:43:08 +00:00
log_error "Sorry, wget does not support file upload"
2019-05-20 15:26:21 +00:00
BOTSENT[OK]="false"
2020-06-26 07:07:00 +00:00
[ -n "${BASHBOT_EVENT_SEND[*]}" ] && event_send "upload" "$@" &
2020-11-29 16:20:57 +00:00
}
else
# ups, no curl AND no wget
if [ -n "${BASHBOT_WGET}" ]; then
2020-12-26 20:18:18 +00:00
printf "${RED}Error: You set BASHBOT_WGET but no wget found!${NN}"
2020-11-29 16:20:57 +00:00
else
2020-12-26 20:18:18 +00:00
printf "${RED}Error: curl and wget not found, install curl!${NN}"
2020-11-29 16:20:57 +00:00
fi
exit_source 8
2020-11-29 16:20:57 +00:00
fi
fi
2020-06-08 19:47:36 +00:00
# retry sendJson
# $1 function $2 sleep $3 ... $n arguments
2020-06-08 19:47:36 +00:00
sendJsonRetry(){
local retry="$1"; shift
[[ "$1" =~ ^\ *[${o9o9o9}.]+\ *$ ]] && sleep "$1"; shift
2021-01-06 14:33:07 +00:00
printf "%(%c)T: RETRY %s %s %s\n" -1 "${retry}" "$1" "${2:0:60}"
2020-06-08 19:47:36 +00:00
case "${retry}" in
'sendJson'*)
sendJson "$@"
;;
'sendUpload'*)
sendUpload "$@"
;;
2020-06-26 07:11:10 +00:00
'send_album'*)
send_album "$@"
;;
2020-06-08 19:47:36 +00:00
*)
2020-06-27 08:43:08 +00:00
log_error "Error: unknown function ${retry}, cannot retry"
2020-06-10 13:56:34 +00:00
return
2020-06-08 19:47:36 +00:00
;;
esac
[ "${BOTSENT[OK]}" = "true" ] && log_error "Retry OK:${retry} $1 ${2:0:60}"
2020-06-10 13:56:34 +00:00
} >>"${ERRORLOG}"
2020-06-08 19:47:36 +00:00
# process sendJson result
# stdout is written to ERROR.log
# $1 result $2 function $3 .. $n original arguments, $3 is Chat_id
sendJsonResult(){
2020-06-20 18:12:36 +00:00
local offset=0
2020-06-08 18:51:15 +00:00
BOTSENT=( )
Json2Array 'UPD' <<<"$1"
2020-12-15 19:53:42 +00:00
[ -n "${BASHBOTDEBUG}" ] && log_message "New Result ==========\n$1"
BOTSENT[OK]="${UPD["ok"]}"
if [ "${BOTSENT[OK]}" = "true" ]; then
BOTSENT[ID]="${UPD["result,message_id"]}"
BOTSENT[CHAT]="${UPD["result,chat,id"]}"
[ -n "${UPD["result"]}" ] && BOTSENT[RESULT]="${UPD["result"]}"
return
2020-06-23 14:35:50 +00:00
# hot path everything OK!
else
2020-06-08 18:51:15 +00:00
# oops something went wrong!
2021-01-06 14:33:07 +00:00
if [ -n "$1" ]; then
BOTSENT[ERROR]="${UPD["error_code"]}"
BOTSENT[DESCRIPTION]="${UPD["description"]}"
[ -n "${UPD["parameters,retry_after"]}" ] && BOTSENT[RETRY]="${UPD["parameters,retry_after"]}"
else
BOTSENT[OK]="false"
BOTSENT[ERROR]="999"
BOTSENT[DESCRIPTION]="Send to telegram not possible, timeout/broken/no connection"
fi
2020-06-08 18:51:15 +00:00
# log error
2020-06-20 18:12:36 +00:00
[[ "${BOTSENT[ERROR]}" = "400" && "${BOTSENT[DESCRIPTION]}" == *"starting at byte offset"* ]] &&\
offset="${BOTSENT[DESCRIPTION]%* }"
2021-01-06 14:33:07 +00:00
printf "%(%c)T: RESULT=%s FUNC=%s CHAT[ID]=%s ERROR=%s DESC=%s ACTION=%s\n" -1\
"${BOTSENT[OK]}" "$2" "$3" "${BOTSENT[ERROR]}" "${BOTSENT[DESCRIPTION]}" "${4:${offset}:100}"
2020-06-09 10:27:28 +00:00
# warm path, do not retry on error, also if we use wegt
2020-06-11 06:33:59 +00:00
[ -n "${BASHBOT_RETRY}${BASHBOT_WGET}" ] && return
2020-06-08 19:47:36 +00:00
# OK, we can retry sendJson, let's see what's failed
# throttled, telegram say we send too many messages
2020-06-08 18:51:15 +00:00
if [ -n "${BOTSENT[RETRY]}" ]; then
BASHBOT_RETRY="$(( ++BOTSENT[RETRY] ))"
printf "Retry %s in %s seconds ...\n" "$2" "${BASHBOT_RETRY}"
sendJsonRetry "$2" "${BASHBOT_RETRY}" "${@:3}"
unset BASHBOT_RETRY
return
2020-06-08 18:51:15 +00:00
fi
2020-06-08 19:47:36 +00:00
# timeout, failed connection or blocked
2020-06-08 18:51:15 +00:00
if [ "${BOTSENT[ERROR]}" == "999" ];then
# check if default curl and args are OK
if ! curl -sL -k -m 2 "${URL}" >/dev/null 2>&1 ; then
printf "%(%c)T: BASHBOT IP Address seems blocked!\n" -1
# user provided function to recover or notify block
if _exec_if_function bashbotBlockRecover; then
BASHBOT_RETRY="2"
printf "bashbotBlockRecover returned true, retry %s ...\n" "$2"
sendJsonRetry "$2" "${BASHBOT_RETRY}" "${@:3}"
unset BASHBOT_RETRY
fi
# seems not blocked, try if blockrecover and default curl args working
elif [ -n "${BASHBOT_CURL_ARGS}" ] || [ "${BASHBOT_CURL}" != "curl" ]; then
printf "Problem with \"%s %s\"? retry %s with default config ...\n"\
"${BASHBOT_CURL}" "${BASHBOT_CURL_ARGS}" "$2"
BASHBOT_RETRY="2"; BASHBOT_CURL="curl"; BASHBOT_CURL_ARGS=""
_exec_if_function bashbotBlockRecover
sendJsonRetry "$2" "${BASHBOT_RETRY}" "${@:3}"
unset BASHBOT_RETRY
fi
[ -n "${BOTSENT[ERROR]}" ] && processError "$3" "${BOTSENT[ERROR]}" "$4" "" "${BOTSENT[DESCRIPTION]}" "$5" "$6"
2020-06-08 18:51:15 +00:00
fi
fi
} >>"${ERRORLOG}"
# convert common telegram entities to JSON
# title caption description markup inlinekeyboard
title2Json(){
local title caption desc markup keyboard
[ -n "$1" ] && title=',"title":"'$(JsonEscape "$1")'"'
[ -n "$2" ] && caption=',"caption":"'$(JsonEscape "$2")'"'
[ -n "$3" ] && desc=',"description":"'$(JsonEscape "$3")'"'
[ -n "$4" ] && markup=',"parse_mode":"'"$4"'"'
[ -n "$5" ] && keyboard=',"reply_markup":"'$(JsonEscape "$5")'"'
printf '%s\n' "${title}${caption}${desc}${markup}${keyboard}"
}
2020-12-03 19:43:20 +00:00
# get bot name and id from telegram
getBotName() {
declare -A BOTARRAY
2021-01-04 22:08:09 +00:00
Json2Array 'BOTARRAY' <<<"$(getJson "${ME_URL}" | "${JSONSHFILE}" -b -n 2>/dev/null)"
2020-12-03 19:43:20 +00:00
[ -z "${BOTARRAY["result","username"]}" ] && return 1
# save botname and id
setConfigKey "botname" "${BOTARRAY["result","username"]}"
setConfigKey "botid" "${BOTARRAY["result","id"]}"
2020-12-26 20:18:18 +00:00
printf "${BOTARRAY["result","username"]}\n"
2016-01-06 16:11:56 +00:00
}
2020-06-23 14:35:50 +00:00
# pure bash implementation, done by KayM (@gnadelwartz)
2019-05-14 15:56:23 +00:00
# see https://stackoverflow.com/a/55666449/9381171
JsonDecode() {
2021-01-15 12:09:32 +00:00
local remain U out="$1"
2019-05-14 15:56:23 +00:00
local regexp='(.*)\\u[dD]([0-9a-fA-F]{3})\\u[dD]([0-9a-fA-F]{3})(.*)'
2021-01-04 22:08:09 +00:00
while [[ "${out}" =~ ${regexp} ]] ; do
U=$(( ( (0xd${BASH_REMATCH[2]} & 0x3ff) <<10 ) | ( 0xd${BASH_REMATCH[3]} & 0x3ff ) + 0x10000 ))
remain="$(printf '\\U%8.8x' "${U}")${BASH_REMATCH[4]}${remain}"
out="${BASH_REMATCH[1]}"
done
2021-01-15 12:09:32 +00:00
printf "%b\n" "${out}${remain}"
2019-05-14 15:56:23 +00:00
}
2019-06-04 16:04:52 +00:00
EVENT_SEND="0"
2021-01-29 19:48:31 +00:00
declare -Ax BASHBOT_EVENT_SEND
2019-06-04 16:04:52 +00:00
event_send() {
# max recursion level 5 to avoid fork bombs
2021-01-04 22:08:09 +00:00
(( EVENT_SEND++ )); [ "${EVENT_SEND}" -gt "5" ] && return
2019-06-04 16:04:52 +00:00
# shellcheck disable=SC2153
for key in "${!BASHBOT_EVENT_SEND[@]}"
do
_exec_if_function "${BASHBOT_EVENT_SEND[${key}]}" "$@"
done
}
2021-03-04 12:58:04 +00:00
# cleanup activities on startup, called from startbot and resume background jobs
# $1 action, timestamp for action is saved in config
bot_cleanup() {
# cleanup countfile on startup
jssh_deleteKeyDB "CLEAN_COUNTER_DATABASE_ON_STARTUP" "${COUNTFILE}"
[ -f "${COUNTFILE}.jssh.flock" ] && rm -f "${COUNTFILE}.jssh.flock"
# store action time and cleanup botconfig on startup
[ -n "$1" ] && jssh_updateKeyDB "$1" "$(_date)" "${BOTCONFIG}"
[ -f "${BOTCONFIG}.jssh.flock" ] && rm -f "${BOTCONFIG}.jssh.flock"
}
2019-04-22 19:05:52 +00:00
2021-01-27 15:19:43 +00:00
# fallback version, full version is in bin/bashbot_init.in.sh
2019-04-22 19:05:52 +00:00
# initialize bot environment, user and permissions
bot_init() {
2021-02-01 18:37:24 +00:00
if [ -n "${BASHBOT_HOME}" ] && ! cd "${BASHBOT_HOME}"; then
printf "Can't change to BASHBOT_HOME"
exit 1
fi
2021-02-01 11:58:57 +00:00
# initialize addons
2021-01-27 15:29:58 +00:00
printf "Initialize addons ...\n"
2019-12-07 12:25:50 +00:00
for addons in "${ADDONDIR:-.}"/*.sh ; do
2019-05-25 17:31:20 +00:00
# shellcheck source=./modules/aliases.sh
2021-01-27 15:19:43 +00:00
[ -r "${addons}" ] && source "${addons}" "init" "$1"
2019-05-25 17:31:20 +00:00
done
2020-12-26 20:18:18 +00:00
printf "Done.\n"
2021-01-10 17:23:37 +00:00
# adjust permissions
2021-01-27 15:19:43 +00:00
printf "Adjusting files and permissions ...\n"
2021-01-10 17:23:37 +00:00
chmod 711 .
chmod -R o-w ./*
chmod -R u+w "${COUNTFILE}"* "${BLOCKEDFILE}"* "${DATADIR}" logs "${LOGDIR}/"*.log 2>/dev/null
chmod -R o-r,o-w "${COUNTFILE}"* "${BLOCKEDFILE}"* "${DATADIR}" "${BOTACL}" 2>/dev/null
# jsshDB must writeable by owner
find . -name '*.jssh*' -exec chmod u+w \{\} +
printf "Done.\n"
2021-01-27 15:19:43 +00:00
_exec_if_function my_init
2019-04-22 19:05:52 +00:00
}
2019-05-26 15:03:58 +00:00
if ! _is_function send_message ; then
2020-12-26 20:18:18 +00:00
printf "${RED}ERROR: send_message is not available, did you deactivate ${MODULEDIR}/sendMessage.sh?${NN}"
exit_source 1
2019-05-26 15:03:58 +00:00
fi
# check if JSON.awk exist and has x flag
JSONAWKFILE="${JSONSHFILE%.sh}.awk"
if [ -x "${JSONAWKFILE}" ] && _exists awk ; then
JSONSHFILE="JsonAwk"; JsonAwk() { "${JSONAWKFILE}" -v "BRIEF=8" -v "STRICT=0" -; }
fi
2016-03-20 21:34:55 +00:00
# source the script with source as param to use functions in other scripts
2019-04-20 14:26:16 +00:00
# do not execute if read from other scripts
2021-03-03 12:27:07 +00:00
BOTADMIN="$(getConfigKey "botadmin")"
2020-06-19 16:47:18 +00:00
if [ -z "${SOURCE}" ]; then
2019-04-22 19:05:52 +00:00
##############
# internal options only for use from bashbot and developers
2020-11-29 14:34:00 +00:00
# shellcheck disable=SC2221,SC2222
case "$1" in
# update botname when starting only
2020-06-22 12:10:36 +00:00
"botname"|"start"*)
2020-06-12 19:18:32 +00:00
ME="$(getBotName)"
if [ -n "${ME}" ]; then
# ok we have a connection and got botname, save it
2020-12-27 10:26:38 +00:00
[ -n "${INTERACTIVE}" ] && printf "${GREY}Bottoken is valid ...${NN}"
2020-06-19 10:49:18 +00:00
jssh_updateKeyDB "botname" "${ME}" "${BOTCONFIG}"
rm -f "${BOTCONFIG}.jssh.flock"
2020-06-12 19:18:32 +00:00
else
2020-12-26 20:18:18 +00:00
printf "${GREY}Info: Can't get Botname from Telegram, try cached one ...${NN}"
2020-06-12 19:18:32 +00:00
ME="$(getConfigKey "botname")"
2021-01-04 22:08:09 +00:00
if [ -z "${ME}" ]; then
2020-12-26 20:18:18 +00:00
printf "${RED}ERROR: No cached botname, can't continue! ...${NN}"
2020-06-12 19:18:32 +00:00
exit 1
fi
fi
2020-12-27 10:26:38 +00:00
[ -n "${INTERACTIVE}" ] && printf "Bot Name: %s\n" "${ME}"
2020-06-12 19:18:32 +00:00
[ "$1" = "botname" ] && exit
;;&
2020-06-23 14:35:50 +00:00
# used to send output of background and interactive to chats
2021-01-20 18:50:19 +00:00
"outproc") # $2 chat_id $3 identifier of job, internal use only!
2020-12-26 20:18:18 +00:00
[ -z "$3" ] && printf "No job identifier\n" && exit 3
[ -z "$2" ] && printf "No chat to send to\n" && exit 3
ME="$(getConfigKey "botname")"
2020-06-12 19:18:32 +00:00
# read until terminated
2019-05-19 15:31:55 +00:00
while read -r line ;do
2021-01-04 22:08:09 +00:00
[ -n "${line}" ] && send_message "$2" "${line}"
2019-05-18 18:25:18 +00:00
done
2020-06-12 19:18:32 +00:00
# cleanup datadir, keep logfile if not empty
2019-05-28 18:44:40 +00:00
rm -f -r "${DATADIR:-.}/$3"
[ -s "${DATADIR:-.}/$3.log" ] || rm -f "${DATADIR:-.}/$3.log"
debug_checks "end outproc" "$@"
2019-04-22 19:05:52 +00:00
exit
;;
2021-01-29 19:48:31 +00:00
# finally starts the read update loop, internal use only
2019-04-22 19:05:52 +00:00
"startbot" )
2021-03-03 17:37:57 +00:00
_exec_if_function start_bot "$2" "polling mode"
_exec_if_function get_updates "$2"
debug_checks "end startbot" "$@"
2019-04-22 19:05:52 +00:00
exit
;;
2020-06-12 19:18:32 +00:00
# run after every update to update files and adjust permissions
"init")
2021-01-27 12:58:55 +00:00
# shellcheck source=./bin/bashbot._init.inc.sh"
[ -r "${BASHBOT_HOME:-.}/bin/bashbot_init.inc.sh" ] && source "${BASHBOT_HOME:-.}/bin/bashbot_init.inc.sh"
2019-05-25 17:31:20 +00:00
bot_init "$2"
debug_checks "end init" "$@"
2019-04-22 19:05:52 +00:00
exit
;;
2020-12-27 10:18:42 +00:00
# stats deprecated
"stats"|"count")
printf "${ORANGE}Stats is a separate command now, see bin/bashbot_stats.sh --help${NN}"
"${BASHBOT_HOME:-.}"/bin/bashbot_stats.sh --help
exit
;;
2020-12-27 10:18:42 +00:00
# broadcast deprecated
'broadcast')
2020-12-27 10:18:42 +00:00
printf "${ORANGE}Broadcast is a separate command now, see bin/send_broadcast.sh --help${NN}"
"${BASHBOT_HOME:-.}"/bin/send_broadcast.sh --help
exit
;;
# does what it says
"status")
ME="$(getConfigKey "botname")"
SESSION="${ME:-_bot}-startbot"
BOTPID="$(proclist "${SESSION}")"
if [ -n "${BOTPID}" ]; then
2020-12-26 20:18:18 +00:00
printf "${GREEN}Bot is running with UID ${RUNUSER}.${NN}"
exit
else
2020-12-26 20:18:18 +00:00
printf "${ORANGE}No Bot running with UID ${RUNUSER}.${NN}"
exit 5
fi
debug_checks "end status" "$@"
;;
# start bot as background job and check if bot is running
2016-04-17 12:13:28 +00:00
"start")
2021-01-29 19:55:54 +00:00
SESSION="${ME:-_bot}-startbot"
BOTPID="$(proclist "${SESSION}")"
2021-01-31 11:40:13 +00:00
if _is_function process_update; then
2021-01-29 19:48:31 +00:00
# shellcheck disable=SC2086
2021-02-01 08:02:22 +00:00
[ -n "${BOTPID}" ] && kill ${BOTPID} && printf "${GREY}Stop already running bot ...${NN}"
2021-01-29 19:48:31 +00:00
nohup "${SCRIPT}" "startbot" "$2" "${SESSION}" &>/dev/null &
printf "Session Name: %s\n" "${SESSION}"
sleep 1
else
printf "${ORANGE}Update processing disabled, bot can only send messages.${NN}"
2021-02-01 08:02:22 +00:00
[ -n "${BOTPID}" ] && printf "${ORANGE}Already running bot found ...${NN}"
2021-01-29 19:48:31 +00:00
fi
if [ -n "$(proclist "${SESSION}")" ]; then
2020-12-26 20:18:18 +00:00
printf "${GREEN}Bot started successfully.${NN}"
else
2020-12-26 20:18:18 +00:00
printf "${RED}An error occurred while starting the bot.${NN}"
exit 5
fi
debug_checks "end start" "$@"
2016-04-17 12:13:28 +00:00
;;
2020-06-12 19:18:32 +00:00
# does what it says
"stop")
ME="$(getConfigKey "botname")"
SESSION="${ME:-_bot}-startbot"
BOTPID="$(proclist "${SESSION}")"
if [ -n "${BOTPID}" ]; then
# shellcheck disable=SC2086
if kill ${BOTPID}; then
# inform botadmin about stop
2021-03-03 17:37:57 +00:00
send_normal_message "${BOTADMIN}" "Bot ${ME} polling mode stopped ..." &
2020-12-26 20:18:18 +00:00
printf "${GREEN}OK. Bot stopped successfully.${NN}"
else
2020-12-26 20:18:18 +00:00
printf "${RED}An error occurred while stopping bot.${NN}"
exit 5
fi
else
2020-12-26 20:18:18 +00:00
printf "${ORANGE}No Bot running with UID ${RUNUSER}.${NN}"
fi
debug_checks "end stop" "$@"
exit
2019-03-25 10:15:07 +00:00
;;
2020-06-23 14:35:50 +00:00
# suspend, resume or kill background jobs
2021-02-25 18:58:00 +00:00
"suspendb"*|"resumeb"*|'restartb'*|"killb"*)
2020-12-26 20:18:18 +00:00
_is_function job_control || { printf "${RED}Module background is not available!${NN}"; exit 3; }
2020-06-22 12:10:36 +00:00
ME="$(getConfigKey "botname")"
job_control "$1"
debug_checks "end background $1" "$@"
;;
*)
2020-12-26 20:18:18 +00:00
printf "${RED}${REALME##*/}: unknown command${NN}"
2021-01-03 21:39:36 +00:00
printf "${ORANGE}Available commands: ${GREY}${BOTCOMMANDS}${NN}" && exit
exit 4
2016-04-17 12:13:28 +00:00
;;
2019-04-20 14:26:16 +00:00
esac
2019-04-20 14:26:16 +00:00
# warn if root
if [[ "${UID}" -eq "0" ]] ; then
2021-01-03 21:39:36 +00:00
printf "\n${ORANGE}WARNING: ${SCRIPT} was started as ROOT (UID 0)!${NN}"
2020-12-26 20:18:18 +00:00
printf "${ORANGE}You are at HIGH RISK when running a Telegram BOT with root privileges!${NN}"
2019-04-20 14:26:16 +00:00
fi
fi # end source