| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595 | #!/usr/bin/env bash# SPDX-License-Identifier: AGPL-3.0-or-later# shellcheck disable=SC2001# shellcheck source=utils/lib.shsource "$(dirname "${BASH_SOURCE[0]}")/lib.sh"# shellcheck source=utils/lib_install.shsource "${REPO_ROOT}/utils/lib_install.sh"# ----------------------------------------------------------------------------# config# ----------------------------------------------------------------------------PUBLIC_HOST="${PUBLIC_HOST:-$(echo "$PUBLIC_URL" | sed -e 's/[^/]*\/\/\([^@]*@\)\?\([^:/]*\).*/\2/')}"FILTRON_URL_PATH="${FILTRON_URL_PATH:-$(echo "${PUBLIC_URL}" \| sed -e 's,^.*://[^/]*\(/.*\),\1,g')}"[[ "${FILTRON_URL_PATH}" == "${PUBLIC_URL}" ]] && FILTRON_URL_PATH=/FILTRON_ETC="/etc/filtron"FILTRON_RULES="$FILTRON_ETC/rules.json"FILTRON_RULES_TEMPLATE="${FILTRON_RULES_TEMPLATE:-${REPO_ROOT}/utils/templates/etc/filtron/rules.json}"FILTRON_API="${FILTRON_API:-127.0.0.1:4005}"FILTRON_LISTEN="${FILTRON_LISTEN:-127.0.0.1:4004}"# The filtron target is the SearXNG installation, listenning on server.port at# server.bind_address.  The default of FILTRON_TARGET is taken from the YAML# configuration, do not change this value without reinstalling the entire# SearXNG suite including filtron & morty.FILTRON_TARGET="${SEARXNG_BIND_ADDRESS}:${SEARXNG_PORT}"SERVICE_NAME="filtron"SERVICE_USER="${SERVICE_USER:-${SERVICE_NAME}}"SERVICE_HOME_BASE="${SERVICE_HOME_BASE:-/usr/local}"SERVICE_HOME="${SERVICE_HOME_BASE}/${SERVICE_USER}"SERVICE_SYSTEMD_UNIT="${SYSTEMD_UNITS}/${SERVICE_NAME}.service"# shellcheck disable=SC2034SERVICE_GROUP="${SERVICE_USER}"# shellcheck disable=SC2034SERVICE_GROUP="${SERVICE_USER}"GO_ENV="${SERVICE_HOME}/.go_env"GO_PKG_URL="https://dl.google.com/go/go1.13.5.linux-amd64.tar.gz"GO_TAR=$(basename "$GO_PKG_URL")APACHE_FILTRON_SITE="searx.conf"NGINX_FILTRON_SITE="searx.conf"# shellcheck disable=SC2034CONFIG_FILES=(    "${FILTRON_RULES}"    "${SERVICE_SYSTEMD_UNIT}")# ----------------------------------------------------------------------------usage() {# ----------------------------------------------------------------------------    # shellcheck disable=SC1117    cat <<EOFusage::  $(basename "$0") shell  $(basename "$0") install    [all|user|rules]  $(basename "$0") update     [filtron]  $(basename "$0") remove     [all]  $(basename "$0") activate   [service]  $(basename "$0") deactivate [service]  $(basename "$0") inspect    [service]  $(basename "$0") option     [debug-on|debug-off]  $(basename "$0") apache     [install|remove]  $(basename "$0") nginx      [install|remove]shell  start interactive shell from user ${SERVICE_USER}install / remove  :all:        complete setup of filtron service  :user:       add/remove service user '$SERVICE_USER' ($SERVICE_HOME)  :rules:      reinstall filtron rules $FILTRON_RULESupdate filtron  Update filtron installation ($SERVICE_HOME)activate service  activate and start service daemon (systemd unit)deactivate service  stop and deactivate service daemon (systemd unit)inspect service  show service status and logoption  set one of the available optionsapache (${PUBLIC_URL})  :install: apache site with a reverse proxy (ProxyPass)  :remove:  apache site ${APACHE_FILTRON_SITE}nginx (${PUBLIC_URL})  :install: nginx site with a reverse proxy (ProxyPass)  :remove:  nginx site ${NGINX_FILTRON_SITE}filtron rules: ${FILTRON_RULES_TEMPLATE}---- sourced ${DOT_CONFIG} :  SERVICE_USER        : ${SERVICE_USER}  SERVICE_HOME        : ${SERVICE_HOME}  FILTRON_TARGET      : ${FILTRON_TARGET}  FILTRON_API         : ${FILTRON_API}  FILTRON_LISTEN      : ${FILTRON_LISTEN}  FILTRON_URL_PATH    : ${FILTRON_URL_PATH}EOF    install_log_searx_instance    [[ -n ${1} ]] &&  err_msg "$1"}main() {    required_commands \        sudo install git wget curl \        || exit    local _usage="unknown or missing $1 command $2"    case $1 in        --getenv)  var="$2"; echo "${!var}"; exit 0;;        -h|--help) usage; exit 0;;        shell)            sudo_or_exit            interactive_shell "${SERVICE_USER}"            ;;        inspect)            case $2 in                service)                    sudo_or_exit                    inspect_service                    ;;                *) usage "$_usage"; exit 42;;            esac ;;        install)            rst_title "$SERVICE_NAME" part            sudo_or_exit            case $2 in                all) install_all ;;                user) assert_user ;;                rules)                    install_rules                    systemd_restart_service "${SERVICE_NAME}"                    ;;                *) usage "$_usage"; exit 42;;            esac ;;        update)            sudo_or_exit            case $2 in                filtron) update_filtron ;;                *) usage "$_usage"; exit 42;;            esac ;;        remove)            sudo_or_exit            case $2 in                all) remove_all;;                user) drop_service_account "${SERVICE_USER}" ;;                *) usage "$_usage"; exit 42;;            esac ;;        activate)            sudo_or_exit            case $2 in                service)  systemd_activate_service "${SERVICE_NAME}" ;;                *) usage "$_usage"; exit 42;;            esac ;;        deactivate)            sudo_or_exit            case $2 in                service)  systemd_deactivate_service "${SERVICE_NAME}" ;;                *) usage "$_usage"; exit 42;;            esac ;;        apache)            sudo_or_exit            case $2 in                install) install_apache_site ;;                remove) remove_apache_site ;;                *) usage "$_usage"; exit 42;;            esac ;;        nginx)            sudo_or_exit            case $2 in                install) install_nginx_site ;;                remove) remove_nginx_site ;;                *) usage "$_usage"; exit 42;;            esac ;;        option)            sudo_or_exit            case $2 in                debug-on)  echo; enable_debug ;;                debug-off)  echo; disable_debug ;;                *) usage "$_usage"; exit 42;;            esac ;;        doc) rst-doc ;;        *) usage "unknown or missing command $1"; exit 42;;    esac}install_all() {    rst_title "Install $SERVICE_NAME (service)"    assert_user    wait_key    install_go "${GO_PKG_URL}" "${GO_TAR}" "${SERVICE_USER}"    wait_key    install_filtron    install_rules    wait_key    systemd_install_service "${SERVICE_NAME}" "${SERVICE_SYSTEMD_UNIT}"    wait_key    echo    if ! service_is_available "http://${FILTRON_LISTEN}" ; then        err_msg "Filtron is not listening on: http://${FILTRON_LISTEN}"    fi    if apache_is_installed; then        info_msg "Apache is installed on this host."        if ask_yn "Do you want to install a reverse proxy (ProxyPass)" Yn; then            install_apache_site        fi    elif nginx_is_installed; then        info_msg "nginx is installed on this host."        if ask_yn "Do you want to install a reverse proxy (ProxyPass)" Yn; then            install_nginx_site        fi    fi    if ask_yn "Do you want to inspect the installation?" Ny; then        inspect_service    fi}remove_all() {    rst_title "De-Install $SERVICE_NAME (service)"    rst_para "\It goes without saying that this script can only be used to removeinstallations that were installed with this script."    if ! systemd_remove_service "${SERVICE_NAME}" "${SERVICE_SYSTEMD_UNIT}"; then        return 42    fi    drop_service_account "${SERVICE_USER}"    rm -r "$FILTRON_ETC" 2>&1 | prefix_stdout    if service_is_available "${PUBLIC_URL}"; then        MSG="** Don't forget to remove your public site! (${PUBLIC_URL}) **" wait_key 10    fi}assert_user() {    rst_title "user $SERVICE_USER" section    echo    tee_stderr 1 <<EOF | bash | prefix_stdoutuseradd --shell /bin/bash --system \ --home-dir "$SERVICE_HOME" \ --comment 'Reverse HTTP proxy to filter requests' $SERVICE_USERmkdir "$SERVICE_HOME"chown -R "$SERVICE_GROUP:$SERVICE_GROUP" "$SERVICE_HOME"groups $SERVICE_USEREOF    SERVICE_HOME="$(sudo -i -u "$SERVICE_USER" echo \$HOME)"    export SERVICE_HOME    echo "export SERVICE_HOME=$SERVICE_HOME"    cat > "$GO_ENV" <<EOFexport GOPATH=\$HOME/go-appsexport PATH=\$PATH:\$HOME/local/go/bin:\$GOPATH/binEOF    echo "Environment $GO_ENV has been setup."    tee_stderr <<EOF | sudo -i -u "$SERVICE_USER"grep -qFs -- 'source $GO_ENV' ~/.profile || echo 'source $GO_ENV' >> ~/.profileEOF}filtron_is_installed() {    [[ -f $SERVICE_HOME/go-apps/bin/filtron ]]}_svcpr="  ${_Yellow}|${SERVICE_USER}|${_creset} "install_filtron() {    rst_title "Install filtron in user's ~/go-apps" section    echo    tee_stderr <<EOF | sudo -i -u "$SERVICE_USER" 2>&1 | prefix_stdout "$_svcpr"go get -v -u github.com/asciimoo/filtronEOF}update_filtron() {    rst_title "Update filtron" section    echo    tee_stderr <<EOF | sudo -i -u "$SERVICE_USER" 2>&1 | prefix_stdout "$_svcpr"go get -v -u github.com/asciimoo/filtronEOF}install_rules() {    rst_title "Install filtron rules"    echo    if [[ ! -f "${FILTRON_RULES}" ]]; then        info_msg "install rules ${FILTRON_RULES_TEMPLATE}"        info_msg "  --> ${FILTRON_RULES}"        mkdir -p "$(dirname "${FILTRON_RULES}")"        cp "${FILTRON_RULES_TEMPLATE}" "${FILTRON_RULES}"        return    fi    if cmp --silent "${FILTRON_RULES}" "${FILTRON_RULES_TEMPLATE}"; then        info_msg "${FILTRON_RULES} is up to date with"        info_msg "${FILTRON_RULES_TEMPLATE}"        return    fi    rst_para "Diff between origin's rules file (+) and current (-):"    echo "${FILTRON_RULES}" "${FILTRON_RULES_TEMPLATE}"    $DIFF_CMD "${FILTRON_RULES}" "${FILTRON_RULES_TEMPLATE}"    local action    choose_one action "What should happen to the rules file? " \           "keep configuration unchanged" \           "use origin rules" \           "start interactive shell"    case $action in        "keep configuration unchanged")            info_msg "leave rules file unchanged"            ;;        "use origin rules")            backup_file "${FILTRON_RULES}"            info_msg "install origin rules"            cp "${FILTRON_RULES_TEMPLATE}" "${FILTRON_RULES}"            ;;        "start interactive shell")            backup_file "${FILTRON_RULES}"            echo -e "// exit with [${_BCyan}CTRL-D${_creset}]"            sudo -H -i            rst_para 'Diff between new rules file (-) and current (+):'            echo            $DIFF_CMD "${FILTRON_RULES_TEMPLATE}" "${FILTRON_RULES}"            wait_key            ;;    esac}inspect_service() {    rst_title "service status & log"    cat <<EOFsourced ${DOT_CONFIG} :  SERVICE_USER        : ${SERVICE_USER}  SERVICE_HOME        : ${SERVICE_HOME}  FILTRON_TARGET      : ${FILTRON_TARGET}  FILTRON_API         : ${FILTRON_API}  FILTRON_LISTEN      : ${FILTRON_LISTEN}  FILTRON_URL_PATH    : ${FILTRON_URL_PATH}EOF    install_log_searx_instance    if service_account_is_available "$SERVICE_USER"; then        info_msg "service account $SERVICE_USER available."    else        err_msg "service account $SERVICE_USER not available!"    fi    if go_is_available "$SERVICE_USER"; then        info_msg "~$SERVICE_USER: go is installed"    else        err_msg "~$SERVICE_USER: go is not installed"    fi    if filtron_is_installed; then        info_msg "~$SERVICE_USER: filtron app is installed"    else        err_msg "~$SERVICE_USER: filtron app is not installed!"    fi    if ! service_is_available "http://${FILTRON_API}"; then        err_msg "API not available at: http://${FILTRON_API}"    fi    if ! service_is_available "http://${FILTRON_LISTEN}" ; then        err_msg "Filtron is not listening on: http://${FILTRON_LISTEN}"    fi    if service_is_available "http://${FILTRON_TARGET}" ; then        info_msg "Filtron's target is available at: http://${FILTRON_TARGET}"    fi    if ! service_is_available "${PUBLIC_URL}"; then        warn_msg "Public service at ${PUBLIC_URL} is not available!"        if ! in_container; then            warn_msg "Check if public name is correct and routed or use the public IP from above."        fi    fi    if in_container; then        lxc_suite_info    else        info_msg "public URL   --> ${PUBLIC_URL}"        info_msg "internal URL --> http://${FILTRON_LISTEN}"    fi    local _debug_on    if ask_yn "Enable filtron debug mode?"; then        enable_debug        _debug_on=1    fi    echo    systemctl --no-pager -l status "${SERVICE_NAME}"    echo    info_msg "public URL --> ${PUBLIC_URL}"    # shellcheck disable=SC2059    printf "// use ${_BCyan}CTRL-C${_creset} to stop monitoring the log"    read -r -s -n1 -t 5    echo    while true;  do        trap break 2        journalctl -f -u "${SERVICE_NAME}"    done    if [[ $_debug_on == 1 ]]; then        disable_debug    fi    return 0}enable_debug() {    info_msg "try to enable debug mode ..."    python <<EOFimport sys, jsondebug = {    u'name': u'debug request'    , u'filters': []    , u'interval': 0    , u'limit': 0    , u'actions': [{u'name': u'log'}]}with open('$FILTRON_RULES') as rules:    j = json.load(rules)pos = Nonefor i in range(len(j)):    if j[i].get('name') == 'debug request':        pos = i        breakif pos is not None:    j[pos] = debugelse:    j.append(debug)with open('$FILTRON_RULES', 'w') as rules:    json.dump(j, rules, indent=2, sort_keys=True)EOF    systemctl restart "${SERVICE_NAME}.service"}disable_debug() {    info_msg "try to disable debug mode ..."    python <<EOFimport sys, jsonwith open('$FILTRON_RULES') as rules:    j = json.load(rules)pos = Nonefor i in range(len(j)):    if j[i].get('name') == 'debug request':        pos = i        breakif pos is not None:    del j[pos]    with open('$FILTRON_RULES', 'w') as rules:         json.dump(j, rules, indent=2, sort_keys=True)EOF    systemctl restart "${SERVICE_NAME}.service"}install_apache_site() {    rst_title "Install Apache site $APACHE_FILTRON_SITE"    rst_para "\This installs a reverse proxy (ProxyPass) into apache site (${APACHE_FILTRON_SITE})"    ! apache_is_installed && info_msg "Apache is not installed."    if ! ask_yn "Do you really want to continue?" Yn; then        return    else        install_apache    fi    "${REPO_ROOT}/utils/searx.sh" install uwsgi    apache_install_site --variant=filtron "${APACHE_FILTRON_SITE}"    info_msg "testing public url .."    if ! service_is_available "${PUBLIC_URL}"; then        err_msg "Public service at ${PUBLIC_URL} is not available!"    fi}remove_apache_site() {    rst_title "Remove Apache site $APACHE_FILTRON_SITE"    rst_para "\This removes apache site ${APACHE_FILTRON_SITE}."    ! apache_is_installed && err_msg "Apache is not installed."    if ! ask_yn "Do you really want to continue?" Yn; then        return    fi    apache_remove_site "$APACHE_FILTRON_SITE"}install_nginx_site() {    rst_title "Install nginx site $NGINX_FILTRON_SITE"    rst_para "\This installs a reverse proxy (ProxyPass) into nginx site (${NGINX_FILTRON_SITE})"    ! nginx_is_installed && info_msg "nginx is not installed."    if ! ask_yn "Do you really want to continue?" Yn; then        return    else        install_nginx    fi    "${REPO_ROOT}/utils/searx.sh" install uwsgi    # shellcheck disable=SC2034    SEARX_SRC=$("${REPO_ROOT}/utils/searx.sh" --getenv SEARX_SRC)    # shellcheck disable=SC2034    SEARX_URL_PATH=$("${REPO_ROOT}/utils/searx.sh" --getenv SEARX_URL_PATH)    nginx_install_app --variant=filtron "${NGINX_FILTRON_SITE}"    info_msg "testing public url .."    if ! service_is_available "${PUBLIC_URL}"; then        err_msg "Public service at ${PUBLIC_URL} is not available!"    fi}remove_nginx_site() {    rst_title "Remove nginx site $NGINX_FILTRON_SITE"    rst_para "\This removes nginx site ${NGINX_FILTRON_SITE}."    ! nginx_is_installed && err_msg "nginx is not installed."    if ! ask_yn "Do you really want to continue?" Yn; then        return    fi    nginx_remove_site "$FILTRON_FILTRON_SITE"}rst-doc() {    eval "echo \"$(< "${REPO_ROOT}/docs/build-templates/filtron.rst")\""    echo -e "\n.. START install systemd unit"    cat <<EOF.. tabs::   .. group-tab:: systemd      .. code:: bashEOF    eval "echo \"$(< "${TEMPLATES}/${SERVICE_SYSTEMD_UNIT}")\"" | prefix_stdout "         "    echo -e "\n.. END install systemd unit"    # for DIST_NAME in ubuntu-20.04 arch fedora centos; do    #     (    #         DIST_ID=${DIST_NAME%-*}    #         DIST_VERS=${DIST_NAME#*-}    #         [[ $DIST_VERS =~ $DIST_ID ]] && DIST_VERS=    #         # ...    #     )    # done}# ----------------------------------------------------------------------------main "$@"# ----------------------------------------------------------------------------
 |