123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397 |
- package api
- import (
- "bytes"
- "fmt"
- "path"
- "regexp"
- "sort"
- "strings"
- "text/template"
- "github.com/aws/aws-sdk-go/internal/util"
- )
- // A ShapeRef defines the usage of a shape within the API.
- type ShapeRef struct {
- API *API `json:"-"`
- Shape *Shape `json:"-"`
- Documentation string
- ShapeName string `json:"shape"`
- Location string
- LocationName string
- QueryName string
- Flattened bool
- Streaming bool
- XMLAttribute bool
- XMLNamespace XMLInfo
- Payload string
- }
- // A XMLInfo defines URL and prefix for Shapes when rendered as XML
- type XMLInfo struct {
- Prefix string
- URI string
- }
- // A Shape defines the definition of a shape type
- type Shape struct {
- API *API `json:"-"`
- ShapeName string
- Documentation string
- MemberRefs map[string]*ShapeRef `json:"members"`
- MemberRef ShapeRef `json:"member"`
- KeyRef ShapeRef `json:"key"`
- ValueRef ShapeRef `json:"value"`
- Required []string
- Payload string
- Type string
- Exception bool
- Enum []string
- EnumConsts []string
- Flattened bool
- Streaming bool
- Location string
- LocationName string
- XMLNamespace XMLInfo
- refs []*ShapeRef // References to this shape
- resolvePkg string // use this package in the goType() if present
- }
- // Rename changes the name of the Shape to newName. Also updates
- // the associated API's reference to use newName.
- func (s *Shape) Rename(newName string) {
- for _, r := range s.refs {
- r.ShapeName = newName
- }
- delete(s.API.Shapes, s.ShapeName)
- s.API.Shapes[newName] = s
- s.ShapeName = newName
- }
- // MemberNames returns a slice of struct member names.
- func (s *Shape) MemberNames() []string {
- i, names := 0, make([]string, len(s.MemberRefs))
- for n := range s.MemberRefs {
- names[i] = n
- i++
- }
- sort.Strings(names)
- return names
- }
- // GoTypeWithPkgName returns a shape's type as a string with the package name in
- // <packageName>.<type> format. Package naming only applies to structures.
- func (s *Shape) GoTypeWithPkgName() string {
- return goType(s, true)
- }
- // GoType returns a shape's Go type
- func (s *Shape) GoType() string {
- return goType(s, false)
- }
- // GoType returns a shape ref's Go type.
- func (ref *ShapeRef) GoType() string {
- if ref.Shape == nil {
- panic(fmt.Errorf("missing shape definition on reference for %#v", ref))
- }
- return ref.Shape.GoType()
- }
- // GoTypeWithPkgName returns a shape's type as a string with the package name in
- // <packageName>.<type> format. Package naming only applies to structures.
- func (ref *ShapeRef) GoTypeWithPkgName() string {
- if ref.Shape == nil {
- panic(fmt.Errorf("missing shape definition on reference for %#v", ref))
- }
- return ref.Shape.GoTypeWithPkgName()
- }
- // Returns a string version of the Shape's type.
- // If withPkgName is true, the package name will be added as a prefix
- func goType(s *Shape, withPkgName bool) string {
- switch s.Type {
- case "structure":
- if withPkgName || s.resolvePkg != "" {
- pkg := s.resolvePkg
- if pkg != "" {
- s.API.imports[pkg] = true
- pkg = path.Base(pkg)
- } else {
- pkg = s.API.PackageName()
- }
- return fmt.Sprintf("*%s.%s", pkg, s.ShapeName)
- }
- return "*" + s.ShapeName
- case "map":
- return "map[string]" + s.ValueRef.GoType()
- case "list":
- return "[]" + s.MemberRef.GoType()
- case "boolean":
- return "*bool"
- case "string", "character":
- return "*string"
- case "blob":
- return "[]byte"
- case "integer", "long":
- return "*int64"
- case "float", "double":
- return "*float64"
- case "timestamp":
- s.API.imports["time"] = true
- return "*time.Time"
- default:
- panic("Unsupported shape type: " + s.Type)
- }
- }
- // GoTypeElem returns the Go type for the Shape. If the shape type is a pointer just
- // the type will be returned minus the pointer *.
- func (s *Shape) GoTypeElem() string {
- t := s.GoType()
- if strings.HasPrefix(t, "*") {
- return t[1:]
- }
- return t
- }
- // GoTypeElem returns the Go type for the Shape. If the shape type is a pointer just
- // the type will be returned minus the pointer *.
- func (ref *ShapeRef) GoTypeElem() string {
- if ref.Shape == nil {
- panic(fmt.Errorf("missing shape definition on reference for %#v", ref))
- }
- return ref.Shape.GoTypeElem()
- }
- // GoTags returns the rendered tags string for the ShapeRef
- func (ref *ShapeRef) GoTags(toplevel bool, isRequired bool) string {
- code := "`"
- if ref.Location != "" {
- code += `location:"` + ref.Location + `" `
- } else if ref.Shape.Location != "" {
- code += `location:"` + ref.Shape.Location + `" `
- }
- if ref.LocationName != "" {
- code += `locationName:"` + ref.LocationName + `" `
- } else if ref.Shape.LocationName != "" {
- code += `locationName:"` + ref.Shape.LocationName + `" `
- }
- if ref.QueryName != "" {
- code += `queryName:"` + ref.QueryName + `" `
- }
- if ref.Shape.MemberRef.LocationName != "" {
- code += `locationNameList:"` + ref.Shape.MemberRef.LocationName + `" `
- }
- if ref.Shape.KeyRef.LocationName != "" {
- code += `locationNameKey:"` + ref.Shape.KeyRef.LocationName + `" `
- }
- if ref.Shape.ValueRef.LocationName != "" {
- code += `locationNameValue:"` + ref.Shape.ValueRef.LocationName + `" `
- }
- code += `type:"` + ref.Shape.Type + `" `
- // embed the timestamp type for easier lookups
- if ref.Shape.Type == "timestamp" {
- code += `timestampFormat:"`
- if ref.Location == "header" {
- code += "rfc822"
- } else {
- switch ref.API.Metadata.Protocol {
- case "json", "rest-json":
- code += "unix"
- case "rest-xml", "ec2", "query":
- code += "iso8601"
- }
- }
- code += `" `
- }
- if ref.Shape.Flattened || ref.Flattened {
- code += `flattened:"true" `
- }
- if ref.XMLAttribute {
- code += `xmlAttribute:"true" `
- }
- if isRequired {
- code += `required:"true" `
- }
- if ref.Shape.IsEnum() {
- code += `enum:"` + ref.ShapeName + `" `
- }
- if toplevel {
- if ref.Shape.Payload != "" {
- code += `payload:"` + ref.Shape.Payload + `" `
- }
- if ref.XMLNamespace.Prefix != "" {
- code += `xmlPrefix:"` + ref.XMLNamespace.Prefix + `" `
- } else if ref.Shape.XMLNamespace.Prefix != "" {
- code += `xmlPrefix:"` + ref.Shape.XMLNamespace.Prefix + `" `
- }
- if ref.XMLNamespace.URI != "" {
- code += `xmlURI:"` + ref.XMLNamespace.URI + `" `
- } else if ref.Shape.XMLNamespace.URI != "" {
- code += `xmlURI:"` + ref.Shape.XMLNamespace.URI + `" `
- }
- }
- return strings.TrimSpace(code) + "`"
- }
- // Docstring returns the godocs formated documentation
- func (ref *ShapeRef) Docstring() string {
- if ref.Documentation != "" {
- return ref.Documentation
- }
- return ref.Shape.Docstring()
- }
- // Docstring returns the godocs formated documentation
- func (s *Shape) Docstring() string {
- return s.Documentation
- }
- const goCodeStringerTmpl = `
- // String returns the string representation
- func (s {{ .ShapeName }}) String() string {
- return awsutil.Prettify(s)
- }
- // GoString returns the string representation
- func (s {{ .ShapeName }}) GoString() string {
- return s.String()
- }
- `
- func (s *Shape) goCodeStringers() string {
- tmpl := template.Must(template.New("goCodeStringerTmpl").Parse(goCodeStringerTmpl))
- w := bytes.Buffer{}
- if err := tmpl.Execute(&w, s); err != nil {
- panic(fmt.Sprintln("Unexpected error executing goCodeStringers template", err))
- }
- return w.String()
- }
- var enumStrip = regexp.MustCompile(`[^a-zA-Z0-9_:\./-]`)
- var enumDelims = regexp.MustCompile(`[-_:\./]+`)
- var enumCamelCase = regexp.MustCompile(`([a-z])([A-Z])`)
- // EnumName returns the Nth enum in the shapes Enum list
- func (s *Shape) EnumName(n int) string {
- enum := s.Enum[n]
- enum = enumStrip.ReplaceAllLiteralString(enum, "")
- enum = enumCamelCase.ReplaceAllString(enum, "$1-$2")
- parts := enumDelims.Split(enum, -1)
- for i, v := range parts {
- v = strings.ToLower(v)
- parts[i] = ""
- if len(v) > 0 {
- parts[i] = strings.ToUpper(v[0:1])
- }
- if len(v) > 1 {
- parts[i] += v[1:]
- }
- }
- enum = strings.Join(parts, "")
- enum = strings.ToUpper(enum[0:1]) + enum[1:]
- return enum
- }
- // GoCode returns the rendered Go code for the Shape.
- func (s *Shape) GoCode() string {
- code := s.Docstring()
- if !s.IsEnum() {
- code += "type " + s.ShapeName + " "
- }
- switch {
- case s.Type == "structure":
- code += "struct {\n"
- for _, n := range s.MemberNames() {
- m := s.MemberRefs[n]
- code += m.Docstring()
- if (m.Streaming || m.Shape.Streaming) && s.Payload == n {
- rtype := "io.ReadSeeker"
- if len(s.refs) > 1 {
- rtype = "aws.ReaderSeekCloser"
- } else if strings.HasSuffix(s.ShapeName, "Output") {
- rtype = "io.ReadCloser"
- }
- s.API.imports["io"] = true
- code += n + " " + rtype + " " + m.GoTags(false, s.IsRequired(n)) + "\n\n"
- } else {
- code += n + " " + m.GoType() + " " + m.GoTags(false, s.IsRequired(n)) + "\n\n"
- }
- }
- metaStruct := "metadata" + s.ShapeName
- ref := &ShapeRef{ShapeName: s.ShapeName, API: s.API, Shape: s}
- code += "\n" + metaStruct + " `json:\"-\" xml:\"-\"`\n"
- code += "}\n\n"
- code += "type " + metaStruct + " struct {\n"
- code += "SDKShapeTraits bool " + ref.GoTags(true, false)
- code += "}"
- if !s.API.NoStringerMethods {
- code += s.goCodeStringers()
- }
- case s.IsEnum():
- code += "const (\n"
- for n, e := range s.Enum {
- code += fmt.Sprintf("\t// @enum %s\n\t%s = %q\n",
- s.ShapeName, s.EnumConsts[n], e)
- }
- code += ")"
- default:
- panic("Cannot generate toplevel shape for " + s.Type)
- }
- return util.GoFmt(code)
- }
- // IsEnum returns whether this shape is an enum list
- func (s *Shape) IsEnum() bool {
- return s.Type == "string" && len(s.Enum) > 0
- }
- // IsRequired returns if member is a required field.
- func (s *Shape) IsRequired(member string) bool {
- for _, n := range s.Required {
- if n == member {
- return true
- }
- }
- return false
- }
- // IsInternal returns whether the shape was defined in this package
- func (s *Shape) IsInternal() bool {
- return s.resolvePkg == ""
- }
- // removeRef removes a shape reference from the list of references this
- // shape is used in.
- func (s *Shape) removeRef(ref *ShapeRef) {
- r := s.refs
- for i := 0; i < len(r); i++ {
- if r[i] == ref {
- j := i + 1
- copy(r[i:], r[j:])
- for k, n := len(r)-j+i, len(r); k < n; k++ {
- r[k] = nil // free up the end of the list
- } // for k
- s.refs = r[:len(r)-j+i]
- break
- }
- }
- }
|