#include "version.h" #include "base/file_utils.h" #include "base/output_file_requirements.h" #include "caching/commands.h" #include "caching/metadata.h" #include "caching/restore_emitter.h" #include "caching/xml_format.h" #include "persistent-data/file_utils.h" #include #include #include #include #include #include using namespace boost; using namespace caching; using namespace persistent_data; using namespace file_utils; using namespace std; //---------------------------------------------------------------- namespace { unique_ptr create_monitor(bool quiet) { if (!quiet && isatty(fileno(stdout))) return create_progress_bar("Restoring"); else return create_quiet_progress_monitor(); } struct flags { flags() : metadata_version(1), clean_shutdown(true), quiet(false), override_metadata_version(false), override_version(1) { } optional input; optional output; uint32_t metadata_version; bool clean_shutdown; bool quiet; bool override_metadata_version; unsigned override_version; }; void override_version(metadata::ptr md, flags const &fs) { md->sb_.version = fs.override_version; md->commit(fs.clean_shutdown); } int restore(flags const &fs) { bool metadata_touched = false; try { block_manager::ptr bm = open_bm(*fs.output, block_manager::READ_WRITE); check_file_exists(*fs.input); ifstream in(fs.input->c_str(), ifstream::in); metadata_touched = true; metadata::ptr md(new metadata(bm, metadata::CREATE)); emitter::ptr restorer = create_restore_emitter(md, fs.metadata_version, fs.clean_shutdown ? CLEAN_SHUTDOWN : NO_CLEAN_SHUTDOWN); unique_ptr monitor = create_monitor(fs.quiet); parse_xml(in, restorer, get_file_length(*fs.input), *monitor); if (fs.override_metadata_version) override_version(md, fs); } catch (std::exception &e) { if (metadata_touched) zero_superblock(*fs.output); cerr << e.what() << endl; return 1; } return 0; } } //---------------------------------------------------------------- cache_restore_cmd::cache_restore_cmd() : command("cache_restore") { } void cache_restore_cmd::usage(std::ostream &out) const { out << "Usage: " << get_name() << " [options]" << endl << "Options:" << endl << " {-h|--help}\n" << " {-i|--input} \n" << " {-o|--output} \n" << " {-q|--quiet}\n" << " {--metadata-version} <1 or 2>\n" << " {-V|--version}\n" << "\n" << " {--debug-override-metadata-version} " << endl << " {--omit-clean-shutdown}" << endl; } int cache_restore_cmd::run(int argc, char **argv) { int c; flags fs; char const *short_opts = "hi:o:qV"; option const long_opts[] = { { "debug-override-metadata-version", required_argument, NULL, 0 }, { "omit-clean-shutdown", no_argument, NULL, 1 }, { "metadata-version", required_argument, NULL, 2 }, { "help", no_argument, NULL, 'h'}, { "input", required_argument, NULL, 'i' }, { "output", required_argument, NULL, 'o'}, { "quiet", no_argument, NULL, 'q'}, { "version", no_argument, NULL, 'V'}, { NULL, no_argument, NULL, 0 } }; while ((c = getopt_long(argc, argv, short_opts, long_opts, NULL)) != -1) { switch(c) { case 0: fs.override_metadata_version = true; fs.override_version = lexical_cast(optarg); break; case 1: fs.clean_shutdown = false; break; case 2: fs.metadata_version = lexical_cast(optarg); if ((fs.metadata_version < MIN_METADATA_VERSION) || (fs.metadata_version > MAX_METADATA_VERSION)) { cerr << "Bad metadata version\n\n"; return 1; } break; case 'h': usage(cout); return 0; case 'i': fs.input = optional(string(optarg)); break; case 'o': fs.output = optional(string(optarg)); break; case 'q': fs.quiet = true; break; case 'V': cout << THIN_PROVISIONING_TOOLS_VERSION << endl; return 0; default: usage(cerr); return 1; } } if (argc != optind) { usage(cerr); return 1; } if (!fs.input) { cerr << "No input file provided." << endl << endl; usage(cerr); return 1; } if (fs.output) check_output_file_requirements(*fs.output); else { cerr << "No output file provided." << endl << endl; usage(cerr); return 1; } return restore(fs); } //----------------------------------------------------------------