Monday, June 14, 2021

How-To: Automated Company-Wide IP Blocking via Azure Firewall and Azure Functions

 

Summary 

One of the top 10 Azure consuming companies has multiple country government-mandated requirements to block egress to and ingress from IP addresses and IP address ranges on a dynamic embargoed/sanctioned IP list. In other words, various country governments across the globe forbid the company from sending traffic to certain addresses.

  • This is across more than 20 Azure regions and over 4,000 Subscriptions for dozens of lines of business 
  • This includes regions with existing Azure Firewalls, those with a 3rd party Network Virtual Appliance functioning as a firewall, and regions containing both Web Application Gateway and an Azure Firewall (see Figure 1 below).
  • This list of embargoed IPs is currently 5,000 address ranges and could grow to 75,000 over the next 2-3 years.
  • The company's global security group provides these addresses in a text file and mandates that changes to these IP addresses should be applied into production within 24 hours.
  • They also mandate that there should be no lapse in coverage when the list is updating (no period of time a blocked IP could get through).

 

The company needed a very agile solution that could translate a dynamically changing list of IP addresses and CIDR ranges into active policy applied globally across any Azure region and can scale to huge numbers of subscriptions. It needed to be very low maintenance, work quickly, and not allow for bypassing. 

 

Solution Summary: The Microsoft CE DSE team came up with a solution leveraging Azure Firewall Policy and Azure Function Apps to enforce a parent Network Rule policy that is based on automatically updating IP Groups to control traffic to these embargoed locations across their global enterprise. 

 

Details 

Global Network 

The global network has a disparate blend of network topologies, managed by dozens of lines of business (LoB). Some have both an App Gateway publishing websites and an Azure Firewall controlling general traffic (Scenario 1, below). Others have only an Azure Firewall (Scenario 2), and a handful have 3rd party Network Virtual Appliances (NVA) controlling ingress and egress (Scenario 3). In each of these scenarios, Azure Firewall policy is used to enforce the network traffic. Additionally, Azure Firewall Manager, could be added into the mix to push this policy from a single, central location to multiple firewalls (however, it does incur additional cost). Representative sample network topologies are shown in Figure 1, below. 

 

Figure 1Figure 1

 

Solution Elements 

The building blocks of the solution are as follows in Figure 2 below. The placement of resources into separate Resource Groups as implemented is not a necessity. 

 

Storage 

A storage account containing a blob storage container is required as the target location for the controlled IP address list. 

 

Azure Function App 

The function app provides the automation behind the solution. It is configured to watch and trigger off any text files getting dropped into the blob storage container. It runs a simple PowerShell script that runs the logic to split the IP addresses into the desired number of pieces, and then runs Set-AzIPGroup in parallel processing to assign those ranges to the IP Groups.  

 

Azure Firewall 

A Network Rules Collection rule inside of Azure Firewall Policy defines the action (deny) and is linked to the IP Groups. Azure Firewall Manager can optionally be used to push this parent policy to any number of Azure Firewalls in the Azure Tenant, even across regions. 

 

Figure 2Figure 2

 

 

 

High Level Flow 

Here's how the pieces fit together (see Figure 3 below). As mentioned, the company's global security department receives mandated embargoed IP range lists from various government and regulatory agencies and compiles them into a simple text file, with one IP/IP CIDR range per line. 

 

This is the sequence of events: 

  1. Global Security drops the text file with the IP addresses/ranges into an Azure Storage Blob Container.
  2. An Azure Function watches this container, and any time a text file gets dropped, it parses this text file using a PowerShell script that… 
  3. Determines the number of IP addresses to include for each IP Group, based on a user-defined parameter that controls the limit. This lets the customer choose to spread the IPs out over a defined list of IP Groups or load each IP Group full before starting the next one. Then… 
  4. The Azure function updates the IP Groups.  
    1. In tests of up to 13,000 IP addresses, this takes less than 5 minutes, typically, with runs up to about 11 minutes when reducing from 8,000+ IPs down to only a handful of addresses 
    2. Because no security rule is being deleted and then re-created, only the IP Group re-defined, there is no lapse of coverage. Blocked IPs will remain blocked during the updates and only be allowed if they drop out from the updated list. 
  5. The process repeats itself each time Global Security drops a new version of the file in the blob storage.

 

Figure 3Figure 3

 

 

  

End Result 

