/* * 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 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 = "/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 = "/tmp/.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"; ////// 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; } 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); 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 string mopts = "dirs="; mopts += change_root.path_cstr(); mopts += "=rw:"; mopts += active_root.path_cstr(); mopts += "=ro"; if (mount("unionfs", work_root.path_cstr(), "unionfs", 0, mopts.c_str()) != 0) { output_internal("setup session mount failed [%s][%s]\n", strerror(errno), work_root.path_cstr()); 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 (umount(wstr.c_str()) != 0) { output_internal("teardown session umount failed [%s][%s]\n", strerror(errno), wstr.c_str()); 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)); } ////// 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); } /* XXX currently "committed marking" is done inside commit. * TODO move "committed marking" out of commit and into low-level * implementation (here). */ /* return whether current "cfg path" has been committed, i.e., whether * the set or delete operation on the path has been processed by commit. * is_set: whether the operation is set (for sanity check as there can * be only one operation on the path). */ bool UnionfsCstore::marked_committed(const tr1::shared_ptr& def, bool is_set) { FsPath cpath = mutable_cfg_path; string com_str = cpath.path_cstr(); com_str += "/"; if (def->isLeafValue()) { // path includes leaf value. construct the right string. string val; cpath.pop(val); val = _unescape_path_name(val); /* XXX current commit implementation escapes value strings for * single-value nodes but not for multi-value nodes for some * reason. the following match current behavior. */ if (!def->isMulti()) { val = _escape_path_name(val); } com_str = cpath.path_cstr(); com_str += "/value:"; com_str += val; } com_str = (is_set ? "+ " : "- ") + com_str; return committed_marker_exists(com_str); } bool UnionfsCstore::validate_val_impl(const tr1::shared_ptr& def, char *value) { /* XXX filesystem paths/accesses are completely embedded in var ref lib. * for now, treat the lib as a unionfs-specific implementation. * generalizing it will need a rewrite. * set the handle to be used during validate_value() for var ref * processing. this is a global var in cli_new.c. */ var_ref_handle = (void *) this; bool ret = validate_value(def->getDef(), value); var_ref_handle = NULL; return ret; } 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(); } } 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); } void UnionfsCstore::get_all_child_dir_names(const FsPath& root, vector& nodes) { if (!path_exists(root) || !path_is_directory(root)) { // not a valid root. nop. return; } try { b_fs::directory_iterator di(root.path_cstr()); for (; di != b_fs::directory_iterator(); ++di) { // must be directory if (!path_is_directory(di->path().file_string().c_str())) { continue; } // name cannot start with "." string cname = di->path().filename(); if (cname.length() < 1 || cname[0] == '.') { continue; } // found one nodes.push_back(_unescape_path_name(cname)); } } catch (...) { return; } } bool UnionfsCstore::write_file(const FsPath& file, const string& data) { 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); fout.open(file.path_cstr(), ios_base::out | ios_base::trunc); 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; } /* return whether specified "commited marker" exists in the * "committed marker file". */ bool UnionfsCstore::committed_marker_exists(const string& marker) { bool ret = false; try { ifstream fin(C_COMMITTED_MARKER_FILE.c_str()); while (!fin.eof() && !fin.bad() && !fin.fail()) { string line; getline(fin, line); if (line == marker) { ret = true; break; } } fin.close(); } catch (...) { ret = false; } return ret; } /* 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) { string src_str = src.path_cstr(); string dst_str = dst.path_cstr(); b_fs::create_directory(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 { b_fs::copy_file(oname, nname); } } } 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); } } // end namespace unionfs } // end namespace cstore