HipChat is Atlassian’s alternative to Slack and its solution to team collaboration chats. Atlassian Connect offers developer tools to bootstrap applications, connect to Atlassian’s cloud products with easy and in combination with HipChat’s REST APIs allows us to write integrations for such a chat server in no time.

In the following tutorial I’d like to show how to write an integration within a few steps using Atlassian Connect, Node.js and Express and how to connect the integration to a HipChat server.

Finally on the one hand I’m going to explain how to speed up local development with ngrok, an in-memory database and nodemon for automatic application restarts and on the other hand I’m going to demonstrate how to configure the application for production, running with a Redis key-value store on Heroku.

step5 search result display in hipchat 1024x747
Figure 1. Blog article search integration interacting with a HipChat room

Prerequisites

The only thing we need is having Node.js and the NPM package manager installed (you might want to install ngrok, too but it is optional).

$ node --version
v0.10.22
$ npm --version
1.3.14

What we’re going to build

We’re going to build an integration for HipChat that allows us to search for blog articles from this blog, www.hascode.com by entering a command in the following syntax into a HipChat room where the integration is enabled: /hascode tag.

Afterwards, a list of links to matching blog articles should be displayed in the room. If there are more then 10 search hits, a link to display all search results should be displayed rather than polluting the chat room with a long list of articles.

The screenshot above should give us a good impression of the interaction of the integration within HipChat.

The following, hopefully not too confusing mind-map demonstrates the technology stack used to build the integration and to communicate with the chat server.

Technologies used for building a HipChat Integration.

hipchat integration building mindmap 1024x560

Integrations are not deployed within the HipChat server but must be run as a separate server instance. HipChat is only configured to call this remote instance on specific events, called web-hooks.

That’s why our approach of building an integration is to create a server application with Nodejs and Express here.

Communication Flow Explained

So what is happening when we install our integration to a HipChat room or run our blog article search?

The following simplified sequence diagram tries to explain the process flow here:

  • When the integration is installed into a room, HipChat reads the plug-in descriptor file and scans for web-hooks, permissions and install hooks.

  • HipChat sends a request to our remote application with a specific JSON structure that contains important information like the OAuth2  credentials.

  • If this happens, our remote application stores this information in a database and responds with a success message and status code

  • HipChat stores the plug-in meta-data, web-hooks etc and confirms the successful installation to the requesting user

  • When now in a chat-room the blog article search is triggered by a user, HipChat forwards the event to our remote application

  • The remote application now sends a search request to the blog, parses the result, generates a view-friendly message and sends the message to the chat-room using the REST-API and OAuth2

hipchat integration communication seq diagram 1024x824
Figure 2. Communication flow as a sequence diagram

Atlassian Connect Express

Atlassian Connect Express is a toolkit that eases the process of creating Atlassian Connect addons for us providing utility and helper libraries to bootstrap new projects, configure Express.js based web applications and connect to an existing HipChat instance using Oauth2 and a REST client.

Installation with NPM

The atlassian-connect command line tool is installed using NPM:

npm i -g atlas-connect

Creating a new Connect Project

We’re now ready to create a new connect project using the command line interface:

$ atlas-connect new -t hipchat hipchat-integration-tutorial
Downloading template [===================] 100% 0.0s
  hipchat-integration-tutorial/
  hipchat-integration-tutorial/.gitignore
  hipchat-integration-tutorial/Procfile
  [..]

hipchat-integration-tutorial has been created. `cd hipchat-integration-tutorial` then run `npm install` to install all dependencies.

For information on what to do next, visit:

  http://j.mp/get-started-with-atlassian-connect-express.

Afterwards we have a full configured project and may take a first look at our application. We may remove the handlebar templates (*.hbs), addon.css and addon.js as we won’t need them in the following tutorial.

Finally our initial directory structure should look similar to this:

.
├── app.js
├── atlassian-connect.json
├── config.json
├── lib
│   └── hipchat.js
├── package.json
├── Procfile
├── public
│   ├── css
│   │   └── addon.css
│   └── js
│       └── addon.js
├── README.md
├── routes
│   └── index.js
└── views
    ├── config.hbs
    └── layout.hbs

Dependency Management with NPM

The dependencies our application needs are specified in the package.json file in the root directory – this is what ours looks like:

