A simple app that makes use of physical web beacons to make a convenient way of displaying a staff in/out board in an office without any user interaction. Users wear or carry a small coin-battery powered bluetooth low energy beacon, which emits a URL once a second which can be detected at a range of around 10 metres. The URL is the address of a semantic web representation of that person, eg:
HTTP/1.1 200 OK
Etag: "26-bf3632e463ecd614b8aafcf1df80c13f"
Date: Sun, 01 Mar 2015 11:51:42 GMT
Content-Type: application/json;charset=utf-8
Content-Length: 233
{
"_id":"12ac9c3c2e6d03f2e49eec1be25332ce",
"_rev":"26-bf3632e463ecd614b8aafcf1df80c13f",
"type":"person",
"displayName":"Andrew",
"userName":"andrew.betts",
"lastSeenLocation":"Andrew's Macbook",
"lastSeenDate":"2015-03-01T11:35:05.357Z"
}
Fixed sensors mounted in the physical environment detect the beacons, and make an HTTP GET request to the advertised URL, fetching the current revision of the person's data. The sensor then updates it with the sensor's location, and makes an HTTP PUT request to update the resource.
Companies with multiple locations and staff who are mobile often find it hard to know where individuals are, and whether it's reasonable to bother them: they could be working at another office, or could be on holiday. With beacons, it's possible to easily know when a user is at work and where they are.
Conventional means of manually updating status are prone to error because users tend to forget to do it, security data from keycard swipes is often hard to extract from proprietary systems, tracking based on DHCP allocation is easily misled by VPNs, and tracking based on smartphone GPS has many privacy issues. Beacons seem to offer an open standards solution that's also possible to operate securely such that the organisation cannot track the user when outside the office, and the signal broadcast by the beacon does not identify the user in a way that's useful to any third party.
I purchased some pre-made beacons from Blesh, but if you're more of a hacker than me I'm told they're not hard to build. A guide and bill of materials is available on Openbeacon. The blesh beacons ship in plastic cases around the size of a ginger nut biscuit:
But a lot of that is actually empty space. I removed all the cases since it was then easier to get to the battery and configure button. To configure a beacon:
- Install the "Physical Web" app from the iOS or Android app store
- Make sure bluetooth is enabled on your phone
- Activate the beacon (in my case, that meant removing a plastic tab that was preventing the battery from making contact)
- Launch the app, and make sure it detects the beacon
- Press the cog to enter setup mode
- Press the button on the beacon
- Wait for the app to read the current URL from the beacon (I found this would sometimes hang. A restart of the beacon and the phone fixed it)
- Enter a new URL. The URL can only be at most approximately 17 characters, so use a URL shortening service like bit.ly.
Raspberry Pis make excellent sensors. I used a Raspberry Pi 1 Model B+, with a Bluetooth 4 USB dongle like this one, and a MicroSD card preloaded with NOOBS. Here's how I set it up:
Connect a USB keyboard, ethernet cable, HDMI monitor, and NOOBS SD card to the Pi. Now plug it in (you can either use any Micro USB power adapter, or tether the Pi to any convenient USB port using a Micro-USB to USB cable). The Pi will boot to the NOOBS screen. Choose Raspbian, press i to install it, and wait ages.
When it's done, you should get a screen with other install options, I skipped all those and pressed finish. Then you get a shell prompt. Type ifconfig
to get the Pi's IP address. Now connect to that IP from your computer with the username pi
and password raspberry
.
Assuming that works, you can now detach the keyboard and monitor from the Pi (if you're feeling confident), and continue setting it up over an SSH session.
sudo apt-get update
sudo apt-get install libdbus-1-dev libdbus-glib-1-dev libglib2.0-dev libical-dev libreadline-dev libudev-dev libusb-dev libbluetooth-dev make
mkdir bluez
cd bluez
sudo wget www.kernel.org/pub/linux/bluetooth/bluez-5.28.tar.xz
xz -d bluez-5.28.tar.xz
tar xvf bluez-5.28.tar
cd bluez-5.28/
./configure --disable-systemd
make
sudo make install
sudo shutdown -h now
Now insert the USB dongly thing, power cycle and log back in.
cd /usr/local/bin
sudo ln -s ~/bluez/bluez-5.28/attrib/gatttool
cd ~/bluez/bluez-5.28
tools/hciconfig
sudo tools/hciconfig hci0 up
tools/hciconfig
sudo hcitool lescan
Should find the beacon
cd ~
wget http://node-arm.herokuapp.com/node_latest_armhf.deb
sudo dpkg -i node_latest_armhf.deb
cd ~
git clone git://github.com/triblondon/uribeacon-checkin.git
cd uribeacon-checkin
npm install
LOCATION_DESC="FT Labs office" node index.js
On the Pi you may need to run using sudo
to give Node access to the bluetooth hardware.
The server can be anything that serves JSON and permits clients to perform GET and PUT operations, but a CouchDB database works pretty well. I used Cloudant. Set up a new database, allow public read/write (this ought to be authenticated, but it's only an in/out board). Create a record for each person with at least lastSeenLocation
and lastSeenDate
on each record. You can add whatever other fields you want, the sensor doesn't care and will not overwrite them.
Cloudant then provides an API URL for the document, which will look something like:
https:// {accountName} .cloudant.com/ {databaseName} / {docID}
If you're using Couch, you can report the current location of all people using a view. This view code will return a list of people seen within the last day and their locations:
function(doc) {
if (doc.type === "person" && doc.displayName && Date.parse(doc.lastSeenDate) > (Date.now() - (60*60*24*1000))) {
emit(doc.displayName, doc.lastSeenLocation);
}
}
Future development ideas:
- Authentication (or use URL that's only accessible within private network)
- UI to display the view