Skip to content

hashicorp/terraform-plugin-docs

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

terraform-plugin-docs

This repository contains tools and packages for creating Terraform plugin docs (currently only provider plugins). The primary way users will interact with this is the tfplugindocs CLI tool to generate and validate plugin documentation.

tfplugindocs

The tfplugindocs CLI has three main commands, migrate, validate and generate (generate is the default). This tool will let you generate documentation for your provider from live example .tf files and markdown templates. It will also export schema information from the provider (using terraform providers schema -json), and sync the schema with the reference documents.

If your documentation only consists of simple examples and schema information, the tool can also generate missing template files to make website creation extremely simple for most providers.

Installation

You can install a copy of the binary manually from the releases, or you can optionally use the tools.go model for tool installation.

Note

Here is a brief ./tools/tools.go example from https://github.com/hashicorp/terraform-provider-scaffolding-framework:

//go:build tools

package tools

import (
  // Documentation generation
  _ "github.com/hashicorp/terraform-plugin-docs/cmd/tfplugindocs"
)

Then run the following to install and verify tfplugindocs:

export GOBIN=$PWD/bin
export PATH=$GOBIN:$PATH
go install github.com/hashicorp/terraform-plugin-docs/cmd/tfplugindocs
which tfplugindocs

Usage

$ tfplugindocs --help
Usage: tfplugindocs [--version] [--help] <command> [<args>]

Available commands are:
                the generate command is run by default
    generate    generates a plugin website from code, templates, and examples
    migrate     migrates website files from either the legacy rendered website directory (`website/docs/r`) or the docs rendered website directory (`docs/resources`) to the tfplugindocs supported structure (`templates/`).
    validate    validates a plugin website

generate command:

$ tfplugindocs generate --help

Usage: tfplugindocs generate [<args>]

    --examples-dir <ARG>             examples directory based on provider-dir                                                                                           (default: "examples")
    --ignore-deprecated <ARG>        don't generate documentation for deprecated resources and data-sources                                                             (default: "false")
    --provider-dir <ARG>             relative or absolute path to the root provider code directory when running the command outside the root provider code directory
    --provider-name <ARG>            provider name, as used in Terraform configurations; defaults to the --provider-dir short name (after removing `terraform-provider-` prefix)
    --providers-schema <ARG>         path to the providers schema JSON file, which contains the output of the terraform providers schema -json command. Setting this flag will skip building the provider and calling Terraform CLI
    --rendered-provider-name <ARG>   provider name, as generated in documentation (ex. page titles, ...); defaults to the --provider-name
    --rendered-website-dir <ARG>     output directory based on provider-dir                                                                                             (default: "docs")
    --tf-version <ARG>               terraform binary version to download. If not provided, will look for a terraform binary in the local environment. If not found in the environment, will download the latest version of Terraform
    --website-source-dir <ARG>       templates directory based on provider-dir                                                                                          (default: "templates")
    --website-temp-dir <ARG>         temporary directory (used during generation)

validate command:

$ tfplugindocs validate --help

Usage: tfplugindocs validate [<args>]

    --allowed-guide-subcategories <ARG>           comma separated list of allowed guide frontmatter subcategories
    --allowed-guide-subcategories-file <ARG>      path to newline separated file of allowed guide frontmatter subcategories
    --allowed-resource-subcategories <ARG>        comma separated list of allowed resource frontmatter subcategories
    --allowed-resource-subcategories-file <ARG>   path to newline separated file of allowed resource frontmatter subcategories
    --provider-dir <ARG>                          relative or absolute path to the root provider code directory; this will default to the current working directory if not set
    --provider-name <ARG>                         provider name, as used in Terraform configurations; defaults to the --provider-dir short name (after removing `terraform-provider-` prefix)
    --providers-schema <ARG>                      path to the providers schema JSON file, which contains the output of the terraform providers schema -json command. Setting this flag will skip building the provider and calling Terraform CLI
    --tf-version <ARG>                            terraform binary version to download. If not provided, will look for a terraform binary in the local environment. If not found in the environment, will download the latest version of Terraform

migrate command:

$ tfplugindocs migrate --help

