Skip to content

Latest commit

 

History

History
297 lines (216 loc) · 9.87 KB

README.md

File metadata and controls

297 lines (216 loc) · 9.87 KB

🖼 Tela

Node.js CI

Tela is a small framework for building Intercom Canvas Kit applications in node. Write your apps as simple classes, run a server to receive calls to your app with one line, and get your canvasses type-checked with TypeScript.

⬇️ Install

Simply install with npm or yarn. Tela is available on the NPM registry or GitHub Packages.

npm install @efstajas/tela

or

yarn add @efstajas/tela

🎬 Getting started

In Tela, you define Canvas Kit apps as simple classes that match the webhooks generated by Intercom for an app. Afterwards, you can use Tela to serve your apps with one line of code.

💡 This guide assumes you use TypeScript — everything also works with plain JS too, but syntax may be different. You really should use TypeScript either way!

1️⃣ Create your app

Create a new file named your-app.app.ts. Within that, export a new class YourName that implements App, and return an array consisting of a single text component:

//your-app.app.ts
import { App, Component } from '@efstajas/tela'

export default class ExampleApp implements App {
  public initialize = (body): Component[] => {
    return [
      {
        type: 'text',
        text: 'Hello world',
        style: 'paragraph'
      }
    ]
  }
}

Congratulations, you just implemented your first app. initialize is a handler — and initialize specifically is required, but you can also add configure and submit handlers if you need them. For your convenience, these handlers are a bit magic: Instead of you handling response sending and canvas creation manually in each handler, you simply return either an array of Components or a Promise that resolves to an array of Components directly. Tela will automatically take care of wrapping your components to create a valid canvas definition and send it back to Intercom once your handler has resolved. Anyway — let's run our new app!

2️⃣ Register your app

In a new file index.ts, import your app, create a new Tela instance, and register your app:

//index.ts
import Tela from '@efstajas/tela'
import YourApp from './your-app.app.ts'

const tela = new Tela()

tela.registerApp('example-app', new YourApp())

The first argument to registerApp is your app name — please make sure it's a URL-safe string, since it'll be used as a path for the server later. Please make sure you pass a new instance of your app, not the class itself!

3️⃣ Start the server

Now that our app is registered, we can go ahead and start our server:

tela.listen(8000)

Go ahead and run the script — congratulations, your server is now listening on port 8000. Try calling POST /example-app/initialize, and you'll see that you receive back a valid canvas defininion with the text component you defined in your initialize handler.

To get your app up and running in Intercom, go read the official Intercom Canvas Kit documentation. It's quite simple!

😎 Advanced usage

🤚 Handlers

In each app you can define up to three handlers: initialize, submit and configure. To understand what they're for, it's best to read Intercom's documentation. In addition to a synchronous handler like in the Getting Started guide, you can also create async handlers, for example if you need to get some data from an API to create your components:

//your-app.ts
import { App, Component } from '@efstajas/tela'

export default class ExampleApp implements App {
  public initialize = async (body): Promise<Component[]> => {
    const {
      customer
    } = body
    const userId = customer.id

    const userData = await someApiService.getUser(userId)

    return [
      {
        type: 'text',
        text: `The user's favorite color is ${userData.favoriteColor}.`,
        style: 'paragraph'
      }
    ]
  }
}

If you need to read information that Intercom sends with the requests, worry not: The first argument for your handler is Intercom's request body.

🖕 Middlewares

Sometimes, you want to perform some logic after every incoming request and pass down some data to individual handlers. For that, you can use the registerMiddleware function.

Let's say for example that we want to parse Intercom's locale context value from the incoming request, initialize an i18next instance for localization, and then pass it into each handler of our app for convenient usage.

tela.registerMiddleware(async (req, middlewareContext) => {
  const body = req.body

  const {
    context: intercomContext
  } = body

  const browserLanguage = (intercomContext && intercomContext.locale) || 'en'

  const t = await i18n(browserLanguage)

  return {
    ...middlewareContext,
    t
  }
})

As you can see, registerMiddleware accepts any handler (promise or synchronous function). Your middleware gets the full express req object, as well as the previous' middleware's middlewareContext which will include what was returned by the previous middleware in the chain.

