parent
							
								
									84b7a80e1b
								
							
						
					
					
						commit
						db1760cb72
					
				| @ -0,0 +1,480 @@ | ||||
| /* | ||||
| Copyright (c) 2019-2024 Digital Energy Cloud Solutions LLC. All Rights Reserved. | ||||
| Authors: | ||||
| Petr Krutov, <petr.krutov@digitalenergy.online> | ||||
| Stanislav Solovev, <spsolovev@digitalenergy.online> | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /* | ||||
| Terraform DECORT provider - manage resources provided by DECORT (Digital Energy Cloud | ||||
| Orchestration Technology) with Terraform by Hashicorp. | ||||
| 
 | ||||
| Source code: https://repository.basistech.ru/BASIS/terraform-provider-decort
 | ||||
| 
 | ||||
| Please see README.md to learn where to place source code so that it | ||||
| builds seamlessly. | ||||
| 
 | ||||
| Documentation: https://repository.basistech.ru/BASIS/terraform-provider-decort/wiki
 | ||||
| */ | ||||
| 
 | ||||
| package disks | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/google/uuid" | ||||
| 	log "github.com/sirupsen/logrus" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/disks" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/constants" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/controller" | ||||
| 
 | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/diag" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
| 
 | ||||
| func dataSourceDiskReplicationRead(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("dataSourceDiskReplicationRead: called for disk with ID: %s", d.Id()) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	req := disks.ReplicationStatusRequest{ | ||||
| 		DiskID: uint64(d.Get("disk_id").(int)), | ||||
| 	} | ||||
| 
 | ||||
| 	status, err := c.CloudAPI().Disks().ReplicationStatus(ctx, req) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	id := uuid.New() | ||||
| 	d.SetId(id.String()) | ||||
| 
 | ||||
| 	disk, err := utilityDiskReplicaCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	flattenDiskReplica(d, disk, status) | ||||
| 
 | ||||
| 	log.Debugf("dataSourceDiskReplicationRead: read complete for disk with ID: %s", d.Id()) | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func dataSourceDiskReplicationSchemaMake() map[string]*schema.Schema { | ||||
| 	rets := map[string]*schema.Schema{ | ||||
| 		"disk_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Id of primary disk", | ||||
| 		}, | ||||
| 		"replica_disk_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Id of secondary disk", | ||||
| 		}, | ||||
| 		"status_replication": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Status of replication", | ||||
| 		}, | ||||
| 		"account_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "The unique ID of the subscriber-owner of the disk", | ||||
| 		}, | ||||
| 		"account_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "The name of the subscriber '(account') to whom this disk belongs", | ||||
| 		}, | ||||
| 		"acl": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		// "boot_partition": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Number of disk partitions",
 | ||||
| 		// },
 | ||||
| 		"computes": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"compute_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"compute_name": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"created_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Created time", | ||||
| 		}, | ||||
| 		"deleted_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Deleted time", | ||||
| 		}, | ||||
| 		"desc": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Description of disk", | ||||
| 		}, | ||||
| 		"destruction_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Time of final deletion", | ||||
| 		}, | ||||
| 		"devicename": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Name of the device", | ||||
| 		}, | ||||
| 		// "disk_path": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk path",
 | ||||
| 		// },
 | ||||
| 		"gid": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the grid (platform)", | ||||
| 		}, | ||||
| 		// "guid": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk ID on the storage side",
 | ||||
| 		// },
 | ||||
| 		"image_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Image ID", | ||||
| 		}, | ||||
| 		"images": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeString, | ||||
| 			}, | ||||
| 			Description: "IDs of images using the disk", | ||||
| 		}, | ||||
| 		"iotune": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"read_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of bytes to read per second", | ||||
| 					}, | ||||
| 					"read_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of bytes to read", | ||||
| 					}, | ||||
| 					"read_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of io read operations per second", | ||||
| 					}, | ||||
| 					"read_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of io read operations", | ||||
| 					}, | ||||
| 					"size_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Size of io operations", | ||||
| 					}, | ||||
| 					"total_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Total size bytes per second", | ||||
| 					}, | ||||
| 					"total_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum total size of bytes per second", | ||||
| 					}, | ||||
| 					"total_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Total number of io operations per second", | ||||
| 					}, | ||||
| 					"total_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum total number of io operations per second", | ||||
| 					}, | ||||
| 					"write_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of bytes to write per second", | ||||
| 					}, | ||||
| 					"write_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of bytes to write per second", | ||||
| 					}, | ||||
| 					"write_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of write operations per second", | ||||
| 					}, | ||||
| 					"write_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of write operations per second", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		// "iqn": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk IQN",
 | ||||
| 		// },
 | ||||
| 		// "login": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Login to access the disk",
 | ||||
| 		// },
 | ||||
| 		// "milestones": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Milestones",
 | ||||
| 		// },
 | ||||
| 		"disk_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Name of disk", | ||||
| 		}, | ||||
| 		"order": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk order", | ||||
| 		}, | ||||
| 		"params": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk params", | ||||
| 		}, | ||||
| 		"parent_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the parent disk", | ||||
| 		}, | ||||
| 		// "passwd": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Password to access the disk",
 | ||||
| 		// },
 | ||||
| 		"pci_slot": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the pci slot to which the disk is connected", | ||||
| 		}, | ||||
| 		"pool": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Pool for disk location", | ||||
| 		}, | ||||
| 		"present_to": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		// "purge_attempts": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Number of deletion attempts",
 | ||||
| 		// },
 | ||||
| 		"purge_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Time of the last deletion attempt", | ||||
| 		}, | ||||
| 		"replication": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"disk_id": { | ||||
| 						Type: schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"pool_id": { | ||||
| 						Type: schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"role": { | ||||
| 						Type: schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"self_volume_id": { | ||||
| 						Type: schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"storage_id": { | ||||
| 						Type: schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"volume_id": { | ||||
| 						Type: schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Description: "Replication status", | ||||
| 		}, | ||||
| 		// "reality_device_number": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Reality device number",
 | ||||
| 		// },
 | ||||
| 		// "reference_id": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "ID of the reference to the disk",
 | ||||
| 		// },
 | ||||
| 		"res_id": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Resource ID", | ||||
| 		}, | ||||
| 		"res_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Name of the resource", | ||||
| 		}, | ||||
| 		"role": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk role", | ||||
| 		}, | ||||
| 		"sep_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Storage endpoint provider ID to create disk", | ||||
| 		}, | ||||
| 		"sep_type": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Type SEP. Defines the type of storage system and contains one of the values set in the cloud platform", | ||||
| 		}, | ||||
| 		"shareable": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"size_max": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Size in GB", | ||||
| 		}, | ||||
| 		"size_used": { | ||||
| 			Type:        schema.TypeFloat, | ||||
| 			Computed:    true, | ||||
| 			Description: "Number of used space, in GB", | ||||
| 		}, | ||||
| 		"snapshots": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"guid": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "ID of the snapshot", | ||||
| 					}, | ||||
| 					"label": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "Name of the snapshot", | ||||
| 					}, | ||||
| 					"res_id": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "Reference to the snapshot", | ||||
| 					}, | ||||
| 					"snap_set_guid": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "The set snapshot ID", | ||||
| 					}, | ||||
| 					"snap_set_time": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "The set time of the snapshot", | ||||
| 					}, | ||||
| 					"timestamp": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Snapshot time", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"status": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk status", | ||||
| 		}, | ||||
| 		"tech_status": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Technical status of the disk", | ||||
| 		}, | ||||
| 		"type": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "The type of disk in terms of its role in compute: 'B=Boot, D=Data, T=Temp'", | ||||
| 		}, | ||||
| 		"vmid": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Virtual Machine ID (Deprecated)", | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	return rets | ||||
| } | ||||
| 
 | ||||