Usage: tfplugindocs migrate [<args>]

    --examples-dir <ARG>             examples directory based on provider-dir                                                                                           (default: "examples")
    --provider-dir <ARG>             relative or absolute path to the root provider code directory when running the command outside the root provider code directory
    --templates-dir <ARG>            new website templates directory based on provider-dir; files will be migrated to this directory                                    (default: "templates")
    --provider-name <ARG>            provider name, as used in Terraform configurations; defaults to the --provider-dir short name (after removing `terraform-provider-` prefix)

How it Works

When you run tfplugindocs, by default from the root directory of a provider codebase, the tool takes the following actions:

  • Copy all the templates and static files to a temporary directory
  • Build (go build) a temporary binary of the provider source code
  • Collect schema information using terraform providers schema -json
  • Generate a default provider template file, if missing (index.md)
  • Generate resource template files, if missing
  • Generate data source template files, if missing
  • Generate function template files, if missing (Requires Terraform v1.8.0+)
  • Generate ephemeral resource template files, if missing (Requires Terraform v1.10.0+)
  • Generate action template files, if missing (Requires Terraform v1.14.0+)
  • Generate list resource templeate files, if missing (Requires Terraform v1.14.0+)
  • Copy all non-template files to the output website directory

Note

Non-template files that already exist in the output website directory will not be overwritten.

  • Process all the remaining templates to generate files for the output website directory

For inspiration, you can look at the templates and output of the terraform-provider-random and terraform-provider-tls. You can browse their respective docs on the Terraform Registry, here and here.

Usage of Terraform binary

If the --providers-schema flag is not provided, tfplugindocs will use the Terraform binary to generate the provider schema with the commands:

We recommend using the latest version of Terraform when using tfplugindocs, however, the version can be specified with the --tf-version flag if needed.

About the id attribute

If the provider schema didn't set id for the given resource/data-source, the documentation generated will place it under the "Read Only" section and provide a simple description.

Otherwise, the provider developer can set an arbitrary description like this:

    // ...
    Schema: map[string]*schema.Schema{
        // ...
        "id": {
            Type:     schema.TypeString,
            Computed: true,
            Description: "Unique identifier for this resource",
		},
        // ...
    }
    // ...

Validate subcommand

The validate subcommand can be used to validate the provider website documentation against the Terraform Registry's provider documentation guidelines and provider documentation best practices. The current checks in the validate command are:

Check Description
InvalidDirectoriesCheck Checks for valid subdirectory structure and throws an error if an invalid Terraform Provider documentation subdirectory is found.
MixedDirectoriesCheck Throws an error if both legacy documentation (/website/docs) and registry documentation (/docs) are found.
FileSizeCheck Throws an error if the documentation file is above the registry storage limit.
FileExtensionCheck Throws an error if the extension of the given file is not a valid registry documentation extension.
FrontMatterCheck Checks the YAML frontmatter of documentation for missing required fields or invalid fields. Optionally, checks that the subcategory is within the specified allow list.
FileMismatchCheck Throws an error if the names/number of resources/datasources/functions in the provider schema does not match the names/number of files in the corresponding documentation directory.

All check errors are wrapped and returned as a single error message to stderr.

Migrate subcommand

The migrate subcommand can be used to migrate website files from either the legacy rendered website directory (website/docs/r) or the docs rendered website directory (docs/resources) to the tfplugindocs supported structure (templates/). Markdown files in the rendered website directory will be converted to tfplugindocs templates. The legacy website/ directory will be removed after migration to avoid Terraform Registry ingress issues.

