The Azure Functions Files Episode 1 - The Case of the Consumption Plan File shares
For the last few months I’ve been building a large enterprise API composed of many (30+) Azure Functions v2 microservices, with an Azure API Management Gateway in front of them.
This week I hit one of those classic Azure pitfalls that are incredibly difficult and time-consuming to track down.
Somethings not right…
There was a minor string formatting bug in one of the microservices. No problem, easy fix.
I write a patch, run my unit and integration tests. Perfect. Everything is looking nice, green and regression free.
I do some manual testing locally and all signals are go. So I commit and push my changes and wait for CI and CD agents in Azure Pipelines to do their thing.
Once the service is deployed, I start my last round of manual testing in our Azure Staging environment.
… The changes I’d just deployed were not there. The endpoint was working, but the formatting bug was still there.
Trial and Azure
So I did some detective work:
- Check Azure Pipelines - Everything deployed OK
- Check Kudu - Logged into Kudu and checked all the directories with the console. Everything fine there too.
- Wrote and deployed a patch that forced all of the strings in the endpoint’s response payload to upper-case. A super obvious change, no chance I’m missing that… It didn’t deploy either.
After an hour or so, I was stumped. So I started reading through the Azure docs and I found this hidden away in there
The part that caught my attention is below:
WEBSITE_CONTENTSHARE For consumption plans only. The file path to the function app code and configuration. Used with WEBSITE_CONTENTAZUREFILECONNECTIONSTRING. Default is a unique string that begins with the function app name.
Consumption Plans are scalable in ways that App Service plans are not, so when we can, we deploy our Functions to Consumption Plans, and the Function in question was on a Consumption Plan.
So I checked the Functions App Settings. Everything looked good. We had both a WEBSITE_CONTENTAZUREFILECONNECTIONSTRING
and WEBSITE_CONTENTSHARE
.
At this point I got curious, and lucky I did otherwise this would have taken me days to find.
So I created a new Function App Instance and a Fresh CI/CD Pipeline to deploy to it and then compared the settings on the old set up to the new setup.
Suspicious Storage
When you create a new Function App instance in the Azure Portal, you need to create a couple of support resources, namely:
- A Storage Account; and
- An optional Application Insights instance
For Function Apps that run on an App Service Plan, the Function is deployed to a dynamically provisioned VM.
But… in the case of Functions that run on Consumption Plans. The Function App is deployed to a File Share that is automatically created under the Storage Account for that Function App Instance.
So I opened up Azure Storage Explorer and did some snooping around in the Storage Accounts list and I found this:
There were 2 File Shares.
Resolution
The creation of file shares for Consumption Plans is entirely automated by Azure. So it took my a while to figure out what had happened.
At some point, the WEBSITE_CONTENTSHARE
Application Setting had been deleted. But instead of failing to deploy, Azure had silently recovered by creating a new File Share under the Storage Account and deploying the Function App there.
But, during the creation of a CD Stage in Azure Pipelines, when you link a Consumption Plan Function App in Azure Pipelines as a deployment target, it fetches the File Share to deploy to and stores it as the upload target for the Release.
So we were deploying to the old file share, but the Function App settings were pointing to the new file share, which, while it was running a working build of the Function App, wasn’t getting any of the updates.