重用配置在 Terraform 中创建类似的资源

Posted

技术标签:

【中文标题】重用配置在 Terraform 中创建类似的资源【英文标题】:Reuse the configuration to create similar resources in Terraform 【发布时间】:2020-11-24 07:16:23 【问题描述】:

我有下面的 TF 文件,它将创建一个函数 - FirstFunction。这完美无缺。

resource "azurerm_function_app" "**firstfunction**" 
  name                = **var.firstfunctionname**
  location            = azurerm_resource_group.resourcegroupX.location
  resource_group_name = azurerm_resource_group.resourcegroupX.name
  app_service_plan_id = azurerm_app_service_plan.appserviceplan.id
  https_only          = "true"
  client_affinity_enabled = "true"
  
  app_settings =    
    NS                     = azurerm_eventhub_namespace.eventhubns.name
    Hub                    = azurerm_eventhub.**firsteventhub**.name    
    propertyX       = "**firstproperty**"
    LogRef                       = "$azurerm_storage_account.store.primary_blob_endpoint$azurerm_storage_container.**firstlogs**.name"
 
 
 
 resource "azurerm_app_service_virtual_network_swift_connection" "**firstvnet**" 
  app_service_id = azurerm_function_app.**firstfunction**.id
  subnet_id      = azurerm_subnet.snet.id


在文件中看到用****括起来的部分,需要改成创建SecondFunction、ThirdFunction等...

我现在的方法是创建多个 TF 文件,复制相同的代码并更改 ** 中包含的部分。

我通读了模块系统,但了解模块系统的限制是我无法引用在同一个 TF 根模块中创建的其他组件,如下所示 例如,在 TF 文件中,我将位置称为 位置 = azurerm_resource_group.resourcegroupX.location 如果我将其作为一个模块进行,则该位置应称为 location = var.location_name 其中 location_name 应定义为变量。我无法引用使用相同根模块创建的组件。

您能否提出一个解决方案,让我可以根据类似的代码创建多个组件?请注意,在上面的示例中,我在一个 TF 文件中创建了 2 个资源,并且它们都是相关的。

【问题讨论】:

【参考方案1】:

最简单的方法是在资源中使用 count 属性。它可以帮助您在同一代码中创建多个相同的资源。

resource "azurerm_function_app" "myfunction" 
  count               = number_your_need     # how many resources you want to create 
  name                = "$var.firstfunctionname-$count.index" 
  location            = azurerm_resource_group.resourcegroupX.location
  resource_group_name = azurerm_resource_group.resourcegroupX.name
  app_service_plan_id = azurerm_app_service_plan.appserviceplan.id
  https_only          = "true"
  client_affinity_enabled = "true"
  
  app_settings =    
    NS                     = azurerm_eventhub_namespace.eventhubns.name
    Hub                    = azurerm_eventhub.**firsteventhub**.name   
    propertyX       = "**firstproperty**"
    LogRef                       = "$azurerm_storage_account.store.primary_blob_endpoint$azurerm_storage_container.**firstlogs**.name"
   
 
 
 resource "azurerm_app_service_virtual_network_swift_connection" "**firstvnet**" 
  count          = number      # how many you need to create
  app_service_id = element(azurerm_function_app.myfunction[*].id, count.index)
  subnet_id      = azurerm_subnet.snet.id


您需要为azurerm_eventhub 创建相同的解决方案。例如:

resource "azurerm_eventhub" "myeventhub" 
  count  = number    # how many you need to create 
  name   = "$var.eventhub_name-$count.index"
  ...

然后你可以像这样在函数应用中引用它:

app_settings =    
    NS                     = azurerm_eventhub_namespace.eventhubns.name
    Hub                    = element(azurerm_eventhub.myeventhub[*].name, count.index)  
    propertyX       = "**firstproperty**"
    LogRef                       = "$azurerm_storage_account.store.primary_blob_endpoint$azurerm_storage_container.**firstlogs**.name"
 

所有用 **** 括起来的部分也是如此。

【讨论】:

上面的代码可以用for_each写吗?上述 sn-p 的问题在于,函数的 app_settings 的任何更改都会导致该函数被删除并重新创建。根据文档,如果我们使用 count 进行循环,terraform 会删除并重新创建组件。 @Vaya 的源码,可以使用for_each。但它的结果是一样的。而且需要创建多个资源,为什么要更改app_settings?不设置在创建时间? @Vaya 如果你使用for_each,它没有count index。

以上是关于重用配置在 Terraform 中创建类似的资源的主要内容,如果未能解决你的问题,请参考以下文章

在同一资源组中创建两个 VM,但 Terraform 不希望销毁第一个

Terraform 在 AKS 节点资源组中创建入口应用程序网关

Terraform 在子 AWS 账户中创建资源

Terraform - 尝试使用服务主体在 Azure 中创建资源并从 keyvault 中提取该 SP 密钥

通过 terraform 在 aws 中创建 VPC 问题

如何在 terraform 中创建地图