Writing effecient Dockerfiles is important to keep us productive. Ideally, you want to be able to break your build into cacheable layers for things that don't change that often, like dependencies, so they can be skipped next time; but a huge pain point is that Docker doesn't support globbing in their COPY command. As such, the community has had a fun time trying to work around this limitation as nearly every programming stack and/or package manager typically relies on a directory structure for determining dependencies and restoring them.
Andrew Lock's blog posts show his progression through this challenge and with the help of some readers he finally settled a few approaches documented here.
Here's another approach, taking advantage of the fact that dotnet core global tools are cross-platform and (in my case anyway) we've already got dotnet core installed in the base Docker image.
First, install this tool by adding the following line to your Dockerfile.
Note: it's probably a good idea to pin to a specific version to avoid breaking changes by appending --version 0.0.x
.
RUN dotnet tool install -g dotnet-references
If you're using a Linux image, you'll need to then add the following to make the tool available to be called directly:
ENV PATH="${PATH}:/root/.dotnet/tools"
Next, copy over a .sln file that serves as an "entry point" for tracking down the rest of your .csproj files. It must be copied into the correct location relative to the "root".
COPY MyCompany.MySolution.sln ./The/Correct/Directory/Path/
(Alternatively, you can use a .csproj as an entry, coming soon...)
Copy over your .csproj files using whatever technique works best for your situation.
# This results in a layer for each step, but it is what it is...
COPY *.csproj ./
COPY */*.csproj ./
COPY */*/*.csproj ./
COPY some-other-dir/*/*.csproj ./
Run this dotnet global tool at your sln file (or csproj file, coming soon), give it the "root" where all the .csproj files were copied, and optionally pass true
to delete .csproj files that aren't actuallly in the dependency graph from the entry point.
dotnet references fix -ep ./The/Correct/Directory/Path/MyCompany.MySolution.sln -wd . -rupf
(see full details of args here)
⚠️ If the dotnet cli complains with "Specify which project file to use because this '...' contains more than one project file.", you've run into a limitation of the cli. Essentially it can't execute when the current working directory you're in contains more than one project file (and potentially even solution files too). Since we can pass anything to the -ep and -wd arguments, it doesn't matter what directory you executedotnet references
from, so the easiest way to workaround this is to just go to a random directory (you've already got or that you create) and execute from there, adjusting the -ep and -wd arguments as necessary. (eg.RUN mkdir temp && cd temp && dotnet references fix -ep ../MyCompany.MySolution.sln -wd .. -rupf
)
You can run a command like so after to validate the files are now in the right structure, ready for a NuGet restore to be fired.
RUN ls . -alR
(todo, windows cmd)
You'll now be able to run a restore
against the same entry point, and Docker will cache the result in a layer.
Then just complete your Dockerfile as per usual, copying over the rest of your src, doing your build
, test
, publish
, etc. as needed.
Running each separately may seem counter-intuitive, but allows Docker to cache at each layer. When running build
, test
or publish
, you can also pass --no-restore
to explicitly document that the previous restore
step has taken care of that.
I hope you've found this helpful and you've been able to progress through what's quite a frustrating functional hole in Dockerfile builds. BuildKit is on the way, hopefully that alleviates so much hoop jumping.
Thanks to Andrew Lock and his readers for sharing their approaches with the community.