| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424 |
- // 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 qcloud
- import (
- "context"
- "fmt"
- "sort"
- "strconv"
- "strings"
- "time"
- "yunion.io/x/jsonutils"
- "yunion.io/x/log"
- "yunion.io/x/pkg/utils"
- billing_api "yunion.io/x/cloudmux/pkg/apis/billing"
- api "yunion.io/x/cloudmux/pkg/apis/compute"
- "yunion.io/x/cloudmux/pkg/cloudprovider"
- "yunion.io/x/cloudmux/pkg/multicloud"
- )
- type Placement struct {
- ProjectId int
- Zone string
- }
- type SDisk struct {
- storage *SStorage
- multicloud.SDisk
- QcloudTags
- Attached bool
- AutoRenewFlagError bool
- CreateTime time.Time
- DeadlineError bool
- DeadlineTime time.Time
- DifferDaysOfDeadline int
- DiskChargeType string
- DiskId string
- DiskName string
- DiskSize int
- DiskState string
- DiskType string
- DiskUsage string
- Encrypt bool
- InstanceId string
- IsReturnable bool
- Placement Placement
- Portable bool
- RenewFlag string
- ReturnFailCode int
- RollbackPercent int
- Rollbacking bool
- SnapshotAbility bool
- DeleteWithInstance bool
- }
- type SDiskSet []SDisk
- func (v SDiskSet) Len() int {
- return len(v)
- }
- func (v SDiskSet) Swap(i, j int) {
- v[i], v[j] = v[j], v[i]
- }
- func (v SDiskSet) Less(i, j int) bool {
- if v[i].DiskUsage == "SYSTEM_DISK" || v[j].DiskUsage == "DATA_DISK" {
- return true
- }
- return false
- }
- func (self *SRegion) GetDisks(instanceId string, zoneId string, category string, diskIds []string, offset int, limit int) ([]SDisk, int, error) {
- if limit > 50 || limit <= 0 {
- limit = 50
- }
- params := make(map[string]string)
- params["Limit"] = fmt.Sprintf("%d", limit)
- params["Offset"] = fmt.Sprintf("%d", offset)
- filter := 0
- if len(zoneId) > 0 {
- params[fmt.Sprintf("Filters.%d.Name", filter)] = "zone"
- params[fmt.Sprintf("Filters.%d.Values.0", filter)] = zoneId
- filter++
- }
- if len(instanceId) > 0 {
- params[fmt.Sprintf("Filters.%d.Name", filter)] = "instance-id"
- params[fmt.Sprintf("Filters.%d.Values.0", filter)] = instanceId
- filter++
- }
- if len(category) > 0 {
- params[fmt.Sprintf("Filters.%d.Name", filter)] = "disk-type"
- params[fmt.Sprintf("Filters.%d.Values.0", filter)] = category
- filter++
- }
- if diskIds != nil && len(diskIds) > 0 {
- for index, diskId := range diskIds {
- params[fmt.Sprintf("DiskIds.%d", index)] = diskId
- }
- }
- body, err := self.cbsRequest("DescribeDisks", params)
- if err != nil {
- log.Errorf("GetDisks fail %s", err)
- return nil, 0, err
- }
- disks := make([]SDisk, 0)
- err = body.Unmarshal(&disks, "DiskSet")
- if err != nil {
- log.Errorf("Unmarshal disk details fail %s", err)
- return nil, 0, err
- }
- total, _ := body.Float("TotalCount")
- sort.Sort(SDiskSet(disks))
- return disks, int(total), nil
- }
- func (self *SRegion) GetDisk(diskId string) (*SDisk, error) {
- disks, total, err := self.GetDisks("", "", "", []string{diskId}, 0, 1)
- if err != nil {
- return nil, err
- }
- if total != 1 {
- return nil, cloudprovider.ErrNotFound
- }
- return &disks[0], nil
- }
- func (self *SDisk) GetId() string {
- return self.DiskId
- }
- func (self *SRegion) DeleteDisk(diskIds []string) error {
- params := make(map[string]string)
- params["Region"] = self.Region
- for idx, id := range diskIds {
- params[fmt.Sprintf("DiskIds.%d", idx)] = id
- }
- _, err := self.cbsRequest("TerminateDisks", params)
- return err
- }
- func (self *SDisk) Delete(ctx context.Context) error {
- return self.storage.zone.region.DeleteDisk([]string{self.DiskId})
- }
- func (self *SRegion) ResizeDisk(ctx context.Context, diskId string, sizeGb int64) error {
- params := make(map[string]string)
- params["DiskId"] = diskId
- params["DiskSize"] = fmt.Sprintf("%d", sizeGb)
- startTime := time.Now()
- for {
- _, err := self.cbsRequest("ResizeDisk", params)
- if err != nil {
- if strings.Index(err.Error(), "Code=InvalidDisk.Busy") > 0 {
- log.Infof("The disk is busy, try later ...")
- time.Sleep(10 * time.Second)
- if time.Now().Sub(startTime) > time.Minute*20 {
- return cloudprovider.ErrTimeout
- }
- continue
- }
- }
- return err
- }
- }
- func (self *SDisk) Resize(ctx context.Context, sizeMb int64) error {
- return self.storage.zone.region.ResizeDisk(ctx, self.DiskId, sizeMb/1024)
- }
- func (self *SDisk) GetName() string {
- if len(self.DiskName) > 0 && self.DiskName != "未命名" {
- return self.DiskName
- }
- return self.DiskId
- }
- func (self *SDisk) GetGlobalId() string {
- return self.DiskId
- }
- func (self *SDisk) IsEmulated() bool {
- return false
- }
- func (self *SDisk) GetIStorage() (cloudprovider.ICloudStorage, error) {
- return self.storage, nil
- }
- func (self *SDisk) GetStatus() string {
- switch self.DiskState {
- case "ATTACHING", "DETACHING", "EXPANDING", "ROLLBACKING":
- return api.DISK_ALLOCATING
- default:
- return api.DISK_READY
- }
- }
- func (self *SDisk) Refresh() error {
- new, err := self.storage.zone.region.GetDisk(self.DiskId)
- if err != nil {
- return err
- }
- return jsonutils.Update(self, new)
- }
- func (self *SDisk) CreateISnapshot(ctx context.Context, name, desc string) (cloudprovider.ICloudSnapshot, error) {
- snapshotId, err := self.storage.zone.region.CreateSnapshot(self.DiskId, name, desc)
- if err != nil {
- log.Errorf("createSnapshot fail %s", err)
- return nil, err
- }
- snapshots, total, err := self.storage.zone.region.GetSnapshots("", "", "", []string{snapshotId}, 0, 1)
- if err != nil {
- return nil, err
- }
- if total == 1 {
- snapshot := &snapshots[0]
- err := cloudprovider.WaitStatus(snapshot, api.SNAPSHOT_READY, 15*time.Second, 3600*time.Second)
- if err != nil {
- return nil, err
- }
- return snapshot, nil
- }
- return nil, nil
- }
- func (self *SDisk) GetDiskType() string {
- switch self.DiskUsage {
- case "SYSTEM_DISK":
- return api.DISK_TYPE_SYS
- case "DATA_DISK":
- return api.DISK_TYPE_DATA
- default:
- return api.DISK_TYPE_DATA
- }
- }
- func (self *SDisk) GetFsFormat() string {
- return ""
- }
- func (self *SDisk) GetIsNonPersistent() bool {
- return false
- }
- func (self *SDisk) GetDriver() string {
- return "scsi"
- }
- func (self *SDisk) GetCacheMode() string {
- return "none"
- }
- func (self *SDisk) GetMountpoint() string {
- return ""
- }
- func (self *SDisk) GetBillingType() string {
- switch self.DiskChargeType {
- case "PREPAID":
- return billing_api.BILLING_TYPE_PREPAID
- case "POSTPAID_BY_HOUR":
- return billing_api.BILLING_TYPE_POSTPAID
- default:
- return billing_api.BILLING_TYPE_PREPAID
- }
- }
- func (self *SDisk) GetDiskFormat() string {
- return "vhd"
- }
- func (self *SDisk) GetDiskSizeMB() int {
- return self.DiskSize * 1024
- }
- func (self *SDisk) GetIsAutoDelete() bool {
- return self.DeleteWithInstance
- }
- func (self *SDisk) GetCreatedAt() time.Time {
- // 2019-12-25 09:00:43 #非UTC时间
- return self.CreateTime.Add(time.Hour * -8)
- }
- func (self *SDisk) GetExpiredAt() time.Time {
- if self.DeadlineTime.IsZero() {
- return time.Time{}
- }
- return self.DeadlineTime.Add(time.Hour * -8)
- }
- func (self *SDisk) SetTags(tags map[string]string, replace bool) error {
- return self.storage.zone.region.SetResourceTags("cvm", "volume", []string{self.DiskId}, tags, replace)
- }
- func (self *SDisk) GetISnapshot(snapshotId string) (cloudprovider.ICloudSnapshot, error) {
- snapshots, total, err := self.storage.zone.region.GetSnapshots("", "", "", []string{snapshotId}, 0, 1)
- if err != nil {
- return nil, err
- }
- if total == 1 {
- return &snapshots[0], nil
- }
- return nil, cloudprovider.ErrNotFound
- }
- func (self *SDisk) GetISnapshots() ([]cloudprovider.ICloudSnapshot, error) {
- snapshots := make([]SSnapshot, 0)
- for {
- parts, total, err := self.storage.zone.region.GetSnapshots("", self.DiskId, "", []string{}, 0, 20)
- if err != nil {
- log.Errorf("GetDisks fail %s", err)
- return nil, err
- }
- snapshots = append(snapshots, parts...)
- if len(snapshots) >= total {
- break
- }
- }
- isnapshots := make([]cloudprovider.ICloudSnapshot, len(snapshots))
- for i := 0; i < len(snapshots); i++ {
- snapshots[i].region = self.storage.zone.region
- isnapshots[i] = &snapshots[i]
- }
- return isnapshots, nil
- }
- func (self *SDisk) GetTemplateId() string {
- //return self.ImageId
- return ""
- }
- func (self *SRegion) ResetDisk(diskId, snapshotId string) error {
- params := make(map[string]string)
- params["Region"] = self.Region
- params["DiskId"] = diskId
- params["SnapshotId"] = snapshotId
- _, err := self.cbsRequest("ApplySnapshot", params)
- if err != nil {
- log.Errorf("ResetDisk %s to snapshot %s fail %s", diskId, snapshotId, err)
- return err
- }
- return nil
- }
- func (self *SDisk) Reset(ctx context.Context, snapshotId string) (string, error) {
- return "", self.storage.zone.region.ResetDisk(self.DiskId, snapshotId)
- }
- func (self *SRegion) CreateDisk(zoneId string, category string, opts *cloudprovider.DiskCreateConfig) (string, error) {
- params := make(map[string]string)
- params["Region"] = self.Region
- params["DiskType"] = category
- params["DiskChargeType"] = "POSTPAID_BY_HOUR"
- // [TencentCloudSDKError] Code=InvalidParameter, Message=DiskName: vdisk_stress-testvm-qcloud-1_1560117118026502729, length is 48, out of range [0,20] (e11d6c4007e4), RequestId=a8409994-0357-42e9-b028-e11d6c4007e4
- if len(opts.Name) > 20 {
- opts.Name = opts.Name[:20]
- }
- params["DiskName"] = opts.Name
- params["Placement.Zone"] = zoneId
- if len(opts.ProjectId) > 0 {
- params["Placement.ProjectId"] = opts.ProjectId
- }
- //params["Encrypted"] = "false"
- params["DiskSize"] = fmt.Sprintf("%d", opts.SizeGb)
- params["ClientToken"] = utils.GenRequestId(20)
- tagIdx := 0
- for k, v := range opts.Tags {
- params[fmt.Sprintf("Tags.%d.Key", tagIdx)] = k
- params[fmt.Sprintf("Tags.%d.Value", tagIdx)] = v
- tagIdx += 1
- }
- body, err := self.cbsRequest("CreateDisks", params)
- if err != nil {
- return "", err
- }
- diskIDSet := []string{}
- err = body.Unmarshal(&diskIDSet, "DiskIdSet")
- if err != nil {
- return "", err
- }
- if len(diskIDSet) < 1 {
- return "", fmt.Errorf("Create Disk error")
- }
- return diskIDSet[0], nil
- }
- func (disk *SDisk) GetAccessPath() string {
- return ""
- }
- func (self *SDisk) Rebuild(ctx context.Context) error {
- // TODO
- return cloudprovider.ErrNotSupported
- }
- func (self *SDisk) GetProjectId() string {
- return strconv.Itoa(self.Placement.ProjectId)
- }
|