Como comparar a versão de um programa em um shell script?

14

Suponha que eu queira comparar a gccversão para ver se o sistema possui a versão mínima instalada ou não.

Para verificar a gccversão, executei o seguinte

gcc --version | head -n1 | cut -d" " -f4

A saída foi

4.8.5

Então, escrevi uma ifdeclaração simples para comparar esta versão com outro valor

if [ "$(gcc --version | head -n1 | cut -d" " -f4)" -lt 5.0.0 ]; then
    echo "Less than 5.0.0"
else
    echo "Greater than 5.0.0"
fi

Mas gera um erro:

[: integer expression expected: 4.8.5

Eu entendi meu erro que eu estava usando seqüências de caracteres para comparar e -ltrequer número inteiro. Então, existe alguma outra maneira de comparar as versões?

Abhimanyu Saharan
fonte
@ 123 Nada acontece
Abhimanyu Saharan
1
Há também uma pergunta sobre estouro de pilha com várias sugestões diferentes para comparar seqüências de caracteres de versão.
n.st
1
Muito mais simples do que usar tubos:gcc -dumpversion
Victor Lamoine

Respostas:

21

Não sei se é bonito, mas está funcionando para todos os formatos de versão que conheço.

#!/bin/bash
currentver="$(gcc -dumpversion)"
requiredver="5.0.0"
 if [ "$(printf '%s\n' "$requiredver" "$currentver" | sort -V | head -n1)" = "$requiredver" ]; then 
        echo "Greater than or equal to 5.0.0"
 else
        echo "Less than 5.0.0"
 fi

( Nota: versão melhorada pelo usuário 'curinga': https://unix.stackexchange.com/users/135943/wildcard , condição adicional removida)

Luciano Andress Martini
fonte
3
No começo, pensei que isso era horrível e depois percebi que a beleza do script de shell está precisamente em abusar de ferramentas como essa. +1
Boicote SE para Monica Cellio
2
Isso é interrompido se houver sinais '%' na declaração de impressão. Melhor substituir printf "$requiredver\n$currentver"por printf '%s\n' "$requiredver" "$currentver".
PHK
1
-Vé uma extensão GNU, sort(1)portanto, esta solução não é portátil.
stefanct
1
sort -nfunciona da mesma maneira no caso de versões numéricas.
Rockallite #
1
@LucianoAndressMartini, veja o que você acha da minha edição.
Curinga
2

Aqui, dou uma solução para comparar as versões do Unix Kernel. E deve funcionar para outros, como o gcc. Eu só ligo para o número da primeira versão 2, mas você pode adicionar outra camada de lógica. É um liner e eu escrevi em várias linhas para entender.

check_linux_version() {
    version_good=$(uname -r | awk 'BEGIN{ FS="."}; 
    { if ($1 < 4) { print "N"; } 
      else if ($1 == 4) { 
          if ($2 < 4) { print "N"; } 
          else { print "Y"; } 
      } 
      else { print "Y"; }
    }')

    #if [ "$current" \< "$expected" ]; then
    if [ "$version_good" = "N" ]; then
        current=$(uname -r)
        echo current linux version too low
        echo current Linux: $current
        echo required 4.4 minimum
        return 1
    fi
}

Você pode modificar isso e usá-lo para verificação da versão do gcc.

Kemin Zhou
fonte
+1 é compatível com outro Unix-like? (Minha solução não é)
Luciano Andress Martini
1

Costumávamos fazer muita verificação de versão em um makefile GNU. Nós percorremos as instalações de makefile. Tivemos que detectar Binutils antigos e compiladores de bugs e contorná-los em tempo real.

O padrão que usamos foi:

#!/usr/bin/env bash

CC=$(command -v gcc)
GREP=$(command -v grep)

# Fixup CC and GREP as needed. It may be needed on AIX, BSDs, and Solaris
if [[ -f "/usr/gnu/bin/grep" ]]; then
    GREP="/usr/gnu/bin/grep"
elif [[ -f "/usr/linux/bin/grep" ]]; then
    GREP="/usr/linux/bin/grep"
elif [[ -f "/usr/xpg4/bin/grep" ]]; then
    GREP="/usr/xpg4/bin/grep"
fi

# Check compiler for GCC 4.8 or later
GCC48_OR_LATER=$("$CXX" -v 2>&1 | "$GREP" -i -c -E "gcc version (4\.[8-9]|[5-9]\.)")
if [[ "$GCC48_OR_LATER" -ne 0 ]];
then
   ...
fi

# Check assembler for GAS 2.19 or later
GAS219_OR_LATER=$("$CXX" -xc -c /dev/null -Wa,-v -o/dev/null 2>&1 | "$GREP" -c -E "GNU assembler version (2\.19|2\.[2-9]|[3-9])")
if [[ "$GAS219_OR_LATER" -ne 0 ]];
then
   ...
fi

fonte
Agradável! Isso funciona e é super fácil de estender e muito portátil!
Brad Parks
1

Versão mais curta:

version_greater_equal()
{
    printf '%s\n%s\n' "$2" "$1" | sort -V -C
}

version_greater_equal "${gcc_version}" 8.2 || die "need 8.2 or above"
Marcha
fonte
(1) Essa é uma variação menor das respostas já dadas. Você pode adicionar valor adicionando uma explicação que ainda não foi lançada. (2)  printf '%s\n'é bom o suficiente; printfrepetirá a string de formato conforme necessário.
G-Man diz 'Reinstate Monica'
Normalmente, prefiro editar as respostas existentes, mas excluir metade delas é complicado: outras pessoas podem ver valor onde eu não. O mesmo para explicações detalhadas. Menos é mais.
MarcH
Eu sei que printfrepete a string de formato, mas a sintaxe (falta de!) Para isso é IMHO obscura; então eu uso isso somente quando necessário = quando o número de argumentos é grande ou variável.
MarcH
1
function version_compare () {
  function sub_ver () {
    local len=${#1}
    temp=${1%%"."*} && indexOf=`echo ${1%%"."*} | echo ${#temp}`
    echo -e "${1:0:indexOf}"
  }
  function cut_dot () {
    local offset=${#1}
    local length=${#2}
    echo -e "${2:((++offset)):length}"
  }
  if [ -z "$1" ] || [ -z "$2" ]; then
    echo "=" && exit 0
  fi
  local v1=`echo -e "${1}" | tr -d '[[:space:]]'`
  local v2=`echo -e "${2}" | tr -d '[[:space:]]'`
  local v1_sub=`sub_ver $v1`
  local v2_sub=`sub_ver $v2`
  if (( v1_sub > v2_sub )); then
    echo ">"
  elif (( v1_sub < v2_sub )); then
    echo "<"
  else
    version_compare `cut_dot $v1_sub $v1` `cut_dot $v2_sub $v2`
  fi
}

### Usage:

version_compare "1.2.3" "1.2.4"
# Output: <

O crédito vai para @Shellman

Xaqron
fonte