libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
path.hxx File Reference
#include <string>
#include <string_view>
#include <cosmos/dso_export.h>
#include <cosmos/SysString.hxx>

Go to the source code of this file.

Functions

std::string cosmos::fs::normalize_path (const std::string_view path)
 Takes an input path and returns a normalized version of it.
 
std::string cosmos::fs::canonicalize_path (const SysString path)
 Normalizes path and resolves any symbolic link components.
 

Detailed Description

File system path related helpers.

Definition in file path.hxx.

Function Documentation

◆ canonicalize_path()

COSMOS_API std::string cosmos::fs::canonicalize_path ( const SysString path)

Normalizes path and resolves any symbolic link components.

Definition at line 64 of file path.cxx.

64 {
65 // implementing this on foot is non-trivial so rely on realpath() for
66 // the time being.
67 std::string ret;
68 ret.resize(max::PATH);
69 if (::realpath(path.raw(), ret.data()) == nullptr) {
70 cosmos_throw (FileError(path, "realpath()"));
71 }
72
73 ret.resize(std::strlen(ret.data()));
74 return ret;
75}

◆ normalize_path()

COSMOS_API std::string cosmos::fs::normalize_path ( const std::string_view path)

Takes an input path and returns a normalized version of it.

A normalized path is an absolute path without any redundant separators ('/') or relative path components (".", "..").

To achieve this, this function eliminates redundant components and expands the current working directory, if necessary. Retrieving the CWD is the only potential system call performed by this function.

Symbolic links will not be resolved. If you want this you can use canonicalize_path() instead.

Definition at line 17 of file path.cxx.

17 {
18 std::string ret;
19
20 size_t slashes = 0;
21 size_t start = 0;
22 auto end = path.find('/');
23
24 while (start < path.size()) {
25 const auto component = path.substr(start, end - start);
26
27 if (component.size() > 0) {
28
29 // path starts without a leading /, so it's a relative
30 // path, add the CWD
31 if (!slashes && ret.empty()) {
32 ret.append(get_working_dir());
33 }
34
35 // remove the last component
36 if (component == "..") {
37 // don't remove the root directory
38 if (ret.size() > 1) {
39 ret.erase(ret.rfind('/'));
40 }
41 slashes++;
42 // references to the CWD are simply ignored
43 } else if (component == ".") {
44 slashes++;
45 } else {
46 // add exactly this component
47 ret.push_back('/');
48 ret.append(component);
49 }
50 } else {
51 slashes++;
52 }
53
54 start = end != path.npos ? end + 1 : end;
55 end = path.find('/', start);
56 }
57
58 if (slashes && ret.empty())
59 ret.push_back('/');
60
61 return ret;
62}