This document describes how to integrate CleanroomLogger into your application.
Integration is the act of embedding the CleanroomLogger.framework
binary (and its required CleanroomASL.framework
dependency) into your project, thereby exposing the API it provides to your code.
CleanroomLogger is built as a Swift framework, and as such, it has the following base platform requirements:
Platform | Minimum OS version |
---|---|
Apple TV | tvOS 9.0 |
Apple Watch | watchOS 2.0 |
iPhone/iPad | iOS 8.0 |
Mac | OS X 10.10 |
CleanroomLogger is Swift 2.2 compliant and therefore requires Xcode 7.3 or higher to compile.
- Options for integration
- Instructions for integration using Carthage
- Instructions for manual integration
Some familiarity with the Terminal application, the bash command line, and the git
command is assumed.
The steps below have been tested with git 2.6.4 (Apple Git-63), although they should be compatible with a wide range of recent git versions.
When developing for a platform with a simulator, there are some kinks in the process:
-
Simulators use a different processor architecture than real devices. As a result, frameworks compiled only for device won’t work in the simulator, and vice-versa. Sure, you could use
lipo
to stitch together a universal binary and use that instead, but... -
Apple will not accept App Store binaries containing simulator code. That means if you go the universal binary route, you now need a custom build step to make the universal binary, and another step to un-make the universal binary when you’re building for submission. Okay, well, why not just build two frameworks: one specifically for the simulator and the other for devices?
-
Xcode won’t be happy if you import two separate frameworks with the same symbols. Xcode won’t notice that there really is no conflict because the frameworks are compiled for different processor architectures. All Xcode will care about is that you’re importing two separate frameworks that both claim to have the same module name.
For these reasons, we do not release Cleanroom projects as framework binaries. Instead, we provide the source, the Xcode project file, and options for integrating so that you can use it from within your code and submit an app that Apple will accept. (Or, at the very least, if Apple doesn’t accept your app, we don’t want it to be the fault of this project!)
There are two supported options for integration:
-
Carthage integration — Explains how to use the Carthage dependency manager for adding CleanroomLogger to your project.
-
Manual integration — Demonstrates the steps for adding CleanroomLogger by embedding
CleanroomLogger.xcodeproj
within your own Xcode project. (Note: Manual integration is a bit more involved than using Carthage).
Whether you choose one over the other largely depends on your preferences and—in the case of Carthage—whether you’re already using that solution for other dependencies.
Carthage is a third-party package dependency manager for iOS and Mac OS X. Carthage works by building frameworks for each of a project’s dependencies.
Before attempting any of the steps below, you should verify that Carthage is available on your system. To do that, open Terminal and execute the command:
carthage version
If Carthage is available, the version you have installed will be shown.
As of this writing, the current version of Carthage is 0.15.2.
If Carthage is not present, you will see an error that looks like:
-bash: carthage: command not found
Installing Carthage is beyond the scope of this document. If you do not have Carthage installed but would like to use it, you can find installation instructions on the project page.
When building a framework for a platform with a simulator, Carthage creates a universal binary, allowing CleanroomLogger.framework
to work in the simulator as well as on actual devices.
However, because Apple will not accept App Store submissions containing universal binary code, Carthage requires the addition of a build step that strips all unused architectures out of the universal binaries. That way, when building for the simulator, device code is removed; conversely, when creating a device build, simulator code is removed. This keeps Apple happy, while also making it easy to switch back and forth between running on the device and in the simulator.
Carthage integration is a little simpler than manual integration:
- Update the
Cartfile
with an entry for CleanroomLogger - Download and build CleanroomLogger
- Add
CleanroomLogger.framework
andCleanroomASL.framework
to your application target - Create a build phase to strip the extra processor architectures from the Carthage frameworks (not necessary for Mac OS X builds)
We’ll start in the Terminal, by cd
ing into to your project’s root directory. The commands you’ll need to issue below can all be done from this location.
In your project’s root directory, edit the file named Cartfile
—creating it if necessary—to add the following line:
github "emaloney/CleanroomLogger"
The command carthage update
causes Carthage to download and build CleanroomLogger.
By default, carthage update
builds all platform targets in a project. Normally, this is not what you want; you’ll usually use CleanroomLogger on just one platform at a time.
To speed up the build process—and to avoid trying to build for a platform that might not be supported by the version of Xcode you have—it is strongly recommended that you pass the --platform
argument to carthage update
:
To build for | Run the command |
---|---|
Apple TV | carthage update --platform tvos |
Apple Watch | carthage update --platform watchos |
iPhone/iPad | carthage update --platform ios |
Mac | carthage update --platform mac |
Carthage puts its files within a top-level directory called Carthage
at the root of your project’s directory structure (i.e., the Carthage
directory is a sibling of the Cartfile
). Within this directory are two more directories: Build
, which contains any frameworks built by Carthage; and Checkouts
, which contains fully populated directory structures for each repository specified in the Cartfile
.
Note: By default, Carthage builds a framework for each platform supported by the project. You can limit the build to a specific platform by specifying a value for the
--platform
argument when invoking thecarthage
command.
Once Carthage is done building CleanroomLogger and its dependencies, you can use the open
command to locate the framework binaries in Finder:
open Carthage/Build/iOS
The command above opens the directory containing the iOS framework binaries; to locate the Mac OS X binaries, execute:
open Carthage/Build/Mac
If all went well, the Carthage build directories should contain the files:
CleanroomLogger.framework
CleanroomASL.framework
If those files aren’t present, something went wrong with the build.
In Xcode, select the General tab in the build settings for your application target. Scroll to the bottom of the screen to reveal the section entitled Embedded Binaries (the second-to-last section).
Go back to Finder, select the files CleanroomLogger.framework
and CleanroomASL.framework
, and then drag them into the list area directly below Embedded Binaries.
If successful, you should see CleanroomLogger.framework
and CleanroomASL.framework
listed under both the Embedded Binaries and Linked Frameworks and Libraries sections.
Note: You do not need to perform this step when building for Mac OS X. This step is only necessary when building for targets that can be run in a simulator.
In Xcode, select the Build Phases tab in the build settings for your application target.
At the top-left corner of the list of build phases, you will see a “+
” icon. Click that icon and add a “New Run Script Phase”.
Then, in the script editor area just below the Shell line, add the following text:
"$PROJECT_DIR"/Carthage/Checkouts/CleanroomLogger/BuildControl/bin/stripCarthageFrameworks.sh
This script will ensure that any frameworks built by Carthage are stripped of unnecessary processor architectures. Without this step, Apple would reject your app submission because the frameworks would be included as universal binaries, which isn’t allowed in App Store submissions.
Once you’ve done this, try building your application. If you don’t see any errors, you’re all done integrating CleanroomLogger!
But before you start coding, skip to the Adding the Swift import section to see how you can import CleanroomLogger for use in your Swift code.
Manual integration involves embedding CleanroomLogger.xcodeproj
directly in your Xcode project.
This will ensure that CleanroomLogger is built with the exact same settings you’re using for your app. You won’t have to fiddle with different settings for different architectures — when you’re running in the simulator, it will work; when you then switch to building for device, it’ll work, too.
You’ll also be able to step into CleanroomLogger code directly in the debugger without worrying about .dSYM
resolution, which is very helpful.
Manual integration is a bit involved; there are five high-level tasks that you’ll need to perform:
- Download the CleanroomLogger source into your project structure
- Embed
CleanroomLogger.xcodeproj
in your Xcode project - Build
CleanroomLogger.framework
; this will also cause theCleanroomASL.framework
dependency to be built - Add
CleanroomLogger.framework
andCleanroomASL.framework
to your application target - Fix the way Xcode references the frameworks you added in Step 4
Launch Terminal on your Mac, and cd
to the directory that contains your application.
For our integration examples, we’re going to be showing the top-level CleanroomLogger
directory inside a Libraries
directory at the root level of of your application’s source.
You do not need to use this structure, although we’d recommend it, if only to make the following examples work for you without translation.
If you do not already have a Libraries
directory, create one:
mkdir Libraries
Next, cd
into Libraries
and follow the instructions below.
If you’re already using git for version control, we recommend adding CleanroomLogger to your project as a submodule. This will allow you to “lock” your codebase to specific versions of CleanroomLogger, making it easier to incorporate new versions on whatever schedule works best for you.
If you’re using some other form of version control of if you’re not using version control at all—shame on you!—then you’ll want to clone the CleanroomLogger repository. We suggest putting the CleanroomLogger clone somewhere within your application’s directory structure, so that it is included in whatever version control regimen you’re using.
Important: Skip this section if you plan to download CleanroomLogger using
git clone
.
From within the Libraries
directory, issue the following commands to download CleanroomLogger:
git submodule add https://github.com/emaloney/CleanroomLogger
git submodule update --init --recursive
Next, you’re ready to embed the CleanroomLogger.xcodeproj
project file in your Xcode project.
Important: Skip this section if you already performed the tasks outlined in “Downloading CleanroomLogger as a submodule” above.
From within the Libraries
directory, issue the following command to clone the CleanroomLogger repository:
git clone --recursive https://github.com/emaloney/CleanroomLogger
In the Terminal, the command open CleanroomLogger
to open the folder containing the CleanroomLogger source in the Finder. This will reveal the CleanroomLogger.xcodeproj
Xcode project and all files needed to build CleanroomLogger.framework
and its CleanroomASL.framework
dependency.
Then, open your application in Xcode, and drag CleanroomLogger.xcodeproj
into the Xcode project browser. This will embed CleanroomLogger in your project and allow you to add the targets built by CleanroomLogger to your project.
Before we can add CleanroomLogger.framework
to your app, we have to build it, so Xcode has more information about the framework.
Important: The next step will only work when the framework is built for a device-based run destination. That means that you must either select the “My Mac” or “iOS Device” run destination before building, or you must select an actual external device (an option that’s only available when such a device is connected to your development machine).
Once a device-based run destination has been selected, select the appropriate build scheme for the target platform: “CleanroomLogger-iOS”, “CleanroomLogger-OSX”, “CleanroomLogger-tvOS” or “CleanroomLogger-watchOS”.
Then, select Build (⌘B) from the Product menu.
Once the build is complete, open CleanroomLogger.xcodeproj
in the project navigator and find the “Products” group. Open that, and right-click on CleanroomLogger.framework
. Select Show in Finder. This will open the folder containing the framework binaries you just built.
You may see several files in this folder; the ones we’re concerned with are:
CleanroomLogger.framework
CleanroomASL.framework
If those files aren’t present, something went wrong with the build.
In Xcode, select the General tab in the build settings for your application target. Scroll to the bottom of the screen to reveal the section entitled Embedded Binaries (the second-to-last section).
Go back to Finder, select the files CleanroomLogger.framework
and CleanroomASL.framework
, and then drag them into the list area directly below Embedded Binaries.
If successful, you should see CleanroomLogger.framework
and CleanroomASL.framework
listed under both the Embedded Binaries and Linked Frameworks and Libraries sections.
Unfortunately, Xcode will reference the frameworks you just added in a way that will eventually cause you pain, particularly if multiple developers are sharing the same project file (in which case the pain will be felt almost immediately).
So, to make things sane again, you’ll need to make sure Xcode references CleanroomLogger.framework
and CleanroomASL.framework
using a “Relative to Build Products” location.
To do this, repeat the following steps for each framework:
- Locate the framework in the Xcode project browser
- Select the framework
- Ensure the Xcode project window’s Utilities pane is open
- Show the File Inspector in the Utilities pane
- Under the Identity and Type section, find the dropdown for the Location setting
- If the Location dropdown does not show “Relative to Build Products” as the setting, select “Relative to Build Products”
Once you’ve done this for each framework, you’re all done integrating CleanroomLogger!
Once CleanroomLogger has been successfully integrated, all you will need to do is add the following import
statement to any Swift source file where you want to use CleanroomLogger:
import CleanroomLogger
Want to learn more about CleanroomLogger? Check out the README.
Happy coding!