Posso examinar o tipo de uma variável bash?

Posso examinar o tipo de uma variável bash?

Estou escrevendo uma função que gera datas. Gostaria de permitir que o usuário personalize a saída fornecendo argumentos datecom uma variável de ambiente. Para preservar o espaço em branco nas strings de formato, gostaria de aceitar argumentos em uma matriz como esta:

function argdates {
    while [ $# -gt 0 ] && date "${DATE_ARGS[@]}" -d @$1
    do shift
    done
}

O usuário pode querer usar um array se tiver espaços em uma string de formato de data:

DATE_ARGS=( -u "+%y/%U, %I%p %a" )
argdates 1476395008 1493172224

# output:
# 16/41, 09PM Thu
# 17/17, 02AM Wed

Mas neste caso, um array pode ser um exagero:

DATE_ARGS="-u -Iseconds"
argdates 1476395008 1493172224

# output:
# date: invalid option -- ' '
# Try 'date --help' for more information.

# output should be:
# 2016-10-13T21:43:28+00:00
# 2017-04-26T02:03:44+00:00

Não quero exigir um array para casos simples como este. É possível saber de que tipo é a variável?

Responder1

Parece-me que você pode querer deixar sua função passar qualquer opção de linha de comando diretamente para o GNU dateenquanto trata especialmente a não opção numérica:

argdates () {
    local -a opts
    local timestamp

    while [ "$#" -gt 0 ] && [ "$1" != '--' ] && [[ "$1" != [0-9]* ]]; do
        opts+=( "$1" )
        shift
    done
    [ "$1" = '--' ] && shift

    for timestamp do
        date "${opts[@]}" -d "@$timestamp"
    done

    # or, with a single invocation of "date":
    # printf '@%s\n' "$@" | date "${opts[@]}" -f -
}

Esta bashfunção percorreria seus argumentos de linha de comando e os salvaria no optsarray, até encontrar um argumento que seja --(a forma padrão de sinalizar o fim das opções) ou que comece com um dígito. Cada argumento salvo optsé retirado da lista de argumentos da linha de comando.

Uma vez encontrado um argumento que não é uma opção para date, assumimos que o restante dos argumentos são carimbos de data e hora da época UNIX e fazemos um loop sobre eles, chamando datecom nossas opções salvas para cada carimbo de data e hora. Veja o comentário no código para saber como fazer esse loop com mais eficiência.

Exemplos de invocações:

$ argdates 1476395008 1493172224
Thu Oct 13 23:43:28 CEST 2016
Wed Apr 26 04:03:44 CEST 2017
$ argdates -- 1476395008 1493172224
Thu Oct 13 23:43:28 CEST 2016
Wed Apr 26 04:03:44 CEST 2017
$ argdates +%D 1476395008 1493172224
10/13/16
04/26/17
$ argdates +%F 1476395008 1493172224
2016-10-13
2017-04-26
$ argdates -u 1476395008 1493172224
Thu Oct 13 21:43:28 UTC 2016
Wed Apr 26 02:03:44 UTC 2017
$ argdates -u -Iseconds 1476395008 1493172224
2016-10-13T21:43:28+00:00
2017-04-26T02:03:44+00:00

Responder2

Você poderia fazer isso muito mais facilmente desistindo de usar arrays e deixando o usuário especificar DATE_ARGSsimplesmente como uma string que deve ser inserida na linha de comando de date:

$ argdates(){
    for d; do printf %s "$DATE_ARGS" | xargs date -d "@$d"; done
}
$ DATE_ARGS='-u "+%y/%U, %I%p %a"' argdates 1476395008 1493172224
16/41, 09PM Thu
17/17, 02AM Wed
$ DATE_ARGS='-u -Iseconds' argdates 1476395008 1493172224
2016-10-13T21:43:28+00:00
2017-04-26T02:03:44+00:00

informação relacionada