This repository holds several sample Gradle builds which demonstrate how to use Gradle to build Swift/C++ libraries and applications as introduced in our blog post.
Each sample build is listed below with a bit of information related to the features of Gradle that are demonstrated in that build. Each sample is functionally the same for both Swift and C++ languages.
Each C++ sample works on macOS, Linux and Windows with GCC, Clang and Visual C++.
Each Swift sample works on macOS and Linux, with Swift 3 and later.
You can generate build-scans with these samples by running Gradle with --scan
. At the end of the build, you will be prompted to upload build data to scans.gradle.com.
As an example of adding more data to a build scan, you can also run any sample with -I ../../build-scan/buildScanUserData.gradle
in combination with --scan
. This will add custom values that describe what is being built like these.
If you want to contribute an improvement to the samples, please refer to the samples-dev
subproject.
If you have a use case that isn't covered by an existing sample, open an issue for gradle-native. Please describe what you're trying to accomplish so we can help you find a solution.
All of the C++ samples have Visual Studio support, added by applying the visual-studio
plugin. To open a sample build in Visual Studio:
> cd <sample-dir>
> ./gradlew openVisualStudio
All of the samples have Xcode support, added by applying the xcode
plugin. To open a sample build in Xcode:
> cd <sample-dir>
> ./gradlew openXcode
All Swift samples demonstrate XCTest support in Gradle. As a user, you can either interact with the test code as you would usually do through Xcode or you can run the test directly from the command line:
> cd <sample-dir>
> ./gradlew test
The C++ sample simple-library
demonstrates some basic Google test support in Gradle. This is currently not as refined as the XCTest support.
To run the tests from the command line:
> cd <sample-dir>
> ./gradlew test
The swiftc
compiler has a built-in incremental compilation feature that tries to reduce the number of .swift
files that need to be recompiled on each build by analyzing the dependencies between all files.
Gradle enables Swift incremental compilation by default, so no extra configuration is required to take advantage of this feature with your Swift projects.
The Swift and C++ plugins add a 'debug' and 'release' variant for each library or application. By default, the assemble
task will build the debug variant only.
You can also use the assembleDebug
and assembleRelease
tasks to build a specific variant, or both variants.
At this stage, there are no convenience tasks to build all variants of a library or application.
Some of the C++ samples are configured to publish binaries to a local Maven repository. For these samples you can run:
> cd <sample-dir>
> ./gradle publish
> tree ../repo/
This will build and publish the debug and release binaries. The binaries are published to a repository in the cpp/repo
directory.
This sample shows how a simple Swift or C++ application can be built with Gradle. The application has no dependencies and the build has minimal configuration.
Although there is currently no out-of-the-box support for building applications and libraries from C, there is also a sample build that shows how the C++ support can be configured to build a C application.
To build and run the application:
> cd cpp/application
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
To build and run the application:
> cd swift/application
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!
To build and run the application:
> cd c/application
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
This sample shows how a Swift or C++ library can be built with Gradle. The library has no dependencies. The build is configured to add unit tests. The C++ sample also adds binary publishing to a Maven repository.
To build the library:
> cd cpp/simple-library
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> find build/lib/main/debug
build/lib/main/debug/liblist.dylib
To run the unit tests for the library:
> ./gradlew test
BUILD SUCCESSFUL in 1s
To build the library:
> cd swift/simple-library
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> find build/lib/main/debug
build/lib/main/debug/libList.dylib
To run the unit tests for the library:
> ./gradlew test
> open build/reports/tests/xcTest/index.html
This sample shows how a C++ or Swift application and several libraries can be built with Gradle and linked together. The sample is structured as a multi-project build, with the application and each library as separate projects in this build. Dependencies are added using project dependencies.
In this sample, the application and libraries all use the same implementation language. Mixing C++ and Swift is shown in another sample below.
To build and run the application:
> cd cpp/transitive-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
The build script also demonstrates how to configure convenience tasks like assembleDebuggable
, which will assemble all "debuggable" binaries.
> ./gradlew assembleDebuggable
BUILD SUCCESSFUL in 1s
To build and run the application:
> cd swift/transitive-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/App
Hello, World!
This sample shows how Gradle is able to compile Windows Resources (rc
) files and link them into a native binary.
This sample applies the 'cpp-application'
plugin.
This sample requires you have VisualCpp toolchain installed
To build and run the application: (Note the application only runs and build on Windows)
> cd cpp/windows-resources
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/exe/main/debug/app.exe
Hello, World!
This sample shows how Gradle is able to compile code using precompiled headers.
This sample applies the 'cpp-application'
plugin.
To build and run the application:
> cd cpp/precompiled-headers
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/exe/main/debug/app
Hello, World!
This sample shows how several otherwise independent Swift or C++ libraries can be built together with Gradle. The sample is structured as separate builds for each of the libraries and a composite build that includes these.
To build and run the application:
> cd cpp/composite-build
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
To build and run the application:
> cd swift/composite-build
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!
This sample shows how to publish C++ libraries to a Maven repository and use them from another build. This is currently not supported for Swift.
To use the sample, first publish a library to the repository using the simple-library
build:
> cd cpp/binary-dependencies
> ./gradlew -p ../simple-library publish
BUILD SUCCESSFUL in 1s
You can find the repository in the cpp/repo
directory.
Next, build the application that uses the library from this repository.
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
The build is also configured to download the Google test binaries from a Maven repository to build and run the unit tests:
> ./gradlew test
BUILD SUCCESSFUL in 1s
This sample shows how to use pre-built binaries that are already available on the local machine. Currently, Gradle does not offer a convenient way to do this but it is possible to configure Gradle to use these binaries.
To use the sample, first create the binaries using the simple-library
build:
> cd cpp/prebuilt-binaries
> ./gradlew -p ../simple-library assembleDebug assembleRelease
BUILD SUCCESSFUL in 1s
Next, build and run the application that uses these binaries:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
To use the sample, first create the binaries using the simple-library
build:
> cd swift/prebuilt-binaries
> ./gradlew -p ../simple-library assembleDebug assembleRelease
BUILD SUCCESSFUL in 1s
Next, build and run the application that uses these binaries:
> ./gradlew assemble
BUILD SUCCESSFUL in 0s
> ./build/install/main/debug/App
Hello, World!
This sample shows how to configure Gradle to use a source layout that is different to its conventions. In this case, the sample uses the typical layout for a Swift Package Manager package.
It contains an application and a single library. The source files for the application and libraries are all under a single Sources
directory.
This sample also includes a Swift Package Manager build file, so the same source can be built using Swift Package Manager
> cd cpp/swift-package-manager
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/app/install/main/debug/app
Hello, World!
> cd swift/swift-package-manager
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/app/install/main/debug/App
Hello, World!
This sample demonstrates using a task to generate source code before building a Swift or C++ application.
> cd cpp/source-generation
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
Generated sources will be under build/generated
.
> cd swift/source-generation
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!
Generated sources will be under build/generated
.
This sample demonstrates using external source dependencies to build Swift and C++ applications that require two libraries. The source for the libraries are hosted in separate Git repositories and declared as 'source dependencies' of the application. When Gradle builds the application, it first checks out a revision of the library source and uses this to build the binaries for the library.
The Git repositories to use are declared in the build's settings.gradle
and then the libraries are referenced in the same way as binary dependencies in the build files.
To use this sample, build and run the application:
> cd swift/source-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
World!
This will clone version 1.0
of the Git repository at https://github.com/gradle/native-samples-swift-library
and build the library binaries.
You can see the application's output is incorrect. The build is configured to use version 1.0
of the utilities library from this repository and this version contains a bug. Let's fix this.
Version 1.1
of the library contains a bug fix. Update the application to use the new version:
> cd ../..
> edit build.gradle # change dependency on org.gradle.swift-samples:utilities:1.0 to org.gradle.swift-samples:utilities:1.1
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!
Dynamic dependencies are also supported, so you could also use 1.+
, [1.1,2.0]
or latest.release
. Gradle matches the tags of the Git repository to determine which Git revision to use. Branches are also supported, but use a different syntax. See the following sample.
To use this sample, build and run the application:
> cd cpp/source-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
World!
This will clone version 1.0
of the Git repository at https://github.com/gradle/native-samples-cpp-library
and build the library binaries.
The application output is incorrect because of a bug in the utilities library.
Update the application to use a new version that contains a fix:
> cd ../..
> edit build.gradle # change dependency on org.gradle.cpp-samples:utilities:1.0 to org.gradle.cpp-samples:utilities:1.1
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
This sample shows how a source dependency on a particular branch can be used.
To use this sample, create the Git repositories containing the libraries:
> cd swift/dependency-on-upstream-branch/app
> ./gradlew -p ../../.. generateRepos
Now you can build and run the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
World!
You can see the application's output is incorrect. The build is configured to use the most recent changes from the 'release' branch of the utilities library and this branch contains a bug. Let's fix this.
Edit the source of the utilities library to fix the bug:
> cd repos/utilities-library
> git checkout release
> edit src/main/swift/Util.swift # follow the instructions to fix the bug in function join()
> git commit -a -m 'fixed bug'
There's no need to create a tag, as Gradle will take care of checking out the new branch tip.
Now build and run the application:
> cd ../..
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!
To use this sample, create the Git repositories containing the libraries:
> cd cpp/dependency-on-upstream-branch/app
> ./gradlew -p ../../.. generateRepos
Build and run the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
World!
Edit the source of the utilities library to fix the bug:
> cd repos/utilities-library
> git checkout release
> edit src/main/cpp/join.cpp # follow the instructions to fix the bug in function join()
> git commit -a -m 'fixed bug'
Now build the application:
> cd ../..
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
This sample demonstrates building and using static libraries.
> cd cpp/static-library
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
> cd swift/static-library
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/App
Hello, World!
Application with operating system specific library dependencies (operating-system-specific-dependencies)
This sample demonstrates an application that has dependencies on different libraries for each operating system. Currently, there are no conveniences for using libraries that are installed on the build machine.
> cd swift/operating-system-specific-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> app/build/install/main/debug/App
Hello, World!
The application selects the 'MacOsConsole' library that prints the output in blue when building on macOS. On Linux, it selects the 'LinuxConsole' library that prints the output in green. Each console library is configured to only build on specific operating systems.
> cd cpp/operating-system-specific-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
The application selects the 'ansiConsole' library on macOS and Linux and the 'winConsole' library when built on Windows. The output is blue on macOS and green on Linux and Windows. Each console library is configured to only build on specific operating systems, while the application is configured to build on all operating systems (Windows, Linux, macOS).
This sample demonstrates using a C++ library from Swift.
> cd swift/cpp-dependencies
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
There are three projects in this sample: a shared C++ library (:list
), a static C++ library (:listStatic
), and a swift application with a dependency on the shared C++ library (:app
).
By default, when building the application, only the shared library will be built and used. To change the application to use the static library instead, change the dependency in app/build.gradle to depend on the :listStatic
project instead of :list
. You'll also need to change app/src/main/swift/LinkedList.swift to import the "listStatic" module (rather than the "list" module).
This sample demonstrates using multiple versions of Swift in a single build. There are two projects that build identical applications. One is written in Swift 3 compatible code (swift3-app
) and one is written with Swift 4 compatible code (swift4-app
). When running the application, it will print a message about which version of Swift was used.
If you have the Swift 4 compiler installed, you can build both applications:
# NOTE: Needs Swift 4 tool chain
> cd swift/swift-versions
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./swift4-app/build/install/main/debug/App
Built for Swift 4
Hello, World!
> ./swift3-app/build/install/main/debug/App
Built for Swift 3
Hello, World!
By default, the tests for a given Swift production component will be compiled for the same version of Swift. For instance, in swift3-app
, the production and test code will be built with Swift 3 source compatibility.
If you have the Swift 3 compiler installed, you can only build the Swift 3 application. Attempting to build the Swift 4 application will fail.
> cd swift/swift-versions
> ./gradlew swift3-app:assemble
BUILD SUCCESSFUL in 1s
> ./swift3-app/build/install/main/debug/app
Built for Swift 3
Hello, World!
Currently, Gradle does not offer a convenience to ignore projects that are not buildable due to missing or incompatible tool chains.
Gradle can also consume source dependencies that come from repositories without Gradle builds. When declaring a source dependency's repository information, you can instruct Gradle to inject plugins into the source dependency. These plugins can configure a Gradle build based on the contents of the repository.
To use the sample, create the Git repositories containing the libraries:
> cd cpp/injected-plugins
> ./gradlew -p ../.. generateRepos
Now build the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
In the "repos" directory, you can find the source code without any Gradle configuration. The utilities
and list
builds are configured with the utilities-build
and list-build
plugins.
To use the sample, create the Git repositories containing the libraries:
> cd swift/injected-plugins
> ./gradlew -p ../.. generateRepos
Now build the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, from Gradle build
Hello, World!
In the "repos" directory, you can find the source code without any Gradle configuration. The utilities
and list
builds are configured with the utilities-build
and list-build
plugins.
This sample demonstrates integrating a library that is built by CMake into a Gradle build. The sample is structured as a multi-project build. There are two projects, 'app' and 'utilities', which are built by Gradle. Both of these depend on a library 'list' that is built using CMake. The 'list' library has a Gradle project that wraps the CMake build and exposes its outputs in a way that other Gradle builds can consume.
The sample packages the CMake integration logic as a 'cmake-library' plugin and applies the plugin to the 'library' project as a source dependency.
To use the sample, first create the Git repository containing the sample plugin:
> cd cpp/cmake-library
> ./gradlew -p ../.. generateRepos
Now build the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
This generates the CMake build for the 'list' library, if required, and builds the libraries and application.
This sample demonstrates using Gradle's dependency management features to coordinate building an application and libraries built by CMake. The sample is composed of an application and two libraries. Each of these is hosted in a separate Git repository and connected together using source dependencies.
The sample packages the CMake integration logic as a 'cmake-application' and 'cmake-library' plugin and applies these to the different builds.
To use the sample, first create the Git repository containing the sample plugin:
> cd cpp/cmake-source-dependencies/app
> ./gradlew -p ../../.. generateRepos
Now build the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/debug/app
Hello, World!
This generates the CMake builds for each of the libraries and the application, then builds and links the libraries and applications.
This sample shows how libraries built with Gradle can be used by projects that are built with Swift Package Manager, without having to maintain separate Gradle and Swift PM build for the library.
The sample is made up of an application built using Swift PM, and two libraries that are built using Gradle. The sample also includes a 'release' plugin that takes care of generating a Swift PM build from a Gradle build.
To use the sample, setup the Git repositories for the libraries:
> ./gradlew generateRepos
Next, create a release of the list library that can be used by Swift PM. This generates a Package.swift
file to be used by Swift PM, commits the changes and creates a tag:
> cd swift/swift-package-manager-publish/list-library
> ./gradlew build release
BUILD SUCCESSFUL in 1s
Do the same for the utilities library:
> cd ../utilities-library
> ./gradlew build release
BUILD SUCCESSFUL in 1s
Now build the application using Swift PM:
> cd ../app
> swift build
> ./gradlew generateRepos
> cd cpp/swift-package-manager-publish/list-library
> ./gradlew build release
BUILD SUCCESSFUL in 1s
> cd ../utilities-library
> ./gradlew build release
BUILD SUCCESSFUL in 1s
> cd ../app
> swift build
Existing system libraries can be wrapped in user defined module.modulemap
files.
This sample demonstrates a Swift application that uses libcurl to fetch example.com
.
> cd swift/system-library-as-module
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
This sample shows how to provision tool chains used by a Gradle build instead of the system tool chains. Please note the sample doesn't provision any tool chain for Windows yet. The sample can only provision tool chain at configuration. We use the buildSrc
included build to use tasks for the provisioning.
This sample demonstrates a Swift tool chain provisioning under Linux.
> cd swift/provisionable-tool-chains
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
This sample demonstrates a Clang tool chain provisioning under macOS and Linux.
> cd cpp/provisionable-tool-chains
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
This sample demonstrates using Gradle's dependency management features to coordinate building libraries built by Autotools (i.e. configure
and make
). The sample is composed of an application and a curl library. The curl library is downloaded from the Curl home page and then built using the Autotools configuration provided with the library. The application is built with Gradle, linking the curl library statically. When run, the application uses curl to query a REST service to lookup the capital of Sweden and prints this to the console.
To use the sample, create the Git repositories containing the libraries:
> cd cpp/autotools-library
> ./gradlew -p ../.. generateRepos
Now build the application:
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
This sample demonstrates some basic Google test support in Gradle, building the GoogleTest library from source.
To run the tests from the command line:
> cd cpp/library-with-tests
> ./gradlew -p ../.. generateRepos
> ./gradlew assemble
> ./gradlew test
This sample demonstrates header-only libraries.
To use the sample, run the tests from the command line:
> cd cpp/header-only-library
> ./gradlew assemble
> ./gradlew test
This sample demonstrates a iOS 11.2 application build for the iPhone simulator, allowing editing of the iOS specific files inside Xcode.
To use the sample, build the application:
> cd swift/ios-application
> ./gradlew assemble
Now install the application into an iOS simulator running iOS 11.2 by drag and dropping the app into a running simulator.
Finally, you can develop the application using Xcode IDE to edit the storyboard and asset catalog. Note that the sample doesn't allow running the iOS application from the IDE.
> ./gradlew openXcode
This sample demonstrates how to publish macros to downstream projects.
To build and run the application:
> cd cpp/publish-macros
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./app/build/install/main/debug/app
Hello, World!
This sample shows how a simple Swift or C++ application can target multiple machines with Gradle. The application has no dependencies and the build has minimal configuration.
To build and run the application for the target machine of the same type as the current host:
> cd cpp/multiple-target-machines
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/app
Hello, World!
To build and run the application for the target machine of the same type as the current host:
> cd swift/multiple-target-machines
> ./gradlew assemble
BUILD SUCCESSFUL in 1s
> ./build/install/main/debug/App
Hello, World!