Managing dependencies in ARM template
Managing dependencies in an Azure Resource Manager (ARM) template is crucial for ensuring that resources are deployed in the correct order, particularly when one resource depends on another. ARM templates allow you to specify and manage these dependencies to ensure the right sequence of actions is followed during deployment.
In ARM templates, dependencies can be managed using the dependsOn
property, which ensures that certain resources are created or updated before others. Let's explore how to use dependsOn
effectively and also look at implicit dependencies.
1. dependsOn
Property
The dependsOn
property explicitly defines the order in which resources are deployed in an ARM template. This is useful when one resource depends on another, such as when a virtual machine (VM) requires a network interface card (NIC) to be created first, or a virtual network (VNet) must be deployed before a subnet.
Syntax of dependsOn
xxxxxxxxxx
31"dependsOn": [
2 "<resourceId>"
3]
The dependsOn
property is an array of resource IDs that defines the order of resource deployment.
If a resource in the dependsOn
array fails, the deployment of the dependent resource will not proceed.
Example of dependsOn
Suppose you want to deploy a virtual machine that depends on a virtual network and a network interface. Here's how you would define that dependency:
xxxxxxxxxx
601"resources": [
2 {
3 "type": "Microsoft.Network/virtualNetworks",
4 "apiVersion": "2023-03-01",
5 "location": "East US",
6 "name": "myVnet",
7 "properties": {
8 "addressSpace": {
9 "addressPrefixes": ["10.0.0.0/16"]
10 }
11 }
12 },
13 {
14 "type": "Microsoft.Network/networkInterfaces",
15 "apiVersion": "2023-03-01",
16 "location": "East US",
17 "name": "myNIC",
18 "dependsOn": [
19 "[resourceId('Microsoft.Network/virtualNetworks', 'myVnet')]"
20 ],
21 "properties": {
22 "ipConfigurations": [
23 {
24 "name": "ipconfig1",
25 "properties": {
26 "subnet": {
27 "id": "[resourceId('Microsoft.Network/virtualNetworks/subnets', 'myVnet', 'default')]"
28 }
29 }
30 }
31 ]
32 }
33 },
34 {
35 "type": "Microsoft.Compute/virtualMachines",
36 "apiVersion": "2023-03-01",
37 "location": "East US",
38 "name": "myVM",
39 "dependsOn": [
40 "[resourceId('Microsoft.Network/networkInterfaces', 'myNIC')]"
41 ],
42 "properties": {
43 "hardwareProfile": {
44 "vmSize": "Standard_DS1_v2"
45 },
46 "osProfile": {
47 "computerName": "myVM",
48 "adminUsername": "adminuser",
49 "adminPassword": "password123"
50 },
51 "networkProfile": {
52 "networkInterfaces": [
53 {
54 "id": "[resourceId('Microsoft.Network/networkInterfaces', 'myNIC')]"
55 }
56 ]
57 }
58 }
59 }
60]
Explanation:
Virtual Network (
myVnet
): This resource is created first, and there's nodependsOn
because it's the first resource.Network Interface (
myNIC
): This depends on the virtual networkmyVnet
(specified in thedependsOn
array), as the NIC needs to be associated with a subnet in the virtual network.Virtual Machine (
myVM
): This depends on the network interfacemyNIC
(specified in thedependsOn
array), because the VM needs the NIC to be created and associated with it.
2. Implicit Dependencies
In addition to dependsOn
, ARM templates also have implicit dependencies. Azure implicitly knows the order in which resources should be deployed based on their references to other resources.
Example of Implicit Dependency
Consider the following example where a virtual machine depends on a network interface, but no dependsOn
property is used:
xxxxxxxxxx
431"resources": [
2 {
3 "type": "Microsoft.Network/networkInterfaces",
4 "apiVersion": "2023-03-01",
5 "location": "East US",
6 "name": "myNIC",
7 "properties": {
8 "ipConfigurations": [
9 {
10 "name": "ipconfig1",
11 "properties": {
12 "subnet": {
13 "id": "[resourceId('Microsoft.Network/virtualNetworks/subnets', 'myVnet', 'default')]"
14 }
15 }
16 }
17 ]
18 }
19 },
20 {
21 "type": "Microsoft.Compute/virtualMachines",
22 "apiVersion": "2023-03-01",
23 "location": "East US",
24 "name": "myVM",
25 "properties": {
26 "hardwareProfile": {
27 "vmSize": "Standard_DS1_v2"
28 },
29 "osProfile": {
30 "computerName": "myVM",
31 "adminUsername": "adminuser",
32 "adminPassword": "password123"
33 },
34 "networkProfile": {
35 "networkInterfaces": [
36 {
37 "id": "[resourceId('Microsoft.Network/networkInterfaces', 'myNIC')]"
38 }
39 ]
40 }
41 }
42 }
43]
In this case:
The network interface (
myNIC
) is created before the virtual machine (myVM
), because the VM references the NIC via thenetworkInterfaces
property.Azure will implicitly understand that the network interface must be deployed before the VM, even though no
dependsOn
is explicitly provided.
3. Circular Dependencies
Circular dependencies occur when two or more resources depend on each other, creating an endless loop. ARM templates do not allow circular dependencies, and you will receive an error if such a situation is detected.
For example, consider the following scenario:
Resource A depends on Resource B.
Resource B depends on Resource A.
This is a circular dependency, and ARM will not allow it.
To avoid circular dependencies:
Reassess the resources' relationships.
Ensure that dependencies are only unidirectional (i.e., one resource depends on another, not vice versa).
4. Implicit Ordering of Resources
Although resources are deployed in parallel by default, ARM ensures that resources that have dependencies (either implicit or explicit) are deployed in the correct order.
For example:
If a virtual machine requires a storage account, the storage account will be created before the virtual machine, even if no
dependsOn
is explicitly defined.If two resources are independent (i.e., they don’t reference each other), they will be deployed in parallel, unless otherwise specified by
dependsOn
.
5. Using dependsOn
with Loops or Multiple Resources
In more complex templates, you may want to use loops (like copy
loops) to deploy multiple similar resources, and sometimes the order of those resources matters.
For example, deploying multiple NICs and ensuring they are created before VMs:
xxxxxxxxxx
541"resources": [
2 {
3 "type": "Microsoft.Network/networkInterfaces",
4 "apiVersion": "2023-03-01",
5 "location": "East US",
6 "name": "[concat('nic', copyIndex())]",
7 "copy": {
8 "name": "nicLoop",
9 "count": 3
10 },
11 "properties": {
12 "ipConfigurations": [
13 {
14 "name": "ipconfig1",
15 "properties": {
16 "subnet": {
17 "id": "[resourceId('Microsoft.Network/virtualNetworks/subnets', 'myVnet', 'default')]"
18 }
19 }
20 }
21 ]
22 }
23 },
24 {
25 "type": "Microsoft.Compute/virtualMachines",
26 "apiVersion": "2023-03-01",
27 "location": "East US",
28 "name": "[concat('myVM', copyIndex())]",
29 "dependsOn": [
30 "[resourceId('Microsoft.Network/networkInterfaces', concat('nic', copyIndex()))]"
31 ],
32 "copy": {
33 "name": "vmLoop",
34 "count": 3
35 },
36 "properties": {
37 "hardwareProfile": {
38 "vmSize": "Standard_DS1_v2"
39 },
40 "osProfile": {
41 "computerName": "[concat('myVM', copyIndex())]",
42 "adminUsername": "adminuser",
43 "adminPassword": "password123"
44 },
45 "networkProfile": {
46 "networkInterfaces": [
47 {
48 "id": "[resourceId('Microsoft.Network/networkInterfaces', concat('nic', copyIndex()))]"
49 }
50 ]
51 }
52 }
53 }
54]
Here:
The dependsOn
ensures that each virtual machine depends on the NIC created in the same copy loop.
6. Summary
Managing dependencies in ARM templates is essential for ensuring that resources are deployed in the correct order. You can use the dependsOn
property to explicitly manage dependencies, but implicit dependencies often make this less necessary. For complex scenarios, such as loops or conditional dependencies, dependsOn
can help you define precise deployment sequences. Circular dependencies should be avoided, and careful consideration of resource relationships will ensure smooth deployments.
Leave a Reply