Skip to content

Commit

Permalink
updated socketMode package
Browse files Browse the repository at this point in the history
  • Loading branch information
stevengill committed Dec 14, 2020
1 parent 7874399 commit c159234
Show file tree
Hide file tree
Showing 6 changed files with 274 additions and 18 deletions.
239 changes: 239 additions & 0 deletions docs/_packages/socket_mode.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,239 @@
---
title: Socket Mode
permalink: /socket-mode
order: 7
anchor_links_header: Usage
---

# Slack Socket Mode


## Installation

```shell
$ npm install @slack/socket-mode
```

---

### Initialize the client

The package exports an `SocketModeClient` class. Your app will create an instance of the class for each workspace it
communicates with. Creating an instance requires a `app level token` from Slack. Apps connect to the `Socket Mode` API using the newly introduced `app level token`, which start with `xapp`.

Note: `Socket Mode` requires the scope of `connections:write`. In your browser, navigate to your app's app config and go to the `OAuth and Permissions` section to add the scope.



```javascript
const { SocketModeClient } = require('@slack/socket-mode');

// Read a token from the environment variables
const appToken = process.env.SLACK_APP_TOKEN;

// Initialize
const client = new SocketModeClient({appToken});
```

### Connect to Slack

Data from Slack will begin to flow to your program once the client is connected. You'll also be able to send data to
Slack after the connection is established. Connecting is as easy as calling the `.start()` method.

```javascript
const { SocketModeClient } = require('@slack/socket-mode');
const appToken = process.env.SLACK_APP_TOKEN;

const socketModeClient = new SocketModeClient(appToken);

(async () => {
// Connect to Slack
await socketModeClient.start();
})();
```
---

### Listen for an event

Apps register functions, called listeners, to be triggered when an event of a specific type is received by the client.
If you've used Node's [`EventEmitter`](https://nodejs.org/api/events.html#events_class_eventemitter) pattern
before, then you're already familiar with how this works, since the client is an `EventEmitter`.

The `event` argument passed to the listener is an object. It's contents corresponds to the [type of
event](https://api.slack.com/events) its registered for.

```javascript
const { SocketModeClient } = require('@slack/socket-mode');
const appToken = process.env.SLACK_APP_TOKEN;

const socketModeClient = new SocketModeClient(appToken);

// Attach listeners to events by type. See: https://api.slack.com/events/message
socketModeClient.on('message', (event) => {
console.log(event);
});

(async () => {
await socketModeClient.start();
})();
```

---

### Send a message

To respond to events and send messages back into slack, it is recommend to use the `@slack/web-api` package with a `bot token`.

```javascript
const { SocketModeClient } = require('@slack/socket-mode');
const { WebClient } = require('@slack/web-api');

const socketModeClient = new SocketModeClient(process.env.SLACK_APP_TOKEN);
const webclient = new WebClient(process.env.BOT_TOKEN);

// Attach listeners to events by type. See: https://api.slack.com/events/message
socketModeClient.on('member_joined_channel', async ({event, body, ack}) => {
try {
// send acknowledgement back to slack over the socketMode websocket connection
// this is so slack knows you have received the event and are processing it
await ack();
await webclient.chat.postMessage({
blocks: [
{
type: 'section',
text: {
type: 'mrkdwn',
text: `Welcome to the channel, <@${event.user}>. We're here to help. Let us know if you have an issue.`,
},
accessory: {
type: 'button',
text: {
type: 'plain_text',
text: 'Get Help',
},
value: 'get_help',
},
},
],
channel: event.channel,
});
} catch (error) {
console.log('An error occurred', error);
}
});
```
---

### Lifecycle events

The client's connection to Slack has a lifecycle. This means the client can be seen as a state machine which transitions
through a few states as it connects, disconnects, reconnects, and synchronizes with Slack. The client emits an event
for each state it transitions to throughout its lifecycle. If your app simply needs to know whether the client is
connected or not, the `.connected` boolean property can be checked.

In the table below, the client's states are listed, which are also the names of the events you can use to observe
the transition to that state. The table also includes description for the state, and arguments that a listener would
receive.

| Event Name | Arguments | Description |
|-----------------|-----------------|-------------|
| `connecting` | | The client is in the process of connecting to the platform. |
| `authenticated` | `(connectData)` - the response from `apps.connections.open` | The client has authenticated with the platform. This is a sub-state of `connecting`. |
| `connected` | | The client is connected to the platform and incoming events will start being emitted. |
| `ready` | | The client is ready to send outgoing messages. This is a sub-state of `connected` |
| `disconnecting` | | The client is no longer connected to the platform and cleaning up its resources. It will soon transition to `disconnected`. |
| `reconnecting` | | The client is no longer connected to the platform and cleaning up its resources. It will soon transition to `connecting`. |
| `disconnected` | `(error)` | The client is not connected to the platform. This is a steady state - no attempt to connect is occurring. The `error` argument will be `undefined` when the client initiated the disconnect (normal). |

The client also emits events that are part of its lifecycle, but aren't states. Instead, they represent specific
moments that might be interesting to your app. The following table lists these events, their description, and includes
the arguments that a listener would receive.

| Event Name | Arguments | Description |
|-----------------|-----------|-------------|
| `error` | `(error)` | An error has occurred. See [error handling](#handle-errors) for details. |
| `slack_event` | `(eventType, event)` | An incoming Slack event has been received. |
| `unable_to_socket_mode_start` | `(error)` | A problem occurred while connecting, a reconnect may or may not occur. |

---

### Logging

The `SocketModeClient` will log interesting information to the console by default. You can use the `logLevel` to decide how
much information, or how interesting the information needs to be, in order for it to be output. There are a few possible
log levels, which you can find in the `LogLevel` export. By default, the value is set to `LogLevel.INFO`. While you're
in development, its sometimes helpful to set this to the most verbose: `LogLevel.DEBUG`.

```javascript
// Import LogLevel from the package
const { SocketModeClient, LogLevel } = require('@slack/socket-mode');
const appToken = process.env.SLACK_APP_TOKEN;

