My first CocoaPod described in this post sparked a discussion with the CocoaPods team. There are a few things I didn’t like about the project organization.
- The name of the workspace was “Demo”. I would expect the project to have the same name as the pod being written.
- The demo project contained the Podfile, requiring developers to cd into Demo, then run pod install. I would expect the Podfile to live in the root of the project.
- The demo project contained tests for the pod. I would expect the demo to be a demo and to have a test project that unit tests the pod being written.
All these issues can be addressed. I’ve reorganized two of my pods, ARASCIIImageSwizzle and ARTiledImageView and fixed all the issues above.
Folder Structure
Fixing the folder structure and naming is fairly straightforward. You can move the contents of the Demo folder one level up. This brings the Podfile to the root. The Podfile can also explicitly name the workspace.
workspace 'ARASCIISwizzle'
pod 'ARASCIISwizzle', :path => 'ARASCIISwizzle.podspec'
xcodeproj 'Demo.xcodeproj'
You’ll have to update .travis.yml to build a different workspace and to reference the development pod in the same folder.
language: objective-c
before_install:
- brew update
- brew uninstall xctool
- brew install xctool
- export LANG=en_US.UTF-8
- gem i cocoapods --no-ri --no-rdoc
- pod install
xcode_workspace: ARASCIISwizzle.xcworkspace
xcode_scheme: Demo
xcode_sdk: iphonesimulator
You can see this commit in ARTiledImageView that accomplishes the same.
Integration Tests
The tests that are part of the Demo project are really integration tests. I’ve renamed those consequently with a bulk replace of file names and within the source code. You can see it in this commit in ARTiledImageView.
Unit Tests
Let’s build a clean unit test project for the pod, independent of the Demo project.
Create a new empty project in XCode called “Tests” or, if you prefer, “UnitTests”.
Add a target.
Choose Cocoa Touch Unit Testing bundle.
Open Manage Schemes and make sure the scheme is Shared. Then edit the Test project’s scheme and check the Run box under the Build action.
The Podfile must now reference “Demo”, “Tests” and the “IntegrationTests” project. The syntax for that is a bit backwards, discussed in https://github.com/CocoaPods/CocoaPods/issues/1922.
workspace 'ARASCIISwizzle'
pod 'ARASCIISwizzle', :path => 'ARASCIISwizzle.podspec'
xcodeproj 'Demo.xcodeproj'
target 'Demo' do
pod 'FLKAutoLayout', '~> 0.1.1'
xcodeproj 'Demo.xcodeproj'
end
target 'IntegrationTests' do
pod 'Specta', '~> 0.2.1'
pod 'Expecta', '~> 0.2.3'
pod 'FBSnapshotTestCase', :head
pod 'EXPMatchers+FBSnapshotTest', :head
xcodeproj 'Demo.xcodeproj'
end
target 'Tests' do
pod 'Specta', '~> 0.2.1'
pod 'Expecta', '~> 0.2.3'
pod 'FBSnapshotTestCase', :head
pod 'EXPMatchers+FBSnapshotTest', :head
pod 'OCMock', '~> 2.2.3'
xcodeproj 'Tests.xcodeproj'
end
Travis will build “Tests” and “IntegrationTests” in a matrix.
xcode_scheme:
- Demo
- Tests
You can see the results of the above in this commit in ARASCIISwizzle.
Conclusion
This is a lot closer to the Rubygems-style organization where everything is accessible from the top and tests live at the same level as pod classes.