diff --git a/CHANGELOG.md b/CHANGELOG.md index b693d326..53aeaf75 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,22 @@ # EDGEGRID GOLANG RELEASE NOTES +## 2.9.0 (Jan. 24, 2022) + +#### FEATURES/ENHANCEMENTS: + +* CLOUDLETS + * Support for VP cloudlet type (Visitor Prioritization) + * Support for CD cloudlet type (Continuous Deployment / Phased Release) + * Support for FR cloudlet type (Forward Rewrite) + * Support for AP cloudlet type (API Prioritization) + +* APPSEC + * Add support for Evasive Path Match feature + * Deprecate individual policy protection interface methods + +* NETWORK LISTS + * Include ContractID and GroupID in GetNetworkListResponse + ## 2.8.1 (Nov. 30, 2021) #### FEATURES/ENHANCEMENTS: @@ -79,7 +96,7 @@ ## 2.4.0 (Mar 29, 2021) PAPI - Secure by default -*PAPI +* PAPI * Support to provision default certs as part of hostnames request * New cert status object in hostnames response if it exists diff --git a/go.mod b/go.mod index fd85d5a4..2a526f20 100644 --- a/go.mod +++ b/go.mod @@ -1,20 +1,28 @@ module github.com/akamai/AkamaiOPEN-edgegrid-golang/v2 -go 1.14 +go 1.17 require ( github.com/apex/log v1.9.0 - github.com/go-ozzo/ozzo-validation v3.6.0+incompatible github.com/go-ozzo/ozzo-validation/v4 v4.3.0 github.com/google/uuid v1.1.1 - github.com/kr/text v0.2.0 // indirect github.com/mitchellh/go-homedir v1.1.0 - github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e // indirect - github.com/pkg/errors v0.9.1 // indirect - github.com/smartystreets/goconvey v1.6.4 // indirect github.com/spf13/cast v1.3.1 github.com/stretchr/testify v1.6.1 github.com/tj/assert v0.0.3 - gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f // indirect gopkg.in/ini.v1 v1.51.1 ) + +require ( + github.com/asaskevich/govalidator v0.0.0-20200108200545-475eaeb16496 // indirect + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/kr/text v0.2.0 // indirect + github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/smartystreets/goconvey v1.6.4 // indirect + gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f // indirect + gopkg.in/yaml.v3 v3.0.0-20200605160147-a5ece683394c // indirect +) + +retract v2.8.0 \ No newline at end of file diff --git a/go.sum b/go.sum index 3d715acc..65046146 100644 --- a/go.sum +++ b/go.sum @@ -14,10 +14,6 @@ github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= -github.com/go-ozzo/ozzo-validation v3.6.0+incompatible h1:msy24VGS42fKO9K1vLz82/GeYW1cILu7Nuuj1N3BBkE= -github.com/go-ozzo/ozzo-validation v3.6.0+incompatible/go.mod h1:gsEKFIVnabGBt6mXmxK0MoFy+cZoTJY6mu5Ll3LVLBU= -github.com/go-ozzo/ozzo-validation/v4 v4.2.2 h1:5uhbQAuRK6taB9orHJXA5GtOCuQbsHktskg8aWciC68= -github.com/go-ozzo/ozzo-validation/v4 v4.2.2/go.mod h1:2NKgrcHl3z6cJs+3Oo940FPRiTzuqKbvfrL2RxCj6Ew= github.com/go-ozzo/ozzo-validation/v4 v4.3.0 h1:byhDUpfEwjsVQb1vBunvIjh2BHQ9ead57VkAEY4V+Es= github.com/go-ozzo/ozzo-validation/v4 v4.3.0/go.mod h1:2NKgrcHl3z6cJs+3Oo940FPRiTzuqKbvfrL2RxCj6Ew= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= @@ -32,10 +28,8 @@ github.com/jpillora/backoff v0.0.0-20180909062703-3050d21c67d7/go.mod h1:2iMrUgb github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7C0MuV77Wo= github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= -github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs= github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= -github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= @@ -50,7 +44,6 @@ github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWb github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/gomega v1.5.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= -github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= @@ -58,7 +51,6 @@ github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZb github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/rogpeppe/fastuuid v1.1.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= -github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d h1:zE9ykElWQ6/NYmHa3jpm/yHnI4xSofP+UP6SpjHcSeM= github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= github.com/smartystreets/assertions v1.0.0 h1:UVQPSSmc3qtTi+zPPkCXvZX9VvW/xT/NsRvKfwY81a8= github.com/smartystreets/assertions v1.0.0/go.mod h1:kHHU4qYBaI3q23Pp3VPrmWhuIUrLW/7eUrw0BU5VaoM= @@ -70,9 +62,7 @@ github.com/spf13/cast v1.3.1 h1:nFm6S0SMdyzrzcmThSipiEubIDy8WEXKNZ0UOgiRpng= github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= -github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= -github.com/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= @@ -98,9 +88,7 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= -gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= @@ -109,7 +97,6 @@ gopkg.in/ini.v1 v1.51.1 h1:GyboHr4UqMiLUybYjd22ZjQIKEJEpgtLXtuGbR21Oho= gopkg.in/ini.v1 v1.51.1/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20200605160147-a5ece683394c h1:grhR+C34yXImVGp7EzNk+DTIk+323eIUWOmEevy6bDo= diff --git a/pkg/appsec/advanced_settings_evasive_path_match.go b/pkg/appsec/advanced_settings_evasive_path_match.go new file mode 100644 index 00000000..23935bb4 --- /dev/null +++ b/pkg/appsec/advanced_settings_evasive_path_match.go @@ -0,0 +1,190 @@ +package appsec + +import ( + "context" + "fmt" + "net/http" + + validation "github.com/go-ozzo/ozzo-validation/v4" +) + +type ( + // The AdvancedSettingsEvasivePathMatch supports retrieving or modifying the Evasive Path Match setting. + // + // https://developer.akamai.com/api/cloud_security/application_security/v1.html + AdvancedSettingsEvasivePathMatch interface { + // GetAdvancedSettingsEvasivePathMatch retrieves the Evasive Path Match setting + GetAdvancedSettingsEvasivePathMatch(ctx context.Context, params GetAdvancedSettingsEvasivePathMatchRequest) (*GetAdvancedSettingsEvasivePathMatchResponse, error) + // UpdateAdvancedSettingsEvasivePathMatch modifies the Evasive Path Match setting + UpdateAdvancedSettingsEvasivePathMatch(ctx context.Context, params UpdateAdvancedSettingsEvasivePathMatchRequest) (*UpdateAdvancedSettingsEvasivePathMatchResponse, error) + // RemoveAdvancedSettingsEvasivePathMatch removes the Evasive Path Match setting + RemoveAdvancedSettingsEvasivePathMatch(ctx context.Context, params RemoveAdvancedSettingsEvasivePathMatchRequest) (*RemoveAdvancedSettingsEvasivePathMatchResponse, error) + } + + // GetAdvancedSettingsEvasivePathMatchRequest is used to retrieve the EvasivePathMatch setting + GetAdvancedSettingsEvasivePathMatchRequest struct { + ConfigID int `json:"-"` + Version int `json:"-"` + PolicyID string `json:"-"` + } + + // GetAdvancedSettingsEvasivePathMatchResponse returns the EvasivePathMatch setting + GetAdvancedSettingsEvasivePathMatchResponse struct { + EnablePathMatch bool `json:"enablePathMatch"` + } + + // UpdateAdvancedSettingsEvasivePathMatchRequest is used to update the EvasivePathMatch setting + UpdateAdvancedSettingsEvasivePathMatchRequest struct { + ConfigID int `json:"-"` + Version int `json:"-"` + PolicyID string `json:"-"` + EnablePathMatch bool `json:"enablePathMatch"` + } + + // UpdateAdvancedSettingsEvasivePathMatchResponse returns the result of updating the EvasivePathMatch setting + UpdateAdvancedSettingsEvasivePathMatchResponse struct { + EnablePathMatch bool `json:"enablePathMatch"` + } + + // RemoveAdvancedSettingsEvasivePathMatchRequest is used to clear the EvasivePathMatch setting + RemoveAdvancedSettingsEvasivePathMatchRequest struct { + ConfigID int `json:"-"` + Version int `json:"-"` + PolicyID string `json:"-"` + EnablePathMatch bool `json:"enablePathMatch"` + } + + // RemoveAdvancedSettingsEvasivePathMatchResponse returns the result of clearing the EvasivePathMatch setting + RemoveAdvancedSettingsEvasivePathMatchResponse struct { + ConfigID int `json:"-"` + Version int `json:"-"` + PolicyID string `json:"-"` + EnablePathMatch bool `json:"enablePathMatch"` + } +) + +// Validate validates GetAdvancedSettingssEvasivePathMatchRequest +func (v GetAdvancedSettingsEvasivePathMatchRequest) Validate() error { + return validation.Errors{ + "ConfigID": validation.Validate(v.ConfigID, validation.Required), + "Version": validation.Validate(v.Version, validation.Required), + }.Filter() +} + +// Validate validates UpdateAdvancedSettingsEvasivePathMatchRequest +func (v UpdateAdvancedSettingsEvasivePathMatchRequest) Validate() error { + return validation.Errors{ + "ConfigID": validation.Validate(v.ConfigID, validation.Required), + "Version": validation.Validate(v.Version, validation.Required), + }.Filter() +} + +// Validate validates UpdateAdvancedSettingsEvasivePathMatchRequest +func (v RemoveAdvancedSettingsEvasivePathMatchRequest) Validate() error { + return validation.Errors{ + "ConfigID": validation.Validate(v.ConfigID, validation.Required), + "Version": validation.Validate(v.Version, validation.Required), + }.Filter() +} + +func (p *appsec) GetAdvancedSettingsEvasivePathMatch(ctx context.Context, params GetAdvancedSettingsEvasivePathMatchRequest) (*GetAdvancedSettingsEvasivePathMatchResponse, error) { + if err := params.Validate(); err != nil { + return nil, fmt.Errorf("%w: %s", ErrStructValidation, err.Error()) + } + + logger := p.Log(ctx) + logger.Debug("GetAdvancedSettingsLoggings") + + var rval GetAdvancedSettingsEvasivePathMatchResponse + var uri string + + if params.PolicyID != "" { + uri = fmt.Sprintf( + "/appsec/v1/configs/%d/versions/%d/security-policies/%s/advanced-settings/evasive-path-match", + params.ConfigID, + params.Version, + params.PolicyID) + } else { + uri = fmt.Sprintf( + "/appsec/v1/configs/%d/versions/%d/advanced-settings/evasive-path-match", + params.ConfigID, + params.Version) + } + + req, err := http.NewRequestWithContext(ctx, http.MethodGet, uri, nil) + if err != nil { + return nil, fmt.Errorf("failed to create getadvancedsettingsloggings request: %w", err) + } + + resp, err := p.Exec(req, &rval) + if err != nil { + return nil, fmt.Errorf("getadvancedsettingsloggings request failed: %w", err) + } + + if resp.StatusCode != http.StatusOK { + return nil, p.Error(resp) + } + + return &rval, nil +} + +func (p *appsec) UpdateAdvancedSettingsEvasivePathMatch(ctx context.Context, params UpdateAdvancedSettingsEvasivePathMatchRequest) (*UpdateAdvancedSettingsEvasivePathMatchResponse, error) { + if err := params.Validate(); err != nil { + return nil, fmt.Errorf("%w: %s", ErrStructValidation, err.Error()) + } + + logger := p.Log(ctx) + logger.Debug("UpdateAdvancedSettingsLogging") + + var putURL string + if params.PolicyID != "" { + putURL = fmt.Sprintf( + "/appsec/v1/configs/%d/versions/%d/security-policies/%s/advanced-settings/evasive-path-match", + params.ConfigID, + params.Version, + params.PolicyID) + } else { + putURL = fmt.Sprintf( + "/appsec/v1/configs/%d/versions/%d/advanced-settings/evasive-path-match", + params.ConfigID, + params.Version) + } + + req, err := http.NewRequestWithContext(ctx, http.MethodPut, putURL, nil) + if err != nil { + return nil, fmt.Errorf("failed to create create AdvancedSettingsLoggingrequest: %w", err) + } + + req.Header.Set("Content-Type", "application/json") + var rval UpdateAdvancedSettingsEvasivePathMatchResponse + resp, err := p.Exec(req, &rval, params) + if err != nil { + return nil, fmt.Errorf("create AdvancedSettingsLogging request failed: %w", err) + } + + if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusCreated { + return nil, p.Error(resp) + } + + return &rval, nil +} + +func (p *appsec) RemoveAdvancedSettingsEvasivePathMatch(ctx context.Context, params RemoveAdvancedSettingsEvasivePathMatchRequest) (*RemoveAdvancedSettingsEvasivePathMatchResponse, error) { + request := UpdateAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: params.ConfigID, + Version: params.Version, + PolicyID: params.PolicyID, + EnablePathMatch: false, + } + _, err := p.UpdateAdvancedSettingsEvasivePathMatch(ctx, request) + if err != nil { + return nil, fmt.Errorf("UpdateAdvancedSettingsEvasivePathMatch request failed: %w", err) + } + response := RemoveAdvancedSettingsEvasivePathMatchResponse{ + ConfigID: params.ConfigID, + Version: params.Version, + PolicyID: params.PolicyID, + EnablePathMatch: false, + } + return &response, nil +} diff --git a/pkg/appsec/advanced_settings_evasive_path_match_test.go b/pkg/appsec/advanced_settings_evasive_path_match_test.go new file mode 100644 index 00000000..a051059e --- /dev/null +++ b/pkg/appsec/advanced_settings_evasive_path_match_test.go @@ -0,0 +1,243 @@ +package appsec + +import ( + "context" + "encoding/json" + "errors" + "net/http" + "net/http/httptest" + "testing" + + "github.com/akamai/AkamaiOPEN-edgegrid-golang/v2/pkg/session" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" +) + +func TestApsec_ListAdvancedSettingsEvasivePathMatch(t *testing.T) { + + result := GetAdvancedSettingsEvasivePathMatchResponse{} + + respData := compactJSON(loadFixtureBytes("testdata/TestAdvancedSettingsEvasivePathMatch/AdvancedSettingsEvasivePathMatch.json")) + json.Unmarshal([]byte(respData), &result) + + tests := map[string]struct { + params GetAdvancedSettingsEvasivePathMatchRequest + responseStatus int + responseBody string + expectedPath string + expectedResponse *GetAdvancedSettingsEvasivePathMatchResponse + withError error + headers http.Header + }{ + "200 OK": { + params: GetAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + headers: http.Header{ + "Content-Type": []string{"application/json"}, + }, + responseStatus: http.StatusOK, + responseBody: string(respData), + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + expectedResponse: &result, + }, + "500 internal server error": { + params: GetAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + headers: http.Header{}, + responseStatus: http.StatusInternalServerError, + responseBody: ` +{ + "type": "internal_error", + "title": "Internal Server Error", + "detail": "Error fetching AdvancedSettingsEvasivePathMatch", + "status": 500 +}`, + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + withError: &Error{ + Type: "internal_error", + Title: "Internal Server Error", + Detail: "Error fetching AdvancedSettingsEvasivePathMatch", + StatusCode: http.StatusInternalServerError, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + mockServer := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + assert.Equal(t, test.expectedPath, r.URL.String()) + assert.Equal(t, http.MethodGet, r.Method) + w.WriteHeader(test.responseStatus) + _, err := w.Write([]byte(test.responseBody)) + assert.NoError(t, err) + })) + client := mockAPIClient(t, mockServer) + result, err := client.GetAdvancedSettingsEvasivePathMatch( + session.ContextWithOptions( + context.Background(), + session.WithContextHeaders(test.headers), + ), + test.params) + if test.withError != nil { + assert.True(t, errors.Is(err, test.withError), "want: %s; got: %s", test.withError, err) + return + } + require.NoError(t, err) + assert.Equal(t, test.expectedResponse, result) + }) + } +} + +// Test AdvancedSettingsEvasivePathMatch +func TestAppSec_GetAdvancedSettingsEvasivePathmatch(t *testing.T) { + + result := GetAdvancedSettingsEvasivePathMatchResponse{} + + respData := compactJSON(loadFixtureBytes("testdata/TestAdvancedSettingsEvasivePathMatch/AdvancedSettingsEvasivePathMatch.json")) + json.Unmarshal([]byte(respData), &result) + + tests := map[string]struct { + params GetAdvancedSettingsEvasivePathMatchRequest + responseStatus int + responseBody string + expectedPath string + expectedResponse *GetAdvancedSettingsEvasivePathMatchResponse + withError error + }{ + "200 OK": { + params: GetAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + responseStatus: http.StatusOK, + responseBody: respData, + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + expectedResponse: &result, + }, + "500 internal server error": { + params: GetAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + responseStatus: http.StatusInternalServerError, + responseBody: (` +{ + "type": "internal_error", + "title": "Internal Server Error", + "detail": "Error fetching AdvancedSettingsEvasivePathMatch" +}`), + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + withError: &Error{ + Type: "internal_error", + Title: "Internal Server Error", + Detail: "Error fetching AdvancedSettingsEvasivePathMatch", + StatusCode: http.StatusInternalServerError, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + mockServer := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + assert.Equal(t, test.expectedPath, r.URL.String()) + assert.Equal(t, http.MethodGet, r.Method) + w.WriteHeader(test.responseStatus) + _, err := w.Write([]byte(test.responseBody)) + assert.NoError(t, err) + })) + client := mockAPIClient(t, mockServer) + result, err := client.GetAdvancedSettingsEvasivePathMatch(context.Background(), test.params) + if test.withError != nil { + assert.True(t, errors.Is(err, test.withError), "want: %s; got: %s", test.withError, err) + return + } + require.NoError(t, err) + assert.Equal(t, test.expectedResponse, result) + }) + } +} + +// Test Update AdvancedSettingsEvasivePathMatch. +func TestAppSec_UpdateAdvancedSettingsEvasivePathMatch(t *testing.T) { + result := UpdateAdvancedSettingsEvasivePathMatchResponse{} + + respData := compactJSON(loadFixtureBytes("testdata/TestAdvancedSettingsEvasivePathMatch/AdvancedSettingsEvasivePathMatch.json")) + json.Unmarshal([]byte(respData), &result) + + req := UpdateAdvancedSettingsEvasivePathMatchRequest{} + + reqData := compactJSON(loadFixtureBytes("testdata/TestAdvancedSettingsEvasivePathMatch/AdvancedSettingsEvasivePathMatch.json")) + json.Unmarshal([]byte(reqData), &req) + + tests := map[string]struct { + params UpdateAdvancedSettingsEvasivePathMatchRequest + responseStatus int + responseBody string + expectedPath string + expectedResponse *UpdateAdvancedSettingsEvasivePathMatchResponse + withError error + headers http.Header + }{ + "200 Success": { + params: UpdateAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + headers: http.Header{ + "Content-Type": []string{"application/json;charset=UTF-8"}, + }, + responseStatus: http.StatusCreated, + responseBody: respData, + expectedResponse: &result, + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + }, + "500 internal server error": { + params: UpdateAdvancedSettingsEvasivePathMatchRequest{ + ConfigID: 43253, + Version: 15, + }, + responseStatus: http.StatusInternalServerError, + responseBody: (` +{ + "type": "internal_error", + "title": "Internal Server Error", + "detail": "Error creating AdvancedSettingsEvasivePathMatch" +}`), + expectedPath: "/appsec/v1/configs/43253/versions/15/advanced-settings/evasive-path-match", + withError: &Error{ + Type: "internal_error", + Title: "Internal Server Error", + Detail: "Error creating AdvancedSettingsEvasivePathMatch", + StatusCode: http.StatusInternalServerError, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + mockServer := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + assert.Equal(t, http.MethodPut, r.Method) + w.WriteHeader(test.responseStatus) + if len(test.responseBody) > 0 { + _, err := w.Write([]byte(test.responseBody)) + assert.NoError(t, err) + } + })) + client := mockAPIClient(t, mockServer) + result, err := client.UpdateAdvancedSettingsEvasivePathMatch( + session.ContextWithOptions( + context.Background(), + session.WithContextHeaders(test.headers)), test.params) + if test.withError != nil { + assert.True(t, errors.Is(err, test.withError), "want: %s; got: %s", test.withError, err) + return + } + require.NoError(t, err) + assert.Equal(t, test.expectedResponse, result) + }) + } +} diff --git a/pkg/appsec/appsec.go b/pkg/appsec/appsec.go index 8c58791b..dedd4987 100644 --- a/pkg/appsec/appsec.go +++ b/pkg/appsec/appsec.go @@ -16,6 +16,7 @@ type ( // APPSEC is the appsec api interface APPSEC interface { Activations + AdvancedSettingsEvasivePathMatch AdvancedSettingsLogging AdvancedSettingsPrefetch AdvancedSettingsPragma diff --git a/pkg/appsec/export_configuration.go b/pkg/appsec/export_configuration.go index 70c06c29..1fb529e7 100644 --- a/pkg/appsec/export_configuration.go +++ b/pkg/appsec/export_configuration.go @@ -206,6 +206,7 @@ type ( SlowPost *SlowPostexp `json:"slowPost,omitempty"` LoggingOverrides *LoggingOverridesexp `json:"loggingOverrides,omitempty"` PragmaHeader *GetAdvancedSettingsPragmaResponse `json:"pragmaHeader,omitempty"` + EvasivePathMatch *EvasivePathMatchexp `json:"evasivePathMatch,omitempty"` } `json:"securityPolicies"` Siem *Siemexp `json:"siem,omitempty"` AdvancedOptions *AdvancedOptionsexp `json:"advancedOptions,omitempty"` @@ -403,6 +404,7 @@ type ( SlowPost *SlowPostexp `json:"slowPost,omitempty"` LoggingOverrides *LoggingOverridesexp `json:"loggingOverrides,omitempty"` PragmaHeader *GetAdvancedSettingsPragmaResponse `json:"pragmaHeader,omitempty"` + EvasivePathMatch *EvasivePathMatchexp `json:"evasivePathMatch,omitempty"` } `json:"securityPolicies"` Siem *Siemexp `json:"siem,omitempty"` AdvancedOptions *AdvancedOptionsexp `json:"advancedOptions,omitempty"` @@ -459,8 +461,9 @@ type ( // AdvancedOptionsexp is returned as part of GetExportConfigurationResponse. AdvancedOptionsexp struct { - Logging *Loggingexp `json:"logging"` - Prefetch struct { + Logging *Loggingexp `json:"logging"` + EvasivePathMatch *EvasivePathMatchexp `json:"evasivePathMatch,omitempty"` + Prefetch struct { AllExtensions bool `json:"allExtensions"` EnableAppLayer bool `json:"enableAppLayer"` EnableRateControls bool `json:"enableRateControls"` @@ -608,6 +611,11 @@ type ( } `json:"standardHeaders"` } + // EvasivePathMatchexp contains the EnablePathMatch setting + EvasivePathMatchexp struct { + EnablePathMatch bool `json:"enabled"` + } + // ConditionsExp is returned as part of GetExportConfigurationResponse. ConditionsExp []struct { Type string `json:"type"` diff --git a/pkg/cloudlets/loadbalancer_activation.go b/pkg/cloudlets/loadbalancer_activation.go index 593d9b1e..0253a44f 100644 --- a/pkg/cloudlets/loadbalancer_activation.go +++ b/pkg/cloudlets/loadbalancer_activation.go @@ -120,8 +120,8 @@ func (v ListLoadBalancerActivationsRequest) Validate() error { //Validate validates LoadBalancerVersionActivation Struct func (v LoadBalancerVersionActivation) Validate() error { return validation.Errors{ - "Network": validation.Validate(v.Network, validation.In(LoadBalancerActivationNetworkStaging, LoadBalancerActivationNetworkProduction).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'STAGING', 'PRODUCTION' or '' (empty)", v.Network))), + "Network": validation.Validate(v.Network, validation.Required, validation.In(LoadBalancerActivationNetworkStaging, LoadBalancerActivationNetworkProduction).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'STAGING' or 'PRODUCTION'", v.Network))), "Version": validation.Validate(v.Version, validation.Min(0)), }.Filter() } diff --git a/pkg/cloudlets/loadbalancer_activation_test.go b/pkg/cloudlets/loadbalancer_activation_test.go index 042d6e13..493add5e 100644 --- a/pkg/cloudlets/loadbalancer_activation_test.go +++ b/pkg/cloudlets/loadbalancer_activation_test.go @@ -247,18 +247,21 @@ func TestActivateLoadBalancerVersion(t *testing.T) { assert.True(t, errors.Is(err, want), "want: %s; got: %s", want, err) }, }, - "Validation Error": { + "Validation Errors": { params: ActivateLoadBalancerVersionRequest{ OriginID: "", Async: false, LoadBalancerVersionActivation: LoadBalancerVersionActivation{ - Network: LoadBalancerActivationNetworkStaging, + Network: "", DryRun: false, - Version: 1, + Version: -1, }, }, responseStatus: http.StatusInternalServerError, withError: func(t *testing.T, err error) { + assert.Containsf(t, err.Error(), "Network: cannot be blank", "want: %s; got: %s", ErrStructValidation, err) + assert.Containsf(t, err.Error(), "OriginID: cannot be blank", "want: %s; got: %s", ErrStructValidation, err) + assert.Containsf(t, err.Error(), "Version: must be no less than 0", "want: %s; got: %s", ErrStructValidation, err) assert.True(t, errors.Is(err, ErrStructValidation), "want: %s; got: %s", ErrStructValidation, err) }, }, diff --git a/pkg/cloudlets/match_rule.go b/pkg/cloudlets/match_rule.go new file mode 100644 index 00000000..a7eae021 --- /dev/null +++ b/pkg/cloudlets/match_rule.go @@ -0,0 +1,851 @@ +package cloudlets + +import ( + "encoding/json" + "errors" + "fmt" + + "github.com/akamai/AkamaiOPEN-edgegrid-golang/v2/pkg/edgegriderr" + + validation "github.com/go-ozzo/ozzo-validation/v4" +) + +type ( + // MatchRule is base interface for MarchRuleALB and MatchRuleER + MatchRule interface { + // cloudletType is a private method to ensure that only match rules for supported cloudlets can be used + cloudletType() string + Validate() error + } + + // MatchRules is an array of *MarchRuleALB or *MatchRuleER depending on the cloudletId (9 or 0) of the policy + MatchRules []MatchRule + + // MatchRuleALB represents a match rule resource for create or update resource + MatchRuleALB struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaALB `json:"matches,omitempty"` + MatchURL string `json:"matchURL,omitempty"` + MatchesAlways bool `json:"matchesAlways"` + ForwardSettings ForwardSettingsALB `json:"forwardSettings"` + Disabled bool `json:"disabled,omitempty"` + } + + // ForwardSettingsALB represents forward settings for ALB + ForwardSettingsALB struct { + OriginID string `json:"originId"` + } + + // MatchRuleAP represents an API Prioritization match rule resource for create or update + MatchRuleAP struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaAP `json:"matches,omitempty"` + MatchURL string `json:"matchURL,omitempty"` + PassThroughPercent *float64 `json:"passThroughPercent"` + Disabled bool `json:"disabled,omitempty"` + } + + // MatchRulePR represents a match rule resource for create or update resource + MatchRulePR struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaPR `json:"matches,omitempty"` + MatchURL string `json:"matchURL,omitempty"` + ForwardSettings ForwardSettingsPR `json:"forwardSettings"` + Disabled bool `json:"disabled,omitempty"` + MatchesAlways bool `json:"matchesAlways,omitempty"` + } + + // ForwardSettingsPR represents forward settings for CD aka PR + ForwardSettingsPR struct { + OriginID string `json:"originId"` + Percent int `json:"percent"` + } + + // MatchRuleER represents a match rule resource for create or update resource + MatchRuleER struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaER `json:"matches,omitempty"` + UseRelativeURL string `json:"useRelativeUrl,omitempty"` + StatusCode int `json:"statusCode"` + RedirectURL string `json:"redirectURL"` + MatchURL string `json:"matchURL,omitempty"` + UseIncomingQueryString bool `json:"useIncomingQueryString"` + UseIncomingSchemeAndHost bool `json:"useIncomingSchemeAndHost"` + Disabled bool `json:"disabled,omitempty"` + } + + // MatchRuleFR represents a match rule resource for create or update resource + MatchRuleFR struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaFR `json:"matches,omitempty"` + MatchURL string `json:"matchURL,omitempty"` + ForwardSettings ForwardSettingsFR `json:"forwardSettings"` + Disabled bool `json:"disabled,omitempty"` + } + + // ForwardSettingsFR represents forward settings for FR + ForwardSettingsFR struct { + PathAndQS string `json:"pathAndQS,omitempty"` + UseIncomingQueryString bool `json:"useIncomingQueryString,omitempty"` + OriginID string `json:"originId,omitempty"` + } + + // MatchRuleVP represents a match rule resource for create or update resource + MatchRuleVP struct { + Name string `json:"name,omitempty"` + Type MatchRuleType `json:"type,omitempty"` + Start int64 `json:"start,omitempty"` + End int64 `json:"end,omitempty"` + ID int64 `json:"id,omitempty"` + Matches []MatchCriteriaVP `json:"matches,omitempty"` + MatchURL string `json:"matchURL,omitempty"` + PassThroughPercent *float64 `json:"passThroughPercent"` + Disabled bool `json:"disabled,omitempty"` + } + + // MatchCriteria represents a match criteria resource for match rule for cloudlet + MatchCriteria struct { + MatchType string `json:"matchType,omitempty"` + MatchValue string `json:"matchValue,omitempty"` + MatchOperator MatchOperator `json:"matchOperator,omitempty"` + CaseSensitive bool `json:"caseSensitive"` + Negate bool `json:"negate"` + CheckIPs CheckIPs `json:"checkIPs,omitempty"` + ObjectMatchValue interface{} `json:"objectMatchValue,omitempty"` + } + + // MatchCriteriaALB represents a match criteria resource for match rule for cloudlet ALB + // ObjectMatchValue can contain ObjectMatchValueObject, ObjectMatchValueSimple or ObjectMatchValueRange + MatchCriteriaALB MatchCriteria + + // MatchCriteriaAP represents a match criteria resource for match rule for cloudlet AP + // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple + MatchCriteriaAP MatchCriteria + + // MatchCriteriaPR represents a match criteria resource for match rule for cloudlet CD aka PR + // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple + MatchCriteriaPR MatchCriteria + + // MatchCriteriaER represents a match criteria resource for match rule for cloudlet ER + // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple + MatchCriteriaER MatchCriteria + + // MatchCriteriaFR represents a match criteria resource for match rule for cloudlet FR + // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple + MatchCriteriaFR MatchCriteria + + // MatchCriteriaVP represents a match criteria resource for match rule for cloudlet VP + // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple + MatchCriteriaVP MatchCriteria + + // ObjectMatchValueObject represents an object match value resource for match criteria of type object + ObjectMatchValueObject struct { + Name string `json:"name"` + Type ObjectMatchValueObjectType `json:"type"` + NameCaseSensitive bool `json:"nameCaseSensitive"` + NameHasWildcard bool `json:"nameHasWildcard"` + Options *Options `json:"options,omitempty"` + } + + // ObjectMatchValueSimple represents an object match value resource for match criteria of type simple + ObjectMatchValueSimple struct { + Type ObjectMatchValueSimpleType `json:"type"` + Value []string `json:"value,omitempty"` + } + + // ObjectMatchValueRange represents an object match value resource for match criteria of type range + ObjectMatchValueRange struct { + Type ObjectMatchValueRangeType `json:"type"` + Value []int64 `json:"value,omitempty"` + } + + // Options represents an option resource for ObjectMatchValueObject + Options struct { + Value []string `json:"value,omitempty"` + ValueHasWildcard bool `json:"valueHasWildcard,omitempty"` + ValueCaseSensitive bool `json:"valueCaseSensitive,omitempty"` + ValueEscaped bool `json:"valueEscaped,omitempty"` + } + + //MatchRuleType enum type + MatchRuleType string + // MatchRuleFormat enum type + MatchRuleFormat string + // MatchOperator enum type + MatchOperator string + // CheckIPs enum type + CheckIPs string + // ObjectMatchValueRangeType enum type + ObjectMatchValueRangeType string + // ObjectMatchValueSimpleType enum type + ObjectMatchValueSimpleType string + // ObjectMatchValueObjectType enum type + ObjectMatchValueObjectType string +) + +const ( + // MatchRuleTypeALB represents rule type for ALB cloudlets + MatchRuleTypeALB MatchRuleType = "albMatchRule" + // MatchRuleTypeAP represents rule type for AP cloudlets + MatchRuleTypeAP MatchRuleType = "apMatchRule" + // MatchRuleTypePR represents rule type for PR aka PR cloudlets + MatchRuleTypePR MatchRuleType = "cdMatchRule" + // MatchRuleTypeER represents rule type for ER cloudlets + MatchRuleTypeER MatchRuleType = "erMatchRule" + // MatchRuleTypeFR represents rule type for FR cloudlets + MatchRuleTypeFR MatchRuleType = "frMatchRule" + // MatchRuleTypeVP represents rule type for VP cloudlets + MatchRuleTypeVP MatchRuleType = "vpMatchRule" +) + +const ( + // MatchRuleFormat10 represents default match rule format + MatchRuleFormat10 MatchRuleFormat = "1.0" +) + +const ( + // MatchOperatorContains represents contains operator + MatchOperatorContains MatchOperator = "contains" + // MatchOperatorExists represents exists operator + MatchOperatorExists MatchOperator = "exists" + // MatchOperatorEquals represents equals operator + MatchOperatorEquals MatchOperator = "equals" +) + +const ( + // CheckIPsConnectingIP represents connecting ip option + CheckIPsConnectingIP CheckIPs = "CONNECTING_IP" + // CheckIPsXFFHeaders represents xff headers option + CheckIPsXFFHeaders CheckIPs = "XFF_HEADERS" + // CheckIPsConnectingIPXFFHeaders represents connecting ip + xff headers option + CheckIPsConnectingIPXFFHeaders CheckIPs = "CONNECTING_IP XFF_HEADERS" +) + +const ( + // Range represents range option + Range ObjectMatchValueRangeType = "range" + // Simple represents simple option + Simple ObjectMatchValueSimpleType = "simple" + // Object represents object option + Object ObjectMatchValueObjectType = "object" +) + +var ( + // ErrUnmarshallMatchCriteriaALB is returned when unmarshalling of MatchCriteriaALB fails + ErrUnmarshallMatchCriteriaALB = errors.New("unmarshalling MatchCriteriaALB") + // ErrUnmarshallMatchCriteriaAP is returned when unmarshalling of MatchCriteriaAP fails + ErrUnmarshallMatchCriteriaAP = errors.New("unmarshalling MatchCriteriaAP") + // ErrUnmarshallMatchCriteriaPR is returned when unmarshalling of MatchCriteriaPR fails + ErrUnmarshallMatchCriteriaPR = errors.New("unmarshalling MatchCriteriaPR") + // ErrUnmarshallMatchCriteriaER is returned when unmarshalling of MatchCriteriaER fails + ErrUnmarshallMatchCriteriaER = errors.New("unmarshalling MatchCriteriaER") + // ErrUnmarshallMatchCriteriaFR is returned when unmarshalling of MatchCriteriaFR fails + ErrUnmarshallMatchCriteriaFR = errors.New("unmarshalling MatchCriteriaFR") + // ErrUnmarshallMatchCriteriaVP is returned when unmarshalling of MatchCriteriaVP fails + ErrUnmarshallMatchCriteriaVP = errors.New("unmarshalling MatchCriteriaVP") + // ErrUnmarshallMatchRules is returned when unmarshalling of MatchRules fails + ErrUnmarshallMatchRules = errors.New("unmarshalling MatchRules") +) + +// matchRuleHandlers contains mapping between name of the type for MatchRule and its implementation +// It makes the UnmarshalJSON more compact and easier to support more cloudlet types +var matchRuleHandlers = map[string]func() MatchRule{ + "albMatchRule": func() MatchRule { return &MatchRuleALB{} }, + "apMatchRule": func() MatchRule { return &MatchRuleAP{} }, + "cdMatchRule": func() MatchRule { return &MatchRulePR{} }, + "erMatchRule": func() MatchRule { return &MatchRuleER{} }, + "vpMatchRule": func() MatchRule { return &MatchRuleVP{} }, + "frMatchRule": func() MatchRule { return &MatchRuleFR{} }, +} + +// objectALBMatchValueHandlers contains mapping between name of the type for ObjectMatchValue and its implementation +// It makes the UnmarshalJSON more compact and easier to support more types +var objectALBMatchValueHandlers = map[string]func() interface{}{ + "object": func() interface{} { return &ObjectMatchValueObject{} }, + "range": func() interface{} { return &ObjectMatchValueRange{} }, + "simple": func() interface{} { return &ObjectMatchValueSimple{} }, +} + +// simpleObjectMatchValueHandlers contains mapping between name of the types (simple or object) for ObjectMatchValue and their implementations +// It makes the UnmarshalJSON more compact and easier to support more types +var simpleObjectMatchValueHandlers = map[string]func() interface{}{ + "object": func() interface{} { return &ObjectMatchValueObject{} }, + "simple": func() interface{} { return &ObjectMatchValueSimple{} }, +} + +// Validate validates MatchRules +func (m MatchRules) Validate() error { + type matchRules MatchRules + + errs := validation.Errors{ + "MatchRules": validation.Validate(matchRules(m), validation.Length(0, 5000)), + } + return edgegriderr.ParseValidationErrors(errs) +} + +// Validate validates MatchRuleALB +func (m MatchRuleALB) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeALB).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'albMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "ForwardSettings.OriginID": validation.Validate(m.ForwardSettings.OriginID, validation.Required, validation.Length(0, 8192)), + "Matches": validation.Validate(m.Matches), + }.Filter() +} + +// Validate validates MatchRuleAP +func (m MatchRuleAP) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeAP).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'apMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "PassThroughPercent": validation.Validate(m.PassThroughPercent, validation.By(passThroughPercentValidation)), + "Matches": validation.Validate(m.Matches), + }.Filter() +} + +// Validate validates MatchRulePR +func (m MatchRulePR) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypePR).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'cdMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "ForwardSettings": validation.Validate(m.ForwardSettings, validation.Required), + "ForwardSettings.OriginID": validation.Validate(m.ForwardSettings.OriginID, validation.Required, validation.Length(0, 8192)), + "ForwardSettings.Percent": validation.Validate(m.ForwardSettings.Percent, validation.Required, validation.Min(1), validation.Max(100)), + "Matches": validation.Validate(m.Matches), + }.Filter() +} + +// Validate validates MatchRuleER +func (m MatchRuleER) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeER).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'erMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "RedirectURL": validation.Validate(m.RedirectURL, validation.Required, validation.Length(1, 8192)), + "UseRelativeURL": validation.Validate(m.UseRelativeURL, validation.In("none", "copy_scheme_hostname", "relative_url").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'none', 'copy_scheme_hostname', 'relative_url' or '' (empty)", (&m).UseRelativeURL))), + "StatusCode": validation.Validate(m.StatusCode, validation.Required, validation.In(301, 302, 303, 307, 308).Error( + fmt.Sprintf("value '%d' is invalid. Must be one of: 301, 302, 303, 307 or 308", (&m).StatusCode))), + "Matches": validation.Validate(m.Matches), + }.Filter() +} + +// Validate validates MatchRuleFR +func (m MatchRuleFR) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeFR).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'frMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "Matches": validation.Validate(m.Matches), + "ForwardSettings": validation.Validate(m.ForwardSettings, validation.Required), + "ForwardSettings.PathAndQS": validation.Validate(m.ForwardSettings.PathAndQS, validation.Length(1, 8192)), + "ForwardSettings.OriginID": validation.Validate(m.ForwardSettings.OriginID, validation.Length(0, 8192)), + }.Filter() +} + +// Validate validates MatchRuleVP +func (m MatchRuleVP) Validate() error { + return validation.Errors{ + "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeVP).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'vpMatchRule'", (&m).Type))), + "Name": validation.Validate(m.Name, validation.Length(0, 8192)), + "Start": validation.Validate(m.Start, validation.Min(0)), + "End": validation.Validate(m.End, validation.Min(0)), + "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), + "PassThroughPercent": validation.Validate(m.PassThroughPercent, validation.By(passThroughPercentValidation)), + "Matches": validation.Validate(m.Matches), + }.Filter() +} + +// Validate validates MatchCriteriaALB +func (m MatchCriteriaALB) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.In("clientip", "continent", "cookie", "countrycode", + "deviceCharacteristics", "extension", "header", "hostname", "method", "path", "protocol", "proxy", "query", "regioncode", "range").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'clientip', 'continent', 'cookie', 'countrycode', 'deviceCharacteristics', "+ + "'extension', 'header', 'hostname', 'method', 'path', 'protocol', 'proxy', 'query', 'regioncode', 'range' or '' (empty)", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrRangeOrObjectValidation)), + }.Filter() +} + +// Validate validates MatchCriteriaAP +func (m MatchCriteriaAP) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.In( + "header", "hostname", "path", "extension", "query", "cookie", "deviceCharacteristics", "clientip", + "continent", "countrycode", "regioncode", "protocol", "method", "proxy").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'cookie', "+ + "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy'", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrObjectValidation)), + }.Filter() +} + +// Validate validates MatchCriteriaPR +func (m MatchCriteriaPR) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.In("header", "hostname", "path", "extension", + "query", "cookie", "deviceCharacteristics", "clientip", "continent", "countrycode", "regioncode", "protocol", + "method", "proxy").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'cookie', "+ + "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy'", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrObjectValidation)), + }.Filter() +} + +// Validate validates MatchCriteriaER +func (m MatchCriteriaER) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.In("header", "hostname", "path", "extension", "query", + "regex", "cookie", "deviceCharacteristics", "clientip", "continent", "countrycode", "regioncode", "protocol", "method", "proxy").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'regex', 'cookie', "+ + "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy' or '' (empty)", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrObjectValidation)), + }.Filter() +} + +// Validate validates MatchCriteriaFR +func (m MatchCriteriaFR) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.Required, validation.In("header", "hostname", "path", "extension", "query", "regex", + "cookie", "deviceCharacteristics", "clientip", "continent", "countrycode", "regioncode", "protocol", "method", "proxy").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'regex', 'cookie', "+ + "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy'", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrObjectValidation)), + }.Filter() +} + +// Validate validates MatchCriteriaVP +func (m MatchCriteriaVP) Validate() error { + return validation.Errors{ + "MatchType": validation.Validate(m.MatchType, validation.In("header", "hostname", "path", "extension", "query", + "cookie", "deviceCharacteristics", "clientip", "continent", "countrycode", "regioncode", "protocol", "method", "proxy").Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'cookie', "+ + "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy'", (&m).MatchType))), + "MatchValue": validation.Validate(m.MatchValue, validation.Length(1, 8192), validation.Required.When(m.ObjectMatchValue == nil).Error("cannot be blank when ObjectMatchValue is blank"), + validation.Empty.When(m.ObjectMatchValue != nil).Error("must be blank when ObjectMatchValue is set")), + "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), + "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( + fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), + "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == "").Error("cannot be blank when MatchValue is blank"), + validation.Empty.When(m.MatchValue != "").Error("must be blank when MatchValue is set"), validation.By(objectMatchValueSimpleOrObjectValidation)), + }.Filter() +} + +func objectMatchValueSimpleOrObjectValidation(value interface{}) error { + if value == nil { + return nil + } + switch value.(type) { + case *ObjectMatchValueObject, *ObjectMatchValueSimple: + return nil + default: + return fmt.Errorf("type %T is invalid. Must be one of: 'simple' or 'object'", value) + } +} + +func objectMatchValueSimpleOrRangeOrObjectValidation(value interface{}) error { + if value == nil { + return nil + } + switch value.(type) { + case *ObjectMatchValueObject, *ObjectMatchValueSimple, *ObjectMatchValueRange: + return nil + default: + return fmt.Errorf("type %T is invalid. Must be one of: 'simple', 'range' or 'object'", value) + } +} + +func passThroughPercentValidation(value interface{}) error { + v, ok := value.(*float64) + if !ok { + return fmt.Errorf("type %T is invalid. Must be *float64", value) + } + if v == nil { + return fmt.Errorf("cannot be blank") + } + if *v < -1 { + return fmt.Errorf("must be no less than -1") + } + if *v > 100 { + return fmt.Errorf("must be no greater than 100") + } + return nil +} + +// Validate validates ObjectMatchValueRange +func (o ObjectMatchValueRange) Validate() error { + return validation.Errors{ + "Type": validation.Validate(o.Type, validation.In(Range).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'range'", (&o).Type))), + }.Filter() +} + +// Validate validates ObjectMatchValueSimple +func (o ObjectMatchValueSimple) Validate() error { + return validation.Errors{ + "Type": validation.Validate(o.Type, validation.In(Simple).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'simple'", (&o).Type))), + }.Filter() +} + +// Validate validates ObjectMatchValueObject +func (o ObjectMatchValueObject) Validate() error { + return validation.Errors{ + "Name": validation.Validate(o.Name, validation.Required, validation.Length(0, 8192)), + "Type": validation.Validate(o.Type, validation.Required, validation.In(Object).Error( + fmt.Sprintf("value '%s' is invalid. Must be: 'object'", (&o).Type))), + }.Filter() +} + +func (m MatchRuleALB) cloudletType() string { + return "albMatchRule" +} + +func (m MatchRuleAP) cloudletType() string { + return "apMatchRule" +} + +func (m MatchRulePR) cloudletType() string { + return "cdMatchRule" +} + +func (m MatchRuleER) cloudletType() string { + return "erMatchRule" +} + +func (m MatchRuleFR) cloudletType() string { + return "frMatchRule" +} + +func (m MatchRuleVP) cloudletType() string { + return "vpMatchRule" +} + +// UnmarshalJSON helps to un-marshall items of MatchRules array as proper instances of *MatchRuleALB or *MatchRuleER +func (m *MatchRules) UnmarshalJSON(b []byte) error { + data := make([]map[string]interface{}, 0) + if err := json.Unmarshal(b, &data); err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) + } + for _, matchRule := range data { + cloudletType, ok := matchRule["type"] + if !ok { + return fmt.Errorf("%w: match rule entry should contain 'type' field", ErrUnmarshallMatchRules) + } + cloudletTypeName, ok := cloudletType.(string) + if !ok { + return fmt.Errorf("%w: 'type' field on match rule entry should be a string", ErrUnmarshallMatchRules) + } + byteArr, err := json.Marshal(matchRule) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) + } + + matchRuleType, ok := matchRuleHandlers[cloudletTypeName] + if !ok { + return fmt.Errorf("%w: unsupported match rule type: %s", ErrUnmarshallMatchRules, cloudletTypeName) + } + dst := matchRuleType() + err = json.Unmarshal(byteArr, dst) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) + } + *m = append(*m, dst) + } + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaALB as proper instance of *ObjectMatchValueObject, *ObjectMatchValueSimple or *ObjectMatchValueRange +func (m *MatchCriteriaALB) UnmarshalJSON(b []byte) error { + // matchCriteriaALB is an alias for MatchCriteriaALB for un-marshalling purposes + type matchCriteriaALB MatchCriteriaALB + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaALB)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) + } + + createObjectMatchValue, ok := objectALBMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaALB, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaAP as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple +func (m *MatchCriteriaAP) UnmarshalJSON(b []byte) error { + // matchCriteriaER is an alias for MatchCriteriaER for un-marshalling purposes + type matchCriteriaAP MatchCriteriaAP + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaAP)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaAP, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaAP, err) + } + + createObjectMatchValue, ok := simpleObjectMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaAP, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaAP, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaPR as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple +func (m *MatchCriteriaPR) UnmarshalJSON(b []byte) error { + // matchCriteriaPR is an alias for MatchCriteriaPR for un-marshalling purposes + type matchCriteriaPR MatchCriteriaPR + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaPR)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaPR, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaPR, err) + } + + createObjectMatchValue, ok := simpleObjectMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaPR, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaPR, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaER as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple +func (m *MatchCriteriaER) UnmarshalJSON(b []byte) error { + // matchCriteriaER is an alias for MatchCriteriaER for un-marshalling purposes + type matchCriteriaER MatchCriteriaER + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaER)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) + } + + createObjectMatchValue, ok := simpleObjectMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaER, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaFR as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple +func (m *MatchCriteriaFR) UnmarshalJSON(b []byte) error { + // matchCriteriaFR is an alias for MatchCriteriaFR for un-marshalling purposes + type matchCriteriaFR MatchCriteriaFR + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaFR)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaFR, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaFR, err) + } + + createObjectMatchValue, ok := simpleObjectMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaFR, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaFR, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaVP as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple +func (m *MatchCriteriaVP) UnmarshalJSON(b []byte) error { + // matchCriteriaVP is an alias for MatchCriteriaVP for un-marshalling purposes + type matchCriteriaVP MatchCriteriaVP + + // populate common attributes using default json unmarshaler using aliased type + err := json.Unmarshal(b, (*matchCriteriaVP)(m)) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaVP, err) + } + if m.ObjectMatchValue == nil { + return nil + } + + objectMatchValueTypeName, err := getObjectMatchValueType(m.ObjectMatchValue) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaVP, err) + } + + createObjectMatchValue, ok := simpleObjectMatchValueHandlers[objectMatchValueTypeName] + if !ok { + return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaVP, objectMatchValueTypeName) + } + convertedObjectMatchValue, err := convertObjectMatchValue(m.ObjectMatchValue, createObjectMatchValue()) + if err != nil { + return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaVP, err) + } + m.ObjectMatchValue = convertedObjectMatchValue + + return nil +} + +func getObjectMatchValueType(omv interface{}) (string, error) { + objectMatchValueMap, ok := omv.(map[string]interface{}) + if !ok { + return "", fmt.Errorf("structure of objectMatchValue should be 'map', but was '%T'", omv) + } + objectMatchValueType, ok := objectMatchValueMap["type"] + if !ok { + return "", fmt.Errorf("objectMatchValue should contain 'type' field") + } + objectMatchValueTypeName, ok := objectMatchValueType.(string) + if !ok { + return "", fmt.Errorf("'type' should be a string") + } + return objectMatchValueTypeName, nil +} + +func convertObjectMatchValue(in, out interface{}) (interface{}, error) { + marshal, err := json.Marshal(in) + if err != nil { + return nil, fmt.Errorf("%s", err) + } + err = json.Unmarshal(marshal, out) + if err != nil { + return nil, fmt.Errorf("%s", err) + } + + return out, nil +} diff --git a/pkg/cloudlets/match_rule_test.go b/pkg/cloudlets/match_rule_test.go new file mode 100644 index 00000000..3da7ee22 --- /dev/null +++ b/pkg/cloudlets/match_rule_test.go @@ -0,0 +1,1186 @@ +package cloudlets + +import ( + "encoding/json" + "errors" + "strings" + "testing" + + "github.com/stretchr/testify/require" + "github.com/tj/assert" + + "github.com/akamai/AkamaiOPEN-edgegrid-golang/v2/pkg/cloudlets/tools" +) + +func TestUnmarshalJSONMatchRules(t *testing.T) { + tests := map[string]struct { + withError error + responseBody string + expectedObject MatchRules + }{ + "valid MatchRuleALB": { + responseBody: ` + [ + { + "type": "albMatchRule", + "end": 0, + "forwardSettings": { + "originId": "alb_test_krk_dc1_only" + }, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "protocol", + "matchValue": "https", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "range", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + expectedObject: MatchRules{ + &MatchRuleALB{ + Type: "albMatchRule", + End: 0, + ForwardSettings: ForwardSettingsALB{ + OriginID: "alb_test_krk_dc1_only", + }, + ID: 0, + MatchURL: "", + Matches: []MatchCriteriaALB{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "protocol", + MatchValue: "https", + Negate: false, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "range", + Negate: false, + ObjectMatchValue: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Name: "Rule3", + Start: 0, + }, + }, + }, + + "invalid objectMatchValue type for ALB - foo": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: objectMatchValue has unexpected type: 'foo'"), + responseBody: ` + [ + { + "type": "albMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "foo", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "wrong type for object value type": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: 'type' should be a string"), + responseBody: ` + [ + { + "type": "albMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": 1, + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "missing object value type": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: objectMatchValue should contain 'type' field"), + responseBody: ` + [ + { + "type": "albMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "invalid object value": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: structure of objectMatchValue should be 'map', but was 'string'"), + responseBody: ` + [ + { + "type": "albMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": "" + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "invalid MatchRuleXX": { + responseBody: ` + [ + { + "type": "xxMatchRule" + } + ] +`, + withError: errors.New("unmarshalling MatchRules: unsupported match rule type: xxMatchRule"), + }, + + "invalid type": { + withError: errors.New("unmarshalling MatchRules: 'type' field on match rule entry should be a string"), + responseBody: ` + [ + { + "type": 1 + } + ] +`, + }, + + "invalid JSON": { + withError: errors.New("unexpected end of JSON input"), + responseBody: ` + [ + { + "type": "albMatchRule" + } + +`, + }, + + "missing type": { + withError: errors.New("unmarshalling MatchRules: match rule entry should contain 'type' field"), + responseBody: ` + [ + { + } + ] +`, + }, + + "invalid objectMatchValue type for PR - range": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaPR: objectMatchValue has unexpected type: 'range'"), + responseBody: ` + [ + { + "type": "cdMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "invalid objectMatchValue type for ER - range": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaER: objectMatchValue has unexpected type: 'range'"), + responseBody: ` + [ + { + "type": "erMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "invalid objectMatchValue type for FR - range": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaFR: objectMatchValue has unexpected type: 'range'"), + responseBody: ` + [ + { + "type": "frMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "valid MatchRulePR": { + responseBody: ` + [ + { + "type": "cdMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "forwardSettings": { + "originId": "fr_test_krk_dc2", + "percent": 62 + }, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "protocol", + "matchValue": "https", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + expectedObject: MatchRules{ + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Matches: []MatchCriteriaPR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "protocol", + MatchValue: "https", + Negate: false, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Name: "Rule3", + Start: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "fr_test_krk_dc2", + Percent: 62, + }, + }, + }, + }, + + "valid MatchRuleFR": { + responseBody: ` + [ + { + "type": "frMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "forwardSettings": {}, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "protocol", + "matchValue": "https", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + expectedObject: MatchRules{ + &MatchRuleFR{ + Type: "frMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Matches: []MatchCriteriaFR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "protocol", + MatchValue: "https", + Negate: false, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Name: "Rule3", + Start: 0, + }, + }, + }, + + "invalid objectMatchValue type for VP - range": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaVP: objectMatchValue has unexpected type: 'range'"), + responseBody: ` + [ + { + "type": "vpMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + "invalid objectMatchValue type for AP - range": { + withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaAP: objectMatchValue has unexpected type: 'range'"), + responseBody: ` + [ + { + "type": "apMatchRule", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "range", + "value": [ + 1, + 50 + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + }, + + "valid MatchRuleVP": { + responseBody: ` + [ + { + "type": "vpMatchRule", + "end": 0, + "passThroughPercent": 50.50, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "protocol", + "matchValue": "https", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + expectedObject: MatchRules{ + &MatchRuleVP{ + Type: "vpMatchRule", + End: 0, + PassThroughPercent: tools.Float64Ptr(50.50), + ID: 0, + MatchURL: "", + Matches: []MatchCriteriaVP{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "protocol", + MatchValue: "https", + Negate: false, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Name: "Rule3", + Start: 0, + }, + }, + }, + "valid MatchRuleAP": { + responseBody: ` + [ + { + "type": "apMatchRule", + "end": 0, + "passThroughPercent": 50.50, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "protocol", + "matchValue": "https", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "Rule3", + "start": 0 + } + ] +`, + expectedObject: MatchRules{ + &MatchRuleAP{ + Type: "apMatchRule", + End: 0, + PassThroughPercent: tools.Float64Ptr(50.50), + ID: 0, + MatchURL: "", + Matches: []MatchCriteriaAP{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "protocol", + MatchValue: "https", + Negate: false, + }, + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Name: "Rule3", + Start: 0, + }, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + var matchRules MatchRules + err := json.Unmarshal([]byte(test.responseBody), &matchRules) + + if test.withError != nil { + assert.Equal(t, test.withError.Error(), err.Error()) + return + } + require.NoError(t, err) + assert.Equal(t, test.expectedObject, matchRules) + }) + } +} + +func TestGetObjectMatchValueType(t *testing.T) { + tests := map[string]struct { + withError error + input interface{} + expected string + }{ + "success getting objectMatchValue type": { + input: map[string]interface{}{ + "type": "range", + "value": []int{1, 50}, + }, + expected: "range", + }, + "error getting objectMatchValue type - invalid type": { + withError: errors.New("structure of objectMatchValue should be 'map', but was 'string'"), + input: "stringType", + }, + "error getting objectMatchValue type - missing type": { + withError: errors.New("objectMatchValue should contain 'type' field"), + input: map[string]interface{}{ + "value": []int{1, 50}, + }, + }, + "error getting objectMatchValue type - type not string": { + withError: errors.New("'type' should be a string"), + input: map[string]interface{}{ + "type": 50, + "value": []int{1, 50}, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + objectMatchValueType, err := getObjectMatchValueType(test.input) + + if test.withError != nil { + assert.Equal(t, test.withError.Error(), err.Error()) + return + } + require.NoError(t, err) + assert.Equal(t, test.expected, objectMatchValueType) + }) + } +} + +func TestConvertObjectMatchValue(t *testing.T) { + tests := map[string]struct { + withError bool + input map[string]interface{} + output interface{} + expected interface{} + }{ + "success converting objectMatchValueRange": { + input: map[string]interface{}{ + "type": "range", + "value": []int{1, 50}, + }, + output: &ObjectMatchValueRange{}, + expected: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + "success converting objectMatchValueSimple": { + input: map[string]interface{}{ + "type": "simple", + "value": []string{"GET"}, + }, + output: &ObjectMatchValueSimple{}, + expected: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + "success converting objectMatchValueObject": { + input: map[string]interface{}{ + "type": "object", + "name": "ER", + "options": map[string]interface{}{ + "value": []string{ + "text/html*", + "text/css*", + "application/x-javascript*", + }, + "valueHasWildcard": true, + }, + }, + output: &ObjectMatchValueObject{}, + expected: &ObjectMatchValueObject{ + Type: "object", + Name: "ER", + Options: &Options{ + Value: []string{ + "text/html*", + "text/css*", + "application/x-javascript*", + }, + ValueHasWildcard: true, + }, + }, + }, + "error converting objectMatchValue": { + withError: true, + input: map[string]interface{}{ + "type": "range", + "value": []int{1, 50}, + }, + output: &ObjectMatchValueSimple{}, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + convertedObjectMatchValue, err := convertObjectMatchValue(test.input, test.output) + + if test.withError == true { + require.Error(t, err) + return + } + require.NoError(t, err) + assert.Equal(t, test.expected, convertedObjectMatchValue) + }) + } +} + +func TestValidateMatchRules(t *testing.T) { + tests := map[string]struct { + input MatchRules + withError string + }{ + "valid match rules ALB": { + input: MatchRules{ + MatchRuleALB{ + Type: "albMatchRule", + ForwardSettings: ForwardSettingsALB{ + OriginID: "testOriginID", + }, + }, + MatchRuleALB{ + Type: "albMatchRule", + Start: 1, + End: 2, + ForwardSettings: ForwardSettingsALB{ + OriginID: "testOriginID", + }, + }, + }, + }, + "invalid match rules ALB": { + input: MatchRules{ + MatchRuleALB{ + Type: "matchRule", + }, + MatchRuleALB{ + Type: "albMatchRule", + Start: -1, + End: -1, + ForwardSettings: ForwardSettingsALB{ + OriginID: "testOriginID", + }, + }, + }, + withError: ` +MatchRules[0]: { + ForwardSettings.OriginID: cannot be blank + Type: value 'matchRule' is invalid. Must be: 'albMatchRule' +} +MatchRules[1]: { + End: must be no less than 0 + Start: must be no less than 0 +}`, + }, + "valid match rules AP": { + input: MatchRules{ + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(-1), + }, + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(50.5), + }, + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(0), + }, + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(100), + }, + }, + }, + "invalid match rules AP": { + input: MatchRules{ + MatchRuleAP{ + Type: "matchRule", + }, + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(100.1), + }, + MatchRuleAP{ + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(-1.1), + }, + }, + withError: ` +MatchRules[0]: { + PassThroughPercent: cannot be blank + Type: value 'matchRule' is invalid. Must be: 'apMatchRule' +} +MatchRules[1]: { + PassThroughPercent: must be no greater than 100 +} +MatchRules[2]: { + PassThroughPercent: must be no less than -1 +}`, + }, + "valid match rules CD": { + input: MatchRules{ + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{ + OriginID: "testOriginID", + Percent: 100, + }, + }, + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{ + OriginID: "testOriginID", + Percent: 1, + }, + }, + }, + }, + "invalid match rules CD": { + input: MatchRules{ + MatchRulePR{ + Type: "matchRule", + }, + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{}, + }, + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{ + OriginID: "testOriginID", + Percent: 101, + }, + }, + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{ + OriginID: "testOriginID", + Percent: -1, + }, + }, + MatchRulePR{ + Type: "cdMatchRule", + ForwardSettings: ForwardSettingsPR{ + OriginID: "testOriginID", + Percent: 0, + }, + }, + }, + withError: ` +MatchRules[0]: { + ForwardSettings.OriginID: cannot be blank + ForwardSettings.Percent: cannot be blank + Type: value 'matchRule' is invalid. Must be: 'cdMatchRule' +} +MatchRules[1]: { + ForwardSettings.OriginID: cannot be blank + ForwardSettings.Percent: cannot be blank +} +MatchRules[2]: { + ForwardSettings.Percent: must be no greater than 100 +} +MatchRules[3]: { + ForwardSettings.Percent: must be no less than 1 +} +MatchRules[4]: { + ForwardSettings.Percent: cannot be blank +}`, + }, + "valid match rules ER": { + input: MatchRules{ + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + UseRelativeURL: "none", + StatusCode: 301, + }, + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + StatusCode: 301, + }, + }, + }, + "invalid match rules ER": { + input: MatchRules{ + MatchRuleER{ + Type: "matchRule", + }, + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + UseRelativeURL: "test", + StatusCode: 404, + }, + }, + withError: ` +MatchRules[0]: { + RedirectURL: cannot be blank + StatusCode: cannot be blank + Type: value 'matchRule' is invalid. Must be: 'erMatchRule' +} +MatchRules[1]: { + StatusCode: value '404' is invalid. Must be one of: 301, 302, 303, 307 or 308 + UseRelativeURL: value 'test' is invalid. Must be one of: 'none', 'copy_scheme_hostname', 'relative_url' or '' (empty) +}`, + }, + "valid match rules FR": { + input: MatchRules{ + MatchRuleFR{ + Type: "frMatchRule", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "test", + OriginID: "testOriginID", + }, + }, + MatchRuleFR{ + Type: "frMatchRule", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "test", + OriginID: "testOriginID", + }, + }, + }, + }, + "invalid match rules FR": { + input: MatchRules{ + MatchRuleFR{ + Type: "matchRule", + }, + MatchRuleFR{ + Type: "frMatchRule", + ForwardSettings: ForwardSettingsFR{ + OriginID: "testOriginID", + PathAndQS: "", + }, + }, + }, + withError: ` +MatchRules[0]: { + Type: value 'matchRule' is invalid. Must be: 'frMatchRule' +}`, + }, + "valid match rules VP": { + input: MatchRules{ + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(-1), + }, + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(50.5), + }, + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(0), + }, + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(100), + }, + }, + }, + "invalid match rules VP": { + input: MatchRules{ + MatchRuleVP{ + Type: "matchRule", + }, + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(100.1), + }, + MatchRuleVP{ + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(-1.1), + }, + }, + withError: ` +MatchRules[0]: { + PassThroughPercent: cannot be blank + Type: value 'matchRule' is invalid. Must be: 'vpMatchRule' +} +MatchRules[1]: { + PassThroughPercent: must be no greater than 100 +} +MatchRules[2]: { + PassThroughPercent: must be no less than -1 +}`, + }, + "valid match criteria - matchValue": { + input: MatchRules{ + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + StatusCode: 301, + Matches: []MatchCriteriaER{ + { + MatchType: "method", + MatchOperator: "equals", + CheckIPs: "CONNECTING_IP", + MatchValue: "https", + }, + }, + }, + }, + }, + "valid match criteria - object match value": { + input: MatchRules{ + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + StatusCode: 301, + Matches: []MatchCriteriaER{ + { + MatchType: "header", + MatchOperator: "equals", + CheckIPs: "CONNECTING_IP", + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{ + "GET", + }, + }, + }, + }, + }, + }, + }, + "invalid match criteria - matchValue and omv combinations": { + input: MatchRules{ + MatchRuleER{ + Type: "erMatchRule", + RedirectURL: "abc.com", + StatusCode: 301, + Matches: []MatchCriteriaER{ + { + MatchType: "header", + MatchOperator: "equals", + CheckIPs: "CONNECTING_IP", + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{ + "GET", + }, + }, + MatchValue: "GET", + }, + { + MatchType: "header", + MatchOperator: "equals", + CheckIPs: "CONNECTING_IP", + }, + }, + }, + }, + withError: ` +MatchRules[0]: { + Matches[0]: { + MatchValue: must be blank when ObjectMatchValue is set + ObjectMatchValue: must be blank when MatchValue is set + } + Matches[1]: { + MatchValue: cannot be blank when ObjectMatchValue is blank + ObjectMatchValue: cannot be blank when MatchValue is blank + } +}`, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + err := test.input.Validate() + if test.withError != "" { + require.Error(t, err) + assert.Equal(t, strings.TrimPrefix(test.withError, "\n"), err.Error()) + return + } + require.NoError(t, err) + }) + } +} diff --git a/pkg/cloudlets/policy_version.go b/pkg/cloudlets/policy_version.go index 85ae19b2..2071fabf 100644 --- a/pkg/cloudlets/policy_version.go +++ b/pkg/cloudlets/policy_version.go @@ -2,7 +2,6 @@ package cloudlets import ( "context" - "encoding/json" "errors" "fmt" "net/http" @@ -112,156 +111,6 @@ type ( PolicyID int64 Version int64 } - - // MatchRule is base interface for MarchRuleALB and MatchRuleER - MatchRule interface { - // cloudletType is a private method to ensure that only match rules for supported cloudlets can be used - cloudletType() string - Validate() error - } - - // MatchRules is an array of *MarchRuleALB or *MatchRuleER depending on the cloudletId (9 or 0) of the policy - MatchRules []MatchRule - - // MatchRuleALB represents a match rule resource for create or update resource - MatchRuleALB struct { - Name string `json:"name,omitempty"` - Type MatchRuleType `json:"type,omitempty"` - Start int `json:"start,omitempty"` - End int `json:"end,omitempty"` - ID int64 `json:"id,omitempty"` - Matches []MatchCriteriaALB `json:"matches,omitempty"` - MatchURL string `json:"matchURL,omitempty"` - MatchesAlways bool `json:"matchesAlways"` - ForwardSettings ForwardSettings `json:"forwardSettings"` - Disabled bool `json:"disabled,omitempty"` - } - - // ForwardSettings represents forward settings - ForwardSettings struct { - OriginID string `json:"originId"` - } - - // MatchRuleER represents a match rule resource for create or update resource - MatchRuleER struct { - Name string `json:"name,omitempty"` - Type MatchRuleType `json:"type,omitempty"` - Start int `json:"start,omitempty"` - End int `json:"end,omitempty"` - ID int64 `json:"id,omitempty"` - Matches []MatchCriteriaER `json:"matches,omitempty"` - UseRelativeURL string `json:"useRelativeUrl,omitempty"` - StatusCode int `json:"statusCode"` - RedirectURL string `json:"redirectURL"` - MatchURL string `json:"matchURL,omitempty"` - UseIncomingQueryString bool `json:"useIncomingQueryString"` - UseIncomingSchemeAndHost bool `json:"useIncomingSchemeAndHost"` - Disabled bool `json:"disabled,omitempty"` - } - - // MatchCriteria represents a match criteria resource for match rule for cloudlet - MatchCriteria struct { - MatchType string `json:"matchType,omitempty"` - MatchValue string `json:"matchValue,omitempty"` - MatchOperator MatchOperator `json:"matchOperator,omitempty"` - CaseSensitive bool `json:"caseSensitive"` - Negate bool `json:"negate"` - CheckIPs CheckIPs `json:"checkIPs,omitempty"` - ObjectMatchValue interface{} `json:"objectMatchValue,omitempty"` - } - - // MatchCriteriaALB represents a match criteria resource for match rule for cloudlet ALB - // ObjectMatchValue can contain ObjectMatchValueObject, ObjectMatchValueSimple or ObjectMatchValueRange - MatchCriteriaALB MatchCriteria - - // MatchCriteriaER represents a match criteria resource for match rule for cloudlet ER - // ObjectMatchValue can contain ObjectMatchValueObject or ObjectMatchValueSimple - MatchCriteriaER MatchCriteria - - // ObjectMatchValueObject represents an object match value resource for match criteria of type object - ObjectMatchValueObject struct { - Name string `json:"name"` - Type ObjectMatchValueObjectType `json:"type"` - NameCaseSensitive bool `json:"nameCaseSensitive"` - NameHasWildcard bool `json:"nameHasWildcard"` - Options *Options `json:"options,omitempty"` - } - - // ObjectMatchValueSimple represents an object match value resource for match criteria of type simple - ObjectMatchValueSimple struct { - Type ObjectMatchValueSimpleType `json:"type"` - Value []string `json:"value,omitempty"` - } - - // ObjectMatchValueRange represents an object match value resource for match criteria of type range - ObjectMatchValueRange struct { - Type ObjectMatchValueRangeType `json:"type"` - Value []int64 `json:"value,omitempty"` - } - - // Options represents an option resource for ObjectMatchValueObject - Options struct { - Value []string `json:"value,omitempty"` - ValueHasWildcard bool `json:"valueHasWildcard,omitempty"` - ValueCaseSensitive bool `json:"valueCaseSensitive,omitempty"` - ValueEscaped bool `json:"valueEscaped,omitempty"` - } - - //MatchRuleType enum type - MatchRuleType string - // MatchRuleFormat enum type - MatchRuleFormat string - // MatchOperator enum type - MatchOperator string - // CheckIPs enum type - CheckIPs string - // ObjectMatchValueRangeType enum type - ObjectMatchValueRangeType string - // ObjectMatchValueSimpleType enum type - ObjectMatchValueSimpleType string - // ObjectMatchValueObjectType enum type - ObjectMatchValueObjectType string -) - -const ( - // MatchRuleTypeALB represents rule type for ALB cloudlets - MatchRuleTypeALB MatchRuleType = "albMatchRule" - // MatchRuleTypeER represents rule type for ER cloudlets - MatchRuleTypeER MatchRuleType = "erMatchRule" -) - -const ( - // MatchRuleFormat10 represents default match rule format - MatchRuleFormat10 MatchRuleFormat = "1.0" - // MatchRuleFormatDefault represents default match rule format - MatchRuleFormatDefault = MatchRuleFormat10 -) - -const ( - // MatchOperatorContains represents contains operator - MatchOperatorContains MatchOperator = "contains" - // MatchOperatorExists represents exists operator - MatchOperatorExists MatchOperator = "exists" - // MatchOperatorEquals represents equals operator - MatchOperatorEquals MatchOperator = "equals" -) - -const ( - // CheckIPsConnectingIP represents connecting ip option - CheckIPsConnectingIP CheckIPs = "CONNECTING_IP" - // CheckIPsXFFHeaders represents xff headers option - CheckIPsXFFHeaders CheckIPs = "XFF_HEADERS" - // CheckIPsConnectingIPXFFHeaders represents connecting ip + xff headers option - CheckIPsConnectingIPXFFHeaders CheckIPs = "CONNECTING_IP XFF_HEADERS" -) - -const ( - // Range represents range option - Range ObjectMatchValueRangeType = "range" - // Simple represents simple option - Simple ObjectMatchValueSimpleType = "simple" - // Object represents object option - Object ObjectMatchValueObjectType = "object" ) // Validate validates ListPolicyVersionsRequest @@ -284,119 +133,6 @@ func (c CreatePolicyVersionRequest) Validate() error { return edgegriderr.ParseValidationErrors(errs) } -// Validate validates MatchRuleALB -func (m MatchRuleALB) Validate() error { - return validation.Errors{ - "Type": validation.Validate(m.Type, validation.In(MatchRuleTypeALB).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'albMatchRule' or '' (empty)", (&m).Type))), - "Name": validation.Validate(m.Name, validation.Length(0, 8192)), - "Start": validation.Validate(m.Start, validation.Min(0)), - "End": validation.Validate(m.End, validation.Min(0)), - "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), - "ForwardSettings.OriginID": validation.Validate(m.ForwardSettings.OriginID, validation.Required, validation.Length(0, 8192)), - "Matches": validation.Validate(m.Matches), - }.Filter() -} - -// Validate validates MatchRuleER -func (m MatchRuleER) Validate() error { - return validation.Errors{ - "Type": validation.Validate(m.Type, validation.Required, validation.In(MatchRuleTypeER).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'erMatchRule' or '' (empty)", (&m).Type))), - "Name": validation.Validate(m.Name, validation.Length(0, 8192)), - "Start": validation.Validate(m.Start, validation.Min(0)), - "End": validation.Validate(m.End, validation.Min(0)), - "MatchURL": validation.Validate(m.MatchURL, validation.Length(0, 8192)), - "RedirectURL": validation.Validate(m.RedirectURL, validation.Required, validation.Length(1, 8192)), - "UseRelativeURL": validation.Validate(m.UseRelativeURL, validation.In("none", "copy_scheme_hostname", "relative_url").Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'none', 'copy_scheme_hostname', 'relative_url' or '' (empty)", (&m).UseRelativeURL))), - "StatusCode": validation.Validate(m.StatusCode, validation.Required, validation.In(301, 302, 303, 307, 308).Error( - fmt.Sprintf("value '%d' is invalid. Must be one of: 301, 302, 303, 307 or 308", (&m).StatusCode))), - "Matches": validation.Validate(m.Matches), - }.Filter() -} - -// Validate validates MatchCriteriaALB -func (m MatchCriteriaALB) Validate() error { - return validation.Errors{ - "MatchType": validation.Validate(m.MatchType, validation.In("clientip", "continent", "cookie", "countrycode", - "deviceCharacteristics", "extension", "header", "hostname", "method", "path", "protocol", "proxy", "query", "regioncode", "range").Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'clientip', 'continent', 'cookie', 'countrycode', 'deviceCharacteristics', "+ - "'extension', 'header', 'hostname', 'method', 'path', 'protocol', 'proxy', 'query', 'regioncode', 'range' or '' (empty)", (&m).MatchType))), - "MatchValue": validation.Validate(m.MatchValue, validation.Length(0, 8192), validation.Required.When(m.ObjectMatchValue == nil)), - "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), - "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), - "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == ""), validation.By(objectMatchValueALBValidation)), - }.Filter() -} - -func objectMatchValueALBValidation(value interface{}) error { - if value == nil { - return nil - } - switch value.(type) { - case *ObjectMatchValueObject, *ObjectMatchValueSimple, *ObjectMatchValueRange: - return nil - default: - return fmt.Errorf("type %T is invalid. Must be one of: 'simple', 'range' or 'object'", value) - } -} - -// Validate validates MatchCriteriaER -func (m MatchCriteriaER) Validate() error { - return validation.Errors{ - "MatchType": validation.Validate(m.MatchType, validation.In("header", "hostname", "path", "extension", "query", - "regex", "cookie", "deviceCharacteristics", "clientip", "continent", "countrycode", "regioncode", "protocol", "method", "proxy").Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'header', 'hostname', 'path', 'extension', 'query', 'regex', 'cookie', "+ - "'deviceCharacteristics', 'clientip', 'continent', 'countrycode', 'regioncode', 'protocol', 'method', 'proxy' or '' (empty)", (&m).MatchType))), - "MatchValue": validation.Validate(m.MatchValue, validation.Length(0, 8192)), - "MatchOperator": validation.Validate(m.MatchOperator, validation.In(MatchOperatorContains, MatchOperatorExists, MatchOperatorEquals).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'contains', 'exists', 'equals' or '' (empty)", (&m).MatchOperator))), - "CheckIPs": validation.Validate(m.CheckIPs, validation.In(CheckIPsConnectingIP, CheckIPsXFFHeaders, CheckIPsConnectingIPXFFHeaders).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'CONNECTING_IP', 'XFF_HEADERS', 'CONNECTING_IP XFF_HEADERS' or '' (empty)", (&m).CheckIPs))), - "ObjectMatchValue": validation.Validate(m.ObjectMatchValue, validation.Required.When(m.MatchValue == ""), validation.By(objectMatchValueERValidation)), - }.Filter() -} - -func objectMatchValueERValidation(value interface{}) error { - if value == nil { - return nil - } - switch value.(type) { - case *ObjectMatchValueObject, *ObjectMatchValueSimple: - return nil - default: - return fmt.Errorf("type %T is invalid. Must be one of: 'simple' or 'object'", value) - } -} - -// Validate validates ObjectMatchValueRange -func (o ObjectMatchValueRange) Validate() error { - return validation.Errors{ - "Type": validation.Validate(o.Type, validation.In(Range).Error( - fmt.Sprintf("value '%s' is invalid. Must be: 'range'", (&o).Type))), - }.Filter() -} - -// Validate validates ObjectMatchValueSimple -func (o ObjectMatchValueSimple) Validate() error { - return validation.Errors{ - "Type": validation.Validate(o.Type, validation.In(Simple).Error( - fmt.Sprintf("value '%s' is invalid. Must be: 'simple'", (&o).Type))), - }.Filter() -} - -// Validate validates ObjectMatchValueObject -func (o ObjectMatchValueObject) Validate() error { - return validation.Errors{ - "Name": validation.Validate(o.Name, validation.Required, validation.Length(0, 8192)), - "Type": validation.Validate(o.Type, validation.Required, validation.In(Object).Error( - fmt.Sprintf("value '%s' is invalid. Must be: 'object'", (&o).Type))), - }.Filter() -} - // Validate validates UpdatePolicyVersionRequest func (o UpdatePolicyVersionRequest) Validate() error { errs := validation.Errors{ @@ -419,176 +155,8 @@ var ( ErrDeletePolicyVersion = errors.New("delete policy versions") // ErrUpdatePolicyVersion is returned when UpdatePolicyVersion fails ErrUpdatePolicyVersion = errors.New("update policy versions") - // ErrUnmarshallMatchCriteriaALB is returned when unmarshalling of MatchCriteriaALB fails - ErrUnmarshallMatchCriteriaALB = errors.New("unmarshalling MatchCriteriaALB") - // ErrUnmarshallMatchCriteriaER is returned when unmarshalling of MatchCriteriaER fails - ErrUnmarshallMatchCriteriaER = errors.New("unmarshalling MatchCriteriaER") - // ErrUnmarshallMatchRules is returned when unmarshalling of MatchRules fails - ErrUnmarshallMatchRules = errors.New("unmarshalling MatchRules") ) -func (m MatchRuleALB) cloudletType() string { - return "albMatchRule" -} - -func (m MatchRuleER) cloudletType() string { - return "erMatchRule" -} - -// matchRuleHandlers contains mapping between name of the type for MatchRule and its implementation -// It makes the UnmarshalJSON more compact and easier to support more cloudlet types -var matchRuleHandlers = map[string]func() MatchRule{ - "albMatchRule": func() MatchRule { return &MatchRuleALB{} }, - "erMatchRule": func() MatchRule { return &MatchRuleER{} }, -} - -// UnmarshalJSON helps to un-marshall items of MatchRules array as proper instances of *MatchRuleALB or *MatchRuleER -func (m *MatchRules) UnmarshalJSON(b []byte) error { - data := make([]map[string]interface{}, 0) - if err := json.Unmarshal(b, &data); err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) - } - for _, matchRule := range data { - cloudletType, ok := matchRule["type"] - if !ok { - return fmt.Errorf("%w: match rule entry should contain 'type' field", ErrUnmarshallMatchRules) - } - cloudletTypeName, ok := cloudletType.(string) - if !ok { - return fmt.Errorf("%w: 'type' field on match rule entry should be a string", ErrUnmarshallMatchRules) - } - byteArr, err := json.Marshal(matchRule) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) - } - - matchRuleType, ok := matchRuleHandlers[cloudletTypeName] - if !ok { - return fmt.Errorf("%w: unsupported match rule type: %s", ErrUnmarshallMatchRules, cloudletTypeName) - } - dst := matchRuleType() - err = json.Unmarshal(byteArr, dst) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchRules, err) - } - *m = append(*m, dst) - } - return nil -} - -// objectALBMatchValueHandlers contains mapping between name of the type for ObjectMatchValue and its implementation -// It makes the UnmarshalJSON more compact and easier to support more types -var objectALBMatchValueHandlers = map[string]func() interface{}{ - "object": func() interface{} { return &ObjectMatchValueObject{} }, - "range": func() interface{} { return &ObjectMatchValueRange{} }, - "simple": func() interface{} { return &ObjectMatchValueSimple{} }, -} - -// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaALB as proper instance of *ObjectMatchValueObject, *ObjectMatchValueSimple or *ObjectMatchValueRange -func (m *MatchCriteriaALB) UnmarshalJSON(b []byte) error { - // matchCriteriaALB is an alias for MatchCriteriaALB for un-marshalling purposes - type matchCriteriaALB MatchCriteriaALB - - // populate common attributes using default json unmarshaler using aliased type - err := json.Unmarshal(b, (*matchCriteriaALB)(m)) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) - } - if m.ObjectMatchValue == nil { - return nil - } - objectMatchValue, ok := m.ObjectMatchValue.(interface{}) - if !ok { - return fmt.Errorf("%w: objectMatchValue should be of type 'interface{}', but was '%T'", ErrUnmarshallMatchCriteriaALB, m.ObjectMatchValue) - } - - objectMatchValueMap, ok := objectMatchValue.(map[string]interface{}) - if !ok { - return fmt.Errorf("%w: structure of objectMatchValue should be 'map', but was '%T'", ErrUnmarshallMatchCriteriaALB, objectMatchValue) - } - objectMatchValueType, ok := objectMatchValueMap["type"] - if !ok { - return fmt.Errorf("%w: objectMatchValue should contain 'type' field", ErrUnmarshallMatchCriteriaALB) - } - objectMatchValueTypeName, ok := objectMatchValueType.(string) - if !ok { - return fmt.Errorf("%w: 'type' should be a string", ErrUnmarshallMatchCriteriaALB) - } - - createObjectMatchValue, ok := objectALBMatchValueHandlers[objectMatchValueTypeName] - if !ok { - return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaALB, objectMatchValueTypeName) - } - convertedObjectMatchValue := createObjectMatchValue() - marshal, err := json.Marshal(objectMatchValue) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) - } - err = json.Unmarshal(marshal, convertedObjectMatchValue) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaALB, err) - } - m.ObjectMatchValue = convertedObjectMatchValue - - return nil -} - -// objectERMatchValueHandlers contains mapping between name of the type for ObjectMatchValue and its implementation -// It makes the UnmarshalJSON more compact and easier to support more types -var objectERMatchValueHandlers = map[string]func() interface{}{ - "object": func() interface{} { return &ObjectMatchValueObject{} }, - "simple": func() interface{} { return &ObjectMatchValueSimple{} }, -} - -// UnmarshalJSON helps to un-marshall field ObjectMatchValue of MatchCriteriaER as proper instance of *ObjectMatchValueObject or *ObjectMatchValueSimple -func (m *MatchCriteriaER) UnmarshalJSON(b []byte) error { - // matchCriteriaER is an alias for MatchCriteriaER for un-marshalling purposes - type matchCriteriaER MatchCriteriaER - - // populate common attributes using default json unmarshaler using aliased type - err := json.Unmarshal(b, (*matchCriteriaER)(m)) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) - } - if m.ObjectMatchValue == nil { - return nil - } - objectMatchValue, ok := m.ObjectMatchValue.(interface{}) - if !ok { - return fmt.Errorf("%w: objectMatchValue should be of type 'interface{}', but was '%T'", ErrUnmarshallMatchCriteriaER, m.ObjectMatchValue) - } - - objectMatchValueMap, ok := objectMatchValue.(map[string]interface{}) - if !ok { - return fmt.Errorf("%w: structure of objectMatchValue should be 'map', but was '%T'", ErrUnmarshallMatchCriteriaER, objectMatchValue) - } - objectMatchValueType, ok := objectMatchValueMap["type"] - if !ok { - return fmt.Errorf("%w: objectMatchValue should contain 'type' field", ErrUnmarshallMatchCriteriaER) - } - objectMatchValueTypeName, ok := objectMatchValueType.(string) - if !ok { - return fmt.Errorf("%w: 'type' should be a string", ErrUnmarshallMatchCriteriaER) - } - - createObjectMatchValue, ok := objectERMatchValueHandlers[objectMatchValueTypeName] - if !ok { - return fmt.Errorf("%w: objectMatchValue has unexpected type: '%s'", ErrUnmarshallMatchCriteriaER, objectMatchValueTypeName) - } - convertedObjectMatchValue := createObjectMatchValue() - marshal, err := json.Marshal(objectMatchValue) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) - } - err = json.Unmarshal(marshal, convertedObjectMatchValue) - if err != nil { - return fmt.Errorf("%w: %s", ErrUnmarshallMatchCriteriaER, err) - } - m.ObjectMatchValue = convertedObjectMatchValue - - return nil -} - func (c *cloudlets) ListPolicyVersions(ctx context.Context, params ListPolicyVersionsRequest) ([]PolicyVersion, error) { logger := c.Log(ctx) logger.Debug("ListPolicyVersions") diff --git a/pkg/cloudlets/policy_version_activation.go b/pkg/cloudlets/policy_version_activation.go index b806d4b1..37bd9119 100644 --- a/pkg/cloudlets/policy_version_activation.go +++ b/pkg/cloudlets/policy_version_activation.go @@ -87,8 +87,9 @@ func (r ActivatePolicyVersionRequest) Validate() error { "RequestBody.AdditionalPropertyNames": validation.Validate(r.PolicyVersionActivation.AdditionalPropertyNames, validation.Required), "RequestBody.Network": validation.Validate( r.PolicyVersionActivation.Network, + validation.Required, validation.In(PolicyActivationNetworkStaging, PolicyActivationNetworkProduction).Error( - fmt.Sprintf("value '%s' is invalid. Must be one of: 'staging', 'prod' or '' (empty)", (&r).PolicyVersionActivation.Network)), + fmt.Sprintf("value '%s' is invalid. Must be one of: 'staging' or 'prod'", (&r).PolicyVersionActivation.Network)), ), } return edgegriderr.ParseValidationErrors(errs) diff --git a/pkg/cloudlets/policy_version_activation_test.go b/pkg/cloudlets/policy_version_activation_test.go index 4b023edd..14fce531 100644 --- a/pkg/cloudlets/policy_version_activation_test.go +++ b/pkg/cloudlets/policy_version_activation_test.go @@ -5,7 +5,6 @@ import ( "errors" "net/http" "net/http/httptest" - "regexp" "testing" "github.com/stretchr/testify/assert" @@ -154,7 +153,7 @@ func TestActivatePolicyVersion(t *testing.T) { uri string responseBody string expectedActivation PolicyVersionActivation - withError *regexp.Regexp + withError func(*testing.T, error) }{ "200 Policy version activation": { responseStatus: http.StatusOK, @@ -182,7 +181,9 @@ func TestActivatePolicyVersion(t *testing.T) { }, "any request validation error": { parameters: ActivatePolicyVersionRequest{}, - withError: regexp.MustCompile(ErrStructValidation.Error()), + withError: func(t *testing.T, err error) { + assert.True(t, errors.Is(err, ErrStructValidation), "want: %s; got: %s", ErrStructValidation, err) + }, }, "any kind of server error": { responseStatus: http.StatusInternalServerError, @@ -194,7 +195,9 @@ func TestActivatePolicyVersion(t *testing.T) { AdditionalPropertyNames: []string{"www.rc-cloudlet.com"}, }, }, - withError: regexp.MustCompile(ErrActivatePolicyVersion.Error()), + withError: func(t *testing.T, err error) { + assert.True(t, errors.Is(err, ErrActivatePolicyVersion), "want: %s; got: %s", ErrActivatePolicyVersion, err) + }, }, "property name not existing": { responseStatus: http.StatusBadRequest, @@ -206,7 +209,10 @@ func TestActivatePolicyVersion(t *testing.T) { AdditionalPropertyNames: []string{"www.rc-cloudlet.com"}, }, }, - withError: regexp.MustCompile(`"Requested propertyName \\"XYZ\\" does not exist"`), + withError: func(t *testing.T, err error) { + assert.Containsf(t, err.Error(), "Requested propertyName \\\"XYZ\\\" does not exist", "want: %s; got: %s", ErrActivatePolicyVersion, err) + assert.True(t, errors.Is(err, ErrActivatePolicyVersion), "want: %s; got: %s", ErrActivatePolicyVersion, err) + }, responseBody: ` { "detail": "Requested propertyName \"XYZ\" does not exist", @@ -217,17 +223,21 @@ func TestActivatePolicyVersion(t *testing.T) { } `, }, - "empty property names": { + "validation errors": { responseStatus: http.StatusBadRequest, parameters: ActivatePolicyVersionRequest{ PolicyID: 1234, Version: 1, PolicyVersionActivation: PolicyVersionActivation{ - Network: PolicyActivationNetworkStaging, + Network: "", AdditionalPropertyNames: []string{}, }, }, - withError: regexp.MustCompile("struct validation:\nRequestBody.AdditionalPropertyNames: cannot be blank"), + withError: func(t *testing.T, err error) { + assert.Containsf(t, err.Error(), "RequestBody.Network: cannot be blank", "want: %s; got: %s", ErrStructValidation, err) + assert.Containsf(t, err.Error(), "RequestBody.AdditionalPropertyNames: cannot be blank", "want: %s; got: %s", ErrStructValidation, err) + assert.True(t, errors.Is(err, ErrStructValidation), "want: %s; got: %s", ErrStructValidation, err) + }, }, } @@ -242,7 +252,7 @@ func TestActivatePolicyVersion(t *testing.T) { client := mockAPIClient(t, mockServer) err := client.ActivatePolicyVersion(context.Background(), test.parameters) if test.withError != nil { - assert.True(t, test.withError.MatchString(err.Error()), "want: %s; got: %s", test.withError, err) + test.withError(t, err) return } require.NoError(t, err) diff --git a/pkg/cloudlets/policy_version_test.go b/pkg/cloudlets/policy_version_test.go index fe60e88c..dba943c7 100644 --- a/pkg/cloudlets/policy_version_test.go +++ b/pkg/cloudlets/policy_version_test.go @@ -1,8 +1,8 @@ package cloudlets import ( + "bytes" "context" - "encoding/json" "errors" "net/http" "net/http/httptest" @@ -344,6 +344,99 @@ func TestGetPolicyVersion(t *testing.T) { }, }, }, + "200 OK, PR rule": { + request: GetPolicyVersionRequest{ + PolicyID: 276858, + Version: 6, + }, + responseStatus: http.StatusOK, + responseBody: `{ + "activations": [], + "createDate": 1638547203265, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1638547203265, + "location": "/cloudlets/api/v2/policies/325401/versions/3", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "cdMatchRule", + "akaRuleId": "b151ca68e51f5a61", + "end": 0, + "forwardSettings": { + "originId": "fr_test_krk_dc2", + "percent": 11 + }, + "id": 0, + "location": "/cloudlets/api/v2/policies/325401/versions/3/rules/b151ca68e51f5a61", + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "rule 1", + "start": 0 + } + ], + "policyId": 325401, + "revisionId": 4888857, + "rulesLocked": false, + "version": 3 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions/6?omitRules=false", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1638547203265, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1638547203265, + Location: "/cloudlets/api/v2/policies/325401/versions/3", + MatchRuleFormat: "1.0", + MatchRules: MatchRules{ + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "fr_test_krk_dc2", + Percent: 11, + }, + ID: 0, + Matches: []MatchCriteriaPR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{ + "GET"}, + }, + }, + }, + Name: "rule 1", + Start: 0, + }, + }, + PolicyID: 325401, + RevisionID: 4888857, + RulesLocked: false, + Version: 3, + }, + }, "200 OK, ER rule with disabled=false": { request: GetPolicyVersionRequest{ PolicyID: 276858, @@ -472,7 +565,74 @@ func TestGetPolicyVersion(t *testing.T) { }, }, }, - + "200 OK, FR with disabled rule": { + request: GetPolicyVersionRequest{ + PolicyID: 276858, + Version: 6, + }, + responseStatus: http.StatusOK, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [{ + "type": "frMatchRule", + "disabled": true, + "end": 0, + "id": 0, + "matchURL": null, + "forwardSettings": { + "pathAndQS": "/test_images/simpleimg.jpg", + "useIncomingQueryString": true, + "originId": "1234" + }, + "name": "rule 1", + "start": 0 + }], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions/6?omitRules=false", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleFR{ + Name: "rule 1", + Type: "frMatchRule", + Start: 0, + End: 0, + ID: 0, + MatchURL: "", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/simpleimg.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + Disabled: true, + }, + }, + }, + }, "500 internal server error": { request: GetPolicyVersionRequest{ PolicyID: 1, @@ -522,6 +682,7 @@ func TestGetPolicyVersion(t *testing.T) { func TestCreatePolicyVersion(t *testing.T) { tests := map[string]struct { request CreatePolicyVersionRequest + requestBody string responseStatus int responseBody string expectedPath string @@ -597,7 +758,7 @@ func TestCreatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "Rule3", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_dc1_only", }, ID: 0, @@ -617,7 +778,7 @@ func TestCreatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "Rule1", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_0_100", }, ID: 0, @@ -707,7 +868,7 @@ func TestCreatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_dc1_only", }, ID: 0, @@ -735,7 +896,7 @@ func TestCreatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_0_100", }, ID: 0, @@ -759,7 +920,6 @@ func TestCreatePolicyVersion(t *testing.T) { Version: 2, }, }, - "201 created, complex ALB with objectMatchValue - object": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ @@ -786,7 +946,7 @@ func TestCreatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "alb rule", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -860,7 +1020,7 @@ func TestCreatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -892,7 +1052,6 @@ func TestCreatePolicyVersion(t *testing.T) { Version: 796, }, }, - "201 created, complex ALB with objectMatchValue - simple": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ @@ -917,7 +1076,7 @@ func TestCreatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "alb rule", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -987,7 +1146,7 @@ func TestCreatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -1015,7 +1174,6 @@ func TestCreatePolicyVersion(t *testing.T) { Version: 797, }, }, - "201 created, complex ALB with objectMatchValue - range": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ @@ -1040,7 +1198,7 @@ func TestCreatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "alb rule", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -1111,7 +1269,7 @@ func TestCreatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_mutable", }, ID: 0, @@ -1140,20 +1298,21 @@ func TestCreatePolicyVersion(t *testing.T) { }, }, - "201 created, complex ER": { + "201 created, complex PR": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ MatchRules: MatchRules{ - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - UseRelativeURL: "copy_scheme_hostname", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rul3", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { MatchType: "hostname", MatchValue: "3333.dom", @@ -1177,30 +1336,29 @@ func TestCreatePolicyVersion(t *testing.T) { }, }, }, - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - UseRelativeURL: "none", - Name: "rule 2", - MatchURL: "ddd.aaa", - RedirectURL: "sss.com", - StatusCode: 301, - UseIncomingQueryString: true, - ID: 0, + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rule 2", + MatchURL: "ddd.aaa", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, }, - &MatchRuleER{ - Type: "erMatchRule", - ID: 0, - Name: "r1", - Start: 0, - End: 0, - MatchURL: "abc.com", - StatusCode: 301, - RedirectURL: "/ddd", - UseIncomingQueryString: false, - UseIncomingSchemeAndHost: true, - UseRelativeURL: "copy_scheme_hostname", + &MatchRulePR{ + Type: "cdMatchRule", + ID: 0, + Name: "r1", + Start: 0, + End: 0, + MatchURL: "abc.com", + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, }, }, }, @@ -1219,7 +1377,7 @@ func TestCreatePolicyVersion(t *testing.T) { "matchRuleFormat": "1.0", "matchRules": [ { - "type": "erMatchRule", + "type": "cdMatchRule", "end": 0, "id": 0, "matchURL": null, @@ -1249,13 +1407,13 @@ func TestCreatePolicyVersion(t *testing.T) { "name": "rul3", "redirectURL": "/abc/sss", "start": 0, - "statusCode": 307, - "useIncomingQueryString": false, - "useIncomingSchemeAndHost": true, - "useRelativeUrl": "copy_scheme_hostname" + "forwardSettings": { + "originId": "some_origin", + "percent": 10 + } }, { - "type": "erMatchRule", + "type": "cdMatchRule", "end": 0, "id": 0, "matchURL": "ddd.aaa", @@ -1267,7 +1425,7 @@ func TestCreatePolicyVersion(t *testing.T) { "useRelativeUrl": "none" }, { - "type": "erMatchRule", + "type": "cdMatchRule", "end": 0, "id": 0, "matchURL": "abc.com", @@ -1301,19 +1459,18 @@ func TestCreatePolicyVersion(t *testing.T) { RulesLocked: false, Version: 6, MatchRules: MatchRules{ - &MatchRuleER{ - Type: "erMatchRule", - End: 0, - ID: 0, - MatchURL: "", - Name: "rul3", - RedirectURL: "/abc/sss", - Start: 0, - StatusCode: 307, - UseIncomingQueryString: false, - UseIncomingSchemeAndHost: true, - UseRelativeURL: "copy_scheme_hostname", - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + Start: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { MatchType: "hostname", MatchValue: "3333.dom", @@ -1337,49 +1494,40 @@ func TestCreatePolicyVersion(t *testing.T) { }, }, }, - &MatchRuleER{ - Type: "erMatchRule", - End: 0, - ID: 0, - MatchURL: "ddd.aaa", - Name: "rule 2", - RedirectURL: "sss.com", - Start: 0, - StatusCode: 301, - UseIncomingQueryString: true, - UseRelativeURL: "none", + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "ddd.aaa", + Name: "rule 2", + Start: 0, }, - &MatchRuleER{ - Type: "erMatchRule", - End: 0, - ID: 0, - MatchURL: "abc.com", - Name: "r1", - RedirectURL: "/ddd", - Start: 0, - StatusCode: 301, - UseIncomingQueryString: false, - UseIncomingSchemeAndHost: true, - UseRelativeURL: "copy_scheme_hostname", + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "abc.com", + Name: "r1", + Start: 0, }, }, }, }, - - "201 created, complex ER with objectMatchValue - simple": { + "201 created, complex PR with objectMatchValue - simple": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ MatchRules: MatchRules{ - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - UseRelativeURL: "copy_scheme_hostname", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rul3", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { CaseSensitive: true, MatchOperator: "equals", @@ -1396,6 +1544,7 @@ func TestCreatePolicyVersion(t *testing.T) { }, PolicyID: 276858, }, + requestBody: `{"matchRules":[{"name":"rul3","type":"cdMatchRule","matches":[{"matchType":"method","matchOperator":"equals","caseSensitive":true,"negate":false,"objectMatchValue":{"type":"simple","value":["GET"]}}],"forwardSettings":{"originId":"some_origin","percent":10}}]}`, responseStatus: http.StatusCreated, responseBody: `{ "activations": [], @@ -1409,7 +1558,7 @@ func TestCreatePolicyVersion(t *testing.T) { "matchRuleFormat": "1.0", "matchRules": [ { - "type": "erMatchRule", + "type": "cdMatchRule", "end": 0, "id": 0, "matchURL": null, @@ -1430,10 +1579,10 @@ func TestCreatePolicyVersion(t *testing.T) { "name": "rul3", "redirectURL": "/abc/sss", "start": 0, - "statusCode": 307, - "useIncomingQueryString": false, - "useIncomingSchemeAndHost": true, - "useRelativeUrl": "copy_scheme_hostname" + "forwardSettings": { + "originId": "some_origin", + "percent": 10 + } } ], "policyId": 276858, @@ -1457,19 +1606,18 @@ func TestCreatePolicyVersion(t *testing.T) { RulesLocked: false, Version: 6, MatchRules: MatchRules{ - &MatchRuleER{ - Type: "erMatchRule", - End: 0, - ID: 0, - MatchURL: "", - Name: "rul3", - RedirectURL: "/abc/sss", - Start: 0, - StatusCode: 307, - UseIncomingQueryString: false, - UseIncomingSchemeAndHost: true, - UseRelativeURL: "copy_scheme_hostname", - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + Start: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { CaseSensitive: true, MatchOperator: "equals", @@ -1485,28 +1633,28 @@ func TestCreatePolicyVersion(t *testing.T) { }, }, }, - - "201 created, complex ER with objectMatchValue - object": { + "201 created, complex PR with objectMatchValue - object": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ MatchRules: MatchRules{ - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - UseRelativeURL: "copy_scheme_hostname", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rul3", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { MatchOperator: "equals", MatchType: "header", Negate: false, ObjectMatchValue: &ObjectMatchValueObject{ Type: "object", - Name: "ER", + Name: "PR", Options: &Options{ Value: []string{ "text/html*", @@ -1536,7 +1684,7 @@ func TestCreatePolicyVersion(t *testing.T) { "matchRuleFormat": "1.0", "matchRules": [ { - "type": "erMatchRule", + "type": "cdMatchRule", "end": 0, "id": 0, "matchURL": null, @@ -1547,7 +1695,7 @@ func TestCreatePolicyVersion(t *testing.T) { "negate": false, "objectMatchValue": { "type": "object", - "name": "ER", + "name": "PR", "options": { "value": [ "text/html*", @@ -1562,10 +1710,10 @@ func TestCreatePolicyVersion(t *testing.T) { "name": "rul3", "redirectURL": "/abc/sss", "start": 0, - "statusCode": 307, - "useIncomingQueryString": false, - "useIncomingSchemeAndHost": true, - "useRelativeUrl": "copy_scheme_hostname" + "forwardSettings": { + "originId": "some_origin", + "percent": 10 + } } ], "policyId": 276858, @@ -1589,26 +1737,25 @@ func TestCreatePolicyVersion(t *testing.T) { RulesLocked: false, Version: 6, MatchRules: MatchRules{ - &MatchRuleER{ - Type: "erMatchRule", - End: 0, - ID: 0, - MatchURL: "", - Name: "rul3", - RedirectURL: "/abc/sss", - Start: 0, - StatusCode: 307, - UseIncomingQueryString: false, - UseIncomingSchemeAndHost: true, - UseRelativeURL: "copy_scheme_hostname", - Matches: []MatchCriteriaER{ + &MatchRulePR{ + Type: "cdMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + Start: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ { MatchOperator: "equals", MatchType: "hostname", Negate: false, ObjectMatchValue: &ObjectMatchValueObject{ Type: "object", - Name: "ER", + Name: "PR", Options: &Options{ Value: []string{ "text/html*", @@ -1624,15 +1771,145 @@ func TestCreatePolicyVersion(t *testing.T) { }, }, }, + "validation error, complex PR with unavailable objectMatchValue type - range": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rul3", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ + { + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + "validation error, complex PR missing forwardSettings": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rul3", + ID: 0, + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + Matches: []MatchCriteriaPR{ + { + MatchType: "hostname", + MatchValue: "3333.dom", + MatchOperator: "equals", + CaseSensitive: true, + Negate: false, + }, + { + MatchType: "cookie", + MatchValue: "cookie=cookievalue", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + { + MatchType: "extension", + MatchValue: "txt", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + }, + }, + &MatchRulePR{ + Start: 0, + End: 0, + Type: "cdMatchRule", + Name: "rule 2", + MatchURL: "ddd.aaa", + ID: 0, + }, + &MatchRulePR{ + Type: "cdMatchRule", + ID: 0, + Name: "r1", + Start: 0, + End: 0, + MatchURL: "abc.com", + ForwardSettings: ForwardSettingsPR{ + OriginID: "some_origin", + Percent: 10, + }, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, - "201 created, ER with empty/no useRelativeURL": { + "201 created, complex ER": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ MatchRules: MatchRules{ + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + UseRelativeURL: "copy_scheme_hostname", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + Matches: []MatchCriteriaER{ + { + MatchType: "hostname", + MatchValue: "3333.dom", + MatchOperator: "equals", + CaseSensitive: true, + Negate: false, + }, + { + MatchType: "cookie", + MatchValue: "cookie=cookievalue", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + { + MatchType: "extension", + MatchValue: "txt", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + }, + }, &MatchRuleER{ Start: 0, End: 0, Type: "erMatchRule", + UseRelativeURL: "none", Name: "rule 2", MatchURL: "ddd.aaa", RedirectURL: "sss.com", @@ -1651,16 +1928,7 @@ func TestCreatePolicyVersion(t *testing.T) { RedirectURL: "/ddd", UseIncomingQueryString: false, UseIncomingSchemeAndHost: true, - UseRelativeURL: "", - }, - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, + UseRelativeURL: "copy_scheme_hostname", }, }, }, @@ -1678,6 +1946,42 @@ func TestCreatePolicyVersion(t *testing.T) { "location": "/cloudlets/api/v2/policies/276858/versions/6", "matchRuleFormat": "1.0", "matchRules": [ + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": true, + "matchOperator": "equals", + "matchType": "hostname", + "matchValue": "3333.dom", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "cookie", + "matchValue": "cookie=cookievalue", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "extension", + "matchValue": "txt", + "negate": false + } + ], + "name": "rul3", + "redirectURL": "/abc/sss", + "start": 0, + "statusCode": 307, + "useIncomingQueryString": false, + "useIncomingSchemeAndHost": true, + "useRelativeUrl": "copy_scheme_hostname" + }, { "type": "erMatchRule", "end": 0, @@ -1687,7 +1991,8 @@ func TestCreatePolicyVersion(t *testing.T) { "redirectURL": "sss.com", "start": 0, "statusCode": 301, - "useIncomingQueryString": true + "useIncomingQueryString": true, + "useRelativeUrl": "none" }, { "type": "erMatchRule", @@ -1701,16 +2006,7 @@ func TestCreatePolicyVersion(t *testing.T) { "useIncomingQueryString": false, "useIncomingSchemeAndHost": true, "useRelativeUrl": "copy_scheme_hostname" - }, - { - "type": "erMatchRule", - "end": 0, - "id": 0, - "name": "rul3", - "redirectURL": "/abc/sss", - "start": 0, - "statusCode": 307 - } + } ], "policyId": 276858, "revisionId": 4815968, @@ -1733,6 +2029,42 @@ func TestCreatePolicyVersion(t *testing.T) { RulesLocked: false, Version: 6, MatchRules: MatchRules{ + &MatchRuleER{ + Type: "erMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + RedirectURL: "/abc/sss", + Start: 0, + StatusCode: 307, + UseIncomingQueryString: false, + UseIncomingSchemeAndHost: true, + UseRelativeURL: "copy_scheme_hostname", + Matches: []MatchCriteriaER{ + { + MatchType: "hostname", + MatchValue: "3333.dom", + MatchOperator: "equals", + CaseSensitive: true, + Negate: false, + }, + { + MatchType: "cookie", + MatchValue: "cookie=cookievalue", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + { + MatchType: "extension", + MatchValue: "txt", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + }, + }, &MatchRuleER{ Type: "erMatchRule", End: 0, @@ -1743,6 +2075,7 @@ func TestCreatePolicyVersion(t *testing.T) { Start: 0, StatusCode: 301, UseIncomingQueryString: true, + UseRelativeURL: "none", }, &MatchRuleER{ Type: "erMatchRule", @@ -1757,43 +2090,1401 @@ func TestCreatePolicyVersion(t *testing.T) { UseIncomingSchemeAndHost: true, UseRelativeURL: "copy_scheme_hostname", }, - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, + }, + }, + }, + "201 created, complex ER with objectMatchValue - simple": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + UseRelativeURL: "copy_scheme_hostname", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + Matches: []MatchCriteriaER{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": true, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "rul3", + "redirectURL": "/abc/sss", + "start": 0, + "statusCode": 307, + "useIncomingQueryString": false, + "useIncomingSchemeAndHost": true, + "useRelativeUrl": "copy_scheme_hostname" + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleER{ + Type: "erMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + RedirectURL: "/abc/sss", + Start: 0, + StatusCode: 307, + UseIncomingQueryString: false, + UseIncomingSchemeAndHost: true, + UseRelativeURL: "copy_scheme_hostname", + Matches: []MatchCriteriaER{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + }, + "201 created, complex ER with objectMatchValue - object": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + UseRelativeURL: "copy_scheme_hostname", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + Matches: []MatchCriteriaER{ + { + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Type: "object", + Name: "ER", + Options: &Options{ + Value: []string{ + "text/html*", + "text/css*", + "application/x-javascript*", + }, + ValueHasWildcard: true, + }, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "matchOperator": "equals", + "matchType": "hostname", + "negate": false, + "objectMatchValue": { + "type": "object", + "name": "ER", + "options": { + "value": [ + "text/html*", + "text/css*", + "application/x-javascript*" + ], + "valueHasWildcard": true + } + } + } + ], + "name": "rul3", + "redirectURL": "/abc/sss", + "start": 0, + "statusCode": 307, + "useIncomingQueryString": false, + "useIncomingSchemeAndHost": true, + "useRelativeUrl": "copy_scheme_hostname" + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleER{ + Type: "erMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + RedirectURL: "/abc/sss", + Start: 0, + StatusCode: 307, + UseIncomingQueryString: false, + UseIncomingSchemeAndHost: true, + UseRelativeURL: "copy_scheme_hostname", + Matches: []MatchCriteriaER{ + { + MatchOperator: "equals", + MatchType: "hostname", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Type: "object", + Name: "ER", + Options: &Options{ + Value: []string{ + "text/html*", + "text/css*", + "application/x-javascript*", + }, + ValueHasWildcard: true, + }, + }, + }, + }, + }, + }, + }, + }, + "201 created, ER with empty/no useRelativeURL": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + Name: "rule 2", + MatchURL: "ddd.aaa", + RedirectURL: "sss.com", + StatusCode: 301, + UseIncomingQueryString: true, + ID: 0, + }, + &MatchRuleER{ + Type: "erMatchRule", + ID: 0, + Name: "r1", + Start: 0, + End: 0, + MatchURL: "abc.com", + StatusCode: 301, + RedirectURL: "/ddd", + UseIncomingQueryString: false, + UseIncomingSchemeAndHost: true, + UseRelativeURL: "", + }, + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "matchURL": "ddd.aaa", + "name": "rule 2", + "redirectURL": "sss.com", + "start": 0, + "statusCode": 301, + "useIncomingQueryString": true + }, + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "matchURL": "abc.com", + "name": "r1", + "redirectURL": "/ddd", + "start": 0, + "statusCode": 301, + "useIncomingQueryString": false, + "useIncomingSchemeAndHost": true, + "useRelativeUrl": "copy_scheme_hostname" + }, + { + "type": "erMatchRule", + "end": 0, + "id": 0, + "name": "rul3", + "redirectURL": "/abc/sss", + "start": 0, + "statusCode": 307 + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleER{ + Type: "erMatchRule", + End: 0, + ID: 0, + MatchURL: "ddd.aaa", + Name: "rule 2", + RedirectURL: "sss.com", + Start: 0, + StatusCode: 301, + UseIncomingQueryString: true, + }, + &MatchRuleER{ + Type: "erMatchRule", + End: 0, + ID: 0, + MatchURL: "abc.com", + Name: "r1", + RedirectURL: "/ddd", + Start: 0, + StatusCode: 301, + UseIncomingQueryString: false, + UseIncomingSchemeAndHost: true, + UseRelativeURL: "copy_scheme_hostname", + }, + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + }, + }, + }, + }, + "validation error, complex ER with unavailable objectMatchValue type - range": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleER{ + Start: 0, + End: 0, + Type: "erMatchRule", + UseRelativeURL: "copy_scheme_hostname", + Name: "rul3", + StatusCode: 307, + RedirectURL: "/abc/sss", + ID: 0, + Matches: []MatchCriteriaER{ + { + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + + "201 created, complex FR": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleFR{ + Start: 0, + End: 0, + Type: "frMatchRule", + Name: "rul3", + ID: 0, + Matches: []MatchCriteriaFR{ + { + MatchType: "hostname", + MatchValue: "3333.dom", + MatchOperator: "equals", + CaseSensitive: true, + Negate: false, + }, + { + MatchType: "cookie", + MatchValue: "cookie=cookievalue", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + { + MatchType: "extension", + MatchValue: "txt", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + }, + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/simpleimg.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + &MatchRuleFR{ + Name: "rule 1", + Type: "frMatchRule", + Start: 0, + End: 0, + ID: 0, + MatchURL: "ddd.aaa", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/simpleimg.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + &MatchRuleFR{ + Name: "rule 2", + Type: "frMatchRule", + Start: 0, + End: 0, + ID: 0, + MatchURL: "abc.com", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/otherimage.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "frMatchRule", + "akaRuleId": "893947a3d5a85c1b", + "end": 0, + "forwardSettings": { + "pathAndQS": "/test_images/otherimage.jpg", + "useIncomingQueryString": true, + "originId": "1234" + }, + "id": 0, + "location": "/cloudlets/api/v2/policies/276858/versions/1/rules/893947a3d5a85c1b", + "matchURL": null, + "matches": [ + { + "caseSensitive": true, + "matchOperator": "equals", + "matchType": "hostname", + "matchValue": "3333.dom", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "cookie", + "matchValue": "cookie=cookievalue", + "negate": false + }, + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "extension", + "matchValue": "txt", + "negate": false + } + ], + "name": "rul3", + "start": 0 + }, + { + "type": "frMatchRule", + "akaRuleId": "aa379d230efcded0", + "end": 0, + "forwardSettings": { + "pathAndQS": "/test_images/simpleimg.jpg", + "useIncomingQueryString": true, + "originId": "1234" + }, + "id": 0, + "location": "/cloudlets/api/v2/policies/276858/versions/1/rules/aa379d230efcded0", + "matchURL": "ddd.aaa", + "name": "rule 1", + "start": 0 + }, + { + "type": "frMatchRule", + "akaRuleId": "1afe03d843996766", + "end": 0, + "forwardSettings": { + "pathAndQS": "/test_images/otherimage.jpg", + "useIncomingQueryString": true, + "originId": "1234" + }, + "id": 0, + "location": "/cloudlets/api/v2/policies/276858/versions/1/rules/1afe03d843996766", + "matchURL": "abc.com", + "name": "rule 2", + "start": 0 + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleFR{ + Type: "frMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + Start: 0, + Matches: []MatchCriteriaFR{ + { + MatchType: "hostname", + MatchValue: "3333.dom", + MatchOperator: "equals", + CaseSensitive: true, + Negate: false, + }, + { + MatchType: "cookie", + MatchValue: "cookie=cookievalue", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + { + MatchType: "extension", + MatchValue: "txt", + MatchOperator: "equals", + Negate: false, + CaseSensitive: false, + }, + }, + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/otherimage.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + &MatchRuleFR{ + Name: "rule 1", + Type: "frMatchRule", + Start: 0, + End: 0, + ID: 0, + MatchURL: "ddd.aaa", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/simpleimg.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + &MatchRuleFR{ + Name: "rule 2", + Type: "frMatchRule", + Start: 0, + End: 0, + ID: 0, + MatchURL: "abc.com", + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/otherimage.jpg", + UseIncomingQueryString: true, + OriginID: "1234", + }, + }, + }, + }, + }, + "201 created, complex FR with objectMatchValue - object": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + Description: "New version 1630480693371", + MatchRuleFormat: "1.0", + MatchRules: MatchRules{ + &MatchRuleFR{ + ForwardSettings: ForwardSettingsFR{}, + Matches: []MatchCriteriaFR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Type: "object", + Name: "Accept", + NameCaseSensitive: false, + NameHasWildcard: false, + Options: &Options{ + Value: []string{"asd", "qwe"}, + ValueHasWildcard: false, + ValueCaseSensitive: true, + ValueEscaped: false, + }, + }, + }, + }, + Start: 0, + End: 0, + Type: "frMatchRule", + Name: "rul3", + ID: 0, + }, + }, + }, + PolicyID: 139743, + }, + responseStatus: http.StatusCreated, + responseBody: ` +{ + "activations": [], + "createDate": 1630507099511, + "createdBy": "jsmith", + "deleted": false, + "description": "New version 1630480693371", + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1630507099511, + "location": "/cloudlets/api/v2/policies/139743/versions/798", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "frMatchRule", + "akaRuleId": "f2168e71692e6d9f", + "end": 0, + "forwardSettings": {}, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "header", + "negate": false, + "objectMatchValue": { + "type": "object", + "name": "Accept", + "options": { + "value": [ + "asd", + "qwe" + ], + "valueCaseSensitive": true + } + } + } + ], + "name": "rul3", + "start": 0 + } + ], + "policyId": 139743, + "revisionId": 4819450, + "rulesLocked": false, + "version": 798 +}`, + expectedPath: "/cloudlets/api/v2/policies/139743/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1630507099511, + CreatedBy: "jsmith", + Deleted: false, + Description: "New version 1630480693371", + LastModifiedBy: "jsmith", + LastModifiedDate: 1630507099511, + Location: "/cloudlets/api/v2/policies/139743/versions/798", + MatchRuleFormat: "1.0", + MatchRules: MatchRules{ + &MatchRuleFR{ + ForwardSettings: ForwardSettingsFR{}, + Matches: []MatchCriteriaFR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Name: "Accept", + Type: "object", + NameCaseSensitive: false, + NameHasWildcard: false, + Options: &Options{ + Value: []string{"asd", "qwe"}, + ValueHasWildcard: false, + ValueCaseSensitive: true, + ValueEscaped: false, + }, + }, + }, + }, + Start: 0, + End: 0, + Type: "frMatchRule", + Name: "rul3", + ID: 0, + }, + }, + PolicyID: 139743, + RevisionID: 4819450, + RulesLocked: false, + Version: 798, + }, + }, + "201 created, complex FR with objectMatchValue - simple": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + Description: "New version 1630480693371", + MatchRuleFormat: "1.0", + MatchRules: MatchRules{ + &MatchRuleFR{ + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/otherimage.jpg", + UseIncomingQueryString: true, + }, + Matches: []MatchCriteriaFR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Start: 0, + End: 0, + Type: "frMatchRule", + Name: "rul3", + ID: 0, + }, + }, + }, + PolicyID: 139743, + }, + responseStatus: http.StatusCreated, + responseBody: ` +{ + "activations": [], + "createDate": 1630507099511, + "createdBy": "jsmith", + "deleted": false, + "description": "New version 1630480693371", + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1630507099511, + "location": "/cloudlets/api/v2/policies/139743/versions/798", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "frMatchRule", + "akaRuleId": "f2168e71692e6d9f", + "end": 0, + "forwardSettings": { + "pathAndQS": "/test_images/otherimage.jpg", + "useIncomingQueryString": true + }, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "rul3", + "start": 0 + } + ], + "policyId": 139743, + "revisionId": 4819450, + "rulesLocked": false, + "version": 798 +}`, + expectedPath: "/cloudlets/api/v2/policies/139743/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1630507099511, + CreatedBy: "jsmith", + Deleted: false, + Description: "New version 1630480693371", + LastModifiedBy: "jsmith", + LastModifiedDate: 1630507099511, + Location: "/cloudlets/api/v2/policies/139743/versions/798", + MatchRuleFormat: "1.0", + MatchRules: MatchRules{ + &MatchRuleFR{ + ForwardSettings: ForwardSettingsFR{ + PathAndQS: "/test_images/otherimage.jpg", + UseIncomingQueryString: true, + }, + Matches: []MatchCriteriaFR{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + Start: 0, + End: 0, + Type: "frMatchRule", + Name: "rul3", + ID: 0, + }, + }, + PolicyID: 139743, + RevisionID: 4819450, + RulesLocked: false, + Version: 798, + }, + }, + + "201 created, complex VP with objectMatchValue - simple": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleVP{ + Start: 0, + End: 0, + Type: "vpMatchRule", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(-1), + ID: 0, + Matches: []MatchCriteriaVP{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "vpMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": true, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "rul3", + "start": 0, + "passThroughPercent": -1 + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleVP{ + Type: "vpMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(-1), + Start: 0, + Matches: []MatchCriteriaVP{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + }, + + "201 created, complex AP with objectMatchValue - simple": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleAP{ + Start: 0, + End: 0, + Type: "apMatchRule", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(0), + ID: 0, + Matches: []MatchCriteriaAP{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "apMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": true, + "matchOperator": "equals", + "matchType": "method", + "negate": false, + "objectMatchValue": { + "type": "simple", + "value": [ + "GET" + ] + } + } + ], + "name": "rul3", + "start": 0, + "useIncomingQueryString": false, + "passThroughPercent": -1 + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleAP{ + Type: "apMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(-1), + Start: 0, + Matches: []MatchCriteriaAP{ + { + CaseSensitive: true, + MatchOperator: "equals", + MatchType: "method", + Negate: false, + ObjectMatchValue: &ObjectMatchValueSimple{ + Type: "simple", + Value: []string{"GET"}, + }, + }, + }, + }, + }, + }, + }, + "201 created, complex AP with objectMatchValue - object": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleAP{ + Start: 0, + End: 0, + Type: "apMatchRule", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(-1), + ID: 0, + Matches: []MatchCriteriaAP{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Type: "object", + Name: "AP", + Options: &Options{ + Value: []string{"y"}, + ValueHasWildcard: true, + }, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + responseStatus: http.StatusCreated, + responseBody: `{ + "activations": [], + "createDate": 1629981355165, + "createdBy": "jsmith", + "deleted": false, + "description": null, + "lastModifiedBy": "jsmith", + "lastModifiedDate": 1629981355165, + "location": "/cloudlets/api/v2/policies/276858/versions/6", + "matchRuleFormat": "1.0", + "matchRules": [ + { + "type": "apMatchRule", + "end": 0, + "id": 0, + "matchURL": null, + "matches": [ + { + "caseSensitive": false, + "matchOperator": "equals", + "matchType": "header", + "negate": false, + "objectMatchValue": { + "type": "object", + "name": "AP", + "options": { + "value": [ + "y" + ], + "valueHasWildcard": true + } + } + } + ], + "name": "rul3", + "start": 0, + "passThroughPercent": -1 + } + ], + "policyId": 276858, + "revisionId": 4815968, + "rulesLocked": false, + "version": 6 +}`, + expectedPath: "/cloudlets/api/v2/policies/276858/versions", + expectedResponse: &PolicyVersion{ + Activations: []PolicyActivation{}, + CreateDate: 1629981355165, + CreatedBy: "jsmith", + Deleted: false, + Description: "", + LastModifiedBy: "jsmith", + LastModifiedDate: 1629981355165, + Location: "/cloudlets/api/v2/policies/276858/versions/6", + MatchRuleFormat: "1.0", + PolicyID: 276858, + RevisionID: 4815968, + RulesLocked: false, + Version: 6, + MatchRules: MatchRules{ + &MatchRuleAP{ + Type: "apMatchRule", + End: 0, + ID: 0, + MatchURL: "", + Name: "rul3", + PassThroughPercent: tools.Float64Ptr(-1), + Start: 0, + Matches: []MatchCriteriaAP{ + { + CaseSensitive: false, + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueObject{ + Type: "object", + Name: "AP", + Options: &Options{ + Value: []string{"y"}, + ValueHasWildcard: true, + }, + }, + }, + }, + }, + }, + }, + }, + + "validation error, complex VP with unavailable objectMatchValue type - range": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleVP{ + Start: 0, + End: 0, + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(50.50), + Name: "rul3", + ID: 0, + Matches: []MatchCriteriaVP{ + { + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + + "validation error, complex AP with unavailable objectMatchValue type - range": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleAP{ + Start: 0, + End: 0, + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(50.50), + Name: "rul3", + ID: 0, + Matches: []MatchCriteriaAP{ + { + MatchOperator: "equals", + MatchType: "header", + Negate: false, + ObjectMatchValue: &ObjectMatchValueRange{ + Type: "range", + Value: []int64{1, 50}, + }, + }, + }, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + + "validation error, simple VP missing passThroughPercent": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleVP{ + Start: 0, + End: 0, + Type: "vpMatchRule", + Name: "rul3", + ID: 0, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + + "validation error, simple AP missing passThrughPercent": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleAP{ + Start: 0, + End: 0, + Type: "apMatchRule", + Name: "rul3", + ID: 0, + }, + }, + }, + PolicyID: 276858, + }, + withError: ErrStructValidation, + }, + + "validation error, simple VP passThroughPercent out of range": { + request: CreatePolicyVersionRequest{ + CreatePolicyVersion: CreatePolicyVersion{ + MatchRules: MatchRules{ + &MatchRuleVP{ + Start: 0, + End: 0, + Type: "vpMatchRule", + PassThroughPercent: tools.Float64Ptr(101), + Name: "rul3", + ID: 0, + }, }, }, + PolicyID: 276858, }, + withError: ErrStructValidation, }, - "validation error, complex ER with unavailable objectMatchValue type - range": { + "validation error, simple AP passThroughPercent out of range": { request: CreatePolicyVersionRequest{ CreatePolicyVersion: CreatePolicyVersion{ MatchRules: MatchRules{ - &MatchRuleER{ - Start: 0, - End: 0, - Type: "erMatchRule", - UseRelativeURL: "copy_scheme_hostname", - Name: "rul3", - StatusCode: 307, - RedirectURL: "/abc/sss", - ID: 0, - Matches: []MatchCriteriaER{ - { - MatchOperator: "equals", - MatchType: "header", - Negate: false, - ObjectMatchValue: &ObjectMatchValueRange{ - Type: "range", - Value: []int64{1, 50}, - }, - }, - }, + &MatchRuleAP{ + Start: 0, + End: 0, + Type: "apMatchRule", + PassThroughPercent: tools.Float64Ptr(101), + Name: "rul3", + ID: 0, }, }, }, @@ -1838,6 +3529,13 @@ func TestCreatePolicyVersion(t *testing.T) { mockServer := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { assert.Equal(t, test.expectedPath, r.URL.String()) assert.Equal(t, http.MethodPost, r.Method) + if test.requestBody != "" { + buf := new(bytes.Buffer) + _, err := buf.ReadFrom(r.Body) + assert.NoError(t, err) + req := buf.String() + assert.Equal(t, test.requestBody, req) + } w.WriteHeader(test.responseStatus) _, err := w.Write([]byte(test.responseBody)) assert.NoError(t, err) @@ -1918,6 +3616,7 @@ func TestDeletePolicyVersion(t *testing.T) { func TestUpdatePolicyVersion(t *testing.T) { tests := map[string]struct { request UpdatePolicyVersionRequest + requestBody string responseStatus int responseBody string expectedPath string @@ -1988,7 +3687,7 @@ func TestUpdatePolicyVersion(t *testing.T) { End: 2, Type: "albMatchRule", Name: "Rule3", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_dc1_only", }, ID: 0, @@ -2007,7 +3706,7 @@ func TestUpdatePolicyVersion(t *testing.T) { End: 0, Type: "albMatchRule", Name: "Rule1", - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_0_100", }, ID: 0, @@ -2106,7 +3805,7 @@ func TestUpdatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 2, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_dc1_only", }, ID: 10, @@ -2126,7 +3825,7 @@ func TestUpdatePolicyVersion(t *testing.T) { &MatchRuleALB{ Type: "albMatchRule", End: 0, - ForwardSettings: ForwardSettings{ + ForwardSettings: ForwardSettingsALB{ OriginID: "alb_test_krk_0_100", }, ID: 0, @@ -2217,292 +3916,3 @@ func TestUpdatePolicyVersion(t *testing.T) { }) } } - -func TestUnmarshalJSONMatchRules(t *testing.T) { - tests := map[string]struct { - withError error - responseBody string - expectedObject MatchRules - }{ - "valid MarchRuleALB": { - responseBody: ` - [ - { - "type": "albMatchRule", - "end": 0, - "forwardSettings": { - "originId": "alb_test_krk_dc1_only" - }, - "id": 0, - "matchURL": null, - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "protocol", - "matchValue": "https", - "negate": false - }, - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "range", - "negate": false, - "objectMatchValue": { - "type": "range", - "value": [ - 1, - 50 - ] - } - }, - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": { - "type": "simple", - "value": [ - "GET" - ] - } - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - expectedObject: MatchRules{ - &MatchRuleALB{ - Type: "albMatchRule", - End: 0, - ForwardSettings: ForwardSettings{ - OriginID: "alb_test_krk_dc1_only", - }, - ID: 0, - MatchURL: "", - Matches: []MatchCriteriaALB{ - { - CaseSensitive: false, - MatchOperator: "equals", - MatchType: "protocol", - MatchValue: "https", - Negate: false, - }, - { - CaseSensitive: false, - MatchOperator: "equals", - MatchType: "range", - Negate: false, - ObjectMatchValue: &ObjectMatchValueRange{ - Type: "range", - Value: []int64{1, 50}, - }, - }, - { - CaseSensitive: false, - MatchOperator: "equals", - MatchType: "method", - Negate: false, - ObjectMatchValue: &ObjectMatchValueSimple{ - Type: "simple", - Value: []string{"GET"}, - }, - }, - }, - Name: "Rule3", - Start: 0, - }, - }, - }, - - "invalid objectMatchValue type for ALB - range": { - withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: objectMatchValue has unexpected type: 'foo'"), - responseBody: ` - [ - { - "type": "albMatchRule", - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": { - "type": "foo", - "value": [ - "GET" - ] - } - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - }, - - "wrong type for object value type": { - withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: 'type' should be a string"), - responseBody: ` - [ - { - "type": "albMatchRule", - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": { - "type": 1, - "value": [ - "GET" - ] - } - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - }, - - "missing object value type": { - withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: objectMatchValue should contain 'type' field"), - responseBody: ` - [ - { - "type": "albMatchRule", - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": { - "value": [ - "GET" - ] - } - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - }, - - "invalid object value": { - withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaALB: structure of objectMatchValue should be 'map', but was 'string'"), - responseBody: ` - [ - { - "type": "albMatchRule", - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": "" - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - }, - - "invalid MatchRuleAP": { - responseBody: ` - [ - { - "type": "apMatchRule" - } - ] -`, - withError: errors.New("unmarshalling MatchRules: unsupported match rule type: apMatchRule"), - }, - - "invalid type": { - withError: errors.New("unmarshalling MatchRules: 'type' field on match rule entry should be a string"), - responseBody: ` - [ - { - "type": 1 - } - ] -`, - }, - - "invalid JSON": { - withError: errors.New("unexpected end of JSON input"), - responseBody: ` - [ - { - "type": "albMatchRule" - } - -`, - }, - - "missing type": { - withError: errors.New("unmarshalling MatchRules: match rule entry should contain 'type' field"), - responseBody: ` - [ - { - } - ] -`, - }, - - "invalid objectMatchValue type for ER - range": { - withError: errors.New("unmarshalling MatchRules: unmarshalling MatchCriteriaER: objectMatchValue has unexpected type: 'range'"), - responseBody: ` - [ - { - "type": "erMatchRule", - "matches": [ - { - "caseSensitive": false, - "matchOperator": "equals", - "matchType": "method", - "negate": false, - "objectMatchValue": { - "type": "range", - "value": [ - 1, - 50 - ] - } - } - ], - "name": "Rule3", - "start": 0 - } - ] -`, - }, - } - - for name, test := range tests { - t.Run(name, func(t *testing.T) { - var matchRules MatchRules - err := json.Unmarshal([]byte(test.responseBody), &matchRules) - - if test.withError != nil { - assert.Equal(t, test.withError.Error(), err.Error()) - return - } - require.NoError(t, err) - assert.Equal(t, test.expectedObject, matchRules) - }) - } -} diff --git a/pkg/cloudlets/tools/ptr.go b/pkg/cloudlets/tools/ptr.go index 5584e119..a84c2567 100644 --- a/pkg/cloudlets/tools/ptr.go +++ b/pkg/cloudlets/tools/ptr.go @@ -9,3 +9,8 @@ func IntPtr(i int) *int { func Int64Ptr(i int64) *int64 { return &i } + +// Float64Ptr returns the address of the float64 +func Float64Ptr(f float64) *float64 { + return &f +} diff --git a/pkg/cps/changes.go b/pkg/cps/changes.go index 67349a7c..d737966c 100644 --- a/pkg/cps/changes.go +++ b/pkg/cps/changes.go @@ -7,7 +7,7 @@ import ( "net/http" "net/url" - validation "github.com/go-ozzo/ozzo-validation" + validation "github.com/go-ozzo/ozzo-validation/v4" ) type ( diff --git a/pkg/iam/user.go b/pkg/iam/user.go index b15b27f1..d33e7a58 100644 --- a/pkg/iam/user.go +++ b/pkg/iam/user.go @@ -7,7 +7,7 @@ import ( "net/url" "path" - validation "github.com/go-ozzo/ozzo-validation" + validation "github.com/go-ozzo/ozzo-validation/v4" "github.com/go-ozzo/ozzo-validation/v4/is" "github.com/spf13/cast" ) diff --git a/pkg/networklists/activations.go b/pkg/networklists/activations.go index d35d4418..39056f10 100644 --- a/pkg/networklists/activations.go +++ b/pkg/networklists/activations.go @@ -6,7 +6,7 @@ import ( "net/http" "time" - validation "github.com/go-ozzo/ozzo-validation" + validation "github.com/go-ozzo/ozzo-validation/v4" ) type ( diff --git a/pkg/networklists/network_list.go b/pkg/networklists/network_list.go index 066259a7..6577b8fe 100644 --- a/pkg/networklists/network_list.go +++ b/pkg/networklists/network_list.go @@ -6,7 +6,7 @@ import ( "net/http" "net/url" - validation "github.com/go-ozzo/ozzo-validation" + validation "github.com/go-ozzo/ozzo-validation/v4" ) type ( @@ -43,26 +43,31 @@ type ( // GetNetworkListsResponse contains response from GetNetworkLists method GetNetworkListsResponse struct { - Links *NetworkListsResponseLinks `json:"links,omitempty"` - NetworkLists []struct { - ElementCount int `json:"elementCount"` - Links *NetworkListsLinks `json:"links,omitempty"` - Name string `json:"name"` - NetworkListType string `json:"networkListType"` - ReadOnly bool `json:"readOnly"` - Shared bool `json:"shared"` - SyncPoint int `json:"syncPoint"` - Type string `json:"type"` - UniqueID string `json:"uniqueId"` - AccessControlGroup string `json:"accessControlGroup,omitempty"` - Description string `json:"description,omitempty"` - } `json:"networkLists"` + Links *NetworkListsResponseLinks `json:"links,omitempty"` + NetworkLists []GetNetworkListsResponseListElement `json:"networkLists"` + } + + // GetNetworkListsResponseListElement contains information about a single network list + GetNetworkListsResponseListElement struct { + ElementCount int `json:"elementCount"` + Links *NetworkListsLinks `json:"links,omitempty"` + Name string `json:"name"` + NetworkListType string `json:"networkListType"` + ReadOnly bool `json:"readOnly"` + Shared bool `json:"shared"` + SyncPoint int `json:"syncPoint"` + Type string `json:"type"` + UniqueID string `json:"uniqueId"` + AccessControlGroup string `json:"accessControlGroup,omitempty"` + Description string `json:"description,omitempty"` } // GetNetworkListResponse contains response from GetNetworkList method GetNetworkListResponse struct { Name string `json:"name"` UniqueID string `json:"uniqueId"` + ContractID string `json:"contractId"` + GroupID int `json:"groupId"` SyncPoint int `json:"syncPoint"` Type string `json:"type"` Description string `json:"description,omitempty"` @@ -338,6 +343,7 @@ func (p *networklists) GetNetworkLists(ctx context.Context, params GetNetworkLis return &rval, nil } + rvalfiltered.Links = rval.Links for _, val := range rval.NetworkLists { if (params.Name == "" || params.Name == val.Name) && (params.Type == "" || params.Type == val.Type) { rvalfiltered.NetworkLists = append(rvalfiltered.NetworkLists, val) diff --git a/pkg/networklists/network_list_description.go b/pkg/networklists/network_list_description.go index c3f189ea..ab163656 100644 --- a/pkg/networklists/network_list_description.go +++ b/pkg/networklists/network_list_description.go @@ -5,7 +5,7 @@ import ( "fmt" "net/http" - validation "github.com/go-ozzo/ozzo-validation" + validation "github.com/go-ozzo/ozzo-validation/v4" ) type ( diff --git a/pkg/networklists/network_list_test.go b/pkg/networklists/network_list_test.go index 4411a45d..2ff784ee 100644 --- a/pkg/networklists/network_list_test.go +++ b/pkg/networklists/network_list_test.go @@ -86,6 +86,83 @@ func TestNetworkList_ListNetworkList(t *testing.T) { } } +func TestNetworkList_FilterNetworkLists(t *testing.T) { + + result := GetNetworkListsResponse{} + + respData := compactJSON(loadFixtureBytes("testdata/TestNetworkList/NetworkLists.json")) + json.Unmarshal([]byte(respData), &result) + + expectedResult := GetNetworkListsResponse{} + expectedResponseData := compactJSON(loadFixtureBytes("testdata/TestNetworkList/NetworkLists_GEO.json")) + json.Unmarshal([]byte(expectedResponseData), &expectedResult) + + tests := map[string]struct { + params GetNetworkListsRequest + responseStatus int + responseBody string + expectedPath string + expectedResponse *GetNetworkListsResponse + withError error + headers http.Header + }{ + "200 OK": { + params: GetNetworkListsRequest{Type: "GEO"}, + headers: http.Header{ + "Content-Type": []string{"application/json"}, + }, + responseStatus: http.StatusOK, + responseBody: respData, + expectedPath: "/network-list/v2/network-lists", + expectedResponse: &expectedResult, + }, + "500 internal server error": { + params: GetNetworkListsRequest{}, + headers: http.Header{}, + responseStatus: http.StatusInternalServerError, + responseBody: ` +{ + "type": "internal_error", + "title": "Internal Server Error", + "detail": "Error fetching networklist", + "status": 500 +}`, + expectedPath: "/network-list/v2/network-lists", + withError: &Error{ + Type: "internal_error", + Title: "Internal Server Error", + Detail: "Error fetching networklist", + StatusCode: http.StatusInternalServerError, + }, + }, + } + + for name, test := range tests { + t.Run(name, func(t *testing.T) { + mockServer := httptest.NewTLSServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + assert.Equal(t, test.expectedPath, r.URL.String()) + assert.Equal(t, http.MethodGet, r.Method) + w.WriteHeader(test.responseStatus) + _, err := w.Write([]byte(test.responseBody)) + assert.NoError(t, err) + })) + client := mockAPIClient(t, mockServer) + result, err := client.GetNetworkLists( + session.ContextWithOptions( + context.Background(), + session.WithContextHeaders(test.headers), + ), + test.params) + if test.withError != nil { + assert.True(t, errors.Is(err, test.withError), "want: %s; got: %s", test.withError, err) + return + } + require.NoError(t, err) + assert.Equal(t, test.expectedResponse, result) + }) + } +} + // Test NetworkList func TestNetworkList_GetNetworkList(t *testing.T) { diff --git a/pkg/networklists/testdata/TestNetworkList/NetworkLists_GEO.json b/pkg/networklists/testdata/TestNetworkList/NetworkLists_GEO.json new file mode 100644 index 00000000..affd05e8 --- /dev/null +++ b/pkg/networklists/testdata/TestNetworkList/NetworkLists_GEO.json @@ -0,0 +1,996 @@ +{ + "links": { + "create": { + "href": "/network-list/v2/network-lists", + "method": "POST" + } + }, + "networkLists": [ + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/65654_GEONETWORKLIST", + "method": "PUT" + } + }, + "name": "Geo - Network List", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "65654_GEONETWORKLIST" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/63707_EGPGEOLIST", + "method": "PUT" + } + }, + "name": "EGP Geolist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "63707_EGPGEOLIST" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/55213_GEOBLACKLISTGCO", + "method": "PUT" + } + }, + "name": "Geo Blacklist - GCO", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "55213_GEOBLACKLISTGCO" + }, + { + "accessControlGroup": "ALM - 3-TE78Q.G112653", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/53348_GEOBLACKLISTJENKINS", + "method": "PUT" + } + }, + "name": "GEO Blacklist_Jenkins", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "53348_GEOBLACKLISTJENKINS" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/53076_EXPEDIAGROUPGEOLIST", + "method": "PUT" + } + }, + "name": "Expediagroup GEO list", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "53076_EXPEDIAGROUPGEOLIST" + }, + { + "accessControlGroup": "Car Rentals - 3-TE78Q.G116804", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/52783_APIGEOBLOCK", + "method": "PUT" + } + }, + "name": "API_GEOBlock", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "52783_APIGEOBLOCK" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/52767_GEOBLACKLISTDUO", + "method": "PUT" + } + }, + "name": "Geo Blacklist - Duo", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "52767_GEOBLACKLISTDUO" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/51561_EGENCIAGEOBLOCK", + "method": "PUT" + } + }, + "name": "Egencia GEO Block", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "51561_EGENCIAGEOBLOCK" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 10, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/50337_HOTEISCOMGEOLIST", + "method": "PUT" + } + }, + "name": "hoteis.com GEO list", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 1, + "type": "GEO", + "uniqueId": "50337_HOTEISCOMGEOLIST" + }, + { + "accessControlGroup": "EGENCIA - 3-TE78Q.G122736", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/49997_LEGACYORBITZBLACKLIST", + "method": "PUT" + } + }, + "name": "LegacyOrbitz GEO BlackList", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "49997_LEGACYORBITZBLACKLIST" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/49182_ADTGEOBLACKLIST", + "method": "PUT" + } + }, + "name": "ADT - Geo Blacklist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "49182_ADTGEOBLACKLIST" + }, + { + "accessControlGroup": " Kona Site Defender - C-D5TW8R - C-D5TW8R.G31325", + "elementCount": 1, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/47882_HOTELSBRANDGEOLISTUSED", + "method": "PUT" + } + }, + "name": "Hotels Brand GEO List (Used by Custom Bots)", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 1, + "type": "GEO", + "uniqueId": "47882_HOTELSBRANDGEOLISTUSED" + }, + { + "accessControlGroup": "Car Rentals - 3-TE78Q.G116804", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/46790_CARRENTALSGEOBLOCK", + "method": "PUT" + } + }, + "name": "CarRentals Geo Block", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "46790_CARRENTALSGEOBLOCK" + }, + { + "accessControlGroup": "EPC - 3-TE78Q.G82853", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/45956_EPCGEOBLACKLIST", + "method": "PUT" + } + }, + "name": "EPC GEO Blacklist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "45956_EPCGEOBLACKLIST" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/44831_ECSCGEOBLACKLIST", + "method": "PUT" + } + }, + "name": "ECSC-GEO Blacklist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "44831_ECSCGEOBLACKLIST" + }, + { + "accessControlGroup": "exped_mk_101288 - 3-TE78Q.G112742", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/42687_HOTELSBRANDGEOBLACKLIST", + "method": "PUT" + } + }, + "name": "Hotels Brand Geo Blacklist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "42687_HOTELSBRANDGEOBLACKLIST" + }, + { + "accessControlGroup": "EGENCIA - 3-TE78Q.G122736", + "elementCount": 1, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/40731_BMROLLOUTGEO", + "method": "PUT" + } + }, + "name": "BM rollout-GEO", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 8, + "type": "GEO", + "uniqueId": "40731_BMROLLOUTGEO" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/40713_BMROLLOUTGEO", + "method": "PUT" + } + }, + "name": "Not Used_3-TE78Q ION", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 4, + "type": "GEO", + "uniqueId": "40713_BMROLLOUTGEO" + }, + { + "accessControlGroup": "3-TE78Q ION - 3-TE78Q.G17244", + "description": "case F-CS-2759696", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/40548_GEOBLOCKHOTWIRE", + "method": "PUT" + } + }, + "name": "GEO-Block Hotwire", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 2, + "type": "GEO", + "uniqueId": "40548_GEOBLOCKHOTWIRE" + }, + { + "accessControlGroup": "EGENCIA - 3-TE78Q.G122736", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/40016_TEMPGEOBLOCKLIST", + "method": "PUT" + } + }, + "name": "TEMP Geo-Block List", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "40016_TEMPGEOBLOCKLIST" + }, + { + "accessControlGroup": "EGENCIA - 3-TE78Q.G122736", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/25065_HOMEAWAYGEOBLOCK", + "method": "PUT" + } + }, + "name": "HomeAway Geo Block", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "25065_HOMEAWAYGEOBLOCK" + }, + { + "accessControlGroup": "EGENCIA - 3-TE78Q.G122736", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/17401_CLASSICVACATIONSGEOBLACK", + "method": "PUT" + } + }, + "name": "Classic vacations Geo Black list", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "17401_CLASSICVACATIONSGEOBLACK" + }, + { + "accessControlGroup": " Kona Site Defender - C-D5TW8R - C-D5TW8R.G31325", + "elementCount": 0, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/16865_EANGEOBLACKLIST", + "method": "PUT" + } + }, + "name": "EAN Geo Blacklist", + "networkListType": "networkListResponse", + "readOnly": false, + "shared": false, + "syncPoint": 0, + "type": "GEO", + "uniqueId": "16865_EANGEOBLACKLIST" + }, + { + "elementCount": 27, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/968_ITARCOUNTRYLIST", + "method": "PUT" + } + }, + "name": "International Traffic in Arms Regulations (ITAR) List", + "networkListType": "networkListResponse", + "readOnly": true, + "shared": true, + "syncPoint": 9, + "type": "GEO", + "uniqueId": "968_ITARCOUNTRYLIST" + }, + { + "elementCount": 22, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/969_IEEPACOUNTRYLIST", + "method": "PUT" + } + }, + "name": "International Emergency Economic Powers Act (IEEPA) List", + "networkListType": "networkListResponse", + "readOnly": true, + "shared": true, + "syncPoint": 3, + "type": "GEO", + "uniqueId": "969_IEEPACOUNTRYLIST" + }, + { + "elementCount": 18, + "links": { + "activateInProduction": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST/environments/PRODUCTION/activate", + "method": "POST" + }, + "activateInStaging": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST/environments/STAGING/activate", + "method": "POST" + }, + "appendItems": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST/append", + "method": "POST" + }, + "retrieve": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST" + }, + "statusInProduction": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST/environments/PRODUCTION/status" + }, + "statusInStaging": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST/environments/STAGING/status" + }, + "update": { + "href": "/network-list/v2/network-lists/967_OFACCOUNTRYLIST", + "method": "PUT" + } + }, + "name": "Office of Foreign Asset Control (OFAC) List", + "networkListType": "networkListResponse", + "readOnly": true, + "shared": true, + "syncPoint": 4, + "type": "GEO", + "uniqueId": "967_OFACCOUNTRYLIST" + } + ] +}