This repository contains a Bazel ruleset that can be used to download, build, and consume Swift packages. The rules in this repository build the external Swift packages using rules_swift, rules_apple and native C/C++ rulesets making the Swift package products and targets available as Bazel targets.
This repository is designed to fully replace rules_spm and provide utilities to ease Swift development inside a Bazel workspace.
- Documentation
- Prerequisites
- Quickstart
- Using a Swift package registry
- Tips and Tricks
Be sure to install Xcode.
You will need to install Swift. Make sure
that running swift --version works properly.
Don't forget that rules_swift expects the use of
clang. Hence,
you will need to specify CC=clang before running Bazel.
Finally, help rules_swift and rules_swift_package_manager find the Swift toolchain by ensuring
that a PATH that includes the Swift binary is available in the Bazel actions.
cat >>local.bazelrc <<EOF
build --action_env=PATH
EOFThis approach is necessary to successfully execute the examples on an Ubuntu runner using Github actions. See the CI GitHub workflow for more details.
The following provides a quick introduction on how to set up and use the features in this repository. These instructions assume that you are using Bazel modules to load your external dependencies. If you are using Bazel's legacy external dependency management, we recommend using Bazel's hybrid mode, then follow the steps in this quickstart guide.
Also, check out the examples for more information.
This repository supports bzlmod.
common --enable_bzlmod
2. Configure your MODULE.bazel to use rules_swift_package_manager.
Add a dependency on rules_swift_package_manager.
bazel_dep(name = "rules_swift_package_manager", version = "1.9.0")In addition, add the following to load the external dependencies described in your Package.swift
and Package.resolved files.
swift_deps = use_extension(
    "@rules_swift_package_manager//:extensions.bzl",
    "swift_deps",
)
swift_deps.from_package(
    resolved = "//:Package.resolved",
    swift = "//:Package.swift",
)
use_repo(
    swift_deps,
    "swift_deps_info",  # This is generated by the ruleset.
    # The name of the Swift package repositories will be added to this declaration in step 4 after
    # running `bazel mod tidy`.
    # NOTE: The name of the Bazel external repository for a Swift package is `swiftpkg_xxx` where
    # `xxx` is the Swift package identity, lowercase, with punctuation replaced by `hyphen`. For
    # example, the repository name for apple/swift-nio is `swiftpkg_swift_nio`.
)You will also need to add a dependency on rules_swift.
NOTE: Some Swift package manager features (e.g., resources) use rules from rules_apple. It is a
dependency for rules_swift_package_manager. However, you do not need to declare it unless you use
any of the rules in your project.
The swift_deps module extension will by default generate a swift_package repository which can be used to execute swift package commands.
This is useful if you'd like to control the flags and behavior of swift package, as well as for using the correct swift binary according to the Bazel configured toolchain.
For example, to resolve the Package.swift file:
bazel run @swift_package//:resolveTo update packages to their latest supported version:
bazel run @swift_package//:updateBoth targets support passing arguments as well, so for example, you could update a single package:
bazel run @swift_package//:update -- MyPackageThese targets will update the Package.resolved file defined in swift_deps.from_package.
The targets come with default flags applied to enable the best Bazel compatibility, if you wish to configure it further, you can do so with configure_swift_package:
# MODULE.bazel
swift_deps.configure_swift_package(
    build_path = "spm-build",
    cache_path = "spm-cache",
    dependency_caching = "false",
    manifest_cache = "none",
    manifest_caching = "false",
)If you do not want to use the swift_package repository you can disable it in the swift_deps.from_package call:
swift_deps.from_package(
    declare_swift_package = False,  # <=== Disable the `swift_package` repository
    resolved = "//:Package.resolved",
    swift = "//:Package.swift",
)If you will be using the Gazelle plugin for Swift, you will need to enable the generation of
the swift_deps_info repository by enabling declare_swift_deps_info.
swift_deps.from_package(
    declare_swift_deps_info = True, # <=== Enable swift_deps_info generation for the Gazelle plugin
    resolved = "//:Package.resolved",
    swift = "//:Package.swift",
)Create a minimal Package.swift file that only contains the external dependencies that are directly
used by your Bazel workspace.
// swift-tools-version: 5.7
import PackageDescription
let package = Package(
    name: "my-project",
    dependencies: [
        // Replace these entries with your dependencies.
        .package(url: "https://github.com/apple/swift-argument-parser", from: "1.2.0"),
        .package(url: "https://github.com/apple/swift-log", from: "1.4.4"),
    ]
)The name of the package can be whatever you like. It is required for the manifest, but it is not used by rules_swift_package_manager. If your project is published and consumed as a Swift package, feel free to populate the rest of the manifest so that your package works properly by Swift package manager. Just note that the Swift Gazelle plugin does not use the manifest to generate Bazel build files.
This will invoke Swift Package Manager and resolve all dependencies resulting in creation of
Package.resolved file.
This will update your MODULE.bazel with the correct use_repo declaration.
Build and test your project.
bazel test //...- The Package.swiftfile is used byrules_swift_package_managerto generate information about your project's dependencies.
- The Package.resolvedfile specifies that exact versions of the downloaded dependencies that were identified.
- The MODULE.bazelcontains the declarations for your external dependencies.
You are ready to start coding.
See our document on using a Swift package registry.
The following are a few tips to consider as you work with your repository:
- Are you trying to use a Swift package and it just won't build under Bazel? If you can figure out how to fix it, you can patch the Swift package. Check out our document on patching Swift packages.