package ecs import ( "context" "fmt" "github.com/aliyun/alibaba-cloud-sdk-go/sdk/errors" "github.com/hashicorp/packer-plugin-sdk/uuid" "github.com/aliyun/alibaba-cloud-sdk-go/sdk/responses" "github.com/aliyun/alibaba-cloud-sdk-go/services/ecs" "github.com/hashicorp/packer-plugin-sdk/multistep" packersdk "github.com/hashicorp/packer-plugin-sdk/packer" ) type stepConfigAlicloudEIP struct { AssociatePublicIpAddress bool RegionId string InternetChargeType string InternetMaxBandwidthOut int allocatedId string SSHPrivateIp bool } var allocateEipAddressRetryErrors = []string{ "LastTokenProcessing", } func (s *stepConfigAlicloudEIP) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction { client := state.Get("client").(*ClientWrapper) ui := state.Get("ui").(packersdk.Ui) instance := state.Get("instance").(*ecs.Instance) if s.SSHPrivateIp { ipaddress := instance.VpcAttributes.PrivateIpAddress.IpAddress if len(ipaddress) == 0 { ui.Say("Failed to get private ip of instance") return multistep.ActionHalt } state.Put("ipaddress", ipaddress[0]) return multistep.ActionContinue } ui.Say("Allocating eip...") allocateEipAddressRequest := s.buildAllocateEipAddressRequest(state) allocateEipAddressResponse, err := client.WaitForExpected(&WaitForExpectArgs{ RequestFunc: func() (responses.AcsResponse, error) { return client.AllocateEipAddress(allocateEipAddressRequest) }, EvalFunc: client.EvalCouldRetryResponse(allocateEipAddressRetryErrors, EvalRetryErrorType), }) if err != nil { return halt(state, err, "Error allocating eip") } ipaddress := allocateEipAddressResponse.(*ecs.AllocateEipAddressResponse).EipAddress ui.Message(fmt.Sprintf("Allocated eip: %s", ipaddress)) allocateId := allocateEipAddressResponse.(*ecs.AllocateEipAddressResponse).AllocationId s.allocatedId = allocateId err = s.waitForEipStatus(client, instance.RegionId, s.allocatedId, EipStatusAvailable) if err != nil { return halt(state, err, "Error wait eip available timeout") } associateEipAddressRequest := ecs.CreateAssociateEipAddressRequest() associateEipAddressRequest.AllocationId = allocateId associateEipAddressRequest.InstanceId = instance.InstanceId if _, err := client.AssociateEipAddress(associateEipAddressRequest); err != nil { e, ok := err.(errors.Error) if !ok || e.ErrorCode() != "TaskConflict" { return halt(state, err, "Error associating eip") } ui.Error(fmt.Sprintf("Error associate eip: %s", err)) } err = s.waitForEipStatus(client, instance.RegionId, s.allocatedId, EipStatusInUse) if err != nil { return halt(state, err, "Error wait eip associated timeout") } state.Put("ipaddress", ipaddress) return multistep.ActionContinue } func (s *stepConfigAlicloudEIP) Cleanup(state multistep.StateBag) { if len(s.allocatedId) == 0 { return } cleanUpMessage(state, "EIP") client := state.Get("client").(*ClientWrapper) instance := state.Get("instance").(*ecs.Instance) ui := state.Get("ui").(packersdk.Ui) unassociateEipAddressRequest := ecs.CreateUnassociateEipAddressRequest() unassociateEipAddressRequest.AllocationId = s.allocatedId unassociateEipAddressRequest.InstanceId = instance.InstanceId if _, err := client.UnassociateEipAddress(unassociateEipAddressRequest); err != nil { ui.Say(fmt.Sprintf("Failed to unassociate eip: %s", err)) } if err := s.waitForEipStatus(client, instance.RegionId, s.allocatedId, EipStatusAvailable); err != nil { ui.Say(fmt.Sprintf("Timeout while unassociating eip: %s", err)) } releaseEipAddressRequest := ecs.CreateReleaseEipAddressRequest() releaseEipAddressRequest.AllocationId = s.allocatedId if _, err := client.ReleaseEipAddress(releaseEipAddressRequest); err != nil { ui.Say(fmt.Sprintf("Failed to release eip: %s", err)) } } func (s *stepConfigAlicloudEIP) waitForEipStatus(client *ClientWrapper, regionId string, allocationId string, expectedStatus string) error { describeEipAddressesRequest := ecs.CreateDescribeEipAddressesRequest() describeEipAddressesRequest.RegionId = regionId describeEipAddressesRequest.AllocationId = s.allocatedId _, err := client.WaitForExpected(&WaitForExpectArgs{ RequestFunc: func() (responses.AcsResponse, error) { response, err := client.DescribeEipAddresses(describeEipAddressesRequest) if err == nil && len(response.EipAddresses.EipAddress) == 0 { err = fmt.Errorf("eip allocated is not find") } return response, err }, EvalFunc: func(response responses.AcsResponse, err error) WaitForExpectEvalResult { if err != nil { return WaitForExpectToRetry } eipAddressesResponse := response.(*ecs.DescribeEipAddressesResponse) eipAddresses := eipAddressesResponse.EipAddresses.EipAddress for _, eipAddress := range eipAddresses { if eipAddress.Status == expectedStatus { return WaitForExpectSuccess } } return WaitForExpectToRetry }, RetryTimes: shortRetryTimes, }) return err } func (s *stepConfigAlicloudEIP) buildAllocateEipAddressRequest(state multistep.StateBag) *ecs.AllocateEipAddressRequest { instance := state.Get("instance").(*ecs.Instance) request := ecs.CreateAllocateEipAddressRequest() request.ClientToken = uuid.TimeOrderedUUID() request.RegionId = instance.RegionId request.InternetChargeType = s.InternetChargeType request.Bandwidth = string(convertNumber(s.InternetMaxBandwidthOut)) return request }