I spent a good bit of time over the last two weeks converting our Azure functions from the in-process
to the isolated worker
process model. Overall the transition was fairly simple, but there were a few bumps in the proverbial road worth noting.
Migration Process
Microsoft Learn has a very detailed How To Guide for this migration. The guide includes steps for updating the project file and references, as well as additional packages that are required based on various trigger types.
Since I had a number of functions to process, I followed the guide for the first one, and that worked swimmingly. However, then I got lazy and started the “copy-paste” conversion. In that laziness, I missed a particular section of the project file:
<ItemGroup>
<Using Include="System.Threading.ExecutionContext" Alias="ExecutionContext"/>
</ItemGroup>
Unfortunately, if you forget this, you will not break your local development environment. However, when you publish to a function, it will not execute correctly.
Fixing Dependency Injection
When using the in-process
model, there are some “freebies” that get added to the dependency injection system, as if by magic. ILogger
, in particular, was allowed to be automatically injected into the function (as a function parameter). However, in the in-process
model, you must get ILogger
from either the FunctionContext
or through dependency injection into the class.
As part of our conversion, we removed the function parameters for ILogger and replaced them with service instances retrieved through dependency injection at the class level.
What we did not realize until we got our functions into the test environments was that IHttpContextAccessor
was not available in the isolated
model. Apparently, that particular interface is available as part of the in-process
model automatically, but is not added as part of the isolated
model. So we had to add an instance of IHttpContextAccessor
to our services collection in the Program.cs
file.
It is never easy
Upgrades or migrations are never just “change this and go.” as much as we try to make it easy, there always seems to be a little change here or there that end up being a fly in the ointment. In our case, we simply assumed that IHttpContextAccessor
was there because in-process
put it there, and the code which needed that was a few layers deep in the dependency tree. The only way to find it was to make the change and see what breaks. And that is what keeps quality engineers up at night.