Events
May 19, 6 PM - May 23, 12 AM
Calling all developers, creators, and AI innovators to join us in Seattle @Microsoft Build May 19-22.
Register todayThis browser is no longer supported.
Upgrade to Microsoft Edge to take advantage of the latest features, security updates, and technical support.
This article shows you how to use the for
syntax to iterate over items in a collection. This functionality is supported starting in v0.3.1 onward. You can use loops to define multiple copies of a resource, module, variable, property, or output. Use loops to avoid repeating syntax in your Bicep file and to dynamically set the number of copies to create during deployment. See Quickstart: Create multiple resource instances in Bicep for a quickstart of how to use different for
syntaxes to create multiple resource instances in Bicep.
To use loops to create multiple resources or modules, each instance must have a unique value for the name
property. You can use the index value or unique values in arrays or collections to create the names.
For step-by-step guidance about loops, see the Build flexible Bicep templates by using conditions and loops module in Microsoft Learn.
Loops can be declared by:
Using an integer index. This option works when your scenario is: "I want to create this many instances." The range function creates an array of integers that begins at the start index and contains the number of specified elements. Within the loop, you can use the integer index to modify values. For more information, see Integer index.
[for <index> in range(<startIndex>, <numberOfElements>): {
...
}]
Using items in an array: This option works when your scenario is, "I want to create an instance for each element in an array." Within the loop, you can use the value of the current array element to modify values. For more information, see Array elements.
[for <item> in <collection>: {
...
}]
Using items in a dictionary object: This option works when your scenario is, "I want to create an instance for each item in an object." The items function converts the object to an array. Within the loop, you can use properties from the object to create values. For more information, see Dictionary object.
[for <item> in items(<object>): {
...
}]
Using integer index and items in an array: This option works when your scenario is, "I want to create an instance for each element in an array, but I also need the current index to create another value." For more information, see Loop array and index.
[for (<item>, <index>) in <collection>: {
...
}]
Adding a conditional deployment: This option works when your scenario is, "I want to create multiple instances, but I only want to deploy each instance when a condition is true." For more information, see Loop with condition.
[for <item> in <collection>: if(<condition>) {
...
}]
Using loops in Bicep has these limitations:
map
function.For a simple example of using an index, create a variable that contains an array of strings:
param itemCount int = 5
var stringArray = [for i in range(0, itemCount): 'item${(i + 1)}']
output arrayResult array = stringArray
The output returns an array with the following values:
[
"item1",
"item2",
"item3",
"item4",
"item5"
]
The next example creates the number of storage accounts specified in the storageCount
parameter. It returns three properties for each storage account:
param location string = resourceGroup().location
param storageCount int = 2
resource storageAcct 'Microsoft.Storage/storageAccounts@2023-05-01' = [for i in range(0, storageCount): {
name: '${i}storage${uniqueString(resourceGroup().id)}'
location: location
sku: {
name: 'Standard_LRS'
}
kind: 'Storage'
}]
output storageInfo array = [for i in range(0, storageCount): {
id: storageAcct[i].id
blobEndpoint: storageAcct[i].properties.primaryEndpoints.blob
status: storageAcct[i].properties.statusOfPrimary
}]
Notice the index i
is used in creating the storage account resource name.
The next example deploys a module multiple times:
param location string = resourceGroup().location
param storageCount int = 2
var baseName = 'store${uniqueString(resourceGroup().id)}'
module stgModule './storageAccount.bicep' = [for i in range(0, storageCount): {
name: '${i}deploy${baseName}'
params: {
storageName: '${i}${baseName}'
location: location
}
}]
output storageAccountEndpoints array = [for i in range(0, storageCount): {
endpoint: stgModule[i].outputs.storageEndpoint
}]
The following example creates one storage account for each name provided in the storageNames
parameter. Note the name property for each resource instance must be unique:
param location string = resourceGroup().location
param storageNames array = [
'contoso'
'fabrikam'
'coho'
]
resource storageAcct 'Microsoft.Storage/storageAccounts@2023-05-01' = [for name in storageNames: {
name: '${name}${uniqueString(resourceGroup().id)}'
location: location
sku: {
name: 'Standard_LRS'
}
kind: 'Storage'
}]
The next example iterates over an array to define a property. It creates two subnets within a virtual network. Note the subnet names must be unique:
param rgLocation string = resourceGroup().location
var subnets = [
{
name: 'api'
subnetPrefix: '10.144.0.0/24'
}
{
name: 'worker'
subnetPrefix: '10.144.1.0/24'
}
]
resource vnet 'Microsoft.Network/virtualNetworks@2023-11-01' = {
name: 'vnet'
location: rgLocation
properties: {
addressSpace: {
addressPrefixes: [
'10.144.0.0/20'
]
}
subnets: [for subnet in subnets: {
name: subnet.name
properties: {
addressPrefix: subnet.subnetPrefix
}
}]
}
}
The following example uses both the array element and index value when defining the storage account:
param storageAccountNamePrefix string
var storageConfigurations = [
{
suffix: 'local'
sku: 'Standard_LRS'
}
{
suffix: 'geo'
sku: 'Standard_GRS'
}
]
resource storageAccountResources 'Microsoft.Storage/storageAccounts@2023-05-01' = [for (config, i) in storageConfigurations: {
name: '${storageAccountNamePrefix}${config.suffix}${i}'
location: resourceGroup().location
sku: {
name: config.sku
}
kind: 'StorageV2'
}]
The next example uses both the elements of an array and an index to output information about the new resources:
param location string = resourceGroup().location
param orgNames array = [
'Contoso'
'Fabrikam'
'Coho'
]
resource nsg 'Microsoft.Network/networkSecurityGroups@2023-11-01' = [for name in orgNames: {
name: 'nsg-${name}'
location: location
}]
output deployedNSGs array = [for (name, i) in orgNames: {
orgName: name
nsgName: nsg[i].name
resourceId: nsg[i].id
}]
To iterate over elements in a dictionary object, use the items
function, which converts the object to an array. Use the value
property to get properties on the objects. Note the nsg resource names must be unique.
param nsgValues object = {
nsg1: {
name: 'nsg-westus1'
location: 'westus'
}
nsg2: {
name: 'nsg-east1'
location: 'eastus'
}
}
resource nsg 'Microsoft.Network/networkSecurityGroups@2023-11-01' = [for nsg in items(nsgValues): {
name: nsg.value.name
location: nsg.value.location
}]
For resources and modules, you can add an if
expression with the loop syntax to conditionally deploy the collection.
The following example shows a loop combined with a condition statement. In this example, a single condition is applied to all instances of the module:
param location string = resourceGroup().location
param storageCount int = 2
param createNewStorage bool = true
var baseName = 'store${uniqueString(resourceGroup().id)}'
module stgModule './storageAccount.bicep' = [for i in range(0, storageCount): if(createNewStorage) {
name: '${i}deploy${baseName}'
params: {
storageName: '${i}${baseName}'
location: location
}
}]
The next example shows how to apply a condition that's specific to the current element in the array:
resource parentResources 'Microsoft.Example/examples@2024-06-06' = [for parent in parents: if(parent.enabled) {
name: parent.name
properties: {
children: [for child in parent.children: {
name: child.name
setting: child.settingValue
}]
}
}]
Azure resources are deployed in parallel by default. When you use a loop to create multiple instances of a resource type, those instances are all deployed at the same time. The order in which they're created isn't guaranteed. There isn't a limit to the number of resources deployed in parallel other than the total limit of 800 resources in the Bicep file.
You might not want to update all instances of a resource type at the same time. For example, when updating a production environment, you may want to stagger the updates for only a certain number to update at any one time. You can specify for a subset of the instances to be batched together and deployed at the same time. The other instances wait for that batch to complete.
To serially deploy instances of a resource, add the batchSize
decorator. Set its value to the number of instances to deploy concurrently. A dependency is created during earlier instances in the loop, so it doesn't start one batch until the previous batch completes.
param location string = resourceGroup().location
@batchSize(2)
resource storageAcct 'Microsoft.Storage/storageAccounts@2023-05-01' = [for i in range(0, 4): {
name: '${i}storage${uniqueString(resourceGroup().id)}'
location: location
sku: {
name: 'Standard_LRS'
}
kind: 'Storage'
}]
For sequential deployment, set the batch size to 1.
The batchSize
decorator is in the sys namespace. If you need to differentiate this decorator from another item with the same name, preface the decorator with sys: @sys.batchSize(2)
To create more than one instance of a child resource, both of the following Bicep files support this task:
Nested child resources
param location string = resourceGroup().location
resource stg 'Microsoft.Storage/storageAccounts@2023-05-01' = {
name: 'examplestorage'
location: location
kind: 'StorageV2'
sku: {
name: 'Standard_LRS'
}
resource service 'fileServices' = {
name: 'default'
resource share 'shares' = [for i in range(0, 3): {
name: 'exampleshare${i}'
}]
}
}
Top-level child resources
resource stg 'Microsoft.Storage/storageAccounts@2023-05-01' = {
name: 'examplestorage'
location: resourceGroup().location
kind: 'StorageV2'
sku: {
name: 'Standard_LRS'
}
}
resource service 'Microsoft.Storage/storageAccounts/fileServices@2023-05-01' = {
name: 'default'
parent: stg
}
resource share 'Microsoft.Storage/storageAccounts/fileServices/shares@2023-05-01' = [for i in range(0, 3): {
name: 'exampleshare${i}'
parent: service
}]
The Azure Resource Manager template (ARM template) references
function returns an array of objects that represent a resource collection's runtime states. Since there isn't an explicit references
function in Bicep and symbolic collection usage is employed directly, Bicep translates it to an ARM template that utilizes the ARM template references
function while code generates. For the translation feature that use the references
function to transform symbolic collections into ARM templates, it's necessary to have Bicep CLI version 0.20.X or higher. Additionally, in the bicepconfig.json file, the symbolicNameCodegen
setting should be presented and set to true
.
The outputs of the two samples in Integer index can be written as:
param location string = resourceGroup().location
param storageCount int = 2
resource storageAcct 'Microsoft.Storage/storageAccounts@2023-05-01' = [for i in range(0, storageCount): {
name: '${i}storage${uniqueString(resourceGroup().id)}'
location: location
sku: {
name: 'Standard_LRS'
}
kind: 'Storage'
}]
output storageInfo array = map(storageAcct, store => {
blobEndpoint: store.properties.primaryEndpoints
status: store.properties.statusOfPrimary
})
output storageAccountEndpoints array = map(storageAcct, store => store.properties.primaryEndpoints)
This Bicep file is transpiled into the following ARM JSON template that uses the references
function:
{
"$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentTemplate.json#",
"languageVersion": "1.10-experimental",
"contentVersion": "1.0.0.0",
"parameters": {
"location": {
"type": "string",
"defaultValue": "[resourceGroup().location]"
},
"storageCount": {
"type": "int",
"defaultValue": 2
}
},
"resources": {
"storageAcct": {
"copy": {
"name": "storageAcct",
"count": "[length(range(0, parameters('storageCount')))]"
},
"type": "Microsoft.Storage/storageAccounts",
"apiVersion": "2023-04-01",
"name": "[format('{0}storage{1}', range(0, parameters('storageCount'))[copyIndex()], uniqueString(resourceGroup().id))]",
"location": "[parameters('location')]",
"sku": {
"name": "Standard_LRS"
},
"kind": "Storage"
}
},
"outputs": {
"storageInfo": {
"type": "array",
"value": "[map(references('storageAcct', 'full'), lambda('store', createObject('blobEndpoint', lambdaVariables('store').properties.primaryEndpoints, 'status', lambdaVariables('store').properties.statusOfPrimary)))]"
},
"storageAccountEndpoints": {
"type": "array",
"value": "[map(references('storageAcct', 'full'), lambda('store', lambdaVariables('store').properties.primaryEndpoints))]"
}
}
}
Note in the preceding ARM JSON template, languageVersion
must be set to 1.10-experimental
, and the resource element is an object instead of an array.
To learn how to create Bicep files, see Bicep file structure and syntax.
Events
May 19, 6 PM - May 23, 12 AM
Calling all developers, creators, and AI innovators to join us in Seattle @Microsoft Build May 19-22.
Register todayTraining
Module
Build flexible Bicep templates by using conditions and loops - Training
Build advanced Bicep templates by using conditions to conditionally deploy resources and using loops to deploy multiple instances of resources.