Before scaling Sitecore Azure Web App, you must read the experience I had and going to share in this post.
Azure Web App
In Azure, you can easily change existent resources almost in real time to accomodate business needs, and this is also applies to Web Apps where you can Scale Up and/or Scale Out your application.
Sitecore
Sitecore offers different architecture models to fit client’s needs, and its roles vary depending on the topology is choosen. Nevertheless, you have the ability to scale either vertical and/or horizontally.
The environment
Everything is hosted in Azure, using Web Apps, and provisioned following Sitecore XP Scaled topology. Here is an idea in how the environment looked like
data:image/s3,"s3://crabby-images/02ed2/02ed2e842e863c860014c580eff597b00ce58df8" alt="sitecore architecture azure paas blog vinicius deschamps"
Before moving forward it is important to know about the difference between Azure Web App and App Service Plans
Azure Web Apps vs App Service Plans
During Azure Web App creation, you are prompted to provide the Instance Details (1) that corresponds to the Web App, and App Service Plan (2) which must be one of the following: create new or choose an existent one from the dropdown list.
data:image/s3,"s3://crabby-images/b41ba/b41ba7531cf49789fd0c9d6735a41df3a4394292" alt="azure azure app service vs azure app service plan create blog vinicius deschamps"
The relation between Web App and App Service Plan can be illustrated as the image below
data:image/s3,"s3://crabby-images/8e724/8e72417f1ce1cb55e9d79279dbd73de9455a512a" alt="azure app service vs app service plan blog vinicius deschamps"
N Web Apps can use the same App Service Plan, as long as the resources the App Service Plan can handle.
In addition to that, an idea that helps to understand is to think like
- App Service Plan is the physical or virtual server which has the resources (CPU, memory, etc)
- Web App is the web server where configuration and settings sits
That said, we wanted to Scale Out both CM and CD, and everything went well with the CD, so let’s attempt to perform the same on CM.
Scale Out CM idea
When you Scale Out your application, a Load Balancer is added in front of your Azure Web App, and Sitecore CM expects exactly that for a horizontal scaling
You can set up multiple instances behind a load balancer. All instances must connect to a centrally located master database
https://doc.sitecore.com/developers/100/platform-administration-and-architecture/en/scaling-and-configuring-content-management.html
Thus we have decided to simply Scale Out the CM Web App
Everything went wrong #1
Once all instances were up and running, a couple tests were made and things were working just fine.
However, when multiple users started to work, things started to become out of control with instances randomically restarting out of the blue with the following message
WARN Shutdown message: CONFIG change
Always look for the tiny details
By scaling out the CM, the environment was going from 1 to N servers, and Sitecore has different requirements to make this to happen
InstanceName parameter
You must ensure that each CM instance has a unique name, and that’s OK because the default value set in Sitecore.config file is (machine name and IIS site name), and when you scale out you will get a clone App Service that runs with a unique name.
<!-- INSTANCE NAME
Unique name for Sitecore instance.
Default value: (machine name and IIS site name)
-->
<setting name="InstanceName" value="" />
Publishing
Only one CM instance can perform publishing, and you must ensure that each CM instance points to the publishing CM instance, and again that’s OK because the environment has a dedicated Publishing Service.
ValidationKey and DecryptionKey
Ensure that the values is the same on all the Sitecore instances within the environment and that the IsolateApps modifier is not present in either value, and in the environment configuration these parameters were set fixed and statically.
Indexing
Since a Scale Out simply clone the file system, and doesn’t allow us to have control in each new server’s file system to modify the Indexing configuration, Sitecore CM was getting confuse with multiple servers performing the indexing role.
Now, that the issue has been identified, the idea was to provisioning a new Content Management using its own App Service (and App Service Plan) where the file system could be modified as needed.
So, let’s call this new Content Management CM02 and the existent CM01.
Enable Indexing sub-role CM02
The idea is to have CM02 with the Indexing sub-role, and to accomplish that a modification must be done in the Web.config
<add key="role:define" value="ContentManagement, Indexing, Indexer"/>
Starting in Sitecore 9.2, when you assign the indexing sub-role to a server role, that role manages content indexing automatically and the other server roles that have not been assisgned this sub-role do not index content
https://dev.sitecore.net/Downloads/Sitecore%20Experience%20Platform/92/Sitecore%20Experience%20Platform%2092%20Initial%20Release/Release%20Notes
Disable Indexing sub-role CM01
IMPORTANT: If you are using Sitecore 9.2 and newer, you don’t need to disable indexing on other CMs
For each Index you have configured, you must change its indexUpdateStrategies to manual like the example below
<index id="sitecore_index" type="Sitecore.ContentSearch.SolrProvider.SolrSearchIndex,
Sitecore.ContentSearch.SolrProvider">
<param desc="name">$(id)</param>
<param desc="core">$(id)</param>
<param desc="propertyStore"
ref="contentSearch/indexConfigurations/databasePropertyStore"
param1="$(id)" />
<strategies hint="list:AddStrategy">
<strategy ref="contentSearch/indexConfigurations/indexUpdateStrategies/manual" />
</strategies>
Once these changes were applied, here is how the environment look like
data:image/s3,"s3://crabby-images/55eca/55eca6be54e43759094add617c5424274cf4215a" alt="sitecore architecture azure paas scaled out blog vinicius deschamps"
Everything went wrong #2
Now, we have everything that is required to have a multiple CM environment
- Load Balancer in front of the CMs
- InstanceName parameter with Unique Names
- A Publishing Server
- Same ValidationKey and DecryptionKey for all CMs
- One CM to handle Indexing
And things were looking very promising, well… well… until the instances from the scaled out CM01 started to fall apart and, again, randomly and with the exactly same message.
WARN Shutdown message: CONFIG change
What else could be wrong? After some research, I found something answered our question
Accordingly to Sitecore Managed Cloud documentation
Scale out is not supported for CM
https://doc.sitecore.com/developers/managed-cloud/en/scaling-options-for-managed-cloud.html
The solution
That said, the only possibility would be to have additional CMs each of them in their own App Service Plan.
However, since we are not scaling the CM App Service anymore, we don’t have a load balancer to split the load between the multiple CMs and there are some important things to know
- Azure Load Balancer does not work with Azure App Services, even if you configure VNet for them
- WAF and Application Gateway are not compatible with Content Management servers
So, you either come up with a third party solution that will assist you in have a Load Balancer for your CMs or you can use Azure Front Door but please keep in mind the following
Use of this technology within a Sitecore solution is permitted, but the Sitecore Support team can only assist with Sitecore product issues that are not unique to the use of the technology, that is, a Sitecore product issue that can be replicated without the technology present.
In some cases, the Sitecore Support team may ask for replication of the issue without the use of the technology, although they will try to avoid this.
https://kb.sitecore.net/articles/682999#Notes
In my case, I am going to use it and here is how it suppose to be
data:image/s3,"s3://crabby-images/58535/58535eee6c42f99723ed0bb304eb845111f6bae2" alt="sitecore content management azure front door blog vinicius deschamps"
Configure Azure Front Door
Search for Front Doors, and click Front Doors in Azure Search field
data:image/s3,"s3://crabby-images/2fb35/2fb354a6d26b99bf1a3cf7aa02eb7a1e0f053cf5" alt="azure search front doors blog vinicius deschamps"
Click Add to create a new Front Door
data:image/s3,"s3://crabby-images/22780/2278061e971da838b3118835c52b64fd84de565d" alt="azure front doors add blog vinicius deschamps"
In Basics choose the appropriate Resource Group or create a new one, and Next : Configuration >
data:image/s3,"s3://crabby-images/8a4cb/8a4cbb6d5588b4bb7f03518f9782d046c4ac6777" alt="azure create a front door basics blog vinicius deschamps 1"
At Configuration you will see 3 steps, let’s start by clicking at the + sign on Step 1
data:image/s3,"s3://crabby-images/498f1/498f1e38d5713ab2c345f70ba4e2aa3da8a0d4e6" alt="azure create a front door configuration frontends domains blog vinicius deschamps 1"
In Add a frontend host provide Host name (1), ensure to choose Enabled for Session Affinity (2) and disabled for Web Application Firewall (3) – since Sitecore does not like it – and click Add
data:image/s3,"s3://crabby-images/0a319/0a31917d1786014b6d59cc65bbea391f4e7bd8cb" alt="azure create a front door configuration frontends domains add a frontend host blog vinicius deschamps"
Let’s move to the Step 2 by clicking in the + sign
data:image/s3,"s3://crabby-images/b32cb/b32cb5b8d61d81cba0367653d1e2c53bcdfa5947" alt="azure create a front door configuration backend pools blog vinicius deschamps"
Add a backend pool by giving it a Name (1) and + Add a backend (2)
data:image/s3,"s3://crabby-images/36beb/36bebd95a3033e314753e16aafbd2a8646c87e48" alt="azure create a front door add a backend pool blog vinicius deschamps"
Choose a Backend host type (1), in our case an App Service, then choose the Backend host name (2) and its Backend host header (3), and hit Add
data:image/s3,"s3://crabby-images/31e10/31e1061d82c5f5e11f18b7e9e382e851b2c6d3a6" alt="azure create a front door configuration backend pools add a backend app service blog vinicius deschamps"
Ensure to repeat the step above by clicking + Add a backend until you have all CM Web Apps added
data:image/s3,"s3://crabby-images/7db06/7db0622720a2d1b4c111779a3b225ba817d8afa1" alt="azure create a front door add a backend pool repeat add a backend app services blog vinicius deschamps"
data:image/s3,"s3://crabby-images/7313c/7313c9fd9e632afe1800001f845b07f46344f700" alt="azure create a front door add a backend pool repeat complete add a backend app services blog vinicius deschamps"
Once you have all Web Apps, you can continue on Add a backend pool options, and simply click Add
data:image/s3,"s3://crabby-images/faf3e/faf3e04fa275df76bfc88ca0552909022a93aab7" alt="azure create a front door configuration backend pools add a backend health probe load balancing blog vinicius deschamps"
Finally, let’s configure the Step3 by clicking at + sign
data:image/s3,"s3://crabby-images/90c66/90c6687329fd5baab818703a3e525af686bc047e" alt="azure create a front door routing rules blog vinicius deschamps"
Give it a Name (1), and choose its Frontends/domains (2) and the Backend pool (3), then press Add (4)
data:image/s3,"s3://crabby-images/0d3fd/0d3fd98f26d471324618d79b60aabe6405742e66" alt="azure create a front door routing rules add a rule blog vinicius deschamps"
Now that you have all 3 steps configured hit Review + Create
data:image/s3,"s3://crabby-images/fa775/fa7754fb9c3cfe83f8ca57e3741c52f4e95531c3" alt="azure create a front door review create blog vinicius deschamps"
Review the information and hit Create
data:image/s3,"s3://crabby-images/5ba80/5ba804677f505721b2f572edc36d27250e2dbe2b" alt="azure create a front door review create create blog vinicius deschamps"
You should wait until Front Door is provisioned, and you can access from your browser the host name you have set, in my case is https://blog-vinicius-deschamps.azurefd.net
data:image/s3,"s3://crabby-images/393c0/393c0314384dfc80288168f114b068bea13983cc" alt="azure front door access sitecore cm blog vinicius deschamps"
Great but you probably don’t want to share an azurefd.net domain with your Content Editors so they can access your CM. Let’s see how to configure a custom domain.
Custom domain Azure Front Door
Go to the Front Door (1), select Front Door designer (2), then click in the + (3) at Frontends/domains
data:image/s3,"s3://crabby-images/087ba/087ba9b6ee37dab49adace5da6c3b424e0cb365d" alt="azure front door front door design frontends domains add new domain blog vinicius deschamps"
You should choose a Custom host name, and you will receive a message saying that a CNAME configuration must be done at your DNS
data:image/s3,"s3://crabby-images/6d82d/6d82d5f6a14df334f996d76a975af9ba4f8c99fb" alt="azure front door add a custom domain blog vinicius deschamps"
I am using Azure DNS, so I went to my domain viniciusdeschamps.cloud (1), selected Overview (2) and + Record set (3)
data:image/s3,"s3://crabby-images/d475b/d475bef5535bb6b7deeb20f83eb8c0d76796d0a0" alt="azure dns new record set blog vinicius deschamps"
In Add record set, add a Name (1), select the Type (2) and add the Alias (3), then hit Ok (4)
data:image/s3,"s3://crabby-images/fc2f5/fc2f56a5785ffc9c6988c80d570c41590dbf3182" alt="azure dns add record set azure front door cname blog vinicius deschamps"
Back to Front Door, and type your Custom host name (1) again, choose Enabled (2) for Custom Domain HTTPS, and in a Production environment use your own certificate but in my case I’m going to use Front Door managed (3), and don’t forget to Enabled (4) Session Affinity and press Add (5)
data:image/s3,"s3://crabby-images/9ca3a/9ca3a6f7df75e7759a2dda60fb056116fa208390" alt="azure front door add a custom domain after adding dns cname blog vinicius deschamps"
In Front Door Designer you will see the following warning
The host ‘your-custom-host.domain’ is not present in any of the routing rules.
The host ‘your-custom-host.domain’ is missing a default route ‘/*’ path for both the HTTP and HTTPS protocolos
data:image/s3,"s3://crabby-images/ae0b5/ae0b5d81574bf0d063f0e3e9e966f118d629f8df" alt="azure front door the host is not present in any of the routing rules blog vinicius deschamps"
Select the existing Routing rules
data:image/s3,"s3://crabby-images/afce9/afce9dc27e598750fc3eee26679f55a1d9adf993" alt="azure front door routing rules update existing rule blog vinicius deschamps"
And in Frontends/domains choose the recently added domain from the list
data:image/s3,"s3://crabby-images/58ce6/58ce66487c32e563f5ac1148ef70419bd19c2ef4" alt="azure front door update routing rule frontends domains blog vinicius deschamps"
Then, press Update
data:image/s3,"s3://crabby-images/3c1c6/3c1c6cd0fc23521cd02831bb7b2c88a9d6278f60" alt="azure front door update routing rule hit update button after select new frontends domains blog vinicius deschamps"
Once you have done, click Save in Front Door designer
data:image/s3,"s3://crabby-images/5eced/5eced7456e2afbadfa71e8a0711472d502eb6e2a" alt="azure front door there are pending changes click save to apply blog vinicius deschamps"
data:image/s3,"s3://crabby-images/b4d27/b4d279f0f172dec29b174b5ae4f0d81fceaf8cfe" alt="azure front door with custom domain and new rules blog vinicius deschamps"
Please wait for a while until you get a confirmation the configuration has been applied, in my case, it took a while because I choose the Front Door managed certificate
data:image/s3,"s3://crabby-images/c2ac7/c2ac7bbb9cb8d82c1075d0cc8bfd1b3db4bc7ebf" alt="azure front door updating the custom https configuration blog vinicius deschamps"
data:image/s3,"s3://crabby-images/56016/560168c436d586d0417355a18ae13f4dff1aaa6c" alt="azure front door updated the custom https configuration blog vinicius deschamps"
Sitecore Identity Server Unauthorized_Client
Once the Azure Front Door was updated with the new Custom Domain, I tried to access https://prodcm.viniciusdeschamps.cloud/sitecore and when hit the Identity Server got the following message
Sorry, there was an error: unauthorized_client
And this happens because in the Identity Server configuration you should set from which URLs authentication requests should be accepted
data:image/s3,"s3://crabby-images/a9a83/a9a83bf1584ea573297b952bf3999772d06e98a6" alt="sitecore identity server sorry there was an error unauthorized client blog vinicius deschamps"
In order to fix it, you should go to Sitecore Identity Server App Service (1), Advanced Tools (2) and Go (3)
data:image/s3,"s3://crabby-images/417e9/417e9a3504017d5d8bb62c1c47507539475823fb" alt="azure app service sitecore identity server advanced tools go blog vinicius deschamps"
Select Debug console, and Powershell
data:image/s3,"s3://crabby-images/43530/43530c96e85c64e7a541e1158344612ffb57da04" alt="azure app service kudu debug console powershell blog vinicius deschamps"
Navigate to site\wwwroot\Config\production, and access the following file Sitecore.IdentityServer.Host.xml then modify the following line
<AllowedCorsOriginsGroup1>https://sc91vdsch-cm.azurewebsites.net|https://</AllowedCorsOriginsGroup1>
data:image/s3,"s3://crabby-images/ad9f7/ad9f777126c9dab9b29a88ebccfe8d250d81ccd4" alt="sitecore identityserver host xml add new custom domain blog vinicius deschamps"
By adding all Content Management URLs as follows, and hit Save
<AllowedCorsOriginsGroup1>https://sc91vdsch-cm.azurewebsites.net|https://sc91vdsch-cm02.azurewebsites.net|https://sc91vdsch-cm03.azurewebsites.net|https://prodcm.viniciusdeschamps.cloud</AllowedCorsOriginsGroup1>
data:image/s3,"s3://crabby-images/1410e/1410eae712f7628b2ebaf10000afcca8cae6e1d4" alt="sitecore identityserver host xml add new custom domain save blog vinicius deschamps"
Once the modification is saved, you should Restart the Sitecore Identity Server App Service
data:image/s3,"s3://crabby-images/fa740/fa74018edb32280b5ab27d959ce768ae26dbe38a" alt="azure app service sitecore identity server overview restart blog vinicius deschamps"
data:image/s3,"s3://crabby-images/20017/2001712e617c6e795de9145c08d7934395a3e7f0" alt="azure app service sitecore identity server are you sure you want to restart blog vinicius deschamps"
Once the restart is completed, try to access Sitecore CM using the custom domain configured on Azure Front Door, in my case https://prodcm.viniciusdeschamps.cloud/sitecore and voilá no error
data:image/s3,"s3://crabby-images/b8f3d/b8f3d9cc83eac01b14324cfcf7ae63c39e53a2d9" alt="sitcore identity server fully accessible type credentials blog vinicius deschamps"
I got redirect to the Sitecore Experience Platform, and as you can see to the CM02 but that’s not I want for my end users, so there are additional steps to address it. Well, at least we know the Front Door is splitting the traffic
data:image/s3,"s3://crabby-images/eace0/eace0d560aef18451347f4e93ff14cc34956b261" alt="sitecore not using azure front door custom domain blog vinicius deschamps"
Custom Domains for App Services
Navigate to the App Services, and choose one of the CMs (1), select Custom Domains (2) and copy the Custom Domain Verification ID (3)
data:image/s3,"s3://crabby-images/36d3e/36d3ef1886f5c6b6da2396648088099648a88092" alt="azure app service custom domains custom domain verification id blog vinicius deschamps"
Go to your DNS provider, in my case Azure DNS, and let’s add a new Record Set
data:image/s3,"s3://crabby-images/d475b/d475bef5535bb6b7deeb20f83eb8c0d76796d0a0" alt="azure dns new record set blog vinicius deschamps"
The step below is to check domain ownership, and you should fill the information as follows
- Name – asuid.SUBDOMAIN (e.g asuid.prodcm)
- Type – TXT
- Value – Paste the value you copied from Custom Domain Verification ID
data:image/s3,"s3://crabby-images/2c022/2c022087df1ab3c2b678552bb3beeb8b24eed013" alt="azure dns add record set custom domain verification txt blog vinicius deschamps"
Back to the Custom Domains, and click Add custom domain (3)
data:image/s3,"s3://crabby-images/a543b/a543b388071a629bf5e7245b47ca5c7a2e4a7240" alt="azure app service custom domains add custom domain blog vinicius deschamps"
Type your Custom Domain (1), then click Validate (2)
data:image/s3,"s3://crabby-images/a9a0d/a9a0dacb2ffbc8e0ca8398ed2102bdff5cbf5ae9" alt="azure app service add custom domain validate blog vinicius deschamps"
Sometimes it take a while until the record is sync, and you might see the following message
Domain ownership
To verify domain ownership create CNAME and TXT record with your DNS provider
Since we already have a CNAME configured for the Azure Front Door, the CNAME record is not required
data:image/s3,"s3://crabby-images/64026/640267d0d0ac3aa04f918a6b5cad339fab86e659" alt="azure app service add custom domain add custom domain domain ownership blog vinicius deschamps"
Wait for a while and try to Validate the Custom Domain again, and if everything is OK, you should see the following and hit Add custom domain
data:image/s3,"s3://crabby-images/65dd3/65dd316c6b6bb995296508b084b3f6e152d84287" alt="azure app service add custom domain add custom domain blog vinicius deschamps"
IMPORTANT: You should add the same custom domain to every CM App Service!
And finally, when you access your Azure Front Door custom domain to hit Sitecore you will notice that now you have a single URL for all Content Management, which in my case is https://prodcm.viniciusdeschamps.cloud
data:image/s3,"s3://crabby-images/240c6/240c60852d9bb47aa9e6bb25b424f193c8ca055c" alt="sitecore using azure front door custom domain blog vinicius deschamps"
And that’s it!
I hope you liked, and I’ll see you on my next post.
Photo by Marius Masalar on Unsplash
Deixe um comentário