{
  "name": "com.hascode.hc.blogsearch",
  "version": "1.0.0",
  "private": true,
  "scripts": {
    "start": "node app.js"
  },
  "dependencies": {
    "express": "^3.4.8",
    "express-hbs": "*",
    "static-expiry": ">=0.0.10",
    "request": "^2.33.0",
    "rsvp": "^3.0.3",
    "atlassian-connect-express": "^1.0.0-beta2",
    "atlassian-connect-express-redis": "^0.1.3",
    "atlassian-connect-express-hipchat": "^0.2.0",
    "lodash": "*"
  }
}

To install the dependencies in our project directory, we’re running the following command:

$ npm install

When npm has finished downloading the internet, we can see that our dependencies have been saved in the node_modules directory.

Writing our Integration

Now we’re entering the fun zone: implementing our application..

Project Configuration

This is our config.json and it contains our configuration for the two stages development and production (explained in detail in the following chapter).

{
  "development": {
    [..]
  },
  "production": {
    [..]
  }
}

Development Configuration, ngrok and In-memory Database

This is part of our development stage configuration, the most important settings here are:

  • we’re setting a ngrok tunnel URL as base URL (we don’t really need to because we’re going to pass this information from the command line)

  • the port for our application is set to 3000

  • for persistence we’re using jugglingdb with an in-memory database (we might also use another database e.g. sqlite here, we just needed to set the type to “sqlite” and add the sqlite dependency to our package.json)

  • searchHost, searchBasePath and maxResultsDisplayed are settings that we’re going to use to control the behaviour of our search application

{
  "development": {
    "localBaseUrl": "https://d6f97a22.ngrok.io",
    "usePublicKey": false,
    "watch": false,
    "port": 3000,
    "maxTokenAge": 86400,
    "store": {
      "adapter": "jugglingdb",
      "type": "memory"
    },
    "searchHost": "www.hascode.com",
    "searchBasePath": "/wp-content/byTag.php?tag=",
    "maxResultsDisplayed": 10
  }
  [..]
}

Production Configuration for the Cloud and Redis

The production configuration differs from the development configuration:

  • the port is specified by a placeholder variable (later set by Cloud provider)

  • the localBaseUrl points to the Heroku app URL

  • for persistence we’re using a scalable Redis database now, the connection URL again is set using a placeholder (see chapter Heroku setup)

{
  [..]
  "production": {
    "usePublicKey": false,
    "port": "$PORT",
    "localBaseUrl": "https://hascode-tagsearch.herokuapp.com",
    "store": {
      "adapter": "redis",
      "url": "$REDIS_URL"
    },
    "whitelist": [
      "*.hipchat.com"
    ],
    "searchHost": "www.hascode.com",
    "searchBasePath": "/wp-content/byTag.php?tag=",
    "maxResultsDisplayed": 10
  }
}

In addition to this, we’re adding the following Procfile to tell our cloud provider how to start our application

web: node app.js

Plugin Descriptor

The plug-in descriptor is used to install the integration into the HipChat instance and to display requested permissions and web-hooks to establish interaction between HipChat and our application. First of all we’re adding some description about our plug-in and its vendor by specifying name, description, vendor etc.

More interesting is the capabilities section as here in the node

  • hipchatApiConsumer we’re requesting concrete permissions that are granted to our integration upon install – for our plug-in all we need here is the permission to send notifications as we want to post the result of our blog search into the room.

  • installable we’re specifying a callback URL that is called when someone installs our plug-in or removes it from a HipChat instance. Upon install we’re receiving the OAuth2 credentials and the clientId that we need to store to enable further communication between our application and the remote HipChat instance.

  • webhook we’re specifying a callback URL (web-hook) that is called when a message is posted to a room that begins with “/hascode“.

{
  "name": "hasCode.com Blog Search",
  "description": "Search the hasCode.com Blog for articles for a given tag using /hascode tag",
  "key": "com.hascode.hc.blogsearch",
  "vendor": {
    "name": "Micha Kops",
    "url": "http://www.micha-kops.com"
  },
  "links": {
    "self": "{{localBaseUrl}}/atlassian-connect.json",
    "homepage": "{{localBaseUrl}}/atlassian-connect.json"
  },
  "capabilities": {
    "hipchatApiConsumer": {
      "scopes": [
        "send_notification"
      ]
    },
    "installable": {
      "callbackUrl": "{{localBaseUrl}}/installable"
    },
    "webhook": {
      "url": "{{localBaseUrl}}/message",
      "pattern": "^/hascode",
      "event": "room_message",
      "name": "Blog Search Command"
    }
  }
}

