Introduction
In this guide, we will look at the topic Infrastructure as Code using the example of an Azure Function. But what is Infrastructure as Code? Based from the official Microsoft documentation, Infrastructure as Code can be described as follows:
Infrastructure as Code (IaC) is the management of infrastructure (networks, virtual machines, load balancers, and connection topology) in a descriptive model, using the same versioning as DevOps team uses for source code. Like the principle that the same source code generates the same binary, an IaC model generates the same environment every time it is applied. IaC is a key DevOps practice and is used in conjunction with continuous delivery.
A key property of IaC is the so-called Idempotence, which means that the deployment command always brings the target system into the desired state, regardless of the initial state. This can be achieved either by configuring the target system or by creating a fresh environment.
In simpler terms, IaC ensures that we always have the desired configuration of our target system. In classic on-prem scenarios, the target systems are unique, e.g. staging and production systems may differ. These differences can cause the application to work on the staging system but then fail on the production system (e.g. missing permissions, network access, etc.). IaC tackles this issues and helps to reduce such risks. Additionally, since the required infrastructure is now available as code, it can be easily versioned via Git.
For IaC various tools are around:
- Azure Resource Templates
- Azure Bicep
- Terraform
- Pulumi
- Chef
- Puppet
- others...
The built-in tools for Azure are ARM and Bicep Templates, the other tools listed are third party tools for which additional costs may apply.
In this guide we will cover only ARM and (mainly) Bicep. ARM and Bicep offer the advantage that they are developed directly by Microsoft and therefore always have access to the latest features. The only downside is that these technologies are not suitable for multicloud scenarios, as they are developed exclusively for the Microsoft Azure Cloud.
Also, another plus point for ARM and Bicep, both are free and fully supported by Microsoft đ.
ARM vs. Bicep
ARM and Bicep both offer basically the same functionalities, but Bicep is a newer technology compared to ARM, so it can happen that some ARM features are not yet available in Bicep (for example like User-defined ARM Functions).
ARM templates are JSON files, all resources are represented as JSON. Bicep on the other hand is a declarative language, comparable to Terraform. Let's have a look:
Example ARM Template:
{
"$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentTemplate.json#",
"contentVersion": "1.0.0.0",
"parameters": {
"location": {
"type": "string",
"defaultValue": "[resourceGroup().location]"
},
"storageAccountName": {
"type": "string",
"defaultValue": "[format('toylaunch{0}', uniqueString(resourceGroup().id))]"
}
},
"resources": [
{
"type": "Microsoft.Storage/storageAccounts",
"apiVersion": "2021-06-01",
"name": "[parameters('storageAccountName')]",
"location": "[parameters('location')]",
"sku": {
"name": "Standard_LRS"
},
"kind": "StorageV2",
"properties": {
"accessTier": "Hot"
}
}
]
}
Example Bicep Template:
param location string = resourceGroup().location
param storageAccountName string = 'toylaunch${uniqueString(resourceGroup().id)}'
resource storageAccount 'Microsoft.Storage/storageAccounts@2021-06-01' = {
name: storageAccountName
location: location
sku: {
name: 'Standard_LRS'
}
kind: 'StorageV2'
properties: {
accessTier: 'Hot'
}
}
I think it's obvious that the Bicep template is significantly shorter and even if you've never worked with either technology, it's much easier to see what's going on inside the Bicep template than in the ARM template.
In my opinion, Bicep offers the following advantages over ARM templates:
đ Less code
đ Much easier to read
đ Fully integrated into VS Code with awesome intellisense capabilities
Even if you need ARM templates, you can compile Bicep into ARM templates. This actually happens in the background when you deploy a Bicep template to Azure. During the deployment the Bicep template is compiled into an ARM template and then the ARM template is submitted to Azure.
It is also possible to decompile an ARM template and thus convert an ARM to a Bicep template.
Personally, I think if you don't have a strong reason to use ARM templates, you should definitely use Bicep, as the syntax is much simpler and more readable.
Let's get started
Perquisites
- Azure Account with an active subscription. In case you do not have an Azure Account, go ahead and create one for free here
- Azure CLI
- Bicep Tools for Azure CLI and VS Code
âšī¸ It is absolutely possible to create the Bicep templates with the editor of your choice, however, I strongly recommend you to try the VS Code Extension for Bicep as it greatly simplifies the template development.
Azure Function Resources Overview
Before we actually getting started, let's take a look at what resources we typically need for an Azure Function:
Function App
This is the actual "Function App" service, where the code is running.
App Service Plan
The App Service Plan is comparable to a web server and describes how the Function App is hosted. The App Plan also specifies how the Function App scales, the resources available per instance, and which "advanced" features can be used, such as Azure Virtual Network connectivity. In addition, different prices apply depending on the choice of the hosting plan. The following plans are available:
- Consumption Plan (= serverless)
- Premium Plan
- Dedicated hosting plan
For more information, please checkout the official documentation on Azure Functions hosting options.
Storage Account
When using the Consumption/Premium hosting plan, the function code and binding configuration files are stored in Azure Files in the main storage account. In addition, certain platform features may use the storage account for internal operations, such as Azure Durable Functions.
Fore more information, please checkout the official documentation on Azure Function Storage Considerations.
Application Insights (optional)
Application Insights is the preferred monitoring service of Azure Functions and offers built-in logging functionalities. Application Insights collects log, performance, and error data. By automatically detecting performance anomalies and featuring powerful analytics tools, you can more easily diagnose issues and better understand how your functions are used.
âšī¸ Application insights is an optional service that is billed separately. However, the cost are very low and it is highly recommended to use it with your Function App.
For more information, please checkout the official documentation on Monitoring Azure Functions and Azure Application Insights.
Key Vault (optional)
Azure Key Vault is a cloud service for securely storing and accessing secrets. A secret is anything to which access should be strictly controlled, such as API keys, passwords, certificates, or cryptographic keys.
Azure Key Vault Secrets can only be accessed via a secured and authenticated connection. In the context of the Azure Function, a system or user assigned identity must be created with which the Azure Function authenticates itself to the Key Vault service. The permissions must then be set up on the Key Vault in form of an access policy.
âšī¸ Azure Key Vault is an optional service and is primarily used for secure storage and retrieval of sensitive data. Storing sensitive data directly in the function is not recommended, as this data is stored in plain text, meaning that anyone who has access to the storage account can read this information. In addition to secrets, certificates and keys can also be stored in the Key Vault, and the Key Vault also offers various other functionality such as the encryption of data via public and private key. This service is also charged separately.
To learn more about Azure Key Vault or on how to Integrate Azure Key Vault into your Azure Function, please checkout the official documentation.
Templates
Now that we know what resources we need for our function, we can proceed with the development of the templates.
Storage Account
This template creates the Storage Account, for later use the connection string is constructed and exported. The connection string is needed later in order to connect the Function App to the storage account.
As briefly mentioned before, certain function features, such as Durable Functions, require certain Storage Services. If further storage services are required, they can be added to this template.
@description('Storage Account name') | |
@minLength(3) | |
@maxLength(24) | |
param name string | |
@description('Storage Account location') | |
param location string | |
@description('Storage Account SKU name') | |
@allowed([ | |
'Standard_LRS' | |
'Standard_GRS' | |
'Standard_RAGRS' | |
'Standard_ZRS' | |
'Premium_LRS' | |
'Premium_ZRS' | |
'Standard_GZRS' | |
'Standard_RAGZRS' | |
]) | |
param sku string | |
resource storageAccount 'Microsoft.Storage/storageAccounts@2021-08-01' = { | |
name: name | |
location: location | |
kind: 'StorageV2' | |
sku: { | |
name: sku | |
} | |
properties: { | |
accessTier: 'Hot' | |
supportsHttpsTrafficOnly: true | |
encryption: { | |
keySource: 'Microsoft.Storage' | |
services: { | |
file: { | |
keyType: 'Account' | |
enabled: true | |
} | |
blob: { | |
keyType: 'Account' | |
enabled: true | |
} | |
} | |
} | |
} | |
} | |
var accountName = storageAccount.name | |
var endpointSuffix = environment().suffixes.storage | |
var key = listKeys(storageAccount.id, storageAccount.apiVersion).keys[0].value | |
output storageAccountConnectionString string = 'DefaultEndpointsProtocol=https;AccountName=${accountName};EndpointSuffix=${endpointSuffix};AccountKey=${key}' |
Application Insights
Creates the Application Insights Resource for monitoring the Function App. To connect the Application Insights instance to the Function App, the Instrumentation Key must be exported for later assignment.
@description('Application Insights name') | |
param name string | |
@description('Application Insights location') | |
param location string | |
var kind = 'web' | |
resource applicationInsights 'Microsoft.Insights/components@2020-02-02' = { | |
name: name | |
location: location | |
kind: kind | |
properties: { | |
Application_Type: kind | |
publicNetworkAccessForIngestion: 'Enabled' | |
publicNetworkAccessForQuery: 'Enabled' | |
} | |
} | |
output applicationInsightsKey string = reference(applicationInsights.id, applicationInsights.apiVersion).InstrumentationKey |
App Service Plan
Deploys the App Service Plan of the Function, here the plan id is exported for later usage.
@description('App Service Plan name') | |
param name string | |
@description('App Service Plan location') | |
param location string | |
@description('App Service Plan operating system') | |
@allowed([ | |
'Windows' | |
'Linux' | |
]) | |
param os string | |
var reserved = os == 'Linux' ? true : false | |
resource appServicePlan 'Microsoft.Web/serverfarms@2021-03-01' = { | |
name: name | |
location: location | |
kind: 'functionapp' | |
sku: { | |
name: 'Y1' | |
} | |
properties: { | |
reserved: reserved | |
} | |
} | |
output planId string = appServicePlan.id |
â ī¸ The operating system varies depending on the language, please make sure to use the correct OS based on the chosen programming language, for more details please checkout the official documentation.
Function App (without settings)
Deploys the function app without settings, here also the previously exported plan id is passed into the template (the plan id specifies the selected hostion option of the Azure Function).
Also a Managed Identity is assigned, as discussed before this is needed for the authentication against the Key Vault. The identity information, such as principal id and tenant id, as well as the name of the Function App are exported for later use.
@description('Function App name') | |
param name string | |
@description('Function App location') | |
param location string | |
@description('App Service Plan Id') | |
param planId string | |
var kind = 'functionapp' | |
resource functionApp 'Microsoft.Web/sites@2021-03-01' = { | |
name: name | |
location: location | |
kind: kind | |
properties: { | |
serverFarmId: planId | |
} | |
identity: { | |
type: 'SystemAssigned' | |
} | |
} | |
output functionAppName string = functionApp.name | |
output principalId string = functionApp.identity.principalId | |
output tenantId string = functionApp.identity.tenantId |
Key Vault
Creates the Key Vault. The previously configured managed identity is passed to the template and the necessary permissions for this managed identity are then set up using an access policy. In addition, any required secrets are created here and then exported for later referencing.
@description('Key Vault name') | |
param name string | |
@description('Key vault location') | |
param location string | |
@description('Key Vault SKU') | |
@allowed([ | |
'standard' | |
'premium' | |
]) | |
param sku string | |
@description('Function App principal id') | |
param funcPrincipalId string | |
@description('Function App tenant id') | |
param funcTenantId string | |
@description('Key Vault Secret: Database Connection String') | |
@secure() | |
param databaseConnectionString string | |
resource keyVault 'Microsoft.KeyVault/vaults@2021-10-01' = { | |
name: name | |
location: location | |
properties: { | |
tenantId: subscription().tenantId | |
enabledForTemplateDeployment: true | |
sku: { | |
family: 'A' | |
name: sku | |
} | |
accessPolicies: [ | |
{ | |
objectId: funcPrincipalId | |
tenantId: funcTenantId | |
permissions: { | |
secrets: [ | |
'get' | |
] | |
} | |
} | |
] | |
} | |
} | |
resource databaseConnectionStringSecret 'Microsoft.KeyVault/vaults/secrets@2021-10-01' = { | |
name: '${name}/databaseConnectionString' | |
properties: { | |
value: databaseConnectionString | |
} | |
dependsOn: [ | |
keyVault | |
] | |
} | |
output databaseConnectionStringSecretUri string = databaseConnectionStringSecret.properties.secretUri |
Function App Settings
Creates the Function App Settings. Here various settings of the Function App are set, such as the Storage Account, the Application Insights instance, Function App runtime and any other custom parameters including secrets.
@description('Function App name') | |
param functionAppName string | |
@description('Function App runtime') | |
@allowed([ | |
'dotnet' | |
'node' | |
'python' | |
'java' | |
]) | |
param functionAppRuntime string | |
@description('Application Insights Instrumentation Key') | |
@secure() | |
param applicationInsightsKey string | |
@description('Storage Account connection string') | |
@secure() | |
param storageAccountConnectionString string | |
@description('Key Vault URI Connection String reference') | |
@secure() | |
param databaseConnectionString string | |
var function_extension_version = '~4' | |
var databaseConnectionStringKeyVaultRef = '@Microsoft.KeyVault(SecretUri=${databaseConnectionString})' | |
resource functionAppSettings 'Microsoft.Web/sites/config@2021-03-01' = { | |
name: '${functionAppName}/appsettings' | |
properties: { | |
AzureWebJobsStorage: storageAccountConnectionString | |
WEBSITE_CONTENTAZUREFILECONNECTIONSTRING: storageAccountConnectionString | |
WEBSITE_CONTENTSHARE: toLower(functionAppName) | |
FUNCTIONS_EXTENSION_VERSION: function_extension_version | |
APPINSIGHTS_INSTRUMENTATIONKEY: applicationInsightsKey | |
FUNCTIONS_WORKER_RUNTIME: functionAppRuntime | |
//WEBSITE_TIME_ZONE only available on windows | |
WEBSITE_ADD_SITENAME_BINDINGS_IN_APPHOST_CONFIG: 1 | |
DatabaseConnectionString: databaseConnectionStringKeyVaultRef | |
} | |
} |
Main
In order to create the function this template must be deployed. This template glues everything together and deploys all previously created templates in the correct order. The dependencies of the resources are also defined here, e.g. the function app should only be deployed if the deployment of the storage account, the application insights instance and the app service plan was successful.
@description('Resources location') | |
param location string = resourceGroup().location | |
//----------- Storage Account Parameters ------------ | |
@description('Function Storage Account name') | |
@minLength(3) | |
@maxLength(24) | |
param storageAccountName string | |
@description('Function Storage Account SKU') | |
@allowed([ | |
'Standard_LRS' | |
'Standard_GRS' | |
'Standard_RAGRS' | |
'Standard_ZRS' | |
'Premium_LRS' | |
'Premium_ZRS' | |
'Standard_GZRS' | |
'Standard_RAGZRS' | |
]) | |
param storageAccountSku string = 'Standard_LRS' | |
//----------- Application Insights Parameters ------------ | |
@description('Application Insights name') | |
param applicationInsightsName string | |
//----------- Function App Parameters ------------ | |
@description('Function App Plan name') | |
param planName string | |
@description('Function App Plan operating system') | |
@allowed([ | |
'Windows' | |
'Linux' | |
]) | |
param planOS string | |
@description('Function App name') | |
param functionAppName string | |
@description('Function App runtime') | |
@allowed([ | |
'dotnet' | |
'node' | |
'python' | |
'java' | |
]) | |
param functionAppRuntime string | |
//----------- Key Vault Parameters ------------ | |
@description('Key Vault name') | |
param keyVaultName string | |
@description('Key Vault SKU') | |
@allowed([ | |
'standard' | |
'premium' | |
]) | |
param keyVaultSku string = 'standard' | |
@description('Database Connection String') | |
@secure() | |
param databaseConnectionString string | |
var buildNumber = uniqueString(resourceGroup().id) | |
//----------- Storage Account Deployment ------------ | |
module storageAccountModule 'templates/StorageAccount.bicep' = { | |
name: 'stvmdeploy-${buildNumber}' | |
params: { | |
name: storageAccountName | |
sku: storageAccountSku | |
location: location | |
} | |
} | |
//----------- Application Insights Deployment ------------ | |
module applicationInsightsModule 'templates/ApplicationInsights.bicep' = { | |
name: 'appideploy-${buildNumber}' | |
params: { | |
name: applicationInsightsName | |
location: location | |
} | |
} | |
//----------- App Service Plan Deployment ------------ | |
module appServicePlan 'templates/AppServicePlan.bicep' = { | |
name: 'plandeploy-${buildNumber}' | |
params: { | |
name: planName | |
location: location | |
os: planOS | |
} | |
} | |
//----------- Function App Deployment ------------ | |
module functionAppModule 'templates/FunctionApp.bicep' = { | |
name: 'funcdeploy-${buildNumber}' | |
params: { | |
name: functionAppName | |
location: location | |
planId: appServicePlan.outputs.planId | |
} | |
dependsOn: [ | |
storageAccountModule | |
applicationInsightsModule | |
appServicePlan | |
] | |
} | |
//----------- Key Vault Deployment ------------ | |
module keyVaultModule 'templates/KeyVault.bicep' = { | |
name: 'kvdeploy-${buildNumber}' | |
params: { | |
name: keyVaultName | |
location: location | |
sku: keyVaultSku | |
funcTenantId: functionAppModule.outputs.tenantId | |
funcPrincipalId: functionAppModule.outputs.principalId | |
databaseConnectionString: databaseConnectionString | |
} | |
dependsOn: [ | |
functionAppModule | |
] | |
} | |
//----------- Function App Settings Deployment ------------ | |
module functionAppSettingsModule 'templates/FunctionAppSettings.bicep' = { | |
name: 'siteconf-${buildNumber}' | |
params: { | |
applicationInsightsKey: applicationInsightsModule.outputs.applicationInsightsKey | |
databaseConnectionString: keyVaultModule.outputs.databaseConnectionStringSecretUri | |
functionAppName: functionAppModule.outputs.functionAppName | |
functionAppRuntime: functionAppRuntime | |
storageAccountConnectionString: storageAccountModule.outputs.storageAccountConnectionString | |
} | |
dependsOn: [ | |
functionAppModule | |
] | |
} |
Deployment
For creating the function, a resource group is required in advance. If the desired resource group is already present, this step can be skipped.
az group create -n <name> -l <location>
For the deployment various parameters have to be specified including:
- Location
- Storage Account name
- Storage Account SKU
- Application Insights name
- App Service Plan name
- Function App operating system
- Function App name
- Function App runtime
We have two options for specifying the parameters:
Inline parameters:
az deployment group create \
--resource-group testgroup \
--template-file <path-to-bicep> \
--parameters exampleString='inline string' exampleArray='("value1", "value2")'
Parameter file:
az deployment group create \
--name ExampleDeployment \
--resource-group ExampleGroup \
--template-file storage.bicep \
--parameters @storage.parameters.json
Since we need have to specify a few parameters I suggest that we go with the parameter file variant. A parameter also offers the following advantages:
- Is part of the repository and can therefore be versioned in Git
- A separate configuration can be created for each environment (e.g. staging, production, etc.)
- Facilitates deployment via CI/CD pipelines
The parameter looks like this, please just adjust the parameters as needed:
{ | |
"$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#", | |
"contentVersion": "1.0.0.0", | |
"parameters": { | |
"storageAccountName": { | |
"value": "stvm<storage-account-name>" | |
}, | |
"storageAccountSku": { | |
"value": "Standard_LRS" | |
}, | |
"applicationInsightsName": { | |
"value": "appi-<application-insights-name>" | |
}, | |
"planName": { | |
"value": "plan-<plan-name>" | |
}, | |
"planOS": { | |
"value": "Linux" | |
}, | |
"functionAppName": { | |
"value": "func-<plan-name>" | |
}, | |
"functionAppRuntime": { | |
"value": "python" | |
}, | |
"keyVaultName": { | |
"value": "kv-<key-vault-name>" | |
}, | |
"keyVaultSku": { | |
"value": "standard" | |
}, | |
"databaseConnectionString": { | |
"value": "" | |
} | |
} | |
} |
âšī¸ For Azure there are recommendations from Microsoft on how resources should be named, for more details please checkout Recommended abbreviations for Azure resource types. Of course, these are only recommendations, resources can be named in any way you like.
To deploy the Azure Function we can use the following command:
az deployment group create \
--name <deployment-name> \
--resource-group <resource-group-name> \
--template-file bicep\main.bicep
--parameters @<path-to-parameters>
â ī¸ For some names there are certain limitations, e.g. special characters, character lengths, etc. For example, the name of a storage account must be between 3 and 24 characters, should not contain any special characters and the name must be globally unique. If an error occurs during deployment, please read the error message carefully and adjust the names if necessary.
That's it! Deployment takes a few seconds and then the Azure Function is ready to go.
Conclusion
Infrastructure as Code is an extremely powerful tool and an important building block that paves the way for further DevOps steps, such as the automated creation and configuration of environments via CI/CD mechanics.
On top of that, Microsoft offers with Bicep and VS Code two excellent tools with which the creation of templates is a breeze.
Even if we have only looked at the tip of the iceberg, I hope I could awaken your interest and that you liked my blog post.
Thanks for reading, if you have any questions feel free to leave a comment đ
Here you can find the repository with the full source code:
Deploy an Azure Function using Bicep
This project illustrates how an Azure Function can be deployed as IaC via Bicep.
Description
The project provides the following Bicep templates:
- Function App (without settings)
- App Service Plan
- Storage Account
- Application Insights
- Key Vault
- Function App Settings
The individual resources can be found under the bicep/templates folder, these are all linked and deployed from the main.bicep file.
For more detailed information, please checkout my blog post đ
Getting Started
Perquisites
- Azure Account with an active subscription. In case you do not have an Azure Account, go ahead and create one for free here
- Azure CLI
- Bicep Tools for Azure CLI and VS Code
How to deploy the templates
Create a parameter file
The example parameter file looks like this:
{
"$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentParameters.json#"
"contentVersion": "1.0.0.0",
"parameters": {
"storageAccountName": {
"value": "stvm<storage-account-name>"
âĻ
Top comments (0)