Skip to content

Latest commit

 

History

History
175 lines (119 loc) · 9.78 KB

architecture.md

File metadata and controls

175 lines (119 loc) · 9.78 KB

Architecture

TFLint is a pluggable linter and does not contain any rule implementations. Rules are provided as plugins, these are launched by TFLint as subprocesses and communicate over gRPC.

An important part of understanding TFLint's architecture is that TFLint (host) and a plugin act as both gRPC server/client.

For example, the plugin (client) to the host (server) requests to:

  • Retrieve Terraform configs (e.g. aws_instance.main)
  • Evaluate expressions (e.g. var.foo)
  • Save reported issues to the host server

The host (client) to the plugin (server) requests to:

  • Apply plugin configs
  • Request to run inspections

The plugin system is implemented by TFLint plugin SDK. If you want to know more about *.proto and detailed gRPC server/client implementation, check out the SDK.

Inspection Flow Diagram

The following diagram explains how an inspection is performed when a user executes command:

flowchart TB
  CLI["CLI<br/>(cmd package)"]-->configLoad["Load TFLint config<br/>(tflint.LoadConfig)"]
  configLoad-->tfLoad["Load Terraform config<br/>(terraform.LoadConfig)"]
  tfLoad-->discoverPlugin["Discover plugins<br/>(plugin.Discovery)"]
  discoverPlugin-->launchPlugin["Launch plugin processes<br/>(go-plugin)"]
  launchPlugin-->startRulesetServer["Start RuleSet gRPC server<br/>(go-plugin)"]
  launchPlugin-->applyPluginConfig["Apply configs to plugins<br/>(ruleset.ApplyConfig)"]
  applyPluginConfig-->requestCheck["Request inspection to plugins<br/>(ruleset.Check)"]
  requestCheck-->startRunnerServer["Start Runner gRPC server<br/>(tflint-plugin-sdk)"]
  requestCheck-->printIssues["Print issues<br/>(formatter.Print)"]
  subgraph goroutie
    startRunnerServer-->respondTerraformConfig["Respond to requests for Terraform config<br/>(plugin.GRPCServer)"]
    respondTerraformConfig-->saveIssues["Save issues emitted by plugins<br/>(plugin.GRPCServer)"]
  end
  subgraph plugin
    startRulesetServer-->appliedPluginConfig["Apply plugin config sent from TFLint<br/>(tflint-plugin-sdk)"]
    appliedPluginConfig-->respondCheck["Respond to inspection request<br/>(tflint-plugin-sdk)"]
    respondCheck-->runInspection["Run inspection on each rule<br/>(tflint-plugin-sdk)"]
    runInspection-->requestTerraformConfig["Request to get Terraform config<br/>(tflint-plugin-sdk)"]
    requestTerraformConfig-->requestEmitIssue["Emit issues to TFLint<br/>(tflint-plugin-sdk)"]
  end
  requestEmitIssue-->printIssues
  saveIssues-->printIssues

CLI (cmd package)

The cmd package is the entrypoint of the CLI. cmd.CLI has streams to stdout/stderr and prints a result to the screen.

Depending on the user's instructions, it does the following:

  • Run inspection
  • Initialize TFLint (install plugins)
  • Start language server
  • Print version info
  • Run bundled plugin (internal use)

This package is responsible for parsing CLI flags and arguments. The parsed cmd.Option is converted to tflint.Config and merged with a config file.

Load TFLint config (tflint.LoadConfig)