Generated Base Application

Just for the sake of completeness is this the generated app.js that bootstraps the web application, sets up stuff ..  and luckily for us, we don’t need to change a single line of code here.

// This is the entry point for your add-on, creating and configuring
// your add-on HTTP server

// [Express](http://expressjs.com/) is your friend -- it's the underlying
// web framework that `atlassian-connect-express` uses
var express = require('express');
// You need to load `atlassian-connect-express` to use her godly powers
var ac = require('atlassian-connect-express');
process.env.PWD = process.env.PWD || process.cwd(); // Fix expiry on Windows :(
// Static expiry middleware to help serve static resources efficiently
var expiry = require('static-expiry');
// We use [Handlebars](http://handlebarsjs.com/) as our view engine
// via [express-hbs](https://npmjs.org/package/express-hbs)
var hbs = require('express-hbs');
// We also need a few stock Node modules
var http = require('http');
var path = require('path');
var os = require('os');

// Let's use Redis to store our data
ac.store.register('redis', require('atlassian-connect-express-redis'));

// Anything in ./public is served up as static content
var staticDir = path.join(__dirname, 'public');
// Anything in ./views are HBS templates
var viewsDir = __dirname + '/views';
// Your routes live here; this is the C in MVC
var routes = require('./routes');
// Bootstrap Express
var app = express();
// Bootstrap the `atlassian-connect-express` library
var addon = ac(app);
// You can set this in `config.js`
var port = addon.config.port();
// Declares the environment to use in `config.js`
var devEnv = app.get('env') == 'development';

// Load the HipChat AC compat layer
var hipchat = require('atlassian-connect-express-hipchat')(addon, app);

// The following settings applies to all environments
app.set('port', port);

// Configure the Handlebars view engine
app.engine('hbs', hbs.express3({partialsDir: viewsDir}));
app.set('view engine', 'hbs');
app.set('views', viewsDir);

// Declare any Express [middleware](http://expressjs.com/api.html#middleware) you'd like to use here
app.use(express.favicon());
// Log requests, using an appropriate formatter by env
app.use(express.logger(devEnv ? 'dev' : 'default'));
// Include stock request parsers
app.use(express.urlencoded());
app.use(express.json());
app.use(express.cookieParser());
// Gzip responses when appropriate
app.use(express.compress());
// Enable the ACE global middleware (populates res.locals with add-on related stuff)
app.use(addon.middleware());
// Enable static resource fingerprinting for far future expires caching in production
app.use(expiry(app, {dir: staticDir, debug: devEnv}));
// Add an hbs helper to fingerprint static resource urls
hbs.registerHelper('furl', function (url) {
    return app.locals.furl(url);
});
// Mount the add-on's routes
app.use(app.router);
// Mount the static resource dir
app.use(express.static(staticDir));

// Show nicer errors when in dev mode
if (devEnv) app.use(express.errorHandler());

// Wire up your routes using the express and `atlassian-connect-express` objects
routes(app, addon);

// Boot the damn thing
http.createServer(app).listen(port, function () {
    console.log()
    console.log('Add-on server running at ' + (addon.config.localBaseUrl() || ('http://' + (os.hostname()) + ':' + port)));
    // Enables auto registration/de-registration of add-ons into a host in dev mode
    if (devEnv) addon.register();
});

Express Routes

This is our index.js that contains the core functionality of our application and the routes mapped to our web application.

The imported hipchat client layer eases the communication with the remote HipChat instance and its RESTful webservices (and its authentication mechanism).

We’re specifying the following routes:

  • /message receives the blog search events triggered in HipChat when the /hascode command is used

  • /installable receives install and uninstall events (POST or DELETE)

  • other requests are redirected to the plugin descriptor file

When we receive a search event, we’re sending a new request to the blog search, we’re parsing the JSON response and generate a HTML list for display in the HipChat room.

If the amount of search hits exceeds the given limit of 10, we’re displaying the first 10 search results and additionally display a link to view all search results on the blog.

var http = require('http');

