/* ------------------------------------------------------------------------ */ /* Copyright 2002-2018, 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 HOST_H_ #define HOST_H_ #include "PoolSQL.h" #include "HostTemplate.h" #include "HostShare.h" #include "ClusterableSingle.h" #include "ObjectCollection.h" #include "NebulaLog.h" #include "NebulaUtil.h" using namespace std; /** * The Host class. */ class Host : public PoolObjectSQL, public ClusterableSingle { public: // HOST STATES +----------------+ // | VM DEPLOYMENT | // +----------------+------------+--------+-------+ // | STATE | MONITORING | MANUAL | SCHED | // +----------------+------------+--------+-------+ // | INIT/MONITORED | Yes | Yes | // +----------------+------------+--------+-------+ // | DISABLED | Yes | Yes | No | // +----------------+------------+----------------+ // | OFFLINE | No | No | // +----------------+-----------------------------+ enum HostState { INIT = 0, /**< Initial state for enabled hosts. */ MONITORING_MONITORED = 1, /**< Monitoring the host (from monitored). */ MONITORED = 2, /**< The host has been monitored. */ ERROR = 3, /**< An error ocurrer in host monitoring. */ DISABLED = 4, /**< The host is disabled see above. */ MONITORING_ERROR = 5, /**< Monitoring the host (from error). */ MONITORING_INIT = 6, /**< Monitoring the host (from init). */ MONITORING_DISABLED = 7, /**< Monitoring the host (from disabled). */ OFFLINE = 8 /**< The host is set offline, see above */ }; /** * Function to print the Host object into a string in XML format * @param xml the resulting XML string * @return a reference to the generated string */ string& to_xml(string& xml) const; /** * Rebuilds the object from an xml formatted string * @param xml_str The xml-formatted string * * @return 0 on success, -1 otherwise */ int from_xml(const string &xml_str); /** * Checks if the host is a remote public cloud * @return true if the host is a remote public cloud */ bool is_public_cloud() const; /** * Sets the current host offline, it will not be monitored nor used by the * scheduler, manual VM deployment is also restricted */ void offline(); /** * Sets the current host disable, it will receive monitor updates, manual * deployment of VMs is allowed and the scheduler will not consider this * host. */ void disable(); /** * Enables the current host, it will be monitored and could be used by * the scheduler */ void enable(); /** * Sets the host in error */ void set_error() { state = ERROR; } /** * Updates the Host's last_monitored time stamp. * @param success if the monitored action was successfully performed */ void touch(bool success) { last_monitored = time(0); switch (state) { case OFFLINE: state = OFFLINE; break; case DISABLED: case MONITORING_DISABLED: state = DISABLED; break; case INIT: case ERROR: case MONITORED: case MONITORING_ERROR: case MONITORING_INIT: case MONITORING_MONITORED: if (success == true) { state = MONITORED; } else { state = ERROR; } break; } }; /** * Update host after a successful monitor. It modifies counters, state * and template attributes * @param parse_str string with values to be parsed * @param with_vm_info if monitoring contains VM information * @param lost set of VMs that should be in the host and were not found * @param found VMs running in the host (as expected) and info. * @param reserved_cpu from cluster defaults * @param reserved_mem from cluster defaults * @return 0 on success **/ int update_info(Template &tmpl, bool &with_vm_info, set &lost, map &found, const set &non_shared_ds, const string& reserved_cpu, const string& reserved_mem); /** * Extracts the DS attributes from the given template * @param parse_str string with values to be parsed * @param ds map of DS monitoring information * @param template object parsed from parse_str * * @return 0 on success */ int extract_ds_info( string &parse_str, Template &tmpl, map &ds); /** * Update host after a failed monitor. It state * and template attributes * @param message from the driver * @param vm_ids running on the host */ void error_info(const string& message, set &vm_ids); /** * Inserts the last monitoring, and deletes old monitoring entries. * * @param db pointer to the db * @return 0 on success */ int update_monitoring(SqlDB * db); /** * Retrieves host state * @return HostState code number */ HostState get_state() const { return state; }; /** * Retrieves VMM mad name * @return string vmm mad name */ const string& get_vmm_mad() const { return vmm_mad_name; }; /** * Retrieves IM mad name * @return string im mad name */ const string& get_im_mad() const { return im_mad_name; }; /** * Sets the corresponding monitoring state based on the actual host state */ void set_monitoring_state() { last_monitored = time(0); //Needed to expire this monitor action switch (state) { case ERROR: state = MONITORING_ERROR; break; case MONITORED: state = MONITORING_MONITORED; break; case INIT: state = MONITORING_INIT; break; case DISABLED: state = MONITORING_DISABLED; break; default: break; } }; /** * Retrieves last time the host was monitored * @return time_t last monitored time */ time_t get_last_monitored() const { return last_monitored; }; /** * Get the reserved capacity for this host. Parameters will be only updated * if values are defined in the host. Reserved capacity will be subtracted * from the Host total capacity. * @param cpu reserved cpu (in percentage) * @param mem reserved mem (in KB) */ void get_reserved_capacity(string& cpu, string& mem) { get_template_attribute("RESERVED_CPU", cpu); get_template_attribute("RESERVED_MEM", mem); } // ------------------------------------------------------------------------- // Share functions. // ------------------------------------------------------------------------- long long get_share_running_vms() { return host_share.get_running_vms(); } /** * Adds a new VM to the given share by incrementing the cpu, mem and disk * counters * @param vm_id id of the vm to add to the host * @param cpu needed by the VM (percentage) * @param mem needed by the VM (in KB) * @param disk needed by the VM * @param pci devices needed by th VM * @return 0 on success */ void add_capacity(int vm_id, long long cpu, long long mem, long long disk, vector pci) { if ( vm_collection.add(vm_id) == 0 ) { host_share.add(vm_id, cpu, mem, disk, pci); } else { ostringstream oss; oss << "Trying to add VM " << vm_id << ", that it is already associated to host " << oid << "."; NebulaLog::log("ONE", Log::ERROR, oss); } }; /** * Deletes a new VM from the given share by decrementing the cpu,mem and * disk counters * @param vm_id id of the vm to delete from the host * @param cpu used by the VM (percentage) * @param mem used by the VM (in KB) * @param disk used by the VM * @param pci devices needed by th VM * @return 0 on success */ void del_capacity(int vm_id, long long cpu, long long mem, long long disk, const vector& pci) { if ( vm_collection.del(vm_id) == 0 ) { host_share.del(cpu, mem, disk, pci); } else { ostringstream oss; oss << "Trying to remove VM " << vm_id << ", that it is not associated to host " << oid << "."; NebulaLog::log("ONE", Log::ERROR, oss); } }; /** * Updates the capacity used in a host when a VM is resized * counters * @param cpu increment of cpu requested by the VM * @param mem increment of memory requested by the VM * @param disk not used * @return 0 on success */ void update_capacity(int cpu, long int mem, int disk) { host_share.update(cpu,mem,disk); }; /** * Tests whether a new VM can be hosted by the host or not * @param cpu needed by the VM (percentage) * @param mem needed by the VM (in Kb) * @param disk needed by the VM * @param pci devices needed by the VM * @param error Returns the error reason, if any * @return true if the share can host the VM */ bool test_capacity(long long cpu, long long mem, long long disk, vector &pci, string& error) const { return host_share.test(cpu, mem, disk, pci, error); } /** * Tests whether a new VM can be hosted by the host or not, checking the * PCI devices only. * @param pci devices needed by the VM * @param error Returns the error reason, if any * @return true if the share can host the VM */ bool test_capacity(vector &pci, string& error) const { return host_share.test(pci, error); } /** * Returns a copy of the VM IDs set */ set get_vm_ids() { return vm_collection.clone(); } /** * Factory method for host templates */ Template * get_new_template() const { return new HostTemplate; } /** * Executed after an update operation to process the new template * - encrypt VCENTER_PASSWORD attribute. */ int post_update_template(string& error); /** * Returns the rediscovered VMs (from poff to running) in the previous * monitorization cycle * @return the previous rediscovered VMs (from poff to running) */ const set& get_prev_rediscovered_vms() const { return *prev_rediscovered_vms; } /** * Sets the previous rediscovered VMs (from poff to running). This set * is not stored in the DB, the pool update method is not needed * @param rediscovered_vms the previous rediscovered VMs (from poff to running) */ void set_prev_rediscovered_vms(const set& rediscovered_vms) { *prev_rediscovered_vms = rediscovered_vms; } private: // ------------------------------------------------------------------------- // Friends // ------------------------------------------------------------------------- friend class HostPool; // ------------------------------------------------------------------------- // Host Description // ------------------------------------------------------------------------- /** * The state of the Host */ HostState state; /** * Name of the IM driver used to monitor this host */ string im_mad_name; /** * Name of the VM driver used to execute VMs in this host */ string vmm_mad_name; /** * If Host State = MONITORED last time it got fully monitored or 1 Jan 1970 * Host State = MONITORING* last time it got a signal to be monitored */ time_t last_monitored; // ------------------------------------------------------------------------- // Host Attributes // ------------------------------------------------------------------------- /** * The Share represents the logical capacity associated with the host */ HostShare host_share; /** * Tmp set of lost VM IDs. Used to give lost VMs one grace cycle, in case * they reappear. */ set * tmp_lost_vms; /** * Tmp set of zombie VM IDs. Used to give zombie VMs one grace cycle, in * case they are cleaned. */ set * tmp_zombie_vms; /** * Set that stores the VMs reported as found from the poweroff state. This * is managed from outside the host to avoid deadlocks, as the current * VM state is needed */ set * prev_rediscovered_vms; // ------------------------------------------------------------------------- // VM Collection // ------------------------------------------------------------------------- /** * Stores a collection with the VMs running in the host */ ObjectCollection vm_collection; // ************************************************************************* // Constructor // ************************************************************************* Host(int id, const string& hostname, const string& im_mad_name, const string& vmm_mad_name, int cluster_id, const string& cluster_name); virtual ~Host(); // ************************************************************************* // DataBase implementation (Private) // ************************************************************************* static const char * db_names; static const char * db_bootstrap; static const char * table; static const char * monit_db_names; static const char * monit_db_bootstrap; static const char * monit_table; /** * Execute an INSERT or REPLACE Sql query. * @param db The SQL DB * @param replace Execute an INSERT or a REPLACE * @param error_str Returns the error reason, if any * @return 0 one success */ int insert_replace(SqlDB *db, bool replace, string& error_str); /** * Bootstraps the database table(s) associated to the Host * @return 0 on success */ static int bootstrap(SqlDB * db) { int rc; ostringstream oss_host(Host::db_bootstrap); ostringstream oss_monit(Host::monit_db_bootstrap); rc = db->exec_local_wr(oss_host); rc += db->exec_local_wr(oss_monit); return rc; }; /** * Writes the Host and its associated HostShares in the database. * @param db pointer to the db * @return 0 on success */ int insert(SqlDB *db, string& error_str) { return insert_replace(db, false, error_str); }; /** * Writes/updates the Hosts data fields in the database. * @param db pointer to the db * @return 0 on success */ int update(SqlDB *db) { string error_str; return insert_replace(db, true, error_str); }; }; #endif /*HOST_H_*/