| func DataSourceDiskReplication() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		SchemaVersion: 1, | ||||
| 
 | ||||
| 		ReadContext: dataSourceDiskReplicationRead, | ||||
| 
 | ||||
| 		Timeouts: &schema.ResourceTimeout{ | ||||
| 			Read:    &constants.Timeout30s, | ||||
| 			Default: &constants.Timeout60s, | ||||
| 		}, | ||||
| 
 | ||||
| 		Schema: dataSourceDiskReplicationSchemaMake(), | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,642 @@ | ||||
| /* | ||||
| Copyright (c) 2019-2024 Digital Energy Cloud Solutions LLC. All Rights Reserved. | ||||
| Authors: | ||||
| Petr Krutov, <petr.krutov@digitalenergy.online> | ||||
| Stanislav Solovev, <spsolovev@digitalenergy.online> | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /* | ||||
| Terraform DECORT provider - manage resources provided by DECORT (Digital Energy Cloud | ||||
| Orchestration Technology) with Terraform by Hashicorp. | ||||
| 
 | ||||
| Source code: https://repository.basistech.ru/BASIS/terraform-provider-decort
 | ||||
| 
 | ||||
| Please see README.md to learn where to place source code so that it | ||||
| builds seamlessly. | ||||
| 
 | ||||
| Documentation: https://repository.basistech.ru/BASIS/terraform-provider-decort/wiki
 | ||||
| */ | ||||
| 
 | ||||
| package disks | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"strconv" | ||||
| 
 | ||||
| 	log "github.com/sirupsen/logrus" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/disks" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/constants" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/controller" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/dc" | ||||
| 
 | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/diag" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
| 
 | ||||
| func resourceDiskReplicationCreate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 
 | ||||
| 	log.Debugf("resourceDiskReplicationCreate: called for disk with ID: %d", diskId) | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	err := existDiskID(ctx, diskId, m) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	reqCreate := disks.ReplicateRequest{ | ||||
| 		DiskID:   diskId, | ||||
| 		Name:     d.Get("disk_name").(string), | ||||
| 		SepID:    uint64(d.Get("sep_id").(int)), | ||||
| 		PoolName: d.Get("pool_name").(string), | ||||
| 	} | ||||
| 
 | ||||
| 	diskReplicaId, err := c.CloudAPI().Disks().Replicate(ctx, reqCreate) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	d.SetId(strconv.FormatUint(diskReplicaId, 10)) | ||||
| 	d.Set("replica_disk_id", diskReplicaId) | ||||
| 
 | ||||
| 	log.Debugf("resourceDiskReplicationCreate: create replica complete for disk with ID: %d", diskId) | ||||
| 
 | ||||
| 	warnings := dc.Warnings{} | ||||
| 
 | ||||
| 	if start, ok := d.GetOk("start"); ok && !start.(bool) { | ||||
| 		log.Debugf("resourceDiskReplicationCreate: replication between disk with ID: %d and replica with ID: %d, try to stop", diskId, diskReplicaId) | ||||
| 		reqStop := disks.ReplicationStopRequest{ | ||||
| 			DiskID: diskId, | ||||
| 		} | ||||
| 		_, err = c.CloudAPI().Disks().ReplicationStop(ctx, reqStop) | ||||
| 		if err != nil { | ||||
| 			warnings.Add(err) | ||||
| 		} | ||||
| 		log.Debugf("resourceDiskReplicationCreate: replication between disk with ID: %d and replica with ID: %d, stoped", diskId, diskReplicaId) | ||||
| 	} | ||||
| 	return append(resourceDiskReplicationRead(ctx, d, m), warnings.Get()...) | ||||
| } | ||||
| 
 | ||||
| func resourceDiskReplicationRead(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceDiskReplicationRead: called for disk with ID: %s", d.Id()) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	req := disks.ReplicationStatusRequest{ | ||||
| 		DiskID: uint64(d.Get("disk_id").(int)), | ||||
| 	} | ||||
| 
 | ||||
| 	status, err := c.CloudAPI().Disks().ReplicationStatus(ctx, req) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	diskReplica, err := utilityDiskReplicaCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	flattenDiskReplica(d, diskReplica, status) | ||||
| 
 | ||||
| 	log.Debugf("resourceDiskReplicationRead: read complete for disk with ID: %s", d.Id()) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceDiskReplicationUpdate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	log.Debugf("resourceDiskReplicationUpdate: called for disk with ID: %d", diskId) | ||||
| 
 | ||||