module.exports = function (app, addon) {
    var hipchat = require('../lib/hipchat')(addon);
    var options = {
        host: addon.config.searchHost(),
        path: '/',
        method: 'GET',
        headers: {
            'Content-Type': 'application/json'
        }
    };

    // Root route. This route will serve the `addon.json` unless a homepage URL is
    // specified in `addon.json`.
    app.get('/',
        function (req, res) {
            // Use content-type negotiation to choose the best way to respond
            res.format({
                // If the request content-type is text-html, it will decide which to serve up
                'text/html': function () {
                    res.redirect(addon.descriptor.links.homepage);
                },
                // This logic is here to make sure that the `addon.json` is always
                // served up when requested by the host
                'application/json': function () {
                    res.redirect('/atlassian-connect.json');
                }
            });
        }
    );

    // This is an example route to handle an incoming webhook
    app.post('/message',
        addon.authenticate(),
        function (req, res) {
            var term = req.body.item.message.message.split(' ')[1];
            console.log('searching for given term: ' + term);
            var message = "<a href=\"https://www.hascode.com/\">hasCode.com</a> blog articles for given term: <b>"" + term + ""</b>";
            options.path = addon.config.searchBasePath() + term;
            var clientRequest = http.request(options, function (clientResponse) {
                var output = '';
                clientResponse.setEncoding('utf8');

                clientResponse.on('data', function (chunk) {
                    output += chunk;
                });

                clientResponse.on('end', function () {
                    var hits = JSON.parse(output);
                    var hitsNum = hits.length;
                    console.log(hitsNum + ' results for term ' + term + ' found, max-results set to: ' + addon.config.maxResultsDisplayed());
                    message += ' (' + hitsNum + ' hit/s) <ul>';
                    message += hits.slice(0, addon.config.maxResultsDisplayed()).reduce(function (prev, cur) {
                        return createMessage(prev) + createMessage(cur);
                    }, "");
                    message += '</ul>';
                    if (hits.length > addon.config.maxResultsDisplayed()) {
                        message += '<b><a href="https://www.hascode.com/tag/' + term + '">Show all ' + hitsNum + ' results for "' + term + '"</a></b>'
                    }

                    hipchat.sendMessage(req.clientInfo, req.context.item.room.id, message)
                        .then(function (data) {
                            res.send(200);
                        });
                });
            });
            clientRequest.end();
        }
    );

    // Notify the room that the add-on was installed
    addon.on('installed', function (clientKey, clientInfo, req) {
        hipchat.sendMessage(clientInfo, req.body.roomId, 'The ' + addon.descriptor.name + ' add-on has been installed in this room');
    });

    // Clean up clients when uninstalled
    addon.on('uninstalled', function (id) {
        addon.settings.client.keys(id + ':*', function (err, rep) {
            rep.forEach(function (k) {
                addon.logger.info('Removing key:', k);
                addon.settings.client.del(k);
            });
        });
    });

    function createMessage(hit) {
        if (typeof hit === 'string') {
            return hit;
        }
        return '<li><a href=\"' + hit.url + '\">' + hit.title + '</a></li>';
    }
};

Running the Server Application, Tunnelling with ngrok

Ngrok is a nice tool that allows us to export our application running on localhost to the internet using a SSH tunnel.

The only downside of the free version is, that we’re getting a random sub-domain assigned.

In the following example, we’re tunnelling our application running on port 3000 to the outside world:

$ ngrok http 3000
ngrok by @inconshreveable                                                                                                                                                                                                                                      (Ctrl+C to quit)

Tunnel Status                 online
Version                       2.0.19/2.0.19
Web Interface                 http://127.0.0.1:4040
Forwarding                    http://db6b58f8.ngrok.io -> localhost:3000
Forwarding                    https://db6b58f8.ngrok.io -> localhost:3000

Connections                   ttl     opn     rt1     rt5     p50     p90
                              0       0       0.00    0.00    0.00    0.00

When now starting our application, we might want to override its base-URL by adding the following environment variable set to our ngrok sub-domain:

AC_LOCAL_BASE_URL=https://<subdomain>.ngrok.com node app.js

Automatic Application Restart with Nodemon

During development, we don’t want to reboot our application every-time we’re changing the application logic.

Luckily for us, there is nodemon to help us here: nodemon scans the project directory for file changes and reboots the application if changes were detected.

Nodemon is installed in no time using npm:

sudo npm install -g nodemon

Now when running our application with nodemon it is restarted automatically when a file has been changed e.g.:

$ AC_LOCAL_BASE_URL=https://c569b888.ngrok.io nodemon app.js
3 Aug 19:29:12 - [nodemon] v1.4.0
3 Aug 19:29:12 - [nodemon] to restart at any time, enter `rs`
3 Aug 19:29:12 - [nodemon] watching: *.*
3 Aug 19:29:12 - [nodemon] starting `node app.js`