The outcome of this solution meets the mandates of global security. Global security gets to control the embargoed list of IP addresses/ranges in a manner that can be implemented company-wide, across the globe in minutes, well within their 24 hour SLA. Lines of Business are not able to add exceptions to this list, as it's enforced as a parent policy. Because it is IP Groups getting updated (as opposed to a firewall network rule getting deleted and re-created), there's never a lapse in IP blocking, except when an IP address drops off the embargo list, which is the desired behavior. 

 

This is a very scalable solution that can accommodate the largest customers. There is no limit to the number of IP addresses/ranges that can be added to an IP Group (although the Azure networking team only has tested up to 5,000 and can guarantee performance to that number), with up to 100 IP Groups linked to a firewall rule, and multiple rules allowed, there are few scale scenarios this cannot handle. 

 

There can be substantial cost involved to the customer, as Azure Firewall can add up, however. In my next article I'll cover the same general scenario and mechanism, which relies upon the new Azure Network Manager (ANM) feature instead of Azure Firewall. ANM does not scale as much, but should be (pricing is not set yet, as of this article writing) a much lower price point. 

 

Appendix 

Here are more details about the solution. 

 

Capabilities 

  1. Parses a .txt file dropped into a storage blob container to split it into user-defined sized pieces, corresponding to the number of IP Groups in use.
  2. Updates the IP Groups with the new IP lists.

 Specifics 

  1. An Azure Function app has been created for this process.  
  2. The function watches a specific blob storage container for ANY txt file.  
    1. Currently the file name is not important, but could be used to specify region to run against or other desired effect with some minor code changes.
    2. Text file needs to have each IP address/IP range to block on a separate line.
  3. The function breaks the list of IP prefixes into IP Groups, based on the defined $numOfRules and $chunkSize variables (see Figure 4).
    1. Total number of IP addresses/ranges in the text file gets divided by the $chunkSize across the $numOfRules. 

      Figure 4Figure 4

       

       

  4. The function overwrites any previously defined IP Group entries. It does not remove and recreate the IP Groups, as to preserve existing blocked access. The firewall policy network rules will automatically see this change and update behavior accordingly.
  5. Some of the PoC environment specifics are defined by parameters in the code, such as Resource Group.
  6. Logging is provided through Subscription|Activity Log.

 Function Actions 

  1. Parses the text file that was discovered in the blob storage.
  2. Divides the list up according to $numOfRules and $chunkSize parameters.
  3. Uses Set-AzIpGroup to redefine the IP Group membership.

 Considerations 

  1. IP Groups can have any number of IPs in them, but the product group has only tested up to 5,000 and can guarantee good performance at that level.
    1. This is the number of LINE ITEMs… a line item can be either an IP address (1.2.3.4) or and IP range (1.2.3.0/24) 
  2. There can be up to 100 IP groups, for a max of 500,000 IP addresses/ranges entries that are tested to perform well. Numbers above this are possible, but there is no guarantee of performance. 
  3. Network rules can contain multiple IP Groups in them, so a single rule could be linked to all of the IP Groups in use, for a simpler rule set.
  4. If you want to enforce the Firewall Policy Parent Policy to the child policy, with NO ability to REMOVE the enforcement of that parent policy: 
    1. LoBs lose the ability to set DNS, Threat Intelligence, TLS Inspection, IDPS, Secured Virtual Hubs and other global settings.
      1. If LoBs need to change those, you can initially not lock the parent policy down, set the global settings and then lock it down so that LoB can't change the parent policy and global settings. 
    2. LoBs could add their own additional firewall rules 
  5. The Azure Subscription log can be used so see the modified groups.
  6. Notification of IP Group update failure can be set through automation.
  7. PowerShell can be modified to include file name or have more of the settings exposed as Azure Function configuration values and generally improved.
  8. The company in question was only concerned about egress traffic, as other controls blocked that traffic inbound, but the same IP Groups could be used with a Network Rule to control ingress traffic. 

 

Azure function PowerShell code* (provided by my DSE colleague Damir Zelenic): 

