Understand the structure and syntax of Bicep files
Article
This article describes the structure and syntax of a Bicep file. It presents the different sections of the file and the properties that are available in those sections.
Bicep is a declarative language, which means the elements can appear in any order. Unlike imperative languages, the order of elements doesn't affect how deployment is processed.
The following example shows an implementation of these elements.
metadata description = 'Creates a storage account and a web app'
@description('The prefix to use for the storage account name.')
@minLength(3)
@maxLength(11)
param storagePrefix string
param storageSKU string = 'Standard_LRS'
param location string = resourceGroup().location
var uniqueStorageName = '${storagePrefix}${uniqueString(resourceGroup().id)}'
resource stg 'Microsoft.Storage/storageAccounts@2023-04-01' = {
name: uniqueStorageName
location: location
sku: {
name: storageSKU
}
kind: 'StorageV2'
properties: {
supportsHttpsTrafficOnly: true
}
}
module webModule './webApp.bicep' = {
name: 'webDeploy'
params: {
skuName: 'S1'
location: location
}
}
Metadata
Metadata in Bicep is an untyped value that you can include in your Bicep files. Metadata provides supplementary information about your Bicep files, like name, description, author, and creation date.
Target scope
By default, the target scope is set to resourceGroup. If you deploy at the resource-group level, you don't need to set the target scope in your Bicep file.
Use this decorator to make sure the user provides correct values. This decorator is permitted only on param statements. To declare that a property must be one of a set of predefined values in a type or output statement, use union type syntax. You can also use union type syntax in param statements.
Elevate BCP089 from a warning to an error when a property name of a user-defined data type is likely a typo. For more information, see Elevate error level.
Marks the parameter as secure. The value for a secure parameter isn't saved to the deployment history and isn't logged. For more information, see Secure strings and objects.
Parameters
Use parameters for values that need to vary for different deployments. You can define a default value for the parameter that's used if no value is provided during deployment.
For example, you can add an SKU parameter to specify different sizes for a resource. You might pass in different values depending on whether you're deploying to test or production.
param storageSKU string = 'Standard_LRS'
The parameter is available for use in your Bicep file.
sku: {
name: storageSKU
}
You can add one or more decorators for each parameter. For more information, see Use decorators.
To make your Bicep file more readable, encapsulate complex expressions in a variable. For example, you might add a variable for a resource name that's constructed by concatenating several values together.
var uniqueStorageName = '${storagePrefix}${uniqueString(resourceGroup().id)}'
Apply this variable wherever you need the complex expression.
In your Bicep file, you can create your own functions and also use the standard Bicep functions that are automatically available within your Bicep files. Create your own functions when you have complicated expressions that are used repeatedly in your Bicep files.
Use the resource keyword to define a resource to deploy. Your resource declaration includes a symbolic name for the resource. You use this symbolic name in other parts of the Bicep file to get a value from the resource.
The resource declaration includes the resource type and API version. Within the body of the resource declaration, include properties that are specific to the resource type.
Some resources have a parent/child relationship. You can define a child resource either inside the parent resource or outside of it.
The following example shows how to define a child resource within a parent resource. It contains a storage account with a child resource (file service) that's defined within the storage account. The file service also has a child resource (share) that's defined within it.
The next example shows how to define a child resource outside of the parent resource. You use the parent property to identify a parent/child relationship. The same three resources are defined.
Modules enable you to reuse code from a Bicep file in other Bicep files. In the module declaration, you link to the file to reuse. When you deploy the Bicep file, the resources in the module are also deployed.
The symbolic name enables you to reference the module from somewhere else in the file. For example, you can get an output value from a module by using the symbolic name and the name of the output value.
You can add one or more decorators for each module. For more information, see Use decorators.
Use outputs to return values from the deployment. Typically, you return a value from a deployed resource when you need to reuse that value for another operation.
You can add a resource or module to your Bicep file that's conditionally deployed. During deployment, the condition is evaluated and the result determines whether the resource or module is deployed. Use the if expression to define a conditional deployment.
Use // for single-line comments or /* ... */ for multiline comments.
The following example shows a single-line comment.
// This is your primary NIC.
resource nic1 'Microsoft.Network/networkInterfaces@2023-11-01' = {
...
}
The following example shows a multiline comment.
/*
This Bicep file assumes the key vault already exists and
is in same subscription and resource group as the deployment.
*/
param existingKeyVaultName string
Multi-line strings
You can break a string into multiple lines. Use three single quotation marks ''' to start and end the multi-line string.
Characters within the multi-line string are handled as is. Escape characters are unnecessary. You can't include ''' in the multi-line string. String interpolation isn't currently supported.
You can start your string right after the opening ''' or include a new line. In either case, the resulting string doesn't include a new line. Depending on the line endings in your Bicep file, new lines are interpreted as \r\n or \n.
The following example shows a multi-line string.
var stringVar = '''
this is multi-line
string with formatting
preserved.
'''
The preceding example is equivalent to the following JSON.
"variables": {
"stringVar": "this is multi-line\r\n string with formatting\r\n preserved.\r\n"
}
Multiple-line declarations
You can now use multiple lines in function, array, and object declarations. This feature requires Bicep CLI version 0.7.X or higher.
In the following example, the resourceGroup() definition is broken into multiple lines.
var foo = resourceGroup(
mySubscription,
myRgName)
For multiple-line declaration samples, see arrays and objects.
Known limitations
No support is available for the concept of apiProfile, which is used to map a single apiProfile to a set apiVersion for each resource type.
User-defined functions aren't supported at the moment. An experimental feature is currently accessible. For more information, see User-defined functions in Bicep.
Some Bicep features require a corresponding change to the intermediate language (Azure Resource Manager JSON templates). We announce these features as available after all the required updates are deployed to global Azure. If you use a different environment, such as Azure Stack, there might be a delay in the availability of the feature. The Bicep feature is available only after the intermediate language is also updated in that environment.
Build end-to-end solutions in Microsoft Azure to create Azure Functions, implement and manage web apps, develop solutions utilizing Azure storage, and more.