c - realpath() 不解析符号链接(symbolic link)?

标签 c normalization symlink realpath

我已经读过有关 realpath() 的内容,但是是否有一个函数可以传递基本目录和文件名,可以在不解析符号链接(symbolic link)或检查文件是否实际存在的情况下给出以下结果?或者我是否必须使用修改后的realpath()

"/var/", "../etc///././/passwd" => "/etc/passwd"

最佳答案

这是一个 normalize_path() 函数:

如果给定路径是相对路径,则该函数首先在其前面添加当前工作目录。

然后处理特殊路径组件,如...或空组件,并返回结果。

对于 ..,如果有最后一个组件,则将其删除(/.. 将仅返回 /)。
对于 . 或空组件(双 /),这将被跳过。

该函数确保不会返回空路径(而是返回 /)。

#define _GNU_SOURCE /* memrchr() */

#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <limits.h>

char * normalize_path(const char * src, size_t src_len) {

        char * res;
        size_t res_len;

        const char * ptr = src;
        const char * end = &src[src_len];
        const char * next;

        if (src_len == 0 || src[0] != '/') {

                // relative path

                char pwd[PATH_MAX];
                size_t pwd_len;

                if (getcwd(pwd, sizeof(pwd)) == NULL) {
                        return NULL;
                }

                pwd_len = strlen(pwd);
                res = malloc(pwd_len + 1 + src_len + 1);
                memcpy(res, pwd, pwd_len);
                res_len = pwd_len;
        } else {
                res = malloc((src_len > 0 ? src_len : 1) + 1);
                res_len = 0;
        }

        for (ptr = src; ptr < end; ptr=next+1) {
                size_t len;
                next = memchr(ptr, '/', end-ptr);
                if (next == NULL) {
                        next = end;
                }
                len = next-ptr;
                switch(len) {
                case 2:
                        if (ptr[0] == '.' && ptr[1] == '.') {
                                const char * slash = memrchr(res, '/', res_len);
                                if (slash != NULL) {
                                        res_len = slash - res;
                                }
                                continue;
                        }
                        break;
                case 1:
                        if (ptr[0] == '.') {
                                continue;

                        }
                        break;
                case 0:
                        continue;
                }
                res[res_len++] = '/';
                memcpy(&res[res_len], ptr, len);
                res_len += len;
        }

        if (res_len == 0) {
                res[res_len++] = '/';
        }
        res[res_len] = '\0';
        return res;
}

关于c - realpath() 不解析符号链接(symbolic link)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34201143/

相关文章:

C 二分查找

转换双指针以及如何在 C 中使用

pandas - 无法使用FeatureUnion在Python中组合处理后的数字和分类特征

python - 标准化包含过大值的数据集

git - 如何公开使用子模块,但符号链接(symbolic link)到本地​​的单个克隆?

java - 在一个路径中一起解析符号链接(symbolic link)和 ".."

c - 消息太长 - 缓冲区大于消息大小

c - 如何从用户那里获取链表的数据

MySQL表设计和规范化协助

linux - 在 Linux 中,如何让我的 Perl 脚本像二进制文件一样运行?