// +build codegen package api import ( "io/ioutil" "path/filepath" "strings" ) type service struct { srcName string dstName string serviceVersion string } var mergeServices = map[string]service{ "dynamodbstreams": service{ dstName: "dynamodb", srcName: "streams.dynamodb", }, "wafregional": service{ dstName: "waf", srcName: "waf-regional", serviceVersion: "2015-08-24", }, } // customizationPasses Executes customization logic for the API by package name. func (a *API) customizationPasses() { var svcCustomizations = map[string]func(*API){ "s3": s3Customizations, "cloudfront": cloudfrontCustomizations, "rds": rdsCustomizations, // Disable endpoint resolving for services that require customer // to provide endpoint them selves. "cloudsearchdomain": disableEndpointResolving, "iotdataplane": disableEndpointResolving, } for k, _ := range mergeServices { svcCustomizations[k] = mergeServicesCustomizations } if fn := svcCustomizations[a.PackageName()]; fn != nil { fn(a) } } // s3Customizations customizes the API generation to replace values specific to S3. func s3Customizations(a *API) { var strExpires *Shape for name, s := range a.Shapes { // Remove ContentMD5 members if _, ok := s.MemberRefs["ContentMD5"]; ok { delete(s.MemberRefs, "ContentMD5") } for _, refName := range []string{"Bucket", "SSECustomerKey", "CopySourceSSECustomerKey"} { if ref, ok := s.MemberRefs[refName]; ok { ref.GenerateGetter = true } } // Expires should be a string not time.Time since the format is not // enforced by S3, and any value can be set to this field outside of the SDK. if strings.HasSuffix(name, "Output") { if ref, ok := s.MemberRefs["Expires"]; ok { if strExpires == nil { newShape := *ref.Shape strExpires = &newShape strExpires.Type = "string" strExpires.refs = []*ShapeRef{} } ref.Shape.removeRef(ref) ref.Shape = strExpires ref.Shape.refs = append(ref.Shape.refs, &s.MemberRef) } } } s3CustRemoveHeadObjectModeledErrors(a) } // S3 HeadObject API call incorrect models NoSuchKey as valid // error code that can be returned. This operation does not // return error codes, all error codes are derived from HTTP // status codes. // // aws/aws-sdk-go#1208 func s3CustRemoveHeadObjectModeledErrors(a *API) { op, ok := a.Operations["HeadObject"] if !ok { return } op.Documentation += ` // // See http://docs.aws.amazon.com/AmazonS3/latest/API/ErrorResponses.html#RESTErrorResponses // for more information on returned errors.` op.ErrorRefs = []ShapeRef{} } // cloudfrontCustomizations customized the API generation to replace values // specific to CloudFront. func cloudfrontCustomizations(a *API) { // MaxItems members should always be integers for _, s := range a.Shapes { if ref, ok := s.MemberRefs["MaxItems"]; ok { ref.ShapeName = "Integer" ref.Shape = a.Shapes["Integer"] } } } // mergeServicesCustomizations references any duplicate shapes from DynamoDB func mergeServicesCustomizations(a *API) { info := mergeServices[a.PackageName()] p := strings.Replace(a.path, info.srcName, info.dstName, -1) if info.serviceVersion != "" { index := strings.LastIndex(p, "/") files, _ := ioutil.ReadDir(p[:index]) if len(files) > 1 { panic("New version was introduced") } p = p[:index] + "/" + info.serviceVersion } file := filepath.Join(p, "api-2.json") serviceAPI := API{} serviceAPI.Attach(file) serviceAPI.Setup() for n := range a.Shapes { if _, ok := serviceAPI.Shapes[n]; ok { a.Shapes[n].resolvePkg = "github.com/aws/aws-sdk-go/service/" + info.dstName } } } // rdsCustomizations are customization for the service/rds. This adds non-modeled fields used for presigning. func rdsCustomizations(a *API) { inputs := []string{ "CopyDBSnapshotInput", "CreateDBInstanceReadReplicaInput", "CopyDBClusterSnapshotInput", "CreateDBClusterInput", } for _, input := range inputs { if ref, ok := a.Shapes[input]; ok { ref.MemberRefs["SourceRegion"] = &ShapeRef{ Documentation: docstring(`SourceRegion is the source region where the resource exists. This is not sent over the wire and is only used for presigning. This value should always have the same region as the source ARN.`), ShapeName: "String", Shape: a.Shapes["String"], Ignore: true, } ref.MemberRefs["DestinationRegion"] = &ShapeRef{ Documentation: docstring(`DestinationRegion is used for presigning the request to a given region.`), ShapeName: "String", Shape: a.Shapes["String"], } } } } func disableEndpointResolving(a *API) { a.Metadata.NoResolveEndpoint = true }