diff --git a/.cspell/config.txt b/.cspell/config.txt index d672b2c5..c5b08d09 100644 --- a/.cspell/config.txt +++ b/.cspell/config.txt @@ -198,3 +198,11 @@ difftool apos hpdy tagname +Aftertabs +GKHF +cyclonedx +cpes +UNINDEXED +unindexed +logrus +JSONLINT diff --git a/.ecrc b/.ecrc new file mode 100644 index 00000000..d5485de8 --- /dev/null +++ b/.ecrc @@ -0,0 +1,20 @@ +{ + "Verbose": false, + "Debug": false, + "IgnoreDefaults": false, + "SpacesAftertabs": false, + "NoColor": false, + "Exclude": [ + "/testsData/" + ], + "AllowedContentTypes": [], + "PassedFiles": [], + "Disable": { + "EndOfLine": false, + "Indentation": false, + "IndentSize": false, + "InsertFinalNewline": false, + "TrimTrailingWhitespace": false, + "MaxLineLength": false + } +} diff --git a/.editorconfig b/.editorconfig index ef6f79f3..d2e3815a 100644 --- a/.editorconfig +++ b/.editorconfig @@ -9,3 +9,6 @@ indent_size = 2 indent_style = space insert_final_newline = true trim_trailing_whitespace = true + +[.vscode/*.json] +indent_size = unset diff --git a/.eslintrc.js b/.eslintrc.js index 7aa56036..baf95796 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -24,5 +24,21 @@ module.exports = { commonjs: true, node: true, }, - extends: ['eslint:recommended', 'plugin:prettier/recommended'], + plugins: ['json'], + extends: [ + 'eslint:recommended', + 'plugin:json/recommended', + 'eslint-config-prettier', + ], + rules: { + 'json/*': ['error', {allowComments: false}], + }, + overrides: [ + { + files: ['.vscode/*.json'], + rules: { + 'json/*': ['error', {allowComments: true}], + }, + }, + ], }; diff --git a/.framework-config b/.framework-config index 75f39500..919cd65d 100755 --- a/.framework-config +++ b/.framework-config @@ -2,11 +2,14 @@ # shellcheck disable=SC2034 BASH_TOOLS_ROOT_DIR="$(cd -- "$(dirname -- "${CURRENT_LOADED_ENV_FILE}")" &>/dev/null && pwd -P)" -FRAMEWORK_ROOT_DIR="${FRAMEWORK_ROOT_DIR:-${BASH_TOOLS_ROOT_DIR}/vendor/bash-tools-framework}" -FRAMEWORK_SRC_DIR="${FRAMEWORK_SRC_DIR:-${FRAMEWORK_ROOT_DIR}/src}" -FRAMEWORK_BIN_DIR="${FRAMEWORK_BIN_DIR:-${FRAMEWORK_ROOT_DIR}/bin}" -FRAMEWORK_VENDOR_DIR="${FRAMEWORK_VENDOR_DIR:-${FRAMEWORK_ROOT_DIR}/vendor}" -FRAMEWORK_VENDOR_BIN_DIR="${FRAMEWORK_VENDOR_BIN_DIR:-${FRAMEWORK_ROOT_DIR}/vendor/bin}" +FRAMEWORK_ROOT_DIR="${BASH_TOOLS_ROOT_DIR}/vendor/bash-tools-framework" +FRAMEWORK_SRC_DIR="${FRAMEWORK_ROOT_DIR}/src" +FRAMEWORK_BIN_DIR="${FRAMEWORK_ROOT_DIR}/bin" +FRAMEWORK_VENDOR_DIR="${BASH_TOOLS_ROOT_DIR}/vendor" +FRAMEWORK_VENDOR_BIN_DIR="${FRAMEWORK_VENDOR_DIR}/bin" + +# allows to generate bin file in the right directory +export BASH_TOOLS_ROOT_DIR # compile parameters # srcFile : file that needs to be compiled @@ -16,7 +19,7 @@ FRAMEWORK_VENDOR_BIN_DIR="${FRAMEWORK_VENDOR_BIN_DIR:-${FRAMEWORK_ROOT_DIR}/vend # srcDirs : additional directories where to find the functions COMPILE_PARAMETERS=( --src-dir "${BASH_TOOLS_ROOT_DIR}/src" - --src-dir "${FRAMEWORK_SRC_DIR}" + --src-dir "${FRAMEWORK_ROOT_DIR}/src" --bin-dir "${BASH_TOOLS_ROOT_DIR}/bin" --root-dir "${BASH_TOOLS_ROOT_DIR}" --template-dir "${BASH_TOOLS_ROOT_DIR}/src" @@ -40,6 +43,7 @@ fi # export here all the variables that will be used in your templates export REPOSITORY_URL="${REPOSITORY_URL:-https://github.com/fchastanet/bash-tools}" +SRC_FILE_PATH="${CURRENT_COMPILED_RELATIVE_FILE#/}" BASH_FRAMEWORK_THEME="${BASH_FRAMEWORK_THEME:-default}" BASH_FRAMEWORK_LOG_LEVEL="${BASH_FRAMEWORK_LOG_LEVEL:-0}" diff --git a/.github/workflows/docsify-gh-pages.yml b/.github/workflows/docsify-gh-pages.yml index 7c786cba..ce53e628 100644 --- a/.github/workflows/docsify-gh-pages.yml +++ b/.github/workflows/docsify-gh-pages.yml @@ -31,6 +31,7 @@ jobs: uses: actions/checkout@v4 - name: Set up Docker Buildx + # kics-scan ignore-line uses: docker/setup-buildx-action@v3 - name: docker pull image diff --git a/.github/workflows/lint-test.yml b/.github/workflows/lint-test.yml index 07b508c9..4f60bbde 100644 --- a/.github/workflows/lint-test.yml +++ b/.github/workflows/lint-test.yml @@ -9,6 +9,8 @@ on: # yamllint disable-line rule:truthy - '**' workflow_dispatch: +permissions: read-all + jobs: build: runs-on: ubuntu-22.04 @@ -55,17 +57,21 @@ jobs: runPrecommitTests: false steps: - name: Checkout + # kics-scan ignore-line uses: actions/checkout@v4 - name: Set up Docker Buildx + # kics-scan ignore-line uses: docker/setup-buildx-action@v3 - name: Login to DockerHub + # kics-scan ignore-line uses: docker/login-action@v3 with: username: ${{ secrets.DOCKERHUB_USERNAME }} password: ${{ secrets.DOCKERHUB_TOKEN }} + # kics-scan ignore-line - uses: ouzi-dev/commit-status-updater@v2 with: name: build bash-tools-${{matrix.vendor}}-${{matrix.bashTarVersion}} @@ -74,6 +80,7 @@ jobs: # only if pre-commit - name: Set up Python if: matrix.runPrecommitTests + # kics-scan ignore-line uses: actions/setup-python@v4 with: python-version: 3.9 @@ -82,6 +89,7 @@ jobs: if: matrix.runPrecommitTests run: pip install pre-commit + # kics-scan ignore-line - uses: ouzi-dev/commit-status-updater@v2 if: matrix.runPrecommitTests with: @@ -95,6 +103,7 @@ jobs: - name: Archive results if: matrix.runPrecommitTests && always() continue-on-error: true + # kics-scan ignore-line uses: actions/upload-artifact@v3 with: name: linter-reports @@ -103,6 +112,7 @@ jobs: - name: Create Pull Request if: matrix.runPrecommitTests && failure() + # kics-scan ignore-line uses: peter-evans/create-pull-request@v5 with: branch: update/pre-commit-fixes @@ -112,6 +122,7 @@ jobs: some auto fixes have been generated during pre-commit run labels: updates + # kics-scan ignore-line - uses: ouzi-dev/commit-status-updater@v2 if: matrix.runPrecommitTests && always() with: @@ -139,11 +150,13 @@ jobs: - name: Upload Test Results if: always() + # kics-scan ignore-line uses: actions/upload-artifact@v3 with: name: Test Results ${{matrix.vendor}} ${{matrix.bashTarVersion}} path: logs/report.xml + # kics-scan ignore-line - uses: ouzi-dev/commit-status-updater@v2 if: always() with: @@ -165,19 +178,23 @@ jobs: pull-requests: write steps: - name: Download Artifacts + # kics-scan ignore-line uses: actions/download-artifact@v3 with: path: artifacts - name: Checkstyle aggregation + # kics-scan ignore-line uses: lcollins/checkstyle-github-action@v2.0.0 with: path: 'artifacts/**/*.xml' # run this action to get the workflow conclusion # You can get the conclusion via env (env.WORKFLOW_CONCLUSION) + # kics-scan ignore-line - uses: technote-space/workflow-conclusion-action@v3 + # kics-scan ignore-line - uses: ouzi-dev/commit-status-updater@v2 with: name: build diff --git a/.github/workflows/precommit-autoupdate.yml b/.github/workflows/precommit-autoupdate.yml index e66e4db0..07fa3313 100644 --- a/.github/workflows/precommit-autoupdate.yml +++ b/.github/workflows/precommit-autoupdate.yml @@ -7,13 +7,18 @@ on: schedule: # https://crontab.cronhub.io/ - cron: '30 10 * * *' + +permissions: read-all + jobs: auto-update: runs-on: ubuntu-22.04 steps: + # kics-scan ignore-line - uses: actions/checkout@v4 - name: Set up Python + # kics-scan ignore-line uses: actions/setup-python@v4 with: python-version: 3.9 @@ -25,6 +30,7 @@ jobs: run: pre-commit autoupdate - name: Create Pull Request + # kics-scan ignore-line uses: peter-evans/create-pull-request@v5 with: branch: update/pre-commit-autoupdate diff --git a/.github/workflows/set-github-status-on-pr-approved.yml b/.github/workflows/set-github-status-on-pr-approved.yml index fa00d283..b2885b2b 100644 --- a/.github/workflows/set-github-status-on-pr-approved.yml +++ b/.github/workflows/set-github-status-on-pr-approved.yml @@ -6,13 +6,17 @@ on: pull_request_review: types: [submitted] +permissions: read-all + jobs: build: if: github.event.review.state == 'approved' runs-on: ubuntu-22.04 steps: + # kics-scan ignore-line - uses: actions/checkout@v4 - name: Run the action # You would run your tests before this using the output to set state/desc + # kics-scan ignore-line uses: Sibz/github-status-action@v1 with: authToken: ${{secrets.GITHUB_TOKEN}} diff --git a/.gitignore b/.gitignore index efb08e98..99231b08 100644 --- a/.gitignore +++ b/.gitignore @@ -16,3 +16,8 @@ bin/runBuildContainer /bin/test /bin/buildPushDockerImages /bin/buildPushDockerImage + +# node modules +node_modules/ +package*.json +yarn.lock diff --git a/.gitleaks.toml b/.gitleaks.toml new file mode 100644 index 00000000..0579adfd --- /dev/null +++ b/.gitleaks.toml @@ -0,0 +1,20 @@ + +title = "gitleaks config" + +[extend] +# useDefault will extend the base configuration with the default gitleaks config: +# https://github.com/zricethezav/gitleaks/blob/master/config/gitleaks.toml +useDefault = true + +[allowlist] + description = "Allowlisted files" + paths = [ + '''.automation/test''', + '''megalinter-reports''', + '''.github/linters''', + '''(.*?)/testsData/''', + '''(.*?)tests/data/''', + '''(.*?)tests/tools/data/''', + '''(.*?)gitleaks\.toml$''', + '''(.*?)(png|jpg|gif|doc|docx|pdf|bin|xls|pyc|zip)$''' + ] diff --git a/.grype.yaml b/.grype.yaml new file mode 100644 index 00000000..a1c0278a --- /dev/null +++ b/.grype.yaml @@ -0,0 +1,144 @@ +# enable/disable checking for application updates on startup +# same as GRYPE_CHECK_FOR_APP_UPDATE env var +# check-for-app-update: true + +# allows users to specify which image source should be used to generate the sbom +# valid values are: registry, docker, podman +# same as GRYPE_DEFAULT_IMAGE_PULL_SOURCE env var +# default-image-pull-source: "" + +# same as --name; set the name of the target being analyzed +# name: "" + +# upon scanning, if a severity is found at or above the given severity then the return code will be 1 +# default is unset which will skip this validation (options: negligible, low, medium, high, critical) +# same as --fail-on ; GRYPE_FAIL_ON_SEVERITY env var +fail-on-severity: 'high' + +# the output format of the vulnerability report (options: table, json, cyclonedx) +# same as -o ; GRYPE_OUTPUT env var +# output: "table" + +# suppress all output (except for the vulnerability list) +# same as -q ; GRYPE_QUIET env var +# quiet: false + +# write output report to a file (default is to write to stdout) +# same as --file; GRYPE_FILE env var +# file: "" + +# a list of globs to exclude from scanning, for example: +# exclude: +# - '/etc/**' +# - './out/**/*.json' +# same as --exclude ; GRYPE_EXCLUDE env var +# exclude: [] + +# os and/or architecture to use when referencing container images (e.g. "windows/armv6" or "arm64") +# same as --platform; GRYPE_PLATFORM env var +# platform: "" + +# If using SBOM input, automatically generate CPEs when packages have none +# add-cpes-if-none: false + +# Explicitly specify a linux distribution to use as : like alpine:3.10 +# distro: + +# db: +# check for database updates on execution +# same as GRYPE_DB_AUTO_UPDATE env var +# auto-update: true + +# location to write the vulnerability database cache +# same as GRYPE_DB_CACHE_DIR env var +# cache-dir: "$XDG_CACHE_HOME/grype/db" + +# URL of the vulnerability database +# same as GRYPE_DB_UPDATE_URL env var +# update-url: "https://toolbox-data.anchore.io/grype/databases/listing.json" + +# it ensures db build is no older than the max-allowed-built-age +# set to false to disable check +# validate-age: true + +# Max allowed age for vulnerability database, +# age being the time since it was built +# Default max age is 120h (or five days) +# max-allowed-built-age: "120h" + +# search: +# the search space to look for packages (options: all-layers, squashed) +# same as -s ; GRYPE_SEARCH_SCOPE env var +# scope: "squashed" + +# search within archives that do contain a file index to search against (zip) +# note: for now this only applies to the java package cataloger +# same as GRYPE_PACKAGE_SEARCH_INDEXED_ARCHIVES env var +# indexed-archives: true + +# search within archives that do not contain a file index to search against (tar, tar.gz, tar.bz2, etc) +# note: enabling this may result in a performance impact since all discovered compressed tars will be decompressed +# note: for now this only applies to the java package cataloger +# same as GRYPE_PACKAGE_SEARCH_UNINDEXED_ARCHIVES env var +# unindexed-archives: false + +# options when pulling directly from a registry via the "registry:" scheme +# registry: +# skip TLS verification when communicating with the registry +# same as GRYPE_REGISTRY_INSECURE_SKIP_TLS_VERIFY env var +# insecure-skip-tls-verify: false +# use http instead of https when connecting to the registry +# same as GRYPE_REGISTRY_INSECURE_USE_HTTP env var +# insecure-use-http: false + +# credentials for specific registries +# auth: +# - # the URL to the registry (e.g. "docker.io", "localhost:5000", etc.) +# same as GRYPE_REGISTRY_AUTH_AUTHORITY env var +# authority: "" +# same as GRYPE_REGISTRY_AUTH_USERNAME env var +# username: "" +# same as GRYPE_REGISTRY_AUTH_PASSWORD env var +# password: "" +# note: token and username/password are mutually exclusive +# same as GRYPE_REGISTRY_AUTH_TOKEN env var +# token: "" +# - ... # note, more credentials can be provided via config file only + +# log: +# use structured logging +# same as GRYPE_LOG_STRUCTURED env var +# structured: false + +# the log level; note: detailed logging suppress the ETUI +# same as GRYPE_LOG_LEVEL env var +# Uses logrus logging levels: https://github.com/sirupsen/logrus#level-logging +# level: "error" + +# location to write the log file (default is not to have a log file) +# same as GRYPE_LOG_FILE env var +# file: "" + +# match: +# sets the matchers below to use cpes when trying to find +# vulnerability matches. The stock matcher is the default +# when no primary matcher can be identified +# java: +# using-cpes: true +# python: +# using-cpes: true +# javascript: +# using-cpes: true +# ruby: +# using-cpes: true +# dotnet: +# using-cpes: true +# golang: +# using-cpes: true +# stock: +# using-cpes: true + +ignore: + # Ignored by default; disputed and unwarranted CVE that causes Megalinter to fail + # @link https://nvd.nist.gov/vuln/detail/CVE-2018-20225 + - vulnerability: CVE-2018-20225 diff --git a/.lycheeignore b/.lycheeignore index 2b8a17b1..089d95dd 100644 --- a/.lycheeignore +++ b/.lycheeignore @@ -13,3 +13,4 @@ https://github.com/plantuml/plantuml/releases/download/v@version@/plantuml-@vers https://github.com/docker/docker-credential-helpers/releases/download/v@version@/docker-credential-wincred-v@version@.windows* https://github.com/Versent/saml2aws/releases/download/v@version@/saml2aws_@version@_linux_amd64.tar.gz https://github.com/fchastanet/bash-tools/tree/master/src/_binaries* +Commands.md diff --git a/.markdown-link-check.json b/.markdown-link-check.json index 642f3823..133fa58c 100644 --- a/.markdown-link-check.json +++ b/.markdown-link-check.json @@ -4,6 +4,9 @@ "ignorePatterns": [ { "pattern": "^http://localhost:.*/" + }, + { + "pattern": "^Commands.md$" } ] } diff --git a/.mega-linter-githubAction.yml b/.mega-linter-githubAction.yml new file mode 100644 index 00000000..9a3ef9c0 --- /dev/null +++ b/.mega-linter-githubAction.yml @@ -0,0 +1,4 @@ +--- +EXTENDS: + - .mega-linter.yml +SHOW_ELAPSED_TIME: false diff --git a/.mega-linter-light.yml b/.mega-linter-light.yml index 4378a2da..40144cf8 100644 --- a/.mega-linter-light.yml +++ b/.mega-linter-light.yml @@ -15,3 +15,4 @@ DISABLE_LINTERS: - REPOSITORY_TRUFFLEHOG # REPOSITORY_GRYPE disabled because too slow - REPOSITORY_GRYPE + - JSON_JSONLINT diff --git a/.mega-linter.yml b/.mega-linter.yml index b1f13524..8e0aba5e 100644 --- a/.mega-linter.yml +++ b/.mega-linter.yml @@ -18,23 +18,28 @@ DISABLE_LINTERS: - SPELL_VALE - SPELL_LYCHEE - SPELL_PROSELINT + - JSON_JSONLINT # DISABLE_ERRORS: true # Uncomment if you want MegaLinter to detect errors but not block CI to pass EXCLUDED_DIRECTORIES: - '.history' - '.git' - '.idea' - 'logs' + - 'node_modules' FILTER_REGEX_EXCLUDE: | (?x)( \.git/| \.history/| \.idea/| - ^vendor/ + ^vendor/| + ^node_modules/ ) FILEIO_REPORTER: false PRE_COMMANDS: - command: env cwd: 'workspace' + - command: yarn add --dev eslint-plugin-json eslint-config-prettier + cwd: 'workspace' POST_COMMANDS: # FIX files set as root user # HOST_USER_ID and HOST_GROUP_ID set in package.json @@ -66,13 +71,11 @@ CREDENTIALS_SECRETLINT_CONFIG_FILE: .secretlintrc.yml EDITORCONFIG_EDITORCONFIG_CHECKER_FILTER_REGEX_EXCLUDE: | (?x)( \.git/| - /testsData/.*\.(txt|help)| + /testsData/| ^manualTests/data/| - ^src/Log/testsData| - ^Gemfile.lock$| bin/bash-tpl| ^doc/guides/Options/generate.*\.md$| - ^pages/Commands.md + ^pages/Commands.md| ^.*-megalinter_file_names_cspell.txt ) @@ -95,9 +98,14 @@ JAVASCRIPT_DEFAULT_STYLE: prettier JAVASCRIPT_ES_CONFIG_FILE: .eslintrc.js JAVASCRIPT_ES_FILTER_REGEX_EXCLUDE: (report/) -JSON_JSONLINT_FILTER_REGEX_EXCLUDE: (.vscode/) - -MARKDOWN_MARKDOWN_LINK_CHECK_FILTER_REGEX_EXCLUDE: (report) +MARKDOWN_MARKDOWN_LINK_CHECK_FILTER_REGEX_EXCLUDE: (report/) SPELL_CSPELL_FILTER_REGEX_EXCLUDE: (\.git/|\.history/|IDE/.*/.idea|.*-megalinter_file_names_cspell.txt) SPELL_FILTER_REGEX_EXCLUDE: (\.git/|\.history/|IDE/.*/.idea) + +SPELL_LYCHEE_FILTER_REGEX_EXCLUDE: | + (?x)( + ^pages/Commands.md$ + ) + +JSON_ESLINT_PLUGIN_JSONC_FILE_NAME: .eslintrc.js diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index c73fcf60..92b6068e 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -63,7 +63,7 @@ repos: exclude: /testsData/ - repo: https://github.com/fchastanet/bash-tools-framework - rev: 2.0.2 + rev: 2.1.0 hooks: - id: fixShebangExecutionBit - id: fixShebangExecutionBitGithubActions @@ -98,6 +98,7 @@ repos: - id: buildShFiles - id: buildShFilesGithubAction - id: runUnitTests + args: [-r, src, -j, '1'] # not manual as github will run UT with several versions of bash and arch stages: [pre-commit] - id: megalinterCheckVersion diff --git a/.secretlintrc.json b/.secretlintrc.json new file mode 100644 index 00000000..7a1a5df3 --- /dev/null +++ b/.secretlintrc.json @@ -0,0 +1,7 @@ +{ + "rules": [ + { + "id": "@secretlint/secretlint-rule-preset-recommend" + } + ] +} diff --git a/bin/cli b/bin/cli index ecfad827..f233db79 100755 --- a/bin/cli +++ b/bin/cli @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/cli +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/cli # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1180,6 +1203,7 @@ cliCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1472,11 +1496,11 @@ cliCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "easy connection to docker container")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "easy connection to docker container")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1485,89 +1509,93 @@ cliCommand() { local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(containerArgHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " [${__HELP_OPTION_COLOR}user${__HELP_NORMAL} {single}]" local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(userArgHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " [${__HELP_OPTION_COLOR}commandArg${__HELP_NORMAL} {single}]" local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(commandArgHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}AVAILABLE PROFILES (from ${PROFILES_DIR})${__HELP_NORMAL} This list can be overridden in ${HOME_PROFILES_DIR} @@ -1603,7 +1631,7 @@ variables ${__HELP_OPTION_COLOR}finalUserArg${__HELP_NORMAL}, ${__HELP_OPTION_CO echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 @@ -1613,36 +1641,36 @@ variables ${__HELP_OPTION_COLOR}finalUserArg${__HELP_NORMAL}, ${__HELP_OPTION_CO containerArgHelpCallback() { Conf::load "cliProfiles" "default" echo "container should be the name of a profile from profile list," - echo "check containers list below." $'\n' - echo "If not provided, it will load the container specified in default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' + echo "check containers list below." + echo "If not provided, it will load the container specified in default configuration." + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" echo "Default container: ${__HELP_OPTION_COLOR}${finalContainerArg}${__HELP_NORMAL}" } userArgHelpCallback() { Conf::load "cliProfiles" "default" echo "user to connect on this container" $'\n' - echo "Default user: ${__HELP_OPTION_COLOR}${finalUserArg}${__HELP_NORMAL}" $'\n' - echo " loaded from profile selected as first arg" $'\n' + echo "Default user: ${__HELP_OPTION_COLOR}${finalUserArg}${__HELP_NORMAL}" + echo " loaded from profile selected as first arg" echo " or deduced from default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' echo "if first arg is not a profile" } commandArgHelpCallback() { Conf::load "cliProfiles" "default" echo "The command to execute" $'\n' - echo "Default command: ${__HELP_OPTION_COLOR}${finalCommandArg[*]}${__HELP_NORMAL}" $'\n' - echo " loaded from profile selected as first arg" $'\n' - echo " or deduced from default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" + echo "Default command: ${__HELP_OPTION_COLOR}${finalCommandArg[*]}${__HELP_NORMAL}" + echo " loaded from profile selected as first arg" + echo " or deduced from default configuration." + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' echo "if first arg is not a profile" } optionHelpCallback() { local containers # shellcheck disable=SC2046 - containers="$(Array::wrap ", " 80 0 $(docker ps --format '{{.Names}}'))" + containers="$(Array::wrap2 ", " 80 0 $(docker ps --format '{{.Names}}'))" local profilesList="" Conf::load "cliProfiles" "default" diff --git a/bin/dbImport b/bin/dbImport index 8669380f..db0da220 100755 --- a/bin/dbImport +++ b/bin/dbImport @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImport +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImport # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -110,123 +110,146 @@ Array::contains() { # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1544,6 +1567,7 @@ dbImportCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1961,11 +1985,11 @@ dbImportCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Import source db into target db using eventual table filter")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Import source db into target db using eventual table filter")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--profile|-p ]" "[--tables ]" "[--from-dsn|-f ]" "[--skip-schema|-s]" "[--from-aws|-a ]" "[--target-dsn|-t ]" "[--character-set|-c ]" "[--collation-name|-o ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1974,130 +1998,134 @@ dbImportCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ source/remote\ database) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " [${__HELP_OPTION_COLOR}targetDbName${__HELP_NORMAL} {single}]" local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ target\ database\,\ use\ fromDbName\(without\ extension\)\ if\ not\ provided) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}PROFILE OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--profile${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-p ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(profileOptionHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--tables ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(import\ only\ table\ specified\ in\ the\ list.\ \ If\ aws\ mode\,\ ignore\ profile\ option) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}FROM OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--from-dsn${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-f ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'dsn to use for source database\n\\n this option is incompatible with -a|--from-aws option') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'dsn to use for source database\n\n this option is incompatible with -a|--from-aws option') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--skip-schema${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-s${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(avoid\ to\ import\ the\ schema) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--from-aws${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-a ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(db\ dump\ will\ be\ downloaded\ from\ s3\ instead\ of\ using\ remote\ db.\ The\ value\ \\ is\ the\ name\ of\ the\ file\ without\ s3\ location\ \(Only\ .gz\ or\ tar.gz\ file\ are\ supported\).\ This\ option\ is\ incompatible\ with\ -f\|--from-dsn\ option) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}TARGET OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--target-dsn${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-t ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(dsn\ to\ use\ for\ target\ database\ \(Default:\ default.local\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--character-set${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-c ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(change\ the\ character\ set\ used\ during\ database\ creation\ \(default\ value:\ utf8\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--collation-name${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-o ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(change\ the\ collation\ name\ used\ during\ database\ creation\ \(default\ value:\ utf8_general_ci\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}Default profiles directory:${__HELP_NORMAL} ${PROFILES_DIR-configuration error} @@ -2133,7 +2161,7 @@ ${__HELP_EXAMPLE}TODO${__HELP_NORMAL}""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/dbImportProfile b/bin/dbImportProfile index c5ac70d5..dd8279d9 100755 --- a/bin/dbImportProfile +++ b/bin/dbImportProfile @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImportProfile +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImportProfile # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1273,6 +1296,7 @@ dbImportProfileCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1600,11 +1624,11 @@ dbImportProfileCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "generate optimized profiles to be used by dbImport")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "generate optimized profiles to be used by dbImport")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--profile|-p ]" "[--from-dsn|-f ]" "[--ratio|-r ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1613,96 +1637,100 @@ dbImportProfileCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ source/remote\ database) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--profile${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-p ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ profile\ to\ write\ in\ profiles\ directory.\ \ If\ not\ provided\,\ the\ file\ name\ pattern\ will\ be\ \'auto_\_\.sh\') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--from-dsn${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-f ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(dsn\ to\ use\ for\ source\ database\ \(Default:\ default.remote\)\ if\ not\ provided\,\ the\ file\ name\ pattern\ will\ be\ \'auto_\_\.sh\') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--ratio${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-r ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(define\ the\ ratio\ to\ use\ \(0\ to\ 100%\ -\ default\ 70\).\ \ 0\ means\ profile\ will\ filter\ out\ all\ the\ tables.\ \ 100\ means\ profile\ will\ keep\ all\ the\ tables.\ \ Eg:\ 70\ means\ that\ tables\ with\ size\(table+index\)\ that\ are\ greater\ that\ 70%\ of\ the\ max\ table\ size\ will\ be\ excluded.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}Default profiles directory:${__HELP_NORMAL} ${PROFILES_DIR-configuration error} @@ -1729,7 +1757,7 @@ ${dsnList}""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/dbImportStream b/bin/dbImportStream index 4b213b9b..9c822392 100755 --- a/bin/dbImportStream +++ b/bin/dbImportStream @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImportStream +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImportStream # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -110,123 +110,146 @@ Array::contains() { # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1433,6 +1456,7 @@ dbImportStreamCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1795,11 +1819,11 @@ dbImportStreamCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "stream tar.gz file or gz file through mysql")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "stream tar.gz file or gz file through mysql")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--profile|-p ]" "[--tables ]" "[--target-dsn|-t ]" "[--character-set|-c ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1808,108 +1832,112 @@ dbImportStreamCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ of\ the\ file\ that\ will\ be\ streamed\ through\ mysql) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}argTargetDbName${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ mysql\ target\ database) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}PROFILE OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--profile${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-p ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(profileOptionHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--tables ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(import\ only\ table\ specified\ in\ the\ list.\ \ If\ aws\ mode\,\ ignore\ profile\ option) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}TARGET OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--target-dsn${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-t ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(dsn\ to\ use\ for\ target\ database\ \(Default:\ default.local\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--character-set${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-c ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(change\ the\ character\ set\ used\ during\ database\ creation\ \(default\ value:\ utf8\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}Default profiles directory:${__HELP_NORMAL} ${PROFILES_DIR-configuration error} @@ -1936,7 +1964,7 @@ ${dsnList}""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/dbQueryAllDatabases b/bin/dbQueryAllDatabases index 40ff9db9..964884c0 100755 --- a/bin/dbQueryAllDatabases +++ b/bin/dbQueryAllDatabases @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbQueryAllDatabases +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbQueryAllDatabases # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # EMBED Db::queryOneDatabase as dbQueryOneDatabase @@ -95,123 +95,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1094,8 +1117,8 @@ Log::logSkipped() { # @require Compiler::Embed::requireEmbedBinDir -declare -gx embed_function_DbQueryOneDatabase="${PERSISTENT_TMPDIR:-/tmp}/bin/f94f442fbdd0b7f3ff7df80421405d82/dbQueryOneDatabase" -declare -gx encoded_binary_file_DbQueryOneDatabase="#!/usr/bin/env bash
###############################################################################
# GENERATED FACADE
# DO NOT EDIT IT
# @generated
###############################################################################
# shellcheck disable=SC2288,SC2034
# BIN_FILE=${BIN_FILE}
# FACADE

# ensure that no user aliases could interfere with
# commands used in this script
unalias -a || true
shopt -u expand_aliases

# shellcheck disable=SC2034
((failures = 0)) || true

# Bash will remember & return the highest exit code in a chain of pipes.
# This way you can catch the error inside pipes, e.g. mysqldump | gzip
set -o pipefail
set -o errexit

# Command Substitution can inherit errexit option since bash v4.4
shopt -s inherit_errexit || true

# a log is generated when a command fails
set -o errtrace

# use nullglob so that (file*.php) will return an empty array if no file matches the wildcard
shopt -s nullglob

# ensure regexp are interpreted without accentuated characters
export LC_ALL=POSIX

export TERM=xterm-256color

# avoid interactive install
export DEBIAN_FRONTEND=noninteractive
export DEBCONF_NONINTERACTIVE_SEEN=true

# store command arguments for later usage
# shellcheck disable=SC2034
declare -a BASH_FRAMEWORK_ARGV=("$@")
# shellcheck disable=SC2034
declare -a ORIGINAL_BASH_FRAMEWORK_ARGV=("$@")

# @see https://unix.stackexchange.com/a/386856
# shellcheck disable=SC2317
interruptManagement() {
  # restore SIGINT handler
  trap - INT
  # ensure that Ctrl-C is trapped by this script and not by sub process
  # report to the parent that we have indeed been interrupted
  kill -s INT "$$"
}
trap interruptManagement INT
SCRIPT_NAME=${0##*/}
REAL_SCRIPT_FILE="$(readlink -e "$(realpath "${BASH_SOURCE[0]}")")"
if [[ -n "${EMBED_CURRENT_DIR}" ]]; then
  CURRENT_DIR="${EMBED_CURRENT_DIR}"
else
  CURRENT_DIR="$(cd "$(readlink -e "${REAL_SCRIPT_FILE%/*}")" && pwd -P)"
fi

################################################
# Temp dir management
################################################

KEEP_TEMP_FILES="${KEEP_TEMP_FILES:-0}"
export KEEP_TEMP_FILES

# PERSISTENT_TMPDIR is not deleted by traps
PERSISTENT_TMPDIR="${TMPDIR:-/tmp}/bash-framework"
export PERSISTENT_TMPDIR
mkdir -p "${PERSISTENT_TMPDIR}"

# shellcheck disable=SC2034
TMPDIR="$(mktemp -d -p "${PERSISTENT_TMPDIR:-/tmp}" -t bash-framework-$$-XXXXXX)"
export TMPDIR

# temp dir cleaning
# shellcheck disable=SC2317
cleanOnExit() {
  if [[ "${KEEP_TEMP_FILES:-0}" = "1" ]]; then
    Log::displayInfo "KEEP_TEMP_FILES=1 temp files kept here '${TMPDIR}'"
  elif [[ -n "${TMPDIR+xxx}" ]]; then
    Log::displayDebug "KEEP_TEMP_FILES=0 removing temp files '${TMPDIR}'"
    rm -Rf "${TMPDIR:-/tmp/fake}" >/dev/null 2>&1
  fi
}
trap cleanOnExit EXIT HUP QUIT ABRT TERM

# VAR_MAIN_FUNCTION_VAR_NAME=dbQueryAllDatabasesFacade

# @description used to execute given query when using
# dbScriptAllDatabases
# @arg $1 dsn:String
# @arg $2 db:String
# @env query String
# @env optionSeparator String
# @require Linux::requireExecutedAsUser
Db::queryOneDatabase() {
  # query and optionSeparator are passed via export
  local dsn="$1"
  local db="$2"

  local -A dbInstance
  Database::newInstance dbInstance "${dsn}"
  Database::setQueryOptions dbInstance "${dbInstance[QUERY_OPTIONS]} --connect-timeout=5"

  # identify columns header
  echo -n "@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@"
  Database::skipColumnNames dbInstance 0

  # shellcheck disable=SC2154
  if ! Database::query dbInstance "${query}" "${db}" | sed "s/\t/${optionSeparator}/g"; then
    Log::fatal "database ${db} error" 1>&2
  fi
}

# @description Log namespace provides 2 kind of functions
# - Log::display* allows to display given message with
#   given display level
# - Log::log* allows to log given message with
#   given log level
# Log::display* functions automatically log the message too
# @see Env::requireLoad to load the display and log level from .env file

# @description log level off
export __LEVEL_OFF=0
# @description log level error
export __LEVEL_ERROR=1
# @description log level warning
export __LEVEL_WARNING=2
# @description log level info
export __LEVEL_INFO=3
# @description log level success
export __LEVEL_SUCCESS=3
# @description log level debug
export __LEVEL_DEBUG=4

# @description verbose level off
export __VERBOSE_LEVEL_OFF=0
# @description verbose level info
export __VERBOSE_LEVEL_INFO=1
# @description verbose level info
export __VERBOSE_LEVEL_DEBUG=2
# @description verbose level info
export __VERBOSE_LEVEL_TRACE=3

# @description Display message using debug color (grey)
# @arg $1 message:String the message to display
Log::displayDebug() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_DEBUG)); then
    echo -e "${__DEBUG_COLOR}DEBUG   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logDebug "$1"
}

# @description Display message using info color (bg light blue/fg white)
# @arg $1 message:String the message to display
Log::displayInfo() {
  local type="${2:-INFO}"
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__INFO_COLOR}${type}    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logInfo "$1" "${type}"
}

# @description ensure COMMAND_BIN_DIR env var is set
# and PATH correctly prepared
# @noargs
# @set COMMAND_BIN_DIR string the directory where to find this command
# @set PATH string add directory where to find this command binary
Compiler::Facade::requireCommandBinDir() {
  COMMAND_BIN_DIR="${CURRENT_DIR}"
  Env::pathPrepend "${COMMAND_BIN_DIR}"
}

# @description create a new db instance
# Returns immediately if the instance is already initialized
#
# @arg $1 instanceNewInstance:&Map<String,String> (passed by reference) database instance to use
# @arg $2 dsn:String dsn profile - load the dsn.env profile deduced using rules defined in Conf::getAbsoluteFile
#
# @example
#   declare -Agx dbInstance
#   Database::newInstance dbInstance "default.local"
#
# @exitcode 1 if dns file not able to loaded
Database::newInstance() {
  local -n instanceNewInstance=$1
  local dsn="$2"
  local DSN_FILE

  if [[ -v instanceNewInstance['INITIALIZED'] && "${instanceNewInstance['INITIALIZED']:-0}" == "1" ]]; then
    return
  fi

  # final auth file generated from dns file
  instanceNewInstance['AUTH_FILE']=""
  instanceNewInstance['DSN_FILE']=""

  # check dsn file
  DSN_FILE="$(Conf::getAbsoluteFile "dsn" "${dsn}" "env")" || return 1
  Database::checkDsnFile "${DSN_FILE}" || return 1
  instanceNewInstance['DSN_FILE']="${DSN_FILE}"

  # shellcheck source=/src/Database/testsData/dsn_valid.env
  source "${instanceNewInstance['DSN_FILE']}"

  instanceNewInstance['USER']="${USER}"
  instanceNewInstance['PASSWORD']="${PASSWORD}"
  instanceNewInstance['HOSTNAME']="${HOSTNAME}"
  instanceNewInstance['PORT']="${PORT}"

  # generate authFile for easy authentication
  instanceNewInstance['AUTH_FILE']=$(mktemp -p "${TMPDIR:-/tmp}" -t "mysql.XXXXXXXXXXXX")
  (
    echo "[client]"
    echo "user = ${USER}"
    echo "password = ${PASSWORD}"
    echo "host = ${HOSTNAME}"
    echo "port = ${PORT}"
  ) >"${instanceNewInstance['AUTH_FILE']}"

  # some of those values can be overridden using the dsn file
  # SKIP_COLUMN_NAMES enabled by default
  instanceNewInstance['SKIP_COLUMN_NAMES']="${SKIP_COLUMN_NAMES:-1}"
  instanceNewInstance['SSL_OPTIONS']="${MYSQL_SSL_OPTIONS:---ssl-mode=DISABLED}"
  instanceNewInstance['QUERY_OPTIONS']="${MYSQL_QUERY_OPTIONS:---batch --raw --default-character-set=utf8}"
  instanceNewInstance['DUMP_OPTIONS']="${MYSQL_DUMP_OPTIONS:---default-character-set=utf8 --compress --hex-blob --routines --triggers --single-transaction --set-gtid-purged=OFF --column-statistics=0 ${instanceNewInstance['SSL_OPTIONS']}}"
  instanceNewInstance['DB_IMPORT_OPTIONS']="${DB_IMPORT_OPTIONS:---connect-timeout=5 --batch --raw --default-character-set=utf8}"

  instanceNewInstance['INITIALIZED']=1
}

# @description mysql query on a given db
# @warning could use QUERY_OPTIONS variable from dsn if defined
# @example
#   cat file.sql | Database::query ...
# @arg $1 instanceQuery:&Map<String,String> (passed by reference) database instance to use
# @arg $2 sqlQuery:String (optional) sql query or sql file to execute. if not provided or empty, the command can be piped
# @arg $3 dbName:String (optional) the db name
#
# @exitcode mysql command status code
Database::query() {
  local -n instanceQuery=$1
  local -a mysqlCommand=()
  local -a queryOptions

  mysqlCommand+=(mysql)
  mysqlCommand+=("--defaults-extra-file=${instanceQuery['AUTH_FILE']}")
  IFS=' ' read -r -a queryOptions <<<"${instanceQuery['QUERY_OPTIONS']}"
  mysqlCommand+=("${queryOptions[@]}")
  if [[ "${instanceQuery['SKIP_COLUMN_NAMES']}" = "1" ]]; then
    mysqlCommand+=("-s" "--skip-column-names")
  fi
  # add optional db name
  if [[ -n "${3+x}" ]]; then
    mysqlCommand+=("$3")
  fi
  # add optional sql query
  if [[ -n "${2+x}" && -n "$2" && ! -f "$2" ]]; then
    mysqlCommand+=("-e")
    mysqlCommand+=("$2")
  fi
  Log::displayDebug "$(printf "execute command: '%s'" "${mysqlCommand[*]}")"

  if [[ -f "$2" ]]; then
    "${mysqlCommand[@]}" <"$2"
  else
    "${mysqlCommand[@]}"
  fi
}

# @description set the general options to use on mysql command to query the database
# Differs than setOptions in the way that these options could change each time
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 optionsList:String query options list
Database::setQueryOptions() {
  local -n instanceSetQueryOptions=$1
  # shellcheck disable=SC2034
  instanceSetQueryOptions['QUERY_OPTIONS']="$2"
}

# @description by default we skip the column names
# but sometimes we need column names to display some results
# disable this option temporarily and then restore it to true
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 skipColumnNames:Boolean 0 to disable, 1 to enable (hide column names)
Database::skipColumnNames() {
  local -n instanceSkipColumnNames=$1
  # shellcheck disable=SC2034
  instanceSkipColumnNames['SKIP_COLUMN_NAMES']="$2"
}

# @description prepend directories to the PATH environment variable
# @arg $@ args:String[] list of directories to prepend
# @set PATH update PATH with the directories prepended
Env::pathPrepend() {
  local arg
  for arg in "$@"; do
    if [[ -d "${arg}" && ":${PATH}:" != *":${arg}:"* ]]; then
      PATH="$(realpath "${arg}"):${PATH}"
    fi
  done
}

# @description Display message using error color (red) and exit immediately with error status 1
# @arg $1 message:String the message to display
Log::fatal() {
  echo -e "${__ERROR_COLOR}FATAL   - ${1}${__RESET_COLOR}" >&2
  Log::logFatal "$1"
  exit 1
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logDebug() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_DEBUG)); then
    Log::logMessage "${2:-DEBUG}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logInfo() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-INFO}" "$1"
  fi
}

# @description ensure running user is not root
# @exitcode 1 if current user is root
# @stderr diagnostics information is displayed
Linux::requireExecutedAsUser() {
  if [[ "$(id -u)" = "0" ]]; then
    Log::fatal "this script should be executed as normal user"
  fi
}

# @description get absolute conf file from specified conf folder deduced using these rules
#   * from absolute file (ignores <confFolder> and <extension>)
#   * relative to where script is executed (ignores <confFolder> and <extension>)
#   * from home/.bash-tools/<confFolder>
#   * from framework conf/<confFolder>
#
# @arg $1 confFolder:String the directory name (not the path) to list
# @arg $2 conf:String file to use without extension
# @arg $3 extension:String the extension (.sh by default)
#
# @stdout absolute conf filename
# @exitcode 1 if file is not found in any location
Conf::getAbsoluteFile() {
  local confFolder="$1"
  local conf="$2"
  local extension="${3-.sh}"
  if [[ -n "${extension}" && "${extension:0:1}" != "." ]]; then
    extension=".${extension}"
  fi

  testAbs() {
    local result
    result="$(realpath -e "$1" 2>/dev/null)"
    # shellcheck disable=SC2181
    if [[ "$?" = "0" && -f "${result}" ]]; then
      echo "${result}"
      return 0
    fi
    return 1
  }

  # conf is absolute file (including extension)
  testAbs "${confFolder}${extension}" && return 0
  # conf is absolute file
  testAbs "${confFolder}" && return 0
  # conf is absolute file (including extension)
  testAbs "${conf}${extension}" && return 0
  # conf is absolute file
  testAbs "${conf}" && return 0

  # relative to where script is executed (including extension)
  if [[ -n "${CURRENT_DIR+xxx}" ]]; then
    testAbs "$(File::concatenatePath "${CURRENT_DIR}" "${confFolder}")/${conf}${extension}" && return 0
  fi
  # from home/.bash-tools/<confFolder>
  testAbs "$(File::concatenatePath "${HOME}/.bash-tools" "${confFolder}")/${conf}${extension}" && return 0

  if [[ -n "${FRAMEWORK_ROOT_DIR+xxx}" ]]; then
    # from framework conf/<confFolder> (including extension)
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}${extension}" && return 0

    # from framework conf/<confFolder>
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}" && return 0
  fi

  # file not found
  Log::displayError "conf file '${conf}' not found"

  return 1
}

# @description check if dsn file has all the mandatory variables set
# Mandatory variables are: HOSTNAME, USER, PASSWORD, PORT
#
# @arg $1 dsnFileName:String dsn absolute filename
# @set HOSTNAME loaded from dsn file
# @set PORT loaded from dsn file
# @set USER loaded from dsn file
# @set PASSWORD loaded from dsn file
# @exitcode 0 on valid file
# @exitcode 1 if one of the properties of the conf file is invalid or if file not found
# @stderr log output if error found in conf file
Database::checkDsnFile() {
  local dsnFileName="$1"
  if [[ ! -f "${dsnFileName}" ]]; then
    Log::displayError "dsn file ${dsnFileName} not found"
    return 1
  fi

  (
    unset HOSTNAME PORT PASSWORD USER
    # shellcheck source=/src/Database/testsData/dsn_valid.env
    source "${dsnFileName}"
    if [[ -z ${HOSTNAME+x} ]]; then
      Log::displayError "dsn file ${dsnFileName} : HOSTNAME not provided"
      return 1
    fi
    if [[ -z "${HOSTNAME}" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : HOSTNAME value not provided"
    fi
    if [[ "${HOSTNAME}" = "localhost" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : check that HOSTNAME should not be 127.0.0.1 instead of localhost"
    fi
    if [[ -z "${PORT+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT not provided"
      return 1
    fi
    if ! [[ ${PORT} =~ ^[0-9]+$ ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT invalid"
      return 1
    fi
    if [[ -z "${USER+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : USER not provided"
      return 1
    fi
    if [[ -z "${PASSWORD+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PASSWORD not provided"
      return 1
    fi
  )
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logFatal() {
  Log::logMessage "${2:-FATAL}" "$1"
}

# @description Internal: common log message
# @example text
#   [date]|[levelMsg]|message
#
# @example text
#   2020-01-19 19:20:21|ERROR  |log error
#   2020-01-19 19:20:21|SKIPPED|log skipped
#
# @arg $1 levelMsg:String message's level description (eg: STATUS, ERROR, ...)
# @arg $2 msg:String the message to display
# @env BASH_FRAMEWORK_LOG_FILE String log file to use, do nothing if empty
# @env BASH_FRAMEWORK_LOG_LEVEL int log level log only if > OFF or fatal messages
# @stderr diagnostics information is displayed
# @require Env::requireLoad
# @require Log::requireLoad
Log::logMessage() {
  local levelMsg="$1"
  local msg="$2"
  local date

  if [[ -n "${BASH_FRAMEWORK_LOG_FILE}" ]] && ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    date="$(date '+%Y-%m-%d %H:%M:%S')"
    touch "${BASH_FRAMEWORK_LOG_FILE}"
    printf "%s|%7s|%s\n" "${date}" "${levelMsg}" "${msg}" >>"${BASH_FRAMEWORK_LOG_FILE}"
  fi
}

# @description concatenate 2 paths and ensure the path is correct using realpath -m
# @arg $1 basePath:String
# @arg $2 subPath:String
# @require Linux::requireRealpathCommand
File::concatenatePath() {
  local basePath="$1"
  local subPath="$2"
  local fullPath="${basePath:+${basePath}/}${subPath}"

  realpath -m "${fullPath}" 2>/dev/null
}

# @description Display message using error color (red)
# @arg $1 message:String the message to display
Log::displayError() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_ERROR)); then
    echo -e "${__ERROR_COLOR}ERROR   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logError "$1"
}

# @description Display message using warning color (yellow)
# @arg $1 message:String the message to display
Log::displayWarning() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_WARNING)); then
    echo -e "${__WARNING_COLOR}WARN    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logWarning "$1"
}

# @description ensure env files are loaded
# @arg $@ list of default files to load at the end
# @exitcode 1 if one of env files fails to load
# @stderr diagnostics information is displayed
# shellcheck disable=SC2120
Env::requireLoad() {
  local -a defaultFiles=("$@")
  # get list of possible config files
  local -a configFiles=()
  if [[ -n "${BASH_FRAMEWORK_ENV_FILES[0]+1}" ]]; then
    # BASH_FRAMEWORK_ENV_FILES is an array
    configFiles+=("${BASH_FRAMEWORK_ENV_FILES[@]}")
  fi
  if [[ -f "$(pwd)/.framework-config" ]]; then
    configFiles+=("$(pwd)/.framework-config")
  fi
  if [[ -f "${FRAMEWORK_ROOT_DIR}/.framework-config" ]]; then
    configFiles+=("${FRAMEWORK_ROOT_DIR}/.framework-config")
  fi
  if [[ -n "${optionBashFrameworkConfig}" && -f "${optionBashFrameworkConfig}" ]]; then
    # shellcheck disable=SC2034
    configFiles+=("${optionBashFrameworkConfig}")
  fi
  configFiles+=("${optionEnvFiles[@]}")
  configFiles+=("${defaultFiles[@]}")

  for file in "${configFiles[@]}"; do
    # shellcheck source=/.framework-config
    CURRENT_LOADED_ENV_FILE="${file}" source "${file}" || {
      Log::displayError "while loading config file: ${file}"
      return 1
    }
  done
}

# @description activate or not Log::display* and Log::log* functions
# based on BASH_FRAMEWORK_DISPLAY_LEVEL and BASH_FRAMEWORK_LOG_LEVEL
# environment variables loaded by Env::requireLoad
# try to create log file and rotate it if necessary
# @noargs
# @set BASH_FRAMEWORK_LOG_LEVEL int to OFF level if BASH_FRAMEWORK_LOG_FILE is empty or not writable
# @env BASH_FRAMEWORK_DISPLAY_LEVEL int
# @env BASH_FRAMEWORK_LOG_LEVEL int
# @env BASH_FRAMEWORK_LOG_FILE String
# @env BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION int do log rotation if > 0
# @exitcode 0 always successful
# @stderr diagnostics information about log file is displayed
# @require Env::requireLoad
# @require UI::requireTheme
Log::requireLoad() {
  if [[ -z "${BASH_FRAMEWORK_LOG_FILE:-}" ]]; then
    BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
    export BASH_FRAMEWORK_LOG_LEVEL
  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    if [[ ! -f "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      if
        ! mkdir -p "$(dirname "${BASH_FRAMEWORK_LOG_FILE}")" 2>/dev/null ||
          ! touch --no-create "${BASH_FRAMEWORK_LOG_FILE}" 2>/dev/null
      then
        BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
        echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
      fi
    elif [[ ! -w "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
      echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
    fi

  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    # will always be created even if not in info level
    Log::logMessage "INFO" "Logging to file ${BASH_FRAMEWORK_LOG_FILE} - Log level ${BASH_FRAMEWORK_LOG_LEVEL}"
    if ((BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION > 0)); then
      Log::rotate "${BASH_FRAMEWORK_LOG_FILE}" "${BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION}"
    fi
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logError() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_ERROR)); then
    Log::logMessage "${2:-ERROR}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logWarning() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_WARNING)); then
    Log::logMessage "${2:-WARNING}" "$1"
  fi
}

# @description To be called before logging in the log file
# @arg $1 file:string log file name
# @arg $2 maxLogFilesCount:int maximum number of log files
Log::rotate() {
  local file="$1"
  local maxLogFilesCount="${2:-5}"

  if [[ ! -f "${file}" ]]; then
    Log::displaySkipped "Log file ${file} doesn't exist yet"
    return 0
  fi
  for i in $(seq $((maxLogFilesCount - 1)) -1 1); do
    Log::displayInfo "Log rotation ${file}.${i} to ${file}.$((i + 1))"
    mv "${file}."{"${i}","$((i + 1))"} &>/dev/null || true
  done
  if cp "${file}" "${file}.1" &>/dev/null; then
    echo >"${file}" # reset log file
    Log::displayInfo "Log rotation ${file} to ${file}.1"
  fi
}

# @description ensure command realpath is available
# @exitcode 1 if realpath command not available
# @stderr diagnostics information is displayed
Linux::requireRealpathCommand() {
  Assert::commandExists realpath
}

# @description load color theme
# @noargs
# @env BASH_FRAMEWORK_THEME String theme to use
# @exitcode 0 always successful
UI::requireTheme() {
  UI::theme "${BASH_FRAMEWORK_THEME-default}"
}

# @description check if command specified exists or return 1
# with error and message if not
#
# @arg $1 commandName:String on which existence must be checked
# @arg $2 helpIfNotExists:String a help command to display if the command does not exist
#
# @exitcode 1 if the command specified does not exist
# @stderr diagnostic information + help if second argument is provided
Assert::commandExists() {
  local commandName="$1"
  local helpIfNotExists="$2"

  "${BASH_FRAMEWORK_COMMAND:-command}" -v "${commandName}" >/dev/null 2>/dev/null || {
    Log::displayError "${commandName} is not installed, please install it"
    if [[ -n "${helpIfNotExists}" ]]; then
      Log::displayInfo "${helpIfNotExists}"
    fi
    return 1
  }
  return 0
}

# @description Display message using skip color (yellow)
# @arg $1 message:String the message to display
Log::displaySkipped() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__SKIPPED_COLOR}SKIPPED - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logSkipped "$1"
}

# @description load colors theme constants
# @warning if tty not opened, noColor theme will be chosen
# @arg $1 theme:String the theme to use (default, noColor)
# @arg $@ args:String[]
# @set __ERROR_COLOR String indicate error status
# @set __INFO_COLOR String indicate info status
# @set __SUCCESS_COLOR String indicate success status
# @set __WARNING_COLOR String indicate warning status
# @set __SKIPPED_COLOR String indicate skipped status
# @set __DEBUG_COLOR String indicate debug status
# @set __HELP_COLOR String indicate help status
# @set __TEST_COLOR String not used
# @set __TEST_ERROR_COLOR String not used
# @set __HELP_TITLE_COLOR String used to display help title in help strings
# @set __HELP_OPTION_COLOR String used to display highlight options in help strings
#
# @set __RESET_COLOR String reset default color
#
# @set __HELP_EXAMPLE String to remove
# @set __HELP_TITLE String to remove
# @set __HELP_NORMAL String to remove
# shellcheck disable=SC2034
UI::theme() {
  local theme="${1-default}"
  if [[ ! "${theme}" =~ -force$ ]] && ! Assert::tty; then
    theme="noColor"
  fi
  case "${theme}" in
    default | default-force)
      theme="default"
      ;;
    noColor) ;;
    *)
      Log::fatal "invalid theme provided"
      ;;
  esac
  if [[ "${theme}" = "default" ]]; then
    BASH_FRAMEWORK_THEME="default"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR='\e[31m'         # Red
    __INFO_COLOR='\e[44m'          # white on lightBlue
    __SUCCESS_COLOR='\e[32m'       # Green
    __WARNING_COLOR='\e[33m'       # Yellow
    __SKIPPED_COLOR='\e[33m'       # Yellow
    __DEBUG_COLOR='\e[37m'         # Grey
    __HELP_COLOR='\e[7;49;33m'     # Black on Gold
    __TEST_COLOR='\e[100m'         # Light magenta
    __TEST_ERROR_COLOR='\e[41m'    # white on red
    __HELP_TITLE_COLOR="\e[1;37m"  # Bold
    __HELP_OPTION_COLOR="\e[1;34m" # Blue
    # Internal: reset color
    __RESET_COLOR='\e[0m' # Reset Color
    # shellcheck disable=SC2155,SC2034
    __HELP_EXAMPLE="$(echo -e "\e[2;97m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_TITLE="$(echo -e "\e[1;37m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_NORMAL="$(echo -e "\033[0m")"
  else
    BASH_FRAMEWORK_THEME="noColor"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR=''
    __INFO_COLOR=''
    __SUCCESS_COLOR=''
    __WARNING_COLOR=''
    __SKIPPED_COLOR=''
    __DEBUG_COLOR=''
    __HELP_COLOR=''
    __TEST_COLOR=''
    __TEST_ERROR_COLOR=''
    __HELP_TITLE_COLOR=''
    __HELP_OPTION_COLOR=''
    # Internal: reset color
    __RESET_COLOR=''
    __HELP_EXAMPLE=''
    __HELP_TITLE=''
    __HELP_NORMAL=''
  fi
}

# @description check if tty (interactive mode) is active
# @noargs
# @exitcode 1 if tty not active
# @env NON_INTERACTIVE if 1 consider as not interactive even if environment is interactive
# @env INTERACTIVE if 1 consider as interactive even if environment is not interactive
# @stderr diagnostic information + help if second argument is provided
Assert::tty() {
  if [[ "${NON_INTERACTIVE:-0}" = "1" ]]; then
    return 1
  fi
  if [[ "${INTERACTIVE:-0}" = "1" ]]; then
    return 0
  fi
  [[ -t 1 || -t 2 ]]
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logSkipped() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-SKIPPED}" "$1"
  fi
}

# FUNCTIONS

facade_main_embedFrameworkFunctionbinFiletpl() {
# REQUIRES
Linux::requireExecutedAsUser
Env::requireLoad
Log::requireLoad
Linux::requireRealpathCommand
UI::requireTheme
Compiler::Facade::requireCommandBinDir

# @require Compiler::Facade::requireCommandBinDir

# shellcheck disable=SC2154,SC2016
functionToCall='Db::queryOneDatabase'
"${functionToCall}" "$@"

}

facade_main_embedFrameworkFunctionbinFiletpl "$@"
" +declare -gx embed_function_DbQueryOneDatabase="${PERSISTENT_TMPDIR:-/tmp}/bin/e36a5808727f397f60db02f81c9ecc2e/dbQueryOneDatabase" +declare -gx encoded_binary_file_DbQueryOneDatabase="#!/usr/bin/env bash
###############################################################################
# GENERATED FACADE FROM https://github.com/fchastanet/bash-tools/tree/master/home/wsl/fchastanet/bash-dev-env/vendor/bash-tools-framework/src/Compiler/Embed/embedFrameworkFunction.binFile.tpl
# DO NOT EDIT IT
# @generated
###############################################################################
# shellcheck disable=SC2288,SC2034
# BIN_FILE=${BIN_FILE}
# FACADE

# ensure that no user aliases could interfere with
# commands used in this script
unalias -a || true
shopt -u expand_aliases

# shellcheck disable=SC2034
((failures = 0)) || true

# Bash will remember & return the highest exit code in a chain of pipes.
# This way you can catch the error inside pipes, e.g. mysqldump | gzip
set -o pipefail
set -o errexit

# Command Substitution can inherit errexit option since bash v4.4
shopt -s inherit_errexit || true

# a log is generated when a command fails
set -o errtrace

# use nullglob so that (file*.php) will return an empty array if no file matches the wildcard
shopt -s nullglob

# ensure regexp are interpreted without accentuated characters
export LC_ALL=POSIX

export TERM=xterm-256color

# avoid interactive install
export DEBIAN_FRONTEND=noninteractive
export DEBCONF_NONINTERACTIVE_SEEN=true

# store command arguments for later usage
# shellcheck disable=SC2034
declare -a BASH_FRAMEWORK_ARGV=("$@")
# shellcheck disable=SC2034
declare -a ORIGINAL_BASH_FRAMEWORK_ARGV=("$@")

# @see https://unix.stackexchange.com/a/386856
# shellcheck disable=SC2317
interruptManagement() {
  # restore SIGINT handler
  trap - INT
  # ensure that Ctrl-C is trapped by this script and not by sub process
  # report to the parent that we have indeed been interrupted
  kill -s INT "$$"
}
trap interruptManagement INT
SCRIPT_NAME=${0##*/}
REAL_SCRIPT_FILE="$(readlink -e "$(realpath "${BASH_SOURCE[0]}")")"
if [[ -n "${EMBED_CURRENT_DIR}" ]]; then
  CURRENT_DIR="${EMBED_CURRENT_DIR}"
else
  CURRENT_DIR="$(cd "$(readlink -e "${REAL_SCRIPT_FILE%/*}")" && pwd -P)"
fi

################################################
# Temp dir management
################################################

KEEP_TEMP_FILES="${KEEP_TEMP_FILES:-0}"
export KEEP_TEMP_FILES

# PERSISTENT_TMPDIR is not deleted by traps
PERSISTENT_TMPDIR="${TMPDIR:-/tmp}/bash-framework"
export PERSISTENT_TMPDIR
mkdir -p "${PERSISTENT_TMPDIR}"

# shellcheck disable=SC2034
TMPDIR="$(mktemp -d -p "${PERSISTENT_TMPDIR:-/tmp}" -t bash-framework-$$-XXXXXX)"
export TMPDIR

# temp dir cleaning
# shellcheck disable=SC2317
cleanOnExit() {
  if [[ "${KEEP_TEMP_FILES:-0}" = "1" ]]; then
    Log::displayInfo "KEEP_TEMP_FILES=1 temp files kept here '${TMPDIR}'"
  elif [[ -n "${TMPDIR+xxx}" ]]; then
    Log::displayDebug "KEEP_TEMP_FILES=0 removing temp files '${TMPDIR}'"
    rm -Rf "${TMPDIR:-/tmp/fake}" >/dev/null 2>&1
  fi
}
trap cleanOnExit EXIT HUP QUIT ABRT TERM

# VAR_MAIN_FUNCTION_VAR_NAME=dbQueryAllDatabasesFacade

# @description used to execute given query when using
# dbScriptAllDatabases
# @arg $1 dsn:String
# @arg $2 db:String
# @env query String
# @env optionSeparator String
# @require Linux::requireExecutedAsUser
Db::queryOneDatabase() {
  # query and optionSeparator are passed via export
  local dsn="$1"
  local db="$2"

  local -A dbInstance
  Database::newInstance dbInstance "${dsn}"
  Database::setQueryOptions dbInstance "${dbInstance[QUERY_OPTIONS]} --connect-timeout=5"

  # identify columns header
  echo -n "@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@"
  Database::skipColumnNames dbInstance 0

  # shellcheck disable=SC2154
  if ! Database::query dbInstance "${query}" "${db}" | sed "s/\t/${optionSeparator}/g"; then
    Log::fatal "database ${db} error" 1>&2
  fi
}

# @description Log namespace provides 2 kind of functions
# - Log::display* allows to display given message with
#   given display level
# - Log::log* allows to log given message with
#   given log level
# Log::display* functions automatically log the message too
# @see Env::requireLoad to load the display and log level from .env file

# @description log level off
export __LEVEL_OFF=0
# @description log level error
export __LEVEL_ERROR=1
# @description log level warning
export __LEVEL_WARNING=2
# @description log level info
export __LEVEL_INFO=3
# @description log level success
export __LEVEL_SUCCESS=3
# @description log level debug
export __LEVEL_DEBUG=4

# @description verbose level off
export __VERBOSE_LEVEL_OFF=0
# @description verbose level info
export __VERBOSE_LEVEL_INFO=1
# @description verbose level info
export __VERBOSE_LEVEL_DEBUG=2
# @description verbose level info
export __VERBOSE_LEVEL_TRACE=3

# @description Display message using debug color (grey)
# @arg $1 message:String the message to display
Log::displayDebug() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_DEBUG)); then
    echo -e "${__DEBUG_COLOR}DEBUG   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logDebug "$1"
}

# @description Display message using info color (bg light blue/fg white)
# @arg $1 message:String the message to display
Log::displayInfo() {
  local type="${2:-INFO}"
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__INFO_COLOR}${type}    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logInfo "$1" "${type}"
}

# @description ensure COMMAND_BIN_DIR env var is set
# and PATH correctly prepared
# @noargs
# @set COMMAND_BIN_DIR string the directory where to find this command
# @set PATH string add directory where to find this command binary
Compiler::Facade::requireCommandBinDir() {
  COMMAND_BIN_DIR="${CURRENT_DIR}"
  Env::pathPrepend "${COMMAND_BIN_DIR}"
}

# @description create a new db instance
# Returns immediately if the instance is already initialized
#
# @arg $1 instanceNewInstance:&Map<String,String> (passed by reference) database instance to use
# @arg $2 dsn:String dsn profile - load the dsn.env profile deduced using rules defined in Conf::getAbsoluteFile
#
# @example
#   declare -Agx dbInstance
#   Database::newInstance dbInstance "default.local"
#
# @exitcode 1 if dns file not able to loaded
Database::newInstance() {
  local -n instanceNewInstance=$1
  local dsn="$2"
  local DSN_FILE

  if [[ -v instanceNewInstance['INITIALIZED'] && "${instanceNewInstance['INITIALIZED']:-0}" == "1" ]]; then
    return
  fi

  # final auth file generated from dns file
  instanceNewInstance['AUTH_FILE']=""
  instanceNewInstance['DSN_FILE']=""

  # check dsn file
  DSN_FILE="$(Conf::getAbsoluteFile "dsn" "${dsn}" "env")" || return 1
  Database::checkDsnFile "${DSN_FILE}" || return 1
  instanceNewInstance['DSN_FILE']="${DSN_FILE}"

  # shellcheck source=/src/Database/testsData/dsn_valid.env
  source "${instanceNewInstance['DSN_FILE']}"

  instanceNewInstance['USER']="${USER}"
  instanceNewInstance['PASSWORD']="${PASSWORD}"
  instanceNewInstance['HOSTNAME']="${HOSTNAME}"
  instanceNewInstance['PORT']="${PORT}"

  # generate authFile for easy authentication
  instanceNewInstance['AUTH_FILE']=$(mktemp -p "${TMPDIR:-/tmp}" -t "mysql.XXXXXXXXXXXX")
  (
    echo "[client]"
    echo "user = ${USER}"
    echo "password = ${PASSWORD}"
    echo "host = ${HOSTNAME}"
    echo "port = ${PORT}"
  ) >"${instanceNewInstance['AUTH_FILE']}"

  # some of those values can be overridden using the dsn file
  # SKIP_COLUMN_NAMES enabled by default
  instanceNewInstance['SKIP_COLUMN_NAMES']="${SKIP_COLUMN_NAMES:-1}"
  instanceNewInstance['SSL_OPTIONS']="${MYSQL_SSL_OPTIONS:---ssl-mode=DISABLED}"
  instanceNewInstance['QUERY_OPTIONS']="${MYSQL_QUERY_OPTIONS:---batch --raw --default-character-set=utf8}"
  instanceNewInstance['DUMP_OPTIONS']="${MYSQL_DUMP_OPTIONS:---default-character-set=utf8 --compress --hex-blob --routines --triggers --single-transaction --set-gtid-purged=OFF --column-statistics=0 ${instanceNewInstance['SSL_OPTIONS']}}"
  instanceNewInstance['DB_IMPORT_OPTIONS']="${DB_IMPORT_OPTIONS:---connect-timeout=5 --batch --raw --default-character-set=utf8}"

  instanceNewInstance['INITIALIZED']=1
}

# @description mysql query on a given db
# @warning could use QUERY_OPTIONS variable from dsn if defined
# @example
#   cat file.sql | Database::query ...
# @arg $1 instanceQuery:&Map<String,String> (passed by reference) database instance to use
# @arg $2 sqlQuery:String (optional) sql query or sql file to execute. if not provided or empty, the command can be piped
# @arg $3 dbName:String (optional) the db name
#
# @exitcode mysql command status code
Database::query() {
  local -n instanceQuery=$1
  local -a mysqlCommand=()
  local -a queryOptions

  mysqlCommand+=(mysql)
  mysqlCommand+=("--defaults-extra-file=${instanceQuery['AUTH_FILE']}")
  IFS=' ' read -r -a queryOptions <<<"${instanceQuery['QUERY_OPTIONS']}"
  mysqlCommand+=("${queryOptions[@]}")
  if [[ "${instanceQuery['SKIP_COLUMN_NAMES']}" = "1" ]]; then
    mysqlCommand+=("-s" "--skip-column-names")
  fi
  # add optional db name
  if [[ -n "${3+x}" ]]; then
    mysqlCommand+=("$3")
  fi
  # add optional sql query
  if [[ -n "${2+x}" && -n "$2" && ! -f "$2" ]]; then
    mysqlCommand+=("-e")
    mysqlCommand+=("$2")
  fi
  Log::displayDebug "$(printf "execute command: '%s'" "${mysqlCommand[*]}")"

  if [[ -f "$2" ]]; then
    "${mysqlCommand[@]}" <"$2"
  else
    "${mysqlCommand[@]}"
  fi
}

# @description set the general options to use on mysql command to query the database
# Differs than setOptions in the way that these options could change each time
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 optionsList:String query options list
Database::setQueryOptions() {
  local -n instanceSetQueryOptions=$1
  # shellcheck disable=SC2034
  instanceSetQueryOptions['QUERY_OPTIONS']="$2"
}

# @description by default we skip the column names
# but sometimes we need column names to display some results
# disable this option temporarily and then restore it to true
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 skipColumnNames:Boolean 0 to disable, 1 to enable (hide column names)
Database::skipColumnNames() {
  local -n instanceSkipColumnNames=$1
  # shellcheck disable=SC2034
  instanceSkipColumnNames['SKIP_COLUMN_NAMES']="$2"
}

# @description prepend directories to the PATH environment variable
# @arg $@ args:String[] list of directories to prepend
# @set PATH update PATH with the directories prepended
Env::pathPrepend() {
  local arg
  for arg in "$@"; do
    if [[ -d "${arg}" && ":${PATH}:" != *":${arg}:"* ]]; then
      PATH="$(realpath "${arg}"):${PATH}"
    fi
  done
}

# @description Display message using error color (red) and exit immediately with error status 1
# @arg $1 message:String the message to display
Log::fatal() {
  echo -e "${__ERROR_COLOR}FATAL   - ${1}${__RESET_COLOR}" >&2
  Log::logFatal "$1"
  exit 1
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logDebug() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_DEBUG)); then
    Log::logMessage "${2:-DEBUG}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logInfo() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-INFO}" "$1"
  fi
}

# @description ensure running user is not root
# @exitcode 1 if current user is root
# @stderr diagnostics information is displayed
Linux::requireExecutedAsUser() {
  if [[ "$(id -u)" = "0" ]]; then
    Log::fatal "this script should be executed as normal user"
  fi
}

# @description get absolute conf file from specified conf folder deduced using these rules
#   * from absolute file (ignores <confFolder> and <extension>)
#   * relative to where script is executed (ignores <confFolder> and <extension>)
#   * from home/.bash-tools/<confFolder>
#   * from framework conf/<confFolder>
#
# @arg $1 confFolder:String the directory name (not the path) to list
# @arg $2 conf:String file to use without extension
# @arg $3 extension:String the extension (.sh by default)
#
# @stdout absolute conf filename
# @exitcode 1 if file is not found in any location
Conf::getAbsoluteFile() {
  local confFolder="$1"
  local conf="$2"
  local extension="${3-.sh}"
  if [[ -n "${extension}" && "${extension:0:1}" != "." ]]; then
    extension=".${extension}"
  fi

  testAbs() {
    local result
    result="$(realpath -e "$1" 2>/dev/null)"
    # shellcheck disable=SC2181
    if [[ "$?" = "0" && -f "${result}" ]]; then
      echo "${result}"
      return 0
    fi
    return 1
  }

  # conf is absolute file (including extension)
  testAbs "${confFolder}${extension}" && return 0
  # conf is absolute file
  testAbs "${confFolder}" && return 0
  # conf is absolute file (including extension)
  testAbs "${conf}${extension}" && return 0
  # conf is absolute file
  testAbs "${conf}" && return 0

  # relative to where script is executed (including extension)
  if [[ -n "${CURRENT_DIR+xxx}" ]]; then
    testAbs "$(File::concatenatePath "${CURRENT_DIR}" "${confFolder}")/${conf}${extension}" && return 0
  fi
  # from home/.bash-tools/<confFolder>
  testAbs "$(File::concatenatePath "${HOME}/.bash-tools" "${confFolder}")/${conf}${extension}" && return 0

  if [[ -n "${FRAMEWORK_ROOT_DIR+xxx}" ]]; then
    # from framework conf/<confFolder> (including extension)
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}${extension}" && return 0

    # from framework conf/<confFolder>
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}" && return 0
  fi

  # file not found
  Log::displayError "conf file '${conf}' not found"

  return 1
}

# @description check if dsn file has all the mandatory variables set
# Mandatory variables are: HOSTNAME, USER, PASSWORD, PORT
#
# @arg $1 dsnFileName:String dsn absolute filename
# @set HOSTNAME loaded from dsn file
# @set PORT loaded from dsn file
# @set USER loaded from dsn file
# @set PASSWORD loaded from dsn file
# @exitcode 0 on valid file
# @exitcode 1 if one of the properties of the conf file is invalid or if file not found
# @stderr log output if error found in conf file
Database::checkDsnFile() {
  local dsnFileName="$1"
  if [[ ! -f "${dsnFileName}" ]]; then
    Log::displayError "dsn file ${dsnFileName} not found"
    return 1
  fi

  (
    unset HOSTNAME PORT PASSWORD USER
    # shellcheck source=/src/Database/testsData/dsn_valid.env
    source "${dsnFileName}"
    if [[ -z ${HOSTNAME+x} ]]; then
      Log::displayError "dsn file ${dsnFileName} : HOSTNAME not provided"
      return 1
    fi
    if [[ -z "${HOSTNAME}" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : HOSTNAME value not provided"
    fi
    if [[ "${HOSTNAME}" = "localhost" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : check that HOSTNAME should not be 127.0.0.1 instead of localhost"
    fi
    if [[ -z "${PORT+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT not provided"
      return 1
    fi
    if ! [[ ${PORT} =~ ^[0-9]+$ ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT invalid"
      return 1
    fi
    if [[ -z "${USER+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : USER not provided"
      return 1
    fi
    if [[ -z "${PASSWORD+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PASSWORD not provided"
      return 1
    fi
  )
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logFatal() {
  Log::logMessage "${2:-FATAL}" "$1"
}

# @description Internal: common log message
# @example text
#   [date]|[levelMsg]|message
#
# @example text
#   2020-01-19 19:20:21|ERROR  |log error
#   2020-01-19 19:20:21|SKIPPED|log skipped
#
# @arg $1 levelMsg:String message's level description (eg: STATUS, ERROR, ...)
# @arg $2 msg:String the message to display
# @env BASH_FRAMEWORK_LOG_FILE String log file to use, do nothing if empty
# @env BASH_FRAMEWORK_LOG_LEVEL int log level log only if > OFF or fatal messages
# @stderr diagnostics information is displayed
# @require Env::requireLoad
# @require Log::requireLoad
Log::logMessage() {
  local levelMsg="$1"
  local msg="$2"
  local date

  if [[ -n "${BASH_FRAMEWORK_LOG_FILE}" ]] && ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    date="$(date '+%Y-%m-%d %H:%M:%S')"
    touch "${BASH_FRAMEWORK_LOG_FILE}"
    printf "%s|%7s|%s\n" "${date}" "${levelMsg}" "${msg}" >>"${BASH_FRAMEWORK_LOG_FILE}"
  fi
}

# @description concatenate 2 paths and ensure the path is correct using realpath -m
# @arg $1 basePath:String
# @arg $2 subPath:String
# @require Linux::requireRealpathCommand
File::concatenatePath() {
  local basePath="$1"
  local subPath="$2"
  local fullPath="${basePath:+${basePath}/}${subPath}"

  realpath -m "${fullPath}" 2>/dev/null
}

# @description Display message using error color (red)
# @arg $1 message:String the message to display
Log::displayError() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_ERROR)); then
    echo -e "${__ERROR_COLOR}ERROR   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logError "$1"
}

# @description Display message using warning color (yellow)
# @arg $1 message:String the message to display
Log::displayWarning() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_WARNING)); then
    echo -e "${__WARNING_COLOR}WARN    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logWarning "$1"
}

# @description ensure env files are loaded
# @arg $@ list of default files to load at the end
# @exitcode 1 if one of env files fails to load
# @stderr diagnostics information is displayed
# shellcheck disable=SC2120
Env::requireLoad() {
  local -a defaultFiles=("$@")
  # get list of possible config files
  local -a configFiles=()
  if [[ -n "${BASH_FRAMEWORK_ENV_FILES[0]+1}" ]]; then
    # BASH_FRAMEWORK_ENV_FILES is an array
    configFiles+=("${BASH_FRAMEWORK_ENV_FILES[@]}")
  fi
  if [[ -f "$(pwd)/.framework-config" ]]; then
    configFiles+=("$(pwd)/.framework-config")
  fi
  if [[ -f "${FRAMEWORK_ROOT_DIR}/.framework-config" ]]; then
    configFiles+=("${FRAMEWORK_ROOT_DIR}/.framework-config")
  fi
  if [[ -n "${optionBashFrameworkConfig}" && -f "${optionBashFrameworkConfig}" ]]; then
    # shellcheck disable=SC2034
    configFiles+=("${optionBashFrameworkConfig}")
  fi
  configFiles+=("${optionEnvFiles[@]}")
  configFiles+=("${defaultFiles[@]}")

  for file in "${configFiles[@]}"; do
    # shellcheck source=/.framework-config
    CURRENT_LOADED_ENV_FILE="${file}" source "${file}" || {
      Log::displayError "while loading config file: ${file}"
      return 1
    }
  done
}

# @description activate or not Log::display* and Log::log* functions
# based on BASH_FRAMEWORK_DISPLAY_LEVEL and BASH_FRAMEWORK_LOG_LEVEL
# environment variables loaded by Env::requireLoad
# try to create log file and rotate it if necessary
# @noargs
# @set BASH_FRAMEWORK_LOG_LEVEL int to OFF level if BASH_FRAMEWORK_LOG_FILE is empty or not writable
# @env BASH_FRAMEWORK_DISPLAY_LEVEL int
# @env BASH_FRAMEWORK_LOG_LEVEL int
# @env BASH_FRAMEWORK_LOG_FILE String
# @env BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION int do log rotation if > 0
# @exitcode 0 always successful
# @stderr diagnostics information about log file is displayed
# @require Env::requireLoad
# @require UI::requireTheme
Log::requireLoad() {
  if [[ -z "${BASH_FRAMEWORK_LOG_FILE:-}" ]]; then
    BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
    export BASH_FRAMEWORK_LOG_LEVEL
  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    if [[ ! -f "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      if
        ! mkdir -p "$(dirname "${BASH_FRAMEWORK_LOG_FILE}")" 2>/dev/null ||
          ! touch --no-create "${BASH_FRAMEWORK_LOG_FILE}" 2>/dev/null
      then
        BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
        echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
      fi
    elif [[ ! -w "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
      echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
    fi

  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    # will always be created even if not in info level
    Log::logMessage "INFO" "Logging to file ${BASH_FRAMEWORK_LOG_FILE} - Log level ${BASH_FRAMEWORK_LOG_LEVEL}"
    if ((BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION > 0)); then
      Log::rotate "${BASH_FRAMEWORK_LOG_FILE}" "${BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION}"
    fi
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logError() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_ERROR)); then
    Log::logMessage "${2:-ERROR}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logWarning() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_WARNING)); then
    Log::logMessage "${2:-WARNING}" "$1"
  fi
}

# @description To be called before logging in the log file
# @arg $1 file:string log file name
# @arg $2 maxLogFilesCount:int maximum number of log files
Log::rotate() {
  local file="$1"
  local maxLogFilesCount="${2:-5}"

  if [[ ! -f "${file}" ]]; then
    Log::displaySkipped "Log file ${file} doesn't exist yet"
    return 0
  fi
  for i in $(seq $((maxLogFilesCount - 1)) -1 1); do
    Log::displayInfo "Log rotation ${file}.${i} to ${file}.$((i + 1))"
    mv "${file}."{"${i}","$((i + 1))"} &>/dev/null || true
  done
  if cp "${file}" "${file}.1" &>/dev/null; then
    echo >"${file}" # reset log file
    Log::displayInfo "Log rotation ${file} to ${file}.1"
  fi
}

# @description ensure command realpath is available
# @exitcode 1 if realpath command not available
# @stderr diagnostics information is displayed
Linux::requireRealpathCommand() {
  Assert::commandExists realpath
}

# @description load color theme
# @noargs
# @env BASH_FRAMEWORK_THEME String theme to use
# @exitcode 0 always successful
UI::requireTheme() {
  UI::theme "${BASH_FRAMEWORK_THEME-default}"
}

# @description check if command specified exists or return 1
# with error and message if not
#
# @arg $1 commandName:String on which existence must be checked
# @arg $2 helpIfNotExists:String a help command to display if the command does not exist
#
# @exitcode 1 if the command specified does not exist
# @stderr diagnostic information + help if second argument is provided
Assert::commandExists() {
  local commandName="$1"
  local helpIfNotExists="$2"

  "${BASH_FRAMEWORK_COMMAND:-command}" -v "${commandName}" >/dev/null 2>/dev/null || {
    Log::displayError "${commandName} is not installed, please install it"
    if [[ -n "${helpIfNotExists}" ]]; then
      Log::displayInfo "${helpIfNotExists}"
    fi
    return 1
  }
  return 0
}

# @description Display message using skip color (yellow)
# @arg $1 message:String the message to display
Log::displaySkipped() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__SKIPPED_COLOR}SKIPPED - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logSkipped "$1"
}

# @description load colors theme constants
# @warning if tty not opened, noColor theme will be chosen
# @arg $1 theme:String the theme to use (default, noColor)
# @arg $@ args:String[]
# @set __ERROR_COLOR String indicate error status
# @set __INFO_COLOR String indicate info status
# @set __SUCCESS_COLOR String indicate success status
# @set __WARNING_COLOR String indicate warning status
# @set __SKIPPED_COLOR String indicate skipped status
# @set __DEBUG_COLOR String indicate debug status
# @set __HELP_COLOR String indicate help status
# @set __TEST_COLOR String not used
# @set __TEST_ERROR_COLOR String not used
# @set __HELP_TITLE_COLOR String used to display help title in help strings
# @set __HELP_OPTION_COLOR String used to display highlight options in help strings
#
# @set __RESET_COLOR String reset default color
#
# @set __HELP_EXAMPLE String to remove
# @set __HELP_TITLE String to remove
# @set __HELP_NORMAL String to remove
# shellcheck disable=SC2034
UI::theme() {
  local theme="${1-default}"
  if [[ ! "${theme}" =~ -force$ ]] && ! Assert::tty; then
    theme="noColor"
  fi
  case "${theme}" in
    default | default-force)
      theme="default"
      ;;
    noColor) ;;
    *)
      Log::fatal "invalid theme provided"
      ;;
  esac
  if [[ "${theme}" = "default" ]]; then
    BASH_FRAMEWORK_THEME="default"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR='\e[31m'         # Red
    __INFO_COLOR='\e[44m'          # white on lightBlue
    __SUCCESS_COLOR='\e[32m'       # Green
    __WARNING_COLOR='\e[33m'       # Yellow
    __SKIPPED_COLOR='\e[33m'       # Yellow
    __DEBUG_COLOR='\e[37m'         # Grey
    __HELP_COLOR='\e[7;49;33m'     # Black on Gold
    __TEST_COLOR='\e[100m'         # Light magenta
    __TEST_ERROR_COLOR='\e[41m'    # white on red
    __HELP_TITLE_COLOR="\e[1;37m"  # Bold
    __HELP_OPTION_COLOR="\e[1;34m" # Blue
    # Internal: reset color
    __RESET_COLOR='\e[0m' # Reset Color
    # shellcheck disable=SC2155,SC2034
    __HELP_EXAMPLE="$(echo -e "\e[2;97m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_TITLE="$(echo -e "\e[1;37m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_NORMAL="$(echo -e "\033[0m")"
  else
    BASH_FRAMEWORK_THEME="noColor"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR=''
    __INFO_COLOR=''
    __SUCCESS_COLOR=''
    __WARNING_COLOR=''
    __SKIPPED_COLOR=''
    __DEBUG_COLOR=''
    __HELP_COLOR=''
    __TEST_COLOR=''
    __TEST_ERROR_COLOR=''
    __HELP_TITLE_COLOR=''
    __HELP_OPTION_COLOR=''
    # Internal: reset color
    __RESET_COLOR=''
    __HELP_EXAMPLE=''
    __HELP_TITLE=''
    __HELP_NORMAL=''
  fi
}

# @description check if tty (interactive mode) is active
# @noargs
# @exitcode 1 if tty not active
# @env NON_INTERACTIVE if 1 consider as not interactive even if environment is interactive
# @env INTERACTIVE if 1 consider as interactive even if environment is not interactive
# @stderr diagnostic information + help if second argument is provided
Assert::tty() {
  if [[ "${NON_INTERACTIVE:-0}" = "1" ]]; then
    return 1
  fi
  if [[ "${INTERACTIVE:-0}" = "1" ]]; then
    return 0
  fi
  [[ -t 1 || -t 2 ]]
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logSkipped() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-SKIPPED}" "$1"
  fi
}

# FUNCTIONS

facade_main_embedFrameworkFunctionbinFiletpl() {
# REQUIRES
Linux::requireExecutedAsUser
Env::requireLoad
Log::requireLoad
Linux::requireRealpathCommand
UI::requireTheme
Compiler::Facade::requireCommandBinDir

# @require Compiler::Facade::requireCommandBinDir

# shellcheck disable=SC2154,SC2016
functionToCall='Db::queryOneDatabase'
"${functionToCall}" "$@"

}

facade_main_embedFrameworkFunctionbinFiletpl "$@"
" Compiler::Embed::extractFileFromBase64 \ "${embed_function_DbQueryOneDatabase}" \ @@ -1427,6 +1450,7 @@ dbQueryAllDatabasesCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1769,11 +1793,11 @@ dbQueryAllDatabasesCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Execute a query on multiple databases in order to generate a report with tsv format, query can be parallelized on multiple databases")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Execute a query on multiple databases in order to generate a report with tsv format, query can be parallelized on multiple databases")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--jobs|-j ]" "[--bar|-b]" "[--separator|-s ]" "[--from-dsn|-f ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1782,25 +1806,25 @@ dbQueryAllDatabasesCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=($'Query to execute \n - , try to execute the mysql query provided by the file \n - , search for query file in queries directory (see below) \n - else the argument is interpreted as query string') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--jobs${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-j ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(specify\ the\ number\ of\ db\ to\ query\ in\ parallel) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: 1' echo -e " ${__HELP_OPTION_COLOR}--bar${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-b${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Show\ progress\ as\ a\ progress\ bar.\ In\ the\ bar\ is\ shown:\ %\ of\ jobs\ completed\,\ estimated\ seconds\ left\,\ and\ number\ of\ jobs\ started.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--separator${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-s ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(character\ to\ use\ to\ separate\ mysql\ column) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: |' echo echo -e "${__HELP_TITLE_COLOR}QUERY OPTIONS:${__RESET_COLOR}" @@ -1808,79 +1832,83 @@ dbQueryAllDatabasesCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(target\ mysql\ server) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}LIST OF AVAILABLE DSN:${__HELP_NORMAL} ${dsnList} @@ -1910,7 +1938,7 @@ ${__HELP_EXAMPLE}${example1}${__HELP_NORMAL}""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/dbScriptAllDatabases b/bin/dbScriptAllDatabases index 4fb771f4..4bab4274 100755 --- a/bin/dbScriptAllDatabases +++ b/bin/dbScriptAllDatabases @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbScriptAllDatabases +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbScriptAllDatabases # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # EMBED Db::queryOneDatabase as dbQueryOneDatabase @@ -95,123 +95,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -1014,8 +1037,8 @@ Log::logSkipped() { # @require Compiler::Embed::requireEmbedBinDir -declare -gx embed_function_DbQueryOneDatabase="${PERSISTENT_TMPDIR:-/tmp}/bin/f94f442fbdd0b7f3ff7df80421405d82/dbQueryOneDatabase" -declare -gx encoded_binary_file_DbQueryOneDatabase="#!/usr/bin/env bash
###############################################################################
# GENERATED FACADE
# DO NOT EDIT IT
# @generated
###############################################################################
# shellcheck disable=SC2288,SC2034
# BIN_FILE=${BIN_FILE}
# FACADE

# ensure that no user aliases could interfere with
# commands used in this script
unalias -a || true
shopt -u expand_aliases

# shellcheck disable=SC2034
((failures = 0)) || true

# Bash will remember & return the highest exit code in a chain of pipes.
# This way you can catch the error inside pipes, e.g. mysqldump | gzip
set -o pipefail
set -o errexit

# Command Substitution can inherit errexit option since bash v4.4
shopt -s inherit_errexit || true

# a log is generated when a command fails
set -o errtrace

# use nullglob so that (file*.php) will return an empty array if no file matches the wildcard
shopt -s nullglob

# ensure regexp are interpreted without accentuated characters
export LC_ALL=POSIX

export TERM=xterm-256color

# avoid interactive install
export DEBIAN_FRONTEND=noninteractive
export DEBCONF_NONINTERACTIVE_SEEN=true

# store command arguments for later usage
# shellcheck disable=SC2034
declare -a BASH_FRAMEWORK_ARGV=("$@")
# shellcheck disable=SC2034
declare -a ORIGINAL_BASH_FRAMEWORK_ARGV=("$@")

# @see https://unix.stackexchange.com/a/386856
# shellcheck disable=SC2317
interruptManagement() {
  # restore SIGINT handler
  trap - INT
  # ensure that Ctrl-C is trapped by this script and not by sub process
  # report to the parent that we have indeed been interrupted
  kill -s INT "$$"
}
trap interruptManagement INT
SCRIPT_NAME=${0##*/}
REAL_SCRIPT_FILE="$(readlink -e "$(realpath "${BASH_SOURCE[0]}")")"
if [[ -n "${EMBED_CURRENT_DIR}" ]]; then
  CURRENT_DIR="${EMBED_CURRENT_DIR}"
else
  CURRENT_DIR="$(cd "$(readlink -e "${REAL_SCRIPT_FILE%/*}")" && pwd -P)"
fi

################################################
# Temp dir management
################################################

KEEP_TEMP_FILES="${KEEP_TEMP_FILES:-0}"
export KEEP_TEMP_FILES

# PERSISTENT_TMPDIR is not deleted by traps
PERSISTENT_TMPDIR="${TMPDIR:-/tmp}/bash-framework"
export PERSISTENT_TMPDIR
mkdir -p "${PERSISTENT_TMPDIR}"

# shellcheck disable=SC2034
TMPDIR="$(mktemp -d -p "${PERSISTENT_TMPDIR:-/tmp}" -t bash-framework-$$-XXXXXX)"
export TMPDIR

# temp dir cleaning
# shellcheck disable=SC2317
cleanOnExit() {
  if [[ "${KEEP_TEMP_FILES:-0}" = "1" ]]; then
    Log::displayInfo "KEEP_TEMP_FILES=1 temp files kept here '${TMPDIR}'"
  elif [[ -n "${TMPDIR+xxx}" ]]; then
    Log::displayDebug "KEEP_TEMP_FILES=0 removing temp files '${TMPDIR}'"
    rm -Rf "${TMPDIR:-/tmp/fake}" >/dev/null 2>&1
  fi
}
trap cleanOnExit EXIT HUP QUIT ABRT TERM

# VAR_MAIN_FUNCTION_VAR_NAME=dbQueryAllDatabasesFacade

# @description used to execute given query when using
# dbScriptAllDatabases
# @arg $1 dsn:String
# @arg $2 db:String
# @env query String
# @env optionSeparator String
# @require Linux::requireExecutedAsUser
Db::queryOneDatabase() {
  # query and optionSeparator are passed via export
  local dsn="$1"
  local db="$2"

  local -A dbInstance
  Database::newInstance dbInstance "${dsn}"
  Database::setQueryOptions dbInstance "${dbInstance[QUERY_OPTIONS]} --connect-timeout=5"

  # identify columns header
  echo -n "@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@"
  Database::skipColumnNames dbInstance 0

  # shellcheck disable=SC2154
  if ! Database::query dbInstance "${query}" "${db}" | sed "s/\t/${optionSeparator}/g"; then
    Log::fatal "database ${db} error" 1>&2
  fi
}

# @description Log namespace provides 2 kind of functions
# - Log::display* allows to display given message with
#   given display level
# - Log::log* allows to log given message with
#   given log level
# Log::display* functions automatically log the message too
# @see Env::requireLoad to load the display and log level from .env file

# @description log level off
export __LEVEL_OFF=0
# @description log level error
export __LEVEL_ERROR=1
# @description log level warning
export __LEVEL_WARNING=2
# @description log level info
export __LEVEL_INFO=3
# @description log level success
export __LEVEL_SUCCESS=3
# @description log level debug
export __LEVEL_DEBUG=4

# @description verbose level off
export __VERBOSE_LEVEL_OFF=0
# @description verbose level info
export __VERBOSE_LEVEL_INFO=1
# @description verbose level info
export __VERBOSE_LEVEL_DEBUG=2
# @description verbose level info
export __VERBOSE_LEVEL_TRACE=3

# @description Display message using debug color (grey)
# @arg $1 message:String the message to display
Log::displayDebug() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_DEBUG)); then
    echo -e "${__DEBUG_COLOR}DEBUG   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logDebug "$1"
}

# @description Display message using info color (bg light blue/fg white)
# @arg $1 message:String the message to display
Log::displayInfo() {
  local type="${2:-INFO}"
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__INFO_COLOR}${type}    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logInfo "$1" "${type}"
}

# @description ensure COMMAND_BIN_DIR env var is set
# and PATH correctly prepared
# @noargs
# @set COMMAND_BIN_DIR string the directory where to find this command
# @set PATH string add directory where to find this command binary
Compiler::Facade::requireCommandBinDir() {
  COMMAND_BIN_DIR="${CURRENT_DIR}"
  Env::pathPrepend "${COMMAND_BIN_DIR}"
}

# @description create a new db instance
# Returns immediately if the instance is already initialized
#
# @arg $1 instanceNewInstance:&Map<String,String> (passed by reference) database instance to use
# @arg $2 dsn:String dsn profile - load the dsn.env profile deduced using rules defined in Conf::getAbsoluteFile
#
# @example
#   declare -Agx dbInstance
#   Database::newInstance dbInstance "default.local"
#
# @exitcode 1 if dns file not able to loaded
Database::newInstance() {
  local -n instanceNewInstance=$1
  local dsn="$2"
  local DSN_FILE

  if [[ -v instanceNewInstance['INITIALIZED'] && "${instanceNewInstance['INITIALIZED']:-0}" == "1" ]]; then
    return
  fi

  # final auth file generated from dns file
  instanceNewInstance['AUTH_FILE']=""
  instanceNewInstance['DSN_FILE']=""

  # check dsn file
  DSN_FILE="$(Conf::getAbsoluteFile "dsn" "${dsn}" "env")" || return 1
  Database::checkDsnFile "${DSN_FILE}" || return 1
  instanceNewInstance['DSN_FILE']="${DSN_FILE}"

  # shellcheck source=/src/Database/testsData/dsn_valid.env
  source "${instanceNewInstance['DSN_FILE']}"

  instanceNewInstance['USER']="${USER}"
  instanceNewInstance['PASSWORD']="${PASSWORD}"
  instanceNewInstance['HOSTNAME']="${HOSTNAME}"
  instanceNewInstance['PORT']="${PORT}"

  # generate authFile for easy authentication
  instanceNewInstance['AUTH_FILE']=$(mktemp -p "${TMPDIR:-/tmp}" -t "mysql.XXXXXXXXXXXX")
  (
    echo "[client]"
    echo "user = ${USER}"
    echo "password = ${PASSWORD}"
    echo "host = ${HOSTNAME}"
    echo "port = ${PORT}"
  ) >"${instanceNewInstance['AUTH_FILE']}"

  # some of those values can be overridden using the dsn file
  # SKIP_COLUMN_NAMES enabled by default
  instanceNewInstance['SKIP_COLUMN_NAMES']="${SKIP_COLUMN_NAMES:-1}"
  instanceNewInstance['SSL_OPTIONS']="${MYSQL_SSL_OPTIONS:---ssl-mode=DISABLED}"
  instanceNewInstance['QUERY_OPTIONS']="${MYSQL_QUERY_OPTIONS:---batch --raw --default-character-set=utf8}"
  instanceNewInstance['DUMP_OPTIONS']="${MYSQL_DUMP_OPTIONS:---default-character-set=utf8 --compress --hex-blob --routines --triggers --single-transaction --set-gtid-purged=OFF --column-statistics=0 ${instanceNewInstance['SSL_OPTIONS']}}"
  instanceNewInstance['DB_IMPORT_OPTIONS']="${DB_IMPORT_OPTIONS:---connect-timeout=5 --batch --raw --default-character-set=utf8}"

  instanceNewInstance['INITIALIZED']=1
}

# @description mysql query on a given db
# @warning could use QUERY_OPTIONS variable from dsn if defined
# @example
#   cat file.sql | Database::query ...
# @arg $1 instanceQuery:&Map<String,String> (passed by reference) database instance to use
# @arg $2 sqlQuery:String (optional) sql query or sql file to execute. if not provided or empty, the command can be piped
# @arg $3 dbName:String (optional) the db name
#
# @exitcode mysql command status code
Database::query() {
  local -n instanceQuery=$1
  local -a mysqlCommand=()
  local -a queryOptions

  mysqlCommand+=(mysql)
  mysqlCommand+=("--defaults-extra-file=${instanceQuery['AUTH_FILE']}")
  IFS=' ' read -r -a queryOptions <<<"${instanceQuery['QUERY_OPTIONS']}"
  mysqlCommand+=("${queryOptions[@]}")
  if [[ "${instanceQuery['SKIP_COLUMN_NAMES']}" = "1" ]]; then
    mysqlCommand+=("-s" "--skip-column-names")
  fi
  # add optional db name
  if [[ -n "${3+x}" ]]; then
    mysqlCommand+=("$3")
  fi
  # add optional sql query
  if [[ -n "${2+x}" && -n "$2" && ! -f "$2" ]]; then
    mysqlCommand+=("-e")
    mysqlCommand+=("$2")
  fi
  Log::displayDebug "$(printf "execute command: '%s'" "${mysqlCommand[*]}")"

  if [[ -f "$2" ]]; then
    "${mysqlCommand[@]}" <"$2"
  else
    "${mysqlCommand[@]}"
  fi
}

# @description set the general options to use on mysql command to query the database
# Differs than setOptions in the way that these options could change each time
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 optionsList:String query options list
Database::setQueryOptions() {
  local -n instanceSetQueryOptions=$1
  # shellcheck disable=SC2034
  instanceSetQueryOptions['QUERY_OPTIONS']="$2"
}

# @description by default we skip the column names
# but sometimes we need column names to display some results
# disable this option temporarily and then restore it to true
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 skipColumnNames:Boolean 0 to disable, 1 to enable (hide column names)
Database::skipColumnNames() {
  local -n instanceSkipColumnNames=$1
  # shellcheck disable=SC2034
  instanceSkipColumnNames['SKIP_COLUMN_NAMES']="$2"
}

# @description prepend directories to the PATH environment variable
# @arg $@ args:String[] list of directories to prepend
# @set PATH update PATH with the directories prepended
Env::pathPrepend() {
  local arg
  for arg in "$@"; do
    if [[ -d "${arg}" && ":${PATH}:" != *":${arg}:"* ]]; then
      PATH="$(realpath "${arg}"):${PATH}"
    fi
  done
}

# @description Display message using error color (red) and exit immediately with error status 1
# @arg $1 message:String the message to display
Log::fatal() {
  echo -e "${__ERROR_COLOR}FATAL   - ${1}${__RESET_COLOR}" >&2
  Log::logFatal "$1"
  exit 1
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logDebug() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_DEBUG)); then
    Log::logMessage "${2:-DEBUG}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logInfo() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-INFO}" "$1"
  fi
}

# @description ensure running user is not root
# @exitcode 1 if current user is root
# @stderr diagnostics information is displayed
Linux::requireExecutedAsUser() {
  if [[ "$(id -u)" = "0" ]]; then
    Log::fatal "this script should be executed as normal user"
  fi
}

# @description get absolute conf file from specified conf folder deduced using these rules
#   * from absolute file (ignores <confFolder> and <extension>)
#   * relative to where script is executed (ignores <confFolder> and <extension>)
#   * from home/.bash-tools/<confFolder>
#   * from framework conf/<confFolder>
#
# @arg $1 confFolder:String the directory name (not the path) to list
# @arg $2 conf:String file to use without extension
# @arg $3 extension:String the extension (.sh by default)
#
# @stdout absolute conf filename
# @exitcode 1 if file is not found in any location
Conf::getAbsoluteFile() {
  local confFolder="$1"
  local conf="$2"
  local extension="${3-.sh}"
  if [[ -n "${extension}" && "${extension:0:1}" != "." ]]; then
    extension=".${extension}"
  fi

  testAbs() {
    local result
    result="$(realpath -e "$1" 2>/dev/null)"
    # shellcheck disable=SC2181
    if [[ "$?" = "0" && -f "${result}" ]]; then
      echo "${result}"
      return 0
    fi
    return 1
  }

  # conf is absolute file (including extension)
  testAbs "${confFolder}${extension}" && return 0
  # conf is absolute file
  testAbs "${confFolder}" && return 0
  # conf is absolute file (including extension)
  testAbs "${conf}${extension}" && return 0
  # conf is absolute file
  testAbs "${conf}" && return 0

  # relative to where script is executed (including extension)
  if [[ -n "${CURRENT_DIR+xxx}" ]]; then
    testAbs "$(File::concatenatePath "${CURRENT_DIR}" "${confFolder}")/${conf}${extension}" && return 0
  fi
  # from home/.bash-tools/<confFolder>
  testAbs "$(File::concatenatePath "${HOME}/.bash-tools" "${confFolder}")/${conf}${extension}" && return 0

  if [[ -n "${FRAMEWORK_ROOT_DIR+xxx}" ]]; then
    # from framework conf/<confFolder> (including extension)
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}${extension}" && return 0

    # from framework conf/<confFolder>
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}" && return 0
  fi

  # file not found
  Log::displayError "conf file '${conf}' not found"

  return 1
}

# @description check if dsn file has all the mandatory variables set
# Mandatory variables are: HOSTNAME, USER, PASSWORD, PORT
#
# @arg $1 dsnFileName:String dsn absolute filename
# @set HOSTNAME loaded from dsn file
# @set PORT loaded from dsn file
# @set USER loaded from dsn file
# @set PASSWORD loaded from dsn file
# @exitcode 0 on valid file
# @exitcode 1 if one of the properties of the conf file is invalid or if file not found
# @stderr log output if error found in conf file
Database::checkDsnFile() {
  local dsnFileName="$1"
  if [[ ! -f "${dsnFileName}" ]]; then
    Log::displayError "dsn file ${dsnFileName} not found"
    return 1
  fi

  (
    unset HOSTNAME PORT PASSWORD USER
    # shellcheck source=/src/Database/testsData/dsn_valid.env
    source "${dsnFileName}"
    if [[ -z ${HOSTNAME+x} ]]; then
      Log::displayError "dsn file ${dsnFileName} : HOSTNAME not provided"
      return 1
    fi
    if [[ -z "${HOSTNAME}" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : HOSTNAME value not provided"
    fi
    if [[ "${HOSTNAME}" = "localhost" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : check that HOSTNAME should not be 127.0.0.1 instead of localhost"
    fi
    if [[ -z "${PORT+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT not provided"
      return 1
    fi
    if ! [[ ${PORT} =~ ^[0-9]+$ ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT invalid"
      return 1
    fi
    if [[ -z "${USER+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : USER not provided"
      return 1
    fi
    if [[ -z "${PASSWORD+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PASSWORD not provided"
      return 1
    fi
  )
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logFatal() {
  Log::logMessage "${2:-FATAL}" "$1"
}

# @description Internal: common log message
# @example text
#   [date]|[levelMsg]|message
#
# @example text
#   2020-01-19 19:20:21|ERROR  |log error
#   2020-01-19 19:20:21|SKIPPED|log skipped
#
# @arg $1 levelMsg:String message's level description (eg: STATUS, ERROR, ...)
# @arg $2 msg:String the message to display
# @env BASH_FRAMEWORK_LOG_FILE String log file to use, do nothing if empty
# @env BASH_FRAMEWORK_LOG_LEVEL int log level log only if > OFF or fatal messages
# @stderr diagnostics information is displayed
# @require Env::requireLoad
# @require Log::requireLoad
Log::logMessage() {
  local levelMsg="$1"
  local msg="$2"
  local date

  if [[ -n "${BASH_FRAMEWORK_LOG_FILE}" ]] && ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    date="$(date '+%Y-%m-%d %H:%M:%S')"
    touch "${BASH_FRAMEWORK_LOG_FILE}"
    printf "%s|%7s|%s\n" "${date}" "${levelMsg}" "${msg}" >>"${BASH_FRAMEWORK_LOG_FILE}"
  fi
}

# @description concatenate 2 paths and ensure the path is correct using realpath -m
# @arg $1 basePath:String
# @arg $2 subPath:String
# @require Linux::requireRealpathCommand
File::concatenatePath() {
  local basePath="$1"
  local subPath="$2"
  local fullPath="${basePath:+${basePath}/}${subPath}"

  realpath -m "${fullPath}" 2>/dev/null
}

# @description Display message using error color (red)
# @arg $1 message:String the message to display
Log::displayError() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_ERROR)); then
    echo -e "${__ERROR_COLOR}ERROR   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logError "$1"
}

# @description Display message using warning color (yellow)
# @arg $1 message:String the message to display
Log::displayWarning() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_WARNING)); then
    echo -e "${__WARNING_COLOR}WARN    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logWarning "$1"
}

# @description ensure env files are loaded
# @arg $@ list of default files to load at the end
# @exitcode 1 if one of env files fails to load
# @stderr diagnostics information is displayed
# shellcheck disable=SC2120
Env::requireLoad() {
  local -a defaultFiles=("$@")
  # get list of possible config files
  local -a configFiles=()
  if [[ -n "${BASH_FRAMEWORK_ENV_FILES[0]+1}" ]]; then
    # BASH_FRAMEWORK_ENV_FILES is an array
    configFiles+=("${BASH_FRAMEWORK_ENV_FILES[@]}")
  fi
  if [[ -f "$(pwd)/.framework-config" ]]; then
    configFiles+=("$(pwd)/.framework-config")
  fi
  if [[ -f "${FRAMEWORK_ROOT_DIR}/.framework-config" ]]; then
    configFiles+=("${FRAMEWORK_ROOT_DIR}/.framework-config")
  fi
  if [[ -n "${optionBashFrameworkConfig}" && -f "${optionBashFrameworkConfig}" ]]; then
    # shellcheck disable=SC2034
    configFiles+=("${optionBashFrameworkConfig}")
  fi
  configFiles+=("${optionEnvFiles[@]}")
  configFiles+=("${defaultFiles[@]}")

  for file in "${configFiles[@]}"; do
    # shellcheck source=/.framework-config
    CURRENT_LOADED_ENV_FILE="${file}" source "${file}" || {
      Log::displayError "while loading config file: ${file}"
      return 1
    }
  done
}

# @description activate or not Log::display* and Log::log* functions
# based on BASH_FRAMEWORK_DISPLAY_LEVEL and BASH_FRAMEWORK_LOG_LEVEL
# environment variables loaded by Env::requireLoad
# try to create log file and rotate it if necessary
# @noargs
# @set BASH_FRAMEWORK_LOG_LEVEL int to OFF level if BASH_FRAMEWORK_LOG_FILE is empty or not writable
# @env BASH_FRAMEWORK_DISPLAY_LEVEL int
# @env BASH_FRAMEWORK_LOG_LEVEL int
# @env BASH_FRAMEWORK_LOG_FILE String
# @env BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION int do log rotation if > 0
# @exitcode 0 always successful
# @stderr diagnostics information about log file is displayed
# @require Env::requireLoad
# @require UI::requireTheme
Log::requireLoad() {
  if [[ -z "${BASH_FRAMEWORK_LOG_FILE:-}" ]]; then
    BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
    export BASH_FRAMEWORK_LOG_LEVEL
  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    if [[ ! -f "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      if
        ! mkdir -p "$(dirname "${BASH_FRAMEWORK_LOG_FILE}")" 2>/dev/null ||
          ! touch --no-create "${BASH_FRAMEWORK_LOG_FILE}" 2>/dev/null
      then
        BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
        echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
      fi
    elif [[ ! -w "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
      echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
    fi

  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    # will always be created even if not in info level
    Log::logMessage "INFO" "Logging to file ${BASH_FRAMEWORK_LOG_FILE} - Log level ${BASH_FRAMEWORK_LOG_LEVEL}"
    if ((BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION > 0)); then
      Log::rotate "${BASH_FRAMEWORK_LOG_FILE}" "${BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION}"
    fi
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logError() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_ERROR)); then
    Log::logMessage "${2:-ERROR}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logWarning() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_WARNING)); then
    Log::logMessage "${2:-WARNING}" "$1"
  fi
}

# @description To be called before logging in the log file
# @arg $1 file:string log file name
# @arg $2 maxLogFilesCount:int maximum number of log files
Log::rotate() {
  local file="$1"
  local maxLogFilesCount="${2:-5}"

  if [[ ! -f "${file}" ]]; then
    Log::displaySkipped "Log file ${file} doesn't exist yet"
    return 0
  fi
  for i in $(seq $((maxLogFilesCount - 1)) -1 1); do
    Log::displayInfo "Log rotation ${file}.${i} to ${file}.$((i + 1))"
    mv "${file}."{"${i}","$((i + 1))"} &>/dev/null || true
  done
  if cp "${file}" "${file}.1" &>/dev/null; then
    echo >"${file}" # reset log file
    Log::displayInfo "Log rotation ${file} to ${file}.1"
  fi
}

# @description ensure command realpath is available
# @exitcode 1 if realpath command not available
# @stderr diagnostics information is displayed
Linux::requireRealpathCommand() {
  Assert::commandExists realpath
}

# @description load color theme
# @noargs
# @env BASH_FRAMEWORK_THEME String theme to use
# @exitcode 0 always successful
UI::requireTheme() {
  UI::theme "${BASH_FRAMEWORK_THEME-default}"
}

# @description check if command specified exists or return 1
# with error and message if not
#
# @arg $1 commandName:String on which existence must be checked
# @arg $2 helpIfNotExists:String a help command to display if the command does not exist
#
# @exitcode 1 if the command specified does not exist
# @stderr diagnostic information + help if second argument is provided
Assert::commandExists() {
  local commandName="$1"
  local helpIfNotExists="$2"

  "${BASH_FRAMEWORK_COMMAND:-command}" -v "${commandName}" >/dev/null 2>/dev/null || {
    Log::displayError "${commandName} is not installed, please install it"
    if [[ -n "${helpIfNotExists}" ]]; then
      Log::displayInfo "${helpIfNotExists}"
    fi
    return 1
  }
  return 0
}

# @description Display message using skip color (yellow)
# @arg $1 message:String the message to display
Log::displaySkipped() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__SKIPPED_COLOR}SKIPPED - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logSkipped "$1"
}

# @description load colors theme constants
# @warning if tty not opened, noColor theme will be chosen
# @arg $1 theme:String the theme to use (default, noColor)
# @arg $@ args:String[]
# @set __ERROR_COLOR String indicate error status
# @set __INFO_COLOR String indicate info status
# @set __SUCCESS_COLOR String indicate success status
# @set __WARNING_COLOR String indicate warning status
# @set __SKIPPED_COLOR String indicate skipped status
# @set __DEBUG_COLOR String indicate debug status
# @set __HELP_COLOR String indicate help status
# @set __TEST_COLOR String not used
# @set __TEST_ERROR_COLOR String not used
# @set __HELP_TITLE_COLOR String used to display help title in help strings
# @set __HELP_OPTION_COLOR String used to display highlight options in help strings
#
# @set __RESET_COLOR String reset default color
#
# @set __HELP_EXAMPLE String to remove
# @set __HELP_TITLE String to remove
# @set __HELP_NORMAL String to remove
# shellcheck disable=SC2034
UI::theme() {
  local theme="${1-default}"
  if [[ ! "${theme}" =~ -force$ ]] && ! Assert::tty; then
    theme="noColor"
  fi
  case "${theme}" in
    default | default-force)
      theme="default"
      ;;
    noColor) ;;
    *)
      Log::fatal "invalid theme provided"
      ;;
  esac
  if [[ "${theme}" = "default" ]]; then
    BASH_FRAMEWORK_THEME="default"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR='\e[31m'         # Red
    __INFO_COLOR='\e[44m'          # white on lightBlue
    __SUCCESS_COLOR='\e[32m'       # Green
    __WARNING_COLOR='\e[33m'       # Yellow
    __SKIPPED_COLOR='\e[33m'       # Yellow
    __DEBUG_COLOR='\e[37m'         # Grey
    __HELP_COLOR='\e[7;49;33m'     # Black on Gold
    __TEST_COLOR='\e[100m'         # Light magenta
    __TEST_ERROR_COLOR='\e[41m'    # white on red
    __HELP_TITLE_COLOR="\e[1;37m"  # Bold
    __HELP_OPTION_COLOR="\e[1;34m" # Blue
    # Internal: reset color
    __RESET_COLOR='\e[0m' # Reset Color
    # shellcheck disable=SC2155,SC2034
    __HELP_EXAMPLE="$(echo -e "\e[2;97m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_TITLE="$(echo -e "\e[1;37m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_NORMAL="$(echo -e "\033[0m")"
  else
    BASH_FRAMEWORK_THEME="noColor"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR=''
    __INFO_COLOR=''
    __SUCCESS_COLOR=''
    __WARNING_COLOR=''
    __SKIPPED_COLOR=''
    __DEBUG_COLOR=''
    __HELP_COLOR=''
    __TEST_COLOR=''
    __TEST_ERROR_COLOR=''
    __HELP_TITLE_COLOR=''
    __HELP_OPTION_COLOR=''
    # Internal: reset color
    __RESET_COLOR=''
    __HELP_EXAMPLE=''
    __HELP_TITLE=''
    __HELP_NORMAL=''
  fi
}

# @description check if tty (interactive mode) is active
# @noargs
# @exitcode 1 if tty not active
# @env NON_INTERACTIVE if 1 consider as not interactive even if environment is interactive
# @env INTERACTIVE if 1 consider as interactive even if environment is not interactive
# @stderr diagnostic information + help if second argument is provided
Assert::tty() {
  if [[ "${NON_INTERACTIVE:-0}" = "1" ]]; then
    return 1
  fi
  if [[ "${INTERACTIVE:-0}" = "1" ]]; then
    return 0
  fi
  [[ -t 1 || -t 2 ]]
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logSkipped() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-SKIPPED}" "$1"
  fi
}

# FUNCTIONS

facade_main_embedFrameworkFunctionbinFiletpl() {
# REQUIRES
Linux::requireExecutedAsUser
Env::requireLoad
Log::requireLoad
Linux::requireRealpathCommand
UI::requireTheme
Compiler::Facade::requireCommandBinDir

# @require Compiler::Facade::requireCommandBinDir

# shellcheck disable=SC2154,SC2016
functionToCall='Db::queryOneDatabase'
"${functionToCall}" "$@"

}

facade_main_embedFrameworkFunctionbinFiletpl "$@"
" +declare -gx embed_function_DbQueryOneDatabase="${PERSISTENT_TMPDIR:-/tmp}/bin/e36a5808727f397f60db02f81c9ecc2e/dbQueryOneDatabase" +declare -gx encoded_binary_file_DbQueryOneDatabase="#!/usr/bin/env bash
###############################################################################
# GENERATED FACADE FROM https://github.com/fchastanet/bash-tools/tree/master/home/wsl/fchastanet/bash-dev-env/vendor/bash-tools-framework/src/Compiler/Embed/embedFrameworkFunction.binFile.tpl
# DO NOT EDIT IT
# @generated
###############################################################################
# shellcheck disable=SC2288,SC2034
# BIN_FILE=${BIN_FILE}
# FACADE

# ensure that no user aliases could interfere with
# commands used in this script
unalias -a || true
shopt -u expand_aliases

# shellcheck disable=SC2034
((failures = 0)) || true

# Bash will remember & return the highest exit code in a chain of pipes.
# This way you can catch the error inside pipes, e.g. mysqldump | gzip
set -o pipefail
set -o errexit

# Command Substitution can inherit errexit option since bash v4.4
shopt -s inherit_errexit || true

# a log is generated when a command fails
set -o errtrace

# use nullglob so that (file*.php) will return an empty array if no file matches the wildcard
shopt -s nullglob

# ensure regexp are interpreted without accentuated characters
export LC_ALL=POSIX

export TERM=xterm-256color

# avoid interactive install
export DEBIAN_FRONTEND=noninteractive
export DEBCONF_NONINTERACTIVE_SEEN=true

# store command arguments for later usage
# shellcheck disable=SC2034
declare -a BASH_FRAMEWORK_ARGV=("$@")
# shellcheck disable=SC2034
declare -a ORIGINAL_BASH_FRAMEWORK_ARGV=("$@")

# @see https://unix.stackexchange.com/a/386856
# shellcheck disable=SC2317
interruptManagement() {
  # restore SIGINT handler
  trap - INT
  # ensure that Ctrl-C is trapped by this script and not by sub process
  # report to the parent that we have indeed been interrupted
  kill -s INT "$$"
}
trap interruptManagement INT
SCRIPT_NAME=${0##*/}
REAL_SCRIPT_FILE="$(readlink -e "$(realpath "${BASH_SOURCE[0]}")")"
if [[ -n "${EMBED_CURRENT_DIR}" ]]; then
  CURRENT_DIR="${EMBED_CURRENT_DIR}"
else
  CURRENT_DIR="$(cd "$(readlink -e "${REAL_SCRIPT_FILE%/*}")" && pwd -P)"
fi

################################################
# Temp dir management
################################################

KEEP_TEMP_FILES="${KEEP_TEMP_FILES:-0}"
export KEEP_TEMP_FILES

# PERSISTENT_TMPDIR is not deleted by traps
PERSISTENT_TMPDIR="${TMPDIR:-/tmp}/bash-framework"
export PERSISTENT_TMPDIR
mkdir -p "${PERSISTENT_TMPDIR}"

# shellcheck disable=SC2034
TMPDIR="$(mktemp -d -p "${PERSISTENT_TMPDIR:-/tmp}" -t bash-framework-$$-XXXXXX)"
export TMPDIR

# temp dir cleaning
# shellcheck disable=SC2317
cleanOnExit() {
  if [[ "${KEEP_TEMP_FILES:-0}" = "1" ]]; then
    Log::displayInfo "KEEP_TEMP_FILES=1 temp files kept here '${TMPDIR}'"
  elif [[ -n "${TMPDIR+xxx}" ]]; then
    Log::displayDebug "KEEP_TEMP_FILES=0 removing temp files '${TMPDIR}'"
    rm -Rf "${TMPDIR:-/tmp/fake}" >/dev/null 2>&1
  fi
}
trap cleanOnExit EXIT HUP QUIT ABRT TERM

# VAR_MAIN_FUNCTION_VAR_NAME=dbQueryAllDatabasesFacade

# @description used to execute given query when using
# dbScriptAllDatabases
# @arg $1 dsn:String
# @arg $2 db:String
# @env query String
# @env optionSeparator String
# @require Linux::requireExecutedAsUser
Db::queryOneDatabase() {
  # query and optionSeparator are passed via export
  local dsn="$1"
  local db="$2"

  local -A dbInstance
  Database::newInstance dbInstance "${dsn}"
  Database::setQueryOptions dbInstance "${dbInstance[QUERY_OPTIONS]} --connect-timeout=5"

  # identify columns header
  echo -n "@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@"
  Database::skipColumnNames dbInstance 0

  # shellcheck disable=SC2154
  if ! Database::query dbInstance "${query}" "${db}" | sed "s/\t/${optionSeparator}/g"; then
    Log::fatal "database ${db} error" 1>&2
  fi
}

# @description Log namespace provides 2 kind of functions
# - Log::display* allows to display given message with
#   given display level
# - Log::log* allows to log given message with
#   given log level
# Log::display* functions automatically log the message too
# @see Env::requireLoad to load the display and log level from .env file

# @description log level off
export __LEVEL_OFF=0
# @description log level error
export __LEVEL_ERROR=1
# @description log level warning
export __LEVEL_WARNING=2
# @description log level info
export __LEVEL_INFO=3
# @description log level success
export __LEVEL_SUCCESS=3
# @description log level debug
export __LEVEL_DEBUG=4

# @description verbose level off
export __VERBOSE_LEVEL_OFF=0
# @description verbose level info
export __VERBOSE_LEVEL_INFO=1
# @description verbose level info
export __VERBOSE_LEVEL_DEBUG=2
# @description verbose level info
export __VERBOSE_LEVEL_TRACE=3

# @description Display message using debug color (grey)
# @arg $1 message:String the message to display
Log::displayDebug() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_DEBUG)); then
    echo -e "${__DEBUG_COLOR}DEBUG   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logDebug "$1"
}

# @description Display message using info color (bg light blue/fg white)
# @arg $1 message:String the message to display
Log::displayInfo() {
  local type="${2:-INFO}"
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__INFO_COLOR}${type}    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logInfo "$1" "${type}"
}

# @description ensure COMMAND_BIN_DIR env var is set
# and PATH correctly prepared
# @noargs
# @set COMMAND_BIN_DIR string the directory where to find this command
# @set PATH string add directory where to find this command binary
Compiler::Facade::requireCommandBinDir() {
  COMMAND_BIN_DIR="${CURRENT_DIR}"
  Env::pathPrepend "${COMMAND_BIN_DIR}"
}

# @description create a new db instance
# Returns immediately if the instance is already initialized
#
# @arg $1 instanceNewInstance:&Map<String,String> (passed by reference) database instance to use
# @arg $2 dsn:String dsn profile - load the dsn.env profile deduced using rules defined in Conf::getAbsoluteFile
#
# @example
#   declare -Agx dbInstance
#   Database::newInstance dbInstance "default.local"
#
# @exitcode 1 if dns file not able to loaded
Database::newInstance() {
  local -n instanceNewInstance=$1
  local dsn="$2"
  local DSN_FILE

  if [[ -v instanceNewInstance['INITIALIZED'] && "${instanceNewInstance['INITIALIZED']:-0}" == "1" ]]; then
    return
  fi

  # final auth file generated from dns file
  instanceNewInstance['AUTH_FILE']=""
  instanceNewInstance['DSN_FILE']=""

  # check dsn file
  DSN_FILE="$(Conf::getAbsoluteFile "dsn" "${dsn}" "env")" || return 1
  Database::checkDsnFile "${DSN_FILE}" || return 1
  instanceNewInstance['DSN_FILE']="${DSN_FILE}"

  # shellcheck source=/src/Database/testsData/dsn_valid.env
  source "${instanceNewInstance['DSN_FILE']}"

  instanceNewInstance['USER']="${USER}"
  instanceNewInstance['PASSWORD']="${PASSWORD}"
  instanceNewInstance['HOSTNAME']="${HOSTNAME}"
  instanceNewInstance['PORT']="${PORT}"

  # generate authFile for easy authentication
  instanceNewInstance['AUTH_FILE']=$(mktemp -p "${TMPDIR:-/tmp}" -t "mysql.XXXXXXXXXXXX")
  (
    echo "[client]"
    echo "user = ${USER}"
    echo "password = ${PASSWORD}"
    echo "host = ${HOSTNAME}"
    echo "port = ${PORT}"
  ) >"${instanceNewInstance['AUTH_FILE']}"

  # some of those values can be overridden using the dsn file
  # SKIP_COLUMN_NAMES enabled by default
  instanceNewInstance['SKIP_COLUMN_NAMES']="${SKIP_COLUMN_NAMES:-1}"
  instanceNewInstance['SSL_OPTIONS']="${MYSQL_SSL_OPTIONS:---ssl-mode=DISABLED}"
  instanceNewInstance['QUERY_OPTIONS']="${MYSQL_QUERY_OPTIONS:---batch --raw --default-character-set=utf8}"
  instanceNewInstance['DUMP_OPTIONS']="${MYSQL_DUMP_OPTIONS:---default-character-set=utf8 --compress --hex-blob --routines --triggers --single-transaction --set-gtid-purged=OFF --column-statistics=0 ${instanceNewInstance['SSL_OPTIONS']}}"
  instanceNewInstance['DB_IMPORT_OPTIONS']="${DB_IMPORT_OPTIONS:---connect-timeout=5 --batch --raw --default-character-set=utf8}"

  instanceNewInstance['INITIALIZED']=1
}

# @description mysql query on a given db
# @warning could use QUERY_OPTIONS variable from dsn if defined
# @example
#   cat file.sql | Database::query ...
# @arg $1 instanceQuery:&Map<String,String> (passed by reference) database instance to use
# @arg $2 sqlQuery:String (optional) sql query or sql file to execute. if not provided or empty, the command can be piped
# @arg $3 dbName:String (optional) the db name
#
# @exitcode mysql command status code
Database::query() {
  local -n instanceQuery=$1
  local -a mysqlCommand=()
  local -a queryOptions

  mysqlCommand+=(mysql)
  mysqlCommand+=("--defaults-extra-file=${instanceQuery['AUTH_FILE']}")
  IFS=' ' read -r -a queryOptions <<<"${instanceQuery['QUERY_OPTIONS']}"
  mysqlCommand+=("${queryOptions[@]}")
  if [[ "${instanceQuery['SKIP_COLUMN_NAMES']}" = "1" ]]; then
    mysqlCommand+=("-s" "--skip-column-names")
  fi
  # add optional db name
  if [[ -n "${3+x}" ]]; then
    mysqlCommand+=("$3")
  fi
  # add optional sql query
  if [[ -n "${2+x}" && -n "$2" && ! -f "$2" ]]; then
    mysqlCommand+=("-e")
    mysqlCommand+=("$2")
  fi
  Log::displayDebug "$(printf "execute command: '%s'" "${mysqlCommand[*]}")"

  if [[ -f "$2" ]]; then
    "${mysqlCommand[@]}" <"$2"
  else
    "${mysqlCommand[@]}"
  fi
}

# @description set the general options to use on mysql command to query the database
# Differs than setOptions in the way that these options could change each time
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 optionsList:String query options list
Database::setQueryOptions() {
  local -n instanceSetQueryOptions=$1
  # shellcheck disable=SC2034
  instanceSetQueryOptions['QUERY_OPTIONS']="$2"
}

# @description by default we skip the column names
# but sometimes we need column names to display some results
# disable this option temporarily and then restore it to true
#
# @arg $1 instanceSetQueryOptions:&Map<String,String> (passed by reference) database instance to use
# @arg $2 skipColumnNames:Boolean 0 to disable, 1 to enable (hide column names)
Database::skipColumnNames() {
  local -n instanceSkipColumnNames=$1
  # shellcheck disable=SC2034
  instanceSkipColumnNames['SKIP_COLUMN_NAMES']="$2"
}

# @description prepend directories to the PATH environment variable
# @arg $@ args:String[] list of directories to prepend
# @set PATH update PATH with the directories prepended
Env::pathPrepend() {
  local arg
  for arg in "$@"; do
    if [[ -d "${arg}" && ":${PATH}:" != *":${arg}:"* ]]; then
      PATH="$(realpath "${arg}"):${PATH}"
    fi
  done
}

# @description Display message using error color (red) and exit immediately with error status 1
# @arg $1 message:String the message to display
Log::fatal() {
  echo -e "${__ERROR_COLOR}FATAL   - ${1}${__RESET_COLOR}" >&2
  Log::logFatal "$1"
  exit 1
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logDebug() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_DEBUG)); then
    Log::logMessage "${2:-DEBUG}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logInfo() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-INFO}" "$1"
  fi
}

# @description ensure running user is not root
# @exitcode 1 if current user is root
# @stderr diagnostics information is displayed
Linux::requireExecutedAsUser() {
  if [[ "$(id -u)" = "0" ]]; then
    Log::fatal "this script should be executed as normal user"
  fi
}

# @description get absolute conf file from specified conf folder deduced using these rules
#   * from absolute file (ignores <confFolder> and <extension>)
#   * relative to where script is executed (ignores <confFolder> and <extension>)
#   * from home/.bash-tools/<confFolder>
#   * from framework conf/<confFolder>
#
# @arg $1 confFolder:String the directory name (not the path) to list
# @arg $2 conf:String file to use without extension
# @arg $3 extension:String the extension (.sh by default)
#
# @stdout absolute conf filename
# @exitcode 1 if file is not found in any location
Conf::getAbsoluteFile() {
  local confFolder="$1"
  local conf="$2"
  local extension="${3-.sh}"
  if [[ -n "${extension}" && "${extension:0:1}" != "." ]]; then
    extension=".${extension}"
  fi

  testAbs() {
    local result
    result="$(realpath -e "$1" 2>/dev/null)"
    # shellcheck disable=SC2181
    if [[ "$?" = "0" && -f "${result}" ]]; then
      echo "${result}"
      return 0
    fi
    return 1
  }

  # conf is absolute file (including extension)
  testAbs "${confFolder}${extension}" && return 0
  # conf is absolute file
  testAbs "${confFolder}" && return 0
  # conf is absolute file (including extension)
  testAbs "${conf}${extension}" && return 0
  # conf is absolute file
  testAbs "${conf}" && return 0

  # relative to where script is executed (including extension)
  if [[ -n "${CURRENT_DIR+xxx}" ]]; then
    testAbs "$(File::concatenatePath "${CURRENT_DIR}" "${confFolder}")/${conf}${extension}" && return 0
  fi
  # from home/.bash-tools/<confFolder>
  testAbs "$(File::concatenatePath "${HOME}/.bash-tools" "${confFolder}")/${conf}${extension}" && return 0

  if [[ -n "${FRAMEWORK_ROOT_DIR+xxx}" ]]; then
    # from framework conf/<confFolder> (including extension)
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}${extension}" && return 0

    # from framework conf/<confFolder>
    testAbs "$(File::concatenatePath "${FRAMEWORK_ROOT_DIR}/conf" "${confFolder}")/${conf}" && return 0
  fi

  # file not found
  Log::displayError "conf file '${conf}' not found"

  return 1
}

# @description check if dsn file has all the mandatory variables set
# Mandatory variables are: HOSTNAME, USER, PASSWORD, PORT
#
# @arg $1 dsnFileName:String dsn absolute filename
# @set HOSTNAME loaded from dsn file
# @set PORT loaded from dsn file
# @set USER loaded from dsn file
# @set PASSWORD loaded from dsn file
# @exitcode 0 on valid file
# @exitcode 1 if one of the properties of the conf file is invalid or if file not found
# @stderr log output if error found in conf file
Database::checkDsnFile() {
  local dsnFileName="$1"
  if [[ ! -f "${dsnFileName}" ]]; then
    Log::displayError "dsn file ${dsnFileName} not found"
    return 1
  fi

  (
    unset HOSTNAME PORT PASSWORD USER
    # shellcheck source=/src/Database/testsData/dsn_valid.env
    source "${dsnFileName}"
    if [[ -z ${HOSTNAME+x} ]]; then
      Log::displayError "dsn file ${dsnFileName} : HOSTNAME not provided"
      return 1
    fi
    if [[ -z "${HOSTNAME}" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : HOSTNAME value not provided"
    fi
    if [[ "${HOSTNAME}" = "localhost" ]]; then
      Log::displayWarning "dsn file ${dsnFileName} : check that HOSTNAME should not be 127.0.0.1 instead of localhost"
    fi
    if [[ -z "${PORT+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT not provided"
      return 1
    fi
    if ! [[ ${PORT} =~ ^[0-9]+$ ]]; then
      Log::displayError "dsn file ${dsnFileName} : PORT invalid"
      return 1
    fi
    if [[ -z "${USER+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : USER not provided"
      return 1
    fi
    if [[ -z "${PASSWORD+x}" ]]; then
      Log::displayError "dsn file ${dsnFileName} : PASSWORD not provided"
      return 1
    fi
  )
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logFatal() {
  Log::logMessage "${2:-FATAL}" "$1"
}

# @description Internal: common log message
# @example text
#   [date]|[levelMsg]|message
#
# @example text
#   2020-01-19 19:20:21|ERROR  |log error
#   2020-01-19 19:20:21|SKIPPED|log skipped
#
# @arg $1 levelMsg:String message's level description (eg: STATUS, ERROR, ...)
# @arg $2 msg:String the message to display
# @env BASH_FRAMEWORK_LOG_FILE String log file to use, do nothing if empty
# @env BASH_FRAMEWORK_LOG_LEVEL int log level log only if > OFF or fatal messages
# @stderr diagnostics information is displayed
# @require Env::requireLoad
# @require Log::requireLoad
Log::logMessage() {
  local levelMsg="$1"
  local msg="$2"
  local date

  if [[ -n "${BASH_FRAMEWORK_LOG_FILE}" ]] && ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    date="$(date '+%Y-%m-%d %H:%M:%S')"
    touch "${BASH_FRAMEWORK_LOG_FILE}"
    printf "%s|%7s|%s\n" "${date}" "${levelMsg}" "${msg}" >>"${BASH_FRAMEWORK_LOG_FILE}"
  fi
}

# @description concatenate 2 paths and ensure the path is correct using realpath -m
# @arg $1 basePath:String
# @arg $2 subPath:String
# @require Linux::requireRealpathCommand
File::concatenatePath() {
  local basePath="$1"
  local subPath="$2"
  local fullPath="${basePath:+${basePath}/}${subPath}"

  realpath -m "${fullPath}" 2>/dev/null
}

# @description Display message using error color (red)
# @arg $1 message:String the message to display
Log::displayError() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_ERROR)); then
    echo -e "${__ERROR_COLOR}ERROR   - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logError "$1"
}

# @description Display message using warning color (yellow)
# @arg $1 message:String the message to display
Log::displayWarning() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_WARNING)); then
    echo -e "${__WARNING_COLOR}WARN    - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logWarning "$1"
}

# @description ensure env files are loaded
# @arg $@ list of default files to load at the end
# @exitcode 1 if one of env files fails to load
# @stderr diagnostics information is displayed
# shellcheck disable=SC2120
Env::requireLoad() {
  local -a defaultFiles=("$@")
  # get list of possible config files
  local -a configFiles=()
  if [[ -n "${BASH_FRAMEWORK_ENV_FILES[0]+1}" ]]; then
    # BASH_FRAMEWORK_ENV_FILES is an array
    configFiles+=("${BASH_FRAMEWORK_ENV_FILES[@]}")
  fi
  if [[ -f "$(pwd)/.framework-config" ]]; then
    configFiles+=("$(pwd)/.framework-config")
  fi
  if [[ -f "${FRAMEWORK_ROOT_DIR}/.framework-config" ]]; then
    configFiles+=("${FRAMEWORK_ROOT_DIR}/.framework-config")
  fi
  if [[ -n "${optionBashFrameworkConfig}" && -f "${optionBashFrameworkConfig}" ]]; then
    # shellcheck disable=SC2034
    configFiles+=("${optionBashFrameworkConfig}")
  fi
  configFiles+=("${optionEnvFiles[@]}")
  configFiles+=("${defaultFiles[@]}")

  for file in "${configFiles[@]}"; do
    # shellcheck source=/.framework-config
    CURRENT_LOADED_ENV_FILE="${file}" source "${file}" || {
      Log::displayError "while loading config file: ${file}"
      return 1
    }
  done
}

# @description activate or not Log::display* and Log::log* functions
# based on BASH_FRAMEWORK_DISPLAY_LEVEL and BASH_FRAMEWORK_LOG_LEVEL
# environment variables loaded by Env::requireLoad
# try to create log file and rotate it if necessary
# @noargs
# @set BASH_FRAMEWORK_LOG_LEVEL int to OFF level if BASH_FRAMEWORK_LOG_FILE is empty or not writable
# @env BASH_FRAMEWORK_DISPLAY_LEVEL int
# @env BASH_FRAMEWORK_LOG_LEVEL int
# @env BASH_FRAMEWORK_LOG_FILE String
# @env BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION int do log rotation if > 0
# @exitcode 0 always successful
# @stderr diagnostics information about log file is displayed
# @require Env::requireLoad
# @require UI::requireTheme
Log::requireLoad() {
  if [[ -z "${BASH_FRAMEWORK_LOG_FILE:-}" ]]; then
    BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
    export BASH_FRAMEWORK_LOG_LEVEL
  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    if [[ ! -f "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      if
        ! mkdir -p "$(dirname "${BASH_FRAMEWORK_LOG_FILE}")" 2>/dev/null ||
          ! touch --no-create "${BASH_FRAMEWORK_LOG_FILE}" 2>/dev/null
      then
        BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
        echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
      fi
    elif [[ ! -w "${BASH_FRAMEWORK_LOG_FILE}" ]]; then
      BASH_FRAMEWORK_LOG_LEVEL=${__LEVEL_OFF}
      echo -e "${__ERROR_COLOR}ERROR   - File ${BASH_FRAMEWORK_LOG_FILE} is not writable${__RESET_COLOR}" >&2
    fi

  fi

  if ((BASH_FRAMEWORK_LOG_LEVEL > __LEVEL_OFF)); then
    # will always be created even if not in info level
    Log::logMessage "INFO" "Logging to file ${BASH_FRAMEWORK_LOG_FILE} - Log level ${BASH_FRAMEWORK_LOG_LEVEL}"
    if ((BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION > 0)); then
      Log::rotate "${BASH_FRAMEWORK_LOG_FILE}" "${BASH_FRAMEWORK_LOG_FILE_MAX_ROTATION}"
    fi
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logError() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_ERROR)); then
    Log::logMessage "${2:-ERROR}" "$1"
  fi
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logWarning() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_WARNING)); then
    Log::logMessage "${2:-WARNING}" "$1"
  fi
}

# @description To be called before logging in the log file
# @arg $1 file:string log file name
# @arg $2 maxLogFilesCount:int maximum number of log files
Log::rotate() {
  local file="$1"
  local maxLogFilesCount="${2:-5}"

  if [[ ! -f "${file}" ]]; then
    Log::displaySkipped "Log file ${file} doesn't exist yet"
    return 0
  fi
  for i in $(seq $((maxLogFilesCount - 1)) -1 1); do
    Log::displayInfo "Log rotation ${file}.${i} to ${file}.$((i + 1))"
    mv "${file}."{"${i}","$((i + 1))"} &>/dev/null || true
  done
  if cp "${file}" "${file}.1" &>/dev/null; then
    echo >"${file}" # reset log file
    Log::displayInfo "Log rotation ${file} to ${file}.1"
  fi
}

# @description ensure command realpath is available
# @exitcode 1 if realpath command not available
# @stderr diagnostics information is displayed
Linux::requireRealpathCommand() {
  Assert::commandExists realpath
}

# @description load color theme
# @noargs
# @env BASH_FRAMEWORK_THEME String theme to use
# @exitcode 0 always successful
UI::requireTheme() {
  UI::theme "${BASH_FRAMEWORK_THEME-default}"
}

# @description check if command specified exists or return 1
# with error and message if not
#
# @arg $1 commandName:String on which existence must be checked
# @arg $2 helpIfNotExists:String a help command to display if the command does not exist
#
# @exitcode 1 if the command specified does not exist
# @stderr diagnostic information + help if second argument is provided
Assert::commandExists() {
  local commandName="$1"
  local helpIfNotExists="$2"

  "${BASH_FRAMEWORK_COMMAND:-command}" -v "${commandName}" >/dev/null 2>/dev/null || {
    Log::displayError "${commandName} is not installed, please install it"
    if [[ -n "${helpIfNotExists}" ]]; then
      Log::displayInfo "${helpIfNotExists}"
    fi
    return 1
  }
  return 0
}

# @description Display message using skip color (yellow)
# @arg $1 message:String the message to display
Log::displaySkipped() {
  if ((BASH_FRAMEWORK_DISPLAY_LEVEL >= __LEVEL_INFO)); then
    echo -e "${__SKIPPED_COLOR}SKIPPED - ${1}${__RESET_COLOR}" >&2
  fi
  Log::logSkipped "$1"
}

# @description load colors theme constants
# @warning if tty not opened, noColor theme will be chosen
# @arg $1 theme:String the theme to use (default, noColor)
# @arg $@ args:String[]
# @set __ERROR_COLOR String indicate error status
# @set __INFO_COLOR String indicate info status
# @set __SUCCESS_COLOR String indicate success status
# @set __WARNING_COLOR String indicate warning status
# @set __SKIPPED_COLOR String indicate skipped status
# @set __DEBUG_COLOR String indicate debug status
# @set __HELP_COLOR String indicate help status
# @set __TEST_COLOR String not used
# @set __TEST_ERROR_COLOR String not used
# @set __HELP_TITLE_COLOR String used to display help title in help strings
# @set __HELP_OPTION_COLOR String used to display highlight options in help strings
#
# @set __RESET_COLOR String reset default color
#
# @set __HELP_EXAMPLE String to remove
# @set __HELP_TITLE String to remove
# @set __HELP_NORMAL String to remove
# shellcheck disable=SC2034
UI::theme() {
  local theme="${1-default}"
  if [[ ! "${theme}" =~ -force$ ]] && ! Assert::tty; then
    theme="noColor"
  fi
  case "${theme}" in
    default | default-force)
      theme="default"
      ;;
    noColor) ;;
    *)
      Log::fatal "invalid theme provided"
      ;;
  esac
  if [[ "${theme}" = "default" ]]; then
    BASH_FRAMEWORK_THEME="default"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR='\e[31m'         # Red
    __INFO_COLOR='\e[44m'          # white on lightBlue
    __SUCCESS_COLOR='\e[32m'       # Green
    __WARNING_COLOR='\e[33m'       # Yellow
    __SKIPPED_COLOR='\e[33m'       # Yellow
    __DEBUG_COLOR='\e[37m'         # Grey
    __HELP_COLOR='\e[7;49;33m'     # Black on Gold
    __TEST_COLOR='\e[100m'         # Light magenta
    __TEST_ERROR_COLOR='\e[41m'    # white on red
    __HELP_TITLE_COLOR="\e[1;37m"  # Bold
    __HELP_OPTION_COLOR="\e[1;34m" # Blue
    # Internal: reset color
    __RESET_COLOR='\e[0m' # Reset Color
    # shellcheck disable=SC2155,SC2034
    __HELP_EXAMPLE="$(echo -e "\e[2;97m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_TITLE="$(echo -e "\e[1;37m")"
    # shellcheck disable=SC2155,SC2034
    __HELP_NORMAL="$(echo -e "\033[0m")"
  else
    BASH_FRAMEWORK_THEME="noColor"
    # check colors applicable https://misc.flogisoft.com/bash/tip_colors_and_formatting
    __ERROR_COLOR=''
    __INFO_COLOR=''
    __SUCCESS_COLOR=''
    __WARNING_COLOR=''
    __SKIPPED_COLOR=''
    __DEBUG_COLOR=''
    __HELP_COLOR=''
    __TEST_COLOR=''
    __TEST_ERROR_COLOR=''
    __HELP_TITLE_COLOR=''
    __HELP_OPTION_COLOR=''
    # Internal: reset color
    __RESET_COLOR=''
    __HELP_EXAMPLE=''
    __HELP_TITLE=''
    __HELP_NORMAL=''
  fi
}

# @description check if tty (interactive mode) is active
# @noargs
# @exitcode 1 if tty not active
# @env NON_INTERACTIVE if 1 consider as not interactive even if environment is interactive
# @env INTERACTIVE if 1 consider as interactive even if environment is not interactive
# @stderr diagnostic information + help if second argument is provided
Assert::tty() {
  if [[ "${NON_INTERACTIVE:-0}" = "1" ]]; then
    return 1
  fi
  if [[ "${INTERACTIVE:-0}" = "1" ]]; then
    return 0
  fi
  [[ -t 1 || -t 2 ]]
}

# @description log message to file
# @arg $1 message:String the message to display
Log::logSkipped() {
  if ((BASH_FRAMEWORK_LOG_LEVEL >= __LEVEL_INFO)); then
    Log::logMessage "${2:-SKIPPED}" "$1"
  fi
}

# FUNCTIONS

facade_main_embedFrameworkFunctionbinFiletpl() {
# REQUIRES
Linux::requireExecutedAsUser
Env::requireLoad
Log::requireLoad
Linux::requireRealpathCommand
UI::requireTheme
Compiler::Facade::requireCommandBinDir

# @require Compiler::Facade::requireCommandBinDir

# shellcheck disable=SC2154,SC2016
functionToCall='Db::queryOneDatabase'
"${functionToCall}" "$@"

}

facade_main_embedFrameworkFunctionbinFiletpl "$@"
" Compiler::Embed::extractFileFromBase64 \ "${embed_function_DbQueryOneDatabase}" \ @@ -1350,6 +1373,7 @@ dbScriptAllDatabasesCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1731,11 +1755,11 @@ dbScriptAllDatabasesCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Allows to execute a script on each database of specified mysql server")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Allows to execute a script on each database of specified mysql server")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--jobs|-j ]" "[--bar|-b]" "[--from-dsn|-f ]" "[--database ]" "[--output|-o ]" "[--log-format|-l ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1744,120 +1768,125 @@ dbScriptAllDatabasesCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ script\ that\ will\ be\ executed\ on\ each\ databases) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " [${__HELP_OPTION_COLOR}scriptArguments${__HELP_NORMAL} {list} (optional)]" local -a helpArray # shellcheck disable=SC2054 helpArray=(optional\ parameters\ to\ pass\ to\ the\ script) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--jobs${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-j ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(specify\ the\ number\ of\ db\ to\ query\ in\ parallel) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: 1' echo -e " ${__HELP_OPTION_COLOR}--bar${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-b${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Show\ progress\ as\ a\ progress\ bar.\ In\ the\ bar\ is\ shown:\ %\ of\ jobs\ completed\,\ estimated\ seconds\ left\,\ and\ number\ of\ jobs\ started.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}SCRIPT OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--from-dsn${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-f ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(target\ mysql\ server) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--database ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(if\ provided\ will\ check\ only\ this\ db\,\ otherwise\ script\ will\ be\ executed\ on\ all\ dbs\ of\ mysql\ server) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--output${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-o ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(output\ directory\,\ see\ log-format\ option) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-format${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-l ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(if\ output\ dir\ provided\,\ will\ log\ each\ db\ result\ to\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: none' + echo ' Possible values: none|log' echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}NOTE:${__HELP_NORMAL} the use of output, log-format, verbose options highly depends on the script used @@ -1903,7 +1932,7 @@ ${__HELP_TITLE}USE CASES:${__HELP_NORMAL} echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/doc b/bin/doc index c14e887e..cce6f470 100755 --- a/bin/doc +++ b/bin/doc @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/doc +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/doc # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description checkout usage doc below # # [DockerNamespace usage](DockerUsage.md ':include') @@ -247,6 +270,8 @@ Docker::runBuildContainer() { if [[ "${optionContinuousIntegrationMode}" = "0" ]]; then localDockerRunArgs+=(-v "/tmp:/tmp") fi + localDockerRunArgs+=(-e KEEP_TEMP_FILES="${KEEP_TEMP_FILES}") + localDockerRunArgs+=(-e BATS_FIX_TEST="${BATS_FIX_TEST:-0}") # shellcheck disable=SC2154 Log::displayInfo "Using ${optionVendor}:${optionBashVersion}" @@ -1198,6 +1223,7 @@ docCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1466,11 +1492,11 @@ docCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "generate markdown documentation")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "generate markdown documentation")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--skip-docker-build]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1479,79 +1505,83 @@ docCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(skip\ docker\ image\ build\ if\ option\ provided) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo echo -n -e "${__HELP_TITLE_COLOR}VERSION: ${__RESET_COLOR}" echo '1.0' @@ -1565,7 +1595,7 @@ docCommand() { echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/gitIsAncestorOf b/bin/gitIsAncestorOf index dcb86112..17890304 100755 --- a/bin/gitIsAncestorOf +++ b/bin/gitIsAncestorOf @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitIsAncestorOf +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitIsAncestorOf # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description ensure env files are loaded # @arg $@ list of default files to load at the end # @exitcode 1 if one of env files fails to load @@ -911,6 +934,7 @@ gitIsAncestorOfCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1205,11 +1229,11 @@ gitIsAncestorOfCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "check if commit is inside a given branch")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "check if commit is inside a given branch")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1218,84 +1242,88 @@ gitIsAncestorOfCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ branch\ in\ which\ the\ commit\ will\ be\ searched) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}commit${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ commit\ oid\ to\ check) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}EXIT CODES:${__HELP_NORMAL} ${__HELP_OPTION_COLOR}1${__HELP_NORMAL}: if commit does not exists @@ -1313,7 +1341,7 @@ ${__HELP_OPTION_COLOR}2${__HELP_NORMAL}: if commit is not included in given bran echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/gitIsBranch b/bin/gitIsBranch index 695f839c..b9177817 100755 --- a/bin/gitIsBranch +++ b/bin/gitIsBranch @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitIsBranch +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitIsBranch # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description ensure env files are loaded # @arg $@ list of default files to load at the end # @exitcode 1 if one of env files fails to load @@ -910,6 +933,7 @@ gitIsBranchCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1188,11 +1212,11 @@ gitIsBranchCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "show an error if branchName is not a known branch")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "show an error if branchName is not a known branch")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1201,79 +1225,83 @@ gitIsBranchCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ branch\ name\ to\ check) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo echo -n -e "${__HELP_TITLE_COLOR}VERSION: ${__RESET_COLOR}" echo '1.0' @@ -1287,7 +1315,7 @@ gitIsBranchCommand() { echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/gitRenameBranch b/bin/gitRenameBranch index edae7127..8ab896d6 100755 --- a/bin/gitRenameBranch +++ b/bin/gitRenameBranch @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitRenameBranch +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitRenameBranch # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if tty (interactive mode) is active # @noargs # @exitcode 1 if tty not active @@ -948,6 +971,7 @@ gitRenameBranchCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1272,11 +1296,11 @@ gitRenameBranchCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "rename git local branch, push new branch and delete old branch")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "rename git local branch, push new branch and delete old branch")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--assume-yes|--yes|-y]" "[--push|-p]" "[--delete|-d]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1285,101 +1309,105 @@ gitRenameBranchCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ branch\ name\ to\ check) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " [${__HELP_OPTION_COLOR}oldBranchName${__HELP_NORMAL} {single}]" local -a helpArray # shellcheck disable=SC2054 helpArray=(the\ name\ of\ the\ old\ branch\ if\ not\ current\ one) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--assume-yes${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}--yes${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-y${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054,SC2206 mapfile -t helpArray < <(assumeYesHelpCallback) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--push${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-p${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(push\ the\ new\ branch) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--delete${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-d${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(delete\ the\ old\ remote\ branch) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}EXIT CODES:${__HELP_NORMAL} ${__HELP_OPTION_COLOR}1${__HELP_NORMAL} : if current directory is not a git repository @@ -1406,7 +1434,7 @@ ${__HELP_OPTION_COLOR}9${__HELP_NORMAL} : if failed to push the new branch""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 @@ -1414,8 +1442,8 @@ ${__HELP_OPTION_COLOR}9${__HELP_NORMAL} : if failed to push the new branch""" } assumeYesHelpCallback() { - echo "do not ask for confirmation (use with caution)" $'\n' - echo ' Automatic yes to prompts; assume "y" as answer to all prompts' $'\n' + echo "do not ask for confirmation (use with caution)" + echo ' Automatic yes to prompts; assume "y" as answer to all prompts' echo ' and run non-interactively.' } diff --git a/bin/installRequirements b/bin/installRequirements index 161d9568..945a7c8a 100755 --- a/bin/installRequirements +++ b/bin/installRequirements @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/installRequirements +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/installRequirements # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -110,123 +110,146 @@ Array::join() { # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description ensure env files are loaded # @arg $@ list of default files to load at the end # @exitcode 1 if one of env files fails to load @@ -1022,6 +1045,7 @@ installRequirementsCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1278,11 +1302,11 @@ installRequirementsCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "installs requirements")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "installs requirements")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1291,72 +1315,76 @@ installRequirementsCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}INSTALLS REQUIREMENTS:${__HELP_NORMAL} - fchastanet/bash-tools-framework @@ -1375,7 +1403,7 @@ ${__HELP_TITLE}INSTALLS REQUIREMENTS:${__HELP_NORMAL} echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/mysql2puml b/bin/mysql2puml index b0ee2a1c..c22dd18d 100755 --- a/bin/mysql2puml +++ b/bin/mysql2puml @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/mysql2puml +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/mysql2puml # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -110,123 +110,146 @@ Array::contains() { # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description get absolute conf file from specified conf folder deduced using these rules # * from absolute file (ignores and ) # * relative to where script is executed (ignores and ) @@ -1112,6 +1135,7 @@ mysql2pumlCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1404,11 +1428,11 @@ mysql2pumlCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "convert mysql dump sql schema to plantuml format")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "convert mysql dump sql schema to plantuml format")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--skin ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1417,14 +1441,14 @@ mysql2pumlCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(sql\ filepath\ to\ parse\ \(read\ from\ stdin\ if\ not\ provided\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--skin ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(header\ configuration\ of\ the\ plant\ uml\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: default' echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" @@ -1432,72 +1456,76 @@ mysql2pumlCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}EXAMPLE 1:${__HELP_NORMAL} ${__HELP_EXAMPLE}mysql2puml dump.dql${__HELP_NORMAL} @@ -1526,7 +1554,7 @@ ${__HELP_TITLE}LIST OF AVAILABLE SKINS:${__HELP_NORMAL} echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/upgradeGithubRelease b/bin/upgradeGithubRelease index 20e4b8b1..3c5dc3a5 100755 --- a/bin/upgradeGithubRelease +++ b/bin/upgradeGithubRelease @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/upgradeGithubRelease +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/upgradeGithubRelease # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # Checks if file can be created in folder # @@ -1238,6 +1261,7 @@ upgradeGithubReleaseCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1599,11 +1623,11 @@ upgradeGithubReleaseCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "retrieve latest binary release from github and install it")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "retrieve latest binary release from github and install it")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--version-arg ]" "[--current-version|-c ]" "[--exact-version|-e ]" "[--minimal-version|-m ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1611,108 +1635,112 @@ upgradeGithubReleaseCommand() { echo -e " ${__HELP_OPTION_COLOR}targetFile${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 - helpArray=(the\ binary\ downloaded\ will\ e\ written\ to\ this\ file\ path.\ Ensure\ the\ path\ is\ writable.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'the binary downloaded will be written to this file path.\n\n Ensure the path is writable.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}githubUrlPattern${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'the url pattern to use to download the binary, see examples below. \n @version@ is template variable that will be replaced by the latest \n version tag found on github.') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'the url pattern to use to download the binary, see examples below.\n\n @version@ is template variable that will be replaced by the latest \n\n version tag found on github.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}VERSION MANAGEMENT:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--version-arg ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'The argument that will be provided to the currently installed binary\nto check the version of the software. See options constraints below.') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'The argument that will be provided to the currently installed binary\n\n to check the version of the software. \n\n See options constraints below.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: --version' echo -e " ${__HELP_OPTION_COLOR}--current-version${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-c ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'Sometimes the command to retrieve the version is complicated.\nSome command needs you to parse json or other commands provides\nmultiple sub command versions. In this case you can provide the\nversion you currently have, see examples below.\nSee options constraints below.') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'Sometimes the command to retrieve the version is complicated. \n\n Some command needs you to parse json or other commands provides multiple \n\n sub command versions. In this case you can provide the version you \n\n currently have, see examples below. \n\n See options constraints below.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--exact-version${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-e ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'if provided and currently installed binary is not this exactVersion,\n This exact version of the binary will be installed.\n See options constraints below.') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'if provided and currently installed binary is not this exactVersion,\n\n This exact version of the binary will be installed.\n\n See options constraints below.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--minimal-version${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-m ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=($'if provided and currently installed binary is below this minimalVersion,\na new version of the binary will be installed.\nIf this argument is not provided, the latest binary is unconditionally downloaded from github.\nSee options constraints below.') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=($'if provided and currently installed binary is below this \n\n minimalVersion, a new version of the binary will be installed. \n\n If this argument is not provided, the latest binary is unconditionally \n\n downloaded from github. \n\n See options constraints below.') + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}OPTIONS CONSTRAINTS:${__HELP_NORMAL} @@ -1764,7 +1792,7 @@ ${__HELP_EXAMPLE}upgradeGithubRelease /usr/local/bin/oq --exact-version 1.3.4 -- echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/waitForIt b/bin/waitForIt index 2bc03784..c2dd886e 100755 --- a/bin/waitForIt +++ b/bin/waitForIt @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/waitForIt +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/waitForIt # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -110,123 +110,146 @@ Array::contains() { # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -971,6 +994,7 @@ waitForItCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1322,11 +1346,11 @@ waitForItCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "wait for host:port to be available")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "wait for host:port to be available")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--timeout|-t ]" "--host|-i " "--port|-p " "[--algorithm|--algo ]" "[--exec-command-on-success-only|--strict|-s]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1335,107 +1359,111 @@ waitForItCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(Execute\ command\ with\ args\ after\ the\ test\ finishes\ or\ exit\ with\ status\ code\ if\ no\ command\ provided.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--timeout${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-t ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Timeout\ in\ seconds\,\ zero\ for\ no\ timeout.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: 15' echo -e " ${__HELP_OPTION_COLOR}--host${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-i ${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Host\ or\ IP\ under\ test.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--port${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-p ${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(TCP\ port\ under\ test.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--algorithm${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}--algo ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=($'Algorithm to use Check algorithms list below. \n (default: automatic selection based on commands availability and timeout option value).') - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--exec-command-on-success-only${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}--strict${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-s${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Only\ execute\ sub-command\ if\ the\ test\ succeeds.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--bash-framework-config ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}EXIT STATUS CODES:${__HELP_NORMAL} ${__HELP_OPTION_COLOR}0${__HELP_NORMAL}: the host/port is available @@ -1462,7 +1490,7 @@ ${__HELP_OPTION_COLOR}whileLoopWithTcp${__HELP_NORMAL}: timeout command simulate echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/bin/waitForMysql b/bin/waitForMysql index 6ac9b60a..50f512de 100755 --- a/bin/waitForMysql +++ b/bin/waitForMysql @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/waitForMysql +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/waitForMysql # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description check if command specified exists or return 1 # with error and message if not # @@ -939,6 +962,7 @@ waitForMysqlCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1283,11 +1307,11 @@ waitForMysqlCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "wait for mysql to be ready")" + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "wait for mysql to be ready")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]" "[ARGUMENTS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--timeout|-t ]" "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1296,29 +1320,29 @@ waitForMysqlCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(Mysql\ host\ name) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}mysqlPort${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Mysql\ port) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}mysqlUserArg${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Mysql\ user\ name) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}mysqlPasswordArg${__HELP_NORMAL} {single} (mandatory)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Mysql\ password) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo echo -e "${__HELP_TITLE_COLOR}OPTIONS:${__RESET_COLOR}" echo -e " ${__HELP_OPTION_COLOR}--timeout${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-t ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Timeout\ in\ seconds\,\ zero\ for\ no\ timeout.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo ' Default value: 15' echo echo -e "${__HELP_TITLE_COLOR}GLOBAL OPTIONS:${__RESET_COLOR}" @@ -1326,72 +1350,76 @@ waitForMysqlCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e """ ${__HELP_TITLE}EXIT STATUS CODES:${__HELP_NORMAL} ${__HELP_OPTION_COLOR}0${__HELP_NORMAL}: mysql is available @@ -1410,7 +1438,7 @@ ${__HELP_OPTION_COLOR}2${__HELP_NORMAL}: timeout reached""" echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/conf/dbScripts/extractData b/conf/dbScripts/extractData index 2a801303..b53f7701 100755 --- a/conf/dbScripts/extractData +++ b/conf/dbScripts/extractData @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/conf/dbScripts/extractData +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/conf/dbScripts/extractData # FACADE # ensure that no user aliases could interfere with diff --git a/install b/install index 3c177e36..02423c2e 100755 --- a/install +++ b/install @@ -5,7 +5,7 @@ # @generated ############################################################################### # shellcheck disable=SC2288,SC2034 -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/install +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/install # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=. # FACADE @@ -94,123 +94,146 @@ trap cleanOnExit EXIT HUP QUIT ABRT TERM # @description concat each element of an array with a separator # but wrapping text when line length is more than provided argument -# The algorithm will try not to cut the array element if can +# The algorithm will try not to cut the array element if it can. +# if an arg can be placed on current line it will be, +# otherwise current line is printed and arg is added to the new +# current line +# empty arg is interpreted as a new line. # # @arg $1 glue:String # @arg $2 maxLineLength:int # @arg $3 indentNextLine:int # @arg $@ array:String[] -Array::wrap() { +Array::wrap2() { local glue="${1-}" - local -i glueLength=0 + local -i glueLength="${#glue}" shift || true local -i maxLineLength=$1 shift || true local -i indentNextLine=$1 + shift || true local indentStr="" if ((indentNextLine > 0)); then indentStr="$(head -c "${indentNextLine}" 0)); do - argNoAnsi="$(echo "${arg%%*( )}" | Filters::removeAnsiCodes)" - ((argNoAnsiLength = ${#argNoAnsi})) || true - if (($# < 1 && argNoAnsiLength == 0)); then - break + eatNextSpaces() { + if [[ "${currentChar}" != [[:space:]] ]]; then + ((i--)) || true + return 0 fi - if [[ "${arg}" = $'\n' ]]; then - if [[ "${needEcho}" = "1" ]]; then - needEcho="0" + for (( ; i < textLength; i++)); do + if [[ "${text:${i}:1}" != [[:space:]] ]]; then + ((i--)) || true + break fi - echo "" - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - shift || return 0 - arg="$1" - elif ((argNoAnsiLength < maxLineLength - currentLineLength - glueLength)); then - # arg can be stored as a whole on current line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + done + } + printCurrentLine() { + echo -e "${currentLine}" | sed -E -e 's/[[:blank:]]*$//' + ((isNewline = 1)) + currentLine="${indentStr}" + ((currentLineLength = indentNextLine)) || true + } + nextLine() { + printCurrentLine + eatNextSpaces + } + local currentLine currentChar ansiCode + local -i isAnsiCode currentLineLength=0 isNewline=1 textLength=0 + local text="" + local arg="" + while (($# > 0)); do + arg="$1" + shift || true + if ((${#arg} == 0)) || [[ "${arg}" = $'\n' ]]; then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" + echo + continue + fi + local textFirstLine + textFirstLine="$(echo "${arg}" | head -n 1)" + text="$(echo "${arg}" | sed -E '1d')" + ((textLength = ${#text})) || true + local textFirstLineNoAnsi + textFirstLineNoAnsi="$(echo "${textFirstLine}" | Filters::removeAnsiCodes)" + local -i textFirstLineNoAnsiLength=0 + ((textFirstLineNoAnsiLength = ${#textFirstLineNoAnsi})) || true + + if ((isNewline == 0)); then + glueLength="${#glue}" + else + glueLength="0" + fi + if ((currentLineLength + textFirstLineNoAnsiLength + glueLength > maxLineLength)); then + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + printCurrentLine fi - echo -e -n "${arg}" | sed 's/[\t ]*$//g' - needEcho="1" - ((currentLineLength += argNoAnsiLength)) - ((glueLength = ${#glue})) || true - shift || return 0 - arg="$1" + # restore current arg without considering first line + text="${arg}" + ((textLength = ${#text})) || true else - if ((argNoAnsiLength >= (maxLineLength - indentNextLine))); then - if ((currentLineLength == 0 && firstLine == 0)); then - echo -n "${indentStr}" - ((currentLineLength += indentNextLine)) - fi - # arg can be stored on a whole line - if ((glueLength > 0)); then - echo -e -n "${glue}" - ((currentLineLength += glueLength)) + if ((isNewline == 0)); then + # isNewline = 0 means currentLine is not empty + currentLine+="${glue}" + ((currentLineLength += glueLength)) || true + fi + currentLine+="${textFirstLine}" + isNewline="0" + ((currentLineLength += ${#textFirstLine})) || true + fi + + for ((i = 0; i < textLength; i++)); do + currentChar="${text:${i}:1}" + + if [[ "${currentChar}" = $'\r' ]]; then + # ignore + true + elif [[ "${currentChar}" = "\x1b" ]]; then + isAnsiCode=1 + ansiCode+="${currentChar}" + elif ((isAnsiCode == 1)); then + ansiCode+="${currentChar}" + if [[ "${currentChar}" =~ [mGKHF] ]]; then + isAnsiCode=0 + echo -e "${ansiCode}" + elif [[ "${currentChar}" = $'\n' ]]; then + # invalid ansi code, ignore it + isAnsiCode=0 + ansiCode="" fi - local -i length - ((length = maxLineLength - currentLineLength)) || true - echo -e "${arg:0:${length}}" | sed 's/[\t ]*$//g' - ((currentLineLength = 0)) || true - ((glueLength = 0)) || true - arg="${arg:${length}}" - needEcho="0" else - # arg cannot be stored on a whole line, so we add it on next line as a whole - echo - echo -e -n "${indentStr}${arg}" | sed 's/[\t ]*$//g' - ((glueLength = ${#glue})) || true - ((currentLineLength = argNoAnsiLength)) - arg="" # allows to go to next arg - needEcho="1" - fi - if [[ -z "${arg}" ]]; then - shift || return 0 - arg="$1" + # non ansi code + if [[ "${currentChar}" = $'\n' ]] || ((currentLineLength == maxLineLength)); then + nextLine + elif [[ "${currentChar}" = "\t" ]]; then + if ((currentLineLength + 2 <= maxLineLength)); then + currentLine+=" " + ((isNewline = 0)) || true + ((currentLineLength = currentLineLength + 2)) + else + nextLine + fi + else + currentLine+="${currentChar}" + ((isNewline = 0)) || true + ((++currentLineLength)) + fi fi - fi - ((firstLine = 0)) || true + done done - if [[ "${needEcho}" = "1" ]]; then - echo + if [[ "${currentLine}" != "" ]] && [[ ! "${currentLine}" =~ ^[\ \t]+$ ]]; then + printCurrentLine fi } -#set -x -#Array::wrap ":" 40 0 "Lorem ipsum dolor sit amet," "consectetur adipiscing elit." "Curabitur ac elit id massa" "condimentum finibus." - # @description ensure env files are loaded # @arg $@ list of default files to load at the end # @exitcode 1 if one of env files fails to load @@ -910,6 +933,7 @@ installCommand() { optionNoColor="0" local -i options_parse_optionParsedCountOptionNoColor ((options_parse_optionParsedCountOptionNoColor = 0)) || true + optionTheme="default" local -i options_parse_optionParsedCountOptionTheme ((options_parse_optionParsedCountOptionTheme = 0)) || true optionHelp="0" @@ -1166,13 +1190,13 @@ installCommand() { Log::displayDebug "Command ${SCRIPT_NAME} - parse arguments: ${BASH_FRAMEWORK_ARGV[*]}" Log::displayDebug "Command ${SCRIPT_NAME} - parse filtered arguments: ${BASH_FRAMEWORK_ARGV_FILTERED[*]}" elif [[ "${options_parse_cmd}" = "help" ]]; then - echo -e "$(Array::wrap " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Install dependent softwares and configuration needed to use bash-tools + echo -e "$(Array::wrap2 " " 80 0 "${__HELP_TITLE_COLOR}DESCRIPTION:${__RESET_COLOR}" "Install dependent softwares and configuration needed to use bash-tools - GNU parallel - Install default configuration files")" echo - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" - echo -e "$(Array::wrap " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" "${SCRIPT_NAME}" "[OPTIONS]")" + echo -e "$(Array::wrap2 " " 80 2 "${__HELP_TITLE_COLOR}USAGE:${__RESET_COLOR}" \ "${SCRIPT_NAME}" \ "[--bash-framework-config ]" "[--config]" "[--verbose|-v]" "[-vv]" "[-vvv]" "[--env-file ]" "[--no-color]" "[--theme ]" "[--help|-h]" "[--version]" "[--quiet|-q]" "[--log-level ]" "[--log-file ]" "[--display-level ]")" echo @@ -1181,72 +1205,76 @@ installCommand() { local -a helpArray # shellcheck disable=SC2054 helpArray=(use\ alternate\ bash\ framework\ configuration.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--config${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ configuration) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--verbose${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-v${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(info\ level\ verbose\ mode\ \(alias\ of\ --display-level\ INFO\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(debug\ level\ verbose\ mode\ \(alias\ of\ --display-level\ DEBUG\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}-vvv${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(trace\ level\ verbose\ mode\ \(alias\ of\ --display-level\ TRACE\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--env-file ${__HELP_NORMAL} {list} (optional)" local -a helpArray # shellcheck disable=SC2054 helpArray=(Load\ the\ specified\ env\ file\ \(deprecated\,\ please\ use\ --bash-framework-config\ option\ instead\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--no-color${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Produce\ monochrome\ output.\ alias\ of\ --theme\ noColor.) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--theme ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(choose\ color\ theme\ \(default\,\ default-force\ or\ noColor\)\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(choose\ color\ theme\ -\ default-force\ means\ colors\ will\ be\ produced\ even\ if\ command\ is\ piped) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Default value: default' + echo ' Possible values: default|default-force|noColor' echo -e " ${__HELP_OPTION_COLOR}--help${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-h${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Display\ this\ command\ help) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--version${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Print\ version\ information\ and\ quit) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--quiet${__HELP_NORMAL}, ${__HELP_OPTION_COLOR}-q${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(quiet\ mode\,\ doesn\'t\ display\ any\ output) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--log-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(Set\ log\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(Set\ log\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo -e " ${__HELP_OPTION_COLOR}--log-file ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 helpArray=(Set\ log\ file) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" echo -e " ${__HELP_OPTION_COLOR}--display-level ${__HELP_NORMAL} {single}" local -a helpArray # shellcheck disable=SC2054 - helpArray=(set\ display\ level\ \(one\ of\ OFF\,\ ERROR\,\ WARNING\,\ INFO\,\ DEBUG\,\ TRACE\ value\)) - echo -e " $(Array::wrap " " 76 4 "${helpArray[@]}")" + helpArray=(set\ display\ level) + echo -e " $(Array::wrap2 " " 76 4 "${helpArray[@]}")" + echo ' Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE' echo echo -n -e "${__HELP_TITLE_COLOR}VERSION: ${__RESET_COLOR}" echo '1.0' @@ -1260,7 +1288,7 @@ installCommand() { echo -e "${__HELP_TITLE_COLOR}LICENSE:${__RESET_COLOR}" echo 'MIT License' echo - Array::wrap ' ' 76 4 "$(copyrightCallback)" + Array::wrap2 ' ' 76 4 "$(copyrightCallback)" else Log::displayError "Command ${SCRIPT_NAME} - Option command invalid: '${options_parse_cmd}'" return 1 diff --git a/kics.config b/kics.config index 07532ee4..7484fca7 100644 --- a/kics.config +++ b/kics.config @@ -1,2 +1,2 @@ -exclude-paths: - - "vendor/" +--- +exclude-paths: "vendor/**" diff --git a/src/_binaries/Converters/mysql2puml.options.tpl b/src/_binaries/Converters/mysql2puml.options.tpl index ee06be75..36da7671 100644 --- a/src/_binaries/Converters/mysql2puml.options.tpl +++ b/src/_binaries/Converters/mysql2puml.options.tpl @@ -4,6 +4,7 @@ declare commandFunctionName="mysql2pumlCommand" declare optionSkinDefault="default" declare help="convert mysql dump sql schema to plantuml format" # shellcheck disable=SC2016 +# kics-scan disable=487f4be7-3fd9-4506-a07a-eae252180c08 declare longDescription=''' ${__HELP_TITLE}EXAMPLE 1:${__HELP_NORMAL} ${__HELP_EXAMPLE}mysql2puml dump.dql${__HELP_NORMAL} diff --git a/src/_binaries/Converters/mysql2puml.sh b/src/_binaries/Converters/mysql2puml.sh index 13723c29..eceb16bf 100755 --- a/src/_binaries/Converters/mysql2puml.sh +++ b/src/_binaries/Converters/mysql2puml.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/mysql2puml +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/mysql2puml # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Converters/testsData/mysql2puml.help.txt b/src/_binaries/Converters/testsData/mysql2puml.help.txt index 570d3da5..dc8a81da 100644 --- a/src/_binaries/Converters/testsData/mysql2puml.help.txt +++ b/src/_binaries/Converters/testsData/mysql2puml.help.txt @@ -1,10 +1,11 @@ DESCRIPTION: convert mysql dump sql schema to plantuml format USAGE: mysql2puml [OPTIONS] [ARGUMENTS] -USAGE: mysql2puml [--skin ] [--bash-framework-config ] - [--config] [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] - [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] - [--log-file ] [--display-level ] +USAGE: mysql2puml [--skin ] + [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] + [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] + [--quiet|-q] [--log-level ] [--log-file ] + [--display-level ] ARGUMENTS: [inputSqlFile {single}] @@ -32,8 +33,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -41,11 +44,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE EXAMPLE 1: mysql2puml dump.dql diff --git a/src/_binaries/DbImport/dbImport.options.tpl b/src/_binaries/DbImport/dbImport.options.tpl index ed9fcd87..e8977fef 100644 --- a/src/_binaries/DbImport/dbImport.options.tpl +++ b/src/_binaries/DbImport/dbImport.options.tpl @@ -26,7 +26,7 @@ ${__HELP_EXAMPLE}TODO${__HELP_NORMAL} ${__HELP_TITLE}Example 2: import from S3${__HELP_NORMAL} ${__HELP_EXAMPLE}TODO${__HELP_NORMAL}''' # shellcheck disable=SC2116 -declare defaultFromDsnHelp=$'dsn to use for source database\n\ +declare defaultFromDsnHelp=$'dsn to use for source database\n this option is incompatible with -a|--from-aws option' % diff --git a/src/_binaries/DbImport/dbImport.sh b/src/_binaries/DbImport/dbImport.sh index 02443200..13a0b5ab 100755 --- a/src/_binaries/DbImport/dbImport.sh +++ b/src/_binaries/DbImport/dbImport.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImport +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImport # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/DbImport/dbImportProfile.bats b/src/_binaries/DbImport/dbImportProfile.bats index 9a40d19f..d7a483b6 100755 --- a/src/_binaries/DbImport/dbImportProfile.bats +++ b/src/_binaries/DbImport/dbImportProfile.bats @@ -118,7 +118,8 @@ function Database::dbImportProfile::remote_db_fully_functional_ratio_20 { #@test run "${binDir}/dbImportProfile" --verbose -f default.local -r 20 fromDb 2>&1 [[ -f "${HOME}/tableSizeQuery.sql" ]] - assert_lines_count 3 + + assert_lines_count 3 || assert_output "" assert_line --index 0 --partial "INFO - Using from dsn" assert_line --index 1 --partial "Profile generated - 2/3 tables bigger than 20% of max table size (29MB) automatically excluded" assert_line --index 2 --partial "INFO - File saved in" diff --git a/src/_binaries/DbImport/dbImportProfile.sh b/src/_binaries/DbImport/dbImportProfile.sh index bc4a7c53..559d1ee9 100755 --- a/src/_binaries/DbImport/dbImportProfile.sh +++ b/src/_binaries/DbImport/dbImportProfile.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImportProfile +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImportProfile # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/DbImport/dbImportStream.sh b/src/_binaries/DbImport/dbImportStream.sh index 5e8af990..e716c4eb 100755 --- a/src/_binaries/DbImport/dbImportStream.sh +++ b/src/_binaries/DbImport/dbImportStream.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbImportStream +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbImportStream # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/DbImport/testsData/auto_default.local_fromDb_20.sh b/src/_binaries/DbImport/testsData/auto_default.local_fromDb_20.sh index 33f38ff5..a332f1c1 100755 --- a/src/_binaries/DbImport/testsData/auto_default.local_fromDb_20.sh +++ b/src/_binaries/DbImport/testsData/auto_default.local_fromDb_20.sh @@ -4,5 +4,5 @@ cat | grep -v '^table1$' | # table size 29MB grep -v '^table2$' | # table size 10MB -# grep -v '^table3$' | # table size 4MB -cat + # grep -v '^table3$' | # table size 4MB + cat diff --git a/src/_binaries/DbImport/testsData/auto_default.local_fromDb_70.sh b/src/_binaries/DbImport/testsData/auto_default.local_fromDb_70.sh index 8dcd2195..a6919c07 100755 --- a/src/_binaries/DbImport/testsData/auto_default.local_fromDb_70.sh +++ b/src/_binaries/DbImport/testsData/auto_default.local_fromDb_70.sh @@ -3,6 +3,6 @@ # cat represents the whole list of tables cat | grep -v '^table1$' | # table size 29MB -# grep -v '^table2$' | # table size 10MB -# grep -v '^table3$' | # table size 4MB -cat + # grep -v '^table2$' | # table size 10MB + # grep -v '^table3$' | # table size 4MB + cat diff --git a/src/_binaries/DbImport/testsData/dbImport.help.txt b/src/_binaries/DbImport/testsData/dbImport.help.txt index d0d27835..1ec2c9cb 100644 --- a/src/_binaries/DbImport/testsData/dbImport.help.txt +++ b/src/_binaries/DbImport/testsData/dbImport.help.txt @@ -1,13 +1,15 @@ -DESCRIPTION: Import source db into target db using eventual table filter +DESCRIPTION: +Import source db into target db using eventual table filter USAGE: dbImport [OPTIONS] [ARGUMENTS] -USAGE: dbImport [--profile|-p ] [--tables ] - [--from-dsn|-f ] [--skip-schema|-s] [--from-aws|-a ] - [--target-dsn|-t ] [--character-set|-c ] - [--collation-name|-o ] [--bash-framework-config ] [--config] - [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] - [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] - [--log-file ] [--display-level ] +USAGE: dbImport [--profile|-p ] + [--tables ] [--from-dsn|-f ] + [--skip-schema|-s] [--from-aws|-a ] [--target-dsn|-t ] + [--character-set|-c ] [--collation-name|-o ] + [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] + [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] + [--quiet|-q] [--log-level ] [--log-file ] + [--display-level ] ARGUMENTS: fromDbName {single} (mandatory) @@ -18,22 +20,23 @@ PROFILE OPTIONS: --profile, -p  {single} - the name of the profile to use in order to include or exclude tables (if not - specified in default.sh from 'User profiles directory' if exists or 'De - fault profiles directory') + the name of the profile to use in order to include or exclude tables + (if not specified in default.sh from 'User profiles directory' if exists + or 'Default profiles directory') --tables  {single} import only table specified in the list. If aws mode, ignore profile option FROM OPTIONS: --from-dsn, -f  {single} dsn to use for source database - this option is incompatible with -a|--from-aws option + this option is incompatible with -a|--from-aws option --skip-schema, -s {single} avoid to import the schema --from-aws, -a  {single} db dump will be downloaded from s3 instead of using remote db. The value is the name of the file without s3 location (Only .gz or tar.gz f - ile are supported). This option is incompatible with -f|--from-dsn option + ile are supported). This option is incompatible with -f|--from-dsn optio + n TARGET OPTIONS: --target-dsn, -t  {single} @@ -61,8 +64,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -70,11 +75,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE Default profiles directory: /bash/conf/dbImportProfiles diff --git a/src/_binaries/DbImport/testsData/dbImportProfile.help.txt b/src/_binaries/DbImport/testsData/dbImportProfile.help.txt index eed43c23..543e137e 100644 --- a/src/_binaries/DbImport/testsData/dbImportProfile.help.txt +++ b/src/_binaries/DbImport/testsData/dbImportProfile.help.txt @@ -1,11 +1,12 @@ DESCRIPTION: generate optimized profiles to be used by dbImport USAGE: dbImportProfile [OPTIONS] [ARGUMENTS] -USAGE: dbImportProfile [--profile|-p ] [--from-dsn|-f ] - [--ratio|-r ] [--bash-framework-config ] [--config] - [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] - [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] - [--log-file ] [--display-level ] +USAGE: dbImportProfile [--profile|-p ] + [--from-dsn|-f ] [--ratio|-r ] + [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] + [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] + [--quiet|-q] [--log-level ] [--log-file ] + [--display-level ] ARGUMENTS: fromDbName {single} (mandatory) @@ -21,8 +22,8 @@ --ratio, -r  {single} define the ratio to use (0 to 100% - default 70). 0 means profile will filt er out all the tables. 100 means profile will keep all the tables. Eg: - 70 means that tables with size(table+index) that are greater that 70% o - f the max table size will be excluded. + 70 means that tables with size(table+index) that are greater that 70% of + the max table size will be excluded. GLOBAL OPTIONS: --bash-framework-config  {single} @@ -41,8 +42,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -50,11 +53,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE Default profiles directory: /bash/conf/dbImportProfiles diff --git a/src/_binaries/DbImport/testsData/dbImportStream.help.txt b/src/_binaries/DbImport/testsData/dbImportStream.help.txt index 14739ed6..6c9ee1cb 100644 --- a/src/_binaries/DbImport/testsData/dbImportStream.help.txt +++ b/src/_binaries/DbImport/testsData/dbImportStream.help.txt @@ -16,9 +16,9 @@ PROFILE OPTIONS: --profile, -p  {single} - the name of the profile to use in order to include or exclude tables (if not - specified in default.sh from 'User profiles directory' if exists or 'De - fault profiles directory') + the name of the profile to use in order to include or exclude tables + (if not specified in default.sh from 'User profiles directory' if exists + or 'Default profiles directory') --tables  {single} import only table specified in the list. If aws mode, ignore profile option @@ -45,8 +45,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -54,11 +56,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE Default profiles directory: /bash/conf/dbImportProfiles diff --git a/src/_binaries/DbQueryAllDatabases/dbQueryAllDatabases.sh b/src/_binaries/DbQueryAllDatabases/dbQueryAllDatabases.sh index 4d54b8cc..1d5af64d 100755 --- a/src/_binaries/DbQueryAllDatabases/dbQueryAllDatabases.sh +++ b/src/_binaries/DbQueryAllDatabases/dbQueryAllDatabases.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbQueryAllDatabases +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbQueryAllDatabases # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # EMBED Db::queryOneDatabase as dbQueryOneDatabase diff --git a/src/_binaries/DbQueryAllDatabases/testsData/dbQueryAllDatabases.help.txt b/src/_binaries/DbQueryAllDatabases/testsData/dbQueryAllDatabases.help.txt index 8c57c76a..975d868c 100644 --- a/src/_binaries/DbQueryAllDatabases/testsData/dbQueryAllDatabases.help.txt +++ b/src/_binaries/DbQueryAllDatabases/testsData/dbQueryAllDatabases.help.txt @@ -1,5 +1,6 @@ -DESCRIPTION: Execute a query on multiple databases in order to generate a report - with tsv format, query can be parallelized on multiple databases +DESCRIPTION: +Execute a query on multiple databases in order to generate a report with tsv for +mat, query can be parallelized on multiple databases USAGE: dbQueryAllDatabases [OPTIONS] [ARGUMENTS] USAGE: dbQueryAllDatabases [--jobs|-j ] [--bar|-b] @@ -11,10 +12,10 @@ ARGUMENTS: argQuery {single} (mandatory) - Query to execute - - , try to execute the mysql query provided by the file - - , search for query file in queries directory (see below) - - else the argument is interpreted as query string + Query to execute - , try to execute the mysql query provided by the f + ile + - , search for query file in queries directory (see below) + - else the argument is interpreted as query string OPTIONS: --jobs, -j  {single} @@ -48,8 +49,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -57,11 +60,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE LIST OF AVAILABLE DSN: - dsn_local diff --git a/src/_binaries/DbScriptAllDatabases/dbScriptAllDatabases.sh b/src/_binaries/DbScriptAllDatabases/dbScriptAllDatabases.sh index 2f718e88..eb9f508d 100755 --- a/src/_binaries/DbScriptAllDatabases/dbScriptAllDatabases.sh +++ b/src/_binaries/DbScriptAllDatabases/dbScriptAllDatabases.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/dbScriptAllDatabases +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/dbScriptAllDatabases # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # EMBED Db::queryOneDatabase as dbQueryOneDatabase diff --git a/src/_binaries/DbScriptAllDatabases/extractData.sh b/src/_binaries/DbScriptAllDatabases/extractData.sh index 76609436..a32f2b7a 100755 --- a/src/_binaries/DbScriptAllDatabases/extractData.sh +++ b/src/_binaries/DbScriptAllDatabases/extractData.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/conf/dbScripts/extractData +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/conf/dbScripts/extractData # FACADE # shellcheck disable=SC2154 diff --git a/src/_binaries/DbScriptAllDatabases/testsData/dbScriptAllDatabases.help.txt b/src/_binaries/DbScriptAllDatabases/testsData/dbScriptAllDatabases.help.txt index 4e8b2e67..ce5564d4 100644 --- a/src/_binaries/DbScriptAllDatabases/testsData/dbScriptAllDatabases.help.txt +++ b/src/_binaries/DbScriptAllDatabases/testsData/dbScriptAllDatabases.help.txt @@ -34,6 +34,7 @@ Allows to execute a script on each database of specified mysql server --log-format, -l  {single} if output dir provided, will log each db result to log file Default value: none + Possible values: none|log GLOBAL OPTIONS: --bash-framework-config  {single} @@ -52,8 +53,10 @@ Allows to execute a script on each database of specified mysql server --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -61,11 +64,13 @@ Allows to execute a script on each database of specified mysql server --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE NOTE: the use of output, log-format, verbose options highly depends on the script used diff --git a/src/_binaries/Docker/cli.options.tpl b/src/_binaries/Docker/cli.options.tpl index 81277e2e..1898170b 100644 --- a/src/_binaries/Docker/cli.options.tpl +++ b/src/_binaries/Docker/cli.options.tpl @@ -70,36 +70,36 @@ Options::generateCommand "${options[@]}" containerArgHelpCallback() { Conf::load "cliProfiles" "default" echo "container should be the name of a profile from profile list," - echo "check containers list below." $'\n' - echo "If not provided, it will load the container specified in default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' + echo "check containers list below." + echo "If not provided, it will load the container specified in default configuration." + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" echo "Default container: ${__HELP_OPTION_COLOR}${finalContainerArg}${__HELP_NORMAL}" } userArgHelpCallback() { Conf::load "cliProfiles" "default" echo "user to connect on this container" $'\n' - echo "Default user: ${__HELP_OPTION_COLOR}${finalUserArg}${__HELP_NORMAL}" $'\n' - echo " loaded from profile selected as first arg" $'\n' + echo "Default user: ${__HELP_OPTION_COLOR}${finalUserArg}${__HELP_NORMAL}" + echo " loaded from profile selected as first arg" echo " or deduced from default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' echo "if first arg is not a profile" } commandArgHelpCallback() { Conf::load "cliProfiles" "default" echo "The command to execute" $'\n' - echo "Default command: ${__HELP_OPTION_COLOR}${finalCommandArg[*]}${__HELP_NORMAL}" $'\n' - echo " loaded from profile selected as first arg" $'\n' - echo " or deduced from default configuration." $'\n' - echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" + echo "Default command: ${__HELP_OPTION_COLOR}${finalCommandArg[*]}${__HELP_NORMAL}" + echo " loaded from profile selected as first arg" + echo " or deduced from default configuration." + echo "Default configuration: ${__HELP_OPTION_COLOR}${containerArg}${__HELP_NORMAL}" $'\n' echo "if first arg is not a profile" } optionHelpCallback() { local containers # shellcheck disable=SC2046 - containers="$(Array::wrap ", " 80 0 $(docker ps --format '{{.Names}}'))" + containers="$(Array::wrap2 ", " 80 0 $(docker ps --format '{{.Names}}'))" local profilesList="" Conf::load "cliProfiles" "default" diff --git a/src/_binaries/Docker/cli.sh b/src/_binaries/Docker/cli.sh index 6e0f8aaf..376c78e2 100755 --- a/src/_binaries/Docker/cli.sh +++ b/src/_binaries/Docker/cli.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/cli +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/cli # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Docker/testsData/cli.help.txt b/src/_binaries/Docker/testsData/cli.help.txt index 8fd39f97..db28871b 100644 --- a/src/_binaries/Docker/testsData/cli.help.txt +++ b/src/_binaries/Docker/testsData/cli.help.txt @@ -1,31 +1,37 @@ DESCRIPTION: easy connection to docker container USAGE: cli [OPTIONS] [ARGUMENTS] -USAGE: cli [--bash-framework-config ] [--config] [--verbose|-v] [-vv] - [-vvv] [--env-file ] [--no-color] [--theme ] [--help|-h] - [--version] [--quiet|-q] [--log-level ] [--log-file ] - [--display-level ] +USAGE: cli [--bash-framework-config ] [--config] + [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] + [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] + [--log-file ] [--display-level ] ARGUMENTS: [container {single}] container should be the name of a profile from profile list, check containers list below. If not provided, it will load the container specified in default configu - ration. - Default configuration: default + ration. Default configuration: default Default container: project-apache2 [user {single}] user to connect on this container + Default user: www-data loaded from profile selected as first arg or deduced from default configuration. - Default configuration: default if first arg is not a profile + + Default configuration: default + + if first arg is not a profile [commandArg {single}] The command to execute + Default command: /bin/bash loaded from profile selected as first arg or deduced from default configuration. - Default configuration: default if first arg is not a profile + Default configuration: default + + if first arg is not a profile GLOBAL OPTIONS: --bash-framework-config  {single} @@ -44,8 +50,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -53,11 +61,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE AVAILABLE PROFILES (from /bash/conf/cliProfiles) This list can be overridden in home/.bash-tools/cliProfiles diff --git a/src/_binaries/Git/gitIsAncestorOf.sh b/src/_binaries/Git/gitIsAncestorOf.sh index 66173eb0..86020d89 100755 --- a/src/_binaries/Git/gitIsAncestorOf.sh +++ b/src/_binaries/Git/gitIsAncestorOf.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitIsAncestorOf +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitIsAncestorOf # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Git/gitIsBranch.sh b/src/_binaries/Git/gitIsBranch.sh index bb8af53e..dd10bb03 100755 --- a/src/_binaries/Git/gitIsBranch.sh +++ b/src/_binaries/Git/gitIsBranch.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitIsBranch +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitIsBranch # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Git/gitRenameBranch.options.tpl b/src/_binaries/Git/gitRenameBranch.options.tpl index 4c720a28..bbabcfcf 100644 --- a/src/_binaries/Git/gitRenameBranch.options.tpl +++ b/src/_binaries/Git/gitRenameBranch.options.tpl @@ -74,8 +74,8 @@ Options::generateCommand "${options[@]}" % assumeYesHelpCallback() { - echo "do not ask for confirmation (use with caution)" $'\n' - echo ' Automatic yes to prompts; assume "y" as answer to all prompts' $'\n' + echo "do not ask for confirmation (use with caution)" + echo ' Automatic yes to prompts; assume "y" as answer to all prompts' echo ' and run non-interactively.' } diff --git a/src/_binaries/Git/gitRenameBranch.sh b/src/_binaries/Git/gitRenameBranch.sh index a356254f..17487f57 100755 --- a/src/_binaries/Git/gitRenameBranch.sh +++ b/src/_binaries/Git/gitRenameBranch.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/gitRenameBranch +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/gitRenameBranch # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Git/testsData/gitIsAncestorOf.help.txt b/src/_binaries/Git/testsData/gitIsAncestorOf.help.txt index 65fbf775..903ccfd7 100644 --- a/src/_binaries/Git/testsData/gitIsAncestorOf.help.txt +++ b/src/_binaries/Git/testsData/gitIsAncestorOf.help.txt @@ -1,8 +1,8 @@ DESCRIPTION: check if commit is inside a given branch USAGE: gitIsAncestorOf [OPTIONS] [ARGUMENTS] -USAGE: gitIsAncestorOf [--bash-framework-config ] [--config] - [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] +USAGE: gitIsAncestorOf [--bash-framework-config ] + [--config] [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] [--log-file ] [--display-level ] @@ -29,8 +29,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -38,11 +40,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE EXIT CODES: 1: if commit does not exists diff --git a/src/_binaries/Git/testsData/gitRenameBranch.help.txt b/src/_binaries/Git/testsData/gitRenameBranch.help.txt index 88207f95..eeb66481 100644 --- a/src/_binaries/Git/testsData/gitRenameBranch.help.txt +++ b/src/_binaries/Git/testsData/gitRenameBranch.help.txt @@ -1,10 +1,11 @@ -DESCRIPTION: rename git local branch, push new branch and delete old branch +DESCRIPTION: +rename git local branch, push new branch and delete old branch USAGE: gitRenameBranch [OPTIONS] [ARGUMENTS] -USAGE: gitRenameBranch [--assume-yes|--yes|-y] [--push|-p] [--delete|-d] - [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] - [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] - [--quiet|-q] [--log-level ] [--log-file ] +USAGE: gitRenameBranch [--assume-yes|--yes|-y] [--push|-p] + [--delete|-d] [--bash-framework-config ] [--config] [--verbose|-v] + [-vv] [-vvv] [--env-file ] [--no-color] [--theme ] [--help|-h] + [--version] [--quiet|-q] [--log-level ] [--log-file ] [--display-level ] ARGUMENTS: @@ -40,8 +41,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -49,11 +52,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE EXIT CODES: 1 : if current directory is not a git repository diff --git a/src/_binaries/Git/testsData/upgradeGithubRelease.help.txt b/src/_binaries/Git/testsData/upgradeGithubRelease.help.txt index a1c8fbd7..7623eafe 100644 --- a/src/_binaries/Git/testsData/upgradeGithubRelease.help.txt +++ b/src/_binaries/Git/testsData/upgradeGithubRelease.help.txt @@ -1,4 +1,5 @@ -DESCRIPTION: retrieve latest binary release from github and install it +DESCRIPTION: +retrieve latest binary release from github and install it USAGE: upgradeGithubRelease [OPTIONS] [ARGUMENTS] USAGE: upgradeGithubRelease [--version-arg ] @@ -10,32 +11,34 @@ ARGUMENTS: targetFile {single} (mandatory) - the binary downloaded will e written to this file path. Ensure the path is w - ritable. + the binary downloaded will be written to this file path. + Ensure the path is writable. githubUrlPattern {single} (mandatory) the url pattern to use to download the binary, see examples below. - @version@ is template variable that will be replaced by the latest - version tag found on github. + @version@ is template variable that will be replaced by the latest + version tag found on github. VERSION MANAGEMENT: --version-arg  {single} The argument that will be provided to the currently installed binary - to check the version of the software. See options constraints below. + to check the version of the software. + See options constraints below. Default value: --version --current-version, -c  {single} Sometimes the command to retrieve the version is complicated. - Some command needs you to parse json or other commands provides - multiple sub command versions. In this case you can provide the - version you currently have, see examples below. + Some command needs you to parse json or other commands provides multiple + sub command versions. In this case you can provide the version you + currently have, see examples below. See options constraints below. --exact-version, -e  {single} if provided and currently installed binary is not this exactVersion, - This exact version of the binary will be installed. - See options constraints below. + This exact version of the binary will be installed. + See options constraints below. --minimal-version, -m  {single} - if provided and currently installed binary is below this minimalVersion, - a new version of the binary will be installed. If this argument is not provi - ded, the latest binary is unconditionally downloaded from github. + if provided and currently installed binary is below this + minimalVersion, a new version of the binary will be installed. + If this argument is not provided, the latest binary is unconditionally + downloaded from github. See options constraints below. GLOBAL OPTIONS: @@ -55,8 +58,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -64,11 +69,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE OPTIONS CONSTRAINTS: diff --git a/src/_binaries/Git/upgradeGithubRelease.options.tpl b/src/_binaries/Git/upgradeGithubRelease.options.tpl index 11e79954..d2178adb 100644 --- a/src/_binaries/Git/upgradeGithubRelease.options.tpl +++ b/src/_binaries/Git/upgradeGithubRelease.options.tpl @@ -21,7 +21,8 @@ declare defaultVersionArg="--version" source <( targetFileArgCallback() { :; } Options::generateArg \ - --help "the binary downloaded will e written to this file path. Ensure the path is writable." \ + --help $'the binary downloaded will be written to this file path.\n + Ensure the path is writable.' \ --min 1 \ --max 1 \ --name "targetFile" \ @@ -32,11 +33,9 @@ source <( githubUrlPatternArgCallback() { :; } # shellcheck disable=SC2116 Options::generateArg \ - --help "$(echo \ - "the url pattern to use to download the binary, see examples below." $'\n' \ - "@version@ is template variable that will be replaced by the latest" $'\n' \ - "version tag found on github." \ - )" \ + --help $'the url pattern to use to download the binary, see examples below.\n + @version@ is template variable that will be replaced by the latest \n + version tag found on github.' \ --min 1 \ --max 1 \ --name "githubUrlPattern" \ @@ -52,8 +51,9 @@ source <( Options::generateOption \ --help-value-name "versionArg" \ --default-value "${defaultVersionArg}" \ - --help "The argument that will be provided to the currently installed binary -to check the version of the software. See options constraints below." \ + --help $'The argument that will be provided to the currently installed binary\n + to check the version of the software. \n + See options constraints below.' \ --group groupVersionManagementFunction \ --alt "--version-arg" \ --variable-type "String" \ @@ -63,11 +63,11 @@ to check the version of the software. See options constraints below." \ # shellcheck disable=SC2116 Options::generateOption \ --help-value-name "currentVersion" \ - --help "Sometimes the command to retrieve the version is complicated. -Some command needs you to parse json or other commands provides -multiple sub command versions. In this case you can provide the -version you currently have, see examples below. -See options constraints below." \ + --help $'Sometimes the command to retrieve the version is complicated. \n + Some command needs you to parse json or other commands provides multiple \n + sub command versions. In this case you can provide the version you \n + currently have, see examples below. \n + See options constraints below.' \ --group groupVersionManagementFunction \ --alt "--current-version" \ --alt "-c" \ @@ -78,10 +78,11 @@ See options constraints below." \ # shellcheck disable=SC2116,SC2016 Options::generateOption \ --help-value-name "minimalVersion" \ - --help 'if provided and currently installed binary is below this minimalVersion, -a new version of the binary will be installed. -If this argument is not provided, the latest binary is unconditionally downloaded from github. -See options constraints below.' \ + --help $'if provided and currently installed binary is below this \n + minimalVersion, a new version of the binary will be installed. \n + If this argument is not provided, the latest binary is unconditionally \n + downloaded from github. \n + See options constraints below.' \ --group groupVersionManagementFunction \ --alt "--minimal-version" \ --alt "-m" \ @@ -92,8 +93,8 @@ See options constraints below.' \ # shellcheck disable=SC2116,SC2016 Options::generateOption \ --help-value-name "exactVersion" \ - --help 'if provided and currently installed binary is not this exactVersion, - This exact version of the binary will be installed. + --help $'if provided and currently installed binary is not this exactVersion,\n + This exact version of the binary will be installed.\n See options constraints below.' \ --group groupVersionManagementFunction \ --alt "--exact-version" \ diff --git a/src/_binaries/Git/upgradeGithubRelease.sh b/src/_binaries/Git/upgradeGithubRelease.sh index f7d7ec97..cfee457c 100755 --- a/src/_binaries/Git/upgradeGithubRelease.sh +++ b/src/_binaries/Git/upgradeGithubRelease.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/upgradeGithubRelease +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/upgradeGithubRelease # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Utils/testsData/waitForIt.help.txt b/src/_binaries/Utils/testsData/waitForIt.help.txt index 1ef282b6..940050ba 100644 --- a/src/_binaries/Utils/testsData/waitForIt.help.txt +++ b/src/_binaries/Utils/testsData/waitForIt.help.txt @@ -1,8 +1,9 @@ DESCRIPTION: wait for host:port to be available USAGE: waitForIt [OPTIONS] [ARGUMENTS] -USAGE: waitForIt [--timeout|-t ] --host|-i --port|-p - [--algorithm|--algo ] [--exec-command-on-success-only|--strict|-s] +USAGE: waitForIt [--timeout|-t ] --host|-i + --port|-p [--algorithm|--algo ] + [--exec-command-on-success-only|--strict|-s] [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] [--log-file ] @@ -22,8 +23,8 @@ --port, -p  {single} (mandatory) TCP port under test. --algorithm, --algo  {single} - Algorithm to use Check algorithms list below. (default: automatic selectio - n based on commands availability and timeout option value). + Algorithm to use Check algorithms list below. (default: automatic selection + based on commands availability and timeout option value). --exec-command-on-success-only, --strict, -s {single} Only execute sub-command if the test succeeds. @@ -44,8 +45,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -53,11 +56,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE EXIT STATUS CODES: 0: the host/port is available diff --git a/src/_binaries/Utils/testsData/waitForMysql.help.txt b/src/_binaries/Utils/testsData/waitForMysql.help.txt index 373497f3..dd8dce4c 100644 --- a/src/_binaries/Utils/testsData/waitForMysql.help.txt +++ b/src/_binaries/Utils/testsData/waitForMysql.help.txt @@ -1,10 +1,11 @@ DESCRIPTION: wait for mysql to be ready USAGE: waitForMysql [OPTIONS] [ARGUMENTS] -USAGE: waitForMysql [--timeout|-t ] [--bash-framework-config ] - [--config] [--verbose|-v] [-vv] [-vvv] [--env-file ] [--no-color] - [--theme ] [--help|-h] [--version] [--quiet|-q] [--log-level ] - [--log-file ] [--display-level ] +USAGE: waitForMysql [--timeout|-t ] + [--bash-framework-config ] [--config] [--verbose|-v] [-vv] [-vvv] + [--env-file ] [--no-color] [--theme ] [--help|-h] [--version] + [--quiet|-q] [--log-level ] [--log-file ] + [--display-level ] ARGUMENTS: mysqlHost {single} (mandatory) @@ -38,8 +39,10 @@ --no-color {single} Produce monochrome output. alias of --theme noColor. --theme  {single} - choose color theme (default, default-force or noColor) - default-force means - colors will be produced even if command is piped + choose color theme - default-force means colors will be produced even if com + mand is piped + Default value: default + Possible values: default|default-force|noColor --help, -h {single} Display this command help --version {single} @@ -47,11 +50,13 @@ --quiet, -q {single} quiet mode, doesn't display any output --log-level  {single} - Set log level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + Set log level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE --log-file  {single} Set log file --display-level  {single} - set display level (one of OFF, ERROR, WARNING, INFO, DEBUG, TRACE value) + set display level + Possible values: OFF|ERROR|WARNING|INFO|DEBUG|TRACE EXIT STATUS CODES: 0: mysql is available diff --git a/src/_binaries/Utils/waitForIt.sh b/src/_binaries/Utils/waitForIt.sh index d4c0929b..11ff8264 100755 --- a/src/_binaries/Utils/waitForIt.sh +++ b/src/_binaries/Utils/waitForIt.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/waitForIt +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/waitForIt # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/Utils/waitForMysql.sh b/src/_binaries/Utils/waitForMysql.sh index 28bd2255..aad4c9c8 100755 --- a/src/_binaries/Utils/waitForMysql.sh +++ b/src/_binaries/Utils/waitForMysql.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/waitForMysql +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/waitForMysql # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/build/doc.sh b/src/_binaries/build/doc.sh index e2d6a6a4..a01c80c2 100755 --- a/src/_binaries/build/doc.sh +++ b/src/_binaries/build/doc.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/doc +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/doc # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/build/install.sh b/src/_binaries/build/install.sh index 6acc4ba9..dd75f55d 100755 --- a/src/_binaries/build/install.sh +++ b/src/_binaries/build/install.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/install +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/install # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=. # FACADE # shellcheck disable=SC2034 diff --git a/src/_binaries/build/installRequirements.sh b/src/_binaries/build/installRequirements.sh index 6ec19cbf..0337dd96 100755 --- a/src/_binaries/build/installRequirements.sh +++ b/src/_binaries/build/installRequirements.sh @@ -1,5 +1,5 @@ #!/usr/bin/env bash -# BIN_FILE=${FRAMEWORK_ROOT_DIR}/bin/installRequirements +# BIN_FILE=${BASH_TOOLS_ROOT_DIR}/bin/installRequirements # VAR_RELATIVE_FRAMEWORK_DIR_TO_CURRENT_DIR=.. # FACADE # shellcheck disable=SC2034,SC2154