| 	err := existDiskID(ctx, diskId, m) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	if d.HasChange("start") { | ||||
| 		if err := utilityDiskReplicationUpdateStartStop(ctx, d, m); err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if d.HasChange("pause") { | ||||
| 		if err := utilityDiskReplicationUpdatePause(ctx, d, m); err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if d.HasChange("reverse") { | ||||
| 		if err := utilityDiskReplicationUpdateReverse(ctx, d, m); err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("resourceDiskReplicationUpdate: read complete for disk with ID: %d", diskId) | ||||
| 	return resourceDiskReplicationRead(ctx, d, m) | ||||
| } | ||||
| 
 | ||||
| func resourceDiskReplicationDelete(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	log.Debugf("resourceDiskReplicationDelete: called for disk with ID: %d", diskId) | ||||
| 
 | ||||
| 	disk, err := utilityDiskReplicaCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	if d.Get("start").(bool) { | ||||
| 		reqStop := disks.ReplicationStopRequest{ | ||||
| 			DiskID: uint64(d.Get("disk_id").(int)), | ||||
| 		} | ||||
| 
 | ||||
| 		log.Debugf("resourceDiskReplicationDelete: stop replication for disk with ID: %d", diskId) | ||||
| 		_, err = c.CloudAPI().Disks().ReplicationStop(ctx, reqStop) | ||||
| 		if err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 		log.Debugf("resourceDiskReplicationDelete: stop replication for disk with ID: %d, complete", diskId) | ||||
| 	} | ||||
| 
 | ||||
| 	reqDelete := disks.DeleteRequest{ | ||||
| 		DiskID:      disk.ID, | ||||
| 		Detach:      d.Get("detach").(bool), | ||||
| 		Permanently: d.Get("permanently").(bool), | ||||
| 		Reason:      d.Get("reason").(string), | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("resourceDiskReplicationDelete: delete disk replica for disk with ID: %d", diskId) | ||||
| 	_, err = c.CloudAPI().Disks().Delete(ctx, reqDelete) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 	log.Debugf("resourceDiskReplicationDelete: delete disk replica for disk with ID: %d, complete", diskId) | ||||
| 
 | ||||
| 	d.SetId("") | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceDiskReplicationSchemaMake() map[string]*schema.Schema { | ||||
| 	rets := map[string]*schema.Schema{ | ||||
| 		"disk_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Id of primary disk", | ||||
| 		}, | ||||
| 		"disk_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Required:    true, | ||||
| 			Description: "Name of disk replica", | ||||
| 		}, | ||||
| 		"sep_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Storage endpoint provider ID to create disk replica", | ||||
| 		}, | ||||
| 		"pool_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Required:    true, | ||||
| 			Description: "Pool for disk location", | ||||
| 		}, | ||||
| 		"pause": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Description: "Resume replication", | ||||
| 		}, | ||||
| 		"reverse": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "Reverse replication", | ||||
| 		}, | ||||
| 		"start": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     true, | ||||
| 			Description: "Start/Stop replication", | ||||
| 		}, | ||||
| 		"detach": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "Detach disk from machine first", | ||||
| 		}, | ||||
| 		"permanently": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "Delete disk permanently", | ||||
| 		}, | ||||
| 		"reason": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Description: "Reason for disk deletion", | ||||
| 		}, | ||||
| 		"replica_disk_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Id of replica disk", | ||||
| 		}, | ||||
| 		"status_replication": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Status of replication", | ||||
| 		}, | ||||
| 		"account_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "The unique ID of the subscriber-owner of the disk", | ||||
| 		}, | ||||
| 		"account_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "The name of the subscriber '(account') to whom this disk belongs", | ||||
| 		}, | ||||
| 		"acl": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		// "boot_partition": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Number of disk partitions",
 | ||||
| 		// },
 | ||||
| 		"computes": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"compute_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"compute_name": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"created_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Created time", | ||||
| 		}, | ||||
| 		"deleted_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Deleted time", | ||||
| 		}, | ||||
| 		"desc": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Description of disk", | ||||
| 		}, | ||||
| 		"destruction_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Time of final deletion", | ||||
| 		}, | ||||
| 		"devicename": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Name of the device", | ||||
| 		}, | ||||
| 		// "disk_path": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk path",
 | ||||
| 		// },
 | ||||
| 		"gid": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the grid (platform)", | ||||
| 		}, | ||||
| 		// "guid": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk ID on the storage side",
 | ||||
| 		// },
 | ||||
| 		"image_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Image ID", | ||||
| 		}, | ||||
| 		"images": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeString, | ||||
| 			}, | ||||
| 			Description: "IDs of images using the disk", | ||||
| 		}, | ||||
| 		"iotune": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"read_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of bytes to read per second", | ||||
| 					}, | ||||
| 					"read_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of bytes to read", | ||||
| 					}, | ||||
| 					"read_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of io read operations per second", | ||||
| 					}, | ||||
| 					"read_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of io read operations", | ||||
| 					}, | ||||
| 					"size_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Size of io operations", | ||||
| 					}, | ||||
| 					"total_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Total size bytes per second", | ||||
| 					}, | ||||
| 					"total_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum total size of bytes per second", | ||||
| 					}, | ||||
| 					"total_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Total number of io operations per second", | ||||
| 					}, | ||||
| 					"total_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum total number of io operations per second", | ||||
| 					}, | ||||
| 					"write_bytes_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of bytes to write per second", | ||||
| 					}, | ||||
| 					"write_bytes_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of bytes to write per second", | ||||
| 					}, | ||||
| 					"write_iops_sec": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Number of write operations per second", | ||||
| 					}, | ||||
| 					"write_iops_sec_max": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Maximum number of write operations per second", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		// "iqn": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Disk IQN",
 | ||||
| 		// },
 | ||||
| 		// "login": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Login to access the disk",
 | ||||
| 		// },
 | ||||
| 		// "milestones": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Milestones",
 | ||||
| 		// },
 | ||||
| 		"order": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk order", | ||||
| 		}, | ||||
| 		"params": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk params", | ||||
| 		}, | ||||
| 		"parent_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the parent disk", | ||||
| 		}, | ||||
| 		// "passwd": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Password to access the disk",
 | ||||
| 		// },
 | ||||
| 		"pci_slot": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "ID of the pci slot to which the disk is connected", | ||||
| 		}, | ||||
| 		"present_to": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		// "purge_attempts": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Number of deletion attempts",
 | ||||
| 		// },
 | ||||
| 		"purge_time": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Time of the last deletion attempt", | ||||
| 		}, | ||||
| 		"replication": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"disk_id": { | ||||
| 						Type:     schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"pool_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"role": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"self_volume_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"storage_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"volume_id": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Description: "Replication status", | ||||
| 		}, | ||||
| 		// "reality_device_number": {
 | ||||
| 		// 	Type:        schema.TypeInt,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "Reality device number",
 | ||||
| 		// },
 | ||||
| 		// "reference_id": {
 | ||||
| 		// 	Type:        schema.TypeString,
 | ||||
| 		// 	Computed:    true,
 | ||||
| 		// 	Description: "ID of the reference to the disk",
 | ||||
| 		// },
 | ||||