// Log level is one of the options you can set in the constructor
const socketModeClient = new SocketModeClient({
appToken,
logLevel: LogLevel.DEBUG,
});

(async () => {
await socketModeClient.start();
})();
```

All the log levels, in order of most to least information are: `DEBUG`, `INFO`, `WARN`, and `ERROR`.

<details>
<summary markdown="span">
<strong><i>Sending log output somewhere besides the console</i></strong>
</summary>

You can also choose to have logs sent to a custom logger using the `logger` option. A custom logger needs to implement
specific methods (known as the `Logger` interface):

| Method | Parameters | Return type |
|--------------|-------------------|-------------|
| `setLevel()` | `level: LogLevel` | `void` |
| `setName()` | `name: string` | `void` |
| `debug()` | `...msgs: any[]` | `void` |
| `info()` | `...msgs: any[]` | `void` |
| `warn()` | `...msgs: any[]` | `void` |
| `error()` | `...msgs: any[]` | `void` |

A very simple custom logger might ignore the name and level, and write all messages to a file.

```javascript
const { createWriteStream } = require('fs');
const logWritable = createWriteStream('/var/my_log_file'); // Not shown: close this stream

const socketModeClient = new SocketModeClient(appToken, {
// Creating a logger as a literal object. It's more likely that you'd create a class.
logger: {
debug(...msgs): { logWritable.write('debug: ' + JSON.stringify(msgs)); },
info(...msgs): { logWritable.write('info: ' + JSON.stringify(msgs)); },
warn(...msgs): { logWritable.write('warn: ' + JSON.stringify(msgs)); },
error(...msgs): { logWritable.write('error: ' + JSON.stringify(msgs)); },
setLevel(): { },
setName(): { },
},
});

(async () => {
await socketModeClient.start();
})();
```
</details>

---

## Requirements

This package supports Node v12 LTS and higher. It's highly recommended to use [the latest LTS version of
node](https://github.com/nodejs/Release#release-schedule), and the documentation is written using syntax and features
from that version.

## Getting Help

If you get stuck, we're here to help. The following are the best ways to get assistance working through your issue:

* [Issue Tracker](http://github.com/slackapi/node-slack-sdk/issues) for questions, feature requests, bug reports and
general discussion related to these packages. Try searching before you create a new issue.
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
const { SocketModeClient, LogLevel } = require('.');
const { SocketModeClient, LogLevel } = require('@slack/socket-mode');
const { WebClient } = require('@slack/web-api');


Expand All @@ -8,7 +8,7 @@ const clientOptions = {

const socketModeClient = new SocketModeClient({
logLevel: LogLevel.DEBUG,
token: process.env.APP_TOKEN,
appToken: process.env.APP_TOKEN,
// clientOptions
})
const botToken = process.env.BOT_TOKEN;
Expand Down
17 changes: 17 additions & 0 deletions examples/socket-mode/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
{
"name": "socket-mode-example",
"version": "1.0.0",
"description": "",
"main": "app.js",
"scripts": {
"start": "node app.js"
},
"author": "Slack Technologies, Inc.",
"license": "MIT",
"repository": "slackapi/node-slack-sdk",
"dependencies": {
"@slack/socket-mode": "feat-socket-mode",
"@slack/web-api": "^5.14.0",
}
}

20 changes: 10 additions & 10 deletions packages/socket-mode/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,10 +36,10 @@ Note: `Socket Mode` requires the scope of `connections:write`. In your browser,
const { SocketModeClient } = require('@slack/socket-mode');

// Read a token from the environment variables
const token = process.env.SLACK_APP_TOKEN;
const appToken = process.env.SLACK_APP_TOKEN;

// Initialize
const client = new SocketModeClient({token});
const client = new SocketModeClient({appToken});
```

