Files
cloudpods/pkg/image/models/image_guest.go

642 lines
19 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"
"fmt"
"sort"
"strings"
"time"
"golang.org/x/sync/errgroup"
"yunion.io/x/jsonutils"
"yunion.io/x/log"
"yunion.io/x/pkg/errors"
"yunion.io/x/pkg/tristate"
"yunion.io/x/pkg/utils"
"yunion.io/x/sqlchemy"
"yunion.io/x/onecloud/pkg/apis"
api "yunion.io/x/onecloud/pkg/apis/image"
"yunion.io/x/onecloud/pkg/appsrv"
"yunion.io/x/onecloud/pkg/cloudcommon/db"
"yunion.io/x/onecloud/pkg/cloudcommon/db/lockman"
"yunion.io/x/onecloud/pkg/cloudcommon/db/quotas"
"yunion.io/x/onecloud/pkg/cloudcommon/db/taskman"
"yunion.io/x/onecloud/pkg/httperrors"
"yunion.io/x/onecloud/pkg/image/options"
"yunion.io/x/onecloud/pkg/mcclient"
"yunion.io/x/onecloud/pkg/util/logclient"
"yunion.io/x/onecloud/pkg/util/rbacutils"
"yunion.io/x/onecloud/pkg/util/stringutils2"
)
type SGuestImageManager struct {
db.SSharableVirtualResourceBaseManager
}
var GuestImageManager *SGuestImageManager
func init() {
GuestImageManager = &SGuestImageManager{
db.NewSharableVirtualResourceBaseManager(
SGuestImage{},
"guestimages_tbl",
"guestimage",
"guestimages",
),
}
GuestImageManager.SetVirtualObject(GuestImageManager)
}
type SGuestImage struct {
db.SSharableVirtualResourceBase
Protected tristate.TriState `nullable:"false" default:"true" list:"user" get:"user" create:"optional" update:"user"`
}
func (manager *SGuestImageManager) ValidateCreateData(ctx context.Context, userCred mcclient.TokenCredential,
ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject, data *jsonutils.JSONDict) (*jsonutils.JSONDict, error) {
if !data.Contains("image_number") {
return nil, httperrors.NewMissingParameterError("image_number")
}
imageNum, _ := data.Int("image_number")
pendingUsage := SQuota{Image: int(imageNum)}
data.Set("disk_format", jsonutils.NewString("qcow2"))
keys := imageCreateInput2QuotaKeys(data, ownerId)
pendingUsage.SetKeys(keys)
if err := quotas.CheckSetPendingQuota(ctx, userCred, &pendingUsage); err != nil {
return nil, httperrors.NewOutOfQuotaError("%s", err)
}
return data, nil
}
func (gi *SGuestImage) CustomizeCreate(ctx context.Context, userCred mcclient.TokenCredential,
ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject, data jsonutils.JSONObject) error {
err := gi.SVirtualResourceBase.CustomizeCreate(ctx, userCred, ownerId, query, data)
if err != nil {
return err
}
gi.Status = api.IMAGE_STATUS_QUEUED
return nil
}
func (gi *SGuestImage) PostCreate(ctx context.Context, userCred mcclient.TokenCredential,
ownerId mcclient.IIdentityProvider, query jsonutils.JSONObject, data jsonutils.JSONObject) {
kwargs := data.(*jsonutils.JSONDict)
// get image number
imageNumber, _ := kwargs.Int("image_number")
// deal public params
kwargs.Remove("size")
kwargs.Remove("image_number")
kwargs.Remove("name")
if !kwargs.Contains("images") {
return
}
images, _ := kwargs.GetArray("images")
kwargs.Remove("images")
kwargs.Add(jsonutils.NewString(gi.Id), "guest_image_id")
imageIds := jsonutils.NewArray()
suc := true
// HACK
appParams := appsrv.AppContextGetParams(ctx)
appParams.Request.ContentLength = 0
for i := 0; i < len(images); i++ {
params := jsonutils.DeepCopy(kwargs).(*jsonutils.JSONDict)
image := images[i].(*jsonutils.JSONDict)
for _, key := range image.SortedKeys() {
tmp, _ := image.Get(key)
params.Add(tmp, key)
}
if i == len(images)-1 {
params.Add(jsonutils.NewString(fmt.Sprintf("%s-%s", gi.Name, "root")), "generate_name")
} else {
params.Add(jsonutils.NewString(fmt.Sprintf("%s-%s-%d", gi.Name, "data", i)), "generate_name")
params.Add(jsonutils.JSONTrue, "is_data")
}
model, err := db.DoCreate(ImageManager, ctx, userCred, query, params, ownerId)
if err != nil {
imageIds.Add(jsonutils.NewString(""))
suc = false
break
} else {
func() {
lockman.LockObject(ctx, model)
defer lockman.ReleaseObject(ctx, model)
model.PostCreate(ctx, userCred, ownerId, query, data)
}()
imageIds.Add(jsonutils.NewString(model.GetId()))
}
}
pendingUsage := SQuota{Image: int(imageNumber)}
keys := imageCreateInput2QuotaKeys(data, ownerId)
pendingUsage.SetKeys(keys)
quotas.CancelPendingUsage(ctx, userCred, &pendingUsage, &pendingUsage, true)
if !suc {
gi.SetStatus(userCred, api.IMAGE_STATUS_KILLED, "create subimage failed")
}
gi.SetStatus(userCred, api.IMAGE_STATUS_SAVING, "")
// HACK
tmp := query.(*jsonutils.JSONDict)
tmp.Add(imageIds, "image_ids")
}
func (gi *SGuestImage) ValidateDeleteCondition(ctx context.Context) error {
if gi.Protected.IsTrue() {
return httperrors.NewForbiddenError("image is protected")
}
return nil
}
func (gi *SGuestImage) Delete(ctx context.Context, userCred mcclient.TokenCredential) error {
log.Infof("image delete to nothing")
return nil
}
func (gi *SGuestImage) RealDelete(ctx context.Context, userCred mcclient.TokenCredential) error {
// delete joint
guestJoints, err := GuestImageJointManager.GetByGuestImageId(gi.Id)
if err != nil {
return errors.Wrap(err, "get guest image joint failed")
}
for i := range guestJoints {
guestJoints[i].Delete(ctx, userCred)
}
return gi.SVirtualResourceBase.Delete(ctx, userCred)
}
func (gi *SGuestImage) CustomizeDelete(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject, data jsonutils.JSONObject) error {
images, err := GuestImageJointManager.GetImagesByGuestImageId(gi.Id)
if err != nil {
return errors.Wrap(err, "get images of guest images failed")
}
if len(images) == 0 {
return gi.RealDelete(ctx, userCred)
}
overridePendingDelete := false
purge := false
if query != nil {
overridePendingDelete = jsonutils.QueryBoolean(query, "override_pending_delete", false)
purge = jsonutils.QueryBoolean(query, "purge", false)
}
if gi.Status == api.IMAGE_STATUS_QUEUED {
gi.checkStatus(ctx, userCred)
}
if utils.IsInStringArray(gi.Status, []string{api.IMAGE_STATUS_QUEUED, api.IMAGE_STATUS_KILLED}) {
overridePendingDelete = true
}
return gi.startDeleteTask(ctx, userCred, "", purge, overridePendingDelete)
}
func (gi *SGuestImage) startDeleteTask(ctx context.Context, userCred mcclient.TokenCredential, parentTaskId string,
isPurge bool, overridePendingDelete bool) error {
params := jsonutils.NewDict()
if isPurge {
params.Add(jsonutils.JSONTrue, "purge")
}
if overridePendingDelete {
params.Add(jsonutils.JSONTrue, "override_pending_delete")
}
params.Add(jsonutils.NewString(gi.Status), "image_status")
gi.SetStatus(userCred, api.IMAGE_STATUS_DEACTIVATED, "")
if task, err := taskman.TaskManager.NewTask(ctx, "GuestImageDeleteTask", gi, userCred, params, parentTaskId, "",
nil); err != nil {
return err
} else {
task.ScheduleRun(nil)
}
return nil
}
func (gi *SGuestImage) AllowPerformCancelDelete(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject, data jsonutils.JSONObject) bool {
return db.IsAdminAllowPerform(userCred, gi, "cancel-delete")
}
func (gi *SGuestImage) PerformCancelDelete(ctx context.Context, userCred mcclient.TokenCredential,
query jsonutils.JSONObject, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
if gi.PendingDeleted {
err := gi.DoCancelPendingDelete(ctx, userCred)
return nil, err
}
return nil, nil
}
func (gi *SGuestImage) DoCancelPendingDelete(ctx context.Context, userCred mcclient.TokenCredential) error {
subImages, err := GuestImageJointManager.GetImagesByGuestImageId(gi.Id)
for i := range subImages {
err = subImages[i].DoCancelPendingDelete(ctx, userCred)
if err != nil {
return errors.Wrapf(err, "subimage %s cancel delete error", subImages[i].GetId())
}
}
err = gi.SVirtualResourceBase.DoCancelPendingDelete(ctx, userCred)
if err != nil {
return err
}
_, err = db.Update(gi, func() error {
gi.Status = api.IMAGE_STATUS_ACTIVE
return nil
})
return errors.Wrap(err, "guest image cancel delete error")
}
func (self *SGuestImage) getMoreDetails(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject,
out api.GuestImageDetails) api.GuestImageDetails {
if query.Contains("image_ids") {
out.ImageIds, _ = query.Get("image_ids")
}
if self.Status != api.IMAGE_STATUS_ACTIVE {
self.checkStatus(ctx, userCred)
out.Status = self.Status
}
images, err := GuestImageJointManager.GetImagesByGuestImageId(self.Id)
if err != nil {
return out
}
var size int64 = 0
if len(images) == 0 {
out.Size = size
return out
}
dataImages := make([]api.SubImageInfo, 0, len(images)-1)
var rootImage api.SubImageInfo
for i := range images {
image := images[i]
size += image.Size
if !image.IsData.IsTrue() {
rootImage = api.SubImageInfo{
ID: image.Id,
Name: image.Name,
MinDiskMB: image.MinDiskMB,
DiskFormat: image.DiskFormat,
Size: image.Size,
Status: image.Status,
CreatedAt: image.CreatedAt,
}
out.MinRamMb = image.MinRamMB
out.DiskFormat = image.DiskFormat
continue
}
dataImages = append(dataImages, api.SubImageInfo{
ID: image.Id,
Name: image.Name,
MinDiskMB: image.MinDiskMB,
DiskFormat: image.DiskFormat,
Size: image.Size,
Status: image.Status,
CreatedAt: image.CreatedAt,
})
}
// make sure that the sort of dataimage is fixed
sort.Slice(dataImages, func(i, j int) bool {
return dataImages[i].Name < dataImages[j].Name
})
out.Size = size
out.RootImage = rootImage
out.DataImages = dataImages
// properties of root image
properties, err := ImagePropertyManager.GetProperties(rootImage.ID)
if err != nil {
return out
}
propJson := jsonutils.NewDict()
for k, v := range properties {
propJson.Add(jsonutils.NewString(v), k)
}
out.Properties = propJson
out.DisableDelete = self.Protected.Bool()
return out
}
func (self *SGuestImage) GetExtraDetails(
ctx context.Context,
userCred mcclient.TokenCredential,
query jsonutils.JSONObject,
isList bool,
) (api.GuestImageDetails, error) {
return api.GuestImageDetails{}, nil
}
func (manager *SGuestImageManager) FetchCustomizeColumns(
ctx context.Context,
userCred mcclient.TokenCredential,
query jsonutils.JSONObject,
objs []interface{},
fields stringutils2.SSortedStrings,
isList bool,
) []api.GuestImageDetails {
rows := make([]api.GuestImageDetails, len(objs))
virtRows := manager.SSharableVirtualResourceBaseManager.FetchCustomizeColumns(ctx, userCred, query, objs, fields, isList)
for i := range rows {
rows[i] = api.GuestImageDetails{
SharableVirtualResourceDetails: virtRows[i],
}
guestImage := objs[i].(*SGuestImage)
rows[i] = guestImage.getMoreDetails(ctx, userCred, query, rows[i])
}
return rows
}
func (self *SGuestImage) PostUpdate(ctx context.Context, userCred mcclient.TokenCredential, query jsonutils.JSONObject,
data jsonutils.JSONObject) {
lockman.LockClass(ctx, ImageManager, db.GetLockClassKey(ImageManager, userCred))
defer lockman.ReleaseClass(ctx, ImageManager, db.GetLockClassKey(ImageManager, userCred))
err := self.UpdateSubImage(ctx, userCred, data)
if err != nil {
logclient.AddSimpleActionLog(self, logclient.ACT_UPDATE, nil, userCred, false)
}
}
func (self *SGuestImage) UpdateSubImage(ctx context.Context, userCred mcclient.TokenCredential,
data jsonutils.JSONObject) error {
subImages, err := GuestImageJointManager.GetImagesByFilter(self.GetId(), func(q *sqlchemy.SQuery) *sqlchemy.SQuery {
return q.Asc("name")
})
if err != nil {
return err
}
dict := data.(*jsonutils.JSONDict)
var g errgroup.Group
for i := range subImages {
if f, ok := self.genUpdateImage(ctx, userCred, &subImages[i], i, dict); ok {
g.Go(f)
}
}
return g.Wait()
}
func (self *SGuestImage) genUpdateImage(ctx context.Context, userCred mcclient.TokenCredential, image *SImage,
index int, dict *jsonutils.JSONDict) (func() error, bool) {
if !dict.Contains("name") && image.IsData.IsTrue() {
return nil, false
}
if image.IsData.IsTrue() {
if !dict.Contains("name") {
return nil, false
}
return func() error {
name, _ := dict.GetString("name")
name, err := db.GenerateName(ImageManager, userCred, fmt.Sprintf("%s-%s-%d", name, "data", index))
if err != nil {
return errors.Wrap(err, "fail to generate unique name")
}
_, err = db.Update(image, func() error {
image.Name = name
return nil
})
if err != nil {
return errors.Wrap(err, "modify subimage's name failed")
}
return nil
}, true
}
return func() error {
if dict.Contains("name") {
name, _ := dict.GetString("name")
name, err := db.GenerateName(ImageManager, userCred, fmt.Sprintf("%s-%s", name, "root"))
if err != nil {
return errors.Wrap(err, "fail to generate unique name")
}
_, err = db.Update(image, func() error {
image.Name = name
return nil
})
if err != nil {
return errors.Wrap(err, "modify subimage's name failed")
}
}
if dict.Contains("properties") {
props, _ := dict.Get("properties")
err := ImagePropertyManager.SaveProperties(ctx, userCred, image.GetId(), props)
if err != nil {
return errors.Wrap(err, "save properties error")
}
}
return nil
}, true
}
var checkStatus = map[string]int{
api.IMAGE_STATUS_ACTIVE: 1,
api.IMAGE_STATUS_QUEUED: 2,
api.IMAGE_STATUS_SAVING: 3,
api.IMAGE_STATUS_DEACTIVATED: 4,
api.IMAGE_STATUS_KILLED: 5,
}
func (self *SGuestImage) checkStatus(ctx context.Context, userCred mcclient.TokenCredential) error {
images, err := GuestImageJointManager.GetImagesByGuestImageId(self.Id)
if err != nil {
return err
}
if len(images) == 0 {
return nil
}
status := api.IMAGE_STATUS_ACTIVE
for i := range images {
if checkStatus[images[i].Status] > checkStatus[status] {
status = images[i].Status
}
}
if self.Status != status {
self.SetStatus(userCred, status, "")
self.Status = status
}
return nil
}
func (self *SGuestImage) getSize(ctx context.Context, userCred mcclient.TokenCredential) (int64, error) {
images, err := GuestImageJointManager.GetImagesByGuestImageId(self.Id)
if err != nil {
return 0, err
}
var size int64 = 0
for i := range images {
size += images[i].Size
}
return size, nil
}
func (self *SGuestImageManager) getExpiredPendingDeleteImages() []SGuestImage {
deadline := time.Now().Add(time.Duration(-options.Options.PendingDeleteExpireSeconds) * time.Second)
// there are so many common images of one guest image, so that batch shrink three times
q := self.Query().IsTrue("pending_deleted").LT("pending_deleted_at",
deadline).Limit(options.Options.PendingDeleteMaxCleanBatchSize / 3)
images := make([]SGuestImage, 0)
err := db.FetchModelObjects(self, q, &images)
if err != nil {
log.Errorf("fetch guest images error %s", err)
return nil
}
return images
}
func (self *SGuestImageManager) CleanPendingDeleteImages(ctx context.Context, userCred mcclient.TokenCredential,
isStart bool) {
images := self.getExpiredPendingDeleteImages()
if images == nil {
return
}
for i := range images {
images[i].startDeleteTask(ctx, userCred, "", false, true)
}
}
func (self *SGuestImage) PerformPublic(
ctx context.Context,
userCred mcclient.TokenCredential,
query jsonutils.JSONObject,
input apis.PerformPublicInput,
) (jsonutils.JSONObject, error) {
images, err := GuestImageJointManager.GetImagesByGuestImageId(self.Id)
if err != nil {
return nil, errors.Wrap(err, "fail to fetch subimages of guest image")
}
for i := range images {
_, err := images[i].PerformPublic(ctx, userCred, query, input)
if err != nil {
return nil, errors.Wrapf(err, "fail to public subimage %s", images[i].GetId())
}
}
return self.SSharableVirtualResourceBase.PerformPublic(ctx, userCred, query, input)
}
func (self *SGuestImage) PerformPrivate(
ctx context.Context,
userCred mcclient.TokenCredential,
query jsonutils.JSONObject,
input apis.PerformPrivateInput,
) (jsonutils.JSONObject, error) {
images, err := GuestImageJointManager.GetImagesByGuestImageId(self.Id)
if err != nil {
return nil, errors.Wrap(err, "fail to fetch subimages of guest image")
}
for i := range images {
_, err := images[i].PerformPrivate(ctx, userCred, query, input)
if err != nil {
return nil, errors.Wrapf(err, "fail to private subimage %s", images[i].GetId())
}
}
return self.SSharableVirtualResourceBase.PerformPrivate(ctx, userCred, query, input)
}
// 主机镜像列表
func (manager *SGuestImageManager) ListItemFilter(
ctx context.Context,
q *sqlchemy.SQuery,
userCred mcclient.TokenCredential,
query api.GuestImageListInput,
) (*sqlchemy.SQuery, error) {
q, err := manager.SSharableVirtualResourceBaseManager.ListItemFilter(ctx, q, userCred, query.SharableVirtualResourceListInput)
if err != nil {
return nil, errors.Wrap(err, "SSharableVirtualResourceBaseManager.ListItemFilter")
}
if query.Protected != nil {
if *query.Protected {
q = q.IsTrue("protected")
} else {
q = q.IsFalse("protected")
}
}
return q, nil
}
func (manager *SGuestImageManager) OrderByExtraFields(
ctx context.Context,
q *sqlchemy.SQuery,
userCred mcclient.TokenCredential,
query api.GuestImageListInput,
) (*sqlchemy.SQuery, error) {
var err error
q, err = manager.SSharableVirtualResourceBaseManager.OrderByExtraFields(ctx, q, userCred, query.SharableVirtualResourceListInput)
if err != nil {
return nil, errors.Wrap(err, "SSharableVirtualResourceBaseManager.OrderByExtraFields")
}
return q, nil
}
func (manager *SGuestImageManager) QueryDistinctExtraField(q *sqlchemy.SQuery, field string) (*sqlchemy.SQuery, error) {
var err error
q, err = manager.SSharableVirtualResourceBaseManager.QueryDistinctExtraField(q, field)
if err == nil {
return q, nil
}
return q, httperrors.ErrNotFound
}
func (manager *SGuestImageManager) Usage(scope rbacutils.TRbacScope, ownerId mcclient.IIdentityProvider, prefix string) map[string]int64 {
usages := make(map[string]int64)
count := ImageManager.count(scope, ownerId, api.IMAGE_STATUS_ACTIVE, tristate.False, false, tristate.True)
expandUsageCount(usages, prefix, "guest_image", "", count)
sq := manager.Query()
switch scope {
case rbacutils.ScopeSystem:
// do nothing
case rbacutils.ScopeDomain:
sq = sq.Equals("domain_id", ownerId.GetProjectDomainId())
case rbacutils.ScopeProject:
sq = sq.Equals("tenant_id", ownerId.GetProjectId())
}
cnt, _ := sq.CountWithError()
key := []string{}
if len(prefix) > 0 {
key = append(key, prefix)
}
key = append(key, "guest_image", "count")
usages[strings.Join(key, ".")] = int64(cnt)
return usages
}