Skip to content

Commit

Permalink
Add always run scripts (#65)
Browse files Browse the repository at this point in the history
* Add always run scripts

- Added the ability to set a post_processor_script to always run on update
- Fixed some issues where in some cases the update would but the vm in the wrong state
- Fixed an issue where some errors would bring a nil pointer
- Added an option to wait for network before querying to the datasource vm
- Added a retry mechanism to attempt to get the internal ip on create/update

* fix the documentation
  • Loading branch information
cjlapao authored Nov 21, 2024
1 parent 5929a13 commit 05594a1
Show file tree
Hide file tree
Showing 18 changed files with 456 additions and 116 deletions.
1 change: 1 addition & 0 deletions docs/data-sources/vm.md
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ data "parallels-desktop_vm" "example" {
- `filter` (Block, Optional) Filter block, this is used to filter data sources (see [below for nested schema](#nestedblock--filter))
- `host` (String) Parallels Desktop DevOps Host
- `orchestrator` (String) Parallels Desktop DevOps Orchestrator
- `wait_for_network_up` (Boolean) Wait for network up

### Read-Only

Expand Down
2 changes: 2 additions & 0 deletions docs/resources/clone_vm.md
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--on_destroy_script--retry))
Expand Down Expand Up @@ -223,6 +224,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--post_processor_script--retry))
Expand Down
2 changes: 2 additions & 0 deletions docs/resources/remote_vm.md
Original file line number Diff line number Diff line change
Expand Up @@ -197,6 +197,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--on_destroy_script--retry))
Expand Down Expand Up @@ -231,6 +232,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--post_processor_script--retry))
Expand Down
2 changes: 2 additions & 0 deletions docs/resources/vagrant_box.md
Original file line number Diff line number Diff line change
Expand Up @@ -190,6 +190,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--on_destroy_script--retry))
Expand Down Expand Up @@ -224,6 +225,7 @@ Optional:

Optional:

- `always_run_on_update` (Boolean) Always run on update
- `environment_variables` (Map of String) Environment variables that can be used in the DevOps service, please see documentation to see which variables are available
- `inline` (List of String) Inline script
- `retry` (Block, Optional) Retry settings (see [below for nested schema](#nestedblock--post_processor_script--retry))
Expand Down
112 changes: 83 additions & 29 deletions internal/clone_vm/resource.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,9 +134,9 @@ func (r *CloneVmResource) Create(ctx context.Context, req resource.CreateRequest
}

// Checking if the name is already in use
existingVms, diag := apiclient.GetVms(ctx, hostConfig, "name", data.Name.ValueString())
if diag.HasError() {
resp.Diagnostics.Append(diag...)
existingVms, existingVmDiag := apiclient.GetVms(ctx, hostConfig, "name", data.Name.ValueString())
if existingVmDiag.HasError() {
resp.Diagnostics.Append(existingVmDiag...)
return
}

Expand All @@ -146,9 +146,9 @@ func (r *CloneVmResource) Create(ctx context.Context, req resource.CreateRequest
}

// Checking if we can find the base vm to clone
vm, diag := apiclient.GetVm(ctx, hostConfig, data.BaseVmId.ValueString())
if diag.HasError() {
diag.Append(diag...)
vm, getVmDiag := apiclient.GetVm(ctx, hostConfig, data.BaseVmId.ValueString())
if getVmDiag.HasError() {
resp.Diagnostics.Append(getVmDiag...)
return
}

Expand Down Expand Up @@ -177,9 +177,9 @@ func (r *CloneVmResource) Create(ctx context.Context, req resource.CreateRequest
}

// Checking if we can find the base vm to clone
createdVms, diag := apiclient.GetVms(ctx, hostConfig, "name", data.Name.ValueString())
if diag.HasError() {
diag.Append(diag...)
createdVms, createVmDiag := apiclient.GetVms(ctx, hostConfig, "name", data.Name.ValueString())
if createVmDiag.HasError() {
resp.Diagnostics.Append(createVmDiag...)
return
}
if len(createdVms) != 1 {
Expand All @@ -194,9 +194,9 @@ func (r *CloneVmResource) Create(ctx context.Context, req resource.CreateRequest
// stopping the machine as it might need some operations where the machine needs to be stopped
// add anything here in sequence that needs to be done before the machine is started
// so we do not loose time waiting for the machine to stop
stoppedVm, diag := common.EnsureMachineStopped(ctx, hostConfig, &clonedVm)
if diag.HasError() {
resp.Diagnostics.Append(diag...)
stoppedVm, stoppedVmDiag := common.EnsureMachineStopped(ctx, hostConfig, &clonedVm)
if stoppedVmDiag.HasError() {
resp.Diagnostics.Append(stoppedVmDiag...)
}

// Applying the Specs block
Expand Down Expand Up @@ -331,13 +331,24 @@ func (r *CloneVmResource) Create(ctx context.Context, req resource.CreateRequest

externalIp := ""
internalIp := ""
refreshVm, refreshDiag := apiclient.GetVm(ctx, hostConfig, vm.ID)
if refreshDiag.HasError() {
resp.Diagnostics.Append(refreshDiag...)
return
} else {
externalIp = refreshVm.HostExternalIpAddress
internalIp = refreshVm.InternalIpAddress
retryAttempts := 10
var refreshVm *apimodels.VirtualMachine
var refreshDiag diag.Diagnostics
for {
refreshVm, refreshDiag = apiclient.GetVm(ctx, hostConfig, vm.ID)
if !refreshDiag.HasError() {
externalIp = refreshVm.HostExternalIpAddress
internalIp = refreshVm.InternalIpAddress
}
if internalIp != "" {
time.Sleep(5 * time.Second)
break
}
if retryAttempts == 0 {
internalIp = "-"
break
}
retryAttempts--
}

data.ExternalIp = types.StringValue(externalIp)
Expand Down Expand Up @@ -507,15 +518,16 @@ func (r *CloneVmResource) Update(ctx context.Context, req resource.UpdateRequest
DisableTlsValidation: r.provider.DisableTlsValidation.ValueBool(),
}

vm, diag := apiclient.GetVm(ctx, hostConfig, currentData.ID.ValueString())
if diag.HasError() {
resp.Diagnostics.Append(diag...)
vm, getVmDiag := apiclient.GetVm(ctx, hostConfig, currentData.ID.ValueString())
if getVmDiag.HasError() {
resp.Diagnostics.Append(getVmDiag...)
return
}
if vm == nil {
resp.State.RemoveResource(ctx)
return
}
currentVmState := vm.State

nameChanges := apimodels.NewVmConfigRequest(vm.User)
currentState := vm.State
Expand Down Expand Up @@ -666,13 +678,24 @@ func (r *CloneVmResource) Update(ctx context.Context, req resource.UpdateRequest

externalIp := ""
internalIp := ""
refreshVm, refreshDiag := apiclient.GetVm(ctx, hostConfig, vm.ID)
if refreshDiag.HasError() {
resp.Diagnostics.Append(refreshDiag...)
return
} else {
externalIp = refreshVm.HostExternalIpAddress
internalIp = refreshVm.InternalIpAddress
retryAttempts := 10
var refreshVm *apimodels.VirtualMachine
var refreshDiag diag.Diagnostics
for {
refreshVm, refreshDiag = apiclient.GetVm(ctx, hostConfig, vm.ID)
if !refreshDiag.HasError() {
externalIp = refreshVm.HostExternalIpAddress
internalIp = refreshVm.InternalIpAddress
}
if internalIp != "" {
time.Sleep(5 * time.Second)
break
}
if retryAttempts == 0 {
internalIp = "-"
break
}
retryAttempts--
}

data.ID = types.StringValue(refreshVm.ID)
Expand Down Expand Up @@ -705,6 +728,37 @@ func (r *CloneVmResource) Update(ctx context.Context, req resource.UpdateRequest
}
}

if currentVmState != refreshVm.State {
// If the vm state is desync we nee to set it right
switch currentVmState {
case "running":
if refreshVm.State == "stopped" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpStart)
}
if refreshVm.State == "paused" || refreshVm.State == "suspended" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpResume)
}
case "stopped":
if refreshVm.State == "running" || refreshVm.State == "paused" || refreshVm.State == "suspended" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpStop)
}
case "paused":
if refreshVm.State == "running" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpResume)
}
if refreshVm.State == "stopped" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpStart)
}
case "suspended":
if refreshVm.State == "running" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpResume)
}
if refreshVm.State == "stopped" {
apiclient.SetMachineState(ctx, hostConfig, data.ID.ValueString(), apiclient.MachineStateOpStart)
}
}
}

