在本地实现一个最简单的 Terraform Provider

前言

Terraform 是一种开源工具,用于安全高效地预览,配置和管理云基础架构和资源。在笔者看来,Terraform 最大的优势在于"万物统一",所有的资源都能够通过编写 HCL 配置文件统一纳管。

Terraform 官方提供了一个框架,而具体的实现由各家厂商接入的 Provider 决定,这就大幅降低了推广成本。如下图所示,当用户执行 terraform cli 命令的时候,与 terraform provider 交互,terraform provider 再去调用云上 OpenAPI。换言之,Terraform 是个插座,各个 Provider 是插在插座上的插头,只要满足插口的规范就能接入进去。

本文关注点在于带着读者快速实现一个最简单的 Provider(无需部署,无需编写 chart 包,也无需上传到仓库),保证读者能够在本地 run 起来,并做适当的代码解释。毕竟笔者一直认为对于初学者而言,建立自信心是很重要的。而对于 Terraform 的高级用法、核心原理,笔者会后后续的文章中提到。

快速实现 Provider

开发环境

yaml 复制代码
Golang: 1.19
系统:Mac OS
开发工具:GoLand
Terraform:1.5.7

快速 Run 起来

代码(无脑复制)

首先进入到 main.go,调用 terraform 官方 sdk,启动监听服务。

Golang 复制代码
package main

import (
   "github.com/hashicorp/terraform-plugin-sdk/plugin"
   "github.com/hashicorp/terraform-plugin-sdk/terraform"
)

func main() {
   plugin.Serve(&plugin.ServeOpts{
      ProviderFunc: func() terraform.ResourceProvider {
         return Provider()
      },
   })
}

其次,实现 provider.go 逻辑,定义资源 example_server。

Golang 复制代码
package main

import (
   "github.com/hashicorp/terraform-plugin-sdk/helper/schema"
)

func Provider() *schema.Provider {
   return &schema.Provider{
      ResourcesMap: map[string]*schema.Resource{
         "example_server": resourceServer(),
      },
   }
}

最后,编写 resource_server.go,实现增删改查的逻辑。

Golang 复制代码
package main

import (
   "fmt"

   "github.com/hashicorp/terraform-plugin-sdk/helper/schema"
)

func resourceServer() *schema.Resource {
   return &schema.Resource{
      Create: resourceServerCreate,
      Read:   resourceServerRead,
      Update: resourceServerUpdate,
      Delete: resourceServerDelete,

      Schema: map[string]*schema.Schema{
         "name": &schema.Schema{
            Type:     schema.TypeString,
            Required: true,
         },
      },
   }
}

func resourceServerCreate(d *schema.ResourceData, m interface{}) error {
   name := d.Get("name").(string)
   d.SetId(name)
   println(fmt.Sprintf("start to create resource %s.", name))
   return resourceServerRead(d, m)
}

func resourceServerRead(d *schema.ResourceData, m interface{}) error {
   name := d.Get("name").(string)
   println(fmt.Sprintf("start to read resource %s.", name))
   return nil
}

func resourceServerUpdate(d *schema.ResourceData, m interface{}) error {
   name := d.Get("name").(string)
   println(fmt.Sprintf("start to update resource %s.", name))
   return resourceServerRead(d, m)
}

func resourceServerDelete(d *schema.ResourceData, m interface{}) error {
   name := d.Get("name").(string)
   println(fmt.Sprintf("start to delete resource %s.", name))
   return nil
}

到这儿后,代码部分已经结束了。代码结构如下。

go 复制代码
- main.go
  provider.go
  resource_server.go

本地跑起来

代码完成后,来看下怎么在本地 run 起来。 首先打包成二进制。

shell 复制代码
go mod init
go mod tidy
go build -o terraform-provider-example

将二进制文件复制到指定路径,因为本地是 mac,所以路径为 darwin_arm64。

shell 复制代码
cp terraform-provider-example ~/.terraform.d/plugins/terraform-example.com/exampleprovider/example/1.0.0/darwin_arm64/

编写 main.tf

HCL 复制代码
terraform {
  required_providers {
    example = {
      version = "~> 1.0.0"
      source  = "terraform-example.com/exampleprovider/example"
    }
  }
}

resource "example_server" "my-server-name" {
  name = "2"
}

执行 terraform init、terraform plan 和 terraform apply。

简单解释下代码

上文已经给出了快速在本地 run 起来的方法,如果没有 run 起来,欢迎评论。

那么本章就来简单解释下代码,具体的 terraform 原理后续文章会补上。

首先,来看 provider.go 文件,里面 ResourcesMap 定义了资源类型,比如这里的 example_server。用户在使用的时候需要在 tf 文件里面将类型定义为 example_server。

Golang 复制代码
ResourcesMap: map[string]*schema.Resource{
   "example_server": resourceServer(),
}

其次,再来看下 resource_server.go。这里实现了增删改查的主逻辑,并且定义了 Schema,也就是字段名称。

Golang 复制代码
func resourceServer() *schema.Resource {
   return &schema.Resource{
      Create: resourceServerCreate,
      Read:   resourceServerRead,
      Update: resourceServerUpdate,
      Delete: resourceServerDelete,

      Schema: map[string]*schema.Schema{
         "name": &schema.Schema{
            Type:     schema.TypeString,
            Required: true,
         },
      },
   }
}

这样在后续逻辑中可以获取用户输入的 "name"。

Golang 复制代码
name := d.Get("name").(string)

同样在创建函数输出了 id 字段,会显示在最终 state 中。

Golang 复制代码
d.SetId(name)

结语

好啦,快速入门实现 terraform provider 到此为止了,遇到问题欢迎在评论区讨论。

相关推荐
czlczl200209252 小时前
Zookeeper原理
分布式·zookeeper·云原生
国医中兴3 小时前
数据稠密计算的内存优化:从理论到实践
微服务·云原生·容器·kubernetes·k8s
新缸中之脑4 小时前
AntSpace:Anthropic的秘密PaaS
云原生·云计算·paas
nvd114 小时前
GCP 无服务器事件驱动实战:使用 Terraform 构建 Pub/Sub 推送至 Cloud Run 的微型调度中心
云原生·serverless·terraform
cyber_两只龙宝4 小时前
【Docker】搭建企业级私有harbor仓库全流程详解
linux·运维·docker·云原生·容器
月亮!6 小时前
6大AI测试工具极限压测:微软TuringAI竟率先崩溃
java·人工智能·python·测试工具·microsoft·云原生·压力测试
国医中兴6 小时前
数据稠密计算的并行处理:从理论到实践
微服务·云原生·容器·kubernetes·k8s
学博成6 小时前
备考“系统架构设计师”
微服务·云原生·架构·架构设计师
白驹过隙不负青春6 小时前
Zookeeper版本升级
分布式·zookeeper·云原生
Bruce20489986 小时前
Go 云原生实战:K8s Operator 开发与服务网格(Istio)落地
云原生·golang·kubernetes