2016-12-11 23:05:07 +03:00
/* -------------------------------------------------------------------------- */
2017-05-25 17:07:35 +03:00
/* Copyright 2002-2017, OpenNebula Project, OpenNebula Systems */
2016-12-11 23:05:07 +03:00
/* */
/* 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 VIRTUAL_MACHINE_DISK_H_
# define VIRTUAL_MACHINE_DISK_H_
# include <queue>
# include <set>
# include "VirtualMachineAttribute.h"
# include "Snapshots.h"
2016-12-12 11:25:42 +03:00
class AuthRequest ;
2016-12-11 23:05:07 +03:00
/**
* The VirtualMachine DISK attribute
*/
class VirtualMachineDisk : public VirtualMachineAttribute
{
public :
VirtualMachineDisk ( VectorAttribute * va , int id ) :
VirtualMachineAttribute ( va , id ) , snapshots ( 0 ) { } ;
virtual ~ VirtualMachineDisk ( )
{
delete snapshots ;
} ;
/* ---------------------------------------------------------------------- */
/* DISK get/set functions for boolean disk flags */
/* ATTACH */
/* RESIZE */
2017-07-05 19:07:22 +03:00
/* OPENNEBULA_MANAGED */
/* ALLOW_ORPHANS */
2016-12-11 23:05:07 +03:00
/* CLONING */
/* PERSISTENT */
/* DISK_SNAPSHOT_ACTIVE */
/* ---------------------------------------------------------------------- */
bool is_persistent ( ) const
{
return is_flag ( " PERSISTENT " ) ;
}
2017-07-04 19:01:13 +03:00
bool is_managed ( ) const
{
bool one_managed ;
if ( vector_value ( " OPENNEBULA_MANAGED " , one_managed ) = = - 1 )
{
one_managed = true ;
}
return one_managed ;
}
2017-07-05 19:07:22 +03:00
bool allow_orphans ( ) const
{
bool orphans ;
if ( vector_value ( " ALLOW_ORPHANS " , orphans ) = = - 1 )
{
orphans = false ;
}
return orphans ;
}
2016-12-11 23:05:07 +03:00
void set_attach ( )
{
set_flag ( " ATTACH " ) ;
} ;
void set_resize ( )
{
set_flag ( " RESIZE " ) ;
} ;
void clear_resize ( )
{
clear_flag ( " RESIZE " ) ;
} ;
void clear_cloning ( )
{
clear_flag ( " CLONING " ) ;
} ;
bool is_cloning ( ) const
{
return is_flag ( " CLONING " ) ;
}
void set_active_snapshot ( )
{
set_flag ( " DISK_SNAPSHOT_ACTIVE " ) ;
} ;
void clear_active_snapshot ( )
{
2016-12-12 12:26:55 +03:00
clear_flag ( " DISK_SNAPSHOT_ACTIVE " ) ;
2016-12-11 23:05:07 +03:00
} ;
bool is_active_snapshot ( )
{
return is_flag ( " DISK_SNAPSHOT_ACTIVE " ) ;
}
2016-12-12 12:26:55 +03:00
void set_saveas ( )
{
set_flag ( " HOTPLUG_SAVE_AS_ACTIVE " ) ;
} ;
void clear_saveas ( )
{
clear_flag ( " HOTPLUG_SAVE_AS_ACTIVE " ) ;
} ;
2016-12-11 23:05:07 +03:00
/* ---------------------------------------------------------------------- */
/* Disk attributes, not accesible through vector_value */
/* ---------------------------------------------------------------------- */
/**
* Return the disk id ( " DISK_ID " )
*/
int get_disk_id ( ) const
{
return get_id ( ) ;
}
/**
* Return the " effective " target ( LN_TARGET / CLONE_TARGET )
*/
string get_tm_target ( ) const ;
/**
* Check if the given disk is volatile
*/
bool is_volatile ( ) const ;
/**
* Get the effective uid to get an image . Used in VM parsers
*/
int get_uid ( int _uid ) ;
/**
* Gets the ID of the image associated to the disks
* @ param id the image id , if found
* @ param uid effective user id making the call
* @ return 0 if the disk uses an image , - 1 otherwise
*/
int get_image_id ( int & id , int uid ) ;
/* ---------------------------------------------------------------------- */
/* Image Manager Interface */
/* ---------------------------------------------------------------------- */
/**
* Fills the disk extended information attributes
*/
void extended_info ( int uid ) ;
2016-12-12 11:25:42 +03:00
/**
* Fills the authorization request for this disk based on its Image use
* requirements
* @ param uid of user making the request
* @ param ar auth request
*/
void authorize ( int uid , AuthRequest * ar ) ;
2016-12-11 23:05:07 +03:00
/* ---------------------------------------------------------------------- */
/* Snapshots Interface */
/* ---------------------------------------------------------------------- */
/**
* Set the snapshots for this disks
*/
void set_snapshots ( Snapshots * _snapshots )
{
snapshots = _snapshots ;
} ;
/**
* Return the snapshots of this disk
*/
const Snapshots * get_snapshots ( ) const
{
return snapshots ;
}
/**
* Clear snapshots from the disk and free resources
*/
void clear_snapshots ( )
{
2016-12-24 03:35:33 +03:00
if ( snapshots = = 0 )
{
return ;
}
2016-12-11 23:05:07 +03:00
snapshots - > clear ( ) ;
delete snapshots ;
snapshots = 0 ;
}
/**
* @ return total snapshot size ( virtual ) in mb
*/
long long get_total_snapshot_size ( ) const
{
2016-12-24 03:35:33 +03:00
if ( snapshots = = 0 )
{
return 0 ;
}
2016-12-11 23:05:07 +03:00
return snapshots - > get_total_size ( ) ;
}
/**
* Get the size ( virtual ) in mb of the given snapshot
* @ param id of the snapshot
* @ return size or 0 if not found
*/
long long get_snapshot_size ( int snap_id ) const
{
2016-12-24 03:35:33 +03:00
if ( snapshots = = 0 )
{
return 0 ;
}
2016-12-11 23:05:07 +03:00
return snapshots - > get_snapshot_size ( snap_id ) ;
}
/**
* @ return true if the disk has snapshots
*/
bool has_snapshots ( )
{
return ( snapshots ! = 0 ) ;
}
/**
* Creates a new snapshot of the disk
* @ param name a description for this snapshot
* @ param error if any
* @ return the id of the new snapshot or - 1 if error
*/
int create_snapshot ( const string & name , string & error ) ;
/**
* Sets the snap_id as active , the VM will boot from it next time
* @ param snap_id of the snapshot
* @ return - 1 if error
*/
int revert_snapshot ( int snap_id ) ;
/**
* Deletes the snap_id from the list
* @ param snap_id of the snapshot
* @ param ds_quotas template with snapshot usage for the DS quotas
* @ param vm_quotas template with snapshot usage for the VM quotas
2017-03-31 21:09:27 +03:00
* @ param io delete ds quotas from image owners
* @ param vo delete ds quotas from vm owners
2016-12-11 23:05:07 +03:00
*/
2017-03-31 21:09:27 +03:00
void delete_snapshot ( int snap_id , Template * * ds_quota , Template * * vm_quota ,
bool & io , bool & vo ) ;
2016-12-11 23:05:07 +03:00
2016-12-15 23:12:33 +03:00
/* ---------------------------------------------------------------------- */
2016-12-17 04:49:14 +03:00
/* Disk resize functions */
2016-12-15 23:12:33 +03:00
/* ---------------------------------------------------------------------- */
/**
2016-12-17 04:49:14 +03:00
* Cleans the resize attribute from the disk
* @ param restore the previous size
2016-12-15 23:12:33 +03:00
*/
2016-12-17 04:49:14 +03:00
void clear_resize ( bool restore ) ;
2016-12-15 23:12:33 +03:00
/**
* Calculate the quotas for a resize operation on the disk
* @ param new_size of disk
* @ param dsdeltas increment in datastore usage
* @ param vmdelta increment in system datastore usage
2017-03-30 19:54:58 +03:00
* @ param do_img_owner quotas counter allocated for image uid / gid
* @ param do_vm_owner quotas counter allocated for vm uid / gid
*
2016-12-15 23:12:33 +03:00
*/
2017-03-30 19:54:58 +03:00
void resize_quotas ( long long new_size , Template & dsdelta , Template & vmdelta ,
bool & do_img_owner , bool & do_vm_owner ) ;
2016-12-15 23:12:33 +03:00
2016-12-17 04:49:14 +03:00
/* ---------------------------------------------------------------------- */
/* Disk space usage functions */
/* ---------------------------------------------------------------------- */
/**
* @ return the space required by this disk in the system datastore
*/
long long system_ds_size ( ) ;
2017-03-30 19:54:58 +03:00
/**
* @ return the space required by this disk in the image datastore
*/
long long image_ds_size ( ) ;
2016-12-15 23:12:33 +03:00
/**
* Compute the storage needed by the disk in the system and / or image
* datastore
* @ param ds_id of the datastore
* @ param img_sz size in image datastore needed
* @ param sys_sz size in system datastore needed
*/
void datastore_sizes ( int & ds_id , long long & img_sz , long long & sys_sz ) ;
2016-12-11 23:05:07 +03:00
private :
Snapshots * snapshots ;
} ;
/**
* Set of VirtualMachine DIKS
*/
class VirtualMachineDisks : public VirtualMachineAttributeSet
{
public :
/* ---------------------------------------------------------------------- */
/* Constructor and Initialization functions */
/* ---------------------------------------------------------------------- */
/**
* Creates the VirtualMachineDisk set from a Template with DISK = [ . . . ]
* attributes , in this case the id ' s of each disk is auto assigned
* @ param tmpl template with DISK
*/
VirtualMachineDisks ( Template * tmpl , bool has_id ) :
2016-12-12 12:26:55 +03:00
VirtualMachineAttributeSet ( false )
{
std : : vector < VectorAttribute * > vas ;
tmpl - > get ( DISK_NAME , vas ) ;
init ( vas , has_id ) ;
} ;
2016-12-11 23:05:07 +03:00
/**
* Creates the VirtualMachineDisk set from a vector of DISK VectorAttribute
* The DIKS need to have a DISK_ID assgined to create the disk set .
* @ param va vector of DISK Vector Attributes
*/
VirtualMachineDisks ( vector < VectorAttribute * > & va , bool has_id , bool dispose ) :
2016-12-12 12:26:55 +03:00
VirtualMachineAttributeSet ( dispose )
{
init ( va , has_id ) ;
} ;
2016-12-11 23:05:07 +03:00
/**
* Creates an empty disk set
*/
VirtualMachineDisks ( bool dispose ) :
VirtualMachineAttributeSet ( dispose ) { } ;
virtual ~ VirtualMachineDisks ( ) { } ;
/**
* Function used to initialize the attribute map based on a vector of DISK
*/
void init ( std : : vector < VectorAttribute * > & vas , bool has_id )
{
if ( has_id )
{
init_attribute_map ( DISK_ID_NAME , vas ) ;
}
else
{
init_attribute_map ( " " , vas ) ;
}
}
/* ---------------------------------------------------------------------- */
/* Iterators */
/* ---------------------------------------------------------------------- */
/**
* Generic iterator for the disk set .
*/
class DiskIterator : public AttributeIterator
{
public :
DiskIterator ( ) : AttributeIterator ( ) { } ;
DiskIterator ( const AttributeIterator & dit ) : AttributeIterator ( dit ) { } ;
virtual ~ DiskIterator ( ) { } ;
VirtualMachineDisk * operator * ( ) const
{
return static_cast < VirtualMachineDisk * > ( map_it - > second ) ;
}
} ;
DiskIterator begin ( )
{
2017-04-10 19:41:43 +03:00
DiskIterator it ( ExtendedAttributeSet : : begin ( ) ) ;
2016-12-11 23:05:07 +03:00
return it ;
}
DiskIterator end ( )
{
2017-04-10 19:41:43 +03:00
DiskIterator it ( ExtendedAttributeSet : : end ( ) ) ;
2016-12-11 23:05:07 +03:00
return it ;
}
typedef class DiskIterator disk_iterator ;
/* ---------------------------------------------------------------------- */
/* DISK interface */
/* ---------------------------------------------------------------------- */
/**
* Returns the DISK attribute for a disk
* @ param disk_id of the DISK
* @ return pointer to the attribute ir null if not found
*/
VirtualMachineDisk * get_disk ( int disk_id ) const
{
return static_cast < VirtualMachineDisk * > ( get_attribute ( disk_id ) ) ;
}
/**
* Computes the storage needed in the system datastore . The static version
* uses the disk definitions in the template ( first argument )
* @ return the total disk SIZE that the VM instance needs in the system DS
*/
long long system_ds_size ( ) ;
static long long system_ds_size ( Template * ds_tmpl ) ;
/**
* Completes the information of the disks ( IMAGE_ID , SIZE . . . )
*/
void extended_info ( int uid ) ;
static void extended_info ( int uid , Template * tmpl ) ;
2017-03-30 19:54:58 +03:00
/**
* Computes the storage in the image DS needed for the disks in a VM
* template
* @ param tmpl with DISK descriptions
* @ param ds_quotas templates for quota updates
*/
static void image_ds_quotas ( Template * tmpl , vector < Template * > & ds_quotas ) ;
2016-12-11 23:05:07 +03:00
/**
* Sets Datastore information on volatile disks
*/
bool volatile_info ( int ds_id ) ;
/**
* @ return the total disk SIZE that the VM instance needs in the system DS
*/
void image_ds_size ( std : : map < int , long long > & ds_size ) const ;
/**
* Gets the IDs of the images associated to the disk set
* @ param ids set of image ids
* @ param uid effective user id making the call
*/
void get_image_ids ( set < int > & ids , int uid ) ;
/* ---------------------------------------------------------------------- */
/* Image Manager Interface */
/* ---------------------------------------------------------------------- */
/**
* Get all disk images for this Virtual Machine
* @ param vm_id of the VirtualMachine
* @ param uid of owner
* @ param disks list of DISK Attribute in VirtualMachine Template
* @ param context attribute , 0 if none
* @ param error_str Returns the error reason , if any
* @ return 0 if success
*/
int get_images ( int vm_id , int uid , vector < Attribute * > disks ,
VectorAttribute * context , std : : string & error_str ) ;
/**
* Release the images in the disk set
* @ param vmid id of VM
2016-12-17 04:49:14 +03:00
* @ param img_error true if the image has to be set in error state
* @ param quotas disk space usage to free from image datastores
2016-12-11 23:05:07 +03:00
*/
2017-03-30 19:54:58 +03:00
void release_images ( int vmid , bool img_error , vector < Template * > & quotas ) ;
2016-12-11 23:05:07 +03:00
/* ---------------------------------------------------------------------- */
/* DISK cloning functions */
/* ---------------------------------------------------------------------- */
/**
* Returns true if any of the disks is waiting for an image in LOCKED state
* to become READY
* @ return true if cloning
*/
bool has_cloning ( ) ;
/**
* Returns the image IDs for the disks waiting for the LOCKED state be READY
* @ param ids image ID set
*/
void get_cloning_image_ids ( std : : set < int > & ids ) ;
/**
* Clears the flag for the disks waiting for the given image
*/
void clear_cloning_image_id ( int image_id , const string & source ) ;
/* ---------------------------------------------------------------------- */
/* Attach disk Interface */
/* ---------------------------------------------------------------------- */
2016-12-12 12:26:55 +03:00
/**
* Clear attach status from the attach disk ( ATTACH = YES )
*/
VirtualMachineDisk * delete_attach ( )
{
return static_cast < VirtualMachineDisk * > ( remove_attribute ( " ATTACH " ) ) ;
}
/**
* Get the attach disk ( ATTACH = YES )
*/
VirtualMachineDisk * get_attach ( )
{
return static_cast < VirtualMachineDisk * > ( get_attribute ( " ATTACH " ) ) ;
}
2016-12-11 23:05:07 +03:00
/**
* Sets the attach attribute to the given disk
* @ param disk_id of the DISK
* @ return 0 if the disk_id was found - 1 otherwise
*/
int set_attach ( int disk_id ) ;
/**
* Cleans the attach attribute from the disk
*/
void clear_attach ( )
{
clear_flag ( " ATTACH " ) ;
}
/**
* Prepares a disk to be attached to the virtual machine and adds it to the
* disk set . It checks target assigment and cluster compatibility .
* @ param vmid id of virtual machine
* @ param uid of VM owner
* @ param cluster_id where the VM is running
* @ param vdisk VectorAttribute for the new disk
* @ param vcontext VectorAttribute for the CONTEXT disk , 0 if none
* @ param error
*
* @ return Pointer to the new disk or 0 in case of error
*/
VirtualMachineDisk * set_up_attach ( int vmid , int uid , int cluster_id ,
VectorAttribute * vdisk , VectorAttribute * vcontext , string & error ) ;
2016-12-12 12:26:55 +03:00
/* ---------------------------------------------------------------------- */
/* Save as Interface */
/* ---------------------------------------------------------------------- */
/**
* Get the saveas disk ( HOTPLUG_SAVE_AS_ACTIVE = YES )
*/
VirtualMachineDisk * get_saveas ( )
{
return static_cast < VirtualMachineDisk * > (
get_attribute ( " HOTPLUG_SAVE_AS_ACTIVE " ) ) ;
}
/**
* Mark the disk that is going to be " save as "
* @ param disk_id of the VM
* @ param snap_id of the disk to save , - 1 to select the active snapshot
* @ param iid The image id used by the disk
* @ param size The disk size . This may be different to the original
* image size
* @ param err_str describing the error if any
* @ return - 1 if the image cannot saveas , 0 on success
*/
int set_saveas ( int disk_id , int snap_id , int & iid , long long & size ,
string & err_str ) ;
/**
* Set save attributes for the disk
* @ param disk_id Index of the disk to save
* @ param source to save the disk
* @ param img_id ID of the image this disk will be saved to
*/
int set_saveas ( int disk_id , const string & source , int iid ) ;
/**
* Clears the SAVE_AS_ * attributes of the disk being saved as
* @ return the ID of the image this disk will be saved to or - 1 if it
* is not found .
*/
int clear_saveas ( ) ;
/**
* Get the original image id of the disk . It also checks that the disk can
* be saved_as .
* @ param disk_id Index of the disk to save
* @ param source of the image to save the disk to
* @ param image_id of the image to save the disk to
* @ param tm_mad in use by the disk
* @ param ds_id of the datastore in use by the disk
* @ return - 1 if failure
*/
int get_saveas_info ( int & disk_id , string & source , int & image_id ,
string & snap_id , string & tm_mad , string & ds_id ) ;
2016-12-11 23:05:07 +03:00
/* ---------------------------------------------------------------------- */
/* Resize disk Interface */
/* ---------------------------------------------------------------------- */
2016-12-17 04:49:14 +03:00
VirtualMachineDisk * get_resize ( )
{
return static_cast < VirtualMachineDisk * > ( get_attribute ( " RESIZE " ) ) ;
}
2016-12-11 23:05:07 +03:00
VirtualMachineDisk * delete_resize ( )
{
return static_cast < VirtualMachineDisk * > ( remove_attribute ( " RESIZE " ) ) ;
}
2016-12-17 04:49:14 +03:00
/**
* Sets the resize attribute to the given disk
* @ param disk_id of the DISK
* @ return 0 if the disk_id was found - 1 otherwise
*/
int set_resize ( int disk_id ) ;
2016-12-11 23:05:07 +03:00
2016-12-12 12:26:55 +03:00
/**
* Prepares a disk to be resized .
* @ param disk_id of disk
* @ param size new size for the disk ( needs to be greater than current )
* @ param error
*
* @ return 0 on success
*/
int set_up_resize ( int disk_id , long size , string & error ) ;
2016-12-11 23:05:07 +03:00
/* ---------------------------------------------------------------------- */
/* SNAPSHOT interface */
/* ---------------------------------------------------------------------- */
2016-12-12 12:26:55 +03:00
VirtualMachineDisk * get_active_snapshot ( )
{
return static_cast < VirtualMachineDisk * > (
get_attribute ( " DISK_SNAPSHOT_ACTIVE " ) ) ;
}
2016-12-11 23:05:07 +03:00
/**
* Set the snapshots for a disk
* @ param id of disk
* @ param snapshots of disk ;
*/
void set_snapshots ( int id , Snapshots * snapshots ) ;
/**
* Return the snapshots for the disk
*/
const Snapshots * get_snapshots ( int id , string & error ) const ;
/**
* Set the disk as being snapshotted ( reverted . . . )
* @ param disk_id of the disk
* @ param snap_id of the target snap_id
*/
int set_active_snapshot ( int id , int snap_id ) ;
/**
* Unset the current disk being snapshotted ( reverted . . . )
*/
void clear_active_snapshot ( ) ;
/**
* Get information about the disk to take the snapshot from
* @ param ds_id id of the datastore
* @ param tm_mad used by the datastore
* @ param disk_id of the disk
* @ param snap_id of the snapshot
*/
int get_active_snapshot ( int & ds_id , string & tm_mad , int & disk_id ,
int & snap_id ) ;
/**
* Creates a new snapshot of the given disk
* @ param disk_id of the disk
* @ param name a description for this snapshot
* @ param error if any
* @ return the id of the new snapshot or - 1 if error
*/
int create_snapshot ( int disk_id , const string & name , string & error ) ;
/**
* Sets the snap_id as active , the VM will boot from it next time
* @ param disk_id of the disk
* @ param snap_id of the snapshot
* @ return - 1 if error
*/
int revert_snapshot ( int disk_id , int snap_id ) ;
/**
* Deletes the snap_id from the list
* @ param disk_id of the disk
* @ param snap_id of the snapshot
* @ param ds_quotas template with snapshot usage for the DS quotas
* @ param vm_quotas template with snapshot usage for the VM quotas
2017-03-31 21:09:27 +03:00
* @ param io delete ds quotas from image owners
* @ param vo delete ds quotas from vm owners
2016-12-11 23:05:07 +03:00
*/
void delete_snapshot ( int disk_id , int snap_id , Template * * ds_quota ,
2017-03-31 21:09:27 +03:00
Template * * vm_quota , bool & io , bool & vo ) ;
2016-12-11 23:05:07 +03:00
/**
* Deletes all the disk snapshots for non - persistent disks and for persistent
* disks in no shared system ds .
* @ param vm_quotas The SYSTEM_DISK_SIZE freed by the deleted snapshots
* @ param ds_quotas The DS SIZE freed from image datastores .
*/
void delete_non_persistent_snapshots ( Template * * vm_quotas ,
2017-03-31 21:09:27 +03:00
vector < Template * > & ds_quotas ) ;
2016-12-11 23:05:07 +03:00
2016-12-17 04:49:14 +03:00
/**
* Restores the disk original size for non - persistent and for persistent
* disks in no shared system ds .
* @ param vm_quotas The SYSTEM_DISK_SIZE freed by the deleted snapshots
* @ param ds_quotas The DS SIZE freed from image datastores .
*/
void delete_non_persistent_resizes ( Template * * vm_quotas ,
2017-03-31 21:09:27 +03:00
vector < Template * > & ds_quotas ) ;
2016-12-17 04:49:14 +03:00
2016-12-11 23:05:07 +03:00
protected :
VirtualMachineAttribute * attribute_factory ( VectorAttribute * va ,
int id ) const
{
return new VirtualMachineDisk ( va , id ) ;
} ;
private :
static const char * DISK_NAME ; //"DISK"
static const char * DISK_ID_NAME ; //"DISK_ID"
/**
* Finds the first free target to assign to a disk
* @ param dqueue queue of disks to assign target , each disk is associated
* with its bus ( DEV_PREFIX )
* @ param used_targets that cannot be used
*/
void assign_disk_targets (
std : : queue < pair < std : : string , VirtualMachineDisk * > > & dqueue ,
std : : set < std : : string > & used_targets ) ;
} ;
# endif /*VIRTUAL_MACHINE_DISK_H_*/