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