/* * 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 . */ #ifndef _CSTORE_UNIONFS_H_ #define _CSTORE_UNIONFS_H_ #include #include #include #include #include #include #include #include #include namespace b_fs = boost::filesystem; class UnionfsCstore : public Cstore { public: UnionfsCstore(bool use_edit_level = false); UnionfsCstore(const string& session_id, string& env); virtual ~UnionfsCstore(); ////// public virtual functions declared in base class bool markSessionUnsaved(); bool unmarkSessionUnsaved(); bool sessionUnsaved(); bool sessionChanged(); bool setupSession(); bool teardownSession(); bool inSession(); private: // constants static const string C_ENV_TMPL_ROOT; static const string C_ENV_WORK_ROOT; static const string C_ENV_ACTIVE_ROOT; static const string C_ENV_CHANGE_ROOT; static const string C_ENV_TMP_ROOT; static const string C_DEF_TMPL_ROOT; static const string C_DEF_CFG_ROOT; static const string C_DEF_ACTIVE_ROOT; static const string C_DEF_CHANGE_PREFIX; static const string C_DEF_WORK_PREFIX; static const string C_DEF_TMP_PREFIX; static const string C_MARKER_DEF_VALUE; static const string C_MARKER_DEACTIVATE; static const string C_MARKER_CHANGED; static const string C_MARKER_UNSAVED; static const string C_COMMITTED_MARKER_FILE; static const string C_COMMENT_FILE; static const string C_TAG_NAME; static const string C_VAL_NAME; static const string C_DEF_NAME; /* max size for a file. * currently this includes value file and comment file. */ static const size_t C_UNIONFS_MAX_FILE_SIZE = 262144; // root dirs (constant) b_fs::path work_root; // working root (union) b_fs::path active_root; // active root (readonly part of union) b_fs::path change_root; // change root (r/w part of union) b_fs::path tmp_root; // temp root b_fs::path tmpl_root; // template root // path buffers b_fs::path mutable_cfg_path; // mutable part of config path b_fs::path tmpl_path; // whole template path Cstore::MapT > saved_paths; // saved mutable part of cfg path and whole template path ////// virtual functions defined in base class // begin path modifiers void push_tmpl_path(const string& new_comp) { push_path(tmpl_path, new_comp); }; void push_tmpl_path_tag() { push_tmpl_path(C_TAG_NAME); }; string pop_tmpl_path() { return pop_path(tmpl_path); }; void push_cfg_path(const string& new_comp) { push_path(mutable_cfg_path, new_comp); }; void push_cfg_path_val() { push_cfg_path(C_VAL_NAME); }; string pop_cfg_path() { return pop_path(mutable_cfg_path); }; void append_cfg_path(const vector& path_comps) { for (size_t i = 0; i < path_comps.size(); i++) { push_cfg_path(path_comps[i]); } }; void reset_paths() { tmpl_path = tmpl_root; mutable_cfg_path = ""; }; void save_paths(const void *handle = NULL) { pair p; p.first = mutable_cfg_path; p.second = tmpl_path; saved_paths[handle] = p; }; void restore_paths(const void *handle = NULL) { Cstore::MapT >::iterator it = saved_paths.find(handle); if (it == saved_paths.end()) { exit_internal("restore_paths: handle not found\n"); } pair p = saved_paths[handle]; mutable_cfg_path = p.first; tmpl_path = p.second; }; bool cfg_path_at_root() { return (!mutable_cfg_path.has_parent_path()); }; bool tmpl_path_at_root() { return (tmpl_path.file_string() == tmpl_root.file_string()); }; // end path modifiers // these operate on current tmpl path bool tmpl_node_exists(); bool tmpl_parse(vtw_def& def); // these operate on current work path bool add_node(); bool remove_node(); void get_all_child_node_names_impl(vector& cnodes, bool active_cfg); void get_all_tmpl_child_node_names(vector& cnodes) { get_all_child_dir_names(tmpl_path, cnodes); }; bool write_value_vec(const vector& vvec, bool active_cfg); bool rename_child_node(const string& oname, const string& nname); bool copy_child_node(const string& oname, const string& nname); bool mark_display_default(); bool unmark_display_default(); bool mark_deactivated(); bool unmark_deactivated(); bool unmark_deactivated_descendants(); bool mark_changed_with_ancestors(); bool unmark_changed_with_descendants(); bool remove_comment(); bool set_comment(const string& comment); bool discard_changes(unsigned long long& num_removed); // observers for work path bool cfg_node_changed(); // observers for work path or active path bool cfg_node_exists(bool active_cfg); bool read_value_vec(vector& vvec, bool active_cfg); bool marked_deactivated(bool active_cfg); bool get_comment(string& comment, bool active_cfg); bool marked_display_default(bool active_cfg); // observers during commit operation bool marked_committed(const vtw_def& def, bool is_set); // these operate on both current tmpl and work paths bool validate_val_impl(vtw_def *def, char *value); // observers for "edit/tmpl levels" (for "edit"-related operations). // note that these should be moved to base class in the future. string get_edit_level_path() { return cfg_path_to_str(); }; string get_tmpl_level_path() { return tmpl_path_to_str(); }; void get_edit_level(vector& path_comps); bool edit_level_at_root() { return cfg_path_at_root(); }; // for testing/debugging string cfg_path_to_str(); string tmpl_path_to_str(); ////// private functions b_fs::path get_work_path() { return (work_root / mutable_cfg_path); }; b_fs::path get_active_path() { return (active_root / mutable_cfg_path); }; b_fs::path get_change_path() { return (change_root / mutable_cfg_path); }; void push_path(b_fs::path& old_path, const string& new_comp); string pop_path(b_fs::path& path); void get_all_child_dir_names(b_fs::path root, vector& nodes); bool write_file(const string& file, const string& data); bool create_file(const string& file) { return write_file(file, ""); }; bool read_whole_file(const b_fs::path& file, string& data); bool committed_marker_exists(const string& marker); void recursive_copy_dir(const b_fs::path& src, const b_fs::path& dst); // boost fs operations wrappers bool b_fs_exists(const b_fs::path& path) { try { return b_fs::exists(path); } catch (...) { return false; } }; bool b_fs_is_directory(const b_fs::path& path) { try { return b_fs::is_directory(path); } catch (...) { return false; } }; bool b_fs_is_regular(const b_fs::path& path) { try { return b_fs::is_regular(path); } catch (...) { return false; } }; }; #endif /* _CSTORE_UNIONFS_H_ */