The action uses a GitHub repository as a cache for a .conan directory to speed up very slow builds. It was made specifically to offset the cost of a conan-qt build with QtWebEngine turned on. The cache can be populated in the GitHub action pipeline, or offline on a computer and then pushed to the repo. This second workflow is to help with builds that are too slow for GitHub Actions, where a step cannot take longer than 6 hours, and has limited disk space. See below for how to do an offline filling of the cache.
The setup process for the action requires the creation of a bot account and a GitHub repo to hold the cache. See below for details.
When to use: The conan-cache action was created to make CI builds possible for projects where building conan modules takes a lot of time, "a lot of time" is here measured in hours. The cache is also useful for local builds. If you have an artifactory then that is probably a better solution than this. Upload a prebuilt artifact there. (That might be added as an option to conan-cache in the future).
Works on: Linux, Windows and MacOS
This action was inspired by the builtin GitHub Cache Action, which might be more than adequate for your needs. See here for how you can use it for conan modules. The builtin GitHub Cache Action has some limitations, however, and tends to have a lot of intermittant failures to retrieve the cache, which can be unacceptable when building without takes a long time.
See also this trick using a combination of GitHub Cache Action and Conan Cache.
Required The name of the GitHub bot
Required The personal access token of the GitHub bot
Required The GitHub repo used for this cache
Required An explicit key to store this cache under
Optional Target OS if different from host OS
Optional In number of MB. Files with a size larger than lfs_limit are added to Git Large File Storage (LFS), defaults to 50MB, GitHub supports max 100MB, and will generate a warning at 50MB files.
The state of the cache hit: no hit (0), hit on explicit key (1), hit on fallback key (2). (An explicit key is represented in the repo as a tag, a fallback key is represented as a branch)
Assumed environment
env:
CONAN_USER_HOME: "${{ github.workspace }}/release/"
CONAN_USER_HOME_SHORT: "${{ github.workspace }}/release/short"
Action setup
- name: Cache Conan modules
id: cache-conan
uses: turtlebrowser/conan-cache@master
with:
bot_name: ${{ secrets.BOT_NAME }}
bot_token: ${{ secrets.BOT_TOKEN }}
cache_name: ${{ env.CACHE_GITHUB }}/${{ env.CACHE_GITHUB_REPO }}
key: host-${{ runner.os }}-target-${{ runner.os }}-${{ hashFiles('conanfile.py') }}
target_os: ${{ runner.os }}
lfs_limit: 60
After conan has filled the cache, clean it up
- name: Clean up Conan
run: |
conan remove -f "*" --builds
conan remove -f "*" --src
conan remove -f "*" --system-reqs
Use the cache-hit output
- name: On cache miss add bincrafters remote
if: ${{ steps.cache-conan.outputs.cache-hit == 0 }}
run: conan remote add bincrafters https://api.bintray.com/conan/bincrafters/public-conan
A trick that can be used is to use the GitHub Cache Action as a first level cache. It has limited space and often fails to fetch the cache for unknown reasons, but many times it will suffice and you can save on git lfs bandwith costs. Here is an example of how that could look. Note that the example doesn't use restore-keys, that is because the GitHub Cache Action's cache-hit is a boolean that will only signal if the key was hit (this is not how it works in Conan Cache). Also, if the key doesn't hit, then you most likely have a change in your conanfile.py and should fetch from Conan Cache anyway.
Note: Make sure it has been added to the Conan Cache properly using this action first, otherwise the short paths will be wrong on Windows. They are fixed in save.sh.
# Check if GitHub Cache has it, because that's free
- name: Using the builtin GitHub Cache Action for .conan
id: github-cache-conan
uses: actions/cache@v1
env:
cache-name: cache-conan-modules
with:
path: ${{ env.CONAN_USER_HOME }}
key: host-${{ runner.os }}-target-${{ runner.os }}-${{ hashFiles('conanfile.py') }}
# If GitHub Cache doesn't have it, get from Conan Cache (has git lfs cost)
- name: Cache Conan modules
if: steps.github-cache-conan.outputs.cache-hit != 'true'
id: cache-conan
uses: turtlebrowser/conan-cache@master
with:
bot_name: ${{ secrets.BOT_NAME }}
bot_token: ${{ secrets.BOT_TOKEN }}
cache_name: ${{ env.CACHE_GITHUB }}/${{ env.CACHE_GITHUB_REPO }}
key: host-${{ runner.os }}-target-${{ runner.os }}-${{ hashFiles('conanfile.py') }}
target_os: ${{ runner.os }}
lfs_limit: 60
- Create a GitHub bot account that will be used to manage the conan cache
- Create a GitHub repo for your conan cache
- Add your bot as a collaborator on the cache with write access
- Create a personal access token for your bot [https://help.github.com/en/github/authenticating-to-github/creating-a-personal-access-token-for-the-command-line]
- Add the personal access token as a secret to your repos workflow [https://help.github.com/en/actions/configuring-and-managing-workflows/creating-and-storing-encrypted-secrets]
- Follow the instructions below to set up an empty cache
GitHub account that has the cache repo
Name of the cache repo
Conan variable, sets the directory in which .conan will be found
Conan variable, sets the directory which will be used for Conan short paths (Windows only)
The file size at which a file should be added to Git LFS rather than to git. GitHub sets a max file size of 100MB.
This is the structure that is expected by conan-cache of an empty cache
git clone git@github.com:${CACHE_GITHUB}/${CACHE_GITHUB_REPO}.git
cd ${CACHE_GITHUB_REPO}
mkdir .conan && touch .conan/conan-cache.marker
touch .gitattributes
Fill the .gitattributes with this contents (especially important for Windows)
$ cat .gitattributes
* -text
mkdir short && touch short/conan-cache.marker
git add -A
git commit -m "Setup cache"
git push
git clone git@github.com:${CACHE_GITHUB}/${CACHE_GITHUB_REPO}.git
cd ${CACHE_GITHUB_REPO}
git checkout <branch>
git lfs pull
export CONAN_USER_HOME="c:/release"
export CONAN_USER_HOME_SHORT=${CONAN_USER_HOME}/short
find .conan/ -name .conan_link -exec perl -pi -e 's=CONAN_USER_HOME_SHORT=$ENV{CONAN_USER_HOME_SHORT}=g' {} +
export CONAN_USER_HOME="c:/release"
export CONAN_USER_HOME_SHORT=${CONAN_USER_HOME}/short
git clone git@github.com:${CACHE_GITHUB}/${CACHE_GITHUB_REPO}.git $CONAN_USER_HOME
cd $CONAN_USER_HOME
git checkout -b <branch>
git push -u origin <branch>
export CONAN_USER_HOME="c:/release"
export CONAN_USER_HOME_SHORT=${CONAN_USER_HOME}/short
git clone git@github.com:${CACHE_GITHUB}/${CACHE_GITHUB_REPO}.git $CONAN_USER_HOME
cd $CONAN_USER_HOME
git clean -df
git checkout .
git checkout <branch>
git pull
git lfs pull
find .conan/ -name .conan_link -exec perl -pi -e 's=CONAN_USER_HOME_SHORT=$ENV{CONAN_USER_HOME_SHORT}=g' {} +
The final step is to insert the local CONAN_USER_HOME_SHORT path instead of the placeholder
This step assumes it is being run in the same shell as one of the prep steps.
Build the target project, this will fill the cache. Then remove traces of the build process. Replace hardcoded paths with the placeholder CONAN_USER_HOME_SHORT. Then add any files exeeding the LFS limit for the project to git lfs. Add/remove everything as is with git add -A, commit and push. Start a build on GitHub and the branch will be tested and tagged automatically.
cd <path to project checkout>
git pull
rm -rf build
# Make a cmake switch to turn on UPDATE build obsolete:
# mkdir build && cd build && cmake -DUPDATE_CONAN=ON -DCMAKE_BUILD_TYPE=Release ..
mkdir build && cd build && cmake -DCMAKE_BUILD_TYPE=Release ..
conan remove -f "*" --builds
conan remove -f "*" --src
conan remove -f "*" --system-reqs
cd $CONAN_USER_HOME
find .conan/ -name .conan_link -exec perl -pi -e 's=$ENV{CONAN_USER_HOME_SHORT}=CONAN_USER_HOME_SHORT/=g' {} +
export LFS_LIMIT=50
find .conan short -type f -size +${LFS_LIMIT}M -exec ls -lh {} \; | awk '{ print $9 ": " $5 }'
find .conan short -type f -size +${LFS_LIMIT}M -execdir git lfs track {} \;
git add -A
git commit -m "Local build"
git push
Assumed environment
env:
CONAN_USER_HOME: "${{ github.workspace }}/release/"
CONAN_USER_HOME_SHORT: "${{ github.workspace }}/release/short"
Action setup
- name: Using the builtin GitHub Cache Action for .conan
if: matrix.os == 'windows-latest'
id: cache-conan
uses: actions/cache@v1
env:
cache-name: cache-conan-modules
with:
path: ${{ env.CONAN_USER_HOME }}
key: ${{ runner.os }}-builder-${{ env.cache-name }}-${{ hashFiles('conanfile.py') }}
restore-keys: ${{ runner.os }}-builder-${{ env.cache-name }}-