I want to create my resource groups and my virtual network in this part of the blog series. To create these resources will use a combination of Bicep files, where I split my code into an execution file with parameters and a generic template file. I am doing this to not limit my options for deploying environments. In fact, the last post in this series will be to deploy the same environment but switching the execution code with a PowerShell script.

The first thing I want to do is creating a new branch in my GitHub repository. I will do this from VS Code by clicking the branch icon in the bottom left corner and then selecting “Create new branch” in the top midsection of the screen.

I will name my branch “RG_and_vNet”

I always verify that the branch I just created is also the branch I am working in. I do this by checking the branch name in the lower-left corner.

I can now update my resource group file to contain the resource groups I want to create. I have done this in the code below. Finally, I will save the updated ResourceGroup.bicep file and commit it to GitHub.

targetScope = 'subscription'
param Location string = 'WestEurope'
param companyPrefix string = 'bicep'

var ResourceGroups = [  
  'rg-${companyPrefix}-sharedservices-network-001'
  'rg-${companyPrefix}-sharedservices-vm-001'
  'rg-${companyPrefix}-citrix-network-001'
  'rg-${companyPrefix}-citrix-vm-001'
  'rg-${companyPrefix}-citrix-workers-001'  
]

resource resourcegroups 'Microsoft.Resources/resourceGroups@2021-01-01' = [for ResourceGroup in ResourceGroups: {
  location: Location
  name: ResourceGroup
}]

Next, I want to create my Bicep file for the virtual network in Azure. I found the documentation for the template here, but it contains more options that I want to use, so after I removed all the items I don’t want to use, it looks like this.

param Name string 
param Prefix string 
param Subnets array
param Location string
param dnsServers array

resource vnet 'Microsoft.Network/virtualNetworks@2021-02-01' = {
  name: Name
  location: Location
  properties: {
    addressSpace: {
      addressPrefixes: [
        Prefix
      ]
    }
    dhcpOptions: {
      dnsServers: dnsServers
    }
    subnets: [for Subnet in Subnets: {
      name: Subnet.name
      properties: {
        addressPrefix: Subnet.Prefix
      }
    }]
  }
}
output name string = vnet.name

The code above will create a virtual network with custom DNS servers and an array of subnets. I will save this file in a new folder called “Templates” placed in a new folder called “Network”. I like to organize my code this way to easier find the files I want to edit.

I now have the template for my virtual network, and I am now ready to create the execution code for the virtual networks I want to create. First, I will create a new Bicep file called “DeployNetwork.bicep” and place it in the “Network” folder. I find it easiest to write the modules first and then my parameters and variables afterward. I will create two modules, one for each vNet I want to create. In the code below, you can see that I have an entry in each module for each parameter I created in my template file. I will normally use variables or parameters for my execution code, but this is not strictly needed. I have two modules, one for a Citrix vNet and one for Shared Services vNet. Each of the modules has a scope set for a resource group. The bicep deployment can be scoped to either management group, subscription, or resource group. Since I want to deploy into two resource groups, I need to target my execution file at the subscription level.

module CitrixvNet './Templates/vNet.bicep' = {
  name: 'Citrix-vNet-Deployment'
  params: {
    dnsServers: Citrix_vNet_dnsServers
     Location: Location
     Subnets: Citrix_vNet_Subnets
     Name: Citrix_vNetName
     Prefix: Citrix_vNet_Prefix
  }
  dependsOn: [
    SharedservicevNet
  ]
  scope: resourceGroup(Citrix_ResourceGroup)
}

module SharedservicevNet './Templates/vNet.bicep' = {
  name: 'Sharedservice-vNet-Deployment'
  params: {
    dnsServers: Sharedservice_vNet_dnsServers
     Location: Location
     Subnets: Sharedservice_vNet_Subnets
     Name: Sharedservice_vNet_Name
     Prefix: Sharedservice_vNet_Prefix
  }
  scope: resourceGroup(Sharedservice_ResourceGroup)
}

Now that I have the modules in place, I can see which parameters and variables I need to create, so I will go ahead and do that. Below, I put the targetScope as the first line to indicate that this deployment is targeted at the subscription level. Next, I have a parameter called “companyPrefix”, which has a default value of “bicep”. I created this parameter to allow the execution of different environments based on either the value in the file or a value coming from the Github Actions execution. This could be from a RestAPI call, for instance.

targetScope = 'subscription'
param companyPrefix string = 'bicep'
var Location = 'WestEurope'
var Sharedservice_ResourceGroup = 'rg-${companyPrefix}-sharedservices-network-001'
var Sharedservice_vNet_Name = 'vnet-${companyPrefix}-sharedservices-001'
var Sharedservice_vNet_Prefix = '172.16.0.0/16'
var Sharedservice_vNet_Subnets = [
  {
    name: 'GatewaySubnet'
    prefix: '172.16.0.0/26'
  }
  {
    name: 'snet-sharedservices-adds-001'
    prefix: '172.16.0.64/26'
  }      
]
var Sharedservice_vNet_dnsServers = [
  '192.168.10.10'
]
var Citrix_ResourceGroup = 'rg-${companyPrefix}-citrix-network-001'
var Citrix_vNetName = 'vnet-${companyPrefix}-citrix-001'
var Citrix_vNet_Prefix = '172.17.0.0/16'
var Citrix_vNet_Subnets = [
  {
    name: 'snet-citrix-vm-001'
    prefix: '172.17.0.0/26'
  }
  {
    name: 'snet-citrix-workers-001'
    prefix: '172.17.1.0/24'
  }
  {
    name: 'snet-citrix-workers-002'
    prefix: '172.17.2.0/24'
  }
]
var Citrix_vNet_dnsServers = [
  '192.168.10.10'
]

