/* $Id: merge.c 710 2006-07-27 19:02:06Z bruce $ */ /* path/merge.c - Merge one path into another. * Copyright (C) 2001,2005 Bruce Guenter * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include "path.h" static int path_merge_part(str* path, const char* part, unsigned long len) { if (part[0] == '.') { /* "." parts get dropped */ if (len == 1) return 1; /* ".." parts cause truncation */ if (len == 2 && part[1] == '.') { int sep = str_findlast(path, '/'); /* Don't truncate a leading '/' */ if (sep == 0) sep = 1; /* If no '/' found, empty the string */ else if (sep == -1) sep = 0; str_truncate(path, sep); return 1; } } if (path->len) if (path->s[path->len-1] != '/') if (!str_catc(path, '/')) return 0; return str_catb(path, part, len); } /** Merge two paths together. This function takes a starting path, and merges the second path into it. If the second path starts with a "/", it replaces the existing path completely. If the second path contains any ".." components, the preceding directory component in the current path (if any) is removed. All "." components are removed. The effects of this function are equivalent to changing directory to the first path and then using the second one. */ int path_merge(str* path, const char* start) { /* If the path to merge in starts with a "/", drop the previous path */ if (*start == '/') if (!str_copys(path, "/")) return 0; /* This should possibly use a striter... */ while (*start) { const char* end; while (*start == '/') ++start; if ((end = strchr(start, '/')) != 0) { if (!path_merge_part(path, start, end - start)) return 0; start = end + 1; } else { if (!path_merge_part(path, start, strlen(start))) return 0; break; } } return 1; } #ifdef SELFTEST_MAIN #include "selftest.c" #include "path.h" static str path; void showpath(void) { obuf_putstr(&outbuf, &path); NL(); } MAIN { str_copys(&path, "/"); path_merge(&path, "a"); showpath(); path_merge(&path, "b"); showpath(); path_merge(&path, "../c"); showpath(); path_merge(&path, "/d"); showpath(); str_copys(&path, ""); path_merge(&path, "a"); showpath(); path_merge(&path, "b"); showpath(); path_merge(&path, "../c"); showpath(); path_merge(&path, "/d"); showpath(); } #endif #ifdef SELFTEST_EXP /a /a/b /a/c /d a a/b a/c /d #endif