| 		"res_id": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Resource ID", | ||||
| 		}, | ||||
| 		"res_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Name of the resource", | ||||
| 		}, | ||||
| 		"role": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk role", | ||||
| 		}, | ||||
| 		"sep_type": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Type SEP. Defines the type of storage system and contains one of the values set in the cloud platform", | ||||
| 		}, | ||||
| 		"shareable": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"size_max": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Size in GB", | ||||
| 		}, | ||||
| 		"size_used": { | ||||
| 			Type:        schema.TypeFloat, | ||||
| 			Computed:    true, | ||||
| 			Description: "Number of used space, in GB", | ||||
| 		}, | ||||
| 		"snapshots": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"guid": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "ID of the snapshot", | ||||
| 					}, | ||||
| 					"label": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "Name of the snapshot", | ||||
| 					}, | ||||
| 					"res_id": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "Reference to the snapshot", | ||||
| 					}, | ||||
| 					"snap_set_guid": { | ||||
| 						Type:        schema.TypeString, | ||||
| 						Computed:    true, | ||||
| 						Description: "The set snapshot ID", | ||||
| 					}, | ||||
| 					"snap_set_time": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "The set time of the snapshot", | ||||
| 					}, | ||||
| 					"timestamp": { | ||||
| 						Type:        schema.TypeInt, | ||||
| 						Computed:    true, | ||||
| 						Description: "Snapshot time", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"status": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Disk status", | ||||
| 		}, | ||||
| 		"tech_status": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "Technical status of the disk", | ||||
| 		}, | ||||
| 		"type": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Computed:    true, | ||||
| 			Description: "The type of disk in terms of its role in compute: 'B=Boot, D=Data, T=Temp'", | ||||
| 		}, | ||||
| 		"vmid": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Computed:    true, | ||||
| 			Description: "Virtual Machine ID (Deprecated)", | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	return rets | ||||
| } | ||||
| 
 | ||||
| func ResourceDiskReplication() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		SchemaVersion: 1, | ||||
| 
 | ||||
| 		CreateContext: resourceDiskReplicationCreate, | ||||
| 		ReadContext:   resourceDiskReplicationRead, | ||||
| 		UpdateContext: resourceDiskReplicationUpdate, | ||||
| 		DeleteContext: resourceDiskReplicationDelete, | ||||
| 
 | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 
 | ||||
| 		Timeouts: &schema.ResourceTimeout{ | ||||
| 			Create:  &constants.Timeout600s, | ||||
| 			Read:    &constants.Timeout300s, | ||||
| 			Update:  &constants.Timeout300s, | ||||
| 			Delete:  &constants.Timeout300s, | ||||
| 			Default: &constants.Timeout300s, | ||||
| 		}, | ||||
| 
 | ||||
| 		Schema: resourceDiskReplicationSchemaMake(), | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,174 @@ | ||||
| /* | ||||
| Copyright (c) 2019-2024 Digital Energy Cloud Solutions LLC. All Rights Reserved. | ||||
| Authors: | ||||
| Petr Krutov, <petr.krutov@digitalenergy.online> | ||||
| Stanislav Solovev, <spsolovev@digitalenergy.online> | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /* | ||||
| Terraform DECORT provider - manage resources provided by DECORT (Digital Energy Cloud | ||||
| Orchestration Technology) with Terraform by Hashicorp. | ||||
| 
 | ||||
| Source code: https://repository.basistech.ru/BASIS/terraform-provider-decort
 | ||||
| 
 | ||||
| Please see README.md to learn where to place source code so that it | ||||
| builds seamlessly. | ||||
| 
 | ||||
| Documentation: https://repository.basistech.ru/BASIS/terraform-provider-decort/wiki
 | ||||
| */ | ||||
| 
 | ||||
| package disks | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"strconv" | ||||
| 
 | ||||
| 	log "github.com/sirupsen/logrus" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/disks" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/controller" | ||||
| 
 | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| ) | ||||
| 
 | ||||
