package egoscale import ( "context" "fmt" "net/url" ) // AffinityGroup represents an (anti-)affinity group // // Affinity and Anti-Affinity groups provide a way to influence where VMs should run. // See: http://docs.cloudstack.apache.org/projects/cloudstack-administration/en/stable/virtual_machines.html#affinity-groups type AffinityGroup struct { Account string `json:"account,omitempty" doc:"the account owning the affinity group"` Description string `json:"description,omitempty" doc:"the description of the affinity group"` Domain string `json:"domain,omitempty" doc:"the domain name of the affinity group"` DomainID string `json:"domainid,omitempty" doc:"the domain ID of the affinity group"` ID string `json:"id,omitempty" doc:"the ID of the affinity group"` Name string `json:"name,omitempty" doc:"the name of the affinity group"` Type string `json:"type,omitempty" doc:"the type of the affinity group"` VirtualMachineIDs []string `json:"virtualmachineIds,omitempty" doc:"virtual machine Ids associated with this affinity group "` } // ListRequest builds the ListAffinityGroups request func (ag AffinityGroup) ListRequest() (ListCommand, error) { return &ListAffinityGroups{ ID: ag.ID, Name: ag.Name, }, nil } // Delete removes the given Affinity Group func (ag AffinityGroup) Delete(ctx context.Context, client *Client) error { if ag.ID == "" && ag.Name == "" { return fmt.Errorf("an Affinity Group may only be deleted using ID or Name") } req := &DeleteAffinityGroup{ Account: ag.Account, DomainID: ag.DomainID, } if ag.ID != "" { req.ID = ag.ID } else { req.Name = ag.Name } return client.BooleanRequestWithContext(ctx, req) } // AffinityGroupType represent an affinity group type type AffinityGroupType struct { Type string `json:"type,omitempty" doc:"the type of the affinity group"` } // CreateAffinityGroup (Async) represents a new (anti-)affinity group type CreateAffinityGroup struct { Account string `json:"account,omitempty" doc:"an account for the affinity group. Must be used with domainId."` Description string `json:"description,omitempty" doc:"optional description of the affinity group"` DomainID string `json:"domainid,omitempty" doc:"domainId of the account owning the affinity group"` Name string `json:"name" doc:"name of the affinity group"` Type string `json:"type" doc:"Type of the affinity group from the available affinity/anti-affinity group types"` _ bool `name:"createAffinityGroup" description:"Creates an affinity/anti-affinity group"` } func (CreateAffinityGroup) response() interface{} { return new(AsyncJobResult) } func (CreateAffinityGroup) asyncResponse() interface{} { return new(AffinityGroup) } // UpdateVMAffinityGroup (Async) represents a modification of a (anti-)affinity group type UpdateVMAffinityGroup struct { ID string `json:"id" doc:"The ID of the virtual machine"` AffinityGroupIDs []string `json:"affinitygroupids,omitempty" doc:"comma separated list of affinity groups id that are going to be applied to the virtual machine. Should be passed only when vm is created from a zone with Basic Network support. Mutually exclusive with securitygroupnames parameter"` AffinityGroupNames []string `json:"affinitygroupnames,omitempty" doc:"comma separated list of affinity groups names that are going to be applied to the virtual machine. Should be passed only when vm is created from a zone with Basic Network support. Mutually exclusive with securitygroupids parameter"` _ bool `name:"updateVMAffinityGroup" description:"Updates the affinity/anti-affinity group associations of a virtual machine. The VM has to be stopped and restarted for the new properties to take effect."` } func (req UpdateVMAffinityGroup) onBeforeSend(params url.Values) error { // Either AffinityGroupIDs or AffinityGroupNames must be set if len(req.AffinityGroupIDs) == 0 && len(req.AffinityGroupNames) == 0 { params.Set("affinitygroupids", "") } return nil } func (UpdateVMAffinityGroup) response() interface{} { return new(AsyncJobResult) } func (UpdateVMAffinityGroup) asyncResponse() interface{} { return new(VirtualMachine) } // DeleteAffinityGroup (Async) represents an (anti-)affinity group to be deleted type DeleteAffinityGroup struct { Account string `json:"account,omitempty" doc:"the account of the affinity group. Must be specified with domain ID"` DomainID string `json:"domainid,omitempty" doc:"the domain ID of account owning the affinity group"` ID string `json:"id,omitempty" doc:"The ID of the affinity group. Mutually exclusive with name parameter"` Name string `json:"name,omitempty" doc:"The name of the affinity group. Mutually exclusive with ID parameter"` _ bool `name:"deleteAffinityGroup" description:"Deletes affinity group"` } func (DeleteAffinityGroup) response() interface{} { return new(AsyncJobResult) } func (DeleteAffinityGroup) asyncResponse() interface{} { return new(booleanResponse) } // ListAffinityGroups represents an (anti-)affinity groups search type ListAffinityGroups struct { Account string `json:"account,omitempty" doc:"list resources by account. Must be used with the domainId parameter."` DomainID string `json:"domainid,omitempty" doc:"list only resources belonging to the domain specified"` ID string `json:"id,omitempty" doc:"list the affinity group by the ID provided"` IsRecursive *bool `json:"isrecursive,omitempty" doc:"defaults to false, but if true, lists all resources from the parent specified by the domainId till leaves."` Keyword string `json:"keyword,omitempty" doc:"List by keyword"` ListAll *bool `json:"listall,omitempty" doc:"If set to false, list only resources belonging to the command's caller; if set to true - list resources that the caller is authorized to see. Default value is false"` Name string `json:"name,omitempty" doc:"lists affinity groups by name"` Page int `json:"page,omitempty"` PageSize int `json:"pagesize,omitempty"` Type string `json:"type,omitempty" doc:"lists affinity groups by type"` VirtualMachineID string `json:"virtualmachineid,omitempty" doc:"lists affinity groups by virtual machine ID"` _ bool `name:"listAffinityGroups" description:"Lists affinity groups"` } func (ListAffinityGroups) response() interface{} { return new(ListAffinityGroupsResponse) } // SetPage sets the current page func (ls *ListAffinityGroups) SetPage(page int) { ls.Page = page } // SetPageSize sets the page size func (ls *ListAffinityGroups) SetPageSize(pageSize int) { ls.PageSize = pageSize } func (ListAffinityGroups) each(resp interface{}, callback IterateItemFunc) { vms, ok := resp.(*ListAffinityGroupsResponse) if !ok { callback(nil, fmt.Errorf("wrong type. ListAffinityGroupsResponse expected, got %T", resp)) return } for i := range vms.AffinityGroup { if !callback(&vms.AffinityGroup[i], nil) { break } } } // ListAffinityGroupsResponse represents a list of (anti-)affinity groups type ListAffinityGroupsResponse struct { Count int `json:"count"` AffinityGroup []AffinityGroup `json:"affinitygroup"` } // ListAffinityGroupTypes represents an (anti-)affinity groups search type ListAffinityGroupTypes struct { Keyword string `json:"keyword,omitempty" doc:"List by keyword"` Page int `json:"page,omitempty"` PageSize int `json:"pagesize,omitempty"` _ bool `name:"listAffinityGroupTypes" description:"Lists affinity group types available"` } func (ListAffinityGroupTypes) response() interface{} { return new(ListAffinityGroupTypesResponse) } // ListAffinityGroupTypesResponse represents a list of (anti-)affinity group types type ListAffinityGroupTypesResponse struct { Count int `json:"count"` AffinityGroupType []AffinityGroupType `json:"affinitygrouptype"` }