I now have the code for my virtual networks ready, and before I move on, I want to ensure that it is running as expected using Github actions. First, I will update my deploy.yml file to include the virtual network deployment, so the file now looks like the code below. Notice that I have changed the branch for push triggers to run for my working branch and not the main branch. Normally, I have actions run on either a development branch or the main branch, but this is easier for the current scenario since we are testing this branch.

name: CI

# Controls when the action will run. 
on:
  # Triggers the workflow on push or pull request events but only for the main branch
  push:
    branches: [ RG_and_vNet ]
  pull_request:
    branches: [ main ]

  # Allows you to run this workflow manually from the Actions tab
  workflow_dispatch:

# A workflow run is made up of one or more jobs that can run sequentially or in parallel
jobs:
  build-and-deploy:
    runs-on: ubuntu-latest
    steps:
    - uses: actions/checkout@master

    - name: Login to Azure
      uses: azure/login@v1
      with:
        creds: ${{ secrets.AZURE_CREDENTIALS_BLOGSERIES }}

    - name: Deploy resource group
      uses: azure/arm-deploy@main
      with:
        scope: subscription
        subscriptionId: ${{ secrets.AZURE_SUBSCRIPTION }}
        region: westeurope
        template: ./ResourceGroup/ResourceGroup.bicep

    - name: Deploy virtual network
      uses: azure/arm-deploy@main
      with:
        scope: subscription
        subscriptionId: ${{ secrets.AZURE_SUBSCRIPTION }}
        region: westeurope
        template: ./Network/DeployNetwork.bicep

I now have the following resource groups in Azure and I also have the virtual networks as shown below.

I now know that my code can deploy my resource groups and virtual networks, so the last thing I want to cover in this part of the series is peering the two virtual networks together. So, I will create a new template under my network folder and call it “Peering.bicep”. I found the documentation on network peering here , and I adjusted that to fit my needs, and that created the code below. Notice that I have a lookup for my remote virtual network to make it easier to read and deploy the code.

param virtualNetworkName string
param allowForwardedTraffic bool = true
param allowGatewayTransit bool = false
param allowVirtualNetworkAccess bool = true
param useRemoteGateways bool = true
param remoteResourceGroup string
param remoteVirtualNetworkName string

resource remotevnet 'Microsoft.Network/virtualNetworks@2021-02-01' existing = {
  name: remoteVirtualNetworkName
  scope: resourceGroup(remoteResourceGroup)  
}

resource peering 'Microsoft.Network/virtualNetworks/virtualNetworkPeerings@2021-02-01' = {
  name: '${virtualNetworkName}/Peering-To-${remoteVirtualNetworkName}'
  properties: {
    allowForwardedTraffic: allowForwardedTraffic
    allowGatewayTransit: allowGatewayTransit
    allowVirtualNetworkAccess: allowVirtualNetworkAccess
    useRemoteGateways: useRemoteGateways
    remoteVirtualNetwork: {
      id: remotevnet.id
    }
  }
}

Next, I will add the peering modules to the “DeployNetwork.bicep” file to have all my virtual network configurations in one deployment file. I have added the lines below into the deployment file and notice that I use the output from the virtual network deployment module as input to my peering module.

module CitrixPeering './Templates/Peering.bicep' = {
  name: 'CitrixvNetPeering'
  params: {
    allowForwardedTraffic: true
    allowGatewayTransit: false
    allowVirtualNetworkAccess: true
    remoteResourceGroup: 'rg-${companyPrefix}-sharedservices-network-001'
    remoteVirtualNetworkName: 'vnet-${companyPrefix}-sharedservices-001'
    useRemoteGateways: false
    virtualNetworkName: CitrixvNet.outputs.name
  }
  dependsOn: [
    CitrixvNet
  ]
  scope: resourceGroup(Citrix_ResourceGroup)
}
module SharedservicePeering './Templates/Peering.bicep' = {
  name: 'SharedServicevNetPeering'
  params: {
    allowForwardedTraffic: true
    allowGatewayTransit: true
    allowVirtualNetworkAccess: true
    remoteResourceGroup: 'rg-${companyPrefix}-citrix-network-001'
    remoteVirtualNetworkName: 'vnet-${companyPrefix}-citrix-001'
    useRemoteGateways: false
    virtualNetworkName: SharedservicevNet.outputs.name
  }
  dependsOn: [
    CitrixPeering
  ]
  scope: resourceGroup(Sharedservice_ResourceGroup)
}

I can now save the two new files and commit them to my repository. I don’t need to change anything for the Github action since I updated the deployment file I already configured. After the deployment, I can see that my peering is working.

Before I close this part of the blog series, I will merge my changes into the main branch and delete the working branch for the network. This is done in Github by going to “Pull requests” and then click on “Compare & pull request”.

I fill out the fields as shown below with new title and description.

Github will list my commits and run tests to ensure that my code won’t conflict with the existing code. Finally, when all the lights are green, I will click on “Merge pull request”.

When the pull request has been merged, I will delete the branch I merge code from.

This will end this part of the blog series. Next, I will create the VPN connection from Azure to my on-premises lab environment.

Stay tuned.