This application demonstrates how to build an iOS app that uses a Tessel as an iBeacon, and then posts to an HTTP server each time your iDevice enters the range of a Tessel Beacon.
- iBeacons: A Quick Overview
- Tessel iBeacon Demo App
- Features
- Prerequisites
- Getting Started
- Running Tests
- Application Design
- Philosophies
- Important Objects
- 3rd Party Dependencies
- Backend
- Debugging + Helpful Resources
- App in Action
- Video
- Screenshots
iBeacon is a relatively new technology (albeit, slightly older than Tessel) that was introduced by Apple in iOS 7. It takes advantage of iOS devices' built-in BLE hardware by allowing iDevices to determine their relative proximity to other BLE devices. Those "other" BLE devices are the iBeacons!
Why use an iBeacon? Apple's example use case is a retail store. Companies might want iBeacons within their stores in order to notify customers when there are relevant deals nearby, or to track the foot traffic within certain departments.
Any BLE device can act as an iBeacon - including your Tessel! (To be an "official" iBeacon, you must meet Apple's licensing requirements. This app does not meet Apple's licensing requirements; it's intended to be a tool for developers to learn and to prototype.) Anyway, to function as an iBeacon, a BLE device must advertise:
- 16 byte UUID value (required)
- 2 byte "major" value (optional)
- 2 byte "minor" value (optional)
Apple has a fantastic guide to getting started with iBeacons. It talks about the technology, its limitations and advantages, and the Apple API's around iBeacons. That guide is here.
-
Tessel Beacon Monitoring : User can decide whether or not monitor for boundary changes when entering/exiting the region of the Tessel. If the user decides to monitor for boundary changes, they'll get notified each time they enter the region of a Tessel, regardless of whether the app is currently running in the foreground. As of right now, notifications are on entry only - not on exiting a region. Also, the app will ping http://tessel-beacon-server.herokuapp.com each time it enters the region of a Tessel.
-
Tessel Beacon Ranging : When the app is in the foreground, user can range the Tessel Beacon. While ranging is enabled, the app will display the proximity to the Tessel Beacon in realtime. scans for Tessel beacon.
- You must be registered as an Apple Developer. This is a requirement for running the app on a physical device. (You don't need to be an Apple developer to run the app in the simulator; however, bluetooth does not work in the simulator)
- You should have the Tessel BLE module. The application will guide you through setting it up correctly.
- You should have an iOS device with BLE capabilities. Any device listed here http://en.wikipedia.org/wiki/List_of_iOS_devices#Features as having "Bluetooth 4.0" will support BLE.
- Fork/clone this repo.
- If this is your first time building an iOS app, you'll probably have to install Cocoapods. Cocoapods is like NPM for iOS apps. To install:
sudo gem install cocoapods
. Yes, Cocoapods is a Ruby gem. If you don't have Ruby installed, you'll have to do that first. Dependencies are just turtles all the way down. cd
to your local clone of the repo, and runpod install
.- To run the app, build the TesselBeaconDemo target to an iOS device. As mentioned above, you've got to be a registered Apple Developer to do this.
To run the specs (this app is well-tested), build the Specs target.
A lot of demo iOS apps dump all the logic into view controllers. Look! It's only 1 file. I don't love this - it makes demo apps a very different from "real" applications, and it makes them harder to test.
I believe demo apps are just lightweight versions of production apps, and they should be written up to production standards. If anything, demo code should be better than production code. Demo apps are frequently used by people trying to learn iOS development, and it doesn't do anyone any good if they're led to believe that dumping everything into a view controller is a good idea. It's never a good idea.
That being said, this app uses view controllers for displaying text and handling gestures. Everything beyond that is handled by other objects.
This application has lots of tests, and those tests outline the behavior of each object in great detail. I believe that tests should serve as documentation. However, that's not really what people expect from a Readme. So, here's a quick 1-line description of each object in the app
- MainViewController: Allows user to toggle monitoring and ranging on/off. Gets notified of events from the TesselBeaconManager, and logs those events accordingly.
- RegistrationViewController: Explains what it means to "register" a Tessel, and allows the user to opt-in to doing so.
- TesselInformationViewController: Shows the user their registered Tessel UUID, and allows them to email a code snippet to themself, or copy/paste the Tessel UUID to the clipboard.
- TesselBeaconManager: Wraps a
CLLocationManager
to provide Tessel-specific functionality - i.e, it only handles beacon related location events - not GPS-based location events - TesselRegistrationRepository: Registers the Tessel with the remote server, and maintains knowledge of the registered Tessel UUID.
- TesselCheckinRepository: Posts checkins to the remote server. Could easily be extended to also fetch old checkins from the server.
- Cedar: Cedar as an iOS testing framework; at time of writing, this app has 76 tests.
- KSPromise: KSPromise is a lightweight promise library for Objective-C. It's based on the JavaScript Promises/A spec (http://wiki.commonjs.org/wiki/Promises/A). It's not (yet) a popular pattern in iOS development, but I like to use it - it makes writing beahvioral tests much easier.
- AFNetworking: AFNetworking is a highly popular networking library for iOS. It's almost definitely overkill for this application - this app only makes 2 network requests. However, this application is open-source. If you decide to fork it and use it as a seed for your own application, you'll probably want to make more network requests. AFNetworking is great for that.
This application uses a Rails backend, available on Github. There's a public instance on Heroku It has no security features, but is good enough for testing purposes. If you'd like to use it for anything beyond just building the app, I highly, highly encourage you to boot up your own instance on Heroku.
- If you want to confirm whether the iPhone has pinged the server, you can visit the /api/checkins endpoint. Is your Tessel ID there?
- Apple docs about ranging and monitoring with iBeacons
- Blog post about some of the challenges in using iBeacons, including the permissions that are necessary
Here are some screenshots that highlight features of the demo app.
Screenshot 1: The first time you use the app, you'll register your Tessel Screenshot 2: After registering your Tessel, you'll see the UUID it needs to broadcast Screenshot 3: You can email yourself the code that the Tessel should run! Screenshot 4: The first time you turn on any of the iBeacon features, you'll see a location permission pop-up. Screenshot 5: While the app is in the foreground, you can monitor your proximity to the Tessel