[*_restore] Add progress bar to cache_restore and era_restore.
A lot of refactoring common code between the restore tools.
This commit is contained in:
parent
e31ffe0874
commit
828f654800
4
CHANGES
4
CHANGES
@ -4,9 +4,9 @@ v0.4
|
|||||||
- All tools switch to using libaio. This gives a large performance
|
- All tools switch to using libaio. This gives a large performance
|
||||||
boost, especially to the write focused tools like thin_restore.
|
boost, especially to the write focused tools like thin_restore.
|
||||||
|
|
||||||
- Added a progress monitor to thin_restore
|
- Added a progress monitor to thin_restore, cache_restore and era_restore
|
||||||
|
|
||||||
- Added a --quiet/-q option to thin_restore to turn off the progress bar
|
- Added a --quiet/-q option to *_restore to turn off the progress bar
|
||||||
|
|
||||||
- Removed variable hint size support from cache tools. The kernel
|
- Removed variable hint size support from cache tools. The kernel
|
||||||
still only supports a fixed 32bit width. This will have a side
|
still only supports a fixed 32bit width. This will have a side
|
||||||
|
10
Makefile.in
10
Makefile.in
@ -178,6 +178,7 @@ THIN_CHECK_SOURCE=\
|
|||||||
\
|
\
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
persistent-data/checksum.cc \
|
persistent-data/checksum.cc \
|
||||||
@ -202,6 +203,7 @@ THIN_DELTA_SOURCE=\
|
|||||||
\
|
\
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
persistent-data/checksum.cc \
|
persistent-data/checksum.cc \
|
||||||
@ -269,11 +271,11 @@ thin_restore: $(THIN_RESTORE_OBJECTS) thin-provisioning/thin_restore.o
|
|||||||
|
|
||||||
thin_check: $(THIN_CHECK_OBJECTS) thin-provisioning/thin_check.o
|
thin_check: $(THIN_CHECK_OBJECTS) thin-provisioning/thin_check.o
|
||||||
@echo " [LD] $@"
|
@echo " [LD] $@"
|
||||||
$(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
|
$(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
|
||||||
|
|
||||||
thin_delta: $(THIN_DELTA_OBJECTS) thin-provisioning/thin_delta.o
|
thin_delta: $(THIN_DELTA_OBJECTS) thin-provisioning/thin_delta.o
|
||||||
@echo " [LD] $@"
|
@echo " [LD] $@"
|
||||||
$(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
|
$(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
|
||||||
|
|
||||||
thin_rmap: $(THIN_RMAP_OBJECTS) thin-provisioning/thin_rmap.o
|
thin_rmap: $(THIN_RMAP_OBJECTS) thin-provisioning/thin_rmap.o
|
||||||
@echo " [LD] $@"
|
@echo " [LD] $@"
|
||||||
@ -292,6 +294,7 @@ CACHE_CHECK_SOURCE=\
|
|||||||
base/base64.cc \
|
base/base64.cc \
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
persistent-data/checksum.cc \
|
persistent-data/checksum.cc \
|
||||||
@ -354,6 +357,7 @@ ERA_CHECK_SOURCE=\
|
|||||||
base/base64.cc \
|
base/base64.cc \
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
era/writeset_tree.cc \
|
era/writeset_tree.cc \
|
||||||
@ -421,6 +425,7 @@ ERA_INVALIDATE_SOURCE=\
|
|||||||
base/base64.cc \
|
base/base64.cc \
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
era/writeset_tree.cc \
|
era/writeset_tree.cc \
|
||||||
@ -455,6 +460,7 @@ ERA_RESTORE_SOURCE=\
|
|||||||
base/base64.cc \
|
base/base64.cc \
|
||||||
base/error_state.cc \
|
base/error_state.cc \
|
||||||
base/endian_utils.cc \
|
base/endian_utils.cc \
|
||||||
|
base/progress_monitor.cc \
|
||||||
base/xml_utils.cc \
|
base/xml_utils.cc \
|
||||||
\
|
\
|
||||||
era/writeset_tree.cc \
|
era/writeset_tree.cc \
|
||||||
|
@ -60,16 +60,16 @@ namespace {
|
|||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
base::progress_monitor::ptr
|
std::auto_ptr<base::progress_monitor>
|
||||||
base::create_progress_bar(std::string const &title)
|
base::create_progress_bar(std::string const &title)
|
||||||
{
|
{
|
||||||
return progress_monitor::ptr(new progress_bar(title));
|
return auto_ptr<progress_monitor>(new progress_bar(title));
|
||||||
}
|
}
|
||||||
|
|
||||||
base::progress_monitor::ptr
|
std::auto_ptr<base::progress_monitor>
|
||||||
base::create_quiet_progress_monitor()
|
base::create_quiet_progress_monitor()
|
||||||
{
|
{
|
||||||
return progress_monitor::ptr(new quiet_progress());
|
return auto_ptr<progress_monitor>(new quiet_progress());
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
#define BASE_PROGRESS_MONITOR_H
|
#define BASE_PROGRESS_MONITOR_H
|
||||||
|
|
||||||
#include <boost/shared_ptr.hpp>
|
#include <boost/shared_ptr.hpp>
|
||||||
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
@ -9,15 +10,13 @@
|
|||||||
namespace base {
|
namespace base {
|
||||||
class progress_monitor {
|
class progress_monitor {
|
||||||
public:
|
public:
|
||||||
typedef boost::shared_ptr<progress_monitor> ptr;
|
|
||||||
|
|
||||||
virtual ~progress_monitor() {}
|
virtual ~progress_monitor() {}
|
||||||
|
|
||||||
virtual void update_percent(unsigned) = 0;
|
virtual void update_percent(unsigned) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
progress_monitor::ptr create_progress_bar(std::string const &title);
|
std::auto_ptr<progress_monitor> create_progress_bar(std::string const &title);
|
||||||
progress_monitor::ptr create_quiet_progress_monitor();
|
std::auto_ptr<progress_monitor> create_quiet_progress_monitor();
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -1,5 +1,67 @@
|
|||||||
#include "xml_utils.h"
|
#include "xml_utils.h"
|
||||||
|
|
||||||
|
#include "persistent-data/file_utils.h"
|
||||||
|
#include <fstream>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
using namespace xml_utils;
|
||||||
|
|
||||||
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
|
void
|
||||||
|
xml_parser::parse(std::string const &backup_file, bool quiet)
|
||||||
|
{
|
||||||
|
persistent_data::check_file_exists(backup_file);
|
||||||
|
ifstream in(backup_file.c_str(), ifstream::in);
|
||||||
|
|
||||||
|
std::auto_ptr<base::progress_monitor> monitor = create_monitor(quiet);
|
||||||
|
|
||||||
|
size_t total = 0;
|
||||||
|
size_t input_length = get_file_length(backup_file);
|
||||||
|
|
||||||
|
while (!in.eof()) {
|
||||||
|
char buffer[4096];
|
||||||
|
in.read(buffer, sizeof(buffer));
|
||||||
|
size_t len = in.gcount();
|
||||||
|
int done = in.eof();
|
||||||
|
|
||||||
|
if (!XML_Parse(parser_, buffer, len, done)) {
|
||||||
|
ostringstream out;
|
||||||
|
out << "Parse error at line "
|
||||||
|
<< XML_GetCurrentLineNumber(parser_)
|
||||||
|
<< ":\n"
|
||||||
|
<< XML_ErrorString(XML_GetErrorCode(parser_))
|
||||||
|
<< endl;
|
||||||
|
throw runtime_error(out.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
total += len;
|
||||||
|
monitor->update_percent(total * 100 / input_length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t
|
||||||
|
xml_parser::get_file_length(string const &file) const
|
||||||
|
{
|
||||||
|
struct stat info;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
r = ::stat(file.c_str(), &info);
|
||||||
|
if (r)
|
||||||
|
throw runtime_error("Couldn't stat backup path");
|
||||||
|
|
||||||
|
return info.st_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto_ptr<base::progress_monitor>
|
||||||
|
xml_parser::create_monitor(bool quiet)
|
||||||
|
{
|
||||||
|
if (!quiet && isatty(fileno(stdout)))
|
||||||
|
return base::create_progress_bar("Restoring");
|
||||||
|
else
|
||||||
|
return base::create_quiet_progress_monitor();
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -21,5 +83,4 @@ xml_utils::build_attributes(attributes &a, char const **attr)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
#ifndef BASE_XML_UTILS_H
|
#ifndef BASE_XML_UTILS_H
|
||||||
#define BASE_XML_UTILS_H
|
#define BASE_XML_UTILS_H
|
||||||
|
|
||||||
|
#include <base/progress_monitor.h>
|
||||||
#include <boost/lexical_cast.hpp>
|
#include <boost/lexical_cast.hpp>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
#include <expat.h>
|
#include <expat.h>
|
||||||
|
#include <iosfwd>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
@ -30,7 +32,12 @@ namespace xml_utils {
|
|||||||
return parser_;
|
return parser_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void parse(std::string const &backup_file, bool quiet);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
size_t get_file_length(string const &file) const;
|
||||||
|
auto_ptr<base::progress_monitor> create_monitor(bool quiet);
|
||||||
|
|
||||||
XML_Parser parser_;
|
XML_Parser parser_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -20,11 +20,30 @@ using namespace std;
|
|||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
size_t get_file_length(string const &file) {
|
||||||
|
struct stat info;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
r = ::stat(file.c_str(), &info);
|
||||||
|
if (r)
|
||||||
|
throw runtime_error("Couldn't stat backup path");
|
||||||
|
|
||||||
|
return info.st_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto_ptr<progress_monitor> create_monitor(bool quiet) {
|
||||||
|
if (!quiet && isatty(fileno(stdout)))
|
||||||
|
return create_progress_bar("Restoring");
|
||||||
|
else
|
||||||
|
return create_quiet_progress_monitor();
|
||||||
|
}
|
||||||
|
|
||||||
struct flags {
|
struct flags {
|
||||||
flags()
|
flags()
|
||||||
: metadata_version(1),
|
: metadata_version(1),
|
||||||
override_metadata_version(false),
|
override_metadata_version(false),
|
||||||
clean_shutdown(true) {
|
clean_shutdown(true),
|
||||||
|
quiet(false) {
|
||||||
}
|
}
|
||||||
|
|
||||||
optional<string> input;
|
optional<string> input;
|
||||||
@ -33,6 +52,7 @@ namespace {
|
|||||||
uint32_t metadata_version;
|
uint32_t metadata_version;
|
||||||
bool override_metadata_version;
|
bool override_metadata_version;
|
||||||
bool clean_shutdown;
|
bool clean_shutdown;
|
||||||
|
bool quiet;
|
||||||
};
|
};
|
||||||
|
|
||||||
int restore(flags const &fs) {
|
int restore(flags const &fs) {
|
||||||
@ -48,7 +68,9 @@ namespace {
|
|||||||
|
|
||||||
check_file_exists(*fs.input);
|
check_file_exists(*fs.input);
|
||||||
ifstream in(fs.input->c_str(), ifstream::in);
|
ifstream in(fs.input->c_str(), ifstream::in);
|
||||||
parse_xml(in, restorer);
|
|
||||||
|
auto_ptr<progress_monitor> monitor = create_monitor(fs.quiet);
|
||||||
|
parse_xml(in, restorer, get_file_length(*fs.input), *monitor);
|
||||||
|
|
||||||
} catch (std::exception &e) {
|
} catch (std::exception &e) {
|
||||||
cerr << e.what() << endl;
|
cerr << e.what() << endl;
|
||||||
@ -64,6 +86,7 @@ namespace {
|
|||||||
<< " {-h|--help}" << endl
|
<< " {-h|--help}" << endl
|
||||||
<< " {-i|--input} <input xml file>" << endl
|
<< " {-i|--input} <input xml file>" << endl
|
||||||
<< " {-o|--output} <output device or file>" << endl
|
<< " {-o|--output} <output device or file>" << endl
|
||||||
|
<< " {-q|--quiet}" << endl
|
||||||
<< " {-V|--version}" << endl
|
<< " {-V|--version}" << endl
|
||||||
<< endl
|
<< endl
|
||||||
<< " {--debug-override-metadata-version} <integer>" << endl
|
<< " {--debug-override-metadata-version} <integer>" << endl
|
||||||
@ -77,13 +100,14 @@ int main(int argc, char **argv)
|
|||||||
int c;
|
int c;
|
||||||
flags fs;
|
flags fs;
|
||||||
char const *prog_name = basename(argv[0]);
|
char const *prog_name = basename(argv[0]);
|
||||||
char const *short_opts = "hi:o:V";
|
char const *short_opts = "hi:o:qV";
|
||||||
option const long_opts[] = {
|
option const long_opts[] = {
|
||||||
{ "debug-override-metadata-version", required_argument, NULL, 0 },
|
{ "debug-override-metadata-version", required_argument, NULL, 0 },
|
||||||
{ "omit-clean-shutdown", no_argument, NULL, 1 },
|
{ "omit-clean-shutdown", no_argument, NULL, 1 },
|
||||||
{ "help", no_argument, NULL, 'h'},
|
{ "help", no_argument, NULL, 'h'},
|
||||||
{ "input", required_argument, NULL, 'i' },
|
{ "input", required_argument, NULL, 'i' },
|
||||||
{ "output", required_argument, NULL, 'o'},
|
{ "output", required_argument, NULL, 'o'},
|
||||||
|
{ "quiet", no_argument, NULL, 'q'},
|
||||||
{ "version", no_argument, NULL, 'V'},
|
{ "version", no_argument, NULL, 'V'},
|
||||||
{ NULL, no_argument, NULL, 0 }
|
{ NULL, no_argument, NULL, 0 }
|
||||||
};
|
};
|
||||||
@ -111,6 +135,10 @@ int main(int argc, char **argv)
|
|||||||
fs.output = optional<string>(string(optarg));
|
fs.output = optional<string>(string(optarg));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 'q':
|
||||||
|
fs.quiet = true;
|
||||||
|
break;
|
||||||
|
|
||||||
case 'V':
|
case 'V':
|
||||||
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -236,13 +236,16 @@ caching::create_xml_emitter(ostream &out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
caching::parse_xml(istream &in, emitter::ptr e)
|
caching::parse_xml(istream &in, emitter::ptr e,
|
||||||
|
size_t input_length, base::progress_monitor &monitor)
|
||||||
{
|
{
|
||||||
xml_parser p;
|
xml_parser p;
|
||||||
|
|
||||||
XML_SetUserData(p.get_parser(), e.get());
|
XML_SetUserData(p.get_parser(), e.get());
|
||||||
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
||||||
|
|
||||||
|
size_t total = 0;
|
||||||
|
|
||||||
while (!in.eof()) {
|
while (!in.eof()) {
|
||||||
char buffer[4096];
|
char buffer[4096];
|
||||||
in.read(buffer, sizeof(buffer));
|
in.read(buffer, sizeof(buffer));
|
||||||
@ -258,6 +261,9 @@ caching::parse_xml(istream &in, emitter::ptr e)
|
|||||||
<< endl;
|
<< endl;
|
||||||
throw runtime_error(out.str());
|
throw runtime_error(out.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
total += len;
|
||||||
|
monitor.update_percent(total * 100 / input_length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
#ifndef CACHE_XML_FORMAT_H
|
#ifndef CACHE_XML_FORMAT_H
|
||||||
#define CACHE_XML_FORMAT_H
|
#define CACHE_XML_FORMAT_H
|
||||||
|
|
||||||
|
#include "base/progress_monitor.h"
|
||||||
#include "emitter.h"
|
#include "emitter.h"
|
||||||
|
|
||||||
#include <iosfwd>
|
#include <iosfwd>
|
||||||
@ -9,7 +10,8 @@
|
|||||||
|
|
||||||
namespace caching {
|
namespace caching {
|
||||||
emitter::ptr create_xml_emitter(std::ostream &out);
|
emitter::ptr create_xml_emitter(std::ostream &out);
|
||||||
void parse_xml(std::istream &in, emitter::ptr e);
|
void parse_xml(std::istream &in, emitter::ptr e,
|
||||||
|
size_t input_len, base::progress_monitor &monitor);
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -21,19 +21,22 @@ using namespace std;
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
struct flags {
|
struct flags {
|
||||||
|
flags()
|
||||||
|
: quiet(false) {
|
||||||
|
}
|
||||||
|
|
||||||
optional<string> input;
|
optional<string> input;
|
||||||
optional<string> output;
|
optional<string> output;
|
||||||
|
bool quiet;
|
||||||
};
|
};
|
||||||
|
|
||||||
int restore(flags const &fs) {
|
int restore(flags const &fs, bool quiet) {
|
||||||
try {
|
try {
|
||||||
block_manager<>::ptr bm = open_bm(*fs.output, block_manager<>::READ_WRITE);
|
block_manager<>::ptr bm = open_bm(*fs.output, block_manager<>::READ_WRITE);
|
||||||
metadata::ptr md(new metadata(bm, metadata::CREATE));
|
metadata::ptr md(new metadata(bm, metadata::CREATE));
|
||||||
emitter::ptr restorer = create_restore_emitter(*md);
|
emitter::ptr restorer = create_restore_emitter(*md);
|
||||||
|
|
||||||
check_file_exists(*fs.input);
|
parse_xml(*fs.input, restorer, fs.quiet);
|
||||||
ifstream in(fs.input->c_str(), ifstream::in);
|
|
||||||
parse_xml(in, restorer);
|
|
||||||
|
|
||||||
} catch (std::exception &e) {
|
} catch (std::exception &e) {
|
||||||
cerr << e.what() << endl;
|
cerr << e.what() << endl;
|
||||||
@ -49,6 +52,7 @@ namespace {
|
|||||||
<< " {-h|--help}" << endl
|
<< " {-h|--help}" << endl
|
||||||
<< " {-i|--input} <input xml file>" << endl
|
<< " {-i|--input} <input xml file>" << endl
|
||||||
<< " {-o|--output} <output device or file>" << endl
|
<< " {-o|--output} <output device or file>" << endl
|
||||||
|
<< " {-q|--quiet}" << endl
|
||||||
<< " {-V|--version}" << endl;
|
<< " {-V|--version}" << endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -58,11 +62,12 @@ int main(int argc, char **argv)
|
|||||||
int c;
|
int c;
|
||||||
flags fs;
|
flags fs;
|
||||||
char const *prog_name = basename(argv[0]);
|
char const *prog_name = basename(argv[0]);
|
||||||
char const *short_opts = "hi:o:V";
|
char const *short_opts = "hi:o:qV";
|
||||||
option const long_opts[] = {
|
option const long_opts[] = {
|
||||||
{ "help", no_argument, NULL, 'h'},
|
{ "help", no_argument, NULL, 'h'},
|
||||||
{ "input", required_argument, NULL, 'i' },
|
{ "input", required_argument, NULL, 'i' },
|
||||||
{ "output", required_argument, NULL, 'o'},
|
{ "output", required_argument, NULL, 'o'},
|
||||||
|
{ "quiet", no_argument, NULL, 'q'},
|
||||||
{ "version", no_argument, NULL, 'V'},
|
{ "version", no_argument, NULL, 'V'},
|
||||||
{ NULL, no_argument, NULL, 0 }
|
{ NULL, no_argument, NULL, 0 }
|
||||||
};
|
};
|
||||||
@ -81,6 +86,10 @@ int main(int argc, char **argv)
|
|||||||
fs.output = optional<string>(string(optarg));
|
fs.output = optional<string>(string(optarg));
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case 'q':
|
||||||
|
fs.quiet = true;
|
||||||
|
break;
|
||||||
|
|
||||||
case 'V':
|
case 'V':
|
||||||
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
||||||
return 0;
|
return 0;
|
||||||
@ -108,7 +117,7 @@ int main(int argc, char **argv)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return restore(fs);
|
return restore(fs, fs.quiet);
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -159,29 +159,14 @@ era::create_xml_emitter(std::ostream &out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
era::parse_xml(std::istream &in, emitter::ptr e)
|
era::parse_xml(std::string const &backup_file, emitter::ptr e, bool quiet)
|
||||||
{
|
{
|
||||||
xml_parser p;
|
xml_parser p;
|
||||||
|
|
||||||
XML_SetUserData(p.get_parser(), e.get());
|
XML_SetUserData(p.get_parser(), e.get());
|
||||||
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
||||||
|
|
||||||
while (!in.eof()) {
|
p.parse(backup_file, quiet);
|
||||||
char buffer[4096];
|
|
||||||
in.read(buffer, sizeof(buffer));
|
|
||||||
size_t len = in.gcount();
|
|
||||||
int done = in.eof();
|
|
||||||
|
|
||||||
if (!XML_Parse(p.get_parser(), buffer, len, done)) {
|
|
||||||
ostringstream out;
|
|
||||||
out << "Parse error at line "
|
|
||||||
<< XML_GetCurrentLineNumber(p.get_parser())
|
|
||||||
<< ":\n"
|
|
||||||
<< XML_ErrorString(XML_GetErrorCode(p.get_parser()))
|
|
||||||
<< endl;
|
|
||||||
throw runtime_error(out.str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
#ifndef ERA_XML_FORMAT_H
|
#ifndef ERA_XML_FORMAT_H
|
||||||
#define ERA_XML_FORMAT_H
|
#define ERA_XML_FORMAT_H
|
||||||
|
|
||||||
#include "emitter.h"
|
#include "base/progress_monitor.h"
|
||||||
|
#include "era/emitter.h"
|
||||||
|
|
||||||
#include <iosfwd>
|
#include <iosfwd>
|
||||||
|
|
||||||
@ -9,7 +10,7 @@
|
|||||||
|
|
||||||
namespace era {
|
namespace era {
|
||||||
emitter::ptr create_xml_emitter(std::ostream &out);
|
emitter::ptr create_xml_emitter(std::ostream &out);
|
||||||
void parse_xml(std::istream &in, emitter::ptr e);
|
void parse_xml(std::string const &backup_file, emitter::ptr e, bool quiet);
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -18,6 +18,7 @@ Feature: thin_restore
|
|||||||
{-h|--help}
|
{-h|--help}
|
||||||
{-i|--input} <input xml file>
|
{-i|--input} <input xml file>
|
||||||
{-o|--output} <output device or file>
|
{-o|--output} <output device or file>
|
||||||
|
{-q|--quiet}
|
||||||
{-V|--version}
|
{-V|--version}
|
||||||
|
|
||||||
{--debug-override-metadata-version} <integer>
|
{--debug-override-metadata-version} <integer>
|
||||||
@ -36,6 +37,7 @@ Feature: thin_restore
|
|||||||
{-h|--help}
|
{-h|--help}
|
||||||
{-i|--input} <input xml file>
|
{-i|--input} <input xml file>
|
||||||
{-o|--output} <output device or file>
|
{-o|--output} <output device or file>
|
||||||
|
{-q|--quiet}
|
||||||
{-V|--version}
|
{-V|--version}
|
||||||
|
|
||||||
{--debug-override-metadata-version} <integer>
|
{--debug-override-metadata-version} <integer>
|
||||||
@ -80,3 +82,18 @@ Feature: thin_restore
|
|||||||
And an empty dev file
|
And an empty dev file
|
||||||
When I run cache_restore with -i metadata.xml -o metadata.bin --omit-clean-shutdown
|
When I run cache_restore with -i metadata.xml -o metadata.bin --omit-clean-shutdown
|
||||||
Then it should pass
|
Then it should pass
|
||||||
|
|
||||||
|
Scenario: --quiet is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin --quiet
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
Scenario: -q is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin -q
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
@ -18,6 +18,7 @@ Feature: thin_restore
|
|||||||
{-h|--help}
|
{-h|--help}
|
||||||
{-i|--input} <input xml file>
|
{-i|--input} <input xml file>
|
||||||
{-o|--output} <output device or file>
|
{-o|--output} <output device or file>
|
||||||
|
{-q|--quiet}
|
||||||
{-V|--version}
|
{-V|--version}
|
||||||
|
|
||||||
"""
|
"""
|
||||||
@ -33,6 +34,7 @@ Feature: thin_restore
|
|||||||
{-h|--help}
|
{-h|--help}
|
||||||
{-i|--input} <input xml file>
|
{-i|--input} <input xml file>
|
||||||
{-o|--output} <output device or file>
|
{-o|--output} <output device or file>
|
||||||
|
{-q|--quiet}
|
||||||
{-V|--version}
|
{-V|--version}
|
||||||
|
|
||||||
"""
|
"""
|
||||||
@ -63,4 +65,18 @@ Feature: thin_restore
|
|||||||
When I run era_restore with -i metadata.xml -o metadata.bin
|
When I run era_restore with -i metadata.xml -o metadata.bin
|
||||||
Then it should pass
|
Then it should pass
|
||||||
And the metadata should be valid
|
And the metadata should be valid
|
||||||
|
|
||||||
|
Scenario: --quiet is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin --quiet
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
Scenario: -q is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin -q
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
@ -55,6 +55,20 @@ Feature: thin_restore
|
|||||||
No output file provided.
|
No output file provided.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
Scenario: --quiet is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin --quiet
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
Scenario: -q is accepted
|
||||||
|
Given valid metadata
|
||||||
|
When I run thin_restore with -i metadata.xml -o metadata.bin -q
|
||||||
|
Then it should pass with:
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
Scenario: dump/restore is a noop
|
Scenario: dump/restore is a noop
|
||||||
Given valid metadata
|
Given valid metadata
|
||||||
When I dump
|
When I dump
|
||||||
|
@ -41,35 +41,13 @@ using namespace thin_provisioning;
|
|||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
size_t get_file_length(string const &file) {
|
|
||||||
struct stat info;
|
|
||||||
int r;
|
|
||||||
|
|
||||||
r = ::stat(file.c_str(), &info);
|
|
||||||
if (r)
|
|
||||||
throw runtime_error("Couldn't stat backup path");
|
|
||||||
|
|
||||||
return info.st_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
progress_monitor::ptr create_monitor(bool quiet) {
|
|
||||||
if (!quiet && isatty(fileno(stdout)))
|
|
||||||
return create_progress_bar("Restoring");
|
|
||||||
else
|
|
||||||
return create_quiet_progress_monitor();
|
|
||||||
}
|
|
||||||
|
|
||||||
int restore(string const &backup_file, string const &dev, bool quiet) {
|
int restore(string const &backup_file, string const &dev, bool quiet) {
|
||||||
try {
|
try {
|
||||||
// The block size gets updated by the restorer.
|
// The block size gets updated by the restorer.
|
||||||
metadata::ptr md(new metadata(dev, metadata::CREATE, 128, 0));
|
metadata::ptr md(new metadata(dev, metadata::CREATE, 128, 0));
|
||||||
emitter::ptr restorer = create_restore_emitter(md);
|
emitter::ptr restorer = create_restore_emitter(md);
|
||||||
|
|
||||||
check_file_exists(backup_file);
|
parse_xml(backup_file, restorer, quiet);
|
||||||
ifstream in(backup_file.c_str(), ifstream::in);
|
|
||||||
|
|
||||||
progress_monitor::ptr monitor = create_monitor(quiet);
|
|
||||||
parse_xml(in, restorer, get_file_length(backup_file), monitor);
|
|
||||||
|
|
||||||
} catch (std::exception &e) {
|
} catch (std::exception &e) {
|
||||||
cerr << e.what() << endl;
|
cerr << e.what() << endl;
|
||||||
|
@ -217,35 +217,14 @@ tp::create_xml_emitter(ostream &out)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
tp::parse_xml(std::istream &in, emitter::ptr e,
|
tp::parse_xml(std::string const &backup_file, emitter::ptr e, bool quiet)
|
||||||
size_t input_length, base::progress_monitor::ptr monitor)
|
|
||||||
{
|
{
|
||||||
xml_parser p;
|
xml_parser p;
|
||||||
|
|
||||||
XML_SetUserData(p.get_parser(), e.get());
|
XML_SetUserData(p.get_parser(), e.get());
|
||||||
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
XML_SetElementHandler(p.get_parser(), start_tag, end_tag);
|
||||||
|
|
||||||
size_t total = 0;
|
p.parse(backup_file, quiet);
|
||||||
|
|
||||||
while (!in.eof()) {
|
|
||||||
char buffer[1024 * 1024];
|
|
||||||
in.read(buffer, sizeof(buffer));
|
|
||||||
size_t len = in.gcount();
|
|
||||||
int done = in.eof();
|
|
||||||
|
|
||||||
if (!XML_Parse(p.get_parser(), buffer, len, done)) {
|
|
||||||
ostringstream out;
|
|
||||||
out << "Parse error at line "
|
|
||||||
<< XML_GetCurrentLineNumber(p.get_parser())
|
|
||||||
<< ":\n"
|
|
||||||
<< XML_ErrorString(XML_GetErrorCode(p.get_parser()))
|
|
||||||
<< endl;
|
|
||||||
throw runtime_error(out.str());
|
|
||||||
}
|
|
||||||
|
|
||||||
total += len;
|
|
||||||
monitor->update_percent(total * 100 / input_length);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
@ -28,8 +28,7 @@
|
|||||||
|
|
||||||
namespace thin_provisioning {
|
namespace thin_provisioning {
|
||||||
emitter::ptr create_xml_emitter(std::ostream &out);
|
emitter::ptr create_xml_emitter(std::ostream &out);
|
||||||
void parse_xml(std::istream &in, emitter::ptr e,
|
void parse_xml(std::string const &backup_file, emitter::ptr e, bool quiet);
|
||||||
size_t input_length, base::progress_monitor::ptr p);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------
|
//----------------------------------------------------------------
|
||||||
|
Loading…
Reference in New Issue
Block a user