TOSCA DataTypes to Go (tdt2go) structures generator
The goal of tdt2go is to create a generator that takes Data Types defined in a TOSCA definition file and generate a Go source file with corresponding Go data structures.
tdt2go can be installed using go get
go get -u github.com/ystia/tdt2go/cmd/tdt2goBinaries distributions could be downloaded from github
$ tdt2go --help
tdt2go allows to generate Go source files containing data structures generated from files containing TOSCA data types
Usage:
  tdt2go <tosca_file> [flags]
Flags:
  -e, --exclude strings                regexp patterns of data types fully qualified names to exclude. Only non-matching datatypes will be transformed. Include patterns have the precedence over exclude patterns.
  -f, --file string                    file to be generated, if not defined resulting generated file will be printed on default output.
  -b, --generate-builtin               Generate tosca builtin types as 'range' or 'scalar-unit' for instance along with datatypes in this file. (default: false)
  -h, --help                           help for tdt2go
  -i, --include strings                regexp patterns of data types fully qualified names to include. Only matching datatypes will be transformed. Include patterns have the precedence over exclude patterns.
  -m, --name-mappings stringToString   map of regular expressions and their corresponding remplacements that will be applied to TOSCA datatypes fully qualified names to transform them into Go struct names. This is generally used to keep information from the fully qualified name into the generated name. (default [])
  -p, --package string                 package name as it should appear in source file, defaults to the package name of the current directory.- Support of types inheritance via Go composition
-  Support basic types conversions:
- integer➡️- int
- string➡️- string
- boolean➡️- bool
- float➡️- float64
- timestamp➡️- time.Time
- list➡️ slice with entry_schema support
- map➡️ map with entry_schema support
 
-  Generation of TOSCA builtin types such as version,range,scalar-units ...
- include/exclude filters
-  Type name mapping like tosca\.datatypes\.(.+)➡️Normative${1}sotosca.datatypes.CredentialbecomeNormativeCredential
- Use type or property description on generated comments
-  Make use of TOSCA constraintsanddefault
This example is an extract of the TOSCA normative types.
tosca_definitions_version: tosca_simple_yaml_1_0
metadata:
  template_name: tosca-normative-types
  template_author: TOSCA TC
  template_version: 1.3.0
data_types:
  tosca.datatypes.Root:
    description: The TOSCA root Data Type all other TOSCA base Data Types derive from
  tosca.datatypes.json:
    derived_from: string
  tosca.datatypes.xml:
    derived_from: string
  tosca.datatypes.Credential:
    derived_from: tosca.datatypes.Root
    description: >
      The Credential type is a complex TOSCA data Type used when describing authorization credentials used to access network accessible resources.
    properties:
      protocol:
        type: string
        description: The optional protocol name.
        required: false
      token_type:
        type: string
        description: The required token type.
        default: password
      token:
        type: string
        description: The required token used as a credential for authorization or access to a networked resource.
      keys:
        type: map
        description: The optional list of protocol-specific keys or assertions.
        required: false
        entry_schema:
          type: string
      user:
        type: string
        description: The optional user (name or ID) used for non-token based credentials.
        required: false
  tosca.datatypes.TimeInterval:
    derived_from: tosca.datatypes.Root
    properties:
      start_time:
        type: timestamp
        required: true
      end_time:
        type: timestamp
        required: true
  tosca.datatypes.network.NetworkInfo:
    derived_from: tosca.datatypes.Root
    description: The Network type is a complex TOSCA data type used to describe logical network information.
    properties:
      network_name:
        type: string
        description: >
          The name of the logical network.
          e.g., “public”, “private”, “admin”. etc.
      network_id:
        type: string
        description: The unique ID of for the network generated by the network provider.
      addresses:
        type: list
        description: The list of IP addresses assigned from the underlying network.
        entry_schema:
          type: string
  tosca.datatypes.network.PortInfo:
    derived_from: tosca.datatypes.Root
    description: The PortInfo type is a complex TOSCA data type used to describe network port information.
    properties:
      port_name:
        type: string
        description: The logical network port name.
      port_id:
        type: string
        description: The unique ID for the network port generated by the network provider.
      network_id:
        type: string
        description: The unique ID for the network.
      mac_address:
        type: string
        description: The unique media access control address (MAC address) assigned to the port.
      addresses:
        type: list
        description: The list of IP address(es) assigned to the port.
        entry_schema:
          type: string
  tosca.datatypes.network.PortDef:
    derived_from: integer
    description: The PortDef type is a TOSCA data Type used to define a network port.
    constraints:
      - in_range: [ 1, 65535 ]
  tosca.datatypes.network.PortSpec:
    derived_from: tosca.datatypes.Root
    description: The PortSpec type is a complex TOSCA data Type used when describing port specifications for a network connection.
    properties:
      protocol:
        type: string
        description: The required protocol used on the port.
        required: true
        default: tcp
        constraints:
          - valid_values: [ udp, tcp, igmp ]
      target:
        type: tosca.datatypes.network.PortDef
        description: The optional target port.
      target_range:
        type: range
        description: The optional range for target port.
        constraints:
          - in_range: [ 1, 65535 ]
      source:
        type: tosca.datatypes.network.PortDef
        description: The optional target port.
      source_range:
        type: range
        description: The optional range for source port.
        constraints:
          - in_range: [ 1, 65535 ]
Here are the resulting generated structures (with the builtin types option):
// Code generated by tdt2go
// DO NOT EDIT! ANY CHANGES MAY BE OVERWRITTEN.
package datatypes
import (
   "time"
)
// Credential is the generated representation of tosca.datatypes.Credential data type
type Credential struct {
    Root
    Keys      map[string]string `mapstructure:"keys" json:"keys,omitempty"`
    Protocol  string            `mapstructure:"protocol" json:"protocol,omitempty"`
    Token     string            `mapstructure:"token" json:"token,omitempty"`
    TokenType string            `mapstructure:"token_type" json:"token_type,omitempty"`
    User      string            `mapstructure:"user" json:"user,omitempty"`
}
// Root is the generated representation of tosca.datatypes.Root data type
type Root struct {
}
// TimeInterval is the generated representation of tosca.datatypes.TimeInterval data type
type TimeInterval struct {
    Root
    EndTime   time.Time `mapstructure:"end_time" json:"end_time,omitempty"`
    StartTime time.Time `mapstructure:"start_time" json:"start_time,omitempty"`
}
// Json is the generated representation of tosca.datatypes.json data type
type Json string
// NetworkInfo is the generated representation of tosca.datatypes.network.NetworkInfo data type
type NetworkInfo struct {
    Root
    Addresses   []string `mapstructure:"addresses" json:"addresses,omitempty"`
    NetworkId   string   `mapstructure:"network_id" json:"network_id,omitempty"`
    NetworkName string   `mapstructure:"network_name" json:"network_name,omitempty"`
}
// PortDef is the generated representation of tosca.datatypes.network.PortDef data type
type PortDef int
// PortInfo is the generated representation of tosca.datatypes.network.PortInfo data type
type PortInfo struct {
    Root
    Addresses  []string `mapstructure:"addresses" json:"addresses,omitempty"`
    MacAddress string   `mapstructure:"mac_address" json:"mac_address,omitempty"`
    NetworkId  string   `mapstructure:"network_id" json:"network_id,omitempty"`
    PortId     string   `mapstructure:"port_id" json:"port_id,omitempty"`
    PortName   string   `mapstructure:"port_name" json:"port_name,omitempty"`
}
// PortSpec is the generated representation of tosca.datatypes.network.PortSpec data type
type PortSpec struct {
    Root
    Protocol    string  `mapstructure:"protocol" json:"protocol,omitempty"`
    Source      PortDef `mapstructure:"source" json:"source,omitempty"`
    SourceRange Range   `mapstructure:"source_range" json:"source_range,omitempty"`
    Target      PortDef `mapstructure:"target" json:"target,omitempty"`
    TargetRange Range   `mapstructure:"target_range" json:"target_range,omitempty"`
}
// Xml is the generated representation of tosca.datatypes.xml data type
type Xml string
// Range is the generated representation of tosca:range data type
type Range []uint64
// ScalarUnit is the generated representation of tosca:scalar-unit data type
type ScalarUnit string
// ScalarUnitBitRate is the generated representation of tosca:scalar-unit.bitrate data type
type ScalarUnitBitRate ScalarUnit
// ScalarUnitFrequency is the generated representation of tosca:scalar-unit.frequency data type
type ScalarUnitFrequency ScalarUnit
// ScalarUnitSize is the generated representation of tosca:scalar-unit.size data type
type ScalarUnitSize ScalarUnit
// ScalarUnitTim is the generated representation of tosca:scalar-unit.time data type
type ScalarUnitTim ScalarUnit
// Version is the generated representation of tosca:version data type
type Version stringName mappings allows to rename a generated Go struct name based on its TOSCA fully qualified name using regular expressions. It is also very common to use the go generate command to generate Go code. Here are some gotchas to take into account.
- the $is used in regexp replacements to identify capturing groups. But$is interpreted by go generate and should be replaced by$DOLLAR
- mappings should not be single quoted as it should be using a regular shell
Here is an example:
package mytoscatypes
//go:generate tdt2go -m tosca\.datatypes\.(.*)=TOSCA_$DOLLAR{1} -f struct_normative.go normative-types.ymltdt2go is distributed under Apache 2.0 License.
tdt2go dependencies and attributions could be found here: https://app.fossa.io/reports/9d5e1c45-ef44-4ed7-a433-fe47474d9d56