2018-02-12 18:10:05 +01:00
package egoscale
import (
2018-09-14 10:06:03 +02:00
"context"
"fmt"
2018-02-12 18:10:05 +01:00
"net/url"
"strconv"
2018-09-14 10:06:03 +02:00
"strings"
2018-02-12 18:10:05 +01:00
)
// SecurityGroup represent a firewalling set of rules
type SecurityGroup struct {
2018-09-14 10:06:03 +02:00
Account string ` json:"account,omitempty" doc:"the account owning the security group" `
Description string ` json:"description,omitempty" doc:"the description of the security group" `
Domain string ` json:"domain,omitempty" doc:"the domain name of the security group" `
DomainID * UUID ` json:"domainid,omitempty" doc:"the domain ID of the security group" `
EgressRule [ ] EgressRule ` json:"egressrule,omitempty" doc:"the list of egress rules associated with the security group" `
ID * UUID ` json:"id,omitempty" doc:"the ID of the security group" `
IngressRule [ ] IngressRule ` json:"ingressrule,omitempty" doc:"the list of ingress rules associated with the security group" `
Name string ` json:"name,omitempty" doc:"the name of the security group" `
Tags [ ] ResourceTag ` json:"tags,omitempty" doc:"the list of resource tags associated with the rule" `
2018-02-12 18:10:05 +01:00
}
// ResourceType returns the type of the resource
2018-09-14 10:06:03 +02:00
func ( SecurityGroup ) ResourceType ( ) string {
2018-02-12 18:10:05 +01:00
return "SecurityGroup"
}
2018-09-14 10:06:03 +02:00
// UserSecurityGroup converts a SecurityGroup to a UserSecurityGroup
func ( sg SecurityGroup ) UserSecurityGroup ( ) UserSecurityGroup {
return UserSecurityGroup {
Account : sg . Account ,
Group : sg . Name ,
}
}
// ListRequest builds the ListSecurityGroups request
func ( sg SecurityGroup ) ListRequest ( ) ( ListCommand , error ) {
//TODO add tags
req := & ListSecurityGroups {
Account : sg . Account ,
DomainID : sg . DomainID ,
ID : sg . ID ,
SecurityGroupName : sg . Name ,
}
return req , nil
}
// Delete deletes the given Security Group
func ( sg SecurityGroup ) Delete ( ctx context . Context , client * Client ) error {
if sg . ID == nil && sg . Name == "" {
return fmt . Errorf ( "a SecurityGroup may only be deleted using ID or Name" )
}
req := & DeleteSecurityGroup {
Account : sg . Account ,
DomainID : sg . DomainID ,
}
if sg . ID != nil {
req . ID = sg . ID
} else {
req . Name = sg . Name
}
return client . BooleanRequestWithContext ( ctx , req )
}
// RuleByID returns IngressRule or EgressRule by a rule ID
func ( sg SecurityGroup ) RuleByID ( ruleID UUID ) ( * IngressRule , * EgressRule ) {
for i , in := range sg . IngressRule {
if in . RuleID . Equal ( ruleID ) {
return & sg . IngressRule [ i ] , nil
}
}
for i , out := range sg . EgressRule {
if out . RuleID . Equal ( ruleID ) {
return nil , & sg . EgressRule [ i ]
}
}
return nil , nil
}
2018-02-12 18:10:05 +01:00
// IngressRule represents the ingress rule
type IngressRule struct {
2018-09-14 10:06:03 +02:00
Account string ` json:"account,omitempty" doc:"account owning the security group rule" `
CIDR * CIDR ` json:"cidr,omitempty" doc:"the CIDR notation for the base IP address of the security group rule" `
Description string ` json:"description,omitempty" doc:"description of the security group rule" `
EndPort uint16 ` json:"endport,omitempty" doc:"the ending port of the security group rule " `
IcmpCode uint8 ` json:"icmpcode,omitempty" doc:"the code for the ICMP message response" `
IcmpType uint8 ` json:"icmptype,omitempty" doc:"the type of the ICMP message response" `
Protocol string ` json:"protocol,omitempty" doc:"the protocol of the security group rule" `
RuleID * UUID ` json:"ruleid,omitempty" doc:"the id of the security group rule" `
SecurityGroupID * UUID ` json:"securitygroupid,omitempty" `
SecurityGroupName string ` json:"securitygroupname,omitempty" doc:"security group name" `
StartPort uint16 ` json:"startport,omitempty" doc:"the starting port of the security group rule" `
Tags [ ] ResourceTag ` json:"tags,omitempty" doc:"the list of resource tags associated with the rule" `
2018-02-12 18:10:05 +01:00
UserSecurityGroupList [ ] UserSecurityGroup ` json:"usersecuritygrouplist,omitempty" `
}
// EgressRule represents the ingress rule
type EgressRule IngressRule
// UserSecurityGroup represents the traffic of another security group
type UserSecurityGroup struct {
Group string ` json:"group,omitempty" `
Account string ` json:"account,omitempty" `
}
2018-09-14 10:06:03 +02:00
// String gives the UserSecurityGroup name
func ( usg UserSecurityGroup ) String ( ) string {
return usg . Group
2018-02-12 18:10:05 +01:00
}
// CreateSecurityGroup represents a security group creation
type CreateSecurityGroup struct {
2018-09-14 10:06:03 +02:00
Name string ` json:"name" doc:"name of the security group" `
Account string ` json:"account,omitempty" doc:"an optional account for the security group. Must be used with domainId." `
Description string ` json:"description,omitempty" doc:"the description of the security group" `
DomainID * UUID ` json:"domainid,omitempty" doc:"an optional domainId for the security group. If the account parameter is used, domainId must also be used." `
_ bool ` name:"createSecurityGroup" description:"Creates a security group" `
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( CreateSecurityGroup ) response ( ) interface { } {
return new ( SecurityGroup )
2018-02-12 18:10:05 +01:00
}
// DeleteSecurityGroup represents a security group deletion
type DeleteSecurityGroup struct {
2018-09-14 10:06:03 +02:00
Account string ` json:"account,omitempty" doc:"the account of the security group. Must be specified with domain ID" `
DomainID * UUID ` json:"domainid,omitempty" doc:"the domain ID of account owning the security group" `
ID * UUID ` json:"id,omitempty" doc:"The ID of the security group. Mutually exclusive with name parameter" `
Name string ` json:"name,omitempty" doc:"The ID of the security group. Mutually exclusive with id parameter" `
_ bool ` name:"deleteSecurityGroup" description:"Deletes security group" `
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( DeleteSecurityGroup ) response ( ) interface { } {
return new ( booleanResponse )
2018-02-12 18:10:05 +01:00
}
// AuthorizeSecurityGroupIngress (Async) represents the ingress rule creation
type AuthorizeSecurityGroupIngress struct {
2018-09-14 10:06:03 +02:00
Account string ` json:"account,omitempty" doc:"an optional account for the security group. Must be used with domainId." `
CIDRList [ ] CIDR ` json:"cidrlist,omitempty" doc:"the cidr list associated" `
Description string ` json:"description,omitempty" doc:"the description of the ingress/egress rule" `
DomainID * UUID ` json:"domainid,omitempty" doc:"an optional domainid for the security group. If the account parameter is used, domainid must also be used." `
EndPort uint16 ` json:"endport,omitempty" doc:"end port for this ingress rule" `
IcmpCode uint8 ` json:"icmpcode,omitempty" doc:"error code for this icmp message" `
IcmpType uint8 ` json:"icmptype,omitempty" doc:"type of the icmp message being sent" `
Protocol string ` json:"protocol,omitempty" doc:"TCP is default. UDP, ICMP, ICMPv6, AH, ESP, GRE are the other supported protocols" `
SecurityGroupID * UUID ` json:"securitygroupid,omitempty" doc:"The ID of the security group. Mutually exclusive with securitygroupname parameter" `
SecurityGroupName string ` json:"securitygroupname,omitempty" doc:"The name of the security group. Mutually exclusive with securitygroupid parameter" `
StartPort uint16 ` json:"startport,omitempty" doc:"start port for this ingress rule" `
UserSecurityGroupList [ ] UserSecurityGroup ` json:"usersecuritygrouplist,omitempty" doc:"user to security group mapping" `
_ bool ` name:"authorizeSecurityGroupIngress" description:"Authorizes a particular ingress/egress rule for this security group" `
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( AuthorizeSecurityGroupIngress ) response ( ) interface { } {
return new ( AsyncJobResult )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( AuthorizeSecurityGroupIngress ) asyncResponse ( ) interface { } {
return new ( SecurityGroup )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( req AuthorizeSecurityGroupIngress ) onBeforeSend ( params url . Values ) error {
2018-02-12 18:10:05 +01:00
// ICMP code and type may be zero but can also be omitted...
2018-09-14 10:06:03 +02:00
if strings . HasPrefix ( strings . ToLower ( req . Protocol ) , "icmp" ) {
2018-02-12 18:10:05 +01:00
params . Set ( "icmpcode" , strconv . FormatInt ( int64 ( req . IcmpCode ) , 10 ) )
params . Set ( "icmptype" , strconv . FormatInt ( int64 ( req . IcmpType ) , 10 ) )
}
2018-09-14 10:06:03 +02:00
// StartPort may be zero but can also be omitted...
if req . EndPort != 0 && req . StartPort == 0 {
params . Set ( "startport" , "0" )
}
2018-02-12 18:10:05 +01:00
return nil
}
// AuthorizeSecurityGroupEgress (Async) represents the egress rule creation
type AuthorizeSecurityGroupEgress AuthorizeSecurityGroupIngress
2018-09-14 10:06:03 +02:00
func ( AuthorizeSecurityGroupEgress ) response ( ) interface { } {
return new ( AsyncJobResult )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( AuthorizeSecurityGroupEgress ) asyncResponse ( ) interface { } {
return new ( SecurityGroup )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( req AuthorizeSecurityGroupEgress ) onBeforeSend ( params url . Values ) error {
return ( AuthorizeSecurityGroupIngress ) ( req ) . onBeforeSend ( params )
2018-02-12 18:10:05 +01:00
}
// RevokeSecurityGroupIngress (Async) represents the ingress/egress rule deletion
type RevokeSecurityGroupIngress struct {
2018-09-14 10:06:03 +02:00
ID * UUID ` json:"id" doc:"The ID of the ingress rule" `
_ bool ` name:"revokeSecurityGroupIngress" description:"Deletes a particular ingress rule from this security group" `
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( RevokeSecurityGroupIngress ) response ( ) interface { } {
return new ( AsyncJobResult )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( RevokeSecurityGroupIngress ) asyncResponse ( ) interface { } {
return new ( booleanResponse )
2018-02-12 18:10:05 +01:00
}
// RevokeSecurityGroupEgress (Async) represents the ingress/egress rule deletion
type RevokeSecurityGroupEgress struct {
2018-09-14 10:06:03 +02:00
ID * UUID ` json:"id" doc:"The ID of the egress rule" `
_ bool ` name:"revokeSecurityGroupEgress" description:"Deletes a particular egress rule from this security group" `
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( RevokeSecurityGroupEgress ) response ( ) interface { } {
return new ( AsyncJobResult )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
func ( RevokeSecurityGroupEgress ) asyncResponse ( ) interface { } {
return new ( booleanResponse )
2018-02-12 18:10:05 +01:00
}
// ListSecurityGroups represents a search for security groups
type ListSecurityGroups struct {
2018-09-14 10:06:03 +02:00
Account string ` json:"account,omitempty" doc:"list resources by account. Must be used with the domainId parameter." `
DomainID * UUID ` json:"domainid,omitempty" doc:"list only resources belonging to the domain specified" `
ID * UUID ` json:"id,omitempty" doc:"list the security 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" `
2018-02-12 18:10:05 +01:00
Page int ` json:"page,omitempty" `
PageSize int ` json:"pagesize,omitempty" `
2018-09-14 10:06:03 +02:00
SecurityGroupName string ` json:"securitygroupname,omitempty" doc:"lists security groups by name" `
Tags [ ] ResourceTag ` json:"tags,omitempty" doc:"List resources by tags (key/value pairs)" `
VirtualMachineID * UUID ` json:"virtualmachineid,omitempty" doc:"lists security groups by virtual machine id" `
_ bool ` name:"listSecurityGroups" description:"Lists security groups" `
2018-02-12 18:10:05 +01:00
}
// ListSecurityGroupsResponse represents a list of security groups
type ListSecurityGroupsResponse struct {
Count int ` json:"count" `
SecurityGroup [ ] SecurityGroup ` json:"securitygroup" `
}
2018-09-14 10:06:03 +02:00
func ( ListSecurityGroups ) response ( ) interface { } {
return new ( ListSecurityGroupsResponse )
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
// SetPage sets the current page
func ( lsg * ListSecurityGroups ) SetPage ( page int ) {
lsg . Page = page
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
// SetPageSize sets the page size
func ( lsg * ListSecurityGroups ) SetPageSize ( pageSize int ) {
lsg . PageSize = pageSize
}
2018-02-12 18:10:05 +01:00
2018-09-14 10:06:03 +02:00
func ( ListSecurityGroups ) each ( resp interface { } , callback IterateItemFunc ) {
sgs , ok := resp . ( * ListSecurityGroupsResponse )
if ! ok {
callback ( nil , fmt . Errorf ( "wrong type. ListSecurityGroupsResponse expected, got %T" , resp ) )
return
2018-02-12 18:10:05 +01:00
}
2018-09-14 10:06:03 +02:00
for i := range sgs . SecurityGroup {
if ! callback ( & sgs . SecurityGroup [ i ] , nil ) {
break
2018-02-12 18:10:05 +01:00
}
}
}