mirror of
https://github.com/OpenNebula/one.git
synced 2025-01-18 06:03:39 +03:00
F #5516: Incremental backups for qcow2 disk images
- Adds new configuration attribute MODE: FULL or INCREMENTAL for BACKUP_CONFIG. FULL backups uses a differen backup image. - INCREMENTAL backup information is together with the backup image. Example: <BACKUP_INCREMENTS> <INCREMENT> <DATE><![CDATA[1667770552]]></DATE> <ID><![CDATA[0]]></ID> <PARENT_ID><![CDATA[-1]]></PARENT_ID> <SIZE><![CDATA[172]]></SIZE> <SOURCE><![CDATA[bb828060]]></SOURCE> <TYPE><![CDATA[FULL]]></TYPE> </INCREMENT> <INCREMENT> <DATE><![CDATA[1667770604]]></DATE> <ID><![CDATA[1]]></ID> <PARENT_ID><![CDATA[0]]></PARENT_ID> <SIZE><![CDATA[1]]></SIZE> <SOURCE><![CDATA[ca0de5f6]]></SOURCE> <TYPE><![CDATA[INCREMENT]]></TYPE> </INCREMENT> <INCREMENT> <DATE><![CDATA[1667770700]]></DATE> <ID><![CDATA[2]]></ID> <PARENT_ID><![CDATA[1]]></PARENT_ID> <SIZE><![CDATA[39]]></SIZE> <SOURCE><![CDATA[e9897d6a]]></SOURCE> <TYPE><![CDATA[INCREMENT]]></TYPE> </INCREMENT> </BACKUP_INCREMENTS> This information only appears on incremental backups - Internal BACKUP_CONFIG data includes information about the current active backup and the last increment id. - Backup operation includes a new parameter: reset. This "closes" the current active incremental chain and creates a new FULL backup. - Backup drivers store backups with increment index (0 = FULL) e.g. disk.0.0. - Incremental backups are only allowed for VMs using all disks in qcow2 format. - Backup configuration cannot be changed while doing a VM backup. - Downloader strings includes backup chains <inc_id>:<backup_ref>,... - Restic downloader has been updated to support backup chains. Disk images are rebased across increments.
This commit is contained in:
parent
f9b077fb55
commit
0a46e21129
194
include/BackupIncrements.h
Normal file
194
include/BackupIncrements.h
Normal file
@ -0,0 +1,194 @@
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* Copyright 2002-2022, OpenNebula Project, OpenNebula Systems */
|
||||
/* */
|
||||
/* Licensed under the Apache License, Version 2.0 (the "License"); you may */
|
||||
/* not use this file except in compliance with the License. You may obtain */
|
||||
/* a copy of the License at */
|
||||
/* */
|
||||
/* http://www.apache.org/licenses/LICENSE-2.0 */
|
||||
/* */
|
||||
/* Unless required by applicable law or agreed to in writing, software */
|
||||
/* distributed under the License is distributed on an "AS IS" BASIS, */
|
||||
/* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
|
||||
/* See the License for the specific language governing permissions and */
|
||||
/* limitations under the License. */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
#ifndef BACKUPS_INCREMENTS_H_
|
||||
#define BACKUPS_INCREMENTS_H_
|
||||
|
||||
#include <string>
|
||||
|
||||
#include "ExtendedAttribute.h"
|
||||
#include "NebulaUtil.h"
|
||||
|
||||
/**
|
||||
* The Image INCREMENT attribute
|
||||
*/
|
||||
class Increment : public ExtendedAttribute
|
||||
{
|
||||
public:
|
||||
Increment(VectorAttribute *va, int id): ExtendedAttribute(va, id){};
|
||||
|
||||
~Increment() = default;
|
||||
|
||||
enum Type
|
||||
{
|
||||
FULL = 0, /** < Full backup*/
|
||||
INCREMENT = 1, /** < Forward increment */
|
||||
};
|
||||
|
||||
long long size() const
|
||||
{
|
||||
long long sz = 0;
|
||||
|
||||
vector_value("SIZE", sz);
|
||||
|
||||
return sz;
|
||||
}
|
||||
|
||||
int id() const
|
||||
{
|
||||
return get_id();
|
||||
}
|
||||
|
||||
Type type() const
|
||||
{
|
||||
std::string stype = vector_value("TYPE");
|
||||
|
||||
one_util::toupper(stype);
|
||||
|
||||
if ( stype == "FULL" )
|
||||
{
|
||||
return FULL;
|
||||
}
|
||||
else if ( stype == "INCREMENT" )
|
||||
{
|
||||
return INCREMENT;
|
||||
}
|
||||
|
||||
return FULL;
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Set of INCREMENTS (indexed by ID)
|
||||
*/
|
||||
class IncrementSet : public ExtendedAttributeSet
|
||||
{
|
||||
public:
|
||||
IncrementSet(): ExtendedAttributeSet(false){};
|
||||
|
||||
~IncrementSet() = default;
|
||||
|
||||
void init(Template * tmpl)
|
||||
{
|
||||
std::vector<VectorAttribute *> incs;
|
||||
|
||||
tmpl->get("INCREMENT", incs);
|
||||
|
||||
init_attribute_map("ID", incs);
|
||||
};
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Increment interface */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
VectorAttribute * new_increment(std::string source, long long sz,
|
||||
Increment::Type type);
|
||||
|
||||
Increment * last_increment()
|
||||
{
|
||||
return static_cast<Increment *>(last_attribute());
|
||||
}
|
||||
|
||||
long long total_size();
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Iterators */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
class IncIterator : public AttributeIterator
|
||||
{
|
||||
public:
|
||||
IncIterator():AttributeIterator(){};
|
||||
IncIterator(const AttributeIterator& dit):AttributeIterator(dit){};
|
||||
|
||||
virtual ~IncIterator(){};
|
||||
|
||||
Increment * operator*() const
|
||||
{
|
||||
return static_cast<Increment *>(map_it->second);
|
||||
}
|
||||
};
|
||||
|
||||
IncIterator begin()
|
||||
{
|
||||
IncIterator it(ExtendedAttributeSet::begin());
|
||||
return it;
|
||||
}
|
||||
|
||||
IncIterator end()
|
||||
{
|
||||
IncIterator it(ExtendedAttributeSet::end());
|
||||
return it;
|
||||
}
|
||||
|
||||
typedef class IncIterator inc_iterator;
|
||||
|
||||
private:
|
||||
ExtendedAttribute * attribute_factory(VectorAttribute * va, int id) const
|
||||
{
|
||||
return new Increment(va, id);
|
||||
};
|
||||
};
|
||||
|
||||
/**
|
||||
* This class represents a generic set of references (links) for Image objects
|
||||
* The data model is as follows
|
||||
* <BACKUP_INCREMENTS>
|
||||
* <INCREMENT>
|
||||
* <ID> Unique ID within this backup increment
|
||||
* <TYPE> Of the backup FULL | INCREMENT
|
||||
* <PARENT_ID> ID of the parent increment (backing file)
|
||||
* <SOURCE> Reference in the backup system
|
||||
* <SIZE> Size of this increment
|
||||
* <DATE> When this backup was taken (epoch)
|
||||
*/
|
||||
class BackupIncrements
|
||||
{
|
||||
public:
|
||||
BackupIncrements():_template(false,'=',"BACKUP_INCREMENTS"){};
|
||||
|
||||
~BackupIncrements() = default;
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* XML from/to methods for DB persistency */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
int from_xml_node(const xmlNodePtr node);
|
||||
|
||||
std::string& to_xml(std::string& xml) const
|
||||
{
|
||||
return _template.to_xml(xml);
|
||||
}
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Increments interface */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
int add_increment(std::string source, long long size, Increment::Type type);
|
||||
|
||||
int last_increment_id();
|
||||
|
||||
long long total_size()
|
||||
{
|
||||
return increments.total_size();
|
||||
}
|
||||
|
||||
private:
|
||||
/**
|
||||
* Text representation of the increments
|
||||
*/
|
||||
Template _template;
|
||||
|
||||
IncrementSet increments;
|
||||
};
|
||||
|
||||
#endif /*BACKUPS_H_*/
|
@ -34,6 +34,7 @@ class ObjectXML;
|
||||
* <KEEP_LAST> Just keep the last N backups
|
||||
* <BACKUP_VOLATILE> Backup volatile disks or not
|
||||
* <FS_FREEZE> FS freeze operation to perform on the VM
|
||||
* <MODE> Backup mode
|
||||
* <LAST_DATASTORE_ID> The dastore ID used to store the active backups(*)
|
||||
* <LAST_BACKUP_ID> ID of the active backup(*)
|
||||
* <LAST_BACKUP_SIZE> SIZE of the active backup(*)
|
||||
@ -55,6 +56,43 @@ public:
|
||||
|
||||
~Backups() = default;
|
||||
|
||||
// *************************************************************************
|
||||
// Backup modes
|
||||
// *************************************************************************
|
||||
enum Mode
|
||||
{
|
||||
FULL = 0, /** < Full backups */
|
||||
INCREMENT = 1, /** < Forward increments */
|
||||
};
|
||||
|
||||
static std::string mode_to_str(Mode bm)
|
||||
{
|
||||
switch (bm)
|
||||
{
|
||||
case FULL: return "FULL";
|
||||
case INCREMENT: return "INCREMENT";
|
||||
default: return "";
|
||||
}
|
||||
};
|
||||
|
||||
static Mode str_to_mode(std::string& str_mode)
|
||||
{
|
||||
Mode mode = FULL;
|
||||
|
||||
one_util::toupper(str_mode);
|
||||
|
||||
if ( str_mode == "FULL" )
|
||||
{
|
||||
mode = FULL;
|
||||
}
|
||||
else if ( str_mode == "INCREMENT" )
|
||||
{
|
||||
mode = INCREMENT;
|
||||
}
|
||||
|
||||
return mode;
|
||||
};
|
||||
|
||||
// *************************************************************************
|
||||
// Inititalization functions
|
||||
// *************************************************************************
|
||||
@ -78,20 +116,39 @@ public:
|
||||
* - BACKUP_VOLATILE
|
||||
* - KEEP_LAST
|
||||
* - FS_FREEZE
|
||||
* - MODE
|
||||
*
|
||||
* The following attributes are stored in the configuration and refers
|
||||
* only to the active backup operation
|
||||
* - LAST_DATASTORE_ID
|
||||
* - LAST_BACKUP_ID
|
||||
* - LAST_BACKUP_SIZE
|
||||
*
|
||||
* Incremental backups include a reference to the last increment and full
|
||||
* backup:
|
||||
* - LAST_INCREMENT_ID
|
||||
* - INCREMENTAL_BACKUP_ID
|
||||
*/
|
||||
int parse(std::string& error_str, Template *tmpl);
|
||||
int parse(Template *tmpl, bool can_increment, std::string& error_str);
|
||||
|
||||
/**
|
||||
* @return true if Backup includes configuration attributes
|
||||
*/
|
||||
bool configured()
|
||||
{
|
||||
return config.empty();
|
||||
}
|
||||
|
||||
/**
|
||||
* @return true if the backup needs to include volatile disks
|
||||
*/
|
||||
bool do_volatile() const;
|
||||
|
||||
/**
|
||||
* @return true if the backup needs to include volatile disks
|
||||
*/
|
||||
Mode mode() const;
|
||||
|
||||
/**
|
||||
* Set of functions to manipulate the LAST_* attributes referring to
|
||||
* the active backup operation
|
||||
@ -111,6 +168,16 @@ public:
|
||||
config.replace("LAST_BACKUP_SIZE", size);
|
||||
}
|
||||
|
||||
void last_increment_id(int id)
|
||||
{
|
||||
config.replace("LAST_INCREMENT_ID", id);
|
||||
}
|
||||
|
||||
void incremental_backup_id(int id)
|
||||
{
|
||||
config.replace("INCREMENTAL_BACKUP_ID", id);
|
||||
}
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
|
||||
int last_datastore_id() const
|
||||
@ -140,6 +207,23 @@ public:
|
||||
return sz;
|
||||
}
|
||||
|
||||
int last_increment_id() const
|
||||
{
|
||||
int id;
|
||||
|
||||
config.get("LAST_INCREMENT_ID", id);
|
||||
|
||||
return id;
|
||||
}
|
||||
|
||||
int incremental_backup_id() const
|
||||
{
|
||||
int id;
|
||||
|
||||
config.get("INCREMENTAL_BACKUP_ID", id);
|
||||
|
||||
return id;
|
||||
}
|
||||
/* ---------------------------------------------------------------------- */
|
||||
|
||||
void last_backup_clear()
|
||||
|
@ -465,7 +465,7 @@ public:
|
||||
*
|
||||
* @return 0 on success, -1 otherwise
|
||||
*/
|
||||
int backup(int vid, int bck_ds_id,
|
||||
int backup(int vid, int bck_ds_id, bool reset,
|
||||
const RequestAttributes& ra, std::string& error_str);
|
||||
|
||||
//--------------------------------------------------------------------------
|
||||
|
@ -171,10 +171,15 @@ protected:
|
||||
/* Method to access attributes */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/**
|
||||
* @return attribute by id or 0 if not found
|
||||
* @return attribute by id or nullptr if not found
|
||||
*/
|
||||
ExtendedAttribute * get_attribute(int id) const;
|
||||
|
||||
/**
|
||||
* @return last_attribute or nullptr if empty set
|
||||
*/
|
||||
ExtendedAttribute * last_attribute() const;
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Iterators */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
|
@ -21,6 +21,7 @@
|
||||
#include "ImageTemplate.h"
|
||||
#include "ObjectCollection.h"
|
||||
#include "Snapshots.h"
|
||||
#include "BackupIncrements.h"
|
||||
|
||||
class VirtualMachineDisk;
|
||||
|
||||
@ -607,6 +608,28 @@ public:
|
||||
target_snapshot = -1;
|
||||
};
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Incremental backups interface */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
int add_increment(std::string source, long long size, Increment::Type type)
|
||||
{
|
||||
int rc = increments.add_increment(source, size, type);
|
||||
|
||||
if ( rc == -1 )
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
size_mb = increments.total_size();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int last_increment_id()
|
||||
{
|
||||
return increments.last_increment_id();
|
||||
}
|
||||
|
||||
private:
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
@ -717,6 +740,12 @@ private:
|
||||
*/
|
||||
Snapshots snapshots;
|
||||
|
||||
/**
|
||||
* List of backup increments (only relevant for BACKUP images, of type
|
||||
* incremental)
|
||||
*/
|
||||
BackupIncrements increments;
|
||||
|
||||
/**
|
||||
* ID of the snapshot being processed (if any)
|
||||
*/
|
||||
|
@ -634,7 +634,7 @@ public:
|
||||
VirtualMachineBackup():
|
||||
RequestManagerVirtualMachine("one.vm.backup",
|
||||
"Creates a new backup image for the virtual machine",
|
||||
"A:sii")
|
||||
"A:siib")
|
||||
{
|
||||
vm_action = VMActions::BACKUP_ACTION;
|
||||
}
|
||||
|
@ -1736,9 +1736,9 @@ public:
|
||||
/**
|
||||
*
|
||||
*/
|
||||
void max_backup_size(Template &ds_quota)
|
||||
long long backup_size(Template &ds_quota)
|
||||
{
|
||||
disks.backup_size(ds_quota, _backups.do_volatile());
|
||||
return disks.backup_size(ds_quota, _backups.do_volatile());
|
||||
}
|
||||
|
||||
Backups& backups()
|
||||
|
@ -506,6 +506,19 @@ public:
|
||||
*/
|
||||
void get_image_ids(std::set<int>& ids, int uid);
|
||||
|
||||
/**
|
||||
* Marshall disks in XML format with just essential information
|
||||
* @param xml string to write the disk XML description
|
||||
*/
|
||||
std::string& to_xml_short(std::string& xml);
|
||||
|
||||
/**
|
||||
* Check if a tm_mad is valid for each Virtual Machine Disk and set
|
||||
* clone_target and ln_target
|
||||
* @param tm_mad is the tm_mad for system datastore chosen
|
||||
*/
|
||||
int check_tm_mad(const std::string& tm_mad, std::string& error);
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* Image Manager Interface */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
@ -783,24 +796,20 @@ public:
|
||||
/* ---------------------------------------------------------------------- */
|
||||
/* BACKUP interface */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
|
||||
/** Returns upper limit of the disk size needed to do a VM backup
|
||||
/**
|
||||
* Returns upper limit of the disk size needed to do a VM backup
|
||||
* @param ds_quota The Datastore quota
|
||||
* @param do_volatile consider volatile disks to compute size
|
||||
*/
|
||||
void backup_size(Template &ds_quota, bool do_volatile);
|
||||
long long backup_size(Template &ds_quota, bool do_volatile);
|
||||
|
||||
/**
|
||||
* Marshall disks in XML format with just essential information
|
||||
* @param xml string to write the disk XML description
|
||||
* Returns true if all disks support incremental backups. This requires
|
||||
* QCOW2 format.
|
||||
*
|
||||
* @param do_volatile consider volatile disks for incremental backups
|
||||
*/
|
||||
std::string& to_xml_short(std::string& xml);
|
||||
|
||||
/**
|
||||
* Check if a tm_mad is valid for each Virtual Machine Disk and set
|
||||
* clone_target and ln_target
|
||||
* @param tm_mad is the tm_mad for system datastore chosen
|
||||
*/
|
||||
int check_tm_mad(const std::string& tm_mad, std::string& error);
|
||||
bool backup_increment(bool do_volatile);
|
||||
|
||||
protected:
|
||||
|
||||
|
@ -1885,6 +1885,7 @@ TM_FILES="src/tm_mad/tm_common.sh"
|
||||
|
||||
TM_LIB_FILES="src/tm_mad/lib/kvm.rb \
|
||||
src/tm_mad/lib/tm_action.rb \
|
||||
src/tm_mad/lib/backup_qcow2.rb \
|
||||
src/tm_mad/lib/backup.rb"
|
||||
|
||||
TM_SHARED_FILES="src/tm_mad/shared/clone \
|
||||
|
@ -353,7 +353,19 @@ class OneImageHelper < OpenNebulaHelper::OneHelper
|
||||
return unless vms
|
||||
|
||||
if image.type_str.casecmp('backup').zero?
|
||||
puts format(str, 'BACKUP OF VM', vms[0])
|
||||
CLIHelper.print_header(str_h1 % 'BACKUP INFORMATION', false)
|
||||
puts format(str, 'VM', vms[0])
|
||||
|
||||
if image.has_elements?('/IMAGE/BACKUP_INCREMENTS/INCREMENT')
|
||||
puts format(str, 'TYPE', 'INCREMENTAL')
|
||||
|
||||
puts
|
||||
|
||||
CLIHelper.print_header('BACKUP INCREMENTS', false)
|
||||
format_backup_increments(image)
|
||||
else
|
||||
puts format(str, 'TYPE', 'FULL')
|
||||
end
|
||||
else
|
||||
puts
|
||||
CLIHelper.print_header('VIRTUAL MACHINES', false)
|
||||
@ -418,6 +430,50 @@ class OneImageHelper < OpenNebulaHelper::OneHelper
|
||||
table.show(image_snapshots)
|
||||
end
|
||||
|
||||
def format_backup_increments(image)
|
||||
table=CLIHelper::ShowTable.new(nil, self) do
|
||||
column :ID, 'Increment ID', :size=>3 do |d|
|
||||
d['ID']
|
||||
end
|
||||
|
||||
column :PID, 'Parent increment ID', :size=>3 do |d|
|
||||
d['PARENT_ID']
|
||||
end
|
||||
|
||||
column :TYPE, 'T', :size=>1 do |d|
|
||||
d['TYPE'][0]
|
||||
end
|
||||
|
||||
column :SIZE, '', :left, :size=>8 do |d|
|
||||
if d['SIZE']
|
||||
OpenNebulaHelper.unit_to_str(
|
||||
d['SIZE'].to_i,
|
||||
{},
|
||||
'M'
|
||||
)
|
||||
else
|
||||
'-'
|
||||
end
|
||||
end
|
||||
|
||||
column :DATE, 'Creation date', :size=>15 do |d|
|
||||
OpenNebulaHelper.time_to_str(d['DATE'])
|
||||
end
|
||||
|
||||
column :SOURCE, 'Backup source', :left, :size=>37 do |d|
|
||||
d['SOURCE']
|
||||
end
|
||||
|
||||
|
||||
default :ID, :PID, :TYPE, :SIZE, :DATE, :SOURCE
|
||||
end
|
||||
|
||||
ihash = image.to_hash
|
||||
increments = [ihash['IMAGE']['BACKUP_INCREMENTS']['INCREMENT']].flatten
|
||||
|
||||
table.show(increments)
|
||||
end
|
||||
|
||||
class << self
|
||||
|
||||
def create_image_variables(options, name)
|
||||
|
@ -251,6 +251,12 @@ CommandParser::CmdParser.new(ARGV) do
|
||||
:description => 'Select PCI device by its class ID'
|
||||
}
|
||||
|
||||
RESET_BACKUP = {
|
||||
:name => 'reset',
|
||||
:large => '--reset',
|
||||
:description => 'Creates a new backup image, from a new full backup (only for incremental)'
|
||||
}
|
||||
|
||||
OpenNebulaHelper::TEMPLATE_OPTIONS_VM.delete_if do |v|
|
||||
%w[as_gid as_uid].include?(v[:name])
|
||||
end
|
||||
@ -1520,7 +1526,8 @@ CommandParser::CmdParser.new(ARGV) do
|
||||
command :backup,
|
||||
backup_vm_desc,
|
||||
:vmid,
|
||||
:options => [OneDatastoreHelper::DATASTORE,
|
||||
:options => [RESET_BACKUP,
|
||||
OneDatastoreHelper::DATASTORE,
|
||||
OneVMHelper::SCHEDULE,
|
||||
OneVMHelper::WEEKLY,
|
||||
OneVMHelper::MONTHLY,
|
||||
@ -1533,6 +1540,8 @@ CommandParser::CmdParser.new(ARGV) do
|
||||
exit(-1)
|
||||
end
|
||||
|
||||
reset = options[:reset] == true
|
||||
|
||||
if !options[:schedule].nil?
|
||||
options[:args] = options[:datastore]
|
||||
|
||||
@ -1540,7 +1549,7 @@ CommandParser::CmdParser.new(ARGV) do
|
||||
else
|
||||
|
||||
helper.perform_action(args[0], options, 'Backup') do |vm|
|
||||
rc = vm.backup(options[:datastore])
|
||||
rc = vm.backup(options[:datastore], reset)
|
||||
|
||||
if OpenNebula.is_error?(rc)
|
||||
STDERR.puts "Error creating VM backup: #{rc.message}"
|
||||
|
@ -47,6 +47,18 @@ ExtendedAttribute * ExtendedAttributeSet::get_attribute(int id) const
|
||||
return it->second;
|
||||
}
|
||||
|
||||
ExtendedAttribute * ExtendedAttributeSet::last_attribute() const
|
||||
{
|
||||
auto it = a_set.rbegin();
|
||||
|
||||
if ( it == a_set.rend() )
|
||||
{
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
return it->second;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
|
@ -183,7 +183,8 @@ function restic_env
|
||||
/DATASTORE/TEMPLATE/RESTIC_IONICE \
|
||||
/DATASTORE/TEMPLATE/RESTIC_NICE \
|
||||
/DATASTORE/TEMPLATE/RESTIC_BWLIMIT \
|
||||
/DATASTORE/TEMPLATE/RESTIC_CONNECTIONS)
|
||||
/DATASTORE/TEMPLATE/RESTIC_CONNECTIONS \
|
||||
/DATASTORE/TEMPLATE/RESTIC_TMP_DIR)
|
||||
|
||||
SFTP_SERVER="${XPATH_ELEMENTS[j++]}"
|
||||
SFTP_USER="${XPATH_ELEMENTS[j++]:-oneadmin}"
|
||||
@ -193,6 +194,7 @@ function restic_env
|
||||
NICE="${XPATH_ELEMENTS[j++]}"
|
||||
BWLIMIT="${XPATH_ELEMENTS[j++]}"
|
||||
CONNECTIONS="${XPATH_ELEMENTS[j++]}"
|
||||
TMP_DIR="${XPATH_ELEMENTS[j++]}"
|
||||
|
||||
export RESTIC_REPOSITORY="sftp:${SFTP_USER}@${SFTP_SERVER}:${BASE_PATH}"
|
||||
export RESTIC_PASSWORD="${PASSWORD}"
|
||||
@ -203,6 +205,12 @@ function restic_env
|
||||
RESTIC_ONE_PRECMD="nice -n ${NICE} "
|
||||
fi
|
||||
|
||||
if [ -z "${TMP_DIR}" ]; then
|
||||
TMP_DIR="/var/tmp/"
|
||||
fi
|
||||
|
||||
export RESTIC_TMP_DIR="${TMP_DIR}/`uuidgen`"
|
||||
|
||||
if [ -n "${IONICE}" ]; then
|
||||
RESTIC_ONE_PRECMD="${RESTIC_ONE_PRECMD}ionice -c2 -n ${IONICE} "
|
||||
fi
|
||||
@ -532,7 +540,7 @@ docker://*|dockerfile://*)
|
||||
command="$VAR_LOCATION/remotes/datastore/docker_downloader.sh \"$FROM\""
|
||||
;;
|
||||
restic://*)
|
||||
#pseudo restic url restic://<datastore_id>/<snapshot_id>/<file_name>
|
||||
#pseudo restic url restic://<datastore_id>/<id>:<snapshot_id>,.../<file_name>
|
||||
restic_path=${FROM#restic://}
|
||||
d_id=`echo ${restic_path} | cut -d'/' -f1`
|
||||
s_id=`echo ${restic_path} | cut -d'/' -f2`
|
||||
@ -545,7 +553,37 @@ restic://*)
|
||||
exit -1
|
||||
fi
|
||||
|
||||
command="${RESTIC_ONE_CMD} dump -q ${s_id} /${file}"
|
||||
incs=(${s_id//,/ })
|
||||
|
||||
mkdir -p ${RESTIC_TMP_DIR}
|
||||
|
||||
pushd ${RESTIC_TMP_DIR}
|
||||
|
||||
for i in "${incs[@]}"; do
|
||||
inc_id=`echo $i | cut -d':' -f1`
|
||||
snap_id=`echo $i | cut -d':' -f2`
|
||||
|
||||
${RESTIC_ONE_CMD} dump -q ${snap_id} /${file}.${inc_id} > disk.${inc_id}
|
||||
done
|
||||
|
||||
for i in `ls disk* | sort -r`; do
|
||||
id=`echo $i | cut -d'.' -f2`
|
||||
pid=$((id - 1))
|
||||
|
||||
if [ -f "disk.${pid}" ]; then
|
||||
qemu-img rebase -u -F qcow2 -b "disk.${pid}" "disk.${id}"
|
||||
else
|
||||
qemu-img rebase -u -b '' "disk.${id}"
|
||||
fi
|
||||
done
|
||||
|
||||
qemu-img convert -O qcow2 -m 4 `ls disk* | sort -r | head -1` disk.qcow2
|
||||
|
||||
command="cat `realpath disk.qcow2`"
|
||||
|
||||
clean_command="rm -rf ${RESTIC_TMP_DIR}"
|
||||
|
||||
popd
|
||||
;;
|
||||
rsync://*)
|
||||
# rsync://<ds_id>/<vm_id>/<backup_id>/<file>
|
||||
@ -628,3 +666,8 @@ fi
|
||||
if [ "$TO" != "-" ]; then
|
||||
unarchive "$TO"
|
||||
fi
|
||||
|
||||
# Perform any clean operation
|
||||
if [ -n "${clean_command}" ]; then
|
||||
eval "$clean_command"
|
||||
fi
|
||||
|
@ -2536,7 +2536,7 @@ int DispatchManager::detach_sg(int vid, int nicid, int sgid,
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
int DispatchManager::backup(int vid, int backup_ds_id,
|
||||
int DispatchManager::backup(int vid, int backup_ds_id, bool reset,
|
||||
const RequestAttributes& ra, string& error_str)
|
||||
{
|
||||
ostringstream oss;
|
||||
@ -2587,6 +2587,12 @@ int DispatchManager::backup(int vid, int backup_ds_id,
|
||||
|
||||
vm->backups().last_datastore_id(backup_ds_id);
|
||||
|
||||
if (reset)
|
||||
{
|
||||
vm->backups().last_increment_id(-1);
|
||||
vm->backups().incremental_backup_id(-1);
|
||||
}
|
||||
|
||||
vmm->trigger_backup(vid);
|
||||
|
||||
vm->set_resched(false);
|
||||
|
130
src/image/BackupIncrements.cc
Normal file
130
src/image/BackupIncrements.cc
Normal file
@ -0,0 +1,130 @@
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* Copyright 2002-2022, OpenNebula Project, OpenNebula Systems */
|
||||
/* */
|
||||
/* Licensed under the Apache License, Version 2.0 (the "License"); you may */
|
||||
/* not use this file except in compliance with the License. You may obtain */
|
||||
/* a copy of the License at */
|
||||
/* */
|
||||
/* http://www.apache.org/licenses/LICENSE-2.0 */
|
||||
/* */
|
||||
/* Unless required by applicable law or agreed to in writing, software */
|
||||
/* distributed under the License is distributed on an "AS IS" BASIS, */
|
||||
/* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */
|
||||
/* See the License for the specific language governing permissions and */
|
||||
/* limitations under the License. */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
#include "BackupIncrements.h"
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
int BackupIncrements::from_xml_node(const xmlNodePtr node)
|
||||
{
|
||||
int rc = _template.from_xml_node(node);
|
||||
|
||||
if (rc == -1)
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
increments.init(&_template);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
int BackupIncrements::add_increment(std::string source, long long size,
|
||||
Increment::Type type)
|
||||
{
|
||||
VectorAttribute * va = increments.new_increment(source, size, type);
|
||||
|
||||
if (va == nullptr)
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
_template.set(va);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
int BackupIncrements::last_increment_id()
|
||||
{
|
||||
Increment * li = increments.last_increment();
|
||||
|
||||
if ( li == nullptr )
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
return li->id();
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
VectorAttribute * IncrementSet::new_increment(std::string source, long long sz,
|
||||
Increment::Type type)
|
||||
{
|
||||
Increment * li = last_increment();
|
||||
|
||||
int parent = -1;
|
||||
|
||||
if ( li == nullptr )
|
||||
{
|
||||
if ( type != Increment::FULL )
|
||||
{
|
||||
return nullptr;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
parent = li->id();
|
||||
}
|
||||
|
||||
int iid = parent + 1;
|
||||
|
||||
VectorAttribute * va = new VectorAttribute("INCREMENT");
|
||||
|
||||
switch (type)
|
||||
{
|
||||
case Increment::FULL:
|
||||
va->replace("TYPE", "FULL");
|
||||
break;
|
||||
case Increment::INCREMENT:
|
||||
va->replace("TYPE", "INCREMENT");
|
||||
break;
|
||||
}
|
||||
|
||||
va->replace("DATE", time(0));
|
||||
|
||||
va->replace("SOURCE", source);
|
||||
|
||||
va->replace("SIZE", sz);
|
||||
|
||||
va->replace("PARENT_ID", parent);
|
||||
|
||||
va->replace("ID", iid);
|
||||
|
||||
add_attribute(attribute_factory(va, iid), iid);
|
||||
|
||||
return va;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
long long IncrementSet::total_size()
|
||||
{
|
||||
long long sz = 0;
|
||||
|
||||
for (inc_iterator i = begin(); i != end(); ++i)
|
||||
{
|
||||
sz += (*i)->size();
|
||||
}
|
||||
|
||||
return sz;
|
||||
}
|
@ -390,14 +390,16 @@ int Image::bootstrap(SqlDB * db)
|
||||
|
||||
string& Image::to_xml(string& xml) const
|
||||
{
|
||||
string template_xml;
|
||||
string perms_xml;
|
||||
ostringstream oss;
|
||||
string vm_collection_xml;
|
||||
string clone_collection_xml;
|
||||
string app_clone_collection_xml;
|
||||
string snapshots_xml;
|
||||
string lock_str;
|
||||
string template_xml;
|
||||
string perms_xml;
|
||||
string increments_xml;
|
||||
|
||||
ostringstream oss;
|
||||
|
||||
oss <<
|
||||
"<IMAGE>" <<
|
||||
@ -431,6 +433,7 @@ string& Image::to_xml(string& xml) const
|
||||
app_clone_collection.to_xml(app_clone_collection_xml) <<
|
||||
obj_template->to_xml(template_xml) <<
|
||||
snapshots.to_xml(snapshots_xml) <<
|
||||
increments.to_xml(increments_xml) <<
|
||||
"</IMAGE>";
|
||||
|
||||
xml = oss.str();
|
||||
@ -525,6 +528,16 @@ int Image::from_xml(const string& xml)
|
||||
content.clear();
|
||||
}
|
||||
|
||||
ObjectXML::get_nodes("/IMAGE/BACKUP_INCREMENTS", content);
|
||||
|
||||
if (!content.empty())
|
||||
{
|
||||
rc += increments.from_xml_node(content[0]);
|
||||
|
||||
ObjectXML::free_nodes(content);
|
||||
content.clear();
|
||||
}
|
||||
|
||||
if (rc != 0)
|
||||
{
|
||||
return -1;
|
||||
|
@ -27,7 +27,8 @@ source_files=[
|
||||
'ImageManagerProtocol.cc',
|
||||
'ImageManager.cc',
|
||||
'ImageManagerActions.cc',
|
||||
'ImageTemplate.cc'
|
||||
'ImageTemplate.cc',
|
||||
'BackupIncrements.cc'
|
||||
]
|
||||
|
||||
# Build library
|
||||
|
@ -16,6 +16,7 @@
|
||||
#include <time.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#include "Nebula.h"
|
||||
#include "LifeCycleManager.h"
|
||||
#include "TransferManager.h"
|
||||
#include "DispatchManager.h"
|
||||
@ -1276,24 +1277,32 @@ void LifeCycleManager::trigger_attach_success(int vid)
|
||||
return;
|
||||
}
|
||||
|
||||
if ( vm->get_lcm_state() == VirtualMachine::HOTPLUG )
|
||||
{
|
||||
vm->clear_attach_disk();
|
||||
|
||||
vm->set_state(VirtualMachine::RUNNING);
|
||||
|
||||
vmpool->update(vm.get());
|
||||
vmpool->update_search(vm.get());
|
||||
}
|
||||
else if ( vm->get_lcm_state() == VirtualMachine::HOTPLUG_PROLOG_POWEROFF )
|
||||
if ( vm->get_lcm_state() == VirtualMachine::HOTPLUG ||
|
||||
vm->get_lcm_state() == VirtualMachine::HOTPLUG_PROLOG_POWEROFF )
|
||||
{
|
||||
vm->log("LCM", Log::INFO, "VM Disk successfully attached.");
|
||||
|
||||
vm->clear_attach_disk();
|
||||
|
||||
// Reset incremental backup pointer. After attach a new full backup
|
||||
// is needed.
|
||||
if ( vm->backups().configured() )
|
||||
{
|
||||
vm->backups().last_increment_id(-1);
|
||||
vm->backups().incremental_backup_id(-1);
|
||||
}
|
||||
|
||||
if ( vm->get_lcm_state() == VirtualMachine::HOTPLUG )
|
||||
{
|
||||
vm->set_state(VirtualMachine::RUNNING);
|
||||
}
|
||||
else
|
||||
{
|
||||
dm->trigger_poweroff_success(vid);
|
||||
}
|
||||
|
||||
vmpool->update(vm.get());
|
||||
vmpool->update_search(vm.get());
|
||||
|
||||
dm->trigger_poweroff_success(vid);
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -2638,67 +2647,27 @@ void LifeCycleManager::trigger_resize_failure(int vid)
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
void LifeCycleManager::trigger_backup_success(int vid)
|
||||
|
||||
static int create_backup_image(VirtualMachine * vm, string& msg)
|
||||
{
|
||||
trigger([this, vid] {
|
||||
auto vm = vmpool->get(vid);
|
||||
Nebula& nd = Nebula::instance();
|
||||
|
||||
if ( vm == nullptr )
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
time_t the_time;
|
||||
char tbuffer[80];
|
||||
|
||||
ostringstream oss;
|
||||
|
||||
string ds_name, ds_mad, ds_data;
|
||||
Datastore::DatastoreType ds_type;
|
||||
Image::DiskType ds_dtype;
|
||||
auto ipool = nd.get_ipool();
|
||||
auto dspool = nd.get_dspool();
|
||||
|
||||
int i_id;
|
||||
string error_str;
|
||||
|
||||
// Store quota values
|
||||
Template ds_deltas;
|
||||
long long backup_size = 0;
|
||||
|
||||
int vm_uid = vm->get_uid();
|
||||
int vm_gid = vm->get_gid();
|
||||
|
||||
auto& backups = vm->backups();
|
||||
|
||||
vm->max_backup_size(ds_deltas);
|
||||
|
||||
ds_deltas.add("DATASTORE", backups.last_datastore_id());
|
||||
|
||||
one_util::str_cast(backups.last_backup_size(), backup_size);
|
||||
|
||||
switch(vm->get_lcm_state())
|
||||
{
|
||||
case VirtualMachine::BACKUP:
|
||||
vm->set_state(VirtualMachine::RUNNING);
|
||||
break;
|
||||
|
||||
case VirtualMachine::BACKUP_POWEROFF:
|
||||
vm->set_state(VirtualMachine::POWEROFF);
|
||||
vm->set_state(VirtualMachine::LCM_INIT);
|
||||
break;
|
||||
|
||||
default:
|
||||
vm->log("LCM",Log::ERROR,"backup_success, VM in a wrong state");
|
||||
|
||||
vm.reset();
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Get datastore backup information */
|
||||
/* ------------------------------------------------------------------ */
|
||||
string ds_name, ds_mad, ds_data;
|
||||
|
||||
Datastore::DatastoreType ds_type;
|
||||
Image::DiskType ds_dtype;
|
||||
|
||||
int ds_id = backups.last_datastore_id();
|
||||
|
||||
if (auto ds = dspool->get_ro(ds_id))
|
||||
@ -2712,23 +2681,22 @@ void LifeCycleManager::trigger_backup_success(int vid)
|
||||
}
|
||||
else
|
||||
{
|
||||
vm->log("LCM", Log::ERROR, "backup_success, "
|
||||
"backup datastore does not exist");
|
||||
msg = "backup datastore does not exist";
|
||||
|
||||
vmpool->update(vm.get());
|
||||
|
||||
vm.reset();
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
|
||||
return;
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Create Image for the backup snapshot, add it to the VM */
|
||||
/* Create Image for the backup snapshot */
|
||||
/* ------------------------------------------------------------------ */
|
||||
time_t the_time;
|
||||
char tbuffer[80];
|
||||
|
||||
struct tm * tinfo;
|
||||
ostringstream oss;
|
||||
|
||||
time(&the_time);
|
||||
struct tm * tinfo = localtime(&the_time);
|
||||
tinfo = localtime(&the_time);
|
||||
|
||||
strftime(tbuffer, 80, "%d-%b %H.%M.%S", tinfo); //18-Jun 08.30.15
|
||||
|
||||
@ -2743,41 +2711,164 @@ void LifeCycleManager::trigger_backup_success(int vid)
|
||||
itmp->add("VM_ID", vm->get_oid());
|
||||
itmp->add("TYPE", Image::type_to_str(Image::BACKUP));
|
||||
|
||||
int rc = ipool->allocate( vm->get_uid(), vm->get_gid(), vm->get_uname(),
|
||||
vm->get_gname(), 0177, move(itmp), ds_id, ds_name, ds_dtype,
|
||||
ds_data, ds_type, ds_mad, "-", "", -1, &i_id, error_str);
|
||||
|
||||
int rc = ipool->allocate(vm->get_uid(),
|
||||
vm->get_gid(),
|
||||
vm->get_uname(),
|
||||
vm->get_gname(),
|
||||
0177,
|
||||
move(itmp),
|
||||
ds_id,
|
||||
ds_name,
|
||||
ds_dtype,
|
||||
ds_data,
|
||||
ds_type,
|
||||
ds_mad,
|
||||
"-",
|
||||
"",
|
||||
-1,
|
||||
&i_id,
|
||||
error_str);
|
||||
if ( rc < 0 )
|
||||
{
|
||||
vm->log("LCM",Log::ERROR,"backup_success, "
|
||||
"backup image allocate error: " + error_str);
|
||||
msg = "backup image allocate error: " + error_str;
|
||||
|
||||
vmpool->update(vm.get());
|
||||
return -1;
|
||||
}
|
||||
|
||||
vm.reset();
|
||||
return i_id;
|
||||
}
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
void LifeCycleManager::trigger_backup_success(int vid)
|
||||
{
|
||||
trigger([this, vid] {
|
||||
auto vm = vmpool->get(vid);
|
||||
|
||||
if ( vm == nullptr )
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
std::set<int> iids;
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* State update and Quota values (worst case) */
|
||||
/* ------------------------------------------------------------------ */
|
||||
string error;
|
||||
|
||||
backups.add(i_id);
|
||||
Template ds_deltas;
|
||||
|
||||
backups.remove_last(iids);
|
||||
int vm_uid = vm->get_uid();
|
||||
int vm_gid = vm->get_gid();
|
||||
|
||||
auto& backups = vm->backups();
|
||||
|
||||
int ds_id = backups.last_datastore_id();
|
||||
|
||||
int incremental_id = backups.incremental_backup_id();
|
||||
Backups::Mode mode = backups.mode();
|
||||
|
||||
long long reserved_sz = vm->backup_size(ds_deltas);
|
||||
long long real_sz = 0;
|
||||
|
||||
one_util::str_cast(backups.last_backup_size(), real_sz);
|
||||
|
||||
ds_deltas.add("DATASTORE", backups.last_datastore_id());
|
||||
|
||||
if (mode == Backups::FULL || incremental_id == -1)
|
||||
{
|
||||
ds_deltas.add("IMAGES", 1);
|
||||
}
|
||||
|
||||
switch(vm->get_lcm_state())
|
||||
{
|
||||
case VirtualMachine::BACKUP:
|
||||
vm->set_state(VirtualMachine::RUNNING);
|
||||
break;
|
||||
|
||||
case VirtualMachine::BACKUP_POWEROFF:
|
||||
vm->set_state(VirtualMachine::POWEROFF);
|
||||
vm->set_state(VirtualMachine::LCM_INIT);
|
||||
break;
|
||||
|
||||
default:
|
||||
vm->log("LCM", Log::ERROR, "backup_success, VM in wrong state");
|
||||
vm.reset();
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
return;
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Create Backup image if needed */
|
||||
/* ------------------------------------------------------------------ */
|
||||
int image_id = -1;
|
||||
|
||||
std::set<int> delete_ids;
|
||||
|
||||
if (mode == Backups::FULL || incremental_id == -1)
|
||||
{
|
||||
image_id = create_backup_image(vm.get(), error);
|
||||
|
||||
if ( image_id == -1 )
|
||||
{
|
||||
goto error_image_create;
|
||||
}
|
||||
|
||||
backups.add(image_id);
|
||||
|
||||
backups.remove_last(delete_ids);
|
||||
}
|
||||
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Update backup information for increments */
|
||||
/* ------------------------------------------------------------------ */
|
||||
if (mode == Backups::INCREMENT)
|
||||
{
|
||||
Increment::Type itype;
|
||||
|
||||
if (incremental_id == -1)
|
||||
{
|
||||
backups.incremental_backup_id(image_id);
|
||||
itype = Increment::FULL;
|
||||
}
|
||||
else
|
||||
{
|
||||
image_id = backups.incremental_backup_id();
|
||||
itype = Increment::INCREMENT;
|
||||
}
|
||||
|
||||
if (auto image = ipool->get(image_id))
|
||||
{
|
||||
int rc = image->add_increment(backups.last_backup_id(), real_sz, itype);
|
||||
|
||||
if ( rc == -1)
|
||||
{
|
||||
error = "cannot add increment to backup image";
|
||||
goto error_increment_update;
|
||||
}
|
||||
|
||||
backups.last_increment_id(image->last_increment_id());
|
||||
|
||||
ipool->update(image.get());
|
||||
}
|
||||
else
|
||||
{
|
||||
error = "backup image does not exists";
|
||||
goto error_increment_update;
|
||||
}
|
||||
}
|
||||
|
||||
backups.last_backup_clear();
|
||||
|
||||
vmpool->update(vm.get());
|
||||
|
||||
if ( iids.size() > 0 )
|
||||
if ( delete_ids.size() > 0 )
|
||||
{
|
||||
oss.str("");
|
||||
ostringstream oss;
|
||||
|
||||
oss << "Removing backup snapshots:";
|
||||
|
||||
for (int i : iids)
|
||||
for (int i : delete_ids)
|
||||
{
|
||||
oss << " " << i;
|
||||
}
|
||||
@ -2790,32 +2881,54 @@ void LifeCycleManager::trigger_backup_success(int vid)
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Add image to the datastore and forget keep_last backups */
|
||||
/* ------------------------------------------------------------------ */
|
||||
if (mode == Backups::FULL || incremental_id == -1)
|
||||
{
|
||||
if ( auto ds = dspool->get(ds_id) )
|
||||
{
|
||||
ds->add_image(i_id);
|
||||
ds->add_image(image_id);
|
||||
|
||||
dspool->update(ds.get());
|
||||
}
|
||||
|
||||
for (int i : iids)
|
||||
for (int i : delete_ids)
|
||||
{
|
||||
if ( imagem->delete_image(i, error_str) != 0 )
|
||||
if ( imagem->delete_image(i, error) != 0 )
|
||||
{
|
||||
oss.str("");
|
||||
ostringstream oss;
|
||||
|
||||
oss << "backup_success, cannot remove VM backup " << i
|
||||
<< " : " << error_str;
|
||||
<< " : " << error;
|
||||
|
||||
NebulaLog::error("LCM", oss.str());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Update quotas, count real size of the backup
|
||||
long long reserved_size{0};
|
||||
|
||||
ds_deltas.get("SIZE", reserved_size);
|
||||
ds_deltas.replace("SIZE", reserved_size - backup_size);
|
||||
ds_deltas.add("IMAGES", 0);
|
||||
/* ------------------------------------------------------------------ */
|
||||
/* Update quotas, count real size of the backup */
|
||||
/* ------------------------------------------------------------------ */
|
||||
ds_deltas.replace("SIZE", reserved_sz - real_sz);
|
||||
ds_deltas.replace("IMAGES", 0);
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
|
||||
return;
|
||||
|
||||
error_increment_update:
|
||||
if (incremental_id == -1)
|
||||
{
|
||||
backups.incremental_backup_id(-1);
|
||||
backups.del(image_id);
|
||||
}
|
||||
|
||||
error_image_create:
|
||||
vm->log("LCM", Log::ERROR, "backup_success, " + error);
|
||||
vmpool->update(vm.get());
|
||||
|
||||
vm.reset();
|
||||
|
||||
Quotas::ds_del(vm_uid, vm_gid, &ds_deltas);
|
||||
return;
|
||||
});
|
||||
}
|
||||
|
||||
@ -2849,9 +2962,17 @@ void LifeCycleManager::trigger_backup_failure(int vid)
|
||||
vm_uid = vm->get_uid();
|
||||
vm_gid = vm->get_gid();
|
||||
|
||||
vm->max_backup_size(ds_deltas);
|
||||
int incremental_id = vm->backups().incremental_backup_id();
|
||||
Backups::Mode mode = vm->backups().mode();
|
||||
|
||||
vm->backup_size(ds_deltas);
|
||||
ds_deltas.add("DATASTORE", vm->backups().last_datastore_id());
|
||||
|
||||
if (mode == Backups::FULL || incremental_id == -1)
|
||||
{
|
||||
ds_deltas.add("IMAGES", 1);
|
||||
}
|
||||
|
||||
vm->backups().last_backup_clear();
|
||||
|
||||
vmpool->update(vm.get());
|
||||
|
@ -790,8 +790,8 @@ module OpenNebula
|
||||
# @param ds_id [Integer] Id of the datastore to save the backup
|
||||
# @return [Integer, OpenNebula::Error] ID of the resulting BACKUP image
|
||||
# in case of success, Error otherwise.
|
||||
def backup(ds_id)
|
||||
return @client.call(VM_METHODS[:backup], @pe_id, ds_id)
|
||||
def backup(ds_id, reset = false)
|
||||
return @client.call(VM_METHODS[:backup], @pe_id, ds_id, reset)
|
||||
end
|
||||
|
||||
########################################################################
|
||||
|
@ -3394,7 +3394,7 @@ void VirtualMachineUpdateConf::request_execute(
|
||||
}
|
||||
}
|
||||
|
||||
rc = vm->updateconf(uc_tmpl.get(), att.resp_msg, update_type == 1 ? true : false);
|
||||
rc = vm->updateconf(uc_tmpl.get(), att.resp_msg, update_type == 1);
|
||||
|
||||
// rc = -1 (error), 0 (context changed), 1 (no context change)
|
||||
if ( rc == -1 )
|
||||
@ -3822,6 +3822,9 @@ void VirtualMachineBackup::request_execute(
|
||||
PoolObjectAuth ds_perms;
|
||||
Template quota_tmpl;
|
||||
|
||||
Backups::Mode mode;
|
||||
int li_id;
|
||||
|
||||
ostringstream oss;
|
||||
|
||||
// ------------------------------------------------------------------------
|
||||
@ -3829,6 +3832,12 @@ void VirtualMachineBackup::request_execute(
|
||||
// ------------------------------------------------------------------------
|
||||
int vm_id = xmlrpc_c::value_int(paramList.getInt(1));
|
||||
int backup_ds_id = xmlrpc_c::value_int(paramList.getInt(2));
|
||||
bool reset = false;
|
||||
|
||||
if ( paramList.size() > 3 )
|
||||
{
|
||||
reset = xmlrpc_c::value_boolean(paramList.getBoolean(3));
|
||||
}
|
||||
|
||||
// ------------------------------------------------------------------------
|
||||
// Get VM & Backup Information
|
||||
@ -3837,7 +3846,10 @@ void VirtualMachineBackup::request_execute(
|
||||
{
|
||||
vm->get_permissions(vm_perms);
|
||||
|
||||
vm->max_backup_size(quota_tmpl);
|
||||
vm->backup_size(quota_tmpl);
|
||||
|
||||
mode = vm->backups().mode();
|
||||
li_id = vm->backups().last_increment_id();
|
||||
}
|
||||
else
|
||||
{
|
||||
@ -3885,7 +3897,11 @@ void VirtualMachineBackup::request_execute(
|
||||
// after backup success notification from driver
|
||||
// -------------------------------------------------------------------------
|
||||
quota_tmpl.add("DATASTORE", backup_ds_id);
|
||||
|
||||
if ( mode == Backups::FULL || li_id == -1 || reset )
|
||||
{
|
||||
quota_tmpl.add("IMAGES", 1);
|
||||
}
|
||||
|
||||
RequestAttributes att_quota(vm_perms.uid, vm_perms.gid, att);
|
||||
|
||||
@ -3898,7 +3914,7 @@ void VirtualMachineBackup::request_execute(
|
||||
// ------------------------------------------------------------------------
|
||||
// Create backup
|
||||
// ------------------------------------------------------------------------
|
||||
if (dm->backup(vm_id, backup_ds_id, att, att.resp_msg) < 0)
|
||||
if (dm->backup(vm_id, backup_ds_id, reset, att, att.resp_msg) < 0)
|
||||
{
|
||||
quota_rollback("a_tmpl, Quotas::DATASTORE, att_quota);
|
||||
|
||||
|
@ -20,6 +20,39 @@ require 'CommandManager'
|
||||
|
||||
module TransferManager
|
||||
|
||||
# This class includes methods manage backup images
|
||||
class BackupImage
|
||||
|
||||
def initialize(action_xml)
|
||||
@action = REXML::Document.new(action_xml).root
|
||||
@increments = {}
|
||||
|
||||
prefix = '/DS_DRIVER_ACTION_DATA/IMAGE'
|
||||
|
||||
@action.each_element("#{prefix}/BACKUP_INCREMENTS/INCREMENT") do |inc|
|
||||
id = inc.elements['ID'].text.to_i
|
||||
|
||||
@increments[id] = inc.elements['SOURCE'].text
|
||||
end
|
||||
|
||||
if @increments.empty?
|
||||
@increments[0] = @action.elements["#{prefix}/SOURCE"].text
|
||||
end
|
||||
end
|
||||
|
||||
def last
|
||||
@increments[@increments.keys.last]
|
||||
end
|
||||
|
||||
def snapshots
|
||||
@increments.values
|
||||
end
|
||||
|
||||
def chain
|
||||
@increments.map {|k,v| "#{k}:#{v}"}.join(',')
|
||||
end
|
||||
end
|
||||
|
||||
# This class includes methods to generate a recovery VM template based
|
||||
# on the XML stored in a Backup
|
||||
#
|
||||
@ -72,7 +105,7 @@ module TransferManager
|
||||
@vm = OpenNebula::VirtualMachine.new(xml, nil)
|
||||
|
||||
@base_name = "#{@vm.id}-#{opts[:backup_id]}"
|
||||
@base_url = "#{opts[:proto]}://#{opts[:ds_id]}/#{opts[:backup_id]}"
|
||||
@base_url = "#{opts[:proto]}://#{opts[:ds_id]}/#{opts[:chain]}"
|
||||
|
||||
@ds_id = opts[:ds_id]
|
||||
|
||||
|
717
src/tm_mad/lib/backup_qcow2.rb
Executable file
717
src/tm_mad/lib/backup_qcow2.rb
Executable file
@ -0,0 +1,717 @@
|
||||
#!/usr/bin/env ruby
|
||||
|
||||
# -------------------------------------------------------------------------- #
|
||||
# Copyright 2002-2022, OpenNebula Project, OpenNebula Systems #
|
||||
# #
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may #
|
||||
# not use this file except in compliance with the License. You may obtain #
|
||||
# a copy of the License at #
|
||||
# #
|
||||
# http://www.apache.org/licenses/LICENSE-2.0 #
|
||||
# #
|
||||
# Unless required by applicable law or agreed to in writing, software #
|
||||
# distributed under the License is distributed on an "AS IS" BASIS, #
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. #
|
||||
# See the License for the specific language governing permissions and #
|
||||
# limitations under the License. #
|
||||
#--------------------------------------------------------------------------- #
|
||||
|
||||
require 'json'
|
||||
require 'open3'
|
||||
require 'rexml/document'
|
||||
require 'base64'
|
||||
require 'getoptlong'
|
||||
|
||||
require_relative './kvm'
|
||||
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Helper module to execute commands
|
||||
#---------------------------------------------------------------------------
|
||||
module Command
|
||||
def cmd(command, args, opts = {})
|
||||
opts.each do |key, value|
|
||||
if value.class == Array
|
||||
value.each { |v| command << render_opt(key, v) }
|
||||
else
|
||||
command << render_opt(key, value)
|
||||
end
|
||||
end
|
||||
|
||||
File.write($debug, "#{command} #{args}\n", mode: 'a') if $debug
|
||||
|
||||
out, err, rc = Open3.capture3("#{command} #{args}")
|
||||
|
||||
if rc.exitstatus != 0
|
||||
raise StandardError.new "Error executing '#{command} #{args}':\n#{out} #{err}"
|
||||
end
|
||||
|
||||
out
|
||||
end
|
||||
|
||||
def render_opt(name, value)
|
||||
if name.length == 1
|
||||
opt = " -#{name.to_s.gsub('_','-')}"
|
||||
else
|
||||
opt = " --#{name.to_s.gsub('_','-')}"
|
||||
end
|
||||
|
||||
if value && !value.empty?
|
||||
opt << " " if name[-1] != '='
|
||||
opt << "#{value}"
|
||||
end
|
||||
|
||||
opt
|
||||
end
|
||||
end
|
||||
|
||||
#-------------------------------------------------------------------------------
|
||||
# Setup an NBD server to pull changes, an optional map can be provided
|
||||
#-------------------------------------------------------------------------------
|
||||
module Nbd
|
||||
@@server = nil
|
||||
|
||||
def self.start_nbd(file, map = '')
|
||||
return if @@server != nil
|
||||
|
||||
@@socket = "#{File.realpath(file)}.socket"
|
||||
@@server = fork {
|
||||
args = ['-r', '-k', @@socket, '-f', 'qcow2', '-t']
|
||||
args << '-B' << map unless map.empty?
|
||||
args = file
|
||||
|
||||
exec('qemu-nbd', *args)
|
||||
}
|
||||
|
||||
sleep(1) #TODO inotify or poll for @@socket
|
||||
end
|
||||
|
||||
def self.stop_nbd
|
||||
Process.kill('QUIT', @@server)
|
||||
Process.waitpid(@@server)
|
||||
|
||||
File.unlink(@@socket)
|
||||
|
||||
@@server = nil
|
||||
end
|
||||
|
||||
def self.uri
|
||||
"nbd+unix:///?socket=#{@@socket}"
|
||||
end
|
||||
end
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# This class abstracts the information and several methods to operate over
|
||||
# disk images files
|
||||
# ------------------------------------------------------------------------------
|
||||
class QemuImg
|
||||
include Command
|
||||
|
||||
def initialize(path)
|
||||
@path = path
|
||||
@_info = nil
|
||||
|
||||
@path = File.realpath(path) if File.exist?(path)
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# qemu-img command methods
|
||||
#---------------------------------------------------------------------------
|
||||
QEMU_IMG_COMMANDS = %w[convert create rebase info bitmap]
|
||||
|
||||
QEMU_IMG_COMMANDS.each do |command|
|
||||
define_method(command.to_sym) do |args = '', opts|
|
||||
cmd("qemu-img #{command}", "#{@path} #{args}", opts)
|
||||
end
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Image information methods
|
||||
#---------------------------------------------------------------------------
|
||||
def [](key)
|
||||
if !@_info
|
||||
out = info(:output => 'json', :force_share => '')
|
||||
@_info = JSON.parse(out)
|
||||
end
|
||||
|
||||
@_info[key]
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Pull changes since last checkpoint through the NBD server in this image
|
||||
# 1. Get list of dirty blocks
|
||||
# 2. Create increment qcow2 using NBD as backing file
|
||||
# 3. Pull changes reading (copy-on-write)
|
||||
#
|
||||
# Note: Increment files neeed rebase to reconstruct the increment chain
|
||||
#---------------------------------------------------------------------------
|
||||
def pull_changes(uri, map)
|
||||
exts = if !map || map.empty?
|
||||
data_extents(uri, '')
|
||||
else
|
||||
dirty_extents(uri, map)
|
||||
end
|
||||
|
||||
rc, msg = create(:f => 'qcow2', :F => 'raw', :b => uri)
|
||||
|
||||
return [false, msg] unless rc
|
||||
|
||||
exts.each do |e|
|
||||
cmd('qemu-io', @path,
|
||||
:C => '',
|
||||
:c => "\"r #{e['offset']} #{e['length']}\"",
|
||||
:f => 'qcow2')
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Gets the dirty extent information from the given map using an NBD server
|
||||
#---------------------------------------------------------------------------
|
||||
def dirty_extents(uri, map)
|
||||
extents(uri, map, 'dirty')
|
||||
end
|
||||
|
||||
def data_extents(uri, map)
|
||||
#TODO change for pattern include zero
|
||||
extents(uri, map, 'data')
|
||||
end
|
||||
|
||||
def extents(uri, map, description)
|
||||
opts = { :json => '' }
|
||||
|
||||
if !map.empty?
|
||||
opts[:map=] = map
|
||||
else
|
||||
opts[:map] = ''
|
||||
end
|
||||
|
||||
out = cmd('nbdinfo', uri, opts)
|
||||
|
||||
bmap = JSON.parse(out)
|
||||
exts = []
|
||||
|
||||
bmap.each do |e|
|
||||
next if !e || e['description'] != description
|
||||
exts << e
|
||||
end
|
||||
|
||||
exts
|
||||
end
|
||||
end
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# This class represents a KVM domain, includes information about the associated
|
||||
# OpenNebula VM
|
||||
# ------------------------------------------------------------------------------
|
||||
class KVMDomain
|
||||
|
||||
include TransferManager::KVM
|
||||
include Command
|
||||
|
||||
attr_reader :parent_id, :backup_id, :checkpoint
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# @param vm[REXML::Document] OpenNebula XML VM information
|
||||
# @param opts[Hash] Vm attributes:
|
||||
# - :vm_dir VM folder (/var/lib/one/datastores/<DS_ID>/<VM_ID>)
|
||||
#---------------------------------------------------------------------------
|
||||
def initialize(vm_xml, opts = {})
|
||||
@vm = REXML::Document.new(vm_xml).root
|
||||
|
||||
@vid = @vm.elements['ID'].text
|
||||
@dom = @vm.elements['DEPLOY_ID'].text
|
||||
|
||||
@backup_id = 0
|
||||
@parent_id = -1
|
||||
|
||||
@checkpoint = false
|
||||
|
||||
mode = @vm.elements['BACKUPS/BACKUP_CONFIG/MODE']
|
||||
|
||||
if mode
|
||||
case(mode.text)
|
||||
when 'FULL'
|
||||
@backup_id = 0
|
||||
@parent_id = -1
|
||||
|
||||
@checkpoint = false
|
||||
when 'INCREMENT'
|
||||
li = @vm.elements['BACKUPS/BACKUP_CONFIG/LAST_INCREMENT_ID'].text.to_i
|
||||
|
||||
@backup_id = li + 1
|
||||
@parent_id = li
|
||||
|
||||
@checkpoint = true
|
||||
end
|
||||
end
|
||||
|
||||
@vm_dir = opts[:vm_dir]
|
||||
@tmp_dir = "#{opts[:vm_dir]}/tmp"
|
||||
@bck_dir = "#{opts[:vm_dir]}/backup"
|
||||
|
||||
@socket = "#{opts[:vm_dir]}/backup.socket"
|
||||
@ongoing = false
|
||||
@frozen = nil
|
||||
end
|
||||
|
||||
# "pause" the VM before excuting any FS related operation. The modes are:
|
||||
# - NONE (no operation)
|
||||
# - AGENT (domfsfreeze - domfsthaw)
|
||||
# - SUSPEND (suspend - resume)
|
||||
#
|
||||
# @return [String, String] freeze and thaw commands
|
||||
def fsfreeze
|
||||
@frozen = begin
|
||||
@vm.elements['/VM/BACKUPS/BACKUP_CONFIG/FS_FREEZE'].text.upcase
|
||||
rescue StandardError
|
||||
'NONE'
|
||||
end
|
||||
|
||||
case @frozen
|
||||
when 'AGENT'
|
||||
cmd("#{virsh} domfsfreeze", @dom)
|
||||
when 'SUSPEND'
|
||||
cmd("#{virsh} suspend", @dom)
|
||||
end
|
||||
end
|
||||
|
||||
def fsthaw
|
||||
return unless @frozen
|
||||
|
||||
case @frozen
|
||||
when 'AGENT'
|
||||
cmd("#{virsh} domfsthaw", @dom)
|
||||
when 'SUSPEND'
|
||||
cmd("#{virsh} resume", @dom)
|
||||
end
|
||||
|
||||
ensure
|
||||
@frozen = nil
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Re-define the parent_id checkpoint if not included in the checkpoint-list.
|
||||
# If the checkpoint is not present in storage the methods will fail.
|
||||
#
|
||||
# @param[String] List of disks to include in the checkpoint
|
||||
# @param[Integer] id of the checkpoint to define
|
||||
#---------------------------------------------------------------------------
|
||||
def define_checkpoint(disks_s)
|
||||
return if @parent_id == -1
|
||||
|
||||
#-----------------------------------------------------------------------
|
||||
# Check if the parent_id checkpoint is already defined for this domain
|
||||
#-----------------------------------------------------------------------
|
||||
out = cmd("#{virsh} checkpoint-list", @dom, :name => '')
|
||||
out.strip!
|
||||
|
||||
check_ids = []
|
||||
|
||||
out.each_line do |l|
|
||||
m = l.match(/(one-[0-9]+)-([0-9]+)/)
|
||||
next unless m
|
||||
|
||||
check_ids << m[2].to_i
|
||||
end
|
||||
|
||||
return if check_ids.include? @parent_id
|
||||
|
||||
#-----------------------------------------------------------------------
|
||||
# Try to re-define checkpoint, will fail if not present in storage.
|
||||
# Can be queried using qemu-monitor
|
||||
#
|
||||
# out = cmd("#{virsh} qemu-monitor-command", @dom,
|
||||
# :cmd => '{"execute": "query-block","arguments": {}}')
|
||||
#-----------------------------------------------------------------------
|
||||
disks = disks_s.split ':'
|
||||
tgts = []
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
next unless disks.include? did
|
||||
tgts << d.elements['TARGET'].text
|
||||
end
|
||||
|
||||
return if tgts.empty?
|
||||
|
||||
disks = '<disks>'
|
||||
tgts.each { |tgt| disks << "<disk name='#{tgt}'/>" }
|
||||
disks << '</disks>'
|
||||
|
||||
checkpoint_xml = <<~EOS
|
||||
<domaincheckpoint>
|
||||
<name>one-#{@vid}-#{@parent_id}</name>
|
||||
<creationTime>#{Time.now.to_i}</creationTime>
|
||||
#{disks}
|
||||
</domaincheckpoint>
|
||||
EOS
|
||||
|
||||
cpath = "#{@tmp_dir}/checkpoint.xml"
|
||||
|
||||
File.open(cpath, "w") { |f| f.write(checkpoint_xml) }
|
||||
|
||||
cmd("#{virsh} checkpoint-create", @dom,
|
||||
:xmlfile => cpath, :redefine => '')
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Cleans defined checkpoints up to the last one taken for this backup
|
||||
#---------------------------------------------------------------------------
|
||||
def clean_checkpoints(all = false)
|
||||
return unless @checkpoint
|
||||
|
||||
out = cmd("#{virsh} checkpoint-list", @dom, :name => '')
|
||||
out.strip!
|
||||
|
||||
out.each_line do |l|
|
||||
m = l.match(/(one-[0-9]+)-([0-9]+)/)
|
||||
next if (!m || (!all && m[2].to_i == @backup_id))
|
||||
|
||||
cmd("#{virsh} checkpoint-delete", "#{@dom} #{m[1]}-#{m[2]}")
|
||||
end
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Make a live backup for the VM.
|
||||
# @param [Array] ID of disks that will take part on the backup
|
||||
# @param [Boolean] if true do not generate checkpoint
|
||||
#---------------------------------------------------------------------------
|
||||
def backup_nbd_live(disks_s)
|
||||
disks = disks_s.split ':'
|
||||
|
||||
fsfreeze
|
||||
|
||||
start_backup(disks, @backup_id, @parent_id, @checkpoint)
|
||||
|
||||
fsthaw
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
tgt = d.elements['TARGET'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
ipath = "#{@bck_dir}/disk.#{did}.#{@backup_id}"
|
||||
idisk = QemuImg.new(ipath)
|
||||
|
||||
if @parent_id == -1
|
||||
map = ''
|
||||
else
|
||||
map = "qemu:dirty-bitmap:backup-#{tgt}"
|
||||
end
|
||||
|
||||
idisk.pull_changes(mkuri(tgt), map)
|
||||
end
|
||||
|
||||
ensure
|
||||
fsthaw
|
||||
stop_backup
|
||||
end
|
||||
|
||||
def backup_full_live(disks_s)
|
||||
disks = disks_s.split ':'
|
||||
dspec = []
|
||||
qdisk = {}
|
||||
|
||||
disk_xml = '<disks>'
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
tgt = d.elements['TARGET'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
overlay = "#{@tmp_dir}/overlay_#{did}.qcow2"
|
||||
|
||||
File.open(overlay, "w"){}
|
||||
|
||||
dspec << "#{tgt},file=#{overlay}"
|
||||
|
||||
disk_xml << "<disk name='#{tgt}'/>"
|
||||
|
||||
qdisk[did] = QemuImg.new("#{@vm_dir}/disk.#{did}")
|
||||
end
|
||||
|
||||
disk_xml << '</disks>'
|
||||
|
||||
opts = {
|
||||
:name => "one-#{@vid}-backup",
|
||||
:disk_only => '',
|
||||
:atomic => '',
|
||||
:diskspec => dspec
|
||||
}
|
||||
|
||||
checkpoint_xml = <<~EOS
|
||||
<domaincheckpoint>
|
||||
<name>one-#{@vid}-0</name>
|
||||
#{disk_xml}
|
||||
</domaincheckpoint>
|
||||
EOS
|
||||
|
||||
cpath = "#{@tmp_dir}/checkpoint.xml"
|
||||
|
||||
File.open(cpath, "w") { |f| f.write(checkpoint_xml) }
|
||||
|
||||
fsfreeze
|
||||
|
||||
cmd("#{virsh} snapshot-create-as", @dom, opts)
|
||||
|
||||
cmd("#{virsh} checkpoint-create", @dom, :xmlfile => cpath) if @checkpoint
|
||||
|
||||
fsthaw
|
||||
|
||||
qdisk.each do |did, disk|
|
||||
disk.convert("#{@bck_dir}/disk.#{did}.0", :m => '4', :O => 'qcow2')
|
||||
end
|
||||
ensure
|
||||
fsthaw
|
||||
end
|
||||
|
||||
def stop_backup_full_live(disks_s)
|
||||
disks = disks_s.split ':'
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
tgt = d.elements['TARGET'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
opts = {
|
||||
:base => "#{@vm_dir}/disk.#{did}",
|
||||
:active => '',
|
||||
:pivot => '',
|
||||
:keep_relative => ''
|
||||
}
|
||||
|
||||
cmd("#{virsh} blockcommit", "#{@dom} #{tgt}", opts)
|
||||
end
|
||||
|
||||
cmd("#{virsh} snapshot-delete", "#{@dom}",
|
||||
:snapshotname => "one-#{@vid}-backup",
|
||||
:metadata => '')
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Make a backup for the VM. (see make_backup_live)
|
||||
#---------------------------------------------------------------------------
|
||||
def backup_nbd(disks_s)
|
||||
disks = disks_s.split ':'
|
||||
|
||||
if @parent_id == -1
|
||||
nbd_map = ''
|
||||
map = ''
|
||||
else
|
||||
nbd_map = "one-#{@vid}-#{@parent_id}"
|
||||
map = "qemu:dirty-bitmap:#{nbd_map}"
|
||||
end
|
||||
|
||||
dids = []
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
dids << did if disks.include? did
|
||||
end
|
||||
|
||||
dids.each do |d|
|
||||
idisk = QemuImg.new("#{@bck_dir}/disk.#{d}.#{@backup_id}")
|
||||
|
||||
Nbd.start_nbd("#{@vm_dir}/disk.#{d}", nbd_map)
|
||||
|
||||
idisk.pull_changes(Nbd.uri, map)
|
||||
ensure
|
||||
Nbd.stop_nbd
|
||||
end
|
||||
|
||||
#TODO Check if baking files needs bitmap
|
||||
dids.each do |d|
|
||||
idisk = QemuImg.new("#{@vm_dir}/disk.#{d}")
|
||||
|
||||
idisk.bitmap("one-#{@vid}-#{@backup_id}", :add => '')
|
||||
end if @checkpoint
|
||||
end
|
||||
|
||||
def backup_full(disks_s)
|
||||
disks = disks_s.split ':'
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
sdisk = QemuImg.new("#{@vm_dir}/disk.#{d}")
|
||||
ddisk = "#{@bck_dir}/disk.#{did}.0"
|
||||
|
||||
sdisk.convert(ddisk, :m => 4, :O => 'qcow2')
|
||||
sdisk.bitmap("one-#{@vid}-0", :add => '') if @checkpoint
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Generate nbd URI to query block bitmaps for a device
|
||||
def mkuri(target)
|
||||
"nbd+unix:///#{target}?socket=#{@socket}"
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Start a Backup operation on the domain (See make_backup_live)
|
||||
#---------------------------------------------------------------------------
|
||||
def start_backup(disks, bck_id, pid, checkpoint)
|
||||
parent = "one-#{@vid}-#{pid}"
|
||||
bname = "one-#{@vid}-#{bck_id}"
|
||||
|
||||
parent_xml = "<incremental>#{parent}</incremental>" if pid != -1
|
||||
|
||||
backup_xml = <<~EOS
|
||||
<domainbackup mode='pull'>
|
||||
#{parent_xml}
|
||||
<server transport='unix' socket='#{@socket}'/>
|
||||
<disks>
|
||||
EOS
|
||||
|
||||
checkpoint_xml = <<~EOS
|
||||
<domaincheckpoint>
|
||||
<name>#{bname}</name>
|
||||
<disks>
|
||||
EOS
|
||||
|
||||
@vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
tgt = d.elements['TARGET'].text
|
||||
szm = d.elements['SIZE'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
spath = "#{@tmp_dir}/scracth.#{did}.qcow2"
|
||||
|
||||
simg = QemuImg.new(spath)
|
||||
simg.create("#{szm}M", :f => 'qcow2')
|
||||
|
||||
backup_xml << <<~EOS
|
||||
<disk name='#{tgt}' backup='yes' type='file'>
|
||||
<scratch file='#{spath}'/>
|
||||
</disk>
|
||||
EOS
|
||||
|
||||
checkpoint_xml << "<disk name='#{tgt}'/>"
|
||||
end
|
||||
|
||||
checkpoint_xml << <<~EOS
|
||||
</disks>
|
||||
</domaincheckpoint>
|
||||
EOS
|
||||
|
||||
backup_xml << <<~EOS
|
||||
</disks>
|
||||
</domainbackup>
|
||||
EOS
|
||||
|
||||
backup_path = "#{@tmp_dir}/backup.xml"
|
||||
check_path = "#{@tmp_dir}/checkpoint.xml"
|
||||
|
||||
File.open(backup_path, "w") { |f| f.write(backup_xml) }
|
||||
|
||||
File.open(check_path, "w") { |f| f.write(checkpoint_xml) }
|
||||
|
||||
opts = { :reuse_external => '', :backupxml => backup_path}
|
||||
opts[:checkpointxml] = check_path if checkpoint
|
||||
|
||||
cmd("#{virsh} backup-begin", @dom, opts)
|
||||
|
||||
@ongoing = true
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Stop an ongoing Backup operation on the domain
|
||||
#---------------------------------------------------------------------------
|
||||
def stop_backup
|
||||
return unless @ongoing
|
||||
|
||||
cmd("#{virsh} domjobabort", @dom, {})
|
||||
ensure
|
||||
@ongoing = false
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
opts = GetoptLong.new(
|
||||
['--disks','-d', GetoptLong::REQUIRED_ARGUMENT],
|
||||
['--vmxml','-x', GetoptLong::REQUIRED_ARGUMENT],
|
||||
['--path', '-p', GetoptLong::REQUIRED_ARGUMENT],
|
||||
['--live', '-l', GetoptLong::NO_ARGUMENT],
|
||||
['--debug','-v', GetoptLong::NO_ARGUMENT],
|
||||
['--stop', '-s', GetoptLong::NO_ARGUMENT]
|
||||
)
|
||||
|
||||
begin
|
||||
path = disks = vmxml = ''
|
||||
live = stop = false
|
||||
|
||||
$debug = nil
|
||||
|
||||
opts.each do |opt, arg|
|
||||
case opt
|
||||
when '--disks'
|
||||
disks = arg
|
||||
when '--path'
|
||||
path = arg
|
||||
when '--live'
|
||||
live = true
|
||||
when '--stop'
|
||||
stop = true
|
||||
when '--vmxml'
|
||||
vmxml = arg
|
||||
when '--debug'
|
||||
$debug = "/tmp/one.backup.#{Time.now.to_i}"
|
||||
end
|
||||
end
|
||||
|
||||
vm = KVMDomain.new(Base64.decode64(File.read(vmxml)), :vm_dir => path)
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Stop operation. Only for full backups in live mode. It blockcommits
|
||||
# changes and cleans snapshot.
|
||||
#---------------------------------------------------------------------------
|
||||
if stop
|
||||
vm.stop_backup_full_live(disks) if vm.parent_id == -1 && live
|
||||
exit(0)
|
||||
end
|
||||
|
||||
#---------------------------------------------------------------------------
|
||||
# Backup operation
|
||||
# - (live - full) Creates a snapshot to copy the disks via qemu-convert
|
||||
# all previous defined checkpoints are cleaned.
|
||||
# - (live - increment) starts a backup operation in libvirt and pull changes
|
||||
# through NBD server using qemu-io copy-on-read feature
|
||||
# - (poff - full) copy disks via qemu-convert
|
||||
# - (poff - incremental) starts qemu-nbd server to pull changes from the
|
||||
# last checkpoint
|
||||
#---------------------------------------------------------------------------
|
||||
if live
|
||||
if vm.parent_id == -1
|
||||
vm.clean_checkpoints(true)
|
||||
|
||||
vm.backup_full_live(disks)
|
||||
else
|
||||
vm.define_checkpoint(disks)
|
||||
|
||||
vm.backup_nbd_live(disks)
|
||||
|
||||
vm.clean_checkpoints
|
||||
end
|
||||
else
|
||||
if vm.parent_id == -1
|
||||
vm.backup_full(disks)
|
||||
else
|
||||
vm.backup_nbd(disks)
|
||||
end
|
||||
end
|
||||
|
||||
rescue StandardError => e
|
||||
puts e.message
|
||||
exit(-1)
|
||||
end
|
@ -16,9 +16,10 @@
|
||||
# limitations under the License. #
|
||||
#--------------------------------------------------------------------------- #
|
||||
|
||||
# rubocop:disable Style/ClassAndModuleChildren
|
||||
# rubocop:disable Style/ClassVars
|
||||
module TransferManager::KVM
|
||||
module TransferManager
|
||||
|
||||
module KVM
|
||||
|
||||
KVMRC = '/var/lib/one/remotes/etc/vmm/kvm/kvmrc'
|
||||
|
||||
@ -103,5 +104,6 @@ module TransferManager::KVM
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
# rubocop:enable Style/ClassAndModuleChildren
|
||||
# rubocop:enable Style/ClassVars
|
||||
|
@ -61,48 +61,29 @@ include TransferManager::KVM
|
||||
|
||||
# BACKUP tm_mad host:remote_dir DISK_ID:...:DISK_ID deploy_id vmid dsid
|
||||
|
||||
vm_xml = STDIN.read
|
||||
_vm_xml = STDIN.read
|
||||
|
||||
dir = ARGV[0].split ':'
|
||||
disks = ARGV[1].split ':'
|
||||
deploy_id = ARGV[2]
|
||||
vmid = ARGV[3]
|
||||
disks = ARGV[1]
|
||||
_deploy_id = ARGV[2]
|
||||
_vmid = ARGV[3]
|
||||
_dsid = ARGV[4]
|
||||
|
||||
rhost = dir[0]
|
||||
rdir = dir[1]
|
||||
|
||||
xml_doc = REXML::Document.new(vm_xml)
|
||||
vm = xml_doc.root
|
||||
|
||||
tmp_dir = "#{rdir}/tmp"
|
||||
bck_dir = "#{rdir}/backup"
|
||||
|
||||
cmds = ''
|
||||
|
||||
# TODO: limit BW?
|
||||
|
||||
vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
tgt = d.elements['TARGET'].text
|
||||
base = "#{rdir}/disk.#{did}"
|
||||
|
||||
cmds << "#{virsh} blockcommit #{deploy_id} #{tgt} --active --pivot"\
|
||||
" --base #{base} --keep-relative\n"
|
||||
end
|
||||
qcow2_util = '/var/tmp/one/tm/lib/backup_qcow2.rb'
|
||||
|
||||
post_script = <<~EOS
|
||||
set -ex -o pipefail
|
||||
|
||||
# --------------------------------------------------------------------
|
||||
# Commit changes since backup, pivot disks and delete libvirt snapshot
|
||||
# Commit changes, pivot disks and delete libvirt snapshot (only full)
|
||||
# --------------------------------------------------------------------
|
||||
#{cmds}
|
||||
|
||||
#{virsh} snapshot-delete #{deploy_id} --metadata one-#{vmid}-backup
|
||||
#{qcow2_util} -s -l -d "#{disks}" -x #{bck_dir}/vm.xml -p #{rdir}
|
||||
|
||||
# ----------------------
|
||||
# Cleanup backup folders
|
||||
|
@ -61,7 +61,7 @@ require 'rexml/document'
|
||||
vm_xml = STDIN.read
|
||||
|
||||
dir = ARGV[0].split ':'
|
||||
disks = ARGV[1].split ':'
|
||||
disks = ARGV[1]
|
||||
_deployid = ARGV[2]
|
||||
_vmid = ARGV[3]
|
||||
_dsid = ARGV[4]
|
||||
@ -69,24 +69,8 @@ _dsid = ARGV[4]
|
||||
rhost = dir[0]
|
||||
rdir = dir[1]
|
||||
|
||||
xml_doc = REXML::Document.new(vm_xml)
|
||||
vm = xml_doc.root
|
||||
|
||||
bck_dir = "#{rdir}/backup"
|
||||
|
||||
convert = ''
|
||||
|
||||
# TODO: Make compression configurable. Does it intefere with deduplication?
|
||||
vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
dsrc = "#{rdir}/disk.#{did}"
|
||||
ddst = "#{bck_dir}/disk.#{did}"
|
||||
|
||||
convert << "qemu-img convert -m 4 -O qcow2 #{dsrc} #{ddst}; "
|
||||
end
|
||||
qcow2_util = '/var/tmp/one/tm/lib/backup_qcow2.rb'
|
||||
|
||||
pre_script = <<~EOS
|
||||
set -ex -o pipefail
|
||||
@ -103,7 +87,7 @@ pre_script = <<~EOS
|
||||
# -------------------
|
||||
# Convert and flatten
|
||||
# -------------------
|
||||
#{convert}
|
||||
#{qcow2_util} -d "#{disks}" -x #{bck_dir}/vm.xml -p #{rdir}
|
||||
EOS
|
||||
|
||||
rc = TransferManager::Action.ssh('prebackup',
|
||||
|
@ -65,43 +65,18 @@ include TransferManager::KVM
|
||||
vm_xml = STDIN.read
|
||||
|
||||
dir = ARGV[0].split ':'
|
||||
disks = ARGV[1].split ':'
|
||||
deploy_id = ARGV[2]
|
||||
vmid = ARGV[3]
|
||||
disks = ARGV[1]
|
||||
_deploy_id = ARGV[2]
|
||||
_vmid = ARGV[3]
|
||||
_dsid = ARGV[4]
|
||||
|
||||
rhost = dir[0]
|
||||
rdir = dir[1]
|
||||
|
||||
xml_doc = REXML::Document.new(vm_xml)
|
||||
vm = xml_doc.root
|
||||
|
||||
tmp_dir = "#{rdir}/tmp"
|
||||
bck_dir = "#{rdir}/backup"
|
||||
|
||||
dskspec = ''
|
||||
touch = ''
|
||||
convert = ''
|
||||
|
||||
# TODO: Make compression configurable. Does it intefere with deduplication?
|
||||
vm.elements.each 'TEMPLATE/DISK' do |d|
|
||||
did = d.elements['DISK_ID'].text
|
||||
|
||||
next unless disks.include? did
|
||||
|
||||
tgt = d.elements['TARGET'].text
|
||||
file = "#{tmp_dir}/overlay_#{did}.qcow2"
|
||||
|
||||
dskspec << "--diskspec #{tgt},file=#{file} "
|
||||
touch << "touch #{file}; "
|
||||
|
||||
dsrc = "#{rdir}/disk.#{did}"
|
||||
ddst = "#{bck_dir}/disk.#{did}"
|
||||
|
||||
convert << "qemu-img convert -m 4 -O qcow2 #{dsrc} #{ddst}; "
|
||||
end
|
||||
|
||||
freeze, thaw = fsfreeze(vm, deploy_id)
|
||||
qcow2_util = '/var/tmp/one/tm/lib/backup_qcow2.rb'
|
||||
|
||||
pre_script = <<~EOS
|
||||
set -ex -o pipefail
|
||||
@ -119,25 +94,14 @@ pre_script = <<~EOS
|
||||
|
||||
echo "#{Base64.encode64(vm_xml)}" > #{bck_dir}/vm.xml
|
||||
|
||||
#{touch}
|
||||
|
||||
# --------------------------------------
|
||||
# Create external snapshots for VM disks
|
||||
# Create backup live
|
||||
# --------------------------------------
|
||||
#{freeze}
|
||||
|
||||
#{virsh} snapshot-create-as --domain #{deploy_id} --name one-#{vmid}-backup \
|
||||
--disk-only --atomic #{dskspec}
|
||||
|
||||
#{thaw}
|
||||
|
||||
# -------------------
|
||||
# Convert and flatten
|
||||
# -------------------
|
||||
#{convert}
|
||||
#{qcow2_util} -l -d "#{disks}" -x #{bck_dir}/vm.xml -p #{rdir}
|
||||
EOS
|
||||
|
||||
rc = TransferManager::Action.ssh('prebackup_live', :host => rhost,
|
||||
rc = TransferManager::Action.ssh('prebackup_live',
|
||||
:host => rhost,
|
||||
:cmds => pre_script,
|
||||
:nostdout => false,
|
||||
:nostderr => false)
|
||||
|
@ -46,6 +46,14 @@ bool Backups::do_volatile() const
|
||||
return false;
|
||||
}
|
||||
|
||||
Backups::Mode Backups::mode() const
|
||||
{
|
||||
string mode_s;
|
||||
|
||||
config.get("MODE", mode_s);
|
||||
|
||||
return str_to_mode(mode_s);
|
||||
}
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
@ -109,7 +117,7 @@ int Backups::from_xml(const ObjectXML* xml)
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
int Backups::parse(std::string& error_str, Template *tmpl)
|
||||
int Backups::parse(Template *tmpl, bool can_increment, std::string& error_str)
|
||||
{
|
||||
vector<VectorAttribute *> cfg_a;
|
||||
|
||||
@ -127,7 +135,13 @@ int Backups::parse(std::string& error_str, Template *tmpl)
|
||||
if ( cfg == 0 )
|
||||
{
|
||||
error_str = "Internal error parsing BACKUP_CONFIG attribute.";
|
||||
goto error;
|
||||
|
||||
for (auto &i : cfg_a)
|
||||
{
|
||||
delete i;
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* ---------------------------------------------------------------------- */
|
||||
@ -135,6 +149,7 @@ int Backups::parse(std::string& error_str, Template *tmpl)
|
||||
/* - KEEP_LAST */
|
||||
/* - BACKUP_VOLATILE */
|
||||
/* - FSFREEZE */
|
||||
/* - MODE */
|
||||
/* ---------------------------------------------------------------------- */
|
||||
if (cfg->vector_value("KEEP_LAST", iattr) == 0)
|
||||
{
|
||||
@ -175,19 +190,33 @@ int Backups::parse(std::string& error_str, Template *tmpl)
|
||||
config.replace("FS_FREEZE", "NONE");
|
||||
}
|
||||
|
||||
if (!can_increment) //Only FULL backups for this VM
|
||||
{
|
||||
config.replace("INCREMENTAL_BACKUP_ID", -1);
|
||||
config.replace("LAST_INCREMENT_ID", -1);
|
||||
|
||||
config.replace("MODE", mode_to_str(FULL));
|
||||
}
|
||||
else
|
||||
{
|
||||
sattr = cfg->vector_value("MODE");
|
||||
|
||||
Mode new_mode = str_to_mode(sattr);
|
||||
|
||||
// Reset incremental backup pointers if mode changed to/from FULL
|
||||
if (new_mode != INCREMENT || mode() != INCREMENT)
|
||||
{
|
||||
config.replace("INCREMENTAL_BACKUP_ID", -1);
|
||||
config.replace("LAST_INCREMENT_ID", -1);
|
||||
}
|
||||
|
||||
config.replace("MODE", mode_to_str(new_mode));
|
||||
}
|
||||
|
||||
for (auto &i : cfg_a)
|
||||
{
|
||||
delete i;
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
error:
|
||||
for (auto &i : cfg_a)
|
||||
{
|
||||
delete i;
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
@ -971,16 +971,6 @@ int VirtualMachine::insert(SqlDB * db, string& error_str)
|
||||
goto error_vrouter;
|
||||
}
|
||||
|
||||
// ------------------------------------------------------------------------
|
||||
// Parse the backup attribute
|
||||
// ------------------------------------------------------------------------
|
||||
rc = _backups.parse(error_str, user_obj_template.get());
|
||||
|
||||
if ( rc != 0 )
|
||||
{
|
||||
goto error_backup;
|
||||
}
|
||||
|
||||
// ------------------------------------------------------------------------
|
||||
// Get network leases
|
||||
// ------------------------------------------------------------------------
|
||||
@ -1027,6 +1017,18 @@ int VirtualMachine::insert(SqlDB * db, string& error_str)
|
||||
goto error_boot_order;
|
||||
}
|
||||
|
||||
// ------------------------------------------------------------------------
|
||||
// Parse the backup attribute. It assumes volatile disks will take part of
|
||||
// the backups
|
||||
// ------------------------------------------------------------------------
|
||||
rc = _backups.parse(user_obj_template.get(), disks.backup_increment(true),
|
||||
error_str);
|
||||
|
||||
if ( rc != 0 )
|
||||
{
|
||||
goto error_backup;
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Parse the context & requirements
|
||||
// -------------------------------------------------------------------------
|
||||
@ -1117,20 +1119,11 @@ int VirtualMachine::insert(SqlDB * db, string& error_str)
|
||||
return 0;
|
||||
|
||||
error_update:
|
||||
goto error_rollback;
|
||||
|
||||
error_boot_order:
|
||||
goto error_rollback;
|
||||
|
||||
error_context:
|
||||
goto error_rollback;
|
||||
|
||||
error_requirements:
|
||||
goto error_rollback;
|
||||
|
||||
error_graphics:
|
||||
goto error_rollback;
|
||||
|
||||
error_backup:
|
||||
error_rollback:
|
||||
release_disk_images(quotas, true);
|
||||
|
||||
@ -1171,7 +1164,6 @@ error_os:
|
||||
error_pci:
|
||||
error_defaults:
|
||||
error_vrouter:
|
||||
error_backup:
|
||||
error_public:
|
||||
error_name:
|
||||
error_common:
|
||||
@ -3033,13 +3025,19 @@ int VirtualMachine::updateconf(VirtualMachineTemplate* tmpl, string &err,
|
||||
}
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// Parse backup configuration
|
||||
// Parse backup configuration (if not doing a backup). Uses current value of
|
||||
// BACKUP_VOLATILE attribute.
|
||||
// -------------------------------------------------------------------------
|
||||
if ( _backups.parse(err, tmpl) != 0 )
|
||||
if ( lcm_state != BACKUP && lcm_state != BACKUP_POWEROFF)
|
||||
{
|
||||
bool increment = disks.backup_increment(_backups.do_volatile());
|
||||
|
||||
if ( _backups.parse(tmpl, increment, err) != 0 )
|
||||
{
|
||||
NebulaLog::log("ONE",Log::ERROR, err);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
encrypt();
|
||||
|
||||
|
@ -1533,7 +1533,7 @@ void VirtualMachineDisks::delete_non_persistent_snapshots(Template &vm_quotas,
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
void VirtualMachineDisks::backup_size(Template &ds_quotas, bool do_volatile)
|
||||
long long VirtualMachineDisks::backup_size(Template &ds_quotas, bool do_volatile)
|
||||
{
|
||||
long long size = 0;
|
||||
|
||||
@ -1556,6 +1556,36 @@ void VirtualMachineDisks::backup_size(Template &ds_quotas, bool do_volatile)
|
||||
}
|
||||
|
||||
ds_quotas.add("SIZE", size);
|
||||
|
||||
return size;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
bool VirtualMachineDisks::backup_increment(bool do_volatile)
|
||||
{
|
||||
for (const auto disk : *this)
|
||||
{
|
||||
string type = disk->vector_value("TYPE");
|
||||
|
||||
one_util::toupper(type);
|
||||
|
||||
if ((type == "SWAP") || ((type == "FS") && !do_volatile))
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
string format = disk->vector_value("FORMAT");
|
||||
|
||||
one_util::toupper(format);
|
||||
|
||||
if (format != "QCOW2")
|
||||
{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
@ -246,7 +246,8 @@ std::map<std::string,std::vector<std::string>> VirtualMachineTemplate::UPDATECON
|
||||
"FS_FREEZE",
|
||||
"KEEP_LAST",
|
||||
"BACKUP_VOLATILE",
|
||||
"FREQUENCY_SECONDS"}
|
||||
"FREQUENCY_SECONDS",
|
||||
"MODE"}
|
||||
}
|
||||
};
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user