tflog.Info(ctx, "Updated vm with id "+data.ID.ValueString()+" and name "+data.Name.ValueString())

resp.Diagnostics.Append(resp.State.Set(ctx, &data)...)
Expand Down
12 changes: 8 additions & 4 deletions internal/common/ensure_machine_running.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,20 +21,24 @@ func EnsureMachineRunning(ctx context.Context, hostConfig apiclient.HostConfig,
return vm, diagnostics
}

maxRetries := 10
maxRetries := 20
retryCount := 0
for {
diagnostics = diag.Diagnostics{}
retryCount += 1
if returnVm.State != "running" {
tflog.Info(ctx, "Machine "+returnVm.Name+" is not running, starting it"+fmt.Sprintf("[%v/%v]", retryCount, maxRetries))
result, stateDiag := apiclient.SetMachineState(ctx, hostConfig, returnVm.ID, apiclient.MachineStateOpStart)
op := apiclient.MachineStateOpStart
if returnVm.State == "suspended" || returnVm.State == "paused" {
op = apiclient.MachineStateOpResume
}
result, stateDiag := apiclient.SetMachineState(ctx, hostConfig, returnVm.ID, op)
if stateDiag.HasError() {
diagnostics.Append(stateDiag...)
}

if !result {
diagnostics.AddError("error starting vm", "error starting vm")
diagnostics.AddError("Error starting vm", "Could not set the state of the machine to running")
}

tflog.Info(ctx, "Checking if "+returnVm.Name+" is running")
Expand Down Expand Up @@ -62,7 +66,7 @@ func EnsureMachineRunning(ctx context.Context, hostConfig apiclient.HostConfig,

// We have run out of retries, add an error and break out of the loop
if retryCount >= maxRetries {
diagnostics.AddError("error starting vm", "error starting vm")
diagnostics.AddError("Error starting vm", "Could not verify the state of the machine after starting, retry count exceeded")
break
}

Expand Down
6 changes: 6 additions & 0 deletions internal/common/post_processor_script.go
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,13 @@ func RunPostProcessorScript(ctx context.Context, hostConfig apiclient.HostConfig

func PostProcessorHasChanges(ctx context.Context, planPostProcessorScript, statePostProcessorScript []*postprocessorscript.PostProcessorScript) bool {
for i, script := range planPostProcessorScript {
if script.AlwaysRunOnUpdate.ValueBool() {
return true
}
innerElements := script.Inline.Elements()
if len(innerElements) > 0 && len(statePostProcessorScript) == 0 {
return true
}
if len(innerElements) != len(statePostProcessorScript[i].Inline.Elements()) {
return true
}
Expand Down
4 changes: 2 additions & 2 deletions internal/common/specs.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ func SpecsBlockOnUpdate(ctx context.Context, hostConfig apiclient.HostConfig, vm
}

if hardwareInfo.TotalAvailable.LogicalCpuCount-int64(updateValueInt) <= 0 {
diagnostics.AddError("not enough cpus", "not enough cpus")
diagnostics.AddError("Not enough cpus", "You requested more cpus than available, the machine will need "+updateValue+" cpus and we have "+fmt.Sprintf("%v", hardwareInfo.TotalAvailable.LogicalCpuCount))
return diagnostics
}

Expand All @@ -88,7 +88,7 @@ func SpecsBlockOnUpdate(ctx context.Context, hostConfig apiclient.HostConfig, vm
}

if hardwareInfo.TotalAvailable.MemorySize-float64(updateValueInt) <= 0 {
diagnostics.AddError("not enough memory", "not enough memory")
diagnostics.AddError("Not enough memory", "You requested more memory than available, the machine will need "+updateValue+" memory and we have "+fmt.Sprintf("%v", hardwareInfo.TotalAvailable.MemorySize))
return diagnostics
}

Expand Down
Loading

0 comments on commit 05594a1

Please sign in to comment.