# Input bindings are passed in via param block. 
param([byte[]] $InputBlob, $TriggerMetadata) 
function Split-Array($InputArray, [int]$Size) 
{ 
$parts = [Math]::Ceiling($InputArray.count / $Size) 
$outArray = New-Object System.Collections.Generic.List[psobject] 
for ($i = 1; $i -le $parts; $i++) 
{ 
$start = ($i - 1) * $size 
$end = $i * $size - 1 
if ($end -ge $InputArray.count) { $end = $InputArray.count - 1 } 
$outArray.Add($InputArray[$start..$end]) 
} 
return $outArray 
} 
# Write out the blob name and size to the information log. 
Write-Host "PowerShell Blob trigger function Processed blob! Name: $($TriggerMetadata.Name) Size: $($InputBlob.Length) bytes" 
$startT = Get-Date 
$enc = [System.Text.Encoding]::ASCII 
$addr = $enc.GetString($InputBlob) 
$addrArr = $addr.Split("`n") 
# $addrArr = $addr.Split("`r`n") 
# Write-Host $addrArr 
$numOfRules = 13 #This is user defined; for PoC the rules were pre-created and bound to a firewall network rule 
[int]$chunkSize = $env:chunkSize 
Set-AzContext -SubscriptionName "demo-anmtest2" 
$count = $addrArr.Count 
Write-Host "Number of IP addresses $($count)" 
$ipGroups = Get-AzIpGroup -ResourceGroupName anmtest2-net-rg -Name from_file* 
# if there is only one element in array (number of elements smaller than chunk size) 
# indexing is wrong, index is not 0 but number of elements in "inner" array?! 
if ($chunkSize -ge $count) 
{ 
# first group only 
Write-Output "Adding IPs for single group..." 
$group = $ipGroups[0] 
$group.IpAddresses.RemoveRange(0, $group.IpAddresses.Count) 
$group.IpAddresses.AddRange([System.Collections.Generic.List[string]]$addrArr) 
Set-AzIpGroup -IpGroup $group 
} 
else 
{ 
$ipArr = Split-Array -InputArray $addrArr -Size $chunkSize 
$groupArr = @() 
Write-Output "Adding IPs for multiple groups..." 
for ($i = 0; $i -lt $ipArr.Count; $i++) 
{ 
$group = $ipGroups[$i] 
$group.IpAddresses.RemoveRange(0, $group.IpAddresses.Count) 
$group.IpAddresses.AddRange([System.Collections.Generic.List[string]]$ipArr[$i]) 
$groupArr += $group 
} 
$groupArr | foreach -Parallel { 
Set-AzIpGroup -IpGroup $_ 
} 
} 
# Get rest and remove IPs 
Write-Output "Removing IPs..." 
if ($chunkSize -ge $count) 
{ 
$groupsToRemoveIPs = $ipGroups[1..($numOfRules - 1)] 
} 
else 
{ 
$groupsToRemoveIPs = $ipGroups[$ipArr.Count..($numOfRules - 1)] 
} 
$groupsToRemoveIPs | foreach -Parallel { 
if ($_.IpAddresses.Count -gt 0) 
{ 
$_.IpAddresses.RemoveRange(0, $_.IpAddresses.Count) 
Set-AzIpGroup -IpGroup $_ 
} 
} 
$endT = Get-Date 
$totT = $endT - $startT 
Write-Output $totT 

 *Disclaimer 

The sample scripts are not supported under any Microsoft standard support program or service. The sample scripts are provided AS IS without warranty of any kind. Microsoft further disclaims all implied warranties including, without limitation, any implied warranties of merchantability or of fitness for a particular purpose. The entire risk arising out of the use or performance of the sample scripts and documentation remains with you. In no event shall Microsoft, its authors, or anyone else involved in the creation, production, or delivery of the scripts be liable for any damages whatsoever (including, without limitation, damages for loss of business profits, business interruption, loss of business information, or other pecuniary loss) arising out of the use of or inability to use the sample scripts or documentation, even if Microsoft has been advised of the possibility of such damages. 

 

Representative Test Run Times 

The table below shows run times tested during implementation. We fed a text file containing various numbers of IP ranges (left column) to the Azure Function and waited for completion. The resulting time is shown in the right column. Tests were performed in the sequence shown in the table. 

 

Qty IP Addresses in the Text file 

Function Running time (mm:ss) 

8005 

4:40 

13 

4:00 

8005 

1:45 

13 

3:50 

11157 

3:57 

13 

4:25 

13 

5:25 

13 

3:40 

11157 

4:55 

13 

5:33 

8005 

4:29 

11157 

4:48 

13 

4:38 

11157 

7:53 

13 

9:29 

11157 

3:22 

8005 

5:23 

11157 

2:38 

13 

5:08 

8005 

2:36 

11157 

3:38 

 

Conclusion 

Putting this all together, we end up with a simple to implement and maintain solution for an enterprise’s central security to enforce a set of IP addresses or ranges across all of their Azure assets that traverse the Azure Firewall. Requiring only minutes to go into effect, this combination of Azure Storage blob, Azure Function, IP Groups and Azure Firewalls can scale to handle up to the largest enterprise customers and insure that mandated block lists can be achieved. 

Posted at https://sl.advdat.com/3wosqro