Пример функции realpath в C

Я ищу пример использования функции realpath в программе на языке C. Я не могу найти его в Интернете или в любой из моих книг по программированию на С.

Ответы

Ответ 1

Функция realpath() не описана в стандарте C. Это, однако, описано в POSIX 1997 и POSIX 2008. Если это то, что вы имеете в виду, вот пример:

#include <limits.h> /* PATH_MAX */
#include <stdio.h>
#include <stdlib.h>
int main(void) {
    char buf[PATH_MAX]; /* PATH_MAX incudes the \0 so +1 is not required */
    char *res = realpath("this_source.c", buf);
    if (res) {
        printf("This source is at %s.\n", buf);
    } else {
        perror("realpath");
        exit(EXIT_FAILURE);
    }
    return 0;
}

PATH_MAX определен в <limit.h> (<limit.h> из POSIX 1997)

Ответ 2

Командная строка с одной строкой

Минималистский, но он выполняет эту работу!

Строить

gcc -o realpath -x c - <<< $'#include<stdlib.h>\n#include<stdio.h>\nint main(int c,char**v){puts(realpath(v[1],0));}'

Test

$> ./realpath  ~/../../../usr/./bin/./awk
/bin/gawk 

$> readlink -f ~/../../../usr/./bin/./awk
/bin/gawk

Требования

  • для компиляции и ссылки
  • для <<< и $' ... \n ... '

Краш

Моя минималистическая однострочная командная строка создает исполняемый файл realpath, который создает Segmentation fault, когда путь не существует. Вместо написания блоков if/else для решения этой проблемы в моем ответе я добавил ниже некоторые ссылки, чтобы вы могли взглянуть на реализацию Busybox realpath и readlink.


Реализация Busybox

Для более полного исходного кода ознакомьтесь с этой простой реализацией.

Официальный репозиторий Git

Зеркальное хранилище GitHub

Ответ 3

Что делает функция realpath(), это указать путь к файлу, когда все символические ссылки были разрешены. Это необязательно абсолютный путь, если значение, которое вы поставляете, является относительным именем, но отчасти зависит от того, пересекаете ли вы какие-либо символические ссылки с абсолютными именами для значения ссылки - если вы это сделаете, тогда результат будет абсолютным именем, Кроме того, если относительное имя пересекается с корневым каталогом (или "за пределами", как в "../../../../../..", когда только три уровня находятся в глубине иерархии каталогов).

У вас может быть программа "realpath" уже на вашем компьютере. Здесь (нестандартная) версия, которую я написал.

/*
@(#)File:           $RCSfile: realpath.c,v $
@(#)Version:        $Revision: 1.3 $
@(#)Last changed:   $Date: 2007/10/23 20:23:44 $
@(#)Purpose:        Command to evaluate realpath(3) on given arguments.
@(#)Author:         J Leffler
@(#)Copyright:      (C) JLSS 2007
@(#)Product:        :PRODUCT:
*/

/*TABSTOP=4*/

#if __STDC_VERSION__ >= 199901L
#define _XOPEN_SOURCE 600
#else
#define _XOPEN_SOURCE 500
#endif /* __STDC_VERSION__ */

#include <unistd.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include "stderr.h"

static const char optstr[] = "hlsV";
static const char usestr[] = "[-hslV] given-path [...]";
static const char hlpstr[] =
    "  -h   Print this help message\n"
    "  -l   Long format: print given-path and real-path\n"
    "  -s   Short format: print just real-path\n"
    "  -V   Print version and exit\n"
    ;

enum { FMT_LONG, FMT_SHORT };
static int format_type = FMT_LONG;

#ifndef lint
/* Prevent over-aggressive optimizers from eliminating ID string */
extern const char jlss_id_realpath_c[];
const char jlss_id_realpath_c[] = "@(#)$Id: realpath.c,v 1.3 2007/10/23 20:23:44 jleffler Exp $";
#endif /* lint */

static int eval_realpath(const char *given)
{
    char realname[_POSIX_PATH_MAX];
    int rc = 0;

    if (realpath(given, realname) == 0)
    {
        rc = -1;
        err_sysrem("failed to resolve real path name for %s\n", given);
    }
    else if (format_type == FMT_SHORT)
        printf("%s\n", realname);
    else
        printf("%s %s\n", given, realname);
    return(rc);
}

int main(int argc, char **argv)
{
    int i;
    int rc = EXIT_SUCCESS;
    int opt;

    err_setarg0(argv[0]);
    while ((opt = getopt(argc, argv, optstr)) != -1)
    {
        switch (opt)
        {
        case 'V':
            err_version("REALPATH", &"@(#)$Revision: 1.3 $ ($Date: 2007/10/23 20:23:44 $)"[4]);
            break;
        case 'h':
            err_help(usestr, hlpstr);
            break;
        case 'l':
            format_type = FMT_LONG;
            break;
        case 's':
            format_type = FMT_SHORT;
            break;
        default:
            err_usage(usestr);
            break;
        }
    }

    for (i = optind; i < argc; i++)
    {
        if (eval_realpath(argv[i]) != 0)
            rc = EXIT_FAILURE;
    }

    return(rc);
}

Мне нужно было протестировать некоторое программное обеспечение, которое оценивало безопасность пути, и мне нужно было убедиться, что мой код оценивает данный путь в том же разрешенном местоположении, что и realpath(). Вероятно, было бы разумно расширить его с помощью опции -a, чтобы гарантировать, что имена сопоставляются с абсолютными именами (путем префикса результата getcwd() относительных путей).

(Источник для stderr.c, stderr.h можно найти в Интернете, если вы знаете, где искать. Или свяжитесь со мной - см. мой профиль.)

Ответ 4

Вот так:

int main(int argc, char *argv[]) {
    ...
    char *symlinkpath = argv[1];
    char actualpath [PATH_MAX];
    char *ptr;
    ptr = realpath(symlinkpath, actualpath);
    ...
}

Заимствован из здесь