1

我在获取一个 variable 时遇到了问题"${Error[*]}",它是一个常规索引数组,从它被声明到被检查时一直保持设置。在我看来,必须启动一个子外壳,这样声明才不会生效。我认为使用大括号时没有打开子外壳{ stuff...; }。我想知道如何获取我的变量,Error以坚持我试图写的情况。这是我的脚本示例:

TestFunction () {
    unset Error
    local archive="$1" extlist="$2" && local ext="${archive##*.}"
    shopt -s nocasematch
    local -i run=0
    while [[ "$run" == 0 || -n "${Error[run]}" ]]; do
        (( run++ ))
        local IFS=$'\n\r\t '
        if [[ ! "${Error[*]}" =~ 'cpio' && "$ext" =~ ^(pax|cpio|cpgz|igz|ipa|cab)$ && -n "$(which 'cpio')" ]]; then

    ## Try to cpio the archive. Since cpio cannot handle '.cab' archive, I want to declare an Error ##
            { cpio -ti --quiet <"$archive" 2>'/dev/null' || local -a Error[run]='cpio'; } | grep -Ei '$extlist'
        elif [[ ! "${Error[*]}" =~ 'zipinfo' && "$ext" =~ ^(zip|[jw]ar|ipa|cab)$ && -n "$(which 'unzip')" ]]; then

    ## If cpio fails, then try zipinfo, or unzip on the next run through the loop... ##
            if which 'zipinfo' &>'/dev/null'; then
                { zipinfo -1 "$archive" 2>'/dev/null' || local -a Error[run]='zipinfo'; } | grep -Ei "$scanlist"
            elif which 'unzip' &>'/dev/null'; then
                { unzip -lqq "$archive" 2>'/dev/null' || local -a Error[run]='unzip'; } | gsed -re '/^ +[0-9]+/!d;s|^ +[0-9]+ +[0-9-]+ [0-9:]+ +||' | grep -Ei "$exlist"
            fi
    ## many more elifs... ##
        fi
    done
    shopt -u nocasematch
    return 0
}

Archives='\.(gnutar|7-zip|lharc|toast|7zip|boz|bzi?p2?|cpgz|cpio|gtar|g?z(ip)?|lzma(86)?|t[bg]z2?|ar[cgjk]|bz[2a]?|cb[7rz]|cdr|deb|[dt]lz|dmg|exe|fbz|fgz|gz[aip]|igz|img|iso|lh[az]|lz[hmswx]?|mgz|mpv|mpz|pax|piz|pka|[jrtwx]ar|rpm|s?7-?z|sitx?|m?pkg|sfx|nz|xz)$'
IFS=$'\n'
declare -a List=($(TestFunction '/Users/aesthir/Programming/│My Projects│/Swipe Master/Test Folder/SDKSetup.cab' "$Archives"))
IFS=$' \t\n'


xtrace 输出:

  〔xtrace〕 unset Error
  〔xtrace〕 local 'archive=/Users/aesthir/Programming/│My Projects│/Swipe Master/Test Folder/SDKSetup.cab' 'extlist=\.(gnutar|7-zip|lharc|toast|7zip|boz|bzi?p2?|cpgz|cpio|gtar|g?z(ip)?|lzma(86)?|t[bg]z2?|ar[cgjk]|bz[2a]?|cb[7rz]|cdr|deb|[dt]lz|dmg|exe|fbz|fgz|gz[aip]|igz|img|iso|lh[az]|lz[hmswx]?|mgz|mpv|mpz|pax|piz|pka|[jrtwx]ar|rpm|s?7-?z|sitx?|m?pkg|sfx|nz|xz)$'
  〔xtrace〕 local ext=cab
  〔xtrace〕 shopt -s nocasematch
  〔xtrace〕 local -i run=0
  〔xtrace〕 [[ 0 == 0 ]]
  〔xtrace〕 ((  run++  ))
  〔xtrace〕 local 'IFS=
'
  〔xtrace〕 [[ ! '' =~ cpio ]]
  〔xtrace〕 [[ cab =~ ^(pax|cpio|cpgz|igz|ipa|cab)$ ]]
  〔xtrace〕 which cpio
  〔xtrace〕 [[ -n /usr/bin/cpio ]]
  〔xtrace〕 grep -Ei '$extlist'
  〔xtrace〕 cpio -ti --quiet
  〔xtrace〕 local -a 'Error[run]=cpio'
  〔xtrace〕 [[ 1 == 0 ]]
  〔xtrace〕 [[ -n '' ]]          ## <—— Problem is here... when checking "${Error[run]}", it's unset ##
  〔xtrace〕 shopt -u nocasematch
  〔xtrace〕 return 0


现在显然我知道cpio, zipinfo, 并且unzip无法处理 cab 文件...我故意将 'cab' 放在扩展名列表中以导致错误。

我想留在TestFunction并继续与不同的归档程序循环,直到成功(文件列表被转储,cabextract在这种情况下很乐意这样做)而不重复已经失败的归档程序。

最后,因为这工作正常......

TestFunction () {
    unset Error
    local archive="$1" extlist="$2" && local ext="${archive##*.}"
    local -i run=0
    while [[ "$run" == 0 || -n "${Error[run]}" ]]; do
        (( run++ ))
        local IFS=$'\n\r\t '
        if [[ ! "${Error[*]}" =~ 'cpio' && "$ext" =~ ^(pax|cpio|cpgz|igz|ipa|cab)$ && -n "$(which 'cpio')" ]]; then
            cpio -ti --quiet <"$archive" 2>'/dev/null' || local -a Error[run]='cpio'
        fi
    done
    shopt -u nocasematch
    return 0
}

...我不得不假设问题出在大括号上,因为我想要grep马上得到结果。但是,我需要那些大括号,因为如果没有结果,我不想Error[run]被设置,只有在失败时。由于其他原因,我不想在外面grep (我必须完全重写)。grepcpioTestFunction

没有大量重写的任何快速解决方案?也许echo 'cpio'对一些 fd 和read -u6ing 它以某种方式?

我更希望不必为文件列表设置一个数组,然后for loop | grep遍历每个文件,因为它真的会减慢速度。

4

1 回答 1

3

问题不是大括号,而是管道。因为您使用的是管道,所以对 Error[run] 的分配发生在子shell中,因此当子shell退出时,分配就会消失。

改变:

{ cpio -ti --quiet <"$archive" 2>'/dev/null' || local -a Error[run]='cpio'; } | grep -Ei '$extlist'

到:

cpio -ti --quiet <"$archive" 2>'/dev/null' | grep -Ei "$extlist"
[[ ${PIPESTATUS[0]} -ne 0 ]] && Error[run]='cpio'

(顺便说一句,在 grep 部分需要双引号)

于 2011-08-09T17:31:14.940 回答