Search code examples
iosswiftgitcontinuous-integrationswift-package-manager

Swift Package Manager Dependency Mirroring


I have a project and:

  • the company I'm working with is self-hosting their git
  • the CI can communicate only with company's network

That being said, if I want to install any dependency, I have to ask their dev ops to mirror the target repository and only then I can use it.

The problem arises when I want to implement Crashlytics which has a lot of dependencies. When I import the Firebase, it's fetched from the mirrored repo correctly, but it's dependencies are still being fetched from the original URLs (which makes perfect sense).

The question is - How do I tell Swift Package Manager to swap each URL with mirrors? I have all the dependencies mirrored. I only need to tell SPM to use it.

I have found this proposal which was implemented in Swift 5, but when I go to root of my project and run:

$ swift package config set-mirror --package-url <original URL> --mirror-url <mirror URL>

I get this error:

error: root manifest not found

Any ideas how to do this correctly? Thank you

EDIT:

As Florian correctly pointed out, the proposal works from the package's repository, not my projects! So:

  1. I do clone mirrored repo in my project's root
  2. I run set of commands to set mirror url for each dependency:
swift package config set-mirror \
    --original-url https://github.com/google/GoogleAppMeasurement.git \
    --mirror-url <company's url>/mirrors/githubcom-google-GoogleAppMeasurement
  1. I go back to projects root and run:
xcodebuild -resolvePackageDependencies -project MyProject.xcodeproj -scheme MyAppScheme

But it's still fetching from original urls, not the mirrors!


Solution

  • Having an Xcode project makes this task basically impossible (at the time of writing). Xcode's integration with SPM works fine for most things, but is not (yet?) at par with what SPM can do in pure SPM packages.

    The problem is, that swift package config is always only local to the package and does not have any effect on projects / packages that depend on the package. And with Xcode currently having no counterpart to swift package config, it's not possible to do this at the moment.

    What you could do, however, is to clone all your dependencies locally and then reference them as local packages from Xcode (simply dragging the package folder into the open Xcode project will do so). Xcode will be smart enough to take the dependencies from the local local checkout (or at least it was smart enough last time I tried this).

    Let's hope for a future Xcode version with full SPM support!