### Connect to Slack
Expand All @@ -49,9 +49,9 @@ Slack after the connection is established. Connecting is as easy as calling the

```javascript
const { SocketModeClient } = require('@slack/socket-mode');
const token = process.env.SLACK_APP_TOKEN;
const appToken = process.env.SLACK_APP_TOKEN;

const socketModeClient = new SocketModeClient(token);
const socketModeClient = new SocketModeClient(appToken);

(async () => {
// Connect to Slack
Expand All @@ -71,9 +71,9 @@ event](https://api.slack.com/events) its registered for.

```javascript
const { SocketModeClient } = require('@slack/socket-mode');
const token = process.env.SLACK_APP_TOKEN;
const appToken = process.env.SLACK_APP_TOKEN;

const socketModeClient = new SocketModeClient(token);
const socketModeClient = new SocketModeClient(appToken);

// Attach listeners to events by type. See: https://api.slack.com/events/message
socketModeClient.on('message', (event) => {
Expand Down Expand Up @@ -145,7 +145,7 @@ receive.
| Event Name | Arguments | Description |
|-----------------|-----------------|-------------|
| `connecting` | | The client is in the process of connecting to the platform. |
| `authenticated` | `(connectData)` - the response from `relay.connect` | The client has authenticated with the platform. This is a sub-state of `connecting`. |
| `authenticated` | `(connectData)` - the response from `apps.connections.open` | The client has authenticated with the platform. This is a sub-state of `connecting`. |
| `connected` | | The client is connected to the platform and incoming events will start being emitted. |
| `ready` | | The client is ready to send outgoing messages. This is a sub-state of `connected` |
| `disconnecting` | | The client is no longer connected to the platform and cleaning up its resources. It will soon transition to `disconnected`. |
Expand Down Expand Up @@ -174,11 +174,11 @@ in development, its sometimes helpful to set this to the most verbose: `LogLevel
```javascript
// Import LogLevel from the package
const { SocketModeClient, LogLevel } = require('@slack/socket-mode');
const token = process.env.SLACK_APP_TOKEN;
const appToken = process.env.SLACK_APP_TOKEN;

// Log level is one of the options you can set in the constructor
const socketModeClient = new SocketModeClient({
token,
appToken,
logLevel: LogLevel.DEBUG,
});

Expand Down Expand Up @@ -212,7 +212,7 @@ A very simple custom logger might ignore the name and level, and write all messa
const { createWriteStream } = require('fs');
const logWritable = createWriteStream('/var/my_log_file'); // Not shown: close this stream

const socketModeClient = new SocketModeClient(token, {
const socketModeClient = new SocketModeClient(appToken, {
// Creating a logger as a literal object. It's more likely that you'd create a class.
logger: {
debug(...msgs): { logWritable.write('debug: ' + JSON.stringify(msgs)); },
Expand Down
4 changes: 2 additions & 2 deletions packages/socket-mode/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
"dist/**/*"
],
"engines": {
"node": ">=10.0.0"
"node": ">=12.0.0"
},
"repository": "slackapi/node-slack-sdk",
"homepage": "https://slack.dev/node-slack-sdk/socket-mode",
Expand All @@ -44,7 +44,7 @@
},
"dependencies": {
"@slack/logger": "^2.0.0",
"@slack/web-api": "^5.10.0",
"@slack/web-api": "^5.14.0",
"@types/node": ">=10.0.0",
"@types/p-queue": "^2.3.2",
"@types/ws": "^7.2.5",
Expand Down
8 changes: 4 additions & 4 deletions packages/socket-mode/src/SocketModeClient.ts
Original file line number Diff line number Diff line change
Expand Up @@ -326,12 +326,12 @@ export class SocketModeClient extends EventEmitter {
logLevel = LogLevel.INFO,
autoReconnect = true,
clientPingTimeout = 30000,
token = undefined,
appToken = undefined,
clientOptions = {},
}: SocketModeOptions = {}) {
super();

if (token === undefined) {
if (appToken === undefined) {
throw new Error('Must provide an App Level Token when initalizing a Socket Mode Client');
}

Expand All @@ -351,7 +351,7 @@ export class SocketModeClient extends EventEmitter {

this.webClient = new WebClient('', {
logLevel: this.logger.getLevel(),
headers: { Authorization: `Bearer ${token}` },
headers: { Authorization: `Bearer ${appToken}` },
...clientOptions,
});

Expand Down Expand Up @@ -635,7 +635,7 @@ export default SocketModeClient;
*/

export interface SocketModeOptions {
token?: string; // app level token
appToken?: string; // app level token
logger?: Logger;
logLevel?: LogLevel;
autoReconnect?: boolean;
Expand Down

0 comments on commit c159234

Please sign in to comment.