/* * Copyright (C) 2010 Vyatta, Inc. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * 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, see . */ #include #include #include #include #include #include #include #include #include #include #include namespace cstore { // begin namespace cstore namespace unionfs { // begin namespace unionfs ////// constants // environment vars defining root dirs const string UnionfsCstore::C_ENV_TMPL_ROOT = "VYATTA_CONFIG_TEMPLATE"; const string UnionfsCstore::C_ENV_WORK_ROOT = "VYATTA_TEMP_CONFIG_DIR"; const string UnionfsCstore::C_ENV_ACTIVE_ROOT = "VYATTA_ACTIVE_CONFIGURATION_DIR"; const string UnionfsCstore::C_ENV_CHANGE_ROOT = "VYATTA_CHANGES_ONLY_DIR"; const string UnionfsCstore::C_ENV_TMP_ROOT = "VYATTA_CONFIG_TMP"; // default root dirs/paths const string UnionfsCstore::C_DEF_TMPL_ROOT = "/opt/vyatta/share/vyatta-cfg/templates"; const string UnionfsCstore::C_DEF_CFG_ROOT = "/opt/vyatta/config"; const string UnionfsCstore::C_DEF_ACTIVE_ROOT = UnionfsCstore::C_DEF_CFG_ROOT + "/active"; const string UnionfsCstore::C_DEF_CHANGE_PREFIX = UnionfsCstore::C_DEF_CFG_ROOT + "/tmp/changes_only_"; const string UnionfsCstore::C_DEF_WORK_PREFIX = UnionfsCstore::C_DEF_CFG_ROOT + "/tmp/new_config_"; const string UnionfsCstore::C_DEF_TMP_PREFIX = UnionfsCstore::C_DEF_CFG_ROOT + "/tmp/tmp_"; // markers const string UnionfsCstore::C_MARKER_DEF_VALUE = "def"; const string UnionfsCstore::C_MARKER_DEACTIVATE = ".disable"; const string UnionfsCstore::C_MARKER_CHANGED = ".modified"; const string UnionfsCstore::C_MARKER_UNSAVED = ".unsaved"; const string UnionfsCstore::C_COMMITTED_MARKER_FILE = ".changes"; const string UnionfsCstore::C_COMMENT_FILE = ".comment"; const string UnionfsCstore::C_TAG_NAME = "node.tag"; const string UnionfsCstore::C_VAL_NAME = "node.val"; const string UnionfsCstore::C_DEF_NAME = "node.def"; const string UnionfsCstore::C_COMMIT_LOCK_FILE = "/opt/vyatta/config/.lock"; ////// static static MapT _fs_escape_chars; static MapT _fs_unescape_chars; static void _init_fs_escape_chars() { _fs_escape_chars[-1] = "\%\%\%"; _fs_escape_chars['%'] = "\%25"; _fs_escape_chars['/'] = "\%2F"; _fs_unescape_chars["\%\%\%"] = -1; _fs_unescape_chars["\%25"] = '%'; _fs_unescape_chars["\%2F"] = '/'; } static string _escape_char(char c) { MapT::iterator p = _fs_escape_chars.find(c); if (p != _fs_escape_chars.end()) { return p->second; } else { return string(1, c); } } static MapT _escape_path_name_cache; static string _escape_path_name(const string& path) { MapT::iterator p = _escape_path_name_cache.find(path); if (p != _escape_path_name_cache.end()) { // found escaped string in cache. just return it. return p->second; } // special case for empty string string npath = (path.size() == 0) ? _fs_escape_chars[-1] : ""; for (size_t i = 0; i < path.size(); i++) { npath += _escape_char(path[i]); } // cache it before return _escape_path_name_cache[path] = npath; return npath; } static MapT _unescape_path_name_cache; static string _unescape_path_name(const string& path) { MapT::iterator p = _unescape_path_name_cache.find(path); if (p != _unescape_path_name_cache.end()) { // found unescaped string in cache. just return it. return p->second; } // assume all escape patterns are 3-char string npath = ""; for (size_t i = 0; i < path.size(); i++) { if ((path.size() - i) < 3) { npath += path.substr(i); break; } string s = path.substr(i, 3); MapT::iterator p = _fs_unescape_chars.find(s); if (p != _fs_unescape_chars.end()) { char c = p->second; if (path.size() == 3 && c == -1) { // special case for empty string npath = ""; break; } npath += string(1, _fs_unescape_chars[s]); // skip the escape sequence i += 2; } else { npath += path.substr(i, 1); } } // cache it before return _unescape_path_name_cache[path] = npath; return npath; } ////// constructor/destructor /* "current session" constructor. * this constructor sets up the object from environment. * used when environment is already set up, i.e., when operating on the * "current" config session. e.g., in the following scenarios * configure commands * perl module * shell "current session" api * * note: this also applies when using the cstore in operational mode, * in which case only the template root and the active root will be * valid. */ UnionfsCstore::UnionfsCstore(bool use_edit_level) { // set up root dir strings char *val; if ((val = getenv(C_ENV_TMPL_ROOT.c_str()))) { tmpl_path = val; } else { tmpl_path = C_DEF_TMPL_ROOT; } tmpl_root = tmpl_path; // save a copy of tmpl root if ((val = getenv(C_ENV_WORK_ROOT.c_str()))) { work_root = val; } if ((val = getenv(C_ENV_TMP_ROOT.c_str()))) { tmp_root = val; init_commit_data(); } if ((val = getenv(C_ENV_ACTIVE_ROOT.c_str()))) { active_root = val; } else { active_root = C_DEF_ACTIVE_ROOT; } if ((val = getenv(C_ENV_CHANGE_ROOT.c_str()))) { change_root = val; } /* note: the original perl API module does not use the edit levels * from environment. only the actual CLI operations use them. * so here make it an option. */ mutable_cfg_path = "/"; if (use_edit_level) { // set up path strings if ((val = getenv(C_ENV_EDIT_LEVEL.c_str()))) { mutable_cfg_path = val; } if ((val = getenv(C_ENV_TMPL_LEVEL.c_str())) && val[0] && val[1]) { /* no need to append root (i.e., "/"). level (if exists) always * starts with '/', so only append it if it is at least two chars * (i.e., it is not "/"). */ FsPath tlvl(val); tmpl_path /= tlvl; } } orig_mutable_cfg_path = mutable_cfg_path; orig_tmpl_path = tmpl_path; _init_fs_escape_chars(); } /* "specific session" constructor. * this constructor sets up the object for the specified session ID and * returns an environment string that can be "evaled" to set up the * shell environment. * * used when the session environment needs to be established. this is * mainly for the shell functions that set up configuration sessions. * i.e., the "vyatta-cfg-cmd-wrapper" (on boot or for GUI etc.) and * the cfg completion script (when entering configure mode). * * sid: session ID. * env: (output) environment string. * * note: this does NOT set up the session. caller needs to use the * explicit session setup/teardown functions as needed. */ UnionfsCstore::UnionfsCstore(const string& sid, string& env) : Cstore(env) { tmpl_root = C_DEF_TMPL_ROOT; tmpl_path = tmpl_root; active_root = C_DEF_ACTIVE_ROOT; work_root = (C_DEF_WORK_PREFIX + sid); change_root = (C_DEF_CHANGE_PREFIX + sid); tmp_root = (C_DEF_TMP_PREFIX + sid); init_commit_data(); string declr = " declare -x -r "; // readonly vars env += " umask 002; {"; env += (declr + C_ENV_ACTIVE_ROOT + "=" + active_root.path_cstr()); env += (declr + C_ENV_CHANGE_ROOT + "=" + change_root.path_cstr() + ";"); env += (declr + C_ENV_WORK_ROOT + "=" + work_root.path_cstr() + ";"); env += (declr + C_ENV_TMP_ROOT + "=" + tmp_root.path_cstr() + ";"); env += (declr + C_ENV_TMPL_ROOT + "=" + tmpl_root.path_cstr() + ";"); env += " } >&/dev/null || true"; // set up path strings using level vars char *val; mutable_cfg_path = "/"; if ((val = getenv(C_ENV_EDIT_LEVEL.c_str()))) { mutable_cfg_path = val; } if ((val = getenv(C_ENV_TMPL_LEVEL.c_str())) && val[0] && val[1]) { // see comment in the other constructor FsPath tlvl(val); tmpl_path /= tlvl; } orig_mutable_cfg_path = mutable_cfg_path; orig_tmpl_path = tmpl_path; _init_fs_escape_chars(); } UnionfsCstore::~UnionfsCstore() { } ////// public virtual functions declared in base class bool UnionfsCstore::markSessionUnsaved() { FsPath marker = work_root; marker.push(C_MARKER_UNSAVED); if (path_exists(marker)) { // already marked. treat as success. return true; } if (!create_file(marker)) { output_internal("failed to mark unsaved [%s]\n", marker.path_cstr()); return false; } return true; } bool UnionfsCstore::unmarkSessionUnsaved() { FsPath marker = work_root; marker.push(C_MARKER_UNSAVED); if (!path_exists(marker)) { // not marked. treat as success. return true; } try { b_fs::remove(marker.path_cstr()); } catch (...) { output_internal("failed to unmark unsaved [%s]\n", marker.path_cstr()); return false; } return true; } bool UnionfsCstore::sessionUnsaved() { FsPath marker = work_root; marker.push(C_MARKER_UNSAVED); return path_exists(marker); } bool UnionfsCstore::sessionChanged() { FsPath marker = work_root; marker.push(C_MARKER_CHANGED); return path_exists(marker); } /* set up the session associated with this object. * the session comes from either the environment or the session ID * (see the two different constructors). */ bool UnionfsCstore::setupSession() { if (!path_exists(work_root)) { // session doesn't exist. create dirs. try { b_fs::create_directories(work_root.path_cstr()); b_fs::create_directories(change_root.path_cstr()); b_fs::create_directories(tmp_root.path_cstr()); if (!path_exists(active_root)) { // this should only be needed on boot b_fs::create_directories(active_root.path_cstr()); } } catch (...) { output_internal("setup session failed to create session directories\n"); return false; } // union mount if (!do_mount(change_root, active_root, work_root)) { return false; } } else if (!path_is_directory(work_root)) { output_internal("setup session not dir [%s]\n", work_root.path_cstr()); return false; } return true; } /* tear down the session associated with this object. * the session comes from either the environment or the session ID * (see the two different constructors). */ bool UnionfsCstore::teardownSession() { // check if session exists string wstr = work_root.path_cstr(); if (wstr.empty() || wstr.find(C_DEF_WORK_PREFIX) != 0 || !path_exists(work_root) || !path_is_directory(work_root)) { // no session output_internal("teardown invalid session [%s]\n", wstr.c_str()); return false; } // unmount the work root (union) if (!do_umount(work_root)) { return false; } // remove session directories bool ret = false; try { if (b_fs::remove_all(work_root.path_cstr()) != 0 && b_fs::remove_all(change_root.path_cstr()) != 0 && b_fs::remove_all(tmp_root.path_cstr()) != 0) { ret = true; } } catch (...) { } if (!ret) { output_internal("failed to remove session directories\n"); } return ret; } /* whether an actual config session is associated with this object. * the session comes from either the environment or the session ID * (see the two different constructors). */ bool UnionfsCstore::inSession() { string wstr = work_root.path_cstr(); return (!wstr.empty() && wstr.find(C_DEF_WORK_PREFIX) == 0 && path_exists(work_root) && path_is_directory(work_root)); } bool UnionfsCstore::clearCommittedMarkers() { try { b_fs::remove(commit_marker_file.path_cstr()); } catch (...) { output_internal("failed to clear committed markers\n"); return false; } return true; } bool UnionfsCstore::construct_commit_active(commit::PrioNode& node) { auto_ptr save(create_save_paths()); reset_paths(); append_cfg_path(node.getCommitPath()); FsPath ap(get_active_path()); FsPath wp(get_work_path()); FsPath tap(tmp_active_root); tap /= mutable_cfg_path; if (path_exists(tap)) { output_internal("rm[%s]\n", tap.path_cstr()); if (b_fs::remove_all(tap.path_cstr()) < 1) { output_internal("rm ta failed\n"); return false; } cnode::CfgNode *c = node.getCfgNode(); if (c && c->isTag()) { FsPath p(tap); p.pop(); if (is_directory_empty(p)) { output_internal("rm[%s]\n", p.path_cstr()); if (b_fs::remove_all(p.path_cstr()) < 1) { output_internal("rm tag failed\n"); return false; } } } } else { output_internal("no tap[%s]\n", tap.path_cstr()); } if (node.succeeded()) { // prio subtree succeeded if (path_exists(wp)) { output_internal("cp[%s]->[%s]\n", wp.path_cstr(), tap.path_cstr()); try { recursive_copy_dir(wp, tap, true); } catch (const b_fs::filesystem_error& e) { output_internal("cp w->ta failed[%s]\n", e.what()); return false; } catch (...) { output_internal("cp w->ta failed[unknown exception]\n"); return false; } } else { output_internal("no wp[%s]\n", wp.path_cstr()); } if (!node.hasSubtreeFailure()) { // whole subtree succeeded => stop recursion return true; } // failure present in subtree } else { // prio subtree failed if (path_exists(ap)) { output_internal("cp[%s]->[%s]\n", ap.path_cstr(), tap.path_cstr()); try { recursive_copy_dir(ap, tap, false); } catch (const b_fs::filesystem_error& e) { output_internal("cp a->ta failed[%s]\n", e.what()); return false; } catch (...) { output_internal("cp a->ta failed[unknown exception]\n"); return false; } } else { output_internal("no ap[%s]\n", ap.path_cstr()); } if (!node.hasSubtreeSuccess()) { // whole subtree failed => stop recursion return true; } // success present in subtree } for (size_t i = 0; i < node.numChildNodes(); i++) { if (!construct_commit_active(*(node.childAt(i)))) { return false; } } return true; } bool UnionfsCstore::mark_dir_changed(const FsPath& d, const FsPath& root) { if (!path_is_directory(d)) { output_internal("mark_dir_changed on non-directory [%s]\n", d.path_cstr()); return false; } FsPath marker(d); while (marker.size() >= root.size()) { marker.push(C_MARKER_CHANGED); if (path_exists(marker)) { // reached a node already marked => done break; } if (!create_file(marker)) { output_internal("failed to mark changed [%s]\n", marker.path_cstr()); return false; } marker.pop(); marker.pop(); } return true; } bool UnionfsCstore::sync_dir(const FsPath& src, const FsPath& dst, const FsPath& root) { if (!path_exists(src) || !path_exists(dst)) { output_user("sync_dir with non-existing dir(s)[%s][%s]\n", src.path_cstr(), dst.path_cstr()); return false; } MapT smap; MapT dmap; vector sentries; vector dentries; check_dir_entries(src, &sentries, false); check_dir_entries(dst, &dentries, false); for (size_t i = 0; i < sentries.size(); i++) { smap[sentries[i]] = true; } for (size_t i = 0; i < dentries.size(); i++) { dmap[dentries[i]] = true; if (smap.find(dentries[i]) == smap.end()) { // entry in dst but not in src => delete FsPath d(dst); if (!mark_dir_changed(d, root)) { return false; } push_path(d, dentries[i].c_str()); if (b_fs::remove_all(d.path_cstr()) < 1) { return false; } } else { // entry in both src and dst FsPath s(src); FsPath d(dst); push_path(s, dentries[i].c_str()); push_path(d, dentries[i].c_str()); if (path_is_regular(s) && path_is_regular(d)) { // it's file => compare and replace if necessary string ds, dd; if (!read_whole_file(s, ds) || !read_whole_file(d, dd)) { // error output_user("failed to replace file [%s][%s]\n", s.path_cstr(), d.path_cstr()); return false; } if (ds != dd) { // need to replace if (!write_file(d, ds)) { output_user("failed to write file [%s]\n", d.path_cstr()); return false; } d.pop(); if (!mark_dir_changed(d, root)) { return false; } } } else if (path_is_directory(s) && path_is_directory(d)) { // it's dir => recurse if (!sync_dir(s, d, root)) { return false; } } else { // something is wrong output_user("inconsistent config entry [%s][%s]\n", s.path_cstr(), d.path_cstr()); return false; } } } for (size_t i = 0; i < sentries.size(); i++) { if (dmap.find(sentries[i]) == dmap.end()) { // entry in src but not in dst => copy FsPath s(src); FsPath d(dst); push_path(s, sentries[i].c_str()); push_path(d, sentries[i].c_str()); try { if (path_is_regular(s)) { // it's file b_fs::copy_file(s.path_cstr(), d.path_cstr()); } else { // dir recursive_copy_dir(s, d, true); } d.pop(); if (!mark_dir_changed(d, root)) { return false; } } catch (...) { output_user("copy failed [%s][%s]\n", s.path_cstr(), d.path_cstr()); return false; } } } return true; } bool UnionfsCstore::commitConfig(commit::PrioNode& node) { // make a copy of current "work" dir try { if (path_exists(tmp_work_root)) { output_internal("rm[%s]\n", tmp_work_root.path_cstr()); if (b_fs::remove_all(tmp_work_root.path_cstr()) < 1) { output_internal("rm tw failed\n"); return false; } } output_internal("cp[%s]->[%s]\n", work_root.path_cstr(), tmp_work_root.path_cstr()); recursive_copy_dir(work_root, tmp_work_root, true); } catch (const b_fs::filesystem_error& e) { output_internal("cp w->tw failed[%s]\n", e.what()); return false; } catch (...) { output_internal("cp w->tw failed[unknown exception]\n"); return false; } if (!construct_commit_active(node)) { return false; } if (!do_umount(work_root)) { return false; } if (b_fs::remove_all(change_root.path_cstr()) < 1) { output_internal("failed to remove [%s]\n", change_root.path_cstr()); return false; } /* note: unionfs can't cope with whole directory being removed, so just * remove the content. */ if (!remove_dir_content(active_root.path_cstr())) { output_internal("failed to remove [%s] content\n", active_root.path_cstr()); return false; } try { b_fs::create_directories(change_root.path_cstr()); recursive_copy_dir(tmp_active_root, active_root, true); } catch (const b_fs::filesystem_error& e) { output_internal("cp ta->a failed[%s]\n", e.what()); return false; } catch (...) { output_internal("cp ta->a failed[unknown exception]\n"); return false; } if (!do_mount(change_root, active_root, work_root)) { return false; } if (!sync_dir(tmp_work_root, work_root, work_root)) { return false; } if (b_fs::remove_all(tmp_work_root.path_cstr()) < 1 || b_fs::remove_all(tmp_active_root.path_cstr()) < 1) { output_user("failed to remove temp directories\n"); return false; } // all done return true; } bool UnionfsCstore::getCommitLock() { int fd = creat(C_COMMIT_LOCK_FILE.c_str(), 0777); if (fd < 0) { // should not happen since all commit processes should have write access output_internal("getCommitLock() failed to open lock file\n"); return false; } if (lockf(fd, F_TLOCK, 0) < 0) { // locked by someone else return false; } // got the lock return true; } ////// virtual functions defined in base class /* check if current tmpl_path is a valid tmpl dir. * return true if valid. otherwise return false. */ bool UnionfsCstore::tmpl_node_exists() { return (path_exists(tmpl_path) && path_is_directory(tmpl_path)); } typedef MapT, FsPathHash> ParsedTmplCacheT; static ParsedTmplCacheT _parsed_tmpl_cache; /* parse template at current tmpl_path and return an allocated Ctemplate * pointer if successful. otherwise return 0. */ Ctemplate * UnionfsCstore::tmpl_parse() { FsPath tp = tmpl_path; tp.push(C_DEF_NAME); if (!path_exists(tp) || !path_is_regular(tp)) { // invalid return 0; } ParsedTmplCacheT::iterator p = _parsed_tmpl_cache.find(tp); if (p != _parsed_tmpl_cache.end()) { // found in cache return (new Ctemplate(p->second)); } // new template => parse tr1::shared_ptr def(new vtw_def); vtw_def *_def = def.get(); if (_def && parse_def(_def, tp.path_cstr(), 0) == 0) { // succes => cache and return _parsed_tmpl_cache[tp] = def; return (new Ctemplate(def)); } return 0; } bool UnionfsCstore::cfg_node_exists(bool active_cfg) { FsPath p = (active_cfg ? get_active_path() : get_work_path()); return (path_exists(p) && path_is_directory(p)); } bool UnionfsCstore::add_node() { bool ret = true; try { if (!b_fs::create_directory(get_work_path().path_cstr())) { // already exists. shouldn't call this function. ret = false; } } catch (...) { ret = false; } if (!ret) { output_internal("failed to add node [%s]\n", get_work_path().path_cstr()); } return ret; } bool UnionfsCstore::remove_node() { if (!path_exists(get_work_path()) || !path_is_directory(get_work_path())) { output_internal("remove non-existent node [%s]\n", get_work_path().path_cstr()); return false; } bool ret = false; try { if (b_fs::remove_all(get_work_path().path_cstr()) != 0) { ret = true; } } catch (...) { ret = false; } if (!ret) { output_internal("failed to remove node [%s]\n", get_work_path().path_cstr()); } return ret; } void UnionfsCstore::get_all_child_node_names_impl(vector& cnodes, bool active_cfg) { FsPath p = (active_cfg ? get_active_path() : get_work_path()); get_all_child_dir_names(p, cnodes); /* XXX special cases to emulate original perl API behavior. * original perl listNodes() and listOrigNodes() return everything * under a node (except for ".*"), including "node.val" and "def". * * perl API should operate at abstract level and should not access * such implementation-specific details. however, currently * things like config output depend on this behavior, so this * function needs to return them for now. * * use a whilelist-approach, i.e., only add the following: * node.val * def * * FIXED: perl scripts have been changed to eliminate the use of "def" * and "node.val", so they no longer need to be returned. */ } bool UnionfsCstore::read_value_vec(vector& vvec, bool active_cfg) { FsPath vpath = (active_cfg ? get_active_path() : get_work_path()); vpath.push(C_VAL_NAME); string ostr; if (!read_whole_file(vpath, ostr)) { return false; } /* XXX original implementation used to remove a trailing '\n' after * a read. it was only necessary because it was adding a '\n' when * writing the file. don't remove anything now since we shouldn't * be writing it any more. */ // separate values using newline as delimiter size_t start_idx = 0, idx = 0; for (; idx < ostr.size(); idx++) { if (ostr[idx] == '\n') { // got a value vvec.push_back(ostr.substr(start_idx, (idx - start_idx))); start_idx = idx + 1; } } if (start_idx < ostr.size()) { vvec.push_back(ostr.substr(start_idx, (idx - start_idx))); } else { // last char is a newline => another empty value vvec.push_back(""); } return true; } bool UnionfsCstore::write_value_vec(const vector& vvec, bool active_cfg) { FsPath wp = (active_cfg ? get_active_path() : get_work_path()); wp.push(C_VAL_NAME); if (path_exists(wp) && !path_is_regular(wp)) { // not a file output_internal("failed to write node value (file) [%s]\n", wp.path_cstr()); return false; } string ostr = ""; for (size_t i = 0; i < vvec.size(); i++) { if (i > 0) { // subsequent values require delimiter ostr += "\n"; } ostr += vvec[i]; } if (!write_file(wp, ostr)) { output_internal("failed to write node value (write) [%s]\n", wp.path_cstr()); return false; } return true; } bool UnionfsCstore::rename_child_node(const char *oname, const char *nname) { FsPath opath = get_work_path(); opath.push(oname); FsPath npath = get_work_path(); npath.push(nname); if (!path_exists(opath) || !path_is_directory(opath) || path_exists(npath)) { output_internal("cannot rename node [%s,%s,%s]\n", get_work_path().path_cstr(), oname, nname); return false; } bool ret = true; try { /* somehow b_fs::rename() can't be used here as it considers the operation * "Invalid cross-device link" and fails with an exception, probably due * to unionfs in some way. * do it the hard way. */ recursive_copy_dir(opath, npath); if (b_fs::remove_all(opath.path_cstr()) == 0) { ret = false; } } catch (...) { ret = false; } if (!ret) { output_internal("failed to rename node [%s,%s]\n", opath.path_cstr(), npath.path_cstr()); } return ret; } bool UnionfsCstore::copy_child_node(const char *oname, const char *nname) { FsPath opath = get_work_path(); opath.push(oname); FsPath npath = get_work_path(); npath.push(nname); if (!path_exists(opath) || !path_is_directory(opath) || path_exists(npath)) { output_internal("cannot copy node [%s,%s,%s]\n", get_work_path().path_cstr(), oname, nname); return false; } try { recursive_copy_dir(opath, npath); } catch (...) { output_internal("failed to copy node [%s,%s,%s]\n", get_work_path().path_cstr(), oname, nname); return false; } return true; } bool UnionfsCstore::mark_display_default() { FsPath marker = get_work_path(); marker.push(C_MARKER_DEF_VALUE); if (path_exists(marker)) { // already marked. treat as success. return true; } if (!create_file(marker)) { output_internal("failed to mark default [%s]\n", get_work_path().path_cstr()); return false; } return true; } bool UnionfsCstore::unmark_display_default() { FsPath marker = get_work_path(); marker.push(C_MARKER_DEF_VALUE); if (!path_exists(marker)) { // not marked. treat as success. return true; } try { b_fs::remove(marker.path_cstr()); } catch (...) { output_internal("failed to unmark default [%s]\n", get_work_path().path_cstr()); return false; } return true; } bool UnionfsCstore::marked_display_default(bool active_cfg) { FsPath marker = (active_cfg ? get_active_path() : get_work_path()); marker.push(C_MARKER_DEF_VALUE); return path_exists(marker); } bool UnionfsCstore::marked_deactivated(bool active_cfg) { FsPath marker = (active_cfg ? get_active_path() : get_work_path()); marker.push(C_MARKER_DEACTIVATE); return path_exists(marker); } bool UnionfsCstore::mark_deactivated() { FsPath marker = get_work_path(); marker.push(C_MARKER_DEACTIVATE); if (path_exists(marker)) { // already marked. treat as success. return true; } if (!create_file(marker)) { output_internal("failed to mark deactivated [%s]\n", get_work_path().path_cstr()); return false; } return true; } bool UnionfsCstore::unmark_deactivated() { FsPath marker = get_work_path(); marker.push(C_MARKER_DEACTIVATE); if (!path_exists(marker)) { // not deactivated. treat as success. return true; } try { b_fs::remove(marker.path_cstr()); } catch (...) { output_internal("failed to unmark deactivated [%s]\n", get_work_path().path_cstr()); return false; } return true; } bool UnionfsCstore::unmark_deactivated_descendants() { bool ret = false; do { // sanity check if (!path_is_directory(get_work_path())) { break; } try { vector markers; b_fs::recursive_directory_iterator di(get_work_path().path_cstr()); for (; di != b_fs::recursive_directory_iterator(); ++di) { if (!path_is_regular(di->path().file_string().c_str()) || di->path().filename() != C_MARKER_DEACTIVATE) { // not marker continue; } const char *ppath = di->path().parent_path().file_string().c_str(); if (strcmp(ppath, get_work_path().path_cstr()) == 0) { // don't unmark the node itself continue; } markers.push_back(di->path()); } for (size_t i = 0; i < markers.size(); i++) { b_fs::remove(markers[i]); } } catch (...) { break; } ret = true; } while (0); if (!ret) { output_internal("failed to unmark deactivated descendants [%s]\n", get_work_path().path_cstr()); } return ret; } // mark current work path and all ancestors as "changed" bool UnionfsCstore::mark_changed_with_ancestors() { FsPath opath = mutable_cfg_path; // use a copy bool done = false; while (!done) { FsPath marker = work_root; if (opath.has_parent_path()) { marker /= opath; pop_path(opath); } else { done = true; } if (!path_exists(marker) || !path_is_directory(marker)) { // don't do anything if the node is not there continue; } marker.push(C_MARKER_CHANGED); if (path_exists(marker)) { // reached a node already marked => done break; } if (!create_file(marker)) { output_internal("failed to mark changed [%s]\n", marker.path_cstr()); return false; } } return true; } /* remove all "changed" markers under the current work path. this is used, * e.g., at the end of "commit" to reset a subtree. */ bool UnionfsCstore::unmark_changed_with_descendants() { try { vector markers; b_fs::recursive_directory_iterator di(get_work_path().path_cstr()); for (; di != b_fs::recursive_directory_iterator(); ++di) { if (!path_is_regular(di->path().file_string().c_str()) || di->path().filename() != C_MARKER_CHANGED) { // not marker continue; } markers.push_back(di->path()); } for (size_t i = 0; i < markers.size(); i++) { b_fs::remove(markers[i]); } } catch (...) { output_internal("failed to unmark changed with descendants [%s]\n", get_work_path().path_cstr()); return false; } return true; } // remove the comment at the current work path bool UnionfsCstore::remove_comment() { FsPath cfile = get_work_path(); cfile.push(C_COMMENT_FILE); if (!path_exists(cfile)) { return false; } try { b_fs::remove(cfile.path_cstr()); } catch (...) { output_internal("failed to remove comment [%s]\n", cfile.path_cstr()); return false; } return true; } // set comment at the current work path bool UnionfsCstore::set_comment(const string& comment) { FsPath cfile = get_work_path(); cfile.push(C_COMMENT_FILE); return write_file(cfile, comment); } // discard all changes in working config bool UnionfsCstore::discard_changes(unsigned long long& num_removed) { // need to keep unsaved marker bool unsaved = sessionUnsaved(); bool ret = true; vector files; vector directories; try { // iterate through all entries in change root b_fs::directory_iterator di(change_root.path_cstr()); for (; di != b_fs::directory_iterator(); ++di) { if (path_is_directory(di->path().file_string().c_str())) { directories.push_back(di->path()); } else { files.push_back(di->path()); } } // remove and count num_removed = 0; for (size_t i = 0; i < files.size(); i++) { b_fs::remove(files[i]); num_removed++; } for (size_t i = 0; i < directories.size(); i++) { num_removed += b_fs::remove_all(directories[i]); } } catch (...) { output_internal("discard failed [%s]\n", change_root.path_cstr()); ret = false; } if (unsaved) { // restore unsaved marker num_removed--; markSessionUnsaved(); } return ret; } // get comment at the current work or active path bool UnionfsCstore::get_comment(string& comment, bool active_cfg) { FsPath cfile = (active_cfg ? get_active_path() : get_work_path()); cfile.push(C_COMMENT_FILE); return read_whole_file(cfile, comment); } // whether current work path is "changed" bool UnionfsCstore::cfg_node_changed() { FsPath marker = get_work_path(); marker.push(C_MARKER_CHANGED); return path_exists(marker); } void UnionfsCstore::get_edit_level(Cpath& pcomps) { FsPath opath = mutable_cfg_path; // use a copy vector tmp; while (opath.has_parent_path()) { string last; pop_path(opath, last); tmp.push_back(last); } while (tmp.size() > 0) { pcomps.push(tmp.back()); tmp.pop_back(); } } bool UnionfsCstore::marked_committed(bool is_delete) { string marker; get_committed_marker(is_delete, marker); return find_line_in_file(commit_marker_file, marker); } bool UnionfsCstore::mark_committed(bool is_delete) { string marker; get_committed_marker(is_delete, marker); // write one marker per line return write_file(commit_marker_file, marker + "\n", true); } string UnionfsCstore::cfg_path_to_str() { string cpath = mutable_cfg_path.path_cstr(); if (cpath.length() == 0) { cpath = "/"; } return cpath; } string UnionfsCstore::tmpl_path_to_str() { // return only the mutable part string tpath = tmpl_path.path_cstr(); tpath.erase(0, tmpl_root.length()); if (tpath.length() == 0) { tpath = "/"; } return tpath; } ////// private functions void UnionfsCstore::push_path(FsPath& old_path, const char *new_comp) { string comp = _escape_path_name(new_comp); old_path.push(comp); } void UnionfsCstore::pop_path(FsPath& path) { path.pop(); } void UnionfsCstore::pop_path(FsPath& path, string& last) { path.pop(last); last = _unescape_path_name(last); } bool UnionfsCstore::check_dir_entries(const FsPath& root, vector *cnodes, bool filter_nodes, bool empty_check) { if (!path_exists(root) || !path_is_directory(root)) { // not a valid root => treat as empty return false; } bool found = false; try { b_fs::directory_iterator di(root.path_cstr()); for (; di != b_fs::directory_iterator(); ++di) { string cname = di->path().filename(); if (filter_nodes) { // must be directory if (!path_is_directory(di->path().file_string().c_str())) { continue; } // name cannot start with "." if (cname.length() < 1 || cname[0] == '.') { continue; } } // found one if (empty_check) { // only checking and directory is not empty return true; } if (cnodes) { cnodes->push_back(_unescape_path_name(cname)); } else { found = true; } } } catch (...) { // skip the rest } return (cnodes ? (cnodes->size() > 0) : found); } bool UnionfsCstore::write_file(const char *file, const string& data, bool append) { if (data.size() > C_UNIONFS_MAX_FILE_SIZE) { output_internal("write_file too large\n"); return false; } try { // make sure the path exists FsPath ppath(file); ppath.pop(); b_fs::create_directories(ppath.path_cstr()); // write the file ofstream fout; fout.exceptions(ofstream::failbit | ofstream::badbit); ios_base::openmode mflags = ios_base::out; mflags |= ((!append || !path_exists(file)) ? ios_base::trunc : ios_base::app); // truncate or append fout.open(file, mflags); fout << data; fout.close(); } catch (...) { return false; } return true; } bool UnionfsCstore::read_whole_file(const FsPath& fpath, string& data) { /* must exist, be a regular file, and smaller than limit (we're going * to read the whole thing). */ if (!path_exists(fpath) || !path_is_regular(fpath)) { return false; } try { if (b_fs::file_size(fpath.path_cstr()) > C_UNIONFS_MAX_FILE_SIZE) { output_internal("read_whole_file too large\n"); return false; } stringbuf sbuf; ifstream fin(fpath.path_cstr()); fin >> &sbuf; fin.close(); /* note: if file contains just a newline => (eof() && fail()) * so only checking bad() and eof() (we want whole file). */ if (fin.bad() || !fin.eof()) { // read failed return false; } data = sbuf.str(); } catch (...) { return false; } return true; } /* recursively copy source directory to destination. * will throw exception (from b_fs) if fail. */ void UnionfsCstore::recursive_copy_dir(const FsPath& src, const FsPath& dst, bool filter_dot_entries) { string src_str = src.path_cstr(); string dst_str = dst.path_cstr(); b_fs::create_directories(dst.path_cstr()); b_fs::recursive_directory_iterator di(src_str); for (; di != b_fs::recursive_directory_iterator(); ++di) { const char *oname = di->path().file_string().c_str(); string nname = oname; nname.replace(0, src_str.length(), dst_str); if (path_is_directory(oname)) { b_fs::create_directory(nname); } else { if (filter_dot_entries) { string of = di->path().filename(); if (!of.empty() && of.at(0) == '.') { // filter dot files (with exceptions) if (of != C_COMMENT_FILE) { continue; } } } b_fs::copy_file(di->path(), nname); } } } void UnionfsCstore::get_committed_marker(bool is_delete, string& marker) { marker = (is_delete ? "-" : ""); marker += mutable_cfg_path.path_cstr(); } bool UnionfsCstore::find_line_in_file(const FsPath& file, const string& line) { bool ret = false; try { ifstream fin(file.path_cstr()); while (!fin.eof() && !fin.bad() && !fin.fail()) { string in; getline(fin, in); if (in == line) { ret = true; break; } } fin.close(); } catch (...) { ret = false; } return ret; } bool UnionfsCstore::do_mount(const FsPath& rwdir, const FsPath& rdir, const FsPath& mdir) { string mopts = "dirs="; mopts += rwdir.path_cstr(); mopts += "=rw:"; mopts += rdir.path_cstr(); mopts += "=ro"; if (mount("unionfs", mdir.path_cstr(), "unionfs", 0, mopts.c_str()) != 0) { output_internal("union mount failed [%s][%s]\n", strerror(errno), mdir.path_cstr()); return false; } return true; } bool UnionfsCstore::do_umount(const FsPath& mdir) { if (umount(mdir.path_cstr()) != 0) { output_internal("union umount failed [%s][%s]\n", strerror(errno), mdir.path_cstr()); return false; } return true; } bool UnionfsCstore::path_exists(const char *path) { b_fs::file_status result; if (!b_fs_get_file_status(path, result)) { return false; } return b_fs::exists(result); } bool UnionfsCstore::path_is_directory(const char *path) { b_fs::file_status result; if (!b_fs_get_file_status(path, result)) { return false; } return b_fs::is_directory(result); } bool UnionfsCstore::path_is_regular(const char *path) { b_fs::file_status result; if (!b_fs_get_file_status(path, result)) { return false; } return b_fs::is_regular(result); } bool UnionfsCstore::remove_dir_content(const char *path) { if (!path_is_directory(path)) { return false; } b_fs::directory_iterator di(path); for (; di != b_fs::directory_iterator(); ++di) { if (b_fs::remove_all(di->path()) < 1) { return false; } } return true; } } // end namespace unionfs } // end namespace cstore