2019-05-27 14:30:21 +02:00
|
|
|
#!/bin/bash
|
|
|
|
# file: modules/jsshDB.sh
|
|
|
|
# do not edit, this file will be overwritten on update
|
|
|
|
|
|
|
|
# This file is public domain in the USA and all free countries.
|
|
|
|
# Elsewhere, consider it to be WTFPLv2. (wtfpl.net/txt/copying)
|
|
|
|
#
|
2021-01-06 19:22:24 +01:00
|
|
|
#### $$VERSION$$ v1.25-dev-30-gefca2e0
|
2019-05-27 14:30:21 +02:00
|
|
|
#
|
|
|
|
# source from commands.sh to use jsonDB functions
|
|
|
|
#
|
2019-06-04 12:00:19 +02:00
|
|
|
# jsonDB provides simple functions to read and store bash Arrays
|
|
|
|
# from to file in JSON.sh output format, its a simple key/value storage.
|
2019-05-27 14:30:21 +02:00
|
|
|
|
2020-06-14 20:56:46 +02:00
|
|
|
# will be automatically sourced from bashbot
|
2020-06-23 16:35:50 +02:00
|
|
|
# but can be used independent from bashbot also
|
2020-06-14 20:56:46 +02:00
|
|
|
# e.g. to create scrupts to manage jssh files
|
2020-05-15 11:53:19 +02:00
|
|
|
|
2020-05-14 19:47:37 +02:00
|
|
|
# source once magic, function named like file
|
|
|
|
eval "$(basename "${BASH_SOURCE[0]}")(){ :; }"
|
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
# new feature: serialize / atomic operations:
|
|
|
|
# updates will be done atomic with flock
|
2020-06-23 16:35:50 +02:00
|
|
|
# flock should flock should be available on all system as its part of busybox
|
2020-05-15 11:53:19 +02:00
|
|
|
# tinybox
|
|
|
|
|
|
|
|
# lockfile filename.flock is persistent and will be testet with flock for active lock (file open)
|
2020-06-14 20:44:11 +02:00
|
|
|
export JSSH_LOCKNAME=".flock"
|
|
|
|
|
2021-01-04 12:49:43 +01:00
|
|
|
# in UTF-8 äöü etc. are part of [:alnum:] and ranges (e.g. a-z), but we want ASCII a-z ranges!
|
2021-01-04 16:03:53 +01:00
|
|
|
# for more information see doc/4_expert.md#Character_classes
|
2021-01-04 12:38:20 +01:00
|
|
|
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 12:38:20 +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 12:38:20 +01:00
|
|
|
|
|
|
|
# characters allowed for key in key/value pairs
|
2021-01-04 16:10:43 +01:00
|
|
|
JSSH_KEYOK="[-${azAZo9},._]"
|
2021-01-04 12:38:20 +01:00
|
|
|
|
2021-01-04 12:40:14 +01:00
|
|
|
# read string from stdin and and strip invalid characters
|
|
|
|
# $1 - invalid charcaters are replaced with first character
|
|
|
|
# or deleted if $1 is empty
|
|
|
|
jssh_stripKey() { # tr: we must escape first - in [-a-z...]
|
|
|
|
if [[ "$1" =~ ^${JSSH_KEYOK} ]]; then # tr needs [\-...
|
|
|
|
tr -c "${JSSH_KEYOK/\[-/[\\-}\r\n" "${1:0:1}"
|
|
|
|
else
|
|
|
|
tr -dc "${JSSH_KEYOK/\[-/[\\-}\r\n"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2020-06-14 20:44:11 +02:00
|
|
|
# use flock if command exist
|
|
|
|
if [ "$(LC_ALL=C type -t "flock")" = "file" ]; then
|
2020-05-15 11:53:19 +02:00
|
|
|
|
|
|
|
###############
|
|
|
|
# we have flock
|
|
|
|
# use flock for atomic operations
|
|
|
|
|
|
|
|
# read content of a file in JSON.sh format into given ARRAY
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" upfront
|
2020-05-15 11:53:19 +02:00
|
|
|
# $2 filename, must be relative to BASHBOT_ETC, and not contain '..'
|
|
|
|
jssh_readDB() {
|
2019-06-04 13:16:48 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
2020-05-14 20:33:30 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2019-06-04 12:00:19 +02:00
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-05-20 15:18:23 +02:00
|
|
|
# shared lock, many processes can read, max wait 1s
|
2020-06-14 20:44:11 +02:00
|
|
|
{ flock -s -w 1 200; Json2Array "$1" <"${DB}"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-15 11:53:19 +02:00
|
|
|
}
|
2019-05-27 14:30:21 +02:00
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
# write ARRAY content to a file in JSON.sh format
|
|
|
|
# Warning: old content is overwritten
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" upfront
|
2020-05-15 11:53:19 +02:00
|
|
|
# $2 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
|
|
|
jssh_writeDB() {
|
2019-06-04 13:16:48 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
2020-05-14 20:33:30 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2019-06-04 12:00:19 +02:00
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-05-15 11:53:19 +02:00
|
|
|
# exclusive lock, no other process can read or write, maximum wait to get lock is 10s
|
2020-06-14 20:44:11 +02:00
|
|
|
{ flock -e -w 10 200; Array2Json "$1" >"${DB}"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-15 11:53:19 +02:00
|
|
|
}
|
2019-05-27 14:30:21 +02:00
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
# update/write ARRAY content in file without deleting keys not in ARRAY
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" upfront
|
2020-05-15 11:53:19 +02:00
|
|
|
# $2 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-06-15 11:23:43 +02:00
|
|
|
# complex slow, warpper async
|
2020-05-15 11:53:19 +02:00
|
|
|
jssh_updateDB() {
|
2020-06-23 16:35:50 +02:00
|
|
|
# for atomic update we can't use read/writeDB
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$2" ] && return 1
|
|
|
|
local DB="$2.jssh" # check in async
|
2020-05-15 11:53:19 +02:00
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-06-15 11:23:43 +02:00
|
|
|
{ flock -e -w 10 200; jssh_updateDB_async "$@"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-15 11:53:19 +02:00
|
|
|
}
|
2019-06-04 13:16:48 +02:00
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
# insert, update, apped key/value to jsshDB
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-05-15 11:53:19 +02:00
|
|
|
# $2 key value
|
|
|
|
# $3 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-05-31 11:04:38 +02:00
|
|
|
alias jssh_insertDB=jssh_insertKeyDB # backward compatibility
|
|
|
|
# renamed to be more consistent
|
|
|
|
jssh_insertKeyDB() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2019-06-04 13:16:48 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$3")"
|
2020-05-14 20:33:30 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2019-06-04 12:00:19 +02:00
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-05-20 15:18:23 +02:00
|
|
|
# start atomic update here, exclusive max wait 2, it's append, not overwrite
|
2020-05-15 11:53:19 +02:00
|
|
|
{ flock -e -w 2 200
|
|
|
|
# it's append, but last one counts, its a simple DB ...
|
2020-06-15 11:23:43 +02:00
|
|
|
printf '["%s"]\t"%s"\n' "${1//,/\",\"}" "${2//\"/\\\"}" >>"${DB}"
|
2020-06-14 20:44:11 +02:00
|
|
|
} 200>"${DB}${JSSH_LOCKNAME}"
|
2019-06-04 12:00:19 +02:00
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
}
|
|
|
|
|
2020-05-15 17:45:23 +02:00
|
|
|
# delete key/value from jsshDB
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-05-15 17:45:23 +02:00
|
|
|
# $2 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-06-15 11:23:43 +02:00
|
|
|
# medium complex slow, wrapper async
|
2020-05-15 17:45:23 +02:00
|
|
|
jssh_deleteKeyDB() {
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$2" ] && return 1
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2021-01-05 16:17:34 +01:00
|
|
|
local DB="$2.jssh"
|
2020-05-15 17:45:23 +02:00
|
|
|
# start atomic delete here, exclusive max wait 10s
|
2020-06-16 08:39:54 +02:00
|
|
|
{ flock -e -w 10 200; jssh_deleteKeyDB_async "$@"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-15 17:45:23 +02:00
|
|
|
}
|
|
|
|
|
2020-06-05 09:01:20 +02:00
|
|
|
# get key/value from jsshDB
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-05-20 15:18:23 +02:00
|
|
|
# $2 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-06-05 09:01:20 +02:00
|
|
|
alias jssh_getDB=jssh_getKeyDB
|
2020-05-20 15:18:23 +02:00
|
|
|
jssh_getKeyDB() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2020-05-20 15:18:23 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
2020-06-19 22:43:52 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2020-05-20 15:18:23 +02:00
|
|
|
# start atomic delete here, exclusive max wait 1s
|
2020-05-31 11:04:38 +02:00
|
|
|
{ flock -s -w 1 200
|
2020-06-13 18:40:34 +02:00
|
|
|
[ -r "${DB}" ] && sed -n 's/\["'"$1"'"\]\t*"\(.*\)"/\1/p' <"${DB}" | tail -n 1
|
2020-06-14 20:44:11 +02:00
|
|
|
} 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-20 15:18:23 +02:00
|
|
|
}
|
|
|
|
|
2020-05-15 17:45:23 +02:00
|
|
|
|
2020-05-19 18:21:56 +02:00
|
|
|
# add a value to key, used for conters
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-05-19 18:21:56 +02:00
|
|
|
# $2 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-05-31 11:04:38 +02:00
|
|
|
# $3 optional count, value added to counter, add 1 if empty
|
2020-05-19 19:26:10 +02:00
|
|
|
# side effect: if $3 is not given, we add to end of file to be as fast as possible
|
2020-06-15 11:23:43 +02:00
|
|
|
# complex, wrapper to async
|
2020-05-19 18:21:56 +02:00
|
|
|
jssh_countKeyDB() {
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$2" ] && return 1
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2021-01-05 16:17:34 +01:00
|
|
|
local DB="$2.jssh"
|
2020-05-20 15:18:23 +02:00
|
|
|
# start atomic delete here, exclusive max wait 5
|
2020-06-19 11:13:36 +02:00
|
|
|
{ flock -e -w 5 200; jssh_countKeyDB_async "$@"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-05-19 18:21:56 +02:00
|
|
|
}
|
|
|
|
|
2020-06-05 22:13:18 +02:00
|
|
|
# update key/value in place to jsshDB
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-06-05 09:01:20 +02:00
|
|
|
# $2 key value
|
|
|
|
# $3 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
|
|
|
#no own locking, so async is the same as updatekeyDB
|
|
|
|
jssh_updateKeyDB() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$3" ] && return 1
|
2020-06-12 19:18:22 +02:00
|
|
|
declare -A updARR
|
|
|
|
# shellcheck disable=SC2034
|
|
|
|
updARR["$1"]="$2"
|
2021-01-05 16:17:34 +01:00
|
|
|
jssh_updateDB "updARR" "$3" || return 3
|
2020-06-05 09:01:20 +02:00
|
|
|
}
|
|
|
|
|
2020-06-05 22:13:18 +02:00
|
|
|
# $1 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
|
|
|
jssh_clearDB() {
|
|
|
|
local DB; DB="$(jssh_checkDB "$1")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
2020-06-15 11:23:43 +02:00
|
|
|
{ flock -e -w 10 200; printf '' >"${DB}"; } 200>"${DB}${JSSH_LOCKNAME}"
|
2020-06-05 22:13:18 +02:00
|
|
|
}
|
2020-05-19 18:21:56 +02:00
|
|
|
|
2020-06-15 11:23:43 +02:00
|
|
|
# updates Array if DB file has changed since last call
|
|
|
|
# $1 name of array to update
|
|
|
|
# $2 database
|
|
|
|
# $3 id used to identify caller
|
2020-06-23 16:35:50 +02:00
|
|
|
# medium complex, wrapper async
|
2020-06-15 11:23:43 +02:00
|
|
|
jssh_updateArray() {
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$2" ] && return 1
|
|
|
|
local DB="$2.jssh" # name check in async
|
2020-06-15 11:23:43 +02:00
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-06-18 14:33:13 +02:00
|
|
|
declare -n ARRAY="$1"
|
2021-01-05 16:17:34 +01:00
|
|
|
[[ -z "${ARRAY[*]}" || "${DB}" -nt "${DB}.last$3" ]] && touch "${DB}.last$3" && jssh_readDB "$1" "$2"
|
2020-06-15 11:23:43 +02:00
|
|
|
}
|
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
else
|
|
|
|
#########
|
2020-06-15 11:23:43 +02:00
|
|
|
# we have no flock, use non atomic functions
|
2020-05-31 11:04:38 +02:00
|
|
|
alias jssh_readDB=ssh_readDB_async
|
|
|
|
alias jssh_writeDB=jssh_writeDB_async
|
|
|
|
alias jssh_updateDB=jssh_updateDB_async
|
|
|
|
alias jssh_insertDB=jssh_insertDB_async
|
|
|
|
alias ssh_deleteKeyDB=jssh_deleteKeyDB_async
|
2020-06-05 09:01:20 +02:00
|
|
|
alias jssh_getDB=jssh_getKeyDB_async
|
2020-05-31 11:04:38 +02:00
|
|
|
alias jssh_getKeyDB=jssh_getKeyDB_async
|
|
|
|
alias jssh_countKeyDB=jssh_countKeyDB_async
|
2020-06-05 09:01:20 +02:00
|
|
|
alias jssh_updateKeyDB=jssh_updateKeyDB_async
|
2020-06-05 22:13:18 +02:00
|
|
|
alias jssh_clearDB=jssh_clearDB_async
|
2020-06-15 11:23:43 +02:00
|
|
|
alias jssh_updateArray=updateArray_async
|
2020-05-15 11:53:19 +02:00
|
|
|
fi
|
|
|
|
|
|
|
|
##############
|
|
|
|
# no need for atomic
|
|
|
|
|
|
|
|
# print ARRAY content to stdout instead of file
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" upfront
|
2020-06-08 12:58:36 +02:00
|
|
|
jssh_printDB_async() { jssh_printDB "$@"; }
|
2020-05-15 11:53:19 +02:00
|
|
|
jssh_printDB() {
|
|
|
|
Array2Json "$1"
|
2019-06-04 12:00:19 +02:00
|
|
|
}
|
|
|
|
|
2019-05-27 14:30:21 +02:00
|
|
|
# $1 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
2020-06-08 12:58:36 +02:00
|
|
|
jssh_newDB_async() { jssh_newDB "$@"; }
|
2019-05-27 14:30:21 +02:00
|
|
|
jssh_newDB() {
|
2019-06-04 13:16:48 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$1")"
|
2020-05-14 20:33:30 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2020-06-09 09:15:36 +02:00
|
|
|
[ -f "${DB}" ] && return 2 # already exist
|
|
|
|
touch "${DB}"
|
2019-05-27 14:30:21 +02:00
|
|
|
}
|
2019-05-30 18:02:57 +02:00
|
|
|
|
2020-06-01 11:50:42 +02:00
|
|
|
# $1 filename, check filename, it must be relative to BASHBOT_VAR, and not contain '..'
|
2019-06-04 13:16:48 +02:00
|
|
|
# returns real path to DB file if everything is ok
|
2020-06-14 20:36:14 +02:00
|
|
|
jssh_checkDB_async() { jssh_checkDB "$@"; }
|
2019-06-04 13:16:48 +02:00
|
|
|
jssh_checkDB(){
|
2020-06-01 11:50:42 +02:00
|
|
|
local DB
|
2020-05-14 20:33:30 +02:00
|
|
|
[ -z "$1" ] && return 1
|
2020-12-16 17:46:55 +01:00
|
|
|
[[ "$1" = *'../.'* ]] && return 2
|
2020-06-01 11:50:42 +02:00
|
|
|
if [[ "$1" == "${BASHBOT_VAR:-.}"* ]] || [[ "$1" == "${BASHBOT_DATA:-.}"* ]]; then
|
2021-01-05 16:17:34 +01:00
|
|
|
DB="$1.jssh"
|
2020-06-01 11:50:42 +02:00
|
|
|
else
|
2021-01-05 16:17:34 +01:00
|
|
|
DB="${BASHBOT_VAR:-.}/$1.jssh"
|
2020-05-14 22:00:05 +02:00
|
|
|
fi
|
2020-06-20 20:12:36 +02:00
|
|
|
[ "${DB}" != ".jssh" ] && printf '%s' "${DB}"
|
2019-05-30 18:02:57 +02:00
|
|
|
}
|
2019-06-04 12:00:19 +02:00
|
|
|
|
2020-05-15 11:53:19 +02:00
|
|
|
|
|
|
|
######################
|
2020-06-05 09:01:20 +02:00
|
|
|
# implementations as non atomic functions
|
2020-06-23 16:35:50 +02:00
|
|
|
# can be used explictitly or as fallback if flock is not available
|
2020-05-15 11:53:19 +02:00
|
|
|
jssh_readDB_async() {
|
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
|
|
|
[ ! -f "${DB}" ] && return 2
|
|
|
|
Json2Array "$1" <"${DB}"
|
|
|
|
}
|
|
|
|
|
|
|
|
jssh_writeDB_async() {
|
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
|
|
|
[ ! -f "${DB}" ] && return 2
|
|
|
|
Array2Json "$1" >"${DB}"
|
|
|
|
}
|
|
|
|
|
|
|
|
jssh_updateDB_async() {
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$2" ] && return 1
|
2020-05-15 11:53:19 +02:00
|
|
|
declare -n ARRAY="$1"
|
|
|
|
[ -z "${ARRAY[*]}" ] && return 1
|
2020-05-31 11:04:38 +02:00
|
|
|
declare -A oldARR
|
2020-05-15 17:45:23 +02:00
|
|
|
jssh_readDB_async "oldARR" "$2" || return "$?"
|
2020-05-15 11:53:19 +02:00
|
|
|
if [ -z "${oldARR[*]}" ]; then
|
|
|
|
# no old content
|
2020-05-15 17:45:23 +02:00
|
|
|
jssh_writeDB_async "$1" "$2"
|
2020-05-15 11:53:19 +02:00
|
|
|
else
|
|
|
|
# merge arrays
|
2020-05-31 11:04:38 +02:00
|
|
|
local key
|
|
|
|
for key in "${!ARRAY[@]}"
|
|
|
|
do
|
|
|
|
oldARR["${key}"]="${ARRAY["${key}"]}"
|
|
|
|
done
|
|
|
|
Array2Json "oldARR" >"${DB}"
|
2020-05-15 11:53:19 +02:00
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
2020-06-08 12:58:36 +02:00
|
|
|
jssh_insertDB_async() { jssh_insertKeyDB "$@"; }
|
2020-05-31 11:04:38 +02:00
|
|
|
jssh_insertKeyDB_async() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2020-05-15 11:53:19 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$3")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
|
|
|
[ ! -f "${DB}" ] && return 2
|
|
|
|
# its append, but last one counts, its a simple DB ...
|
2020-06-15 11:23:43 +02:00
|
|
|
printf '["%s"]\t"%s"\n' "${1//,/\",\"}" "${2//\"/\\\"}" >>"${DB}"
|
2020-05-15 11:53:19 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-05-15 17:45:23 +02:00
|
|
|
jssh_deleteKeyDB_async() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2020-05-15 18:49:38 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
2020-06-19 11:13:36 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2020-05-15 17:45:23 +02:00
|
|
|
declare -A oldARR
|
2020-06-10 09:32:53 +02:00
|
|
|
Json2Array "oldARR" <"${DB}"
|
2020-05-15 18:49:38 +02:00
|
|
|
unset oldARR["$1"]
|
2020-06-10 09:32:53 +02:00
|
|
|
Array2Json "oldARR" >"${DB}"
|
2020-05-15 17:45:23 +02:00
|
|
|
}
|
|
|
|
|
2020-05-20 15:18:23 +02:00
|
|
|
jssh_getKeyDB_async() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2020-05-20 15:18:23 +02:00
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
2020-06-19 11:13:36 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2020-06-13 18:40:34 +02:00
|
|
|
[ -r "${DB}" ] && sed -n 's/\["'"$1"'"\]\t*"\(.*\)"/\1/p' <"${DB}" | tail -n 1
|
2020-05-20 15:18:23 +02:00
|
|
|
}
|
|
|
|
|
2020-05-19 18:21:56 +02:00
|
|
|
jssh_countKeyDB_async() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2020-06-14 17:39:18 +02:00
|
|
|
local VAL DB; DB="$(jssh_checkDB "$2")"
|
2020-06-19 11:13:36 +02:00
|
|
|
[ -z "${DB}" ] && return 1
|
2020-06-14 17:00:41 +02:00
|
|
|
# start atomic delete here, exclusive max wait 5
|
|
|
|
if [ -n "$3" ]; then
|
2020-06-14 17:39:18 +02:00
|
|
|
declare -A oldARR
|
|
|
|
Json2Array "oldARR" <"${DB}"
|
2020-06-14 17:00:41 +02:00
|
|
|
(( oldARR["$1"]+="$3" ));
|
|
|
|
Array2Json "oldARR" >"${DB}"
|
2020-06-14 17:39:18 +02:00
|
|
|
elif [ -r "${DB}" ]; then
|
2020-06-14 17:00:41 +02:00
|
|
|
# it's append, but last one counts, its a simple DB ...
|
2020-06-14 17:39:18 +02:00
|
|
|
VAL="$(sed -n 's/\["'"$1"'"\]\t*"\(.*\)"/\1/p' <"${DB}" | tail -n 1)"
|
|
|
|
printf '["%s"]\t"%s"\n' "${1//,/\",\"}" "$((++VAL))" >>"${DB}"
|
2020-06-14 17:00:41 +02:00
|
|
|
fi
|
|
|
|
}
|
2020-05-19 18:21:56 +02:00
|
|
|
|
2020-12-16 17:46:55 +01:00
|
|
|
# update key/value in place to jsshDB
|
2020-06-23 16:35:50 +02:00
|
|
|
# $1 key name, can only contain -a-zA-Z0-9,._
|
2020-06-05 09:01:20 +02:00
|
|
|
# $2 key value
|
|
|
|
# $3 filename (must exist!), must be relative to BASHBOT_ETC, and not contain '..'
|
|
|
|
#no own locking, so async is the same as updatekeyDB
|
2020-06-05 22:13:18 +02:00
|
|
|
jssh_updateKeyDB_async() {
|
2021-01-04 12:38:20 +01:00
|
|
|
[[ "$1" =~ ^${JSSH_KEYOK}+$ ]] || return 3
|
2021-01-05 16:17:34 +01:00
|
|
|
[ -z "$3" ] && return 1
|
2020-06-12 19:18:22 +02:00
|
|
|
declare -A updARR
|
|
|
|
# shellcheck disable=SC2034
|
|
|
|
updARR["$1"]="$2"
|
2021-01-05 16:17:34 +01:00
|
|
|
jssh_updateDB_async "updARR" "$3" || return 3
|
2020-06-05 09:01:20 +02:00
|
|
|
}
|
|
|
|
|
2020-06-05 22:13:18 +02:00
|
|
|
jssh_clearDB_async() {
|
|
|
|
local DB; DB="$(jssh_checkDB "$1")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
|
|
|
printf '' >"${DB}"
|
|
|
|
}
|
|
|
|
|
2020-06-15 11:23:43 +02:00
|
|
|
function jssh_updateArray_async() {
|
|
|
|
local DB; DB="$(jssh_checkDB "$2")"
|
|
|
|
[ -z "${DB}" ] && return 1
|
|
|
|
[ ! -f "${DB}" ] && return 2
|
2020-06-18 14:33:13 +02:00
|
|
|
declare -n ARRAY="$1"
|
2021-01-05 16:17:34 +01:00
|
|
|
[[ -z "${ARRAY[*]}" || "${DB}" -nt "${DB}.last$3" ]] && touch "${DB}.last$3" && jssh_readDB_async "$1" "$2"
|
2020-06-15 11:23:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
##############
|
2020-06-23 16:35:50 +02:00
|
|
|
# these 2 functions does all key/value store "magic"
|
2020-06-15 11:23:43 +02:00
|
|
|
# and convert from/to bash array
|
|
|
|
|
2020-06-13 17:56:43 +02:00
|
|
|
# read JSON.sh style data and asssign to an ARRAY
|
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" before calling
|
|
|
|
Json2Array() {
|
2020-12-29 13:03:46 +01:00
|
|
|
# match ["....."]\t and replace \t with = and print quote true false escape not escaped $
|
2020-06-13 18:40:34 +02:00
|
|
|
# shellcheck disable=SC1091,SC1090
|
2020-12-29 13:03:46 +01:00
|
|
|
[ -z "$1" ] || source <( printf "$1"'=( %s )' "$(sed -E -n -e '/\["[-0-9a-zA-Z_,."]+"\]\+*\t/ s/\t/=/p' -e 's/=(true|false)/="\1"/' -e 's/([^\]|^)\$/\1\\$/g')" )
|
2020-06-13 17:56:43 +02:00
|
|
|
}
|
|
|
|
# get Config Key from jssh file without jsshDB
|
|
|
|
# output ARRAY as JSON.sh style data
|
|
|
|
# $1 ARRAY name, must be declared with "declare -A ARRAY" before calling
|
|
|
|
Array2Json() {
|
|
|
|
[ -z "$1" ] && return 1
|
2021-01-06 19:22:24 +01:00
|
|
|
local key
|
2020-06-13 17:56:43 +02:00
|
|
|
declare -n ARRAY="$1"
|
|
|
|
for key in "${!ARRAY[@]}"
|
|
|
|
do
|
2021-01-05 12:57:07 +01:00
|
|
|
[[ "${key}" =~ ^${JSSH_KEYOK}+$ ]] || continue
|
2021-01-06 19:22:24 +01:00
|
|
|
# in case value contains newline convert to \n
|
|
|
|
: "${ARRAY[${key}]//$'\n'/\\n}"
|
|
|
|
printf '["%s"]\t"%s"\n' "${key//,/\",\"}" "${_//\"/\\\"}"
|
2020-06-13 17:56:43 +02:00
|
|
|
done
|
|
|
|
}
|