/* Copyright (C) 2013-2014 Paul Davis This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program 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 General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include #include #include #include #include #include #include #include "pbd/compose.h" #include "pbd/debug.h" #include "pbd/pathexpand.h" #include "pbd/strsplit.h" #include "pbd/tokenizer.h" using std::string; using std::vector; string PBD::path_expand (string path) { if (path.empty()) { return path; } /* tilde expansion */ if (path[0] == '~') { if (path.length() == 1) { return Glib::get_home_dir(); } if (path[1] == '/') { path.replace (0, 1, Glib::get_home_dir()); } else { /* can't handle ~roger, so just leave it */ } } /* now do $VAR substitution, since wordexp isn't reliable */ regex_t compiled_pattern; const int nmatches = 100; regmatch_t matches[nmatches]; if (regcomp (&compiled_pattern, "\\$([a-zA-Z_][a-zA-Z0-9_]*|\\{[a-zA-Z_][a-zA-Z0-9_]*\\})", REG_EXTENDED)) { std::cerr << "bad regcomp\n"; return path; } while (true) { if (regexec (&compiled_pattern, path.c_str(), nmatches, matches, 0)) { break; } /* matches[0] gives the entire match */ string match = path.substr (matches[0].rm_so, matches[0].rm_eo - matches[0].rm_so); /* try to get match from the environment */ if (match[1] == '{') { /* ${FOO} form */ match = match.substr (2, match.length() - 3); } char* matched_value = getenv (match.c_str()); if (matched_value) { path.replace (matches[0].rm_so, matches[0].rm_eo - matches[0].rm_so, matched_value); } else { path.replace (matches[0].rm_so, matches[0].rm_eo - matches[0].rm_so, string()); } /* go back and do it again with whatever remains after the * substitution */ } regfree (&compiled_pattern); /* canonicalize */ return canonical_path (path); } string PBD::search_path_expand (string path) { if (path.empty()) { return path; } vector s; vector n; split (path, s, G_SEARCHPATH_SEPARATOR); for (vector::iterator i = s.begin(); i != s.end(); ++i) { string exp = path_expand (*i); if (!exp.empty()) { n.push_back (exp); } } string r; for (vector::iterator i = n.begin(); i != n.end(); ++i) { if (!r.empty()) { r += G_SEARCHPATH_SEPARATOR; } r += *i; } return r; } std::vector PBD::parse_path(std::string path, bool check_if_exists) { vector pathlist; vector tmp; PBD::tokenize (path, string(G_SEARCHPATH_SEPARATOR_S), std::back_inserter (tmp)); for(vector::const_iterator i = tmp.begin(); i != tmp.end(); ++i) { if ((*i).empty()) continue; std::string dir; #ifndef PLATFORM_WINDOWS if ((*i).substr(0,1) == "~") { dir = Glib::build_filename(Glib::get_home_dir(), (*i).substr(1)); } else #endif { dir = *i; } if (!check_if_exists || Glib::file_test (dir, Glib::FILE_TEST_IS_DIR)) { pathlist.push_back(dir); } } return pathlist; }