The tflint package provides many features related to TFLint, such as loading a config file (.tflint.hcl) and parsing annotations (# tflint-ignore comments).

The tflint.LoadConfig loads a config file and returns tflint.Config. This config will be used in later steps.

Load Terraform config (terraform.LoadConfig)

The terraform package is a fork of github.com/hashicorp/terraform/internal. This package is responsible for processing the Terraform semantics, such as parsing *.tf files, evaluating expressions, and loading modules.

The terraform.LoadConfig reads *.tf files as a terraform.Config in the given directory. These structures are designed to be as similar to Terraform core. See "The Design of terraform Package" section below for details.

Discover plugins (plugin.Discovery)

The plugin package is responsible for the plugin system. This package contains gRPC server implementation, installation, discovery, etc.

The plugin.Discovery discovers installed plugins and launches plugin binaries as subprocesses. This detailed implementation is hidden by github.com/hashicorp/go-plugin.

Launch plugin processes (go-plugin)

The go-plugin launches a plugin binary as a subprocess. The launched plugin acts as a gRPC server and communicates with the host process.

The plugin server is called "RuleSet" server. Its behavior is implemented by a plugin developer.

Apply configs to plugins (ruleset.ApplyConfig)

The plugin.Discovery returns a client for the RuleSet server it started. Use the ApplyConfig method to send the plugin config described in the config file to the server.

Request inspection to plugins (ruleset.Check)

Similarly, send inspection requests to the server. The server responds to requests and runs an inspection, but the plugin needs access to terraform.Config (imagine runner.GetResourceContent).

For this, the host process launches a gRPC server to respond such requests. Ths host server is called "Runner" server. Its behavior is implemented by TFLint. The plugin is passed a client that corresponds to the Runner server.

Respond to requests for Terraform config (plugin.GRPCServer)

The Runner server responds to requests from plugins to retrieve Terraform configs, evaluate expressions, etc. This implementation is contained in the plugin package.

Save issues emitted by plugins (plugin.GRPCServer)

The Runner server saves issues emitted by plugins (imagine runner.EmitIssue). The saved issues will be printed to the screen in the next step.

Print issues (formatter.Print)

The formatter package processes and outputs issues in formats such as default, JSON, and SARIF.

Inspection Sequence Diagram

The following diagram explains how the host process, RuleSet server, and Runner server each behave in inspection:

sequenceDiagram
  participant TFLint as TFLint (host)
  participant RuleSet as RuleSet (plugin)
  participant Runner as Runner (host as goroutie)
  TFLint->>RuleSet: Start server
  TFLint->>RuleSet: Apply plugin configs
  TFLint->>Runner: Start server
  TFLint->>RuleSet: Request to run inspection
  RuleSet->>Runner: Request to get Terraform configs
  Runner-->>RuleSet: Return resources/modules
  RuleSet->>Runner: Request to evaluate expressions
  Runner-->>RuleSet: Return evaluated values
  RuleSet->>Runner: Emit issues
  RuleSet-->>TFLint: End of inspection
  Runner-->>TFLint: Return issues

The Design of terraform Package

The terraform package is a fork of Terraform internal packages. It is based on the same Terraform core, but with some implementation changes specifically for static analysis.

The following diagram explains the dependencies of each component:

flowchart TB
  files["Terraform config files<br/>(*.tf)"]-->parser["Language parser<br/>(terraform.Parser)"]
  subgraph child module
    modFiles["Terraform config files<br/>(*.tf)"]-->modParser["Language parser<br/>(terraform.Parser)"]
  end
  modParser-->modWalker["Module walker<br/>(terraform.ModuleWalker)"]  
  parser-- terraform.Module -->loader["Config loader<br/>(terraform.LoadConfig)"]
  modWalker-- terraform.Module -->loader
  loader-- terraform.Config -->gatherVars["Gather variables<br/>(terraform.VariableValues)"]
  valuesFile["Variable definitions files<br/>(*.tfvars)"]-- terraform.InputValues -->gatherVars
  cliFlag["CLI flags<br/>(--var)"]-- terraform.InputValues -->gatherVars
  gatherVars-- cty.Value -->evaluator["Evaluator<br/>(terraform.Evaluator)"]
  loader-- terraform.Config -->evaluator

HCL and cty

The underlying technologies of Terraform language are HCL and cty. The Terraform language implements its own semantics on top of these.

TFLint also leverages these technologies, making it highly compatible with Terraform.

Differences from Terraform

The basic architecture is the same as Terraform. The biggest difference is that the terraform.Evaluator is state (terraform.State) independent. See also Terraform Core Architecture Summary .

State is not always available for static analysis as it can only be obtained by running terraform plan/apply. TFLint solves this problem in the same way as Terraform. Terraform has the ability to handle unknown values during first-time planning, and TFLint take advantage of this to always treat dynamic values as unknowns.

It has the same basic design as Terraform, so it has the advantage of being able to easily support future language extensions of Terraform.

Lazy Schema Evaluation

Terraform has a predefined schema and uses it to decode HCL body. This is necessary to strictly define the syntax, but TFLint does not necessarily require a strict schema in order to support many versions of the Terraform language.

Against this background, terraform.Module in TFLint decodes minimal structure (e.g. terraform.Variable, terraform.Resource). The terraform.Module has native hcl.File and returns bodies based on the schema requested during inspection. In other words, it does lazy schema evaluation to achieve syntactic robustness.

For this reason, TFLint take a different approach to implementing for_each and count meta-arguments, and dynamic blocks than Terraform. Instead of being pre-expanded, it is expanded as an extension of hcl.Body when extracted by the schema.