This is the Bicep-based deployment guide for Scenario 1: Azure Container Apps - Internal environment secure baseline.
You can deploy the current LZA directly in your Azure subscription using Azure Dev CLI.
- Visit github.com/Azure/aca-landing-zone-accelerator
- Click on the
Green Code
button. - Navigate to the
CodeSpaces
tab and create a new code space. - Open the terminal by pressing
Ctrl + `
. - Navigate to the scenario folder using the command
cd /workspaces/aca-landing-zone-accelerator/scenarios/aca-internal
. - Login to Azure using the command
azd auth login
. - Use the command
azd up
to deploy, provide environment name and subscription to deploy to. - Finally, use the command
azd down
to clean up resources deployed.
This is the starting point for the instructions on deploying this reference implementation. There is required access and tooling you'll need in order to accomplish this.
-
An Azure subscription
-
The following resource providers registered:
Microsoft.App
Microsoft.ContainerRegistry
Microsoft.ContainerService
Microsoft.KeyVault
-
The user or service principal initiating the deployment process must have the owner role at the subscription level to have the ability to create resource groups and to delegate access to others (Azure Managed Identities created from the IaC deployment).
-
Latest Azure CLI installed (must be at least 2.40), or you can perform this from Azure Cloud Shell by clicking below.
-
Clone/download this repo locally, or even better fork this repository.
🔀 If you have forked this reference implementation repo, you can configure the provided GitHub workflow. Ensure references to this git repository mentioned throughout the walk-through are updated to use your own fork.
git clone https://github.com/Azure/aca-landing-zone-accelerator.git cd aca-landing-zone-accelerator/scenarios/aca-internal/bicep
-
Update naming convention. Optional.
The naming of the resources in this implementation follows the Cloud Adoption Framework's resource naming convention. Your organization might have a naming strategy in place, which possibly deviates from this implementation. In most cases you can modified what is deployed by modifying the following two files:
- naming.module.bicep contains the nameing convention.
- naming-rules.jsonc contains the abbreviations for resources (
resourceTypeAbbreviations
) and Azure regions (regionAbbreviations
) used in the naming convention.
-
🗺️ Choose your deployment experience.
This reference implementation comes with three implementation deployment options. They all result in the same resources and architecture, they simply differ in their user experience; specifically how much is abstracted from your involvement.
-
Follow the "Standalone deployment guide" if you'd like to simply configure a set of parameters and execute a single CLI command to deploy.
This will be your simplest deployment approach, but also the most opaque. This is optimized for "cut to the end."
-
Follow the "Standalone deployment guide with GitHub Actions" if you'd like to simply configure a set of parameters and have GitHub Actions execute the deployment.
This is a variant of the above. A fork of this repo is required for this option, and requires you to create a service principal with appropriate permissions in your Azure Subscription to perform the deployment.
-
Follow the "Standalone deployment guide with Azure Pipelines" if you'd like to simply configure a set of parameters and have Azure Pipelines execute the deployment.
This is a variant of the first deployment experience. A fork of this repo is required for this option, and requires you to create an appropriate service connection for the pipeline to connect to your Azure subscription.
-
Follow the "Step-by-step deployment guide" if you'd like to walk through the deployment at a slower, more deliberate pace.
This will approach will allow you to see the deployment evolve over time, which might give you an insight into the various roles and people in your organization that you need to engage when bringing your workload in this architecture to Azure. This is optimized for "learning."
All of these options allow you to deploy to a single subscription, to experience the full architecture in isolation. Adapting this deployment to your Azure landing zone implementation is not required to complete the deployments.
-
-
Log into Azure from the AZ CLI and select your subscription.
az login
-
Review and update deployment parameters.
The main.parameters.jsonc parameter file is where you can customize your deployment. The defaults are a suitable starting point, but feel free to adjust any to fit your requirements.
Deployment parameters
Name Description Default Example(s) workloadName
A suffix that will be used to name resources in a pattern similar to <resourceAbbreviation>-<applicationName>
. Must be less than 11 characters long, alphanumeric with dashes.lzaaca app-svc-01 environment
The short name of the environment. Up to eight characters long. dev qa, uat, prod tags
Resource tags that you wish to add to all resources. none "value": {
"Environment": "qa",
"CostCenter": CS004"
}
enableTelemetry
Enables or disabled telemetry collection true false hubResourceGroupName
The name of the hub resource group to create the hub resources in. none. This results in a new resource group being created. rg-byo-hub-academo. This results in rg-byo-hub-academo
being used. This must be an empty resource group, do not use an existing resource group used for other purposes.spokeResourceGroupName
The name of the spoke resource group to create the spoke resources in. none. This results in a new resource group being created. rg-byo-spoke-academo. This results in rg-byo-spoke-academo
being used. This must be an empty resource group, do not use an existing resource group used for other purposes.vnetAddressPrefixes
An array of string. The address prefixes to use for the hub virtual network. ["10.0.0.0/24"]
["10.100.0.0/24"]
gatewaySubnetAddressPrefix
CIDR to use for the gatewaySubnet. Must be a subset of the hub CIDR ranges. 10.0.0.0/27 10.100.2.0/27 azureFirewallSubnetAddressPrefix
CIDR to use for the azureFirewallSubnet. Must be a subset of the hub CIDR ranges. 10.0.0.64/26 10.100.2.0/26 enableBastion
Controls if Azure Bastion is deployed. true
false` bastionSubnetAddressPrefix
CIDR to use for the Azure Bastion subnet. Must be a subset of the hub CIDR ranges. 10.0.0.128/26 10.100.2.0/26 vmSize
The size of the virtual machine to create for the jump box. Standard_B2ms
Any one of: VM sizes vmAdminUsername
The username to use for the jump box. azureuser jumpboxadmin
vmAdminPassword
The password to use for the jump box admin user. Password123 🛑 You should change this. Any cryptographically strong password of your choosing. vmLinuxSshAuthorizedKeys
The SSH public key to use for the jump box (if VM is Linux). unusable/garbage value Any SSH keys you wish in the form of ssh-rsa AAAAB6NzC...P38/oqQv description vmJumpboxOSType
The type of OS for the deployed jump box. linux windows vmJumpBoxSubnetAddressPrefix
CIDR to use for the jump box subnet. must be a subset of the hub CIDR ranges. 10.1.2.32/27 10.100.3.128/27 vmAuthenticationType
Authentication type for the Linux jump box. Either password or SSH key. SSH key is recommended. password
sshPublicKey
,password
spokeVNetAddressPrefixes
An array of string. The address prefixes to use for the spoke virtual network. ["10.1.0.0/22"]
["10.101.0./22"]
spokeInfraSubnetAddressPrefix
CIDR of the spoke infrastructure subnet. Must be a subset of the spoke CIDR ranges. 10.1.0.0/23 10.101.0.0/23 spokePrivateEndpointsSubnetAddressPrefix
CIDR of the spoke private endpoint subnet. Must be a subset of the spoke CIDR ranges. 10.1.2.0/27 10.101.2.0/27 spokeApplicationGatewaySubnetAddressPrefix
CIDR of the spoke Application Gateway subnet. Must be a subset of the spoke CIDR ranges. 10.1.3.0/24 10.101.3.0/24 enableApplicationInsights
Controls if Application Insights is deployed and configured. true false enableDaprInstrumentation
Enable Dapr's telemetry. enableApplicationInsights` must also be set to true for this to work. true false deployHelloWorldSample
Deploy a simple, sample application to the infrastructure. If you prefer to deploy the more comprehensive, Dapr-enabled sample app, this needs to be disabled true false, because you plan on deploying the Dapr-enabled application instead. deployRedisCache
Feature flag, if true Azure Cache for Redis (Premium SKU), together with Private Endpoint and the relavant Private DNS Zone will be deployed. false true deployOpenAi
Feature flag, Deploy (or not) an Azure OpenAI account. ATTENTION: At the time of writing this, OpenAI is in preview and only available in limited regions false true ddosProtectionMode
DDoS protection mode for the Public IP of the Application Gateway. Allowed values are "VirtualNetworkInherited", "Enabled" and "Disabled". Enabled VirtualNetworkInherited deployAzurePolicies
If true, Azure Policies will be deployed. true false deployZoneRedundantResources
If true, any resources that support AZ will be deployed in all three AZ. However if the selected region is not supporting AZ, this parameter needs to be set to false. true false -
Deploy the reference implementation.
This will deploy all of the infrastructure to your selected subscription. This will take over 10 minutes to execute.
LOCATION=northeurope # or any location that suits your needs LZA_DEPLOYMENT_NAME=bicepAcaLzaUDRDeployment # or any other value that suits your needs ```bash az deployment sub create \ --template-file main.bicep \ --location $LOCATION \ --name $LZA_DEPLOYMENT_NAME \ --parameters ./main.parameters.jsonc
-
Deploy the Dapr-based workload. Optional.
If you chose to set
deployHelloWorldSample
to false, then proceed to deploy the Dapr-based workload by following the instructions at:
Before cleaning up the resources you might wish to verify the Azure Firewall Rules
When you are done exploring the resources created by the Standalone deployment guide, use the following command to remove the resources you created.
$LZA_DEPLOYMENT_NAME=bicepAcaLzaDeployment # The name of the deployment you used in the previous step
# get the name of the Spoke Resource Group that has been created previously
SPOKE_RESOURCE_GROUP_NAME=$(az deployment sub show -n "$LZA_DEPLOYMENT_NAME" --query properties.outputs.spokeResourceGroupName.value -o tsv)
# get the name of the Hub Resource Group that has been created previously
HUB_RESOURCE_GROUP_NAME=$(az deployment sub show -n "$LZA_DEPLOYMENT_NAME" --query properties.outputs.hubResourceGroupName.value -o tsv)
az group delete -n $SPOKE_RESOURCE_GROUP_NAME
az group delete -n $HUB_RESOURCE_GROUP_NAME
-
Create a new service principal with the owner role on the subscription.
Replace
{subscription-id}
below.az ad sp create-for-rbac --name "myApp" --role owner \ --scopes /subscriptions/{subscription-id} \ --sdk-auth
Note that this command will output the following warning
Option '--sdk-auth' has been deprecated and will be removed in a future release.
. Nevertheless, this method is still strongly recommend as documented by the Azure\login team. -
Copy the output from the prior command.
{ "clientId": "<GUID>", "clientSecret": "<GUID>", "subscriptionId": "<GUID>", "tenantId": "<GUID>", (...) }
-
Navigate to where you forked the GitHub repository and go to Settings > Secrets and variables > Actions > New repository secret.
-
Create a new secret called
AZURE_CREDENTIALS
with the JSON information and press Add Secret. -
On the same screen ( Settings > Secrets and variables > Actions ), you need to add two repository variables. Click on the tab titled Variables, and then click on New repository variable.
-
Add the first variable named
LOCATION
and enter as value, a valid Azure data center location (i.e. northeurope). This will be the region where all of your resources will be deployed. -
Add the second variable named
ENABLE_TEARDOWN
typed as a boolean. If you wish the environment to be cleaned up after some manual approval, or after 120 minutes, then set this variable totrue
. If you don't want automatic clean up of the deployed resources, set this variable tofalse
. You need also to update theCODEOWNERS
file, with the right GitHub handles.
Before cleaning up the resources you might wish to verify the Azure Firewall Rules
If you didn't select automatic clean up of the deployed resources, use the following commands to remove the resources you created.
az group delete -n <your-spoke-resource-group>
az group delete -n <your-hub-resource-group>
-
Navigate into your Azure DevOps projects, click on Project Settings, and then on the left sidebar, under the Pipelines section, click on the Service Connections, and then click New service connection button and create a new Azure Resource Manager service connection.
-
Into your Azure DevOps projects, click on Pipelines on the left sidebar, and then click on Library, and then click on +Variable Group. Name the new Variable Group "ACA-LZA" and then add the following variables:
- location: The location of where you want the Azure resources deployed
- azureServiceConnection: the name of the service connection you created in the previous step
-
Navigate into your Azure DevOps projects and click on Pipelines on the left sidebar.
-
Click New Pipeline in the upper right-hand corner of the window or the create pipeline button in the middle if this is your first pipeline. Select GitHub as the source for your YAML.
-
Select your repository in GitHub. If you don't already have the Azure Pipeline app installed in your GitHub repository, it will prompt you to enable that and redirect you back to this creation screen.
-
Select Existing Azure Pipelines YAML file, select the main branch and the file lza-deployment-bicep.yaml.
-
Once you select the file, click Next and then click Run in the upper right-hand corner of the Review tab. If you don't want to run it immediately, you can click the dropdown on the Run button and choose to save it.
Note When you first run your pipeline, you may need to give the pipeline permission to access the service connection and the variable group. This will only occur the first time you run the pipeline.
Before cleaning up the resources you might wish to verify the Azure Firewall Rules
Use the following commands to remove the resources you created.
az group delete -n <your-spoke-resource-group>
az group delete -n <your-hub-resource-group>
These instructions are spread over a series of dedicated pages for each step along the way. With this method of deployment, you can leverage the step-by-step process considering where possibly different teams (devops, network, operations etc) with different levels of access, are required to coordinate and deploy all of the required resources.
If you have deployed the Hello World Sample Application and you wish to verify your Azure Firewall configuration is set up correctly, you can use the curl
command from your app's debugging console. Follow the steps below:
-
Navigate to your Container App that is configured with Azure Firewall.
-
From the menu on the left, select Console, then select your container that supports the curl command.
-
In the Choose start up command menu, select
/bin/sh
, and select Connect. -
In the console, run
curl -s https://mcr.microsoft.com
. You should see a successful response (because the default LZA deployment adds the application rule ace-allow-rules which among others addsmcr.microsoft.com
to the allowlist for your firewall policies). If you get an error that curl is not found in the container's shell, follow the next steps to install it.a. Run
apk add curl
to add the curl package. If you get an error, most possibly some URL is being blocked by your firewall, so let's investigate that.b. Got to your hub, find your azure firewall, and click on the logs. there run the following query:
AzureDiagnostics | where Category == "AzureFirewallNetworkRule" or Category == "AzureFirewallApplicationRule" | extend msg_original = msg_s | extend msg_s = replace(@'. Action: Deny. Reason: SNI TLS extension was missing.', @' to no_data:no_data. Action: Deny. Rule Collection: default behavior. Rule: SNI TLS extension missing', msg_s) | extend msg_s = replace(@'No rule matched. Proceeding with default action', @'Rule Collection: default behavior. Rule: no rule matched', msg_s) | parse msg_s with * " Web Category: " WebCategory | extend msg_s = replace(@'(. Web Category:).*','', msg_s) | parse msg_s with * ". Rule Collection: " RuleCollection ". Rule: " Rule | extend msg_s = replace(@'(. Rule Collection:).*','', msg_s) | parse msg_s with * ". Rule Collection Group: " RuleCollectionGroup | extend msg_s = replace(@'(. Rule Collection Group:).*','', msg_s) | parse msg_s with * ". Policy: " Policy | extend msg_s = replace(@'(. Policy:).*','', msg_s) | parse msg_s with * ". Signature: " IDSSignatureIDInt ". IDS: " IDSSignatureDescription ". Priority: " IDSPriorityInt ". Classification: " IDSClassification | extend msg_s = replace(@'(. Signature:).*','', msg_s) | parse msg_s with * " was DNAT'ed to " NatDestination | extend msg_s = replace(@"( was DNAT'ed to ).*",". Action: DNAT", msg_s) | parse msg_s with * ". ThreatIntel: " ThreatIntel | extend msg_s = replace(@'(. ThreatIntel:).*','', msg_s) | extend URL = extract(@"(Url: )(.*)(\. Action)",2,msg_s) | extend msg_s=replace(@"(Url: .*)(Action)",@"\2",msg_s) | parse msg_s with Protocol " request from " SourceIP " to " Target ". Action: " Action | extend SourceIP = iif(SourceIP contains ":",strcat_array(split(SourceIP,":",0),""),SourceIP), SourcePort = iif(SourceIP contains ":",strcat_array(split(SourceIP,":",1),""),""), Target = iif(Target contains ":",strcat_array(split(Target,":",0),""),Target), TargetPort = iif(SourceIP contains ":",strcat_array(split(Target,":",1),""),""), Action = iif(Action contains ".",strcat_array(split(Action,".",0),""),Action), Policy = case(RuleCollection contains ":", split(RuleCollection, ":")[0] ,Policy), RuleCollectionGroup = case(RuleCollection contains ":", split(RuleCollection, ":")[1], RuleCollectionGroup), RuleCollection = case(RuleCollection contains ":", split(RuleCollection, ":")[2], RuleCollection), IDSSignatureID = tostring(IDSSignatureIDInt), IDSPriority = tostring(IDSPriorityInt) | project TimeGenerated,Protocol,SourceIP,SourcePort,Target,TargetPort,URL,Action, NatDestination, OperationName,ThreatIntel,IDSSignatureID,IDSSignatureDescription,IDSPriority,IDSClassification,Policy,RuleCollectionGroup,RuleCollection,Rule,WebCategory, msg_original | where Action == "Deny" | order by TimeGenerated desc | limit 100
You should find some calls t with target fqdn
dl-cdn.alpinelinux.org
that are being blocked. This already verifies that the firewall is successfully filtering the egress traffic, but let's fix that, and addcurl
in your container.c. Go to the Azure Firewall > Settings > Rules (Classic) > Application Rule Connection and add an application rule that permits calls to
dl-cdn.alpinelinux.org
with http:80 and https:443 protocols. Wait for the rule to be updated/created and then try again to install curl (apk add curl
).d. Once curl is installed run again
curl -s https://mcr.microsoft.com
, you should see a successful response. -
Run
curl -s https://www.docker.com
(for a URL that doesn't match any of your destination rules). You should get no response, which indicates that your firewall has blocked the request. If you wish you can check the Firewall's logs (with the query found in the previous step) to verify that your call to www.docker.com has been denied.