DEV Community

Cover image for Infrastructure as Code - Deploy an Azure Function using Bicep
Manuel Kanetscheider
Manuel Kanetscheider

Posted on

4 2

Infrastructure as Code - Deploy an Azure Function using Bicep

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:

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"
      }
    }
  ]
}


Enter fullscreen mode Exit fullscreen mode

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'
  }
}


Enter fullscreen mode Exit fullscreen mode

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:
Azure Function Resources Overview

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}'
view raw StorageAccount.bicep hosted with ❤ by GitHub

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
view raw AppServicePlan.bicep hosted with ❤ by GitHub

⚠ī¸ 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
view raw FunctionApp.bicep hosted with ❤ by GitHub

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
view raw KeyVault.bicep hosted with ❤ by GitHub

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
]
}
view raw main.bicep hosted with ❤ by GitHub

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>


Enter fullscreen mode Exit fullscreen mode

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")'


Enter fullscreen mode Exit fullscreen mode

Parameter file:



az deployment group create \
  --name ExampleDeployment \
  --resource-group ExampleGroup \
  --template-file storage.bicep \
  --parameters @storage.parameters.json


Enter fullscreen mode Exit fullscreen mode

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>


Enter fullscreen mode Exit fullscreen mode

⚠ī¸ 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 😃

Azure Function Resources Overview

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>"
â€Ļ
Enter fullscreen mode Exit fullscreen mode

Imagine monitoring actually built for developers

Billboard image

Join Vercel, CrowdStrike, and thousands of other teams that trust Checkly to streamline monitor creation and configuration with Monitoring as Code.

Start Monitoring

Top comments (0)

Image of AssemblyAI

Automatic Speech Recognition with AssemblyAI

Experience near-human accuracy, low-latency performance, and advanced Speech AI capabilities with AssemblyAI's Speech-to-Text API. Sign up today and get $50 in API credit. No credit card required.

Try the API

👋 Kindness is contagious

Explore a sea of insights with this enlightening post, highly esteemed within the nurturing DEV Community. Coders of all stripes are invited to participate and contribute to our shared knowledge.

Expressing gratitude with a simple "thank you" can make a big impact. Leave your thanks in the comments!

On DEV, exchanging ideas smooths our way and strengthens our community bonds. Found this useful? A quick note of thanks to the author can mean a lot.

Okay