Caminho de arquivo inválido usando getopt_long em C
-
26-12-2019 - |
Pergunta
Estou me perguntando por que optarg
retorna um caminho inválido no seguinte caso: --foo=~/.bashrc
mas não se eu deixar um espaço no meio --foo ~/.bashrc
.
E qual seria a solução alternativa para que funcione em ambos os casos.
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
int main(int argc, char *argv[]) {
int opt = 0;
int long_index = 0;
char *f;
static struct option longopt[] = {
{"foo", required_argument, 0, 'd' },
{0,0,0,0}
};
while ((opt = getopt_long(argc, argv,"d:", longopt, &long_index )) != -1) {
switch (opt) {
case 'd' :
printf("\n%s\n", optarg);
f = realpath (optarg, NULL);
if (f) printf("%s\n", f);
break;
default:
exit(1);
}
}
return 0;
}
Saída:
$ ./a.out --foo=~/.bashrc
~/.bashrc
$ ./a.out --foo ~/.bashrc
/home/user/.bashrc
Solução
Isso acontece porque a "expansão do til" é realizada pelo shell:não é um caminho válido por si só.O til ~ é expandido como o diretório inicial apenas no caso de estar no início de um argumento de string, que se parece com um caminho.Por exemplo:
$ echo ~
/home/sigi
$ echo ~/a
/home/sigi/a
$ echo ~root/a
/root/a
$ echo ~a
~a
$ echo a/~
a/~
Se você quiser oferecer esta funcionalidade também no primeiro caso, onde o shell não pode ajudá-lo, ou mais geralmente nas expansões de palavras usadas pelo shell, você pode encontrar todas as informações necessárias para fazer isso sozinho em esta referência.