Simply perform your logic and return an object that contains all previous middleware context and the new context added by this middleware handler. If you call registerMiddleware multiple times, all handlers will be executed for each incoming request.

Within your app's handlers, you can now find your middlewareContext as part of the context argument.

//your-app.ts
import { App, Component } from '@efstajas/tela'

export default class ExampleApp implements App {
  public initialize = async (body, context: HandlerContext): Promise<Component[]> => {
    const {
      middlewareContext
    } = handlerContext

    const { t } = middlewareContext

    return [
      {
        type: 'text',
        text: t('translation.key')
        style: 'paragraph'
      }
    ]
  }
}

🗃 Returning stored_data and content_url

If you need to send Intercom stored data values and / or a content URL for Live Canvasses in addition to components to construct a view, you can return the more verbose HandlerResult or a Promise resolving to a HandlerResult instead:

//your-app.ts
import { App, HandlerResult } from '@efstajas/tela'

export default class ExampleApp implements App {
  public initialize = async (body): Promise<HandlerResult> => {
    return {
      components: [ /* Your view… */ ],
      storedData: {
        foo: 'bar'
      },
      contentUrl: '' // Your Live Canvas Content URL
    }
  }
}

components is of course required, while storedData and contentUrl are optional.

🌍 Handler Context

Alongside the request body passed from Intercom, your handler also receives a context object as the second argument. The context includes the current app name your handler is running in, the app's base endpoint path along with two objects hooks and methods, which you can use to find out the app's other handler's endpoints at runtime.

public initialize = (requestBody, context: HandlerContext) => {
  const {
    endpoint,
    appName,
    hooks,
    methods
  } = context

  console.log(`
    This handler's path is ${endpoint}.
    It's part of app ${appName}.
  `)

  /*
  The hooks and methods objects are helpful for registering a webhook
  with one of your handlers at runtime, for example.
  Let's say that in response to an action on Intercom you want to make an
  API call that establishes a webhook to a handler in this app:

  service.createWebhook({
    url: `${hostname}${hooks.hookName.endpoint}`
  })
  */

  return [
    {
      type: 'text',
      text: 'Hello world',
      style: 'paragraph'
    }
  ]
}

🔌 Receiving webhooks in your app

Often-times, you'll need to listen to external webhooks other than those for Canvas Kit and perform some action in response. You can define external webhook handlers in a public hooks object:

public hooks = {
  hookName: (req, res, next, context) => {
    console.log(`Handling incoming webhook at ${context.endpoint}`)

    res.send(200)
    next()
  }
}

These handlers are defined as standard Express middleware. Each defined hook will be initialized at /appname/hookname* (Note the * — that means that a webhook coming in at /appname/hookname/foobar will still hit your handler).

✅ Starting the server

Of course, you can run multiple apps at the same time by calling registerApp multiple times before calling listen. Each app will be initialized at /appname/handlername. registerApp also returns a Promise that resolves to a context object, including the paths for all handlers that were initialized within your app.

import Tela, { HandlerContext } from '@efstajas/tela'
// Import your apps
import apps from './apps'

const tela = new Tela()

let promises: Promise<void>[] = []

apps.forEach((app) => {
  promises.push(
    tela.registerApp('test', new App())
      .then((context) => {
        console.log('App initialized', context)
      })
      .catch((e) => {
        console.error(e)
      })
  )
})

await Promise.all(promises)

console.log('All apps initialized, starting server…')

tela.listen(8000).then(() => {
  console.log(`Listening at 8000`)
}).catch((e) => {
  console.error(e)
})

🚂 Accessing the internal Express instance

If you need to add your own endpoints not part of an Intercom app to the internal server, like for example a /health endpoint, you can access the internal Express server instance directly. Please note that you should do this and registering apps only before calling listen.

import Tela from '@efstajas/tela'
import App from './app'

const tela = new Tela()

tela.expressInstance.get('/health', (req, res, next) => {
  res.send(200)
  next()
})

tela.listen(8000).then(() => {
  console.log(`Listening at 8000`)
}).catch((e) => {
  console.error(e)
})