mirror of
https://github.com/cpm-cmake/CPM.cmake.git
synced 2025-11-17 22:58:14 -05:00
Compare commits
35 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a575ab57b7 | ||
|
|
ca421f0e0c | ||
|
|
005f202655 | ||
|
|
865648e7bd | ||
|
|
33bdbae902 | ||
|
|
9d3fad01d4 | ||
|
|
11c7ec8afa | ||
|
|
4f7af69925 | ||
|
|
c58e98a0a0 | ||
|
|
a27c66a3ea | ||
|
|
5cb7398a24 | ||
|
|
262f1e0602 | ||
|
|
7cbef3efc8 | ||
|
|
939123d1b4 | ||
|
|
22078d5e0d | ||
|
|
96c268827b | ||
|
|
dad37fbe4e | ||
|
|
fed5f8e8a2 | ||
|
|
d65613e860 | ||
|
|
718ea71759 | ||
|
|
6491382204 | ||
|
|
634800c619 | ||
|
|
f552da96bd | ||
|
|
ea6a8eb895 | ||
|
|
a5c22bf6e8 | ||
|
|
c5cb85b2f1 | ||
|
|
91585e3864 | ||
|
|
9675d46517 | ||
|
|
7078e8286a | ||
|
|
6a0277f16e | ||
|
|
4502bf1e04 | ||
|
|
dd3ba9792c | ||
|
|
de5551e42c | ||
|
|
7644c3a40f | ||
|
|
310efb9b17 |
@@ -34,6 +34,7 @@ parse:
|
||||
HTTP_USERNAME: 1
|
||||
HTTP_PASSWORD: 1
|
||||
EXCLUDE_FROM_ALL: 1
|
||||
SOURCE_SUBDIR: 1
|
||||
OPTIONS: +
|
||||
cpmfindpackage:
|
||||
pargs:
|
||||
|
||||
30
.github/workflows/examples.yml
vendored
Normal file
30
.github/workflows/examples.yml
vendored
Normal file
@@ -0,0 +1,30 @@
|
||||
name: Examples
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
gcc:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: build all
|
||||
env:
|
||||
CC: gcc
|
||||
CXX: g++
|
||||
run: python3 examples/build_all.py
|
||||
|
||||
clang:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: build all
|
||||
env:
|
||||
CC: clang
|
||||
CXX: clang++
|
||||
run: python3 examples/build_all.py
|
||||
22
.github/workflows/macos.yml
vendored
22
.github/workflows/macos.yml
vendored
@@ -1,22 +0,0 @@
|
||||
name: MacOS
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: macos-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v1
|
||||
|
||||
- name: run tests
|
||||
run: |
|
||||
cmake -Htest -Bbuild/test
|
||||
cmake --build build/test --target test-verbose
|
||||
2
.github/workflows/style.yml
vendored
2
.github/workflows/style.yml
vendored
@@ -22,7 +22,7 @@ jobs:
|
||||
pip3 install cmake_format==0.6.11 pyyaml
|
||||
|
||||
- name: configure
|
||||
run: cmake -Htest/style -Bbuild/style
|
||||
run: cmake -Stest/style -Bbuild/style
|
||||
|
||||
- name: check style
|
||||
run: cmake --build build/style --target check-format
|
||||
|
||||
33
.github/workflows/test.yml
vendored
Normal file
33
.github/workflows/test.yml
vendored
Normal file
@@ -0,0 +1,33 @@
|
||||
name: Test
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
tests:
|
||||
runs-on: ${{ matrix.os }}
|
||||
strategy:
|
||||
matrix:
|
||||
# windows-latest is windows-2019 which carries a pretty old version of ruby (2.5)
|
||||
# we need at least ruby 2.7 for the tests
|
||||
# instead of dealing with installing a modern version of ruby on 2019, we'll just use windows-2022 here
|
||||
os: [ubuntu-latest, windows-2022, macos-latest]
|
||||
|
||||
steps:
|
||||
- name: clone
|
||||
uses: actions/checkout@v2
|
||||
|
||||
- name: unit tests
|
||||
run: |
|
||||
cmake -Stest -Bbuild/test
|
||||
cmake --build build/test --target test-verbose
|
||||
|
||||
- name: integration tests
|
||||
run: ruby test/integration/runner.rb
|
||||
env:
|
||||
CPM_INTEGRATION_TEST_DIR: ./build/integration
|
||||
29
.github/workflows/ubuntu.yml
vendored
29
.github/workflows/ubuntu.yml
vendored
@@ -1,29 +0,0 @@
|
||||
name: Ubuntu
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v1
|
||||
|
||||
- name: install updates
|
||||
run: |
|
||||
wget -O cmake.sh https://cmake.org/files/v3.14/cmake-3.14.0-Linux-x86_64.sh
|
||||
sudo sh cmake.sh --skip-license --exclude-subdir --prefix=/usr/local
|
||||
export PATH=/usr/local/bin:$PATH
|
||||
cmake --version
|
||||
|
||||
- name: run tests
|
||||
run: |
|
||||
cmake -Htest -Bbuild/test
|
||||
cmake --build build/test --target test-verbose
|
||||
22
.github/workflows/windows.yml
vendored
22
.github/workflows/windows.yml
vendored
@@ -1,22 +0,0 @@
|
||||
name: Windows
|
||||
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- master
|
||||
pull_request:
|
||||
branches:
|
||||
- master
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: windows-latest
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v1
|
||||
|
||||
- name: run tests
|
||||
run: |
|
||||
cmake -Htest -Bbuild/test
|
||||
cmake --build build/test --target test-verbose
|
||||
50
.travis.yml
50
.travis.yml
@@ -1,50 +0,0 @@
|
||||
language: cpp
|
||||
sudo: require
|
||||
dist: xenial
|
||||
|
||||
common_sources: &all_sources
|
||||
- ubuntu-toolchain-r-test
|
||||
- llvm-toolchain-trusty
|
||||
|
||||
python:
|
||||
- 3.7
|
||||
|
||||
matrix:
|
||||
include:
|
||||
- os: linux
|
||||
compiler: gcc
|
||||
addons: &gcc8
|
||||
apt:
|
||||
sources: *all_sources
|
||||
packages:
|
||||
- g++-8
|
||||
env:
|
||||
- MATRIX_EVAL="export CC=gcc-8; export CXX=g++-8;"
|
||||
|
||||
- os: linux
|
||||
compiler: clang
|
||||
addons:
|
||||
apt:
|
||||
sources: *all_sources
|
||||
packages:
|
||||
- g++-8
|
||||
- clang-6.0
|
||||
env:
|
||||
- MATRIX_EVAL="export CC=clang-6.0; export CXX=clang++-6.0;"
|
||||
|
||||
before_install:
|
||||
# Update compilers
|
||||
- eval "${MATRIX_EVAL}"
|
||||
- echo "CC=$CC CXX=$CXX"
|
||||
# Install a supported cmake version (>= 3.14)
|
||||
- wget -O cmake.sh https://cmake.org/files/v3.14/cmake-3.14.0-Linux-x86_64.sh
|
||||
- sudo sh cmake.sh --skip-license --exclude-subdir --prefix=/usr/local
|
||||
- export PATH=/usr/local/bin:$PATH
|
||||
- cmake --version
|
||||
|
||||
script:
|
||||
# unit tests
|
||||
- cmake -Htest -Bbuild/test
|
||||
- cmake --build build/test --target test-verbose
|
||||
# build examples
|
||||
- python3 examples/build_all.py
|
||||
32
CONTRIBUTORS.md
Normal file
32
CONTRIBUTORS.md
Normal file
@@ -0,0 +1,32 @@
|
||||
# CPM.cmake Contributors
|
||||
|
||||
For detailed breakdown of individual contributions see the [Contributors page on GitHub](https://github.com/cpm-cmake/CPM.cmake/graphs/contributors)
|
||||
|
||||
* [Lars Melchior](https://github.com/TheLartians)
|
||||
* [Borislav Stanimirov](https://github.com/iboB)
|
||||
* [Paul T](https://github.com/DeveloperPaul123)
|
||||
* [Leonardo Lima](https://github.com/leozz37)
|
||||
* [pgorgon-hem](https://github.com/pgorgon-hem)
|
||||
* [Xavier Muller](https://github.com/xmuller)
|
||||
* NVIDIA CORPORATION via [Robert Maynard](https://github.com/robertmaynard)
|
||||
* [Kingsley Chen](https://github.com/kingsamchen)
|
||||
* [alexandreSalconiDenis](https://github.com/alexandreSalconiDenis)
|
||||
* [DNKpp](https://github.com/DNKpp)
|
||||
* [Olivier Le Doeuff](https://github.com/OlivierLDff)
|
||||
* [Stuart Dootson](https://github.com/studoot)
|
||||
* [Tobias Hellmann](https://github.com/Tobi823)
|
||||
* [Giuseppe Cesarano](https://github.com/GiuseppeCesarano)
|
||||
* [Pavel Sokolov](https://github.com/hacker-cb)
|
||||
* [Claus Klein](https://github.com/ClausKlein)
|
||||
* [Clare Macrae](https://github.com/claremacrae)
|
||||
* [Kai Germaschewski](https://github.com/germasch)
|
||||
* [Andrea Barbadoro](https://github.com/andijcr)
|
||||
* [MixusMinimax](https://github.com/MixusMinimax)
|
||||
* [Dan Raviv](https://github.com/danra)
|
||||
* [Prabir Shrestha](https://github.com/prabirshrestha)
|
||||
* [Paul Taylor](https://github.com/trxcllnt)
|
||||
* [Ryan Mast](https://github.com/nightlark)
|
||||
* [Vinpasso](https://github.com/Vinpasso)
|
||||
* [Yotam Gingold](https://github.com/yig)
|
||||
* [jecassis](https://github.com/jecassis)
|
||||
* [Johel Ernesto Guerrero Peña](https://github.com/JohelEGP)
|
||||
2
LICENSE
2
LICENSE
@@ -1,6 +1,6 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2019 Lars Melchior
|
||||
Copyright (c) 2019-2022 Lars Melchior and contributors
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
|
||||
115
README.md
115
README.md
@@ -1,7 +1,3 @@
|
||||
[](https://travis-ci.com/cpm-cmake/CPM.cmake)
|
||||
[](https://github.com/cpm-cmake/CPM.cmake/actions)
|
||||
[](https://github.com/cpm-cmake/CPM.cmake/actions)
|
||||
[](https://github.com/cpm-cmake/CPM.cmake/actions)
|
||||
|
||||
<br />
|
||||
<p align="center">
|
||||
@@ -11,7 +7,7 @@
|
||||
|
||||
# Setup-free CMake dependency management
|
||||
|
||||
CPM.cmake is a CMake script that adds dependency management capabilities to CMake.
|
||||
CPM.cmake is a cross-platform CMake script that adds dependency management capabilities to CMake.
|
||||
It's built as a thin wrapper around CMake's [FetchContent](https://cmake.org/cmake/help/latest/module/FetchContent.html) module that adds version control, caching, a simple API [and more](#comparison-to-pure-fetchcontent--externalproject).
|
||||
|
||||
## Manage everything
|
||||
@@ -44,7 +40,7 @@ CPMAddPackage(
|
||||
The origin may be specified by a `GIT_REPOSITORY`, but other sources, such as direct URLs, are [also supported](https://cmake.org/cmake/help/v3.11/module/ExternalProject.html#external-project-definition).
|
||||
If `GIT_TAG` hasn't been explicitly specified it defaults to `v(VERSION)`, a common convention for git projects.
|
||||
On the other hand, if `VERSION` hasn't been explicitly specified, CPM can automatically identify the version from the git tag in some common cases.
|
||||
`GIT_TAG` can also be set to a specific commit or a branch name such as `master` to always download the most recent version.
|
||||
`GIT_TAG` can also be set to a specific commit or a branch name such as `master`, however this isn't recommended, as such packages will only be updated when the cache is cleared.
|
||||
|
||||
If an additional optional parameter `EXCLUDE_FROM_ALL` is set to a truthy value, then any targets defined inside the dependency won't be built by default. See the [CMake docs](https://cmake.org/cmake/help/latest/prop_tgt/EXCLUDE_FROM_ALL.html) for details.
|
||||
|
||||
@@ -53,13 +49,13 @@ A single-argument compact syntax is also supported:
|
||||
```cmake
|
||||
# A git package from a given uri with a version
|
||||
CPMAddPackage("uri@version")
|
||||
# A git package from a given uri with a git tag or commit hash, or branch name
|
||||
# A git package from a given uri with a git tag or commit hash
|
||||
CPMAddPackage("uri#tag")
|
||||
# A git package with both version and tag provided
|
||||
CPMAddPackage("uri@version#tag")
|
||||
```
|
||||
|
||||
In the shorthand syntax if the URI is of the form `gh:user/name`, it is interpreted as GitHub URI and converted to `https://github.com/user/name.git`. If the URI is of the form `gl:user/name`, it is interpreted as a [GitLab](https://gitlab.com/explore/) URI and coverted to `https://gitlab.com/user/name.git`. Otherwise the URI used verbatim as a git URL. All packages added using the shorthand syntax will be added using the [EXCLUDE_FROM_ALL](https://cmake.org/cmake/help/latest/prop_tgt/EXCLUDE_FROM_ALL.html) flag.
|
||||
In the shorthand syntax if the URI is of the form `gh:user/name`, it is interpreted as GitHub URI and converted to `https://github.com/user/name.git`. If the URI is of the form `gl:user/name`, it is interpreted as a [GitLab](https://gitlab.com/explore/) URI and converted to `https://gitlab.com/user/name.git`. If the URI is of the form `bb:user/name`, it is interpreted as a [Bitbucket](https://bitbucket.org/) URI and converted to `https://bitbucket.org/user/name.git`. Otherwise the URI used verbatim as a git URL. All packages added using the shorthand syntax will be added using the [EXCLUDE_FROM_ALL](https://cmake.org/cmake/help/latest/prop_tgt/EXCLUDE_FROM_ALL.html) flag.
|
||||
|
||||
The single-argument syntax also works for URLs:
|
||||
|
||||
@@ -156,7 +152,7 @@ CPM.cmake is a wrapper for CMake's FetchContent module and adds a number of feat
|
||||
The most notable features are:
|
||||
|
||||
- A simpler to use API
|
||||
- Version checking: CPM.cmake will check the version number of any added dependency and omit a warning if another dependency requires a more recent version.
|
||||
- Version checking: CPM.cmake will check the version number of any added dependency and emit a warning if another dependency requires a more recent version.
|
||||
- Offline builds: CPM.cmake will override CMake's download and update commands, which allows new builds to be configured while offline if all dependencies [are available locally](#cpm_source_cache).
|
||||
- Automatic shallow clone: if a version tag (e.g. `v2.2.0`) is provided and `CPM_SOURCE_CACHE` is used, CPM.cmake will perform a shallow clone of the dependency, which should be significantly faster while using less storage than a full clone.
|
||||
- Overridable: all `CPMAddPackage` can be configured to use `find_package` by setting a [CMake flag](#cpm_use_local_packages), making it easy to integrate into projects that may require local versioning through the system's package manager.
|
||||
@@ -187,6 +183,7 @@ You can use `CPM_SOURCE_CACHE` on GitHub Actions workflows [cache](https://githu
|
||||
If set, CPM will forward all calls to `CPMFindPackage` as `CPMAddPackage`.
|
||||
This is useful to create reproducible builds or to determine if the source parameters have all been set correctly.
|
||||
This can also be set as an environmental variable.
|
||||
This can be controlled on a per package basis with the `CPM_DOWNLOAD_<dependency name>` variable.
|
||||
|
||||
### CPM_USE_LOCAL_PACKAGES
|
||||
|
||||
@@ -196,6 +193,11 @@ These options can also be set as environmental variables.
|
||||
|
||||
In the case that `find_package` requires additional arguments, the parameter `FIND_PACKAGE_ARGUMENTS` may be specified in the `CPMAddPackage` call. The value of this parameter will be forwarded to `find_package`.
|
||||
|
||||
### CPM_USE_NAMED_CACHE_DIRECTORIES
|
||||
|
||||
If set, CPM use additional directory level in cache to improve readability of packages names in IDEs like CLion. It changes cache structure, so all dependencies are downloaded again. There is no problem to mix both structures in one cache directory but then there may be 2 copies of some dependencies.
|
||||
This can also be set as an environmental variable.
|
||||
|
||||
## Local package override
|
||||
|
||||
Library developers are often in the situation where they work on a locally checked out dependency at the same time as on a consumer project.
|
||||
@@ -256,6 +258,14 @@ If you know others, feel free to add them here through a PR.
|
||||
<p align="center"><b>ModernCppStarter</b></p>
|
||||
</a>
|
||||
</td>
|
||||
<td>
|
||||
<a href="https://git.io/liblava">
|
||||
<p align="center">
|
||||
<img src="https://github.com/liblava.png" alt="liblava" width="100pt" />
|
||||
</p>
|
||||
<p align="center"><b>liblava - Modern Vulkan library</b></p>
|
||||
</a>
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
@@ -270,28 +280,55 @@ See the [wiki](https://github.com/cpm-cmake/CPM.cmake/wiki/More-Snippets) for mo
|
||||
CPMAddPackage("gh:catchorg/Catch2@2.5.0")
|
||||
```
|
||||
|
||||
### [Boost (via boost-cmake)](https://github.com/Orphis/boost-cmake)
|
||||
### [Range-v3](https://github.com/ericniebler/range-v3)
|
||||
|
||||
```CMake
|
||||
CPMAddPackage("gh:Orphis/boost-cmake@1.67.0")
|
||||
```
|
||||
|
||||
### [cxxopts](https://github.com/jarro2783/cxxopts)
|
||||
|
||||
```cmake
|
||||
CPMAddPackage(
|
||||
GITHUB_REPOSITORY jarro2783/cxxopts
|
||||
VERSION 2.2.1
|
||||
OPTIONS "CXXOPTS_BUILD_EXAMPLES NO" "CXXOPTS_BUILD_TESTS NO" "CXXOPTS_ENABLE_INSTALL YES"
|
||||
)
|
||||
```Cmake
|
||||
CPMAddPackage("gh:ericniebler/range-v3#0.11.0")
|
||||
```
|
||||
|
||||
### [Yaml-cpp](https://github.com/jbeder/yaml-cpp)
|
||||
|
||||
```CMake
|
||||
# as the tag is in an unusual format, we need to explicitly specify the version
|
||||
CPMAddPackage("gh:jbeder/yaml-cpp#yaml-cpp-0.6.3@0.6.3")
|
||||
```
|
||||
|
||||
### [nlohmann/json](https://github.com/nlohmann/json)
|
||||
|
||||
```cmake
|
||||
CPMAddPackage(
|
||||
NAME nlohmann_json
|
||||
VERSION 3.9.1
|
||||
GITHUB_REPOSITORY nlohmann/json
|
||||
OPTIONS
|
||||
"JSON_BuildTests OFF"
|
||||
)
|
||||
```
|
||||
|
||||
### [Boost ](https://github.com/boostorg/boost)
|
||||
|
||||
```CMake
|
||||
# boost is a huge project and will take a while to download
|
||||
# using `CPM_SOURCE_CACHE` is strongly recommended
|
||||
CPMAddPackage(
|
||||
NAME Boost
|
||||
VERSION 1.77.0
|
||||
GITHUB_REPOSITORY "boostorg/boost"
|
||||
GIT_TAG "boost-1.77.0"
|
||||
)
|
||||
```
|
||||
|
||||
### [cxxopts](https://github.com/jarro2783/cxxopts)
|
||||
|
||||
```cmake
|
||||
# the install option has to be explicitly set to allow installation
|
||||
CPMAddPackage(
|
||||
GITHUB_REPOSITORY jarro2783/cxxopts
|
||||
VERSION 2.2.1
|
||||
OPTIONS "CXXOPTS_BUILD_EXAMPLES NO" "CXXOPTS_BUILD_TESTS NO" "CXXOPTS_ENABLE_INSTALL YES"
|
||||
)
|
||||
```
|
||||
|
||||
### [google/benchmark](https://github.com/google/benchmark)
|
||||
|
||||
```cmake
|
||||
@@ -308,40 +345,6 @@ if(benchmark_ADDED)
|
||||
endif()
|
||||
```
|
||||
|
||||
### [nlohmann/json](https://github.com/nlohmann/json)
|
||||
|
||||
```cmake
|
||||
CPMAddPackage(
|
||||
NAME nlohmann_json
|
||||
VERSION 3.6.1
|
||||
# the git repo is incredibly large, so we download the archived include directory
|
||||
URL https://github.com/nlohmann/json/releases/download/v3.6.1/include.zip
|
||||
URL_HASH SHA256=69cc88207ce91347ea530b227ff0776db82dcb8de6704e1a3d74f4841bc651cf
|
||||
)
|
||||
|
||||
if (nlohmann_json_ADDED)
|
||||
add_library(nlohmann_json INTERFACE IMPORTED)
|
||||
target_include_directories(nlohmann_json INTERFACE ${nlohmann_json_SOURCE_DIR})
|
||||
endif()
|
||||
```
|
||||
|
||||
### [Range-v3](https://github.com/ericniebler/range-v3)
|
||||
|
||||
```Cmake
|
||||
CPMAddPackage(
|
||||
NAME range-v3
|
||||
URL https://github.com/ericniebler/range-v3/archive/0.5.0.zip
|
||||
VERSION 0.5.0
|
||||
# the range-v3 CMakeLists screws with configuration options
|
||||
DOWNLOAD_ONLY True
|
||||
)
|
||||
|
||||
if(range-v3_ADDED)
|
||||
add_library(range-v3 INTERFACE IMPORTED)
|
||||
target_include_directories(range-v3 INTERFACE "${range-v3_SOURCE_DIR}/include")
|
||||
endif()
|
||||
```
|
||||
|
||||
### [Lua](https://www.lua.org)
|
||||
|
||||
```cmake
|
||||
|
||||
274
cmake/CPM.cmake
274
cmake/CPM.cmake
@@ -5,7 +5,7 @@
|
||||
# MIT License
|
||||
# -----------
|
||||
#[[
|
||||
Copyright (c) 2021 Lars Melchior and additional contributors
|
||||
Copyright (c) 2019-2022 Lars Melchior and contributors
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
@@ -57,6 +57,13 @@ See https://github.com/cpm-cmake/CPM.cmake for more information."
|
||||
endif()
|
||||
endif()
|
||||
|
||||
if(CURRENT_CPM_VERSION MATCHES "development-version")
|
||||
message(WARNING "Your project is using an unstable development version of CPM.cmake. \
|
||||
Please update to a recent release if possible. \
|
||||
See https://github.com/cpm-cmake/CPM.cmake for details."
|
||||
)
|
||||
endif()
|
||||
|
||||
set_property(GLOBAL PROPERTY CPM_INITIALIZED true)
|
||||
|
||||
option(CPM_USE_LOCAL_PACKAGES "Always try to use `find_package` to get dependencies"
|
||||
@@ -76,6 +83,10 @@ option(CPM_INCLUDE_ALL_IN_PACKAGE_LOCK
|
||||
"Add all packages added through CPM.cmake to the package lock"
|
||||
$ENV{CPM_INCLUDE_ALL_IN_PACKAGE_LOCK}
|
||||
)
|
||||
option(CPM_USE_NAMED_CACHE_DIRECTORIES
|
||||
"Use additional directory of package name in cache on the most nested level."
|
||||
$ENV{CPM_USE_NAMED_CACHE_DIRECTORIES}
|
||||
)
|
||||
|
||||
set(CPM_VERSION
|
||||
${CURRENT_CPM_VERSION}
|
||||
@@ -223,7 +234,7 @@ function(cpm_create_module_file Name)
|
||||
if(NOT CPM_DONT_UPDATE_MODULE_PATH)
|
||||
# erase any previous modules
|
||||
file(WRITE ${CPM_MODULE_PATH}/Find${Name}.cmake
|
||||
"include(${CPM_FILE})\n${ARGN}\nset(${Name}_FOUND TRUE)"
|
||||
"include(\"${CPM_FILE}\")\n${ARGN}\nset(${Name}_FOUND TRUE)"
|
||||
)
|
||||
endif()
|
||||
endfunction()
|
||||
@@ -240,7 +251,13 @@ function(CPMFindPackage)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
if(CPM_DOWNLOAD_ALL)
|
||||
set(downloadPackage ${CPM_DOWNLOAD_ALL})
|
||||
if(DEFINED CPM_DOWNLOAD_${CPM_ARGS_NAME})
|
||||
set(downloadPackage ${CPM_DOWNLOAD_${CPM_ARGS_NAME}})
|
||||
elseif(DEFINED ENV{CPM_DOWNLOAD_${CPM_ARGS_NAME}})
|
||||
set(downloadPackage $ENV{CPM_DOWNLOAD_${CPM_ARGS_NAME}})
|
||||
endif()
|
||||
if(downloadPackage)
|
||||
CPMAddPackage(${ARGN})
|
||||
cpm_export_variables(${CPM_ARGS_NAME})
|
||||
return()
|
||||
@@ -302,6 +319,9 @@ function(cpm_parse_add_package_single_arg arg outArgs)
|
||||
elseif(scheme STREQUAL "gl")
|
||||
set(out "GITLAB_REPOSITORY;${uri}")
|
||||
set(packageType "git")
|
||||
elseif(scheme STREQUAL "bb")
|
||||
set(out "BITBUCKET_REPOSITORY;${uri}")
|
||||
set(packageType "git")
|
||||
# A CPM-specific scheme was not found. Looks like this is a generic URL so try to determine
|
||||
# type
|
||||
elseif(arg MATCHES ".git/?(@|#|$)")
|
||||
@@ -326,7 +346,7 @@ function(cpm_parse_add_package_single_arg arg outArgs)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
# For all packages we interpret @... as version. Only replace the last occurence. Thus URIs
|
||||
# For all packages we interpret @... as version. Only replace the last occurrence. Thus URIs
|
||||
# containing '@' can be used
|
||||
string(REGEX REPLACE "@([^@]+)$" ";VERSION;\\1" out "${out}")
|
||||
|
||||
@@ -351,6 +371,109 @@ function(cpm_parse_add_package_single_arg arg outArgs)
|
||||
)
|
||||
endfunction()
|
||||
|
||||
# Check that the working directory for a git repo is clean
|
||||
function(cpm_check_git_working_dir_is_clean repoPath gitTag isClean)
|
||||
|
||||
find_package(Git REQUIRED)
|
||||
|
||||
if(NOT GIT_EXECUTABLE)
|
||||
# No git executable, assume directory is clean
|
||||
set(${isClean}
|
||||
TRUE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
return()
|
||||
endif()
|
||||
|
||||
# check for uncommitted changes
|
||||
execute_process(
|
||||
COMMAND ${GIT_EXECUTABLE} status --porcelain
|
||||
RESULT_VARIABLE resultGitStatus
|
||||
OUTPUT_VARIABLE repoStatus
|
||||
OUTPUT_STRIP_TRAILING_WHITESPACE ERROR_QUIET
|
||||
WORKING_DIRECTORY ${repoPath}
|
||||
)
|
||||
if(resultGitStatus)
|
||||
# not supposed to happen, assume clean anyway
|
||||
message(WARNING "Calling git status on folder ${repoPath} failed")
|
||||
set(${isClean}
|
||||
TRUE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
return()
|
||||
endif()
|
||||
|
||||
if(NOT "${repoStatus}" STREQUAL "")
|
||||
set(${isClean}
|
||||
FALSE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
return()
|
||||
endif()
|
||||
|
||||
# check for committed changes
|
||||
execute_process(
|
||||
COMMAND ${GIT_EXECUTABLE} diff -s --exit-code ${gitTag}
|
||||
RESULT_VARIABLE resultGitDiff
|
||||
OUTPUT_STRIP_TRAILING_WHITESPACE OUTPUT_QUIET
|
||||
WORKING_DIRECTORY ${repoPath}
|
||||
)
|
||||
|
||||
if(${resultGitDiff} EQUAL 0)
|
||||
set(${isClean}
|
||||
TRUE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
else()
|
||||
set(${isClean}
|
||||
FALSE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
endif()
|
||||
|
||||
endfunction()
|
||||
|
||||
# method to overwrite internal FetchContent properties, to allow using CPM.cmake to overload
|
||||
# FetchContent calls. As these are internal cmake properties, this method should be used carefully
|
||||
# and may need modification in future CMake versions. Source:
|
||||
# https://github.com/Kitware/CMake/blob/dc3d0b5a0a7d26d43d6cfeb511e224533b5d188f/Modules/FetchContent.cmake#L1152
|
||||
function(cpm_override_fetchcontent contentName)
|
||||
cmake_parse_arguments(PARSE_ARGV 1 arg "" "SOURCE_DIR;BINARY_DIR" "")
|
||||
if(NOT "${arg_UNPARSED_ARGUMENTS}" STREQUAL "")
|
||||
message(FATAL_ERROR "Unsupported arguments: ${arg_UNPARSED_ARGUMENTS}")
|
||||
endif()
|
||||
|
||||
string(TOLOWER ${contentName} contentNameLower)
|
||||
set(prefix "_FetchContent_${contentNameLower}")
|
||||
|
||||
set(propertyName "${prefix}_sourceDir")
|
||||
define_property(
|
||||
GLOBAL
|
||||
PROPERTY ${propertyName}
|
||||
BRIEF_DOCS "Internal implementation detail of FetchContent_Populate()"
|
||||
FULL_DOCS "Details used by FetchContent_Populate() for ${contentName}"
|
||||
)
|
||||
set_property(GLOBAL PROPERTY ${propertyName} "${arg_SOURCE_DIR}")
|
||||
|
||||
set(propertyName "${prefix}_binaryDir")
|
||||
define_property(
|
||||
GLOBAL
|
||||
PROPERTY ${propertyName}
|
||||
BRIEF_DOCS "Internal implementation detail of FetchContent_Populate()"
|
||||
FULL_DOCS "Details used by FetchContent_Populate() for ${contentName}"
|
||||
)
|
||||
set_property(GLOBAL PROPERTY ${propertyName} "${arg_BINARY_DIR}")
|
||||
|
||||
set(propertyName "${prefix}_populated")
|
||||
define_property(
|
||||
GLOBAL
|
||||
PROPERTY ${propertyName}
|
||||
BRIEF_DOCS "Internal implementation detail of FetchContent_Populate()"
|
||||
FULL_DOCS "Details used by FetchContent_Populate() for ${contentName}"
|
||||
)
|
||||
set_property(GLOBAL PROPERTY ${propertyName} TRUE)
|
||||
endfunction()
|
||||
|
||||
# Download and add a package from source
|
||||
function(CPMAddPackage)
|
||||
list(LENGTH ARGN argnLength)
|
||||
@@ -369,6 +492,7 @@ function(CPMAddPackage)
|
||||
DOWNLOAD_ONLY
|
||||
GITHUB_REPOSITORY
|
||||
GITLAB_REPOSITORY
|
||||
BITBUCKET_REPOSITORY
|
||||
GIT_REPOSITORY
|
||||
SOURCE_DIR
|
||||
DOWNLOAD_COMMAND
|
||||
@@ -399,10 +523,10 @@ function(CPMAddPackage)
|
||||
|
||||
if(DEFINED CPM_ARGS_GITHUB_REPOSITORY)
|
||||
set(CPM_ARGS_GIT_REPOSITORY "https://github.com/${CPM_ARGS_GITHUB_REPOSITORY}.git")
|
||||
endif()
|
||||
|
||||
if(DEFINED CPM_ARGS_GITLAB_REPOSITORY)
|
||||
elseif(DEFINED CPM_ARGS_GITLAB_REPOSITORY)
|
||||
set(CPM_ARGS_GIT_REPOSITORY "https://gitlab.com/${CPM_ARGS_GITLAB_REPOSITORY}.git")
|
||||
elseif(DEFINED CPM_ARGS_BITBUCKET_REPOSITORY)
|
||||
set(CPM_ARGS_GIT_REPOSITORY "https://bitbucket.org/${CPM_ARGS_BITBUCKET_REPOSITORY}.git")
|
||||
endif()
|
||||
|
||||
if(DEFINED CPM_ARGS_GIT_REPOSITORY)
|
||||
@@ -466,10 +590,13 @@ function(CPMAddPackage)
|
||||
set(PACKAGE_SOURCE ${CPM_${CPM_ARGS_NAME}_SOURCE})
|
||||
set(CPM_${CPM_ARGS_NAME}_SOURCE "")
|
||||
CPMAddPackage(
|
||||
NAME ${CPM_ARGS_NAME}
|
||||
SOURCE_DIR ${PACKAGE_SOURCE}
|
||||
NAME "${CPM_ARGS_NAME}"
|
||||
SOURCE_DIR "${PACKAGE_SOURCE}"
|
||||
EXCLUDE_FROM_ALL "${CPM_ARGS_EXCLUDE_FROM_ALL}"
|
||||
OPTIONS "${CPM_ARGS_OPTIONS}"
|
||||
SOURCE_SUBDIR "${CPM_ARGS_SOURCE_SUBDIR}"
|
||||
DOWNLOAD_ONLY "${DOWNLOAD_ONLY}"
|
||||
FORCE True
|
||||
OPTIONS ${CPM_ARGS_OPTIONS}
|
||||
)
|
||||
cpm_export_variables(${CPM_ARGS_NAME})
|
||||
return()
|
||||
@@ -512,6 +639,13 @@ function(CPMAddPackage)
|
||||
set(PACKAGE_INFO "${CPM_ARGS_VERSION}")
|
||||
endif()
|
||||
|
||||
if(DEFINED FETCHCONTENT_BASE_DIR)
|
||||
# respect user's FETCHCONTENT_BASE_DIR if set
|
||||
set(CPM_FETCHCONTENT_BASE_DIR ${FETCHCONTENT_BASE_DIR})
|
||||
else()
|
||||
set(CPM_FETCHCONTENT_BASE_DIR ${CMAKE_BINARY_DIR}/_deps)
|
||||
endif()
|
||||
|
||||
if(DEFINED CPM_ARGS_DOWNLOAD_COMMAND)
|
||||
list(APPEND CPM_ARGS_UNPARSED_ARGUMENTS DOWNLOAD_COMMAND ${CPM_ARGS_DOWNLOAD_COMMAND})
|
||||
elseif(DEFINED CPM_ARGS_SOURCE_DIR)
|
||||
@@ -520,21 +654,46 @@ function(CPMAddPackage)
|
||||
string(TOLOWER ${CPM_ARGS_NAME} lower_case_name)
|
||||
set(origin_parameters ${CPM_ARGS_UNPARSED_ARGUMENTS})
|
||||
list(SORT origin_parameters)
|
||||
string(SHA1 origin_hash "${origin_parameters}")
|
||||
set(download_directory ${CPM_SOURCE_CACHE}/${lower_case_name}/${origin_hash})
|
||||
if(CPM_USE_NAMED_CACHE_DIRECTORIES)
|
||||
string(SHA1 origin_hash "${origin_parameters};NEW_CACHE_STRUCTURE_TAG")
|
||||
set(download_directory ${CPM_SOURCE_CACHE}/${lower_case_name}/${origin_hash}/${CPM_ARGS_NAME})
|
||||
else()
|
||||
string(SHA1 origin_hash "${origin_parameters}")
|
||||
set(download_directory ${CPM_SOURCE_CACHE}/${lower_case_name}/${origin_hash})
|
||||
endif()
|
||||
# Expand `download_directory` relative path. This is important because EXISTS doesn't work for
|
||||
# relative paths.
|
||||
get_filename_component(download_directory ${download_directory} ABSOLUTE)
|
||||
list(APPEND CPM_ARGS_UNPARSED_ARGUMENTS SOURCE_DIR ${download_directory})
|
||||
if(EXISTS ${download_directory})
|
||||
# avoid FetchContent modules to improve performance
|
||||
set(${CPM_ARGS_NAME}_BINARY_DIR ${CMAKE_BINARY_DIR}/_deps/${lower_case_name}-build)
|
||||
set(${CPM_ARGS_NAME}_ADDED YES)
|
||||
set(${CPM_ARGS_NAME}_SOURCE_DIR ${download_directory})
|
||||
cpm_store_fetch_properties(
|
||||
${CPM_ARGS_NAME} "${download_directory}"
|
||||
"${CPM_FETCHCONTENT_BASE_DIR}/${lower_case_name}-build"
|
||||
)
|
||||
cpm_get_fetch_properties("${CPM_ARGS_NAME}")
|
||||
|
||||
if(DEFINED CPM_ARGS_GIT_TAG)
|
||||
# warn if cache has been changed since checkout
|
||||
cpm_check_git_working_dir_is_clean(${download_directory} ${CPM_ARGS_GIT_TAG} IS_CLEAN)
|
||||
if(NOT ${IS_CLEAN})
|
||||
message(WARNING "Cache for ${CPM_ARGS_NAME} (${download_directory}) is dirty")
|
||||
endif()
|
||||
endif()
|
||||
|
||||
cpm_add_subdirectory(
|
||||
"${CPM_ARGS_NAME}" "${DOWNLOAD_ONLY}"
|
||||
"${${CPM_ARGS_NAME}_SOURCE_DIR}/${CPM_ARGS_SOURCE_SUBDIR}" "${${CPM_ARGS_NAME}_BINARY_DIR}"
|
||||
"${CPM_ARGS_EXCLUDE_FROM_ALL}" "${CPM_ARGS_OPTIONS}"
|
||||
)
|
||||
set(CPM_SKIP_FETCH TRUE)
|
||||
set(PACKAGE_INFO "${PACKAGE_INFO} at ${download_directory}")
|
||||
|
||||
# As the source dir is already cached/populated, we override the call to FetchContent.
|
||||
set(CPM_SKIP_FETCH TRUE)
|
||||
cpm_override_fetchcontent(
|
||||
"${lower_case_name}" SOURCE_DIR "${${CPM_ARGS_NAME}_SOURCE_DIR}/${CPM_ARGS_SOURCE_SUBDIR}"
|
||||
BINARY_DIR "${${CPM_ARGS_NAME}_BINARY_DIR}"
|
||||
)
|
||||
|
||||
else()
|
||||
# Enable shallow clone when GIT_TAG is not a commit hash. Our guess may not be accurate, but
|
||||
# it should guarantee no commit hash get mis-detected.
|
||||
@@ -546,12 +705,12 @@ function(CPMAddPackage)
|
||||
endif()
|
||||
|
||||
# remove timestamps so CMake will re-download the dependency
|
||||
file(REMOVE_RECURSE ${CMAKE_BINARY_DIR}/_deps/${lower_case_name}-subbuild)
|
||||
file(REMOVE_RECURSE ${CPM_FETCHCONTENT_BASE_DIR}/${lower_case_name}-subbuild)
|
||||
set(PACKAGE_INFO "${PACKAGE_INFO} to ${download_directory}")
|
||||
endif()
|
||||
endif()
|
||||
|
||||
cpm_create_module_file(${CPM_ARGS_NAME} "CPMAddPackage(${ARGN})")
|
||||
cpm_create_module_file(${CPM_ARGS_NAME} "CPMAddPackage(\"${ARGN}\")")
|
||||
|
||||
if(CPM_PACKAGE_LOCK_ENABLED)
|
||||
if((CPM_ARGS_VERSION AND NOT CPM_ARGS_SOURCE_DIR) OR CPM_INCLUDE_ALL_IN_PACKAGE_LOCK)
|
||||
@@ -571,12 +730,14 @@ function(CPMAddPackage)
|
||||
cpm_declare_fetch(
|
||||
"${CPM_ARGS_NAME}" "${CPM_ARGS_VERSION}" "${PACKAGE_INFO}" "${CPM_ARGS_UNPARSED_ARGUMENTS}"
|
||||
)
|
||||
cpm_fetch_package("${CPM_ARGS_NAME}")
|
||||
cpm_add_subdirectory(
|
||||
"${CPM_ARGS_NAME}" "${DOWNLOAD_ONLY}"
|
||||
"${${CPM_ARGS_NAME}_SOURCE_DIR}/${CPM_ARGS_SOURCE_SUBDIR}" "${${CPM_ARGS_NAME}_BINARY_DIR}"
|
||||
"${CPM_ARGS_EXCLUDE_FROM_ALL}" "${CPM_ARGS_OPTIONS}"
|
||||
)
|
||||
cpm_fetch_package("${CPM_ARGS_NAME}" populated)
|
||||
if(${populated})
|
||||
cpm_add_subdirectory(
|
||||
"${CPM_ARGS_NAME}" "${DOWNLOAD_ONLY}"
|
||||
"${${CPM_ARGS_NAME}_SOURCE_DIR}/${CPM_ARGS_SOURCE_SUBDIR}" "${${CPM_ARGS_NAME}_BINARY_DIR}"
|
||||
"${CPM_ARGS_EXCLUDE_FROM_ALL}" "${CPM_ARGS_OPTIONS}"
|
||||
)
|
||||
endif()
|
||||
cpm_get_fetch_properties("${CPM_ARGS_NAME}")
|
||||
endif()
|
||||
|
||||
@@ -610,7 +771,7 @@ macro(cpm_export_variables name)
|
||||
endmacro()
|
||||
|
||||
# declares a package, so that any call to CPMAddPackage for the package name will use these
|
||||
# arguments instead. Previous declarations will not be overriden.
|
||||
# arguments instead. Previous declarations will not be overridden.
|
||||
macro(CPMDeclarePackage Name)
|
||||
if(NOT DEFINED "CPM_DECLARATION_${Name}")
|
||||
set("CPM_DECLARATION_${Name}" "${ARGN}")
|
||||
@@ -633,7 +794,7 @@ function(cpm_add_comment_to_package_lock Name)
|
||||
endif()
|
||||
endfunction()
|
||||
|
||||
# includes the package lock file if it exists and creates a target `cpm-write-package-lock` to
|
||||
# includes the package lock file if it exists and creates a target `cpm-update-package-lock` to
|
||||
# update it
|
||||
macro(CPMUsePackageLock file)
|
||||
if(NOT CPM_DONT_CREATE_PACKAGE_LOCK)
|
||||
@@ -687,18 +848,32 @@ function(cpm_get_fetch_properties PACKAGE)
|
||||
if(${CPM_DRY_RUN})
|
||||
return()
|
||||
endif()
|
||||
FetchContent_GetProperties(${PACKAGE})
|
||||
string(TOLOWER ${PACKAGE} lpackage)
|
||||
|
||||
set(${PACKAGE}_SOURCE_DIR
|
||||
"${${lpackage}_SOURCE_DIR}"
|
||||
"${CPM_PACKAGE_${PACKAGE}_SOURCE_DIR}"
|
||||
PARENT_SCOPE
|
||||
)
|
||||
set(${PACKAGE}_BINARY_DIR
|
||||
"${${lpackage}_BINARY_DIR}"
|
||||
"${CPM_PACKAGE_${PACKAGE}_BINARY_DIR}"
|
||||
PARENT_SCOPE
|
||||
)
|
||||
endfunction()
|
||||
|
||||
function(cpm_store_fetch_properties PACKAGE source_dir binary_dir)
|
||||
if(${CPM_DRY_RUN})
|
||||
return()
|
||||
endif()
|
||||
|
||||
set(CPM_PACKAGE_${PACKAGE}_SOURCE_DIR
|
||||
"${source_dir}"
|
||||
CACHE INTERNAL ""
|
||||
)
|
||||
set(CPM_PACKAGE_${PACKAGE}_BINARY_DIR
|
||||
"${binary_dir}"
|
||||
CACHE INTERNAL ""
|
||||
)
|
||||
endfunction()
|
||||
|
||||
# adds a package as a subdirectory if viable, according to provided options
|
||||
function(
|
||||
cpm_add_subdirectory
|
||||
@@ -720,9 +895,15 @@ function(
|
||||
cmake_policy(SET CMP0077 NEW)
|
||||
set(CMAKE_POLICY_DEFAULT_CMP0077 NEW)
|
||||
|
||||
# the policy allows us to change set(CACHE) without caching
|
||||
if(POLICY CMP0126)
|
||||
cmake_policy(SET CMP0126 NEW)
|
||||
set(CMAKE_POLICY_DEFAULT_CMP0126 NEW)
|
||||
endif()
|
||||
|
||||
foreach(OPTION ${OPTIONS})
|
||||
cpm_parse_option(${OPTION})
|
||||
set(${OPTION_KEY} ${OPTION_VALUE})
|
||||
cpm_parse_option("${OPTION}")
|
||||
set(${OPTION_KEY} "${OPTION_VALUE}")
|
||||
endforeach()
|
||||
endif()
|
||||
set(CPM_OLD_INDENT "${CPM_INDENT}")
|
||||
@@ -734,7 +915,11 @@ endfunction()
|
||||
|
||||
# downloads a previously declared package via FetchContent and exports the variables
|
||||
# `${PACKAGE}_SOURCE_DIR` and `${PACKAGE}_BINARY_DIR` to the parent scope
|
||||
function(cpm_fetch_package PACKAGE)
|
||||
function(cpm_fetch_package PACKAGE populated)
|
||||
set(${populated}
|
||||
FALSE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
if(${CPM_DRY_RUN})
|
||||
message(STATUS "${CPM_INDENT} package ${PACKAGE} not fetched (dry run)")
|
||||
return()
|
||||
@@ -742,11 +927,20 @@ function(cpm_fetch_package PACKAGE)
|
||||
|
||||
FetchContent_GetProperties(${PACKAGE})
|
||||
|
||||
string(TOLOWER "${PACKAGE}" lower_case_name)
|
||||
|
||||
if(NOT ${lower_case_name}_POPULATED)
|
||||
FetchContent_Populate(${PACKAGE})
|
||||
set(${populated}
|
||||
TRUE
|
||||
PARENT_SCOPE
|
||||
)
|
||||
endif()
|
||||
|
||||
string(TOLOWER "${PACKAGE}" lower_case_name)
|
||||
cpm_store_fetch_properties(
|
||||
${CPM_ARGS_NAME} ${${lower_case_name}_SOURCE_DIR} ${${lower_case_name}_BINARY_DIR}
|
||||
)
|
||||
|
||||
set(${PACKAGE}_SOURCE_DIR
|
||||
${${lower_case_name}_SOURCE_DIR}
|
||||
PARENT_SCOPE
|
||||
@@ -759,15 +953,15 @@ endfunction()
|
||||
|
||||
# splits a package option
|
||||
function(cpm_parse_option OPTION)
|
||||
string(REGEX MATCH "^[^ ]+" OPTION_KEY ${OPTION})
|
||||
string(LENGTH ${OPTION} OPTION_LENGTH)
|
||||
string(LENGTH ${OPTION_KEY} OPTION_KEY_LENGTH)
|
||||
string(REGEX MATCH "^[^ ]+" OPTION_KEY "${OPTION}")
|
||||
string(LENGTH "${OPTION}" OPTION_LENGTH)
|
||||
string(LENGTH "${OPTION_KEY}" OPTION_KEY_LENGTH)
|
||||
if(OPTION_KEY_LENGTH STREQUAL OPTION_LENGTH)
|
||||
# no value for key provided, assume user wants to set option to "ON"
|
||||
set(OPTION_VALUE "ON")
|
||||
else()
|
||||
math(EXPR OPTION_KEY_LENGTH "${OPTION_KEY_LENGTH}+1")
|
||||
string(SUBSTRING ${OPTION} "${OPTION_KEY_LENGTH}" "-1" OPTION_VALUE)
|
||||
string(SUBSTRING "${OPTION}" "${OPTION_KEY_LENGTH}" "-1" OPTION_VALUE)
|
||||
endif()
|
||||
set(OPTION_KEY
|
||||
"${OPTION_KEY}"
|
||||
@@ -797,7 +991,7 @@ function(cpm_get_version_from_git_tag GIT_TAG RESULT)
|
||||
endif()
|
||||
endfunction()
|
||||
|
||||
# guesses if the git tag is a commit hash or an actual tag or a branch nane.
|
||||
# guesses if the git tag is a commit hash or an actual tag or a branch name.
|
||||
function(cpm_is_git_tag_commit_hash GIT_TAG RESULT)
|
||||
string(LENGTH "${GIT_TAG}" length)
|
||||
# full hash has 40 characters, and short hash has at least 7 characters.
|
||||
|
||||
7
examples/TestingFramework/CMakeLists.txt
Normal file
7
examples/TestingFramework/CMakeLists.txt
Normal file
@@ -0,0 +1,7 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(CPMTestingFrameworkExample)
|
||||
|
||||
include(../../cmake/CPM.cmake)
|
||||
CPMAddPackage("bb:HEMRND/TestingFramework@0.4.1")
|
||||
add_test_suites(TESTS SomeTest)
|
||||
19
examples/TestingFramework/SomeTest.cpp
Normal file
19
examples/TestingFramework/SomeTest.cpp
Normal file
@@ -0,0 +1,19 @@
|
||||
#include <HEM/TestingFramework.hpp>
|
||||
|
||||
using namespace fakeit;
|
||||
|
||||
class ITest {
|
||||
public:
|
||||
virtual int getInt() = 0;
|
||||
};
|
||||
|
||||
TEST_CASE("Testing Framework Test") {
|
||||
constexpr int someIntValue = 123456;
|
||||
|
||||
Mock<ITest> testMock;
|
||||
When(Method(testMock, getInt)).Return(someIntValue);
|
||||
|
||||
int readValue = testMock.get().getInt();
|
||||
|
||||
REQUIRE(readValue == someIntValue);
|
||||
}
|
||||
@@ -11,14 +11,11 @@ target_compile_features(CPMExampleBoost PRIVATE cxx_std_17)
|
||||
|
||||
include(../../cmake/CPM.cmake)
|
||||
|
||||
CPMFindPackage(
|
||||
CPMAddPackage(
|
||||
NAME Boost
|
||||
GITHUB_REPOSITORY Orphis/boost-cmake
|
||||
VERSION 1.67.0
|
||||
# setting FIND_PACKAGE_ARGUMENTS allow usage with `CPM_USE_LOCAL_PACKAGES`
|
||||
FIND_PACKAGE_ARGUMENTS "COMPONENTS system"
|
||||
VERSION 1.77.0
|
||||
GITHUB_REPOSITORY "boostorg/boost"
|
||||
GIT_TAG "boost-1.77.0"
|
||||
)
|
||||
|
||||
find_package(Threads REQUIRED)
|
||||
|
||||
target_link_libraries(CPMExampleBoost PRIVATE Boost::system Threads::Threads)
|
||||
target_link_libraries(CPMExampleBoost PRIVATE Boost::asio)
|
||||
|
||||
@@ -26,7 +26,7 @@ for example in examples:
|
||||
print("running example %s" % example.name)
|
||||
print("================" + ('=' * len(example.name)))
|
||||
project = Path(".") / 'build' / example.name
|
||||
configure = runCommand('cmake -H%s -B%s' % (example, project))
|
||||
configure = runCommand('cmake -S%s -B%s' % (example, project))
|
||||
print(' ' + '\n '.join([line for line in configure.split('\n') if 'CPM:' in line]))
|
||||
build = runCommand('cmake --build %s -- -j%i' % (project, os.cpu_count() / 2))
|
||||
print(' ' + '\n '.join([line for line in build.split('\n') if 'Built target' in line]))
|
||||
|
||||
@@ -3,8 +3,14 @@
|
||||
|
||||
int main(int argc, char** argv) {
|
||||
cxxopts::Options options("MyProgram", "One line description of MyProgram");
|
||||
options.add_options()("h,help", "Show help")("d,debug", "Enable debugging")(
|
||||
"f,file", "File name", cxxopts::value<std::string>());
|
||||
|
||||
// clang-format off
|
||||
options.add_options()
|
||||
("h,help", "Show help")
|
||||
("d,debug", "Enable debugging")
|
||||
("f,file", "File name", cxxopts::value<std::string>()
|
||||
);
|
||||
// clang-format on
|
||||
|
||||
auto result = options.parse(argc, argv);
|
||||
|
||||
|
||||
@@ -5,22 +5,10 @@ project(CPMJSONExample)
|
||||
# ---- Dependencies ----
|
||||
|
||||
include(../../cmake/CPM.cmake)
|
||||
|
||||
CPMAddPackage(
|
||||
NAME nlohmann_json
|
||||
VERSION 3.9.1
|
||||
# not using the repo as it takes forever to clone
|
||||
URL https://github.com/nlohmann/json/releases/download/v3.9.1/include.zip
|
||||
URL_HASH SHA256=6bea5877b1541d353bd77bdfbdb2696333ae5ed8f9e8cc22df657192218cad91
|
||||
)
|
||||
|
||||
if(nlohmann_json_ADDED)
|
||||
add_library(nlohmann_json INTERFACE)
|
||||
target_include_directories(nlohmann_json SYSTEM INTERFACE ${nlohmann_json_SOURCE_DIR}/include)
|
||||
endif()
|
||||
CPMAddPackage("gh:nlohmann/json@3.10.5")
|
||||
|
||||
# ---- Executable ----
|
||||
|
||||
add_executable(CPMJSONExample main.cpp)
|
||||
target_compile_features(CPMJSONExample PRIVATE cxx_std_17)
|
||||
target_link_libraries(CPMJSONExample nlohmann_json)
|
||||
target_link_libraries(CPMJSONExample nlohmann_json::nlohmann_json)
|
||||
|
||||
@@ -6,18 +6,7 @@ project(CPMRangev3Example)
|
||||
|
||||
include(../../cmake/CPM.cmake)
|
||||
|
||||
CPMAddPackage(
|
||||
NAME range-v3
|
||||
URL https://github.com/ericniebler/range-v3/archive/0.11.0.zip
|
||||
VERSION 0.11.0
|
||||
# the range-v3 CMakeLists screws with configuration options
|
||||
DOWNLOAD_ONLY True
|
||||
)
|
||||
|
||||
if(range-v3_ADDED)
|
||||
add_library(range-v3 INTERFACE IMPORTED)
|
||||
target_include_directories(range-v3 SYSTEM INTERFACE "${range-v3_SOURCE_DIR}/include")
|
||||
endif()
|
||||
CPMAddPackage("gh:ericniebler/range-v3#0.11.0")
|
||||
|
||||
# ---- Executable ----
|
||||
|
||||
|
||||
@@ -32,7 +32,7 @@ auto is_six = [](int i) { return i == 6; };
|
||||
int main() {
|
||||
std::vector<int> v{6, 2, 3, 4, 5, 6};
|
||||
cout << std::boolalpha;
|
||||
cout << "vector: " << ranges::view::all(v) << '\n';
|
||||
cout << "vector: " << ranges::views::all(v) << '\n';
|
||||
|
||||
cout << "vector any_of is_six: " << ranges::any_of(v, is_six) << '\n';
|
||||
cout << "vector all_of is_six: " << ranges::all_of(v, is_six) << '\n';
|
||||
|
||||
2
test/integration/.gitignore
vendored
Normal file
2
test/integration/.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
# Use this to have a local integration test which is for personal experiments
|
||||
test_local.rb
|
||||
44
test/integration/README.md
Normal file
44
test/integration/README.md
Normal file
@@ -0,0 +1,44 @@
|
||||
# CPM.cmake Integration Tests
|
||||
|
||||
The integration tests of CPM.cmake are written in Ruby. They use a custom integration test framework which extends the [Test::Unit](https://www.rubydoc.info/github/test-unit/test-unit/Test/Unit) library.
|
||||
|
||||
They require Ruby 2.7.0 or later.
|
||||
|
||||
## Running tests
|
||||
|
||||
To run all tests from the repo root execute:
|
||||
|
||||
```
|
||||
$ ruby test/integration/runner.rb
|
||||
```
|
||||
|
||||
The runner will run all tests and generate a report of the exeuction.
|
||||
|
||||
The current working directory doesn't matter. If you are in `<repo-root>/test/integration`, you can run simply `$ ruby runner.rb`.
|
||||
|
||||
You can execute with `--help` (`$ ruby runner.rb --help`) to see various configuration options of the runner like running individual tests or test cases, or ones that match a regex.
|
||||
|
||||
The tests themselves are situated in the Ruby scripts prefixed with `test_`. `<repo-root>/test/integration/test_*`. You can also run an individual test script. For example to only run the **basics** test case, you can execute `$ ruby test_basics.rb`
|
||||
|
||||
The tests generate CMake scripts and execute CMake and build toolchains. By default they do this in a directory they generate in your temp path (`/tmp/cpm-test/` on Linux). You can configure the working directory of the tests with an environment variable `CPM_INTEGRATION_TEST_DIR`. For example `$ CPM_INTEGRATION_TEST_DIR=~/mycpmtest; ruby runner.rb`
|
||||
|
||||
## Writing tests
|
||||
|
||||
Writing tests makes use of the custom integration test framework in `lib.rb`. It is a relatively small extension of Ruby's Test::Unit library.
|
||||
|
||||
### The Gist
|
||||
|
||||
* Tests cases are Ruby scripts in this directory. The file names must be prefixed with `test_`
|
||||
* The script should `require_relative './lib'` to allow for individual execution (or else if will only be executable from the runner)
|
||||
* A test case file should contain a single class which inherits from `IntegrationTest`. It *can* contain multiple classes, but that's bad practice as it makes individual execution harder and implies a dependency between the classes.
|
||||
* There should be no dependency between the test scripts. Each should be executable individually and the order in which multiple ones are executed mustn't matter.
|
||||
* The class should contain methods, also prefixed with `test_` which will be executed by the framework. In most cases there would be a single test method per class.
|
||||
* In case there are multiple test methods, they will be executed in the order in which they are defined.
|
||||
* The test methods should contain assertions which check for the expected state of things at varous points of the test's execution.
|
||||
|
||||
### More
|
||||
|
||||
* [A basic tutorial on writing integration tests.](tutorial.md)
|
||||
* [A brief reference of the integration test framework](reference.md)
|
||||
* Make sure you're familiar with the [idiosyncrasies](idiosyncrasies.md) of writing integration tests
|
||||
* [Some tips and tricks](tips.md)
|
||||
98
test/integration/idiosyncrasies.md
Normal file
98
test/integration/idiosyncrasies.md
Normal file
@@ -0,0 +1,98 @@
|
||||
# Notable Idiosyncrasies When Writing Integration Tests
|
||||
|
||||
As an integration test framework based on a unit test framework the one created for CPM.cmake suffers from several idiosyncrasies. Make sure you familiarize yourself with them before writing integration tests.
|
||||
|
||||
## No shared instance variables between methods
|
||||
|
||||
The runner will create an instance of the test class for each test method. This means that instance variables defined in a test method, *will not* be visible in another. For example:
|
||||
|
||||
```ruby
|
||||
class MyTest < IntegrationTest
|
||||
def test_something
|
||||
@x = 123
|
||||
assert_equal 123, @x # Pass. @x is 123
|
||||
end
|
||||
def test_something_else
|
||||
assert_equal 123, @x # Fail! @x would be nil here
|
||||
end
|
||||
end
|
||||
```
|
||||
|
||||
There are hacks around sharing Ruby state between methods, but we choose not to use them. If you want to initialize something for all test methods, use `setup`.
|
||||
|
||||
```ruby
|
||||
class MyTest < IntegrationTest
|
||||
def setup
|
||||
@x = 123
|
||||
end
|
||||
def test_something
|
||||
assert_equal 123, @x # Pass. @x is 123 thanks to setup
|
||||
end
|
||||
def test_something_else
|
||||
assert_equal 123, @x # Pass. @x is 123 thanks to setup
|
||||
end
|
||||
end
|
||||
```
|
||||
|
||||
## `IntegrationTest` makes use of `Test::Unit::TestCase#cleanup`
|
||||
|
||||
After each test method the `cleanup` method is called thanks to Test::Unit. If you require the use of `cleanup` in your own tests, make sure you call `super` to also run `IntegrationTest#cleanup`.
|
||||
|
||||
```ruby
|
||||
class MyTest < IntegrationTest
|
||||
def cleanup
|
||||
super
|
||||
my_cleanup
|
||||
end
|
||||
# ...
|
||||
end
|
||||
```
|
||||
|
||||
## It's better to have assertions in test methods as opposed to helper methods
|
||||
|
||||
Test::Unit will display a helpful message if an assertion has failed. It will also include the line of code in the test method which caused the failure. However if an assertion is not in the test method, it will display the line which calls the method in which it is. So, please try, to have most assertions in test methods (though we acknowledge that in certain cases this is not practical). For example, if you only require scopes, try using lambdas.
|
||||
|
||||
Instead of this:
|
||||
|
||||
```ruby
|
||||
class MyTest < IntegrationTest
|
||||
def test_something
|
||||
do_a
|
||||
do_b
|
||||
do_c
|
||||
end
|
||||
def do_a
|
||||
# ...
|
||||
end
|
||||
def do_b
|
||||
# ...
|
||||
assert false # will display failed line as "do_b"
|
||||
end
|
||||
def do_c
|
||||
# ...
|
||||
end
|
||||
end
|
||||
```
|
||||
|
||||
...write this:
|
||||
|
||||
```ruby
|
||||
class MyTest < IntegrationTest
|
||||
def test_something
|
||||
do_a = -> {
|
||||
# ...
|
||||
}
|
||||
do_b = -> {
|
||||
# ...
|
||||
assert false # will display failed line as "assert false"
|
||||
}
|
||||
do_c = -> {
|
||||
# ...
|
||||
}
|
||||
|
||||
do_a.()
|
||||
do_b.()
|
||||
do_c.()
|
||||
end
|
||||
end
|
||||
```
|
||||
200
test/integration/lib.rb
Normal file
200
test/integration/lib.rb
Normal file
@@ -0,0 +1,200 @@
|
||||
require 'fileutils'
|
||||
require 'open3'
|
||||
require 'tmpdir'
|
||||
require 'test/unit'
|
||||
|
||||
module TestLib
|
||||
TMP_DIR = File.expand_path(ENV['CPM_INTEGRATION_TEST_DIR'] || File.join(Dir.tmpdir, 'cpm-test', Time.now.strftime('%Y_%m_%d-%H_%M_%S')))
|
||||
CPM_PATH = File.expand_path('../../cmake/CPM.cmake', __dir__)
|
||||
|
||||
TEMPLATES_DIR = File.expand_path('templates', __dir__)
|
||||
|
||||
# Environment variables which are read by cpm
|
||||
CPM_ENV = %w(
|
||||
CPM_USE_LOCAL_PACKAGES
|
||||
CPM_LOCAL_PACKAGES_ONLY
|
||||
CPM_DOWNLOAD_ALL
|
||||
CPM_DONT_UPDATE_MODULE_PATH
|
||||
CPM_DONT_CREATE_PACKAGE_LOCK
|
||||
CPM_INCLUDE_ALL_IN_PACKAGE_LOCK
|
||||
CPM_USE_NAMED_CACHE_DIRECTORIES
|
||||
CPM_SOURCE_CACHE
|
||||
)
|
||||
def self.clear_env
|
||||
CPM_ENV.each { ENV[_1] = nil }
|
||||
end
|
||||
end
|
||||
|
||||
puts "Warning: test directory '#{TestLib::TMP_DIR}' already exists" if File.exist?(TestLib::TMP_DIR)
|
||||
raise "Cannot find 'CPM.cmake' at '#{TestLib::CPM_PATH}'" if !File.file?(TestLib::CPM_PATH)
|
||||
|
||||
puts "Running CPM.cmake integration tests"
|
||||
puts "Temp directory: '#{TestLib::TMP_DIR}'"
|
||||
|
||||
# Clean all CPM-related env vars
|
||||
TestLib.clear_env
|
||||
|
||||
class Project
|
||||
def initialize(src_dir, bin_dir)
|
||||
@src_dir = src_dir
|
||||
@bin_dir = bin_dir
|
||||
end
|
||||
|
||||
attr :src_dir, :bin_dir
|
||||
|
||||
def create_file(target_path, text, args = {})
|
||||
target_path = File.join(@src_dir, target_path)
|
||||
|
||||
# tweak args
|
||||
args[:cpm_path] = TestLib::CPM_PATH if !args[:cpm_path]
|
||||
args[:packages] = [args[:package]] if args[:package] # if args contain package, create the array
|
||||
args[:packages] = args[:packages].join("\n") if args[:packages] # join all packages if any
|
||||
|
||||
File.write target_path, text % args
|
||||
end
|
||||
|
||||
def create_file_from_template(target_path, source_path, args = {})
|
||||
source_path = File.join(@src_dir, source_path)
|
||||
raise "#{source_path} doesn't exist" if !File.file?(source_path)
|
||||
src_text = File.read source_path
|
||||
create_file target_path, src_text, args
|
||||
end
|
||||
|
||||
# common function to create ./CMakeLists.txt from ./lists.in.cmake
|
||||
def create_lists_from_default_template(args = {})
|
||||
create_file_from_template 'CMakeLists.txt', 'lists.in.cmake', args
|
||||
end
|
||||
|
||||
CommandResult = Struct.new :out, :err, :status
|
||||
def configure(extra_args = '')
|
||||
CommandResult.new *Open3.capture3("cmake -S #{@src_dir} -B #{@bin_dir} #{extra_args}")
|
||||
end
|
||||
def build(extra_args = '')
|
||||
CommandResult.new *Open3.capture3("cmake --build #{@bin_dir} #{extra_args}")
|
||||
end
|
||||
|
||||
class CMakeCache
|
||||
class Entry
|
||||
def initialize(val, type, advanced, desc)
|
||||
@val = val
|
||||
@type = type
|
||||
@advanced = advanced
|
||||
@desc = desc
|
||||
end
|
||||
attr :val, :type, :advanced, :desc
|
||||
alias_method :advanced?, :advanced
|
||||
def inspect
|
||||
"(#{val.inspect} #{type}" + (advanced? ? ' ADVANCED)' : ')')
|
||||
end
|
||||
end
|
||||
|
||||
Package = Struct.new(:ver, :src_dir, :bin_dir)
|
||||
|
||||
def self.from_dir(dir)
|
||||
entries = {}
|
||||
cur_desc = ''
|
||||
file = File.join(dir, 'CMakeCache.txt')
|
||||
return nil if !File.file?(file)
|
||||
File.readlines(file).each { |line|
|
||||
line.strip!
|
||||
next if line.empty?
|
||||
next if line.start_with? '#' # comment
|
||||
if line.start_with? '//'
|
||||
cur_desc += line[2..]
|
||||
else
|
||||
m = /(.+?)(-ADVANCED)?:([A-Z]+)=(.*)/.match(line)
|
||||
raise "Error parsing '#{line}' in #{file}" if !m
|
||||
entries[m[1]] = Entry.new(m[4], m[3], !!m[2], cur_desc)
|
||||
cur_desc = ''
|
||||
end
|
||||
}
|
||||
CMakeCache.new entries
|
||||
end
|
||||
|
||||
def initialize(entries)
|
||||
@entries = entries
|
||||
|
||||
package_list = self['CPM_PACKAGES']
|
||||
@packages = if package_list
|
||||
# collect package data
|
||||
@packages = package_list.split(';').map { |name|
|
||||
[name, Package.new(
|
||||
self["CPM_PACKAGE_#{name}_VERSION"],
|
||||
self["CPM_PACKAGE_#{name}_SOURCE_DIR"],
|
||||
self["CPM_PACKAGE_#{name}_BINARY_DIR"]
|
||||
)]
|
||||
}.to_h
|
||||
else
|
||||
{}
|
||||
end
|
||||
end
|
||||
|
||||
attr :entries, :packages
|
||||
|
||||
def [](key)
|
||||
e = @entries[key]
|
||||
return nil if !e
|
||||
e.val
|
||||
end
|
||||
end
|
||||
def read_cache
|
||||
CMakeCache.from_dir @bin_dir
|
||||
end
|
||||
end
|
||||
|
||||
class IntegrationTest < Test::Unit::TestCase
|
||||
self.test_order = :defined # run tests in order of defintion (as opposed to alphabetical)
|
||||
|
||||
def cleanup
|
||||
# Clear cpm-related env vars which may have been set by the test
|
||||
TestLib.clear_env
|
||||
end
|
||||
|
||||
# extra assertions
|
||||
|
||||
def assert_success(res)
|
||||
msg = build_message(nil, "command status was expected to be a success, but failed with code <?> and STDERR:\n\n#{res.err}", res.status.to_i)
|
||||
assert_block(msg) { res.status.success? }
|
||||
end
|
||||
|
||||
def assert_same_path(a, b)
|
||||
msg = build_message(nil, "<?> expected but was\n<?>", a, b)
|
||||
assert_block(msg) { File.identical? a, b }
|
||||
end
|
||||
|
||||
# utils
|
||||
class << self
|
||||
def startup
|
||||
@@test_dir = File.join(TestLib::TMP_DIR, self.name.
|
||||
# to-underscore conversion from Rails
|
||||
gsub(/::/, '/').
|
||||
gsub(/([A-Z]+)([A-Z][a-z])/,'\1_\2').
|
||||
gsub(/([a-z\d])([A-Z])/,'\1_\2').
|
||||
tr("-", "_").
|
||||
downcase
|
||||
)
|
||||
end
|
||||
end
|
||||
|
||||
def cur_test_dir
|
||||
@@test_dir
|
||||
end
|
||||
|
||||
def make_project(template_dir = nil)
|
||||
test_name = local_name
|
||||
test_name = test_name[5..] if test_name.start_with?('test_')
|
||||
|
||||
base = File.join(cur_test_dir, test_name)
|
||||
src_dir = base + '-src'
|
||||
|
||||
FileUtils.mkdir_p src_dir
|
||||
|
||||
if template_dir
|
||||
template_dir = File.join(TestLib::TEMPLATES_DIR, template_dir)
|
||||
raise "#{template_dir} is not a directory" if !File.directory?(template_dir)
|
||||
FileUtils.copy_entry template_dir, src_dir
|
||||
end
|
||||
|
||||
Project.new src_dir, base + '-bin'
|
||||
end
|
||||
end
|
||||
65
test/integration/reference.md
Normal file
65
test/integration/reference.md
Normal file
@@ -0,0 +1,65 @@
|
||||
# Integration Test Framework Refernce
|
||||
|
||||
## `TestLib`
|
||||
|
||||
A module for the framework. Provides global data and functionality. For ease of use the utility classes are *not* in this module.
|
||||
|
||||
Provides:
|
||||
|
||||
* `TMP_DIR` - the temporary directory for the current test run
|
||||
* `CPM_PATH` - path to CPM.cmake. The thing that is being tested
|
||||
* `TEMPLATES_DIR` - path to integration test templates
|
||||
* `CPM_ENV` - an array of the names of all environment variables, which CPM.cmake may read
|
||||
* `.clear_env` - a function to clear all aforementioned environment variables
|
||||
|
||||
## `Project`
|
||||
|
||||
A helper class to manage a CMake project.
|
||||
|
||||
Provides:
|
||||
|
||||
* `#initialize(src_dir, bin_dir)` - create a project with a given source and binary directory
|
||||
* `#src_dir`, `#bin_dir` - get project directories
|
||||
* `#create_file(target_path, text, args = {})` - create a file in the project's source directory with a given test. The `args` hash is used to interpolate markup in the text string.
|
||||
* Will set `:cpm_path` in `args` to `TestLib::CPM_PATH` if not already present.
|
||||
* If `:package` is present it will be added to the array `:packages`
|
||||
* Will convert `:packages` form an array to a string
|
||||
* `#create_file_from_template(target_path, source_path, args = {})` - create a file in the project source directory, based on another file in the project source directory. The contents of the file at `source_path` will be read and used in `create_file`
|
||||
* `#create_lists_from_default_template(args = {})` - same as `create_file_from_template('CMakeLists.txt', 'lists.in.cmake', args)`
|
||||
* `::CommandResult` - a struct of:
|
||||
* `out` - the standard output from a command execution
|
||||
* `err` - the standard error output from the execution
|
||||
* `status` - the [`Process::Status`](https://ruby-doc.org/core-2.7.0/Process/Status.html) of the execution
|
||||
* `#configure(extra_args = '') => CommandResult` - configure the project with optional extra args to CMake
|
||||
* `#build(extra_args = '') => CommandResult` - build the project with optional extra args to CMake
|
||||
* `::CMakeCache` - a helper class with the contents of a CMakeCache.txt. Provides:
|
||||
* `::Entry` - a CMake cache entry of:
|
||||
* `val` - the value as string
|
||||
* `type` - the type as string
|
||||
* `advanced?` - whether the entry is an advanced option
|
||||
* `desc` - the description of the entry (can be an empty string)
|
||||
* `::Package` - the CMake cache for a CPM.cmake package. A struct of:
|
||||
* `ver` - the version as string
|
||||
* `src_dir`, `bin_dir` - the source and binary directories of the package
|
||||
* `.from_dir(dir)` - create an instance of `CMakeCache` from `<dir>/CMakeLists.txt`
|
||||
* `#initialize(entries)` - create a cache from a hash of entries by name. Will populate packages.
|
||||
* `#entries => {String => Entry}` - the entries of the cache
|
||||
* `#packages => {String => Package}` - CPM.cmake packages by name found in the cache
|
||||
* `#[](key) => String` - an entry value from an entry name. Created because the value is expected to be needed much more frequently than the entire entry data. To get a full entry use `cache.entries['name']`.
|
||||
* `read_cache => CMakeCache` - reads the CMake cache in the binary directory of the project and returns it as a `CMakeCache` instance
|
||||
|
||||
## `IntegrationTest`
|
||||
|
||||
The class which must be a parent of all integration test case classes. It itself extends `Test::Unit::TestCase` with:
|
||||
|
||||
### Assertions
|
||||
|
||||
* `assert_success(res)` - assert that an instance of `Project::CommandResult` is a success
|
||||
* `assert_same_path(a, b)` - assert that two strings represent the same path. For example on Windows `c:\foo` and `C:\Foo` do.
|
||||
|
||||
### Utils
|
||||
|
||||
* `cur_test_dir` - the directory of the current test case. A subdirectory of `TestLib::TMP_DIR`
|
||||
* `make_project(template_dir = nil)` - create a project from a test method. Will create a the project's source and binary directories as subdirectories of `cur_test_dir`.
|
||||
* Optionally work with a template directory, in which case it will copy the contents of the template directory (one from `templates`) in the project's source directory.
|
||||
|
||||
4
test/integration/runner.rb
Normal file
4
test/integration/runner.rb
Normal file
@@ -0,0 +1,4 @@
|
||||
require_relative './lib'
|
||||
|
||||
exit Test::Unit::AutoRunner::run(true, __dir__)
|
||||
|
||||
7
test/integration/templates/no-deps/lists.in.cmake
Normal file
7
test/integration/templates/no-deps/lists.in.cmake
Normal file
@@ -0,0 +1,7 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(no-deps)
|
||||
|
||||
include("%{cpm_path}")
|
||||
|
||||
add_executable(no-deps main.c)
|
||||
6
test/integration/templates/no-deps/main.c
Normal file
6
test/integration/templates/no-deps/main.c
Normal file
@@ -0,0 +1,6 @@
|
||||
#include <stdio.h>
|
||||
|
||||
int main() {
|
||||
puts("Hello");
|
||||
return 0;
|
||||
}
|
||||
13
test/integration/templates/using-adder/lists.in.cmake
Normal file
13
test/integration/templates/using-adder/lists.in.cmake
Normal file
@@ -0,0 +1,13 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(using-adder)
|
||||
|
||||
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
|
||||
|
||||
include("%{cpm_path}")
|
||||
|
||||
%{packages}
|
||||
|
||||
add_executable(using-adder using-adder.cpp)
|
||||
|
||||
target_link_libraries(using-adder adder)
|
||||
8
test/integration/templates/using-adder/using-adder.cpp
Normal file
8
test/integration/templates/using-adder/using-adder.cpp
Normal file
@@ -0,0 +1,8 @@
|
||||
#include <adder/adder.hpp>
|
||||
#include <cstdio>
|
||||
|
||||
int main() {
|
||||
int sum = adder::add(5, 3);
|
||||
std::printf("%d\n", sum);
|
||||
return 0;
|
||||
}
|
||||
13
test/integration/templates/using-fibadder/lists.in.cmake
Normal file
13
test/integration/templates/using-fibadder/lists.in.cmake
Normal file
@@ -0,0 +1,13 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(using-fibadder)
|
||||
|
||||
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
|
||||
|
||||
include("%{cpm_path}")
|
||||
|
||||
%{packages}
|
||||
|
||||
add_executable(using-fibadder using-fibadder.cpp)
|
||||
|
||||
target_link_libraries(using-fibadder fibadder)
|
||||
@@ -0,0 +1,8 @@
|
||||
#include <cstdio>
|
||||
#include <fibadder/fibadder.hpp>
|
||||
|
||||
int main() {
|
||||
int sum = fibadder::fibadd(6, 7);
|
||||
std::printf("%d\n", sum);
|
||||
return 0;
|
||||
}
|
||||
56
test/integration/test_basics.rb
Normal file
56
test/integration/test_basics.rb
Normal file
@@ -0,0 +1,56 @@
|
||||
require_relative './lib'
|
||||
|
||||
# Tests of cpm caches and vars when no packages are used
|
||||
|
||||
class Basics < IntegrationTest
|
||||
# Test cpm caches with no cpm-related env vars
|
||||
def test_cpm_default
|
||||
prj = make_project 'no-deps'
|
||||
prj.create_lists_from_default_template
|
||||
assert_success prj.configure
|
||||
|
||||
@cache = prj.read_cache
|
||||
|
||||
assert_empty @cache.packages
|
||||
|
||||
assert_same_path TestLib::CPM_PATH, check_and_get('CPM_FILE')
|
||||
assert_same_path File.dirname(TestLib::CPM_PATH), check_and_get('CPM_DIRECTORY')
|
||||
|
||||
assert_equal 'OFF', check_and_get('CPM_DRY_RUN')
|
||||
assert_equal 'CPM:', check_and_get('CPM_INDENT')
|
||||
assert_equal '1.0.0-development-version', check_and_get('CPM_VERSION')
|
||||
|
||||
assert_equal 'OFF', check_and_get('CPM_SOURCE_CACHE', 'PATH')
|
||||
assert_equal 'OFF', check_and_get('CPM_DOWNLOAD_ALL', 'BOOL')
|
||||
assert_equal 'OFF', check_and_get('CPM_LOCAL_PACKAGES_ONLY', 'BOOL')
|
||||
assert_equal 'OFF', check_and_get('CPM_USE_LOCAL_PACKAGES', 'BOOL')
|
||||
assert_equal 'OFF', check_and_get('CPM_USE_NAMED_CACHE_DIRECTORIES', 'BOOL')
|
||||
|
||||
assert_equal 'OFF', check_and_get('CPM_DONT_CREATE_PACKAGE_LOCK', 'BOOL')
|
||||
assert_equal 'OFF', check_and_get('CPM_INCLUDE_ALL_IN_PACKAGE_LOCK', 'BOOL')
|
||||
assert_same_path File.join(prj.bin_dir, 'cpm-package-lock.cmake'), check_and_get('CPM_PACKAGE_LOCK_FILE')
|
||||
|
||||
assert_equal 'OFF', check_and_get('CPM_DONT_UPDATE_MODULE_PATH', 'BOOL')
|
||||
assert_same_path File.join(prj.bin_dir, 'CPM_modules'), check_and_get('CPM_MODULE_PATH')
|
||||
end
|
||||
|
||||
# Test when env CPM_SOURCE_CACHE is set
|
||||
def test_env_cpm_source_cache
|
||||
ENV['CPM_SOURCE_CACHE'] = cur_test_dir
|
||||
|
||||
prj = make_project 'no-deps'
|
||||
prj.create_lists_from_default_template
|
||||
assert_success prj.configure
|
||||
|
||||
@cache = prj.read_cache
|
||||
|
||||
assert_equal cur_test_dir, check_and_get('CPM_SOURCE_CACHE', 'PATH')
|
||||
end
|
||||
|
||||
def check_and_get(key, type = 'INTERNAL')
|
||||
e = @cache.entries[key]
|
||||
assert_not_nil e, key
|
||||
assert_equal type, e.type, key
|
||||
e.val
|
||||
end
|
||||
end
|
||||
43
test/integration/test_fetchcontent_compatibility.rb
Normal file
43
test/integration/test_fetchcontent_compatibility.rb
Normal file
@@ -0,0 +1,43 @@
|
||||
require_relative './lib'
|
||||
|
||||
# Tests FetchContent overriding with CPM
|
||||
|
||||
class FetchContentCompatibility < IntegrationTest
|
||||
def setup
|
||||
@cache_dir = File.join(cur_test_dir, 'cpmcache')
|
||||
ENV['CPM_SOURCE_CACHE'] = @cache_dir
|
||||
end
|
||||
|
||||
def test_add_dependency_cpm_and_fetchcontent
|
||||
prj = make_project 'using-adder'
|
||||
|
||||
prj.create_lists_from_default_template package: <<~PACK
|
||||
CPMAddPackage(
|
||||
NAME testpack-adder
|
||||
GITHUB_REPOSITORY cpm-cmake/testpack-adder
|
||||
VERSION 1.0.0
|
||||
OPTIONS "ADDER_BUILD_TESTS OFF"
|
||||
)
|
||||
|
||||
# should have no effect, as we added the dependency using CPM
|
||||
FetchContent_Declare(
|
||||
testpack-adder
|
||||
GIT_REPOSITORY https://github.com/cpm-cmake/testpack-adder
|
||||
GIT_TAG v1.0.0
|
||||
)
|
||||
FetchContent_MakeAvailable(testpack-adder)
|
||||
PACK
|
||||
|
||||
# configure with unpopulated cache
|
||||
assert_success prj.configure
|
||||
assert_success prj.build
|
||||
|
||||
# cache is populated
|
||||
assert_true File.exist?(File.join(@cache_dir, "testpack-adder"))
|
||||
|
||||
# configure with populated cache
|
||||
assert_success prj.configure
|
||||
assert_success prj.build
|
||||
end
|
||||
|
||||
end
|
||||
91
test/integration/test_simple.rb
Normal file
91
test/integration/test_simple.rb
Normal file
@@ -0,0 +1,91 @@
|
||||
require_relative './lib'
|
||||
|
||||
class Simple < IntegrationTest
|
||||
ADDER_PACKAGE_NAME = 'testpack-adder'
|
||||
|
||||
def test_update_single_package
|
||||
prj = make_project 'using-adder'
|
||||
adder_cache0 = nil
|
||||
adder_ver_file = nil
|
||||
|
||||
create_with_commit_sha = -> {
|
||||
prj.create_lists_from_default_template package:
|
||||
'CPMAddPackage("gh:cpm-cmake/testpack-adder#cad1cd4b4cdf957c5b59e30bc9a1dd200dbfc716")'
|
||||
assert_success prj.configure
|
||||
|
||||
cache = prj.read_cache
|
||||
assert_equal 1, cache.packages.size
|
||||
|
||||
adder_cache = cache.packages[ADDER_PACKAGE_NAME]
|
||||
assert_not_nil adder_cache
|
||||
assert_equal '0', adder_cache.ver
|
||||
assert File.directory? adder_cache.src_dir
|
||||
assert File.directory? adder_cache.bin_dir
|
||||
|
||||
adder_ver_file = File.join(adder_cache.src_dir, 'version')
|
||||
assert File.file? adder_ver_file
|
||||
assert_equal 'initial', File.read(adder_ver_file).strip
|
||||
|
||||
# calculated adder values
|
||||
assert_equal 'ON', cache['ADDER_BUILD_EXAMPLES']
|
||||
assert_equal 'ON', cache['ADDER_BUILD_TESTS']
|
||||
assert_equal adder_cache.src_dir, cache['adder_SOURCE_DIR']
|
||||
assert_equal adder_cache.bin_dir, cache['adder_BINARY_DIR']
|
||||
|
||||
# store for future comparisons
|
||||
adder_cache0 = adder_cache
|
||||
}
|
||||
update_to_version_1 = -> {
|
||||
prj.create_lists_from_default_template package:
|
||||
'CPMAddPackage("gh:cpm-cmake/testpack-adder@1.0.0")'
|
||||
assert_success prj.configure
|
||||
|
||||
cache = prj.read_cache
|
||||
assert_equal 1, cache.packages.size
|
||||
|
||||
adder_cache = cache.packages[ADDER_PACKAGE_NAME]
|
||||
assert_not_nil adder_cache
|
||||
assert_equal '1.0.0', adder_cache.ver
|
||||
|
||||
# dirs shouldn't have changed
|
||||
assert_equal adder_cache0.src_dir, adder_cache.src_dir
|
||||
assert_equal adder_cache0.bin_dir, adder_cache.bin_dir
|
||||
|
||||
assert_equal '1.0.0', File.read(adder_ver_file).strip
|
||||
}
|
||||
update_with_option_off_and_build = -> {
|
||||
prj.create_lists_from_default_template package: <<~PACK
|
||||
CPMAddPackage(
|
||||
NAME testpack-adder
|
||||
GITHUB_REPOSITORY cpm-cmake/testpack-adder
|
||||
VERSION 1.0.0
|
||||
OPTIONS "ADDER_BUILD_TESTS OFF"
|
||||
)
|
||||
PACK
|
||||
assert_success prj.configure
|
||||
assert_success prj.build
|
||||
|
||||
exe_dir = File.join(prj.bin_dir, 'bin')
|
||||
assert File.directory? exe_dir
|
||||
|
||||
exes = Dir[exe_dir + '/**/*'].filter {
|
||||
# on multi-configuration generators (like Visual Studio) the executables will be in bin/<Config>
|
||||
# also filter-out other articacts like .pdb or .dsym
|
||||
!File.directory?(_1) && File.stat(_1).executable?
|
||||
}.map {
|
||||
# remove .exe extension if any (there will be one on Windows)
|
||||
File.basename(_1, '.exe')
|
||||
}.sort
|
||||
|
||||
# we should end up with two executables
|
||||
# * simple - the simple example from adder
|
||||
# * using-adder - for this project
|
||||
# ...and notably no test for adder, which must be disabled from the option override from above
|
||||
assert_equal ['simple', 'using-adder'], exes
|
||||
}
|
||||
|
||||
create_with_commit_sha.()
|
||||
update_to_version_1.()
|
||||
update_with_option_off_and_build.()
|
||||
end
|
||||
end
|
||||
80
test/integration/test_source_cache.rb
Normal file
80
test/integration/test_source_cache.rb
Normal file
@@ -0,0 +1,80 @@
|
||||
require_relative './lib'
|
||||
|
||||
# Tests with source cache
|
||||
|
||||
class SourceCache < IntegrationTest
|
||||
def setup
|
||||
@cache_dir = File.join(cur_test_dir, 'cpmcache')
|
||||
ENV['CPM_SOURCE_CACHE'] = @cache_dir
|
||||
end
|
||||
|
||||
def test_add_remove_dependency
|
||||
prj = make_project 'using-fibadder'
|
||||
|
||||
###################################
|
||||
# create
|
||||
prj.create_lists_from_default_template package: 'CPMAddPackage("gh:cpm-cmake/testpack-fibadder@1.0.0")'
|
||||
assert_success prj.configure
|
||||
|
||||
@cache = prj.read_cache
|
||||
|
||||
# fibadder - adder
|
||||
# \ fibonacci - Format
|
||||
assert_equal 4, @cache.packages.size
|
||||
|
||||
check_package_cache 'testpack-fibadder', '1.0.0', '6a17d24c95c44a169ff8ba173f52876a2ba3d137'
|
||||
check_package_cache 'testpack-adder', '1.0.0', '1a4c153849d8e0cf9a3a245e5f6ab6e4722d8995'
|
||||
check_package_cache 'testpack-fibonacci', '2.0', '332c789cb09b8c2f92342dfb874c82bec643daf6'
|
||||
check_package_cache 'Format.cmake', '1.0', 'c5897bd28c5032d45f7f669c8fb470790d2ae156'
|
||||
|
||||
###################################
|
||||
# add one package with a newer version
|
||||
prj.create_lists_from_default_template packages: [
|
||||
'CPMAddPackage("gh:cpm-cmake/testpack-adder@1.0.1")',
|
||||
'CPMAddPackage("gh:cpm-cmake/testpack-fibadder@1.0.0")',
|
||||
]
|
||||
assert_success prj.configure
|
||||
|
||||
@cache = prj.read_cache
|
||||
assert_equal 4, @cache.packages.size
|
||||
|
||||
check_package_cache 'testpack-fibadder', '1.0.0', '6a17d24c95c44a169ff8ba173f52876a2ba3d137'
|
||||
check_package_cache 'testpack-adder', '1.0.1', '84eb33c1b8db880083cefc2adf4dc3f04778cd44'
|
||||
check_package_cache 'testpack-fibonacci', '2.0', '332c789cb09b8c2f92342dfb874c82bec643daf6'
|
||||
check_package_cache 'Format.cmake', '1.0', 'c5897bd28c5032d45f7f669c8fb470790d2ae156'
|
||||
end
|
||||
|
||||
def test_second_project
|
||||
prj = make_project 'using-fibadder'
|
||||
prj.create_lists_from_default_template package: 'CPMAddPackage("gh:cpm-cmake/testpack-fibadder@1.1.0")'
|
||||
assert_success prj.configure
|
||||
|
||||
@cache = prj.read_cache
|
||||
|
||||
# fibadder - adder
|
||||
# \ fibonacci - Format
|
||||
assert_equal 4, @cache.packages.size
|
||||
|
||||
check_package_cache 'testpack-fibadder', '1.1.0', '603d79d88d7230cc749460a0f476df862aa70ead'
|
||||
check_package_cache 'testpack-adder', '1.0.1', '84eb33c1b8db880083cefc2adf4dc3f04778cd44'
|
||||
check_package_cache 'testpack-fibonacci', '2.0', '332c789cb09b8c2f92342dfb874c82bec643daf6'
|
||||
check_package_cache 'Format.cmake', '1.0', 'c5897bd28c5032d45f7f669c8fb470790d2ae156'
|
||||
end
|
||||
|
||||
def test_cache_dir_contents
|
||||
num_subdirs = -> (name) { Dir["#{File.join(@cache_dir, name.downcase)}/*/"].size }
|
||||
assert_equal 2, num_subdirs.('testpack-fibadder')
|
||||
assert_equal 2, num_subdirs.('testpack-adder')
|
||||
assert_equal 1, num_subdirs.('testpack-fibonacci')
|
||||
assert_equal 1, num_subdirs.('Format.cmake')
|
||||
end
|
||||
|
||||
def check_package_cache(name, ver, dir_sha1)
|
||||
package = @cache.packages[name]
|
||||
assert_not_nil package, name
|
||||
assert_equal ver, package.ver
|
||||
expected_parent_dir = File.join(@cache_dir, name.downcase)
|
||||
assert package.src_dir.start_with?(expected_parent_dir), "#{package.src_dir} must be in #{expected_parent_dir}"
|
||||
assert_equal dir_sha1, File.basename(package.src_dir)
|
||||
end
|
||||
end
|
||||
35
test/integration/tips.md
Normal file
35
test/integration/tips.md
Normal file
@@ -0,0 +1,35 @@
|
||||
# Tips and Tricks
|
||||
|
||||
## Playing and experimenting
|
||||
|
||||
Create a file called `test_local.rb` in this directory to have an integration test which is for your personal experiments and just playing with the integration test framework. `test_local.rb` is gitignored.
|
||||
|
||||
## Speeding-up development
|
||||
|
||||
Running an integration test requires configuring directories with CMake which can be quite slow. To speed-up development of integration tests consider doing the following steps:
|
||||
|
||||
**Work with standalone tests**
|
||||
|
||||
Instead of starting the runner, run just your integration test (`$ ruby test_your_test.rb`). This won't burden the execution with the others.
|
||||
|
||||
**Export the environment variable `CPM_INTEGRATION_TEST_DIR` to some local directory**
|
||||
|
||||
By default the framework generates a new temporary directory for each test run. If you override the temp directory to a specific one, rerunning the tests will work with the binary directories from the previous run and will improve the performance considerably.
|
||||
|
||||
*NOTE HOWEVER* that in certain cases this may not be an option. Some tests might assert that certain artifacts in the temporary directory are missing but upon rerunning in an existing directory they will be there causing the test to fail.
|
||||
|
||||
*ALSO NOTE* that this may silently affect reruns based on CMake caches from previous runs. If your test fails in peculiar ways on reruns, try a clean run. Always do a clean run before declaring a test a success.
|
||||
|
||||
**Set `CPM_SOURCE_CACHE` even if the test doesn't require it**
|
||||
|
||||
This is not a option for tests which explicitly check that there is no source cache. However certain tests may be indiferent to this. For such cases in development, you can add a setup function in the lines of:
|
||||
|
||||
```ruby
|
||||
def setup
|
||||
ENV['CPM_SOURCE_CACHE'] = '/home/myself/.testcpmcache'
|
||||
end
|
||||
```
|
||||
|
||||
Then the packages from your test will be cached and not redownloaded every time which is a dramatic improvement in performance.
|
||||
|
||||
*NOTE HOWEVER* that this may introduce subtle bugs. Always test without this dev-only addition, before declaring a test a success.
|
||||
69
test/integration/tutorial.md
Normal file
69
test/integration/tutorial.md
Normal file
@@ -0,0 +1,69 @@
|
||||
# Integration Test Tutorial
|
||||
|
||||
Let's create an integration test which checks that CPM.cmake can make a specific package available.
|
||||
|
||||
First we do some boilerplate.
|
||||
|
||||
```ruby
|
||||
require_relative './lib'
|
||||
|
||||
class MyTest < IntegrationTest
|
||||
# test that CPM.cmake can make https://github.com/cpm-cmake/testpack-adder/ available as a package
|
||||
def test_make_adder_available
|
||||
end
|
||||
end
|
||||
```
|
||||
|
||||
Now we have our test case class, and the single test method that we will require. Let's focus on the method's contents. The integration test framework provides us with a helper class, `Project`, which can be used for this scenario. A project has an assoiciated pair of source and binary directories in the temporary directory and it provides methods to work with them.
|
||||
|
||||
We start by creating the project:
|
||||
|
||||
```ruby
|
||||
prj = make_project
|
||||
```
|
||||
|
||||
`make_project` is method of IntegrationTest which generates a source and a binary directory for it based on the name of our test class and test method. The project doesn't contain anything yet, so let's create some source files:
|
||||
|
||||
```ruby
|
||||
prj.create_file 'main.cpp', <<~SRC
|
||||
#include <iostream>
|
||||
#include <adder/adder.hpp>
|
||||
int main() {
|
||||
std::cout << adder::add(1, 2) << '\\n';
|
||||
return 0;
|
||||
}
|
||||
SRC
|
||||
prj.create_file 'CMakeLists.txt', <<~SRC
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
project(using-adder)
|
||||
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
|
||||
include("%{cpm_path}")
|
||||
CPMAddPackage("gh:cpm-cmake/testpack-adder@1.0.0")
|
||||
add_executable(using-adder main.cpp)
|
||||
target_link_libraries(using-adder adder)
|
||||
SRC
|
||||
```
|
||||
|
||||
Note the line `include("%{cpm_path}")` when creating `CMakeLists.txt`. It contains a markup `%{cpm_path}`. `Project#create_file` will see such markups and substitute them with the appropriate values (in this case the path to CPM.cmake).
|
||||
|
||||
Now that we have the two files we need it's time we configure our project. We can use the opportunity to assert that the configure is successful as we expect it to be.
|
||||
|
||||
```ruby
|
||||
assert_success prj.configure
|
||||
```
|
||||
|
||||
Now we can read the generated `CMakeCache.txt` and assert that certain values we expect are inside. `Project` provides a method for that: `read_cache`. It will return an instance of `Project::CMakeCache` which contains the data from the cache and provides additional helper functionalities. One of them is `packages`, which is a hash of the CPM.cmake packages in the cache with their versions, binary, source directories. So let's get the cache and assert that there is only one CPM.cmake package inside ant it has the version we expect.
|
||||
|
||||
```ruby
|
||||
cache = prj.read_cache
|
||||
assert_equal 1, cache.packages.size
|
||||
assert_equal '1.0.0', cache.packages['testpack-adder'].ver
|
||||
```
|
||||
|
||||
Finally let's assert that the project can be built. This would mean that CPM.cmake has made the package available to our test project and that it has the appropriate include directories and link libraries to make an executable out of `main.cpp`.
|
||||
|
||||
```ruby
|
||||
assert_success prj.build
|
||||
```
|
||||
|
||||
You can see the entire code for this tutorial in [tutorial.rb](tutorial.rb) in this directory.
|
||||
37
test/integration/tutorial.rb
Normal file
37
test/integration/tutorial.rb
Normal file
@@ -0,0 +1,37 @@
|
||||
# This file is intentionally not prefixed with test_
|
||||
# It is a tutorial for making integration tests and is not to be run from the runner
|
||||
require_relative './lib'
|
||||
|
||||
class Tutorial < IntegrationTest
|
||||
# test that CPM.cmake can make https://github.com/cpm-cmake/testpack-adder/ available as a package
|
||||
def test_make_adder_available
|
||||
prj = make_project
|
||||
|
||||
prj.create_file 'main.cpp', <<~SRC
|
||||
#include <iostream>
|
||||
#include <adder/adder.hpp>
|
||||
int main() {
|
||||
std::cout << adder::add(1, 2) << '\\n';
|
||||
return 0;
|
||||
}
|
||||
SRC
|
||||
|
||||
prj.create_file 'CMakeLists.txt', <<~SRC
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
project(using-adder)
|
||||
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
|
||||
include("%{cpm_path}")
|
||||
CPMAddPackage("gh:cpm-cmake/testpack-adder@1.0.0")
|
||||
add_executable(using-adder main.cpp)
|
||||
target_link_libraries(using-adder adder)
|
||||
SRC
|
||||
|
||||
assert_success prj.configure
|
||||
|
||||
cache = prj.read_cache
|
||||
assert_equal 1, cache.packages.size
|
||||
assert_equal '1.0.0', cache.packages['testpack-adder'].ver
|
||||
|
||||
assert_success prj.build
|
||||
end
|
||||
end
|
||||
@@ -6,7 +6,9 @@ include(${CMAKE_CURRENT_LIST_DIR}/../../cmake/CPM.cmake)
|
||||
|
||||
CPMAddPackage(
|
||||
NAME Format.cmake
|
||||
VERSION 1.6
|
||||
VERSION 1.7.3
|
||||
GITHUB_REPOSITORY TheLartians/Format.cmake
|
||||
OPTIONS "FORMAT_CHECK_CMAKE ON"
|
||||
# We exclude cmake files from integration tests as they contain invalid lines of code which are
|
||||
# used by the integration test scripts
|
||||
OPTIONS "CMAKE_FORMAT_EXCLUDE integration/templates"
|
||||
)
|
||||
|
||||
@@ -33,7 +33,7 @@ set(FIBONACCI_VERSION 1.0)
|
||||
reset_test()
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}"
|
||||
"-DCPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
@@ -76,7 +76,7 @@ reset_test()
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -E env "CPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" ${CMAKE_COMMAND}
|
||||
"-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
"-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
assert_equal(${ret} "0")
|
||||
@@ -86,7 +86,7 @@ assert_exists("${CPM_SOURCE_CACHE_DIR}/fibonacci")
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -E env "CPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" ${CMAKE_COMMAND}
|
||||
"-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}-2"
|
||||
"-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}-2"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
@@ -99,7 +99,7 @@ reset_test()
|
||||
execute_process(
|
||||
COMMAND
|
||||
${CMAKE_COMMAND} -E env "CPM_SOURCE_CACHE=${CMAKE_CURRENT_BINARY_DIR}/junk" ${CMAKE_COMMAND}
|
||||
"-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}"
|
||||
"-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}"
|
||||
"-DCPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
@@ -116,7 +116,7 @@ reset_test()
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -E env "CPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" ${CMAKE_COMMAND}
|
||||
"-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
"-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
assert_equal(${ret} "0")
|
||||
@@ -130,7 +130,7 @@ update_cmake_lists()
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -E env "CPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" ${CMAKE_COMMAND}
|
||||
"-H${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
"-S${CMAKE_CURRENT_LIST_DIR}/remote_dependency" "-B${TEST_BUILD_DIR}" RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
assert_equal(${ret} "0")
|
||||
|
||||
49
test/unit/dirty-cache-check.cmake
Normal file
49
test/unit/dirty-cache-check.cmake
Normal file
@@ -0,0 +1,49 @@
|
||||
include(${CPM_PATH}/CPM.cmake)
|
||||
include(${CPM_PATH}/testing.cmake)
|
||||
|
||||
set(baseDir "${CMAKE_CURRENT_BINARY_DIR}/test_dirty_cache")
|
||||
|
||||
find_package(Git REQUIRED)
|
||||
|
||||
function(git_do dir)
|
||||
execute_process(
|
||||
COMMAND ${GIT_EXECUTABLE} -c user.name='User' -c user.email='user@email.org' ${ARGN}
|
||||
RESULT_VARIABLE result
|
||||
OUTPUT_VARIABLE status
|
||||
OUTPUT_STRIP_TRAILING_WHITESPACE
|
||||
WORKING_DIRECTORY "${dir}"
|
||||
)
|
||||
if(result)
|
||||
message(FATAL_ERROR "git ${ARGN} fail: ${result} ${status}")
|
||||
endif()
|
||||
endfunction()
|
||||
|
||||
file(MAKE_DIRECTORY "${baseDir}")
|
||||
|
||||
file(WRITE "${baseDir}/draft.txt" "this is a test")
|
||||
|
||||
git_do("${baseDir}" init -b main)
|
||||
git_do("${baseDir}" commit --allow-empty -m "empty repo")
|
||||
message(STATUS "empty repo with file")
|
||||
cpm_check_git_working_dir_is_clean(${baseDir} HEAD emptygit_test)
|
||||
assert_falsy(emptygit_test)
|
||||
|
||||
git_do("${baseDir}" add draft.txt)
|
||||
git_do("${baseDir}" commit -m "test change")
|
||||
git_do("${baseDir}" tag v0.0.0)
|
||||
message(STATUS "commit a change")
|
||||
cpm_check_git_working_dir_is_clean(${baseDir} v0.0.0 onecommit_test)
|
||||
assert_truthy(onecommit_test)
|
||||
|
||||
file(WRITE "${baseDir}/draft.txt" "a modification")
|
||||
message(STATUS "dirty repo")
|
||||
cpm_check_git_working_dir_is_clean(${baseDir} v0.0.0 nonemptygit_test)
|
||||
assert_falsy(nonemptygit_test)
|
||||
|
||||
git_do("${baseDir}" add draft.txt)
|
||||
git_do("${baseDir}" commit -m "another change")
|
||||
message(STATUS "repo clean")
|
||||
cpm_check_git_working_dir_is_clean(${baseDir} v0.0.0 twocommit_test)
|
||||
assert_falsy(twocommit_test)
|
||||
|
||||
file(REMOVE_RECURSE "${baseDir}")
|
||||
@@ -11,7 +11,7 @@ function(init_project EXCLUDE_FROM_ALL)
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/broken_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/broken_dependency" "-B${TEST_BUILD_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
38
test/unit/fetchcontent_dependency.cmake
Normal file
38
test/unit/fetchcontent_dependency.cmake
Normal file
@@ -0,0 +1,38 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
include(${CPM_PATH}/testing.cmake)
|
||||
include(CMakePackageConfigHelpers)
|
||||
|
||||
set(CPM_SOURCE_CACHE_DIR "${CMAKE_CURRENT_BINARY_DIR}/CPM")
|
||||
set(TEST_BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/fetchcontent_dependency)
|
||||
|
||||
function(clear_cache)
|
||||
message(STATUS "clearing CPM cache")
|
||||
file(REMOVE_RECURSE ${CPM_SOURCE_CACHE_DIR})
|
||||
assert_not_exists("${CPM_SOURCE_CACHE_DIR}")
|
||||
endfunction()
|
||||
|
||||
function(update_cmake_lists)
|
||||
configure_package_config_file(
|
||||
"${CMAKE_CURRENT_LIST_DIR}/fetchcontent_dependency/CMakeLists.txt.in"
|
||||
"${CMAKE_CURRENT_LIST_DIR}/fetchcontent_dependency/CMakeLists.txt"
|
||||
INSTALL_DESTINATION ${CMAKE_CURRENT_BINARY_DIR}/junk
|
||||
)
|
||||
endfunction()
|
||||
|
||||
function(reset_test)
|
||||
clear_cache()
|
||||
file(REMOVE_RECURSE ${TEST_BUILD_DIR})
|
||||
update_cmake_lists()
|
||||
endfunction()
|
||||
|
||||
# Read CPM_SOURCE_CACHE from arguments
|
||||
|
||||
reset_test()
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/fetchcontent_dependency"
|
||||
"-B${TEST_BUILD_DIR}" "-DCPM_SOURCE_CACHE=${CPM_SOURCE_CACHE_DIR}" RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
assert_equal(${ret} "0")
|
||||
2
test/unit/fetchcontent_dependency/.gitignore
vendored
Normal file
2
test/unit/fetchcontent_dependency/.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
/CMakeLists.txt
|
||||
/package-lock.cmake
|
||||
38
test/unit/fetchcontent_dependency/CMakeLists.txt.in
Normal file
38
test/unit/fetchcontent_dependency/CMakeLists.txt.in
Normal file
@@ -0,0 +1,38 @@
|
||||
# ~~~
|
||||
# ┌────────────────────────┐
|
||||
# │ FetchContentDependency │
|
||||
# └─────┬────────────┬─────┘
|
||||
# │1. │3.
|
||||
# │ │
|
||||
# ┌────────▼────┐ ┌───▼─────────┐
|
||||
# │ Dependency ├───► Fibonacci │
|
||||
# └─────────────┘2. └─────────────┘
|
||||
#
|
||||
# 1. Add Project with CPMAddPackage
|
||||
# 2. Dependency will add Fibonacci with FetchContent
|
||||
# 3. Our project add Fibonacci with CPMAddPackage
|
||||
# ~~~
|
||||
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(CPMTest_FetchContentDependency)
|
||||
|
||||
# ---- Dependencies ----
|
||||
|
||||
include(@CPM_PATH@/CPM.cmake)
|
||||
|
||||
# 1 & 2 Dependency will add Fibonacci using FetchContent (1 & 2)
|
||||
CPMAddPackage(NAME Dependency SOURCE_DIR ${CMAKE_CURRENT_LIST_DIR}/dependency)
|
||||
|
||||
# 3 Add again Fibonacci that have already been populated with FetchContent_MakeAvailable
|
||||
#
|
||||
# * This test should highlight the fact that cpm_add_subdirectory is always called, even when
|
||||
# cpm_fetch_package isn't populating the dependency
|
||||
# * NO_CACHE YES highlight a bug introduced in 32b063eba5c754f833725ed4b9e5f352bc3ca959 where
|
||||
# cpm_fetch_package was checking undefined ${lower_case_name}_POPULATED variable
|
||||
CPMAddPackage(
|
||||
NAME Fibonacci
|
||||
GIT_REPOSITORY https://github.com/cpm-cmake/testpack-fibonacci.git
|
||||
VERSION 2.0
|
||||
NO_CACHE YES
|
||||
)
|
||||
15
test/unit/fetchcontent_dependency/dependency/CMakeLists.txt
Normal file
15
test/unit/fetchcontent_dependency/dependency/CMakeLists.txt
Normal file
@@ -0,0 +1,15 @@
|
||||
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)
|
||||
|
||||
project(CPMTest_Dependency)
|
||||
|
||||
# ---- Dependencies ----
|
||||
|
||||
include(FetchContent)
|
||||
|
||||
FetchContent_Declare(
|
||||
Fibonacci
|
||||
GIT_REPOSITORY https://github.com/cpm-cmake/testpack-fibonacci.git
|
||||
GIT_TAG v2.0
|
||||
)
|
||||
|
||||
FetchContent_MakeAvailable(Fibonacci)
|
||||
@@ -13,7 +13,9 @@ include(@CPM_PATH@/CPM.cmake)
|
||||
CPMAddPackage(
|
||||
NAME Dependency
|
||||
SOURCE_DIR ${CMAKE_CURRENT_LIST_DIR}/dependency
|
||||
OPTIONS "DEFINE_ALTERNATIVE_FUNCTION YES"
|
||||
OPTIONS
|
||||
"DEFINE_ALTERNATIVE_FUNCTION YES"
|
||||
"LIST_ARGUMENT a\\\\;b\\\\;c"
|
||||
EXCLUDE_FROM_ALL YES
|
||||
)
|
||||
|
||||
|
||||
@@ -8,6 +8,12 @@ if(NOT DEFINE_ALTERNATIVE_FUNCTION)
|
||||
message("called external method")
|
||||
endfunction()
|
||||
else()
|
||||
|
||||
# check if list was passed correctly
|
||||
if(NOT "${LIST_ARGUMENT}" STREQUAL "a;b;c")
|
||||
message(FATAL_ERROR "list argument not properly passed to dependency: '${LIST_ARGUMENT}'")
|
||||
endif()
|
||||
|
||||
function(alternative_dependency_function)
|
||||
message("called alternative external method")
|
||||
endfunction()
|
||||
|
||||
@@ -11,7 +11,7 @@ function(init_project_with_dependency TEST_DEPENDENCY_NAME)
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
@@ -12,7 +12,7 @@ configure_package_config_file(
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
@@ -24,7 +24,7 @@ function(configure_with_declare DECLARE_DEPENDENCY)
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -H${CMAKE_CURRENT_LIST_DIR}/local_dependency -B${TEST_BUILD_DIR}
|
||||
COMMAND ${CMAKE_COMMAND} -S${CMAKE_CURRENT_LIST_DIR}/local_dependency -B${TEST_BUILD_DIR}
|
||||
-DCPM_INCLUDE_ALL_IN_PACKAGE_LOCK=1 RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
@@ -12,7 +12,7 @@ configure_package_config_file(
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} -H${CMAKE_CURRENT_LIST_DIR}/local_dependency -B${TEST_BUILD_DIR}
|
||||
COMMAND ${CMAKE_COMMAND} -S${CMAKE_CURRENT_LIST_DIR}/local_dependency -B${TEST_BUILD_DIR}
|
||||
-DCPM_Dependency_SOURCE=${CMAKE_CURRENT_LIST_DIR}/local_dependency/dependency
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
@@ -30,6 +30,12 @@ assert_equal("GITLAB_REPOSITORY;foo/bar" "${args}")
|
||||
cpm_parse_add_package_single_arg("gl:foo/Bar" args)
|
||||
assert_equal("GITLAB_REPOSITORY;foo/Bar" "${args}")
|
||||
|
||||
cpm_parse_add_package_single_arg("bb:foo/bar" args)
|
||||
assert_equal("BITBUCKET_REPOSITORY;foo/bar" "${args}")
|
||||
|
||||
cpm_parse_add_package_single_arg("bb:foo/Bar" args)
|
||||
assert_equal("BITBUCKET_REPOSITORY;foo/Bar" "${args}")
|
||||
|
||||
cpm_parse_add_package_single_arg("https://github.com/cpm-cmake/CPM.cmake.git@0.30.5" args)
|
||||
assert_equal("GIT_REPOSITORY;https://github.com/cpm-cmake/CPM.cmake.git;VERSION;0.30.5" "${args}")
|
||||
|
||||
|
||||
@@ -12,7 +12,7 @@ configure_package_config_file(
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
@@ -10,7 +10,7 @@ configure_package_config_file(
|
||||
)
|
||||
|
||||
execute_process(
|
||||
COMMAND ${CMAKE_COMMAND} "-H${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
COMMAND ${CMAKE_COMMAND} "-S${CMAKE_CURRENT_LIST_DIR}/local_dependency" "-B${TEST_BUILD_DIR}"
|
||||
RESULT_VARIABLE ret
|
||||
)
|
||||
|
||||
|
||||
Reference in New Issue
Block a user