The migrate subcommand takes the following actions:

  1. Determines the rendered website directory based on the --provider-dir argument
  2. Determines the provider name based on the --provider-name argument
  3. Copies the contents of the rendered website directory to the --templates-dir folder (will create this folder if it doesn't exist)
  4. (if the rendered website is using legacy format) Renames docs/d/ and docs/r/ subdirectories to data-sources/ and resources/ respectively
  5. Renames files in the --templates-dir folder to remove the provider shortname prefix from the file name
  6. Change file suffixes for Markdown files to .md.tmpl to create website templates
  7. Extracts code blocks from website docs to create individual example files in --examples-dir (will create this folder if it doesn't exist)
  8. Replace extracted example code in website templates with codefile/tffile template functions referencing the example files.
  9. Copies non-template files to --templates-dir folder
  10. Removes the website/ directory

Conventional Paths

The generation of missing documentation is based on a number of assumptions / conventional paths.

For templates:

NOTE: In the following conventional paths for templates, <data source name>, <resource name>, and <function name> do not include the provider prefix.

Path Description
templates/ Root of templated docs
templates/index.md[.tmpl] Docs index page (or template)
templates/actions.md[.tmpl] Generic action page (or template)
templates/actions/<action type name>.md[.tmpl] Action page (or template)
templates/data-sources.md[.tmpl] Generic data source page (or template)
templates/data-sources/<data source name>.md[.tmpl] Data source page (or template)
templates/ephemeral-resources.md[.tmpl] Generic ephemeral resource page (or template)
templates/ephemeral-resources/<ephemeral resource name>.md[.tmpl] Ephemeral resource page (or template)
templates/functions.md[.tmpl] Generic function page (or template)
templates/functions/<function name>.md[.tmpl] Function page (or template)
templates/list-resources.md[.tmpl] Generic list resource page (or template)
templates/list-resources/<list resource name>.md[.tmpl] List resource page (or template)
templates/resources.md[.tmpl] Generic resource page (or template)
templates/resources/<resource name>.md[.tmpl] Resource page (or template)

Note: the .tmpl extension is necessary, for the file to be correctly handled as a template.

For examples:

NOTE: In the following conventional paths for examples, <data source name> and <resource name> include the provider prefix as well, but the provider prefix is NOT included in<function name>. For example, the data source caller_identity in the aws provider would have an "example" conventional path of: examples/data-sources/aws_caller_identity/data-source.tf

Path Description
examples/ Root of examples
examples/provider/provider<*>.tf Provider example config(s)
examples/actions/<action_type>/action<*>.tf Action example config(s)
examples/data-sources/<data source name>/data-source<*>.tf Data source example config(s)
examples/ephemeral-resources/<ephemeral resource>/ephemeral-resource<*>.tf Ephemeral resource example config(s)
examples/functions/<function name>/function<*>.tf Function example config(s)
examples/list-resources/<list resource>/list-resource<*>.tfquery.hcl List resource example config(s)
examples/resources/<resource name>/resource<*>.tf Resource example config(s)
examples/resources/<resource name>/import.sh Resource example import command
examples/resources/<resource name>/import-by-string-id.tf Resource example import by id config
examples/resources/<resource name>/import-by-identity.tf Resource example import by identity config

Migration

The migrate subcommand assumes the following conventional paths for the rendered website directory:

NOTE: In the following conventional paths for templates, <data source name>, <resource name>, and <function name> do not include the provider prefix. if the --provider-name argument is set, the provider prefix will be removed from the file names during migration.

Legacy website directory structure:

Path Description
website/ Root of website docs
website/docs/guides Root of guides subdirectory
website/docs/index.html.markdown Docs index page
website/docs/actions/<action type name>.html.markdown Action page
website/docs/d/<data source name>.html.markdown Data source page
website/docs/ephemeral-resources/<ephemeral resource name>.html.markdown Ephemeral resource page
website/docs/functons/<function name>.html.markdown Functions page
website/docs/list-resources/<list resource name>.html.markdown List resource page
website/docs/r/<resource name>.html.markdown Resource page

Docs website directory structure:

Path Description
docs/ Root of website docs
docs/guides Root of guides subdirectory
docs/index.html.markdown Docs index page
docs/actions/<action type name>.html.markdown Action page
docs/data-sources/<data source name>.html.markdown Data source page
docs/ephemeral-resources/<ephemeral resource name>.html.markdown Ephemeral resource page
docs/functions/<function name>.html.markdown Function page
docs/list-resources/<list resource name>.html.markdown List resource page
docs/resources/<resource name>.html.markdown Resource page

Files named index (before the first .) in the website docs root directory and files in the website/docs/d/, website/docs/r/, docs/data-sources/, and docs/resources/ subdirectories will be converted to tfplugindocs templates.

The website/docs/guides/ and docs/guides/ subdirectories will be copied as-is to the --templates-dir folder.

All other files in the conventional paths will be ignored.

Templates

The templates are implemented with Go text/template using the following data fields and functions:

Data fields

Provider Fields
Field Type Description
.Description string Provider description
.HasExample bool (Legacy) Is there an example file?
.HasExamples bool Are there example files? Always true if HasExample is true.
.ExampleFile string (Legacy) Path to the file with the Terraform configuration example.
.ExampleFiles string Paths to the files with the Terraform configuration example. Includes ExampleFile.
.ProviderName string Canonical provider name (ex. terraform-provider-random)
.ProviderShortName string Short version of the rendered provider name (ex. random)
.RenderedProviderName string Value provided via argument --rendered-provider-name, otherwise same as .ProviderName
.SchemaMarkdown string a Markdown formatted Provider Schema definition
Managed Resource / Ephemeral Resource / Data Source Fields
Field Type Description
.Name string Name of the resource/data-source (ex. tls_certificate)
.Type string Either Resource or Data Source
.Description string Resource / Data Source description
.HasExample bool (Legacy) Is there an example file?
.HasExamples bool Are there example files? Always true if HasExample is true.
.ExampleFile string (Legacy) Path to the file with the Terraform configuration example.
.ExampleFiles string Paths to the files with Terraform configuration examples. Includes ExampleFile.
.HasImport bool Is there an import shell file? (terraform import shell example)
.ImportFile string Path to the file with the command for importing the resource
.HasImportIDConfig bool Is there an import terraform config file? (import block example with id)
.ImportIDConfigFile string Path to the file with the Terraform configuration for importing the resource by id
.HasImportIdentityConfig bool Is there an import terraform config file? (import block example with identity)
.ImportIdentityConfigFile string Path to the file with the Terraform configuration for importing the resource by identity
.IdentitySchemaMarkdown string a Markdown formatted Resource Identity Schema definition
.ProviderName string Canonical provider name (ex. terraform-provider-random)
.ProviderShortName string Short version of the rendered provider name (ex. random)
.RenderedProviderName string Value provided via argument --rendered-provider-name, otherwise same as .ProviderName
.SchemaMarkdown string a Markdown formatted Resource / Data Source Schema definition
Provider-defined Function Fields
Field Type Description
.Name string Name of the function (ex. echo)
.Type string Returns Function
.Description string Function description
.Summary string Function summary
.HasExample bool (Legacy) Is there an example file?
.HasExamples bool Are there example files? Always true if HasExample is true.
.ExampleFile string (Legacy) Path to the file with the Terraform configuration example
.ExampleFiles string Paths to the files with Terraform configuration examples. Includes ExampleFile.
.ProviderName string Canonical provider name (ex. terraform-provider-random)
.ProviderShortName string Short version of the rendered provider name (ex. random)
.RenderedProviderName string Value provided via argument --rendered-provider-name, otherwise same as .ProviderName
.FunctionSignatureMarkdown string a Markdown formatted Function signature
.FunctionArgumentsMarkdown string a Markdown formatted Function arguments definition
.HasVariadic bool Does this function have a variadic argument?
.FunctionVariadicArgumentMarkdown string a Markdown formatted Function variadic argument definition
Action Fields
Field Type Description
.Name string Name of the action (ex. examplecloud_do_thing)
.Type string Action
.Description string Action description
.HasExample bool (Legacy) Is there an example file?
.HasExamples bool Are there example files? Always true if HasExample is true.
.ExampleFile string (Legacy) Path to the file with the Terraform configuration example.
.ExampleFiles string Paths to the files with Terraform configuration examples. Includes ExampleFile.
.ProviderName string Canonical provider name (ex. terraform-provider-random)
.ProviderShortName string Short version of the rendered provider name (ex. random)
.RenderedProviderName string Value provided via argument --rendered-provider-name, otherwise same as .ProviderName
.SchemaMarkdown string a Markdown formatted Action Schema definition
List Resource Fields
Field Type Description
.Name string Name of the list resource (ex. examplecloud_thing)
.Type string List Resource
.Description string List resource description
.HasExample bool (Legacy) Is there an example file?
.HasExamples bool Are there example files? Always true if HasExample is true.
.ExampleFile string (Legacy) Path to the file with the terraform configuration example
.ExampleFiles string Paths to the files with terraform configuration examples. Includes ExampleFile. The file extension for ExampleFiles should be *.tfquery.hcl.
.ProviderName string Canonical provider name (ex. terraform-provider-random)
.ProviderShortName string Short version of the rendered provider name (ex. random)
.RenderedProviderName string Value provided via argument --rendered-provider-name, otherwise same as .ProviderName
.SchemaMarkdown string a Markdown formatted list resource Schema definition

Template Functions

Function Description
codefile Create a Markdown code block with the content of a file. Path is relative to the repository root.
lower Equivalent to strings.ToLower.
plainmarkdown Render Markdown content as plaintext.
prefixlines Add a prefix to all (newline-separated) lines in a string.
printf Equivalent to fmt.Printf.
split Split string into sub-strings, by a given separator (ex. split .Name "_").
title Equivalent to cases.Title.
tffile A special case of the codefile function, designed for Terraform files (i.e. .tf).
trimspace Equivalent to strings.TrimSpace.
upper Equivalent to strings.ToUpper.

Disclaimer

This is still under development: while it's being used for production-ready providers, you might still find bugs and limitations. In those cases, please report issues or, if you can, submit a pull-request.

Your help and patience is truly appreciated.

Contributing

License Headers

All source code files in this repository (excluding autogenerated files like go.mod, prose, and files excluded in .copywrite.hcl) must have a license header at the top.

This can be autogenerated by running make generate or running go generate ./... in the /tools directory.

Acceptance Tests

This repo uses the testscript package for acceptance testing. There are two types of acceptance tests: full provider build tests in tfplugindocs/testdata/scripts/provider-build and provider schema json tests in tfplugindocs/testdata/scripts/schema-json.

Provider build tests run the default tfplugindocs command which builds the provider source code and runs Terraform to retrieve the schema. These tests require the full provider source code to build a valid provider binary. Schema json tests run the tfplugindocs command with the --providers-schema=<arg> flag to specify a provider schemas json file. This allows the test to skip the provider build and Terraform CLI call, instead using the specified file to generate docs.

You can run make testacc to run the full suite of acceptance tests. By default, the provider build acceptance tests will create a temporary directory in /tmp/tftmp for testing, but you can change this location in cmd/tfplugindocs/main_test.go. The schema json tests uses the testscript package's default work directory.

The test scripts are defined in the tfplugindocs/testdata/scripts directory. Each script includes the test, golden files, and the provider source code or schema JSON file needed to run the test. Each script is a text archive. You can install the txtar CLI locally by running go install golang.org/x/exp/cmd/txtar@latest to extract the files in the test script for debugging or to help with updating.

For example, to unarchive all of the files from a test script:

# Assuming that your working directory is the root of the terraform-plugin-docs repo

# Create a fresh directory to extract to
mkdir tmp && cd tmp

# Extract this specific test script to the current working directory
txtar -x <../cmd/tfplugindocs/testdata/scripts/provider-build/generate/framework_provider_success_no_templates.txtar

New test scripts

Updates to existing tests are typically just made manually, but the txtar CLI can be used to create a new test quickly.

A new text archive can be created from a folder with:

# Assuming that your working directory is the root of the terraform-plugin-docs repo

# Create a fresh directory to make changes to
mkdir tmp && cd tmp

# Update files in ./tmp that you want in the new .txtar

# Archives all of the files in the current directory with a placeholder comment
echo "placeholder comment" | txtar . >../cmd/tfplugindocs/testdata/scripts/provider-build/generate/new.txtar

Once a new text archive is created, the comment header can be replaced with test script commands like env, cmp, exec, etc. These commands control what actually is run during the test along with the assertions, for example:

# Copyright (c) HashiCorp, Inc.
# SPDX-License-Identifier: MPL-2.0

# Successful run of tfplugindocs on a Framework provider with examples but no templates or pre-existing docs.
[!unix] skip
env GOCACHE=$WORK/gocache  # sets an environment variable
env GOMODCACHE=$WORK/gomodcache  # sets an environment variable
exec tfplugindocs --provider-name=terraform-provider-scaffolding  # executes the Go binary for generating documentation
cmp stdout expected-output.txt  # compares stdout from binary with the golden file in the text archive below
cmp docs/index.md expected-index.md  # compares the generated docs with the golden file in the text archive below
cmp docs/data-sources/example.md expected-datasource.md  # compares the generated docs with the golden file in the text archive below
cmp docs/resources/example.md expected-resource.md  # compares the generated docs with the golden file in the text archive below