Add-on server running at https://c569b888.ngrok.io
Auto registration not available with HipChat add-ons
Initialized memory storage adapter

3 Aug 19:30:40 - [nodemon] restarting due to changes...
3 Aug 19:30:40 - [nodemon] starting `node app.js`

Integration Installation in HipChat

Finally our application is ready for integration into a concrete HipChat instance. This is done within a few steps that I’d like to demonstrate here.

Adding the Integration via Plugin Descriptor URL

We may add an integration for a room having sufficient permissions by clicking “Integrations” in the operation menu.

step1 add integration 1024x747
Figure 3. Adding a new integration - step 1

We’re now entering the URL to the deployment descriptor exported by our application (I have used address of my Heroku app – more about Heroku configuration later..).

step2 add by deployment descriptor url 1024x747
Figure 4. Entering the plugin deployment descriptor’s url

Confirm Plugin Installation

HipChat loads the descriptor and displays the following confirmation with additional information about the plug-in vendor and required permissions.

As we can see, the only permission we’re requesting is the permission to send room notifications.

step3 display integration details 1024x747
Figure 5. Display integration details

Finally HipChat displays an overview of the plug-in and the affected chat-rooms:

step4 integration installation overview 1024x747
Figure 6. Integration overview

Our Integration in Action

First of all we should be able to view the response from the on-install hook that confirms that the plug-in was successfully installed to this room.

And as we can see in the following screenshot, we’re now able to search for blog articles by typing /hascode keyword #yeah :)

step5 search result display in hipchat 1024x747
Figure 7. Blog article search results in a HipChat room

Heroku Configuration and Deployment

Our application is not bound for deployment and provision by a specific provider but to give a concrete example, I’d like to demonstrate the configuration and deployment for Heroku.

As a preparation all we need is a Heroku account and a free slot for a Nodejs application ;)

Deployment using Git

We just need to add our application’s Heroku git repository and push our changes and our application is launched automatically:

$ git push heroku master
[..]
remote: -----> Node.js app detected
remote: -----> Creating runtime environment
[..]
remote: -----> Installing binaries
remote:        engines.node (package.json):  unspecified
remote:        engines.npm (package.json):   unspecified (use default)
remote:
remote:        Resolving node version (latest stable) via semver.io...
remote:        Downloading and installing node 0.12.7...
remote:        Using default npm version: 2.11.3
remote:
remote: -----> Restoring cache
remote:        Loading 1 from cacheDirectories (default):
remote:        - node_modules
remote:
remote: -----> Building dependencies
remote:        Pruning any extraneous modules
remote:        Installing node modules (package.json)
remote:
remote: -----> Caching build
remote:        Clearing previous node cache
remote:        Saving 1 cacheDirectories (default):
remote:        - node_modules
remote:
remote: -----> Build succeeded!
remote:        ├── atlassian-connect-express@1.0.4
remote:        ├── atlassian-connect-express-hipchat@0.2.3
remote:        ├── atlassian-connect-express-redis@0.1.3
remote:        ├── express@3.21.1
remote:        ├── express-hbs@0.8.4
remote:        ├── lodash@3.10.0
remote:        ├── request@2.60.0
remote:        ├── rsvp@3.0.20
remote:        └── static-expiry@0.0.11
remote:
remote: -----> Discovering process types
remote:        Procfile declares types -> web
remote:
remote: -----> Compressing... done, 22.1MB
remote: -----> Launching... done, v7
remote:        https://nameofmyapp.herokuapp.com/ deployed to Heroku
remote:
remote: Verifying deploy.... done.
To https://git.heroku.com/nameofmyapp.git
   f7ae602..439be85  master -> master

Adding the Redis Plugin

First, we’re adding the Heroku Redis Plugin to our application – either by using the administration panel in the browser or in the command line using the following command:

heroku addons:create heroku-redis:blogsearch-clients

We may verify that it is installed using the cli:

$ heroku addons | grep REDIS
REDIS       thinking-busily-3739  hascode-tagsearch

In addition we might want to verify that the connection URL to the Redis database is present in our environment:

$ heroku config|grep REDIS
REDIS_URL:            redis://h:1111111111111111111111111111.compute-1.amazonaws.com:10859

Setting the Production Stage

Again we have the choice to set variables on the Heroku dashboard or using the command-line interface. We’re setting the stage NODE_ENV to production using the following command:

$ heroku config:set NODE_ENV=production
Setting config vars and restarting hascode-tagsearch... done, v6
NODE_ENV: production

The following screenshot demonstrates how the config variables are display in the Heroku app settings:

heroku configuration variables overview 1024x647
Figure 8. Configuration Variables Overview in Heroku

Viewing Application Logs

We’re able to view our application logs using the following command (snapshot taken during the installation of the integration into a HipChat room):

$ heroku logs --tail
2015-07-23T20:17:33.597314+00:00 heroku[web.1]: Starting process with command `node index.js`
2015-07-23T20:17:34.785515+00:00 app[web.1]: Detected 512 MB available memory, 512 MB limit per process (WEB_MEMORY)
2015-07-23T20:17:34.785541+00:00 app[web.1]: Recommending WEB_CONCURRENCY=1
2015-07-23T20:17:35.083154+00:00 app[web.1]: Node app is running on port 37451
[..]
2015-07-30T08:40:10.210809+00:00 heroku[router]: at=info method=GET path="/atlassian-connect.json" host=hascode-tagsearch.herokuapp.com request_id=a0bebd4f-0965-4d0d-b987-8bc496bff28e fwd="54.197.167.216" dyno=web.1 connect=1ms service=5ms status=200 bytes=907
2015-07-30T08:40:10.208892+00:00 app[web.1]: ::ffff:10.181.95.28 - - [30/Jul/2015:08:40:10 +0000] "GET /atlassian-connect.json HTTP/1.1" 200 680 "-" "-"
2015-07-30T08:40:10.602429+00:00 heroku[router]: at=info method=POST path="/installable" host=hascode-tagsearch.herokuapp.com request_id=bba03b73-4cd6-4129-85b1-6969446159e7 fwd="54.197.167.216" dyno=web.1 connect=2ms service=240ms status=204 bytes=142
2015-07-30T08:40:10.600201+00:00 app[web.1]: Saved tenant details for 37052e9e-b215-4000-a5b6-b471a0008f3e to database
2015-07-30T08:40:10.600208+00:00 app[web.1]: { clientKey: 'aaaaaaaa-bbbb-cccc-dddd-eeeeeeee',
2015-07-30T08:40:10.600209+00:00 app[web.1]:   oauthSecret: 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx',
2015-07-30T08:40:10.600211+00:00 app[web.1]:   capabilitiesUrl: 'https://api.hipchat.com/v2/capabilities',
2015-07-30T08:40:10.600213+00:00 app[web.1]:   capabilitiesDoc:
2015-07-30T08:40:10.600214+00:00 app[web.1]:    { capabilities: { hipchatApiProvider: [Object], oauth2Provider: [Object] },
2015-07-30T08:40:10.600216+00:00 app[web.1]:      description: 'Group chat and IM built for teams',
2015-07-30T08:40:10.600217+00:00 app[web.1]:      key: 'hipchat',
2015-07-30T08:40:10.600219+00:00 app[web.1]:      links:
2015-07-30T08:40:10.600221+00:00 app[web.1]:       { api: 'https://api.hipchat.com/v2',
2015-07-30T08:40:10.600222+00:00 app[web.1]:         homepage: 'https://www.hipchat.com',
2015-07-30T08:40:10.600224+00:00 app[web.1]:         self: 'https://api.hipchat.com/v2/capabilities' },
2015-07-30T08:40:10.600225+00:00 app[web.1]:      name: 'HipChat',
2015-07-30T08:40:10.600226+00:00 app[web.1]:      vendor: { name: 'Atlassian', url: 'http://atlassian.com' } },
2015-07-30T08:40:10.600228+00:00 app[web.1]:   roomId: 123456,
2015-07-30T08:40:10.600229+00:00 app[web.1]:   groupId: 654321,
2015-07-30T08:40:10.600231+00:00 app[web.1]:   groupName: 'hasCode.com' }
2015-07-30T08:40:10.601568+00:00 app[web.1]: ::ffff:10.171.92.168 - - [30/Jul/2015:08:40:10 +0000] "POST /installable HTTP/1.1" 204 - "-" "-"

Another nice add-on to aggregate and visualize our logs is the Heroku Papertrails plugin – it’s limited but for free!

Tutorial Sources

Please feel free to download the tutorial sources from my GitHub repository, fork it there or clone it using Git:

git clone https://github.com/hascode/hipchat-integration-tutorial.git