1
0
mirror of https://github.com/OpenNebula/one.git synced 2024-12-23 17:33:56 +03:00
one/include/Quota.h
Pavel Czerný 442041fd07
F #6341: Generic Quotas
- Generic quotas are defined and read from oned.conf (QUOTA_VM_ATTRIBUTE)
- Generic quotas react to the running mode (i.e. account only when the VM is
  running). For each quota oned also adds a RUNNIN_* metric
- one.system.config API call has been updated so every user can read
  quota configuration:
    . Sensitive information is hidden (***)
    . Oneadmin group has no longer access to sensitive information
      through the API
- CLI has been updated to render generic quotas
- onedb fsck fixes generic quotas reading the configuration from
  oned.conf

Other changes in this PR:
- Refactor Quota metrics to use std::vector instead of C array
Squashed commit of the following:
- New methods to Template and Attribute classes to render hidden
  attributes as "***"
- Update Quota calls to not include unneeded quotas

co-authored-by:Ruben S. Montero <rsmontero@opennebula.org>
2024-01-08 14:07:03 +01:00

349 lines
11 KiB
C++

/* -------------------------------------------------------------------------- */
/* Copyright 2002-2023, 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 QUOTA_H_
#define QUOTA_H_
#include "Template.h"
// Forward declaration to avoid include cycle
class Quotas;
/**
* This class defines the public interface (pure abstract) for Quota.
*/
class QuotaInterface
{
public:
/**
* Check if the resource allocation will exceed the quota limits. If not
* the usage counters are updated
* @param tmpl template for the resource
* @param default_quotas Quotas that contain the default limits
* @param error string
* @return true if the operation can be performed
*/
virtual bool check(Template* tmpl, Quotas& default_quotas, std::string& error) = 0;
/**
* Decrement usage counters
* @param tmpl template for the resource
*/
virtual void del(Template* tmpl) = 0;
/**
* Set the quotas. If the quota previously exists its limit is updated.
* @param quota_str the quota template in ASCII or XML formats
* @param error describe the error in case of error
*
* @return 0 on success -1 otherwise
*/
virtual int set(std::vector<VectorAttribute*> * quotas, std::string& error) = 0;
/**
* Check if a resource update in usage counters will exceed the
* quota limits. If not the usage counters are updated for that resource
* @param tmpl with increments in MEMORY and CPU
* @param default_quotas Quotas that contain the default limits
* @param error string
* @return true if the operation can be performed
*/
virtual bool update(Template * tmpl, Quotas& default_quotas, std::string& error) = 0;
/**
* Returns the name that identifies the quota in a template
*/
virtual const char * get_quota_name() const = 0;
/**
* Gets a quota identified by its ID.
* @param id of the quota
* @param va The quota, if it is found
* @return 0 on success, -1 if not found
*/
virtual int get_quota(const std::string& id, VectorAttribute **va) = 0;
protected:
QuotaInterface(){};
virtual ~QuotaInterface(){};
};
/**
* The QuotaDecorator class decorates the quota interface with specific
* behavior over the same quota object and counters. Decorators must be derived
* from this class and pass in its constructor the decorated object.
*/
class QuotaDecorator : public QuotaInterface
{
bool check(Template* tmpl, Quotas& default_quotas, std::string& error) override
{
return quota->check(tmpl, default_quotas, error);
}
void del(Template* tmpl) override
{
return quota->del(tmpl);
}
int set(std::vector<VectorAttribute*> * quotas, std::string& error) override
{
return quota->set(quotas, error);
}
bool update(Template * tmpl, Quotas& default_quotas, std::string& error) override
{
return quota->update(tmpl, default_quotas, error);
}
const char * get_quota_name() const override
{
return quota->get_quota_name();
}
int get_quota(const std::string& id, VectorAttribute **va) override
{
return quota->get_quota(id, va);
}
protected:
QuotaDecorator(QuotaInterface * _quota):quota(_quota){};
virtual ~QuotaDecorator(){};
QuotaInterface * quota;
};
/**
* Base class for resource quotas, it provides basic storage and management of
* the quotas. Each resource MUST inherit from it to implement check and
* update methods. Quotas are stored in a template form, each class store the
* limits and usage in a resource specific format.
*/
class Quota: public Template, public QuotaInterface
{
public:
/**
* Set the quotas. If the quota previously exists its limit is updated.
* @param quota_str the quota template in ASCII or XML formats
* @param error describe the error in case of error
*
* @return 0 on success -1 otherwise
*/
int set(std::vector<VectorAttribute*> * quotas, std::string& error) override;
/**
* Check if a resource update in usage counters will exceed the
* quota limits. If not the usage counters are updated for that resource
* @param tmpl with increments in MEMORY and CPU
* @param default_quotas Quotas that contain the default limits
* @param error string
* @return true if the operation can be performed
*/
bool update(Template * tmpl, Quotas& default_quotas, std::string& error) override
{
error = "Update operation for quotas not supported.";
return false;
};
/**
* Returns the name that identifies the quota in a template
*/
const char * get_quota_name() const override
{
return template_name;
}
/**
* Gets a quota identified by its ID.
* @param id of the quota
* @param va The quota, if it is found
* @return 0 on success, -1 if not found
*/
int get_quota(const std::string& id, VectorAttribute **va) override
{
std::map<std::string, Attribute *>::iterator it;
return get_quota(id, va, it);
}
/**
* Value for limit default
*/
static const int DEFAULT;
static const std::string DEFAULT_STR;
/**
* Value for "unlimited" limit
*/
static const int UNLIMITED;
protected:
Quota(const char * quota_name,
const char * _template_name,
const std::vector<std::string>& _metrics,
bool _is_default)
: Template(false, '=', quota_name),
template_name(_template_name),
metrics(_metrics),
is_default(_is_default){};
virtual ~Quota(){};
/**
* Generic Quota Names
*
* template_name = [
* ID = "ID to identify the resource",
* metrics[0] = "Limit for the first metric"
* metrics[0]_USED = "Usage for metric"
* ]
*
* ID & counter fields are optional
*/
/**
* Name of the quota used in the templates
*/
const char * template_name;
/**
* The name of the quota metrics
*/
const std::vector<std::string>& metrics;
/**
* Whether or not this is a default quota. Default quotas do not have usage,
* and can't have a limit of -1
*/
bool is_default;
/**
* Check a given quota for an usage request and update counters if the
* request does not exceed quota limits
* @param qid id that identifies the quota, to be used by get_quota
* @param usage_req usage for each metric
* @param default_quotas Quotas that contain the default limits
* @param error string describing the error
* @return true if the request does not exceed current limits
*/
bool check_quota(const std::string& qid,
std::map<std::string, float>& usage_req,
Quotas& default_quotas,
std::string& error);
/**
* Add usage for a given quota without checking the limits
* @param qid id that identifies the quota, to be used by get_quota
* @param usage_req usage for each metric
*/
void add_quota(const std::string& qid,
std::map<std::string, float>& usage_req);
/**
* Reduce usage from a given quota based on the current consumption
* @param qid id that identifies the quota, to be used by get_quota
* @param usage_req usage for each metric
*/
void del_quota(const std::string& qid,
std::map<std::string, float>& usage_req);
/**
* Gets the default quota identified by its ID.
*
* @param id of the quota
* @param default_quotas Quotas that contain the default limits
* @param va The quota, if it is found
*
* @return 0 on success, -1 if not found
*/
virtual int get_default_quota(const std::string& id,
Quotas& default_quotas,
VectorAttribute **va) = 0;
/**
* Gets a quota identified by its ID.
*
* @param id of the quota
* @param va The quota, if it is found
* @param it The quota iterator, if it is found
*
* @return 0 on success, -1 if not found
*/
virtual int get_quota(
const std::string& id,
VectorAttribute **va,
std::map<std::string, Attribute *>::iterator& it);
/**
* Checks if a quota has 0 limit and usage, and deletes it
*
* @param qid id of the quota
*/
void cleanup_quota(const std::string& qid);
private:
/**
* Creates an empty quota based on the given attribute. The attribute va
* contains the limits for the quota.
* @param va limits for the new quota if 0 limits will be 0
*
* @return a new attribute representing the quota, 0 on error
*/
VectorAttribute * new_quota(const VectorAttribute* va);
/**
* Adds a new quota, it also updates an internal index for fast accessing
* the quotas
* @param quota is the new quota, allocated in the HEAP
*/
void add(VectorAttribute * nq)
{
attributes.insert(make_pair(nq->name(), nq));
}
/**
* Adds a given value to the current quota (vector)
* @param attr the quota;
* @param va_name name of the quota in the vector attribute
* @param num value to add to the current quota;
*/
void add_to_quota(VectorAttribute * attr,
const std::string& va_name,
float num);
/**
* Sets new limit values for the quota
* @param quota to be updated
* @param va attribute with the new limits
*
* @return 0 on success or -1 if wrong limits
*/
int update_limits(VectorAttribute* quota,
const VectorAttribute* va);
/**
* Extract the limits for the defined quota metrics from a given attribute
* @param va the attribute with the limits
* @param limits stores the known limits
* @return 0 on success
*/
int get_limits(const VectorAttribute * va,
std::map<std::string, std::string>& limits) const;
};
#endif /*QUOTA_H_*/