# Terraform模块Module管理,聚合资源的抽取与复用
# 1 简介
最近工作中用到了Terraform,权当学习记录一下,希望能帮助到其它人。
Terraform系列文章如下:
Terraform入门教程,示例展示管理Docker和Kubernetes资源 (opens new window)
Terraform插件Provider管理,搜索、定义、下载 (opens new window)
Terraform状态State管理,让变更有记录 (opens new window)
Terraform模块Module管理,聚合资源的抽取与复用 (opens new window)
Terraform常用命令 (opens new window)
模块是为了便为管理与复用,就跟函数是一样的。一个模块大概有以下文件:
main.tf
:就像是函数入口;README.md
:如函数的声明;variables.tf
: 变量说明,就像函数的入参;outputs.tf
: 输出,如函数的返回值;examples
: 使用示例;
另外,模块也是可以嵌套的,即模块里有子模块,但不建议嵌套太深。
# 2 创建及使用自定义模块
# 2.1 创建模块
我们来通过一个简单示例讲解如何自己创建一个模块。这个模块的功能是在Kubernetes上部署一个nginx,就是创建一个Deployment和一个Service。
这个模块共有两个文件,一个是main.tf
,用来定义Resource,即Deployment和Service。另一个文件是variables.tf
,用来定义这个模块所需要的输入变量。这两个文件都放在当前目录的nginx-kubernetes文件夹下。目录结果如下:
main.tf
文件如下:
resource "kubernetes_deployment" "test" {
metadata {
name = var.applicationName
namespace = var.namespace
}
spec {
replicas = var.replicas
selector {
match_labels = {
app = var.applicationName
}
}
template {
metadata {
labels = {
app = var.applicationName
}
}
spec {
container {
image = var.image
name = "nginx-container"
port {
container_port = 80
}
}
}
}
}
}
resource "kubernetes_service" "test" {
metadata {
name = var.applicationName
namespace = var.namespace
}
spec {
selector = {
app = var.applicationName
}
type = "NodePort"
port {
node_port = var.nodePort
port = 80
target_port = 80
}
}
depends_on = [kubernetes_deployment.test]
}
它就是资源定义,然后把一些变量用var.xxx
的形式替换,这样Terraform解析的时候就会找对应的变量进行赋值。
variables.tf
文件如下:
variable "namespace" {
description = "k8s namespace"
}
variable "applicationName" {
description = "applicationName"
}
variable "image" {
description = "image"
}
variable "replicas" {
description = "deployment replicas"
}
variable "nodePort" {
description = "nodePort"
}
在main.tf
文件使用的变量,都在这里有定义。
# 2.2 使用模块
现在我们已经创建好了模块,接下来要引用它。我们就在当前目录引用即可。代码如下:
provider "kubernetes" {
config_path = "~/.kube/config"
}
module "pkslow-nginx" {
source = "./nginx-kubernetes"
namespace = "pkslow"
applicationName = "pkslow-nginx"
image = "nginx:1.19.5"
replicas = 3
nodePort = 30301
}
source
: 模块来源的地址;namespace
: 命令空间,模块定义的入参;applicationName
: 应用名,模块定义的入参;image
: 镜像,模块定义的入参;replicas
: Pod的数目,模块定义的入参;nodePort
: 端口,模块定义的入参;
引用模块还是很简单的,跟调用函数差不多,就是告诉别人名字和入参。
# 3 使用外部模块
模块的source支持多种类型,如本地路径、Terraform官方仓库、GitHub等。
本地路径:
module "pkslow" {
source = "./pkslow"
}
Terraform仓库:
module "consul" {
source = "hashicorp/consul/aws"
version = "0.1.0"
}
GitHub地址:
module "pkslow" {
source = "github.com/larrydpk/pkslow"
}
如果是用SSH,如下:
module "pkslow" {
source = "git@github.com:larrydpk/pkslow.git"
}
压缩包:
module "vpc" {
source = "https://pkslow.com/vpc-module?archive=zip"
}
# 4 总结
代码请查看:https://github.com/LarryDpk/pkslow-samples
参考: