mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-23 04:51:39 +08:00
182 lines
4.7 KiB
Go
182 lines
4.7 KiB
Go
// Copyright 2019 Yunion
|
|
//
|
|
// 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.
|
|
|
|
package models
|
|
|
|
import (
|
|
"context"
|
|
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/pkg/util/rbacscope"
|
|
|
|
identityapi "yunion.io/x/onecloud/pkg/apis/identity"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db/quotas"
|
|
commonOptions "yunion.io/x/onecloud/pkg/cloudcommon/options"
|
|
"yunion.io/x/onecloud/pkg/compute/options"
|
|
"yunion.io/x/onecloud/pkg/mcclient/auth"
|
|
)
|
|
|
|
var (
|
|
ProjectQuota SProjectQuota
|
|
ProjectQuotaManager *SQuotaManager
|
|
ProjectUsageManager *SQuotaManager
|
|
ProjectPendingUsageManager *SQuotaManager
|
|
)
|
|
|
|
func init() {
|
|
ProjectQuota = SProjectQuota{}
|
|
|
|
ProjectUsageManager = &SQuotaManager{
|
|
SQuotaBaseManager: quotas.NewQuotaUsageManager(ProjectQuota,
|
|
rbacscope.ScopeProject,
|
|
"project_quota_usage_tbl",
|
|
"project_quota_usage",
|
|
"project_quota_usages",
|
|
),
|
|
}
|
|
ProjectPendingUsageManager = &SQuotaManager{
|
|
SQuotaBaseManager: quotas.NewQuotaUsageManager(ProjectQuota,
|
|
rbacscope.ScopeProject,
|
|
"project_quota_pending_usage_tbl",
|
|
"project_quota_pending_usage",
|
|
"project_quota_pending_usages",
|
|
),
|
|
}
|
|
ProjectQuotaManager = &SQuotaManager{
|
|
SQuotaBaseManager: quotas.NewQuotaBaseManager(ProjectQuota,
|
|
rbacscope.ScopeProject,
|
|
"project_quota_tbl",
|
|
ProjectPendingUsageManager,
|
|
ProjectUsageManager,
|
|
"project_quota",
|
|
"project_quotas",
|
|
),
|
|
}
|
|
quotas.Register(ProjectQuotaManager)
|
|
}
|
|
|
|
type SProjectQuota struct {
|
|
quotas.SQuotaBase
|
|
|
|
quotas.SBaseProjectQuotaKeys
|
|
|
|
Secgroup int `default:"-1" allow_zero:"true" json:"secgroup"`
|
|
}
|
|
|
|
func (self *SProjectQuota) GetKeys() quotas.IQuotaKeys {
|
|
return self.SBaseProjectQuotaKeys
|
|
}
|
|
|
|
func (self *SProjectQuota) SetKeys(keys quotas.IQuotaKeys) {
|
|
self.SBaseProjectQuotaKeys = keys.(quotas.SBaseProjectQuotaKeys)
|
|
}
|
|
|
|
func (self *SProjectQuota) FetchSystemQuota() {
|
|
keys := self.SBaseProjectQuotaKeys
|
|
base := 0
|
|
switch options.Options.DefaultQuotaValue {
|
|
case commonOptions.DefaultQuotaUnlimit:
|
|
base = -1
|
|
case commonOptions.DefaultQuotaZero:
|
|
base = 0
|
|
if keys.Scope() == rbacscope.ScopeDomain { // domain level quota
|
|
base = 10
|
|
} else if keys.DomainId == identityapi.DEFAULT_DOMAIN_ID && keys.ProjectId == auth.AdminCredential().GetProjectId() {
|
|
base = 1
|
|
}
|
|
case commonOptions.DefaultQuotaDefault:
|
|
base = 1
|
|
if keys.Scope() == rbacscope.ScopeDomain {
|
|
base = 10
|
|
}
|
|
}
|
|
defaultValue := func(def int) int {
|
|
if base < 0 {
|
|
return -1
|
|
} else {
|
|
return def * base
|
|
}
|
|
}
|
|
self.Secgroup = defaultValue(options.Options.DefaultSecgroupQuota)
|
|
}
|
|
|
|
func (self *SProjectQuota) FetchUsage(ctx context.Context) error {
|
|
regionKeys := self.SBaseProjectQuotaKeys
|
|
|
|
scope := regionKeys.Scope()
|
|
ownerId := regionKeys.OwnerId()
|
|
|
|
self.Secgroup, _ = totalSecurityGroupCount(scope, ownerId)
|
|
return nil
|
|
}
|
|
|
|
func (self *SProjectQuota) ResetNegative() {
|
|
if self.Secgroup < 0 {
|
|
self.Secgroup = 0
|
|
}
|
|
}
|
|
|
|
func (self *SProjectQuota) IsEmpty() bool {
|
|
if self.Secgroup > 0 {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (self *SProjectQuota) Add(quota quotas.IQuota) {
|
|
squota := quota.(*SProjectQuota)
|
|
self.Secgroup = self.Secgroup + quotas.NonNegative(squota.Secgroup)
|
|
}
|
|
|
|
func (self *SProjectQuota) Sub(quota quotas.IQuota) {
|
|
squota := quota.(*SProjectQuota)
|
|
self.Secgroup = nonNegative(self.Secgroup - squota.Secgroup)
|
|
}
|
|
|
|
func (self *SProjectQuota) Allocable(request quotas.IQuota) int {
|
|
squota := request.(*SProjectQuota)
|
|
cnt := -1
|
|
if self.Secgroup >= 0 && squota.Secgroup > 0 && (cnt < 0 || cnt > self.Secgroup/squota.Secgroup) {
|
|
cnt = self.Secgroup / squota.Secgroup
|
|
}
|
|
return cnt
|
|
}
|
|
|
|
func (self *SProjectQuota) Update(quota quotas.IQuota) {
|
|
squota := quota.(*SProjectQuota)
|
|
if squota.Secgroup > 0 {
|
|
self.Secgroup = squota.Secgroup
|
|
}
|
|
}
|
|
|
|
func (used *SProjectQuota) Exceed(request quotas.IQuota, quota quotas.IQuota) error {
|
|
err := quotas.NewOutOfQuotaError()
|
|
sreq := request.(*SProjectQuota)
|
|
squota := quota.(*SProjectQuota)
|
|
if quotas.Exceed(used.Secgroup, sreq.Secgroup, squota.Secgroup) {
|
|
err.Add(used, "secgroup", squota.Secgroup, used.Secgroup, sreq.Secgroup)
|
|
}
|
|
if err.IsError() {
|
|
return err
|
|
} else {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (self *SProjectQuota) ToJSON(prefix string) jsonutils.JSONObject {
|
|
ret := jsonutils.NewDict()
|
|
ret.Add(jsonutils.NewInt(int64(self.Secgroup)), keyName(prefix, "secgroup"))
|
|
return ret
|
|
}
|