| func utilityDiskReplicationUpdateStartStop(ctx context.Context, d *schema.ResourceData, m interface{}) error { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	targetDiskId, _ := strconv.ParseUint(d.Id(), 10, 64) | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicationUpdateStartStop: start update for disk replica with ID: %d", diskId) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	start, ok := d.GetOk("start") | ||||
| 
 | ||||
| 	if ok && start.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicationUpdateStartStop: start disk replication from Disk with ID: %d to Disk with ID: %d", diskId, targetDiskId) | ||||
| 		req := disks.ReplicationStartRequest{ | ||||
| 			DiskID: diskId, | ||||
| 			TargetDiskID: targetDiskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationStart(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicationUpdateStartStop: start disk replication from Disk with ID: %d to Disk with ID: %d, complete", diskId, targetDiskId) | ||||
| 	} | ||||
| 
 | ||||
| 	if ok && !start.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicationUpdateStartStop: stop disk replication from Disk with ID: %d to Disk with ID: %d", targetDiskId, diskId) | ||||
| 		req := disks.ReplicationStopRequest{ | ||||
| 			DiskID: targetDiskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationStop(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicationUpdateStartStop: stop disk replication from Disk with ID: %d to Disk with ID: %d, complete", targetDiskId, diskId) | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicationUpdateStartStop: complete update for disk replica with ID: %d", diskId) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func utilityDiskReplicationUpdatePause(ctx context.Context, d *schema.ResourceData, m interface{}) error { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	log.Debugf("utilityDiskReplicationUpdatePause: start update for disk replica with ID: %d", diskId) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	pause, ok := d.GetOk("pause") | ||||
| 
 | ||||
| 	if ok && pause.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicationUpdatePause: pause disk replication with ID: %d", diskId) | ||||
| 		req := disks.ReplicationSuspendRequest{ | ||||
| 			DiskID: diskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationSuspend(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicationUpdatePause: pause disk replication with ID: %d, complete", diskId) | ||||
| 	} | ||||
| 
 | ||||
| 	if ok && !pause.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicationUpdatePause: resume disk replication with ID: %d", diskId) | ||||
| 		req := disks.ReplicationResumeRequest{ | ||||
| 			DiskID: diskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationResume(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicationUpdatePause: resume disk replication with ID: %d, complete", diskId) | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicationUpdatePause: complete update for disk replica with ID: %d", diskId) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func utilityDiskReplicationUpdateReverse(ctx context.Context, d *schema.ResourceData, m interface{}) error { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	targetDiskId, _ := strconv.ParseUint(d.Id(), 10, 64) | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicaUpdateReverse: start update for disk replica with ID: %d", diskId) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	reverse, ok := d.GetOk("reverse") | ||||
| 
 | ||||
| 	if ok && reverse.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicaUpdateReverse: reverse disk replication from Disk with ID: %d to Disk with ID: %d", diskId, targetDiskId) | ||||
| 		req := disks.ReplicationReverseRequest{ | ||||
| 			DiskID: diskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationReverse(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicaUpdateReverse: reverse disk replication from Disk with ID: %d to Disk with ID: %d, complete", diskId, targetDiskId) | ||||
| 	} | ||||
| 
 | ||||
| 	if ok && !reverse.(bool) { | ||||
| 		log.Debugf("utilityDiskReplicaUpdateReverse: reverse disk replication from Disk with ID: %d to Disk with ID: %d", targetDiskId, diskId) | ||||
| 		req := disks.ReplicationReverseRequest{ | ||||
| 			DiskID: targetDiskId, | ||||
| 		} | ||||
| 		_, err := c.CloudAPI().Disks().ReplicationReverse(ctx, req) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		log.Debugf("utilityDiskReplicaUpdateReverse: reverse disk replication from Disk with ID: %d to Disk with ID: %d, complete", targetDiskId, diskId) | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicaUpdateReverse: complete update for disk replica with ID: %d", diskId) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func utilityDiskReplicaCheckPresence(ctx context.Context, d *schema.ResourceData, m interface{}) (*disks.RecordDisk, error) { | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 
 | ||||
| 	req := disks.GetRequest{} | ||||
| 
 | ||||
| 	if d.Id() != "" { | ||||
| 		diskId, _ := strconv.ParseUint(d.Id(), 10, 64) | ||||
| 		req.DiskID = diskId | ||||
| 	} else { | ||||
| 		req.DiskID = uint64(d.Get("replica_disk_id").(int)) | ||||
| 	} | ||||
| 
 | ||||
| 	log.Debugf("utilityDiskReplicaCheckPresence: load disk") | ||||
| 	disk, err := c.CloudAPI().Disks().Get(ctx, req) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return disk, nil | ||||
| } | ||||
| @ -0,0 +1,508 @@ | ||||
| /* | ||||
| Copyright (c) 2019-2022 Digital Energy Cloud Solutions LLC. All Rights Reserved. | ||||
| Authors: | ||||
| Petr Krutov, <petr.krutov@digitalenergy.online> | ||||
| Stanislav Solovev, <spsolovev@digitalenergy.online> | ||||
| Kasim Baybikov, <kmbaybikov@basistech.ru> | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /* | ||||
| Terraform DECORT provider - manage resources provided by DECORT (Digital Energy Cloud | ||||
| Orchestration Technology) with Terraform by Hashicorp. | ||||
| 
 | ||||
| Source code: https://repository.basistech.ru/BASIS/terraform-provider-decort
 | ||||
| 
 | ||||
| Please see README.md to learn where to place source code so that it | ||||
| builds seamlessly. | ||||
| 
 | ||||
| Documentation: https://repository.basistech.ru/BASIS/terraform-provider-decort/wiki
 | ||||
| */ | ||||
| 
 | ||||
| package image | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/diag" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation" | ||||
| 	log "github.com/sirupsen/logrus" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/compute" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/image" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/tasks" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/constants" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/controller" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/status" | ||||
| ) | ||||
| 
 | ||||
| func resourceImageFromBlankComputeCreate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	computeId := uint64(d.Get("compute_id").(int)) | ||||
| 	name := d.Get("name").(string) | ||||
| 
 | ||||
| 	log.Debugf("resourceImageFromBlankComputeCreate: called for image %s", name) | ||||
| 
 | ||||
| 	err := existComputeID(ctx, computeId, m) | ||||
| 	if err != nil { | ||||
| 		return diag.Errorf("resourceImageFromBlankComputeCreate: can't create Image: %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	if _, ok := d.GetOk("account_id"); ok { | ||||
| 		haveAccountID, err := existAccountID(ctx, d, m) | ||||
| 		if err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 
 | ||||
| 		if !haveAccountID { | ||||
| 			return diag.Errorf("resourceImageFromBlankComputeCreate: can't create Image because AccountID %d is not allowed or does not exist", d.Get("account_id").(int)) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 	req := compute.CreateTemplateFromBlankRequest{ | ||||
| 		ComputeID: computeId, | ||||
| 		Name:      name, | ||||
| 		BootType:  d.Get("boot_type").(string), | ||||
| 		ImageType: d.Get("type").(string), | ||||
| 	} | ||||
| 
 | ||||
| 	if username, ok := d.GetOk("username"); ok { | ||||
| 		req.Username = username.(string) | ||||
| 	} | ||||
| 	if password, ok := d.GetOk("password"); ok { | ||||
| 		req.Password = password.(string) | ||||
| 	} | ||||
| 	if accountId, ok := d.GetOk("account_id"); ok { | ||||
| 		req.AccountID = uint64(accountId.(int)) | ||||
| 	} | ||||
| 	if sepId, ok := d.GetOk("sep_id"); ok { | ||||
| 		req.SepID = uint64(sepId.(int)) | ||||
| 	} | ||||
| 	if poolName, ok := d.GetOk("pool_name"); ok { | ||||
| 		req.PoolName = poolName.(string) | ||||
| 	} | ||||
| 	if hotresize, ok := d.GetOk("hot_resize"); ok { | ||||
| 		req.HotResize = hotresize.(bool) | ||||
| 	} | ||||
| 
 | ||||
| 	var imageId uint64 | ||||
| 	asyncMode := d.Get("async_mode").(bool) | ||||
| 	if !asyncMode { | ||||
| 		imageId, err = c.CloudAPI().Compute().CreateTemplateFromBlank(ctx, req) | ||||
| 		if err != nil { | ||||
| 			d.SetId("") | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} else { | ||||
| 		taskId, err := c.CloudAPI().Compute().CreateTemplateFromBlankAsync(ctx, req) | ||||
| 		if err != nil { | ||||
| 			d.SetId("") | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 
 | ||||
| 		taskReq := tasks.GetRequest{ | ||||
| 			AuditID: strings.Trim(taskId, `"`), | ||||
| 		} | ||||
| 
 | ||||
| 		for { | ||||
| 			task, err := c.CloudAPI().Tasks().Get(ctx, taskReq) | ||||
| 			if err != nil { | ||||
| 				return diag.FromErr(err) | ||||
| 			} | ||||
| 
 | ||||
| 			log.Debugf("resourceImageFromBlankComputeCreate: instance creating - %s", task.Stage) | ||||
| 
 | ||||
| 			if task.Completed { | ||||
| 				if task.Error != "" { | ||||
| 					return diag.FromErr(fmt.Errorf("cannot create image instance: %v", task.Error)) | ||||
| 				} | ||||
| 
 | ||||
| 				id, err := task.Result.ID() | ||||
| 				imageId = uint64(id) | ||||
| 				if err != nil { | ||||
| 					return diag.FromErr(err) | ||||
| 				} | ||||
| 				break | ||||
| 			} | ||||
| 
 | ||||
| 			time.Sleep(time.Second * 20) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	d.SetId(strconv.FormatUint(imageId, 10)) | ||||
| 	d.Set("image_id", imageId) | ||||
| 
 | ||||
| 	_, err = utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	return resourceImageFromBlankComputeRead(ctx, d, m) | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromBlankComputeRead(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromBlankComputeRead: called for %s id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	img, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if img == nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	switch img.Status { | ||||
| 	case status.Modeled: | ||||
| 		return diag.Errorf("The image is in status: %s, please, contact support for more information", img.Status) | ||||
| 	case status.Creating: | ||||
| 	case status.Created: | ||||
| 	case status.Destroyed, status.Purged: | ||||
| 		d.SetId("") | ||||
| 		return diag.Errorf("The resource cannot be updated because it has been destroyed") | ||||
| 		// return resourceImageCreate(ctx, d, m)
 | ||||
| 	} | ||||
| 
 | ||||
| 	flattenImage(d, img) | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromBlankComputeDelete(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromBlankComputeDelete: called for %s, id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	_, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 	req := image.DeleteRequest{ | ||||
| 		ImageID: uint64(d.Get("image_id").(int)), | ||||
| 	} | ||||
| 
 | ||||
| 	if permanently, ok := d.GetOk("permanently"); ok { | ||||
| 		req.Permanently = permanently.(bool) | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = c.CloudAPI().Image().Delete(ctx, req) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	d.SetId("") | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromBlankComputeRename(ctx context.Context, d *schema.ResourceData, m interface{}) error { | ||||
| 	log.Debugf("resourceImageFromBlankComputeRename: called for %s, id: %s", d.Get("name").(string), d.Id()) | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 	req := image.RenameRequest{ | ||||
| 		ImageID: uint64(d.Get("image_id").(int)), | ||||
| 		Name:    d.Get("name").(string), | ||||
| 	} | ||||
| 
 | ||||
| 	_, err := c.CloudAPI().Image().Rename(ctx, req) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromBlankComputeUpdate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromBlankComputeUpdate: called for %s, id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	// we do not allow change of compute_id, but allow resource update after import
 | ||||
| 	old, _ := d.GetChange("compute_id") | ||||
| 	if old.(int) != 0 && d.HasChange("compute_id") { | ||||
| 		return diag.Errorf("resourceImageFromBlankComputeUpdate: can't update Image because compute_id is not allowed to be changed") | ||||
| 	} | ||||
| 
 | ||||
| 	image, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	switch image.Status { | ||||
| 	case status.Modeled: | ||||
| 		return diag.Errorf("The image is in status: %s, please, contact support for more information", image.Status) | ||||
| 	case status.Creating: | ||||
| 	case status.Created: | ||||
| 	case status.Destroyed, status.Purged: | ||||
| 		d.SetId("") | ||||
| 		return diag.Errorf("The resource cannot be updated because it has been destroyed") | ||||
| 		// return resourceImageCreate(ctx, d, m)
 | ||||
| 	} | ||||
| 
 | ||||
| 	if d.HasChange("name") { | ||||
| 		err := resourceImageFromBlankComputeRename(ctx, d, m) | ||||
| 		if err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return resourceImageFromBlankComputeRead(ctx, d, m) | ||||
| } | ||||
| 
 | ||||
| func ResourceImageFromBlankCompute() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		SchemaVersion: 1, | ||||
| 
 | ||||
| 		CreateContext: resourceImageFromBlankComputeCreate, | ||||
| 		ReadContext:   resourceImageFromBlankComputeRead, | ||||
| 		UpdateContext: resourceImageFromBlankComputeUpdate, | ||||
| 		DeleteContext: resourceImageFromBlankComputeDelete, | ||||
| 
 | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 
 | ||||
| 		Timeouts: &schema.ResourceTimeout{ | ||||
| 			Create:  &constants.Timeout30m, | ||||
| 			Read:    &constants.Timeout900s, | ||||
| 			Update:  &constants.Timeout900s, | ||||
| 			Delete:  &constants.Timeout900s, | ||||
| 			Default: &constants.Timeout900s, | ||||
| 		}, | ||||
| 
 | ||||
| 		Schema: resourceImageFromBlankComputeSchemaMake(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromBlankComputeSchemaMake() map[string]*schema.Schema { | ||||
| 	return map[string]*schema.Schema{ | ||||
| 		"compute_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Compute Id", | ||||
| 		}, | ||||
| 		"name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Required:    true, | ||||
| 			Description: "Name of the rescue disk", | ||||
| 		}, | ||||
| 		"boot_type": { | ||||
| 			Type:         schema.TypeString, | ||||
| 			Required:     true, | ||||
| 			ValidateFunc: validation.StringInSlice([]string{"bios", "uefi"}, true), | ||||
| 			Description:  "Boot type of image BIOS or UEFI", | ||||
| 		}, | ||||
| 		"type": { | ||||
| 			Type:         schema.TypeString, | ||||
| 			Required:     true, | ||||
| 			ValidateFunc: validation.StringInSlice([]string{"linux", "windows", "other"}, true), | ||||
| 			Description:  "Image type linux, windows or other", | ||||
| 		}, | ||||
| 
 | ||||
| 		"username": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Optional username for the image", | ||||
| 		}, | ||||
| 		"password": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Optional password for the image", | ||||
| 		}, | ||||
| 		"account_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "AccountId to make the image exclusive", | ||||
| 		}, | ||||
| 		"sep_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "storage endpoint provider ID", | ||||
| 		}, | ||||
| 		"pool_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "pool for image create", | ||||
| 		}, | ||||
| 		"hot_resize": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Does this machine supports hot resize", | ||||
| 		}, | ||||
| 		"async_mode": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "create an image in async/sync mode", | ||||
| 		}, | ||||
| 		"permanently": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "whether to completely delete the image", | ||||
| 		}, | ||||
| 
 | ||||
| 		"image_id": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"unc_path": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"ckey": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"acl": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"architecture": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"bootable": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"compute_ci_id": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"cd_presented_to": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"deleted_time": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"desc": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"drivers": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeString, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"enabled": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"gid": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"guid": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"history": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"guid": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"id": { | ||||
| 						Type:     schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"timestamp": { | ||||
| 						Type:     schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"last_modified": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"link_to": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"milestones": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"image_name": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"network_interface_naming": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"provider_name": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"purge_attempts": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"present_to": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"res_id": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"rescuecd": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"shared_with": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"size": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"status": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"tech_status": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"version": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
| @ -0,0 +1,506 @@ | ||||
| /* | ||||
| Copyright (c) 2019-2022 Digital Energy Cloud Solutions LLC. All Rights Reserved. | ||||
| Authors: | ||||
| Petr Krutov, <petr.krutov@digitalenergy.online> | ||||
| Stanislav Solovev, <spsolovev@digitalenergy.online> | ||||
| Kasim Baybikov, <kmbaybikov@basistech.ru> | ||||
| 
 | ||||
| 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. | ||||
| */ | ||||
| 
 | ||||
| /* | ||||
| Terraform DECORT provider - manage resources provided by DECORT (Digital Energy Cloud | ||||
| Orchestration Technology) with Terraform by Hashicorp. | ||||
| 
 | ||||
| Source code: https://repository.basistech.ru/BASIS/terraform-provider-decort
 | ||||
| 
 | ||||
| Please see README.md to learn where to place source code so that it | ||||
| builds seamlessly. | ||||
| 
 | ||||
| Documentation: https://repository.basistech.ru/BASIS/terraform-provider-decort/wiki
 | ||||
| */ | ||||
| 
 | ||||
| package image | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/diag" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" | ||||
| 	"github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation" | ||||
| 	log "github.com/sirupsen/logrus" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/disks" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/image" | ||||
| 	"repository.basistech.ru/BASIS/decort-golang-sdk/pkg/cloudapi/tasks" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/constants" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/controller" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/statefuncs" | ||||
| 	"repository.basistech.ru/BASIS/terraform-provider-decort/internal/status" | ||||
| ) | ||||
| 
 | ||||
| func resourceImageFromPlatformDiskCreate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	diskId := uint64(d.Get("disk_id").(int)) | ||||
| 	name := d.Get("name").(string) | ||||
| 
 | ||||
| 	log.Debugf("resourceImageFromPlatformDiskCreate: called for image %s", name) | ||||
| 
 | ||||
| 	err := existDiskID(ctx, diskId, m) | ||||
| 	if err != nil { | ||||
| 		return diag.Errorf("resourceImageFromPlatformDiskCreate: can't create Image: %v", err) | ||||
| 	} | ||||
| 
 | ||||
| 	if _, ok := d.GetOk("account_id"); ok { | ||||
| 		haveAccountID, err := existAccountID(ctx, d, m) | ||||
| 		if err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 
 | ||||
| 		if !haveAccountID { | ||||
| 			return diag.Errorf("resourceImageFromPlatformDiskCreate: can't create Image because AccountID %d is not allowed or does not exist", d.Get("account_id").(int)) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 	req := disks.FromPlatformDiskRequest{ | ||||
| 		DiskID:       diskId, | ||||
| 		Name:         name, | ||||
| 		BootType:     d.Get("boot_type").(string), | ||||
| 		ImageType:    d.Get("type").(string), | ||||
| 		Architecture: d.Get("architecture").(string), | ||||
| 		Bootable:     d.Get("bootable").(bool), // default value - true
 | ||||
| 	} | ||||
| 
 | ||||
| 	if username, ok := d.GetOk("username"); ok { | ||||
| 		req.Username = username.(string) | ||||
| 	} | ||||
| 	if password, ok := d.GetOk("password"); ok { | ||||
| 		req.Password = password.(string) | ||||
| 	} | ||||
| 	if accountId, ok := d.GetOk("account_id"); ok { | ||||
| 		req.AccountID = uint64(accountId.(int)) | ||||
| 	} | ||||
| 	if sepId, ok := d.GetOk("sep_id"); ok { | ||||
| 		req.SepID = uint64(sepId.(int)) | ||||
| 	} | ||||
| 	if poolName, ok := d.GetOk("pool_name"); ok { | ||||
| 		req.PoolName = poolName.(string) | ||||
| 	} | ||||
| 	if driversInterface, ok := d.GetOk("drivers"); ok { | ||||
| 		for _, d := range driversInterface.([]interface{}) { | ||||
| 			req.Drivers = append(req.Drivers, d.(string)) | ||||
| 		} | ||||
| 	} | ||||
| 	if hotresize, ok := d.GetOk("hot_resize"); ok { | ||||
| 		req.HotResize = hotresize.(bool) | ||||
| 	} | ||||
| 
 | ||||
| 	var imageId uint64 | ||||
| 	asyncMode := d.Get("async_mode").(bool) | ||||
| 	if !asyncMode { | ||||
| 		imageId, err = c.CloudAPI().Disks().FromPlatformDisk(ctx, req) | ||||
| 		if err != nil { | ||||
| 			d.SetId("") | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} else { | ||||
| 		taskId, err := c.CloudAPI().Disks().FromPlatformDiskAsync(ctx, req) | ||||
| 		if err != nil { | ||||
| 			d.SetId("") | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 
 | ||||
| 		taskReq := tasks.GetRequest{ | ||||
| 			AuditID: strings.Trim(taskId, `"`), | ||||
| 		} | ||||
| 
 | ||||
| 		for { | ||||
| 			task, err := c.CloudAPI().Tasks().Get(ctx, taskReq) | ||||
| 			if err != nil { | ||||
| 				return diag.FromErr(err) | ||||
| 			} | ||||
| 
 | ||||
| 			log.Debugf("resourceImageFromPlatformDiskCreate: instance creating - %s", task.Stage) | ||||
| 
 | ||||
| 			if task.Completed { | ||||
| 				if task.Error != "" { | ||||
| 					return diag.FromErr(fmt.Errorf("cannot create image instance: %v", task.Error)) | ||||
| 				} | ||||
| 
 | ||||
| 				id, err := task.Result.ID() | ||||
| 				imageId = uint64(id) | ||||
| 				if err != nil { | ||||
| 					return diag.FromErr(err) | ||||
| 				} | ||||
| 				break | ||||
| 			} | ||||
| 
 | ||||
| 			time.Sleep(time.Second * 20) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	d.SetId(strconv.FormatUint(imageId, 10)) | ||||
| 	d.Set("image_id", imageId) | ||||
| 
 | ||||
| 	_, err = utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	return resourceImageFromPlatformDiskRead(ctx, d, m) | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromPlatformDiskRead(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromPlatformDiskRead: called for %s id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	img, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if img == nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	switch img.Status { | ||||
| 	case status.Modeled: | ||||
| 		return diag.Errorf("The image is in status: %s, please, contact support for more information", img.Status) | ||||
| 	case status.Creating: | ||||
| 	case status.Created: | ||||
| 	case status.Destroyed, status.Purged: | ||||
| 		d.SetId("") | ||||
| 		return diag.Errorf("The resource cannot be updated because it has been destroyed") | ||||
| 	} | ||||
| 
 | ||||
| 	flattenImage(d, img) | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromPlatformDiskDelete(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromPlatformDiskDelete: called for %s, id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	_, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	c := m.(*controller.ControllerCfg) | ||||
| 	req := image.DeleteRequest{ | ||||
| 		ImageID: uint64(d.Get("image_id").(int)), | ||||
| 	} | ||||
| 
 | ||||
| 	if permanently, ok := d.GetOk("permanently"); ok { | ||||
| 		req.Permanently = permanently.(bool) | ||||
| 	} | ||||
| 
 | ||||
| 	_, err = c.CloudAPI().Image().Delete(ctx, req) | ||||
| 	if err != nil { | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	d.SetId("") | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromPlatformDiskUpdate(ctx context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { | ||||
| 	log.Debugf("resourceImageFromPlatformDiskUpdate: called for %s, id: %s", d.Get("name").(string), d.Id()) | ||||
| 
 | ||||
| 	// we do not allow change of disk_id, but allow resource update after import
 | ||||
| 	old, _ := d.GetChange("disk_id") | ||||
| 	if old.(int) != 0 && d.HasChange("disk_id") { | ||||
| 		return diag.Errorf("resourceImageFromPlatformDiskUpdate: can't update Image because disk_id is not allowed to be changed") | ||||
| 	} | ||||
| 
 | ||||
| 	image, err := utilityImageCheckPresence(ctx, d, m) | ||||
| 	if err != nil { | ||||
| 		d.SetId("") | ||||
| 		return diag.FromErr(err) | ||||
| 	} | ||||
| 
 | ||||
| 	switch image.Status { | ||||
| 	case status.Modeled: | ||||
| 		return diag.Errorf("The image is in status: %s, please, contact support for more information", image.Status) | ||||
| 	case status.Creating: | ||||
| 	case status.Created: | ||||
| 	case status.Destroyed, status.Purged: | ||||
| 		d.SetId("") | ||||
| 		return diag.Errorf("The resource cannot be updated because it has been destroyed") | ||||
| 	} | ||||
| 
 | ||||
| 	if d.HasChange("name") { | ||||
| 		err := resourceImageRename(ctx, d, m) | ||||
| 		if err != nil { | ||||
| 			return diag.FromErr(err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	return resourceImageFromPlatformDiskRead(ctx, d, m) | ||||
| } | ||||
| 
 | ||||
| func ResourceImageFromPlatformDisk() *schema.Resource { | ||||
| 	return &schema.Resource{ | ||||
| 		SchemaVersion: 1, | ||||
| 
 | ||||
| 		CreateContext: resourceImageFromPlatformDiskCreate, | ||||
| 		ReadContext:   resourceImageFromPlatformDiskRead, | ||||
| 		UpdateContext: resourceImageFromPlatformDiskUpdate, | ||||
| 		DeleteContext: resourceImageFromPlatformDiskDelete, | ||||
| 
 | ||||
| 		Importer: &schema.ResourceImporter{ | ||||
| 			StateContext: schema.ImportStatePassthroughContext, | ||||
| 		}, | ||||
| 
 | ||||
| 		Timeouts: &schema.ResourceTimeout{ | ||||
| 			Create:  &constants.Timeout30m, | ||||
| 			Read:    &constants.Timeout900s, | ||||
| 			Update:  &constants.Timeout900s, | ||||
| 			Delete:  &constants.Timeout900s, | ||||
| 			Default: &constants.Timeout900s, | ||||
| 		}, | ||||
| 
 | ||||
| 		Schema: resourceImageFromPlatformDiskSchemaMake(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func resourceImageFromPlatformDiskSchemaMake() map[string]*schema.Schema { | ||||
| 	return map[string]*schema.Schema{ | ||||
| 		"disk_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Required:    true, | ||||
| 			Description: "Disk Id", | ||||
| 		}, | ||||
| 		"name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Required:    true, | ||||
| 			Description: "Name of the rescue disk", | ||||
| 		}, | ||||
| 		"boot_type": { | ||||
| 			Type:         schema.TypeString, | ||||
| 			Required:     true, | ||||
| 			ValidateFunc: validation.StringInSlice([]string{"bios", "uefi"}, true), | ||||
| 			Description:  "Boot type of image BIOS or UEFI", | ||||
| 		}, | ||||
| 		"type": { | ||||
| 			Type:         schema.TypeString, | ||||
| 			Required:     true, | ||||
| 			ValidateFunc: validation.StringInSlice([]string{"linux", "windows", "other"}, true), | ||||
| 			Description:  "Image type linux, windows or other", | ||||
| 		}, | ||||
| 		"architecture": { | ||||
| 			Type:         schema.TypeString, | ||||
| 			Required:     true, | ||||
| 			ValidateFunc: validation.StringInSlice([]string{"X86_64", "PPC64_LE"}, true), | ||||
| 			Description:  "binary architecture of this image, one of X86_64 of PPC64_LE", | ||||
| 		}, | ||||
| 
 | ||||
| 		"username": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Optional username for the image", | ||||
| 		}, | ||||
| 		"password": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Optional password for the image", | ||||
| 		}, | ||||
| 		"account_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "AccountId to make the image exclusive", | ||||
| 		}, | ||||
| 		"sep_id": { | ||||
| 			Type:        schema.TypeInt, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "storage endpoint provider ID", | ||||
| 		}, | ||||
| 		"pool_name": { | ||||
| 			Type:        schema.TypeString, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "pool for image create", | ||||
| 		}, | ||||
| 		"drivers": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Optional: true, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				StateFunc:    statefuncs.StateFuncToUpper, | ||||
| 				ValidateFunc: validation.StringInSlice([]string{"SVA_KVM_X86", "KVM_X86", "KVM_PPC"}, false), // observe case while validating
 | ||||
| 				Type:         schema.TypeString, | ||||
| 			}, | ||||
| 			Description: "List of types of compute suitable for image. Example: [ \"KVM_X86\" ]", | ||||
| 		}, | ||||
| 		"bootable": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     true, | ||||
| 			Description: "bootable image", | ||||
| 		}, | ||||
| 		"hot_resize": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Computed:    true, | ||||
| 			Description: "Does this machine supports hot resize", | ||||
| 		}, | ||||
| 		"async_mode": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "create an image in async/sync mode", | ||||
| 		}, | ||||
| 		"permanently": { | ||||
| 			Type:        schema.TypeBool, | ||||
| 			Optional:    true, | ||||
| 			Default:     false, | ||||
| 			Description: "whether to completely delete the image", | ||||
| 		}, | ||||
| 
 | ||||
| 		"image_id": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"unc_path": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"ckey": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"acl": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"compute_ci_id": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"cd_presented_to": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"deleted_time": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"desc": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"enabled": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"gid": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"guid": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"history": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Resource{ | ||||
| 				Schema: map[string]*schema.Schema{ | ||||
| 					"guid": { | ||||
| 						Type:     schema.TypeString, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"id": { | ||||
| 						Type:     schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 					"timestamp": { | ||||
| 						Type:     schema.TypeInt, | ||||
| 						Computed: true, | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"last_modified": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"link_to": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"milestones": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"image_name": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"network_interface_naming": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"provider_name": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"purge_attempts": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"present_to": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"res_id": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"rescuecd": { | ||||
| 			Type:     schema.TypeBool, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"shared_with": { | ||||
| 			Type:     schema.TypeList, | ||||
| 			Computed: true, | ||||
| 			Elem: &schema.Schema{ | ||||
| 				Type: schema.TypeInt, | ||||
| 			}, | ||||
| 		}, | ||||
| 		"size": { | ||||
| 			Type:     schema.TypeInt, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"status": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"tech_status": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 		"version": { | ||||
| 			Type:     schema.TypeString, | ||||
| 			Computed: true, | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
					Loading…
					
					
				
		Reference in new issue