web-dev-qa-db-ja.com

Cでのrealpath関数の例

Cプログラムでrealpath関数を使用する方法の例を探しています。 WebやCプログラミングの本で見つけられないようです。

16
Ralph

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は <limits.h>POSIX 1997の<limits.h> )で定義されています。

22
pmg

realpath()関数が行うことは、すべてのシンボリックリンクが解決されたときにファイルのパス名を通知することです。指定する値が相対名である場合、必ずしも絶対パス名である必要はありませんが、リンク値の絶対名を使用してシンボリックリンクをトラバースするかどうかに一部依存します。トラバースする場合、出力は結局絶対名になります。 。また、相対名がルートディレクトリまでトラバースする場合(または、ディレクトリ階層の深さが3レベルしかない場合は、「../../../../../ ..」のように「beyond」)。

マシンに「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のソースは、どこを見ればよいかわかっている場合はオンラインで見つけることができます。または私に連絡してください-私のプロファイルを参照してください。)

3

1行のビルドコマンドライン

ミニマリストですが、それは仕事をします!

ビルド

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

テスト

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

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

要件

  • gcc コンパイルとリンク用
  • bash<<<および$' ... \n ... 'の場合

クラッシュ

私のミニマリストの1行のコマンドラインは、パスが存在しない場合にSegmentation faultを生成する実行可能ファイルrealpathを作成します。私の答えの中でその問題を処理するためにif/elseブロックを書く代わりに、realpathreadlinkのBusybox実装を見ることができるようにいくつかのリンクを以下に追加しました。


Busyboxの実装

より完全なソースコードについては、この単純な実装をご覧ください。

公式Gitリポジトリ

GitHubミラーリポジトリ

3
olibre

このような:

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

から借りた ここ

0
fvu