Files
cloudpods/pkg/compute/models/cloudimages.go
2023-08-31 11:26:58 +08:00

189 lines
5.2 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/log"
"yunion.io/x/pkg/errors"
"yunion.io/x/pkg/gotypes"
"yunion.io/x/onecloud/pkg/cloudcommon/db"
"yunion.io/x/onecloud/pkg/mcclient"
"yunion.io/x/onecloud/pkg/util/yunionmeta"
)
type SCloudimageManager struct {
db.SStandaloneResourceBaseManager
db.SExternalizedResourceBaseManager
}
var CloudimageManager *SCloudimageManager
func init() {
CloudimageManager = &SCloudimageManager{
SStandaloneResourceBaseManager: db.NewStandaloneResourceBaseManager(
SCloudimage{},
"cloudimages_tbl",
"cloudimage",
"cloudimages",
),
}
CloudimageManager.SetVirtualObject(CloudimageManager)
}
type SCloudimage struct {
db.SStandaloneResourceBase
db.SExternalizedResourceBase
SCloudregionResourceBase `width:"36" charset:"ascii" nullable:"false" list:"user" default:"default" create:"optional" json:"cloudregion_id" index:"true"`
}
func SyncPublicCloudImages(ctx context.Context, userCred mcclient.TokenCredential, isStart bool) {
if isStart {
cnt, err := CloudimageManager.Query().CountWithError()
if err != nil {
return
}
if cnt > 0 {
log.Infof("Public cloud image has already synced, skip syncing")
return
}
}
regions := []SCloudregion{}
q := CloudregionManager.Query().In("provider", CloudproviderManager.GetPublicProviderProvidersQuery())
err := db.FetchModelObjects(CloudregionManager, q, &regions)
if err != nil {
return
}
meta, err := yunionmeta.FetchYunionmeta(ctx)
if err != nil {
log.Errorf("FetchYunionmeta %v", err)
return
}
index, err := meta.Index(CloudimageManager.Keyword())
if err != nil {
log.Errorf("getServerSkuIndex error: %v", err)
return
}
for i := range regions {
region := &regions[i]
skuMeta := &SCloudimage{}
skuMeta.SetModelManager(CloudimageManager, skuMeta)
skuMeta.Id = region.ExternalId
oldMd5 := db.Metadata.GetStringValue(ctx, skuMeta, db.SKU_METADAT_KEY, userCred)
newMd5, ok := index[region.ExternalId]
if !ok || newMd5 == yunionmeta.EMPTY_MD5 || len(oldMd5) > 0 && newMd5 == oldMd5 {
continue
}
db.Metadata.SetValue(ctx, skuMeta, db.SKU_METADAT_KEY, newMd5, userCred)
err = regions[i].SyncCloudImages(ctx, userCred, !isStart, false)
if err != nil {
continue
}
storagecaches, err := regions[i].GetStoragecaches()
if err != nil {
log.Errorf("GetStoragecaches for region %s(%s) error: %v", regions[i].Name, regions[i].Id, err)
}
for j := range storagecaches {
err = storagecaches[j].CheckCloudimages(ctx, userCred, regions[i].Name, regions[i].Id)
if err != nil {
log.Errorf("SyncSystemImages for region %s(%s) storagecache %s error: %v", regions[i].Name, regions[i].Id, storagecaches[j].Name, err)
}
}
}
return
}
func (self *SCloudimage) syncRemove(ctx context.Context, userCred mcclient.TokenCredential) error {
_image, err := db.FetchByExternalId(CachedimageManager, self.ExternalId)
if err != nil {
if errors.Cause(err) == sql.ErrNoRows {
return self.Delete(ctx, userCred)
}
return errors.Wrapf(err, "db.FetchByExternalId(%s)", self.ExternalId)
}
image := _image.(*SCachedimage)
err = image.ValidateDeleteCondition(ctx, nil)
if err == nil {
image.Delete(ctx, userCred)
}
return self.Delete(ctx, userCred)
}
func (self *SCloudimage) Delete(ctx context.Context, userCred mcclient.TokenCredential) error {
return db.RealDeleteModel(ctx, userCred, self)
}
func (self *SCloudimage) syncWithImage(ctx context.Context, userCred mcclient.TokenCredential, image SCachedimage, region *SCloudregion) error {
meta, err := yunionmeta.FetchYunionmeta(ctx)
if err != nil {
return err
}
skuUrl := fmt.Sprintf("%s/%s/%s.json", meta.ImageBase, region.ExternalId, image.GetGlobalId())
obj, err := db.FetchByExternalId(CachedimageManager, image.GetGlobalId())
if err != nil {
if errors.Cause(err) != sql.ErrNoRows {
return errors.Wrapf(err, "db.FetchByExternalId(%s)", image.GetGlobalId())
}
cachedImage := &SCachedimage{}
cachedImage.SetModelManager(CachedimageManager, cachedImage)
err = meta.Get(skuUrl, cachedImage)
if err != nil {
return errors.Wrapf(err, "Get")
}
cachedImage.IsPublic = true
cachedImage.ProjectId = "system"
err = CachedimageManager.TableSpec().Insert(ctx, cachedImage)
if err != nil {
return errors.Wrapf(err, "Insert cachedimage")
}
return nil
}
cachedImage := obj.(*SCachedimage)
if gotypes.IsNil(cachedImage.Info) {
err = meta.Get(skuUrl, &image)
if err != nil {
return errors.Wrapf(err, "Get")
}
_, err := db.Update(cachedImage, func() error {
cachedImage.Info = image.Info
cachedImage.Size = image.Size
cachedImage.UEFI = image.UEFI
return nil
})
return err
}
return nil
}