Terraform
OpenAPI provider spec generator
Note
Code Generation is currently in tech preview.The OpenAPI Provider Spec Generator is a CLI tool that transforms an OpenAPI Specification (OAS) into a Provider Code Specification. This section contains usage documentation and examples for using the generator.
Overview
The OpenAPI Provider Spec Generator receives two sources as input: an OAS file and a Generator Config file. This generator supports both OpenAPI 3.1.x and OpenAPI 3.0.x.
The generator outputs a Provider Code Specification which, as shown in the diagram above, can be used as input to a provider code generator, such as the Framework Code Generator, to create Terraform Provider code.
Usage
Installation
The CLI tool can be installed with the Go toolchain, via go install
:
go install github.com/hashicorp/terraform-plugin-codegen-openapi/cmd/tfplugingen-openapi@latest
Generate
The primary command for the spec generator is the generate
command, which requires a generator config and an OpenAPI 3.x spec as input:
tfplugingen-openapi generate \
--config <path/to/generator_config.yml> \
--output <output/for/provider_code_spec.json> \
<path/to/openapi_spec.json>
Flags
The available flags for the generate
command:
Flag | Required? | Default | Description |
---|---|---|---|
config | No | ./generator_config.yml | String path to a generator config YAML file |
output | No | ./provider_code_spec.json | String path to destination file for the generated provider code specification JSON. |
Arguments
After the flags is one required positional argument that is the path to an OpenAPI 3.1.x or 3.0.x specification file. This file can be in either JSON or YAML format.
tfplugingen-openapi generate \
--config ./generator_config.yml \
--output ./provider_code_spec.json \
./openapi_spec.yml
tfplugingen-openapi generate \
--config ./generator_config.yml \
--output ./provider_code_spec.json \
./openapi_spec.json
Generator Config
As OpenAPI is designed to describe HTTP APIs in a general format, a developer will need to configure which API operations (GET, POST, etc.) and components map to a given Terraform resource, data source, or provider.
A YAML file (generator config) is used to define these mappings for generating the provider code specification. It also provides configuration options such as aliasing path and query parameters and overriding description fields.
Here is an example of a generator config for the standard Petstore OpenAPI spec.
Provider
For generating the Provider specification, the generator config defines a single provider
object:
provider:
name: examplecloud
schema_ref: '#/components/schemas/examplecloud_provider_schema'
The provider
object has the following properties:
Property | Type | Required | Description |
---|---|---|---|
name | String | Yes | The provider type name (e.g. examplecloud ) |
schema_ref | String | No | JSON schema reference to an existing schema in the OAS. This can be used to map to the provider schema. |
ignores | Array | No | List of OAS properties to ignore during mapping. Each item is a string indicating the location of a property in an OAS schema, dot-separated. See ignores section below. |
For a detailed description of the schema mapping rules, see the Mapping OAS to Provider Code Specification section of the repo's design documentation.
Resources
For generating Resource specifications, the generator config defines a map resources
:
resources:
thing:
create:
path: /thing
method: POST
read:
path: /thing/{id}
method: GET
update:
path: /thing
method: PUT
delete:
path: /thing/{id}
method: DELETE
For an item in the resources
map, the key is the name of the resource (ex. thing
), and the value is an object which has the following properties:
Property | Type | Required? | Description |
---|---|---|---|
create | Object | Yes | Create API operation |
read | Object | No | Read API operation |
update | Object | No | Update API operation (this API operation currently does not affect the mapping) |
delete | Object | No | Delete API operation (this API operation currently does not affect the mapping) |
schema | Object | No | Allows configuration of the generated schema for this resource, see Schema Options |
Each API operation is an object that represents it's location in an OAS, with the following properties:
Property | Type | Required? | Description |
---|---|---|---|
path | String | Yes | Exactly matches the path key in the OAS paths object |
method | String | Yes | Matches the API operation in an OAS path item (GET , POST , PUT , DELETE , etc.) |
For a detailed description of how the generator uses the create
and read
operations to generate the resource schema, see the Mapping OAS to Provider Code Specification section of the repo's design documentation.
Data Sources
For generating Data Source specifications, the generator config defines a map data_sources
:
data_sources:
thing:
read:
path: /thing/{id}
method: GET
For an item in the data_sources
map, the key is the name of the data source (ex. thing
), and the value is an object which has the following properties:
Property | Type | Required? | Description |
---|---|---|---|
read | Object | Yes | Read API operation |
schema | Object | No | Allows configuration of the generated schema for this data source, see Schema Options |
The read
API operation is an object that represents it's location in an OAS, with the following properties:
Property | Type | Required? | Description |
---|---|---|---|
path | String | Yes | Exactly matches the path key in the OAS paths object |
method | String | Yes | Matches the API operation in an OAS path item (GET , POST , PUT , DELETE , etc.) |
For a detailed description of how the generator uses the read
operation to generate the data source schema, see the Mapping OAS to Provider Code Specification section of the repo's design documentation.
Schema Options
There are certain scenarios where you need to configure how a resource/data source is mapped to provider code specification. These options are defined in a schema
object for both resource and data sources:
resources:
thing:
# create, read, update, delete configuration
# ...
schema:
# Configure the 'thing' resource mapping
data_sources:
thing:
# read configuration
# ...
schema:
# Configure the 'thing' data source mapping
The schema
object has the following properties:
Property | Type | Required? | Description |
---|---|---|---|
attributes | Object | No | Defines mapping options for attributes |
ignores | Array | No | List of OAS properties to ignore during mapping. Each item is a string indicating the location of a property in an OAS schema, dot-separated. See ignores section below. |
The attributes
object has the following properties:
Property | Type | Required? | Description |
---|---|---|---|
aliases | Object | No | Allows aliasing of path/query parameter names (e.g. petId -> id ), see below |
overrides | Object | No | Allows overriding attribute information after mapping, like description , see below |
Ignoring OAS Properties
A common scenario that can occur during mapping is having additional properties on an OAS schema that you want to exclude from the final Terraform schema. You may want to exclude properties because:
- The property is needed for an API call, but unrelated to Terraform and not needed in the schema (pagination properties, duplicated properties, etc.)
- The property is not supported by the OpenAPI Provider Spec generator (multi-type or dynamic type properties)
The configuration for ignoring properties is an array of strings, each string representing a location of an OAS property to be ignored. A simple dot notation can be used to ignore child properties of objects. Using the Pet OAS, some valid ignores could look like:
resources:
pet:
# create, read, update, delete configuration
# ...
schema:
ignores:
# Ignore the "id" property
- id
# Ignore the entire nested object "category"
- category
# Ignore the "id" property of the nested object "category"
- category.id
# Ignore the entire nested list "tags"
- tags
# Ignore the "name" property of the nested list "tags"
- tags.name
Note
The ignores apply to all OAS operation schemas (request/response bodies) and parameters.Aliasing OAS Operation Parameters
In an OAS, it's possible for path
or query
parameters to have slightly different names than their associated attributes in a create request body/response body schema. For example, petId
(path parameter) and id
(defined on Pet schema):
// OpenAPI Specification
// ...
"paths": {
"/pet/{petId}": {
"get": {
"tags": [
"pet"
],
"summary": "Find pet by ID",
"description": "Returns a single pet",
"operationId": "getPetById",
"parameters": [
{
"name": "petId",
"in": "path",
"description": "ID of pet to return",
"required": true,
"schema": {
"type": "integer",
"format": "int64"
}
}
],
// ...
},
},
},
"components": {
"schemas": {
"Pet": {
"properties": {
"id": {
"type": "integer",
"format": "int64",
"example": 10
},
// ...
},
},
},
},
Merging petId
with Pet.id
can be achieved by using the schema.aliases
option in a resource or data source:
resources:
pet:
# create, read, update, delete configuration
# ...
schema:
attributes:
aliases:
# Match 'petId' parameter to 'id'
petId: id
Overriding Attribute Descriptions
In an OAS, it's possible for the description
fields of request/response body schemas to either be omitted, or the provided description
may not be written with Terraform nuance in mind.
To override the description
used for the provider code specification for an attribute, you can utilize the schema.overrides
map option in a resource or data source.
The overrides
map key is a string that represents the location of the attribute to be overridden. A simple dot notation can be used to represent nested attributes, like below:
resources:
pet:
# create, read, update, delete configuration
# ...
schema:
attributes:
overrides:
id:
description: The ID of the pet resource, this is a string
category:
description: The pet category, this is a single nested object
"category.id":
description: The ID of the pet category, this is a string
Example
This example shows how to use the generator CLI with a sample OpenAPI spec (Petstore) to generate a provider code specification.
- Start by downloading the Petstore OpenAPI 3.0 spec
curl https://petstore3.swagger.io/api/v3/openapi.json --output openapi.json
- Next, create an empty generator config file
touch generator_config.yml
- Add the following content to the
generator_config.yml
- This config describes a new
pet
resource for thepetstore
provider
- This config describes a new
provider:
name: petstore
resources:
pet:
create:
path: /pet
method: POST
read:
path: /pet/{petId}
method: GET
- Run the
generate
command
tfplugingen-openapi generate \
--config ./generator_config.yml \
--output ./provider_code_spec.json \
./openapi.json
- The resulting provider code specification will contain all the schema information provided by the
create
andread
operations inprovider_code_spec.json
{
"provider": {
"name": "petstore"
},
"resources": [
{
"name": "pet",
"schema": {
"attributes": [
{
"name": "category",
"single_nested": {
"computed_optional_required": "computed_optional",
"attributes": [
{
"name": "id",
"int64": {
"computed_optional_required": "computed_optional"
}
},
{
"name": "name",
"string": {
"computed_optional_required": "computed_optional"
}
}
]
}
},
{
"name": "id",
"int64": {
"computed_optional_required": "computed_optional"
}
},
{
"name": "name",
"string": {
"computed_optional_required": "required"
}
},
{
"name": "photo_urls",
"list": {
"computed_optional_required": "required",
"element_type": {
"string": {}
}
}
},
{
"name": "status",
"string": {
"computed_optional_required": "computed_optional",
"description": "pet status in the store",
"validators": [
{
"custom": {
"imports": [
{
"path": "github.com/hashicorp/terraform-plugin-framework-validators/stringvalidator"
}
],
"schema_definition": "stringvalidator.OneOf(\n\"available\",\n\"pending\",\n\"sold\",\n)"
}
}
]
}
},
{
"name": "tags",
"list_nested": {
"computed_optional_required": "computed_optional",
"nested_object": {
"attributes": [
{
"name": "id",
"int64": {
"computed_optional_required": "computed_optional"
}
},
{
"name": "name",
"string": {
"computed_optional_required": "computed_optional"
}
}
]
}
}
},
{
"name": "pet_id",
"int64": {
"computed_optional_required": "computed_optional",
"description": "ID of pet to return"
}
}
]
}
}
]
}
- This provider code specification file can now be used as input to a provider code generator, such as the Framework Code Generator, to create Terraform Provider code.
- For a full demo using the OpenAPI Spec Generator and the Framework Code Generator on a Terraform provider, see the Workflow Example section.
Known Limitations
As OpenAPI is designed to describe HTTP APIs in general, it doesn't always fully align with Terraform Provider design principles. There are pieces of logic in this generator that make assumptions on what portions of the OAS to use when mapping to the provider code specification, however there are some limitations on what can be supported.
See the Known Limitations section in the repo's design document for more details.