mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-06-05 10:41:14 +08:00
702 lines
25 KiB
Go
702 lines
25 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"
|
|
"database/sql"
|
|
"fmt"
|
|
|
|
"yunion.io/x/cloudmux/pkg/cloudprovider"
|
|
"yunion.io/x/jsonutils"
|
|
"yunion.io/x/log"
|
|
"yunion.io/x/pkg/errors"
|
|
"yunion.io/x/pkg/util/compare"
|
|
"yunion.io/x/pkg/util/netutils"
|
|
"yunion.io/x/pkg/util/rand"
|
|
"yunion.io/x/pkg/util/rbacscope"
|
|
"yunion.io/x/pkg/utils"
|
|
"yunion.io/x/sqlchemy"
|
|
|
|
api "yunion.io/x/onecloud/pkg/apis/compute"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
|
|
"yunion.io/x/onecloud/pkg/cloudcommon/validators"
|
|
"yunion.io/x/onecloud/pkg/httperrors"
|
|
"yunion.io/x/onecloud/pkg/mcclient"
|
|
"yunion.io/x/onecloud/pkg/util/stringutils2"
|
|
)
|
|
|
|
// +onecloud:swagger-gen-model-singular=loadbalancerbackend
|
|
// +onecloud:swagger-gen-model-plural=loadbalancerbackends
|
|
type SLoadbalancerBackendManager struct {
|
|
SLoadbalancerLogSkipper
|
|
db.SStatusStandaloneResourceBaseManager
|
|
db.SExternalizedResourceBaseManager
|
|
SLoadbalancerBackendgroupResourceBaseManager
|
|
}
|
|
|
|
var LoadbalancerBackendManager *SLoadbalancerBackendManager
|
|
|
|
func init() {
|
|
LoadbalancerBackendManager = &SLoadbalancerBackendManager{
|
|
SStatusStandaloneResourceBaseManager: db.NewStatusStandaloneResourceBaseManager(
|
|
SLoadbalancerBackend{},
|
|
"loadbalancerbackends_tbl",
|
|
"loadbalancerbackend",
|
|
"loadbalancerbackends",
|
|
),
|
|
}
|
|
LoadbalancerBackendManager.SetVirtualObject(LoadbalancerBackendManager)
|
|
}
|
|
|
|
type SLoadbalancerBackend struct {
|
|
db.SStatusStandaloneResourceBase
|
|
db.SExternalizedResourceBase
|
|
|
|
SLoadbalancerBackendgroupResourceBase `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional"`
|
|
|
|
BackendId string `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional"`
|
|
BackendType string `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional"`
|
|
BackendRole string `width:"36" charset:"ascii" nullable:"false" list:"user" default:"default" create:"optional"`
|
|
Weight int `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional" update:"user"`
|
|
Address string `width:"36" charset:"ascii" nullable:"true" list:"user" create:"optional"`
|
|
Port int `nullable:"false" list:"user" create:"required" update:"user"`
|
|
|
|
SendProxy string `width:"16" charset:"ascii" nullable:"false" list:"user" create:"optional" update:"user" default:"off"`
|
|
Ssl string `width:"16" charset:"ascii" nullable:"true" list:"user" create:"optional" update:"user" default:"off"`
|
|
}
|
|
|
|
func (manager *SLoadbalancerBackendManager) ResourceScope() rbacscope.TRbacScope {
|
|
return rbacscope.ScopeProject
|
|
}
|
|
|
|
func (self *SLoadbalancerBackend) GetOwnerId() mcclient.IIdentityProvider {
|
|
lbbg, err := self.GetLoadbalancerBackendGroup()
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
return lbbg.GetOwnerId()
|
|
}
|
|
|
|
func (manager *SLoadbalancerBackendManager) FetchOwnerId(ctx context.Context, data jsonutils.JSONObject) (mcclient.IIdentityProvider, error) {
|
|
lbbgId, _ := data.GetString("backend_group_id")
|
|
if len(lbbgId) > 0 {
|
|
lbbg, err := db.FetchById(LoadbalancerBackendGroupManager, lbbgId)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "db.FetchById(LoadbalancerBackendGroupManager, %s)", lbbgId)
|
|
}
|
|
return lbbg.(*SLoadbalancerBackendGroup).GetOwnerId(), nil
|
|
}
|
|
return db.FetchProjectInfo(ctx, data)
|
|
}
|
|
|
|
func (man *SLoadbalancerBackendManager) FilterByOwner(ctx context.Context, q *sqlchemy.SQuery, manager db.FilterByOwnerProvider, userCred mcclient.TokenCredential, ownerId mcclient.IIdentityProvider, scope rbacscope.TRbacScope) *sqlchemy.SQuery {
|
|
if ownerId != nil {
|
|
sq := LoadbalancerBackendGroupManager.Query("id")
|
|
lb := LoadbalancerManager.Query().SubQuery()
|
|
sq = sq.Join(lb, sqlchemy.Equals(sq.Field("loadbalancer_id"), lb.Field("id")))
|
|
switch scope {
|
|
case rbacscope.ScopeProject:
|
|
sq = sq.Filter(sqlchemy.Equals(lb.Field("tenant_id"), ownerId.GetProjectId()))
|
|
return q.In("backend_group_id", sq.SubQuery())
|
|
case rbacscope.ScopeDomain:
|
|
sq = sq.Filter(sqlchemy.Equals(lb.Field("domain_id"), ownerId.GetProjectDomainId()))
|
|
return q.In("backend_group_id", sq.SubQuery())
|
|
}
|
|
}
|
|
return q
|
|
}
|
|
|
|
// 负载均衡后端列表
|
|
func (man *SLoadbalancerBackendManager) ListItemFilter(
|
|
ctx context.Context,
|
|
q *sqlchemy.SQuery,
|
|
userCred mcclient.TokenCredential,
|
|
query api.LoadbalancerBackendListInput,
|
|
) (*sqlchemy.SQuery, error) {
|
|
q, err := man.SStatusStandaloneResourceBaseManager.ListItemFilter(ctx, q, userCred, query.StatusStandaloneResourceListInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SStatusStandaloneResourceBaseManager.ListItemFilter")
|
|
}
|
|
q, err = man.SExternalizedResourceBaseManager.ListItemFilter(ctx, q, userCred, query.ExternalizedResourceBaseListInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SExternalizedResourceBaseManager.ListItemFilter")
|
|
}
|
|
q, err = man.SLoadbalancerBackendgroupResourceBaseManager.ListItemFilter(ctx, q, userCred, query.LoadbalancerBackendGroupFilterListInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SLoadbalancerBackendgroupResourceBaseManager.ListItemFilter")
|
|
}
|
|
|
|
data := jsonutils.Marshal(query).(*jsonutils.JSONDict)
|
|
q, err = validators.ApplyModelFilters(ctx, q, data, []*validators.ModelFilterOptions{
|
|
{Key: "backend", ModelKeyword: "server", OwnerId: userCred}, // NOTE extend this when new backend_type was added
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(query.BackendType) > 0 {
|
|
q = q.In("backend_type", query.BackendType)
|
|
}
|
|
if len(query.BackendRole) > 0 {
|
|
q = q.In("backend_role", query.BackendRole)
|
|
}
|
|
if len(query.Address) > 0 {
|
|
q = q.In("address", query.Address)
|
|
}
|
|
if len(query.SendProxy) > 0 {
|
|
q = q.In("send_proxy", query.SendProxy)
|
|
}
|
|
if len(query.Ssl) > 0 {
|
|
q = q.In("ssl", query.Ssl)
|
|
}
|
|
|
|
return q, nil
|
|
}
|
|
|
|
func (man *SLoadbalancerBackendManager) OrderByExtraFields(
|
|
ctx context.Context,
|
|
q *sqlchemy.SQuery,
|
|
userCred mcclient.TokenCredential,
|
|
query api.LoadbalancerBackendListInput,
|
|
) (*sqlchemy.SQuery, error) {
|
|
var err error
|
|
|
|
q, err = man.SStatusStandaloneResourceBaseManager.OrderByExtraFields(ctx, q, userCred, query.StatusStandaloneResourceListInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SStatusStandaloneResourceBaseManager.OrderByExtraFields")
|
|
}
|
|
q, err = man.SLoadbalancerBackendgroupResourceBaseManager.OrderByExtraFields(ctx, q, userCred, query.LoadbalancerBackendGroupFilterListInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SLoadbalancerBackendgroupResourceBaseManager.OrderByExtraFields")
|
|
}
|
|
|
|
return q, nil
|
|
}
|
|
|
|
func (man *SLoadbalancerBackendManager) QueryDistinctExtraField(q *sqlchemy.SQuery, field string) (*sqlchemy.SQuery, error) {
|
|
var err error
|
|
|
|
q, err = man.SStatusStandaloneResourceBaseManager.QueryDistinctExtraField(q, field)
|
|
if err == nil {
|
|
return q, nil
|
|
}
|
|
q, err = man.SLoadbalancerBackendgroupResourceBaseManager.QueryDistinctExtraField(q, field)
|
|
if err == nil {
|
|
return q, nil
|
|
}
|
|
|
|
return q, httperrors.ErrNotFound
|
|
}
|
|
|
|
func (man *SLoadbalancerBackendManager) ValidateBackendVpc(lb *SLoadbalancer, guest *SGuest, backendgroup *SLoadbalancerBackendGroup) error {
|
|
region, err := lb.GetRegion()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
requireStatus := region.GetDriver().GetBackendStatusForAdd()
|
|
if !utils.IsInStringArray(guest.Status, requireStatus) {
|
|
return httperrors.NewUnsupportOperationError("%s requires the virtual machine state to be %s before it can be added backendgroup, but current state of the virtual machine is %s", region.GetDriver().GetProvider(), requireStatus, guest.Status)
|
|
}
|
|
vpc, err := guest.GetVpc()
|
|
if err != nil {
|
|
return httperrors.NewBadRequestError("%s", err)
|
|
}
|
|
if len(lb.VpcId) > 0 {
|
|
lbVpc, err := lb.GetVpc()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if lbVpc != nil && !lbVpc.IsEmulated && vpc.Id != lb.VpcId {
|
|
return httperrors.NewBadRequestError("guest %s(%s) vpc %s(%s) not same as loadbalancer vpc %s", guest.Name, guest.Id, vpc.Name, vpc.Id, lb.VpcId)
|
|
}
|
|
return nil
|
|
}
|
|
backends, err := backendgroup.GetBackends()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, backend := range backends {
|
|
_server, err := GuestManager.FetchById(backend.BackendId)
|
|
if err != nil {
|
|
return httperrors.NewBadRequestError("failed getting guest %s", backend.BackendId)
|
|
}
|
|
server := _server.(*SGuest)
|
|
_vpc, err := server.GetVpc()
|
|
if err != nil {
|
|
return httperrors.NewBadRequestError("%s", err)
|
|
}
|
|
if _vpc.Id != vpc.Id {
|
|
return httperrors.NewBadRequestError("guest %s(%s) vpc %s(%s) not same as vpc %s(%s)", guest.Name, guest.Id, vpc.Name, vpc.Id, _vpc.Name, _vpc.Id)
|
|
}
|
|
if _server.GetId() == guest.Id {
|
|
return httperrors.NewBadRequestError("guest %s(%s) is already in the backendgroup %s(%s)", guest.Name, guest.Id, backendgroup.Name, backendgroup.Id)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (man *SLoadbalancerBackendManager) ValidateCreateData(ctx context.Context, userCred mcclient.TokenCredential,
|
|
ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject,
|
|
input *api.LoadbalancerBackendCreateInput) (*api.LoadbalancerBackendCreateInput, error) {
|
|
lbbgObj, err := validators.ValidateModel(ctx, userCred, LoadbalancerBackendGroupManager, &input.BackendGroupId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
lbbg := lbbgObj.(*SLoadbalancerBackendGroup)
|
|
if input.Port < 1 || input.Port > 65535 {
|
|
return input, httperrors.NewInputParameterError("invalid port %d", input.Port)
|
|
}
|
|
if input.Weight < 0 || input.Weight > 100 {
|
|
return input, httperrors.NewInputParameterError("invalid weight %d", input.Weight)
|
|
}
|
|
if len(input.SendProxy) == 0 {
|
|
input.SendProxy = api.LB_SENDPROXY_OFF
|
|
}
|
|
if !utils.IsInStringArray(input.SendProxy, api.LB_SENDPROXY_CHOICES) {
|
|
return input, httperrors.NewInputParameterError("invalid send_proxy %s", input.SendProxy)
|
|
}
|
|
if len(input.Ssl) > 0 && !utils.IsInStringArray(input.Ssl, []string{api.LB_BOOL_ON, api.LB_BOOL_OFF}) {
|
|
return input, httperrors.NewInputParameterError("invalid ssl %s", input.Ssl)
|
|
}
|
|
lb, err := lbbg.GetLoadbalancer()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "GetLoadbalancer")
|
|
}
|
|
if len(input.BackendId) == 0 && input.BackendType != api.LB_BACKEND_ADDRESS {
|
|
return nil, httperrors.NewMissingParameterError("backend_id")
|
|
}
|
|
region, err := lb.GetRegion()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
baseName := ""
|
|
switch input.BackendType {
|
|
case api.LB_BACKEND_GUEST:
|
|
guestObj, err := validators.ValidateModel(ctx, userCred, GuestManager, &input.BackendId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
guest := guestObj.(*SGuest)
|
|
input.Address, err = guest.GetAddress()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
baseName = guest.Name
|
|
host, err := guest.GetHost()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "GetHost")
|
|
}
|
|
hRegion, err := host.GetRegion()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if hRegion.Id != region.Id {
|
|
return nil, httperrors.NewInputParameterError("region of host %q (%s) != region of loadbalancer %q (%s))",
|
|
host.Name, host.ZoneId, lb.Name, lb.ZoneId)
|
|
}
|
|
if len(lb.ManagerId) == 0 {
|
|
if !utils.IsInStringArray(host.HostType, []string{api.HOST_TYPE_HYPERVISOR, api.HOST_TYPE_ESXI, api.HOST_TYPE_BAREMETAL}) {
|
|
return nil, httperrors.NewInputParameterError("host type of host %q (%s) should be either hypervisor, baremetal or esxi",
|
|
host.Name, host.HostType)
|
|
}
|
|
} else if host.ManagerId != lb.ManagerId {
|
|
return nil, httperrors.NewInputParameterError("manager of host %q (%s) != manager of loadbalancer %q (%s))",
|
|
host.Name, host.ManagerId, lb.Name, lb.ManagerId)
|
|
}
|
|
case api.LB_BACKEND_HOST:
|
|
hostObj, err := validators.ValidateModel(ctx, userCred, HostManager, &input.BackendId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
host := hostObj.(*SHost)
|
|
if len(host.AccessIp) == 0 {
|
|
return nil, fmt.Errorf("host %s has no access ip", host.GetId())
|
|
}
|
|
hRegion, err := host.GetRegion()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if hRegion.Id != region.Id {
|
|
return nil, httperrors.NewInputParameterError("region of host %q (%s) != region of loadbalancer %q (%s))",
|
|
host.Name, host.ZoneId, lb.Name, lb.ZoneId)
|
|
}
|
|
if host.ManagerId != lb.ManagerId {
|
|
return nil, httperrors.NewInputParameterError("manager of host %q (%s) != manager of loadbalancer %q (%s))",
|
|
host.Name, host.ManagerId, lb.Name, lb.ManagerId)
|
|
}
|
|
input.Address = host.AccessIp
|
|
baseName = host.Name
|
|
case api.LB_BACKEND_IP:
|
|
_, err := netutils.NewIPV4Addr(input.BackendId)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
input.Address = input.BackendId
|
|
baseName = input.Address
|
|
|
|
case api.LB_BACKEND_ADDRESS:
|
|
if len(input.Address) == 0 {
|
|
return nil, httperrors.NewMissingParameterError("address")
|
|
}
|
|
default:
|
|
return input, httperrors.NewInputParameterError("invalid backend_type %s", input.BackendType)
|
|
}
|
|
if len(input.Name) == 0 {
|
|
input.Name = fmt.Sprintf("%s-%s-%s-%s", lbbg.Name, input.BackendType, baseName, rand.String(4))
|
|
}
|
|
input.StatusStandaloneResourceCreateInput, err = man.SStatusStandaloneResourceBaseManager.ValidateCreateData(ctx, userCred, ownerId, query, input.StatusStandaloneResourceCreateInput)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return region.GetDriver().ValidateCreateLoadbalancerBackendData(ctx, userCred, lb, lbbg, input)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) GetCloudproviderId() string {
|
|
lbbg, _ := lbb.GetLoadbalancerBackendGroup()
|
|
if lbbg != nil {
|
|
return lbbg.GetCloudproviderId()
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) GetLoadbalancerBackendGroup() (*SLoadbalancerBackendGroup, error) {
|
|
backendgroup, err := LoadbalancerBackendGroupManager.FetchById(lbb.BackendGroupId)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "GetLoadbalancerBackendGroup(%s)", lbb.BackendGroupId)
|
|
}
|
|
return backendgroup.(*SLoadbalancerBackendGroup), nil
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) GetGuest() *SGuest {
|
|
guest, err := GuestManager.FetchById(lbb.BackendId)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
return guest.(*SGuest)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) GetRegion() (*SCloudregion, error) {
|
|
backendgroup, err := lbb.GetLoadbalancerBackendGroup()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return backendgroup.GetRegion()
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) GetIRegion(ctx context.Context) (cloudprovider.ICloudRegion, error) {
|
|
backendgroup, err := lbb.GetLoadbalancerBackendGroup()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return backendgroup.GetIRegion(ctx)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) ValidateUpdateData(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, input *api.LoadbalancerBackendUpdateInput) (*api.LoadbalancerBackendUpdateInput, error) {
|
|
var err error
|
|
input.StatusStandaloneResourceBaseUpdateInput, err = lbb.SStatusStandaloneResourceBase.ValidateUpdateData(ctx, userCred, query, input.StatusStandaloneResourceBaseUpdateInput)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SStatusStandaloneResourceBase.ValidateUpdateData")
|
|
}
|
|
region, err := lbb.GetRegion()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
lbbg, err := lbb.GetLoadbalancerBackendGroup()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return region.GetDriver().ValidateUpdateLoadbalancerBackendData(ctx, userCred, lbbg, input)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) PostUpdate(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) {
|
|
lbb.SStatusStandaloneResourceBase.PostUpdate(ctx, userCred, query, data)
|
|
if data.Contains("port") || data.Contains("weight") || data.Contains("enabled") {
|
|
params := data.(*jsonutils.JSONDict)
|
|
if !params.Contains("enabled") {
|
|
params.Add(jsonutils.NewBool(lbb.Status == api.LB_STATUS_ENABLED), "enabled")
|
|
}
|
|
lbb.StartLoadBalancerBackendSyncTask(ctx, userCred, params, "")
|
|
}
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) StartLoadBalancerBackendSyncTask(ctx context.Context, userCred mcclient.TokenCredential, params *jsonutils.JSONDict, parentTaskId string) error {
|
|
lbb.SetStatus(ctx, userCred, api.LB_SYNC_CONF, "")
|
|
task, err := taskman.TaskManager.NewTask(ctx, "LoadbalancerBackendSyncTask", lbb, userCred, params, parentTaskId, "", nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return task.ScheduleRun(nil)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) PostCreate(ctx context.Context, userCred mcclient.TokenCredential, ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject, data jsonutils.JSONObject) {
|
|
lbb.SStatusStandaloneResourceBase.PostCreate(ctx, userCred, ownerId, query, data)
|
|
lbb.SetStatus(ctx, userCred, api.LB_CREATING, "")
|
|
err := lbb.StartLoadBalancerBackendCreateTask(ctx, userCred, "")
|
|
if err != nil {
|
|
log.Errorf("Failed to create loadbalancer backend error: %v", err)
|
|
}
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) getVpc(ctx context.Context) (*SVpc, error) {
|
|
if lbb.BackendType != api.LB_BACKEND_GUEST {
|
|
return nil, nil
|
|
}
|
|
guestM, err := GuestManager.FetchById(lbb.BackendId)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "find guest %s", lbb.BackendId)
|
|
}
|
|
guest := guestM.(*SGuest)
|
|
return guest.GetVpc()
|
|
}
|
|
|
|
func (manager *SLoadbalancerBackendManager) FetchCustomizeColumns(
|
|
ctx context.Context,
|
|
userCred mcclient.TokenCredential,
|
|
query jsonutils.JSONObject,
|
|
objs []interface{},
|
|
fields stringutils2.SSortedStrings,
|
|
isList bool,
|
|
) []api.LoadbalancerBackendDetails {
|
|
rows := make([]api.LoadbalancerBackendDetails, len(objs))
|
|
stdRows := manager.SStatusStandaloneResourceBaseManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
|
|
lbbgRows := manager.SLoadbalancerBackendgroupResourceBaseManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
|
|
lbIds := make([]string, len(objs))
|
|
for i := range rows {
|
|
rows[i] = api.LoadbalancerBackendDetails{
|
|
StatusStandaloneResourceDetails: stdRows[i],
|
|
LoadbalancerBackendGroupResourceInfo: lbbgRows[i],
|
|
}
|
|
lbIds[i] = rows[i].LoadbalancerId
|
|
}
|
|
|
|
lbs := map[string]SLoadbalancer{}
|
|
err := db.FetchStandaloneObjectsByIds(LoadbalancerManager, lbIds, &lbs)
|
|
if err != nil {
|
|
return rows
|
|
}
|
|
|
|
virObjs := make([]interface{}, len(objs))
|
|
for i := range rows {
|
|
if lb, ok := lbs[lbIds[i]]; ok {
|
|
virObjs[i] = &lb
|
|
rows[i].ProjectId = lb.ProjectId
|
|
}
|
|
}
|
|
|
|
return rows
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) StartLoadBalancerBackendCreateTask(ctx context.Context, userCred mcclient.TokenCredential, parentTaskId string) error {
|
|
task, err := taskman.TaskManager.NewTask(ctx, "LoadbalancerBackendCreateTask", lbb, userCred, nil, parentTaskId, "", nil)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "NewTask")
|
|
}
|
|
return task.ScheduleRun(nil)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) Delete(ctx context.Context, userCred mcclient.TokenCredential) error {
|
|
return nil
|
|
}
|
|
|
|
func (self *SLoadbalancerBackend) RealDelete(ctx context.Context, userCred mcclient.TokenCredential) error {
|
|
return self.SStatusStandaloneResourceBase.Delete(ctx, userCred)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) PerformPurge(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
|
|
parasm := jsonutils.NewDict()
|
|
parasm.Add(jsonutils.JSONTrue, "purge")
|
|
return nil, lbb.StartLoadBalancerBackendDeleteTask(ctx, userCred, parasm, "")
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) CustomizeDelete(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject, data jsonutils.JSONObject) error {
|
|
lbb.SetStatus(ctx, userCred, api.LB_STATUS_DELETING, "")
|
|
return lbb.StartLoadBalancerBackendDeleteTask(ctx, userCred, jsonutils.NewDict(), "")
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) StartLoadBalancerBackendDeleteTask(ctx context.Context, userCred mcclient.TokenCredential, params *jsonutils.JSONDict, parentTaskId string) error {
|
|
task, err := taskman.TaskManager.NewTask(ctx, "LoadbalancerBackendDeleteTask", lbb, userCred, params, parentTaskId, "", nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
task.ScheduleRun(nil)
|
|
return nil
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) ValidateDeleteCondition(ctx context.Context, info jsonutils.JSONObject) error {
|
|
return lbb.SStatusStandaloneResourceBase.ValidateDeleteCondition(ctx, nil)
|
|
}
|
|
|
|
func (lbbg *SLoadbalancerBackendGroup) SyncLoadbalancerBackends(ctx context.Context, userCred mcclient.TokenCredential, provider *SCloudprovider, exts []cloudprovider.ICloudLoadbalancerBackend) compare.SyncResult {
|
|
lockman.LockRawObject(ctx, LoadbalancerBackendManager.Keyword(), lbbg.Id)
|
|
defer lockman.ReleaseRawObject(ctx, LoadbalancerBackendManager.Keyword(), lbbg.Id)
|
|
|
|
result := compare.SyncResult{}
|
|
dbRes, err := lbbg.GetBackends()
|
|
if err != nil {
|
|
result.Error(err)
|
|
return result
|
|
}
|
|
|
|
removed := []SLoadbalancerBackend{}
|
|
commondb := []SLoadbalancerBackend{}
|
|
commonext := []cloudprovider.ICloudLoadbalancerBackend{}
|
|
added := []cloudprovider.ICloudLoadbalancerBackend{}
|
|
|
|
err = compare.CompareSets(dbRes, exts, &removed, &commondb, &commonext, &added)
|
|
if err != nil {
|
|
result.Error(err)
|
|
return result
|
|
}
|
|
|
|
for i := 0; i < len(removed); i++ {
|
|
err = removed[i].syncRemove(ctx, userCred)
|
|
if err != nil {
|
|
result.DeleteError(err)
|
|
continue
|
|
}
|
|
result.Delete()
|
|
}
|
|
for i := 0; i < len(commondb); i++ {
|
|
err = commondb[i].SyncWithCloudLoadbalancerBackend(ctx, userCred, commonext[i], provider)
|
|
if err != nil {
|
|
result.UpdateError(err)
|
|
continue
|
|
}
|
|
result.Update()
|
|
}
|
|
for i := 0; i < len(added); i++ {
|
|
_, err := lbbg.newFromCloudLoadbalancerBackend(ctx, userCred, added[i], provider)
|
|
if err != nil {
|
|
result.AddError(err)
|
|
continue
|
|
}
|
|
result.Add()
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) constructFieldsFromCloudLoadbalancerBackend(ext cloudprovider.ICloudLoadbalancerBackend, managerId string) error {
|
|
lbb.Status = ext.GetStatus()
|
|
|
|
lbb.Weight = ext.GetWeight()
|
|
lbb.Port = ext.GetPort()
|
|
|
|
lbb.BackendType = ext.GetBackendType()
|
|
lbb.BackendRole = ext.GetBackendRole()
|
|
|
|
ipAddr := ext.GetIpAddress()
|
|
if len(lbb.Address) == 0 || ipAddr != lbb.Address {
|
|
lbb.Address = ipAddr
|
|
}
|
|
|
|
if lbb.BackendType == api.LB_BACKEND_GUEST {
|
|
instance, err := db.FetchByExternalIdAndManagerId(GuestManager, ext.GetBackendId(), func(q *sqlchemy.SQuery) *sqlchemy.SQuery {
|
|
sq := HostManager.Query().SubQuery()
|
|
return q.Join(sq, sqlchemy.Equals(sq.Field("id"), q.Field("host_id"))).Filter(sqlchemy.Equals(sq.Field("manager_id"), managerId))
|
|
})
|
|
if err != nil {
|
|
// 部分弹性伸缩组实例未同步, 忽略找不到实例错误
|
|
if errors.Cause(err) == sql.ErrNoRows {
|
|
return nil
|
|
}
|
|
return errors.Wrapf(err, "FetchByExternalIdAndManagerId %s", ext.GetBackendId())
|
|
}
|
|
|
|
guest := instance.(*SGuest)
|
|
|
|
lbb.BackendId = guest.Id
|
|
address, err := guest.GetAddress()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
lbb.Address = address
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) syncRemove(ctx context.Context, userCred mcclient.TokenCredential) error {
|
|
lockman.LockObject(ctx, lbb)
|
|
defer lockman.ReleaseObject(ctx, lbb)
|
|
|
|
err := lbb.ValidateDeleteCondition(ctx, nil)
|
|
if err != nil { // cannot delete
|
|
lbb.SetStatus(ctx, userCred, api.LB_STATUS_UNKNOWN, "sync to delete")
|
|
return errors.Wrapf(err, "ValidateDeleteCondition")
|
|
}
|
|
return lbb.RealDelete(ctx, userCred)
|
|
}
|
|
|
|
func (lbb *SLoadbalancerBackend) SyncWithCloudLoadbalancerBackend(ctx context.Context, userCred mcclient.TokenCredential, ext cloudprovider.ICloudLoadbalancerBackend, provider *SCloudprovider) error {
|
|
diff, err := db.UpdateWithLock(ctx, lbb, func() error {
|
|
return lbb.constructFieldsFromCloudLoadbalancerBackend(ext, provider.Id)
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
db.OpsLog.LogSyncUpdate(lbb, diff, userCred)
|
|
return nil
|
|
}
|
|
|
|
func (lbbg *SLoadbalancerBackendGroup) newFromCloudLoadbalancerBackend(ctx context.Context, userCred mcclient.TokenCredential, ext cloudprovider.ICloudLoadbalancerBackend, provider *SCloudprovider) (*SLoadbalancerBackend, error) {
|
|
lbb := &SLoadbalancerBackend{}
|
|
lbb.SetModelManager(LoadbalancerBackendManager, lbb)
|
|
|
|
lbb.BackendGroupId = lbbg.Id
|
|
lbb.ExternalId = ext.GetGlobalId()
|
|
|
|
err := lbb.constructFieldsFromCloudLoadbalancerBackend(ext, provider.Id)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "constructFieldsFromCloudLoadbalancerBackend")
|
|
}
|
|
|
|
lbb.Name = ext.GetName()
|
|
err = LoadbalancerBackendManager.TableSpec().Insert(ctx, lbb)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "Insert")
|
|
}
|
|
db.OpsLog.LogEvent(lbb, db.ACT_CREATE, lbb.GetShortDesc(ctx), userCred)
|
|
return lbb, nil
|
|
}
|
|
|
|
func (manager *SLoadbalancerBackendManager) ListItemExportKeys(ctx context.Context,
|
|
q *sqlchemy.SQuery,
|
|
userCred mcclient.TokenCredential,
|
|
keys stringutils2.SSortedStrings,
|
|
) (*sqlchemy.SQuery, error) {
|
|
var err error
|
|
|
|
q, err = manager.SStatusStandaloneResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SStatusStandaloneResourceBaseManager.ListItemExportKeys")
|
|
}
|
|
|
|
if keys.ContainsAny(manager.SLoadbalancerBackendgroupResourceBaseManager.GetExportKeys()...) {
|
|
q, err = manager.SLoadbalancerBackendgroupResourceBaseManager.ListItemExportKeys(ctx, q, userCred, keys)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "SLoadbalancerBackendgroupResourceBaseManager.ListItemExportKeys")
|
|
}
|
|
}
|
|
|
|
return q, nil
|
|
}
|
|
|
|
func (manager *SLoadbalancerBackendManager) InitializeData() error {
|
|
return nil
|
|
}
|