diff --git a/docs/datastores.md b/docs/datastores.md deleted file mode 100644 index 8cee70bd..00000000 --- a/docs/datastores.md +++ /dev/null @@ -1,87 +0,0 @@ -## Datastores - -### Defining a Datastore - -Datastores can be defined with the top level `DefineDatastore` export. Below is -an example of setting up a Datastore: - -```ts -import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; - -export const ReversalsDatastore = DefineDatastore({ - name: "reversals", - attributes: { - id: { - type: Schema.types.string, - }, - original: { - type: Schema.types.string, - }, - reversed: { - type: Schema.types.string, - }, - }, - primary_key: "id", -}); -``` - -### Registering a Datastore to the App - -To register the newly defined Datastore, add it to the array assigned to the -`datastores` parameter while defining the [`Manifest`][manifest]. - -```ts -export default Manifest({ - name: "admiring-ox-50", - description: "Reverse a string", - icon: "assets/icon.png", - functions: [ReverseFunction], - outgoingDomains: [], - datastores: [ReversalsDatastore], - botScopes: [ - "commands", - "chat:write", - "chat:write.public", - "datastore:read", - "datastore:write", - ], -}); -``` - -Note: Registering a Datastore will automatically add `datastore:read` and -`datastore:write` to the App's defined `botScopes`. - -### Using a Datastore in your custom function code - -Now that you have a Datastore all set up, you can use it in your -[`functions`][functions]! Import the -[deno-slack-api](https://github.com/slackapi/deno-slack-api) library, -instantiate your client, and make an API call to one of the Datastore endpoints! - -```ts -import { SlackFunction } from "deno_slack_api/mod.ts"; - -export default SlackFunction(ReverseFunction, async ({ client, inputs }) => { - const original = inputs.stringToReverse; - const recordId = crypto.randomUUID(); - const reversed = inputs.stringToReverse.split("").reverse().join(""); - - const putResp = await client.apps.datastore.put({ - datastore: "reversals", - item: { - id: recordId, - original, - reversed, - }, - }); - if (!putResp.ok) { - return { - error: putResp.error, - }; - } - // ... -}); -``` - -[functions]: ./functions.md -[manifest]: ./manifest.md diff --git a/docs/events.md b/docs/events.md deleted file mode 100644 index bf7313fb..00000000 --- a/docs/events.md +++ /dev/null @@ -1,133 +0,0 @@ -## Events - -Custom events provide a way for Apps to validate -[message metadata](https://api.slack.com/metadata) against a pre-defined schema. - -### Defining an event - -Events can be defined with the top level `DefineEvent` export. Events must be -set up as an `object` type or a [`custom Type`][types] of an `object` type. -Below is an example of setting up a custom Event that can be used during an -incident. - -```ts -const IncidentEvent = DefineEvent({ - name: "incident", - title: "Incident", - type: Schema.types.object, - properties: { - id: { type: Schema.types.string }, - title: { type: Schema.types.string }, - summary: { type: Schema.types.string }, - severity: { type: Schema.types.string }, - date_created: { type: Schema.types.number }, - }, - required: ["id", "title", "summary", "severity"], - additionalProperties: false, // Setting this to false forces the validation to catch any additional properties -}); -``` - -### Registering an event with the app - -To register the newly defined event, add it to the array assigned to the -`events` parameter while defining the [`Manifest`][manifest]. - -Note: All custom events **must** be registered to the [Manifest][manifest] in -order for them to be used. There is no automated registration for events. - -```ts -Manifest({ - ... - events: [IncidentEvent], -}); -``` - -### Referencing events - -There are two places where you can reference your events: - -1. Posting a message to Slack -2. Creating a message metadata trigger - -#### Posting a message to Slack - -Event validation happens against the App's manifest when an App posts a message -to Slack using the -[`metadata` parameter](https://api.slack.com/methods/chat.postMessage#arg_metadata). -If the `event_type` matches the `name` of a custom Event specified in the App's -manifest, it will validate that all required parameters are provided. If it -doesn't meet the validation standards, a warning will be returned in the -response and the message will still be posted, but the metadata will be dropped -from the message. - -```ts -// At workflow authoring time -// This example assumes all required values are passed to the workflow's inputs -MyWorkflow.addStep(Schema.slack.functions.SendMessage, { - channel_id: MyWorkflow.inputs.channel_id, - message: "We have an incident!", - metadata: { - event_type: IncidentEvent, - event_payload: { - id: MyWorkflow.inputs.incident_id, - title: MyWorkflow.inputs.incident_title, - summary: MyWorkflow.inputs.incident_summary, - severity: MyWorkflow.inputs.incident_severity, - date_created: MyWorkflow.inputs.incident_date, // Since this isn't required, it doesn't need to exist to pass validation - }, - }, -}); -``` - -```ts -// At function runtime -// This example assumes all required values are passed to the function's inputs -await client.chat.postMessage({ - channel_id: inputs.channel_id, - message: "We have an incident!", - metadata: { - event_type: IncidentEvent, - event_payload: { - id: inputs.incident_id, - title: inputs.incident_title, - summary: inputs.incident_summary, - severity: inputs.incident_severity, - date_created: inputs.incident_date, // Since this isn't required, it doesn't need to exist to pass validation - }, - }, -}); -``` - -#### Creating a message metadata trigger - -Now that the app has a defined schema for the event, a trigger can be created to -watch for any message posted with the expected metadata. When the schema is met, -the trigger will execute a workflow - -```ts -// A trigger Definition file for the CLI -import { IncidentEvent } from "./manifest.ts"; - -const trigger: Trigger = { - type: "event", - name: "Incident Metadata Posted", - inputs: { - id: "{{data.metadata.event_payload.incident_id}}", - title: "{{data.metadata.event_payload.incident_title}}", - summary: "{{data.metadata.event_payload.incident_summary}}", - severity: "{{data.metadata.event_payload.incident_severity}}", - date_created: "{{data.metadata.event_payload.incident_date}}", - }, - workflow: "#/workflows/start_incident", - event: { - event_type: "slack#/events/message_metadata_posted", - metadata_event_type: IncidentEvent, - channel_ids: ["C012354"], // The channel that needs to be watched for message metadata being posted - }, -}; - -export default trigger; -``` - -[manifest]: ./manifest.md -[types]: ./types.md diff --git a/docs/functions-action-handlers.md b/docs/functions-action-handlers.md deleted file mode 100644 index d91330da..00000000 --- a/docs/functions-action-handlers.md +++ /dev/null @@ -1,264 +0,0 @@ -## Block kit action handlers - -Your application's [functions][functions] can do a wide variety of interesting -things: post messages, create channels, or anything available to developers via -the [Slack API][api]. One of the more compelling features available to app -developers is the ability to use [Block Kit][block-kit] to add richness and -depth to messages in Slack. Even better, [Block Kit][block-kit] supports a -variety of [interactive components][interactivity]! This document explores the -APIs available to app developers building Run-On-Slack applications to leverage -these [interactive components][interactivity] and how applications can respond -to user interactions with these [interactive components][interactivity]. - -If you're already familiar with the main concepts underpinning Block Kit Action -Handlers, then you may want to skip ahead to the -[`addBlockActionsHandler()` method API Reference](#api-reference). - -- [Block kit action handlers](#block-kit-action-handlers) - - [Requirements](#requirements) - - [Posting a message with block kit elements](#posting-a-message-with-block-kit-elements) - - [Adding block action handlers](#adding-block-action-handlers) - - [API reference](#api-reference) - - [`addBlockActionsHandler(constraint, handler)`](#addblockactionshandlerconstraint-handler) - - [`BlockActionConstraintField`](#blockactionconstraintfield) - - [`BlockActionConstraintObject`](#blockactionconstraintobject) - -### Requirements - -Your app needs to have an existing [function][functions] defined, implemented -and working before you can add interactivity handlers like Block Kit Action -Handlers to them. Make sure you have followed our -[functions documentation][functions] and have a function in your app ready that -we can expand with a Block Kit Action Handler. - -As part of exploring how Block Kit Action Handlers work, we'll walk through an -approval flow example. A user would trigger our app's function, which would post -a message with two buttons: Approve and Deny. Once someone clicks either button, -our app will handle these button interactions - these Block Kit Actions - and -update the original message with either an "Approved!" or "Denied!" text. - -For the purposes of walking through this approval flow example, let us assume -the following [function][functions] definition (that we will store in a file -called `definition.ts` under the `functions/approval/` subdirectory inside your -app): - -```typescript -import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; - -export const ApprovalFunction = DefineFunction({ - callback_id: "review_approval", - title: "Approval", - description: "Get approval for a request", - source_file: "functions/approval/mod.ts", // <-- important! Make sure this is where the logic for your function - which we will write in the next section - exists. - input_parameters: { - properties: { - requester_id: { - type: Schema.slack.types.user_id, - description: "Requester", - }, - approval_channel_id: { - type: Schema.slack.types.channel_id, - description: "Approval channel", - }, - }, - required: [ - "requester_id", - "approval_channel_id", - ], - }, - output_parameters: { - properties: { - approved: { - type: Schema.types.boolean, - description: "Approved", - }, - reviewer: { - type: Schema.slack.types.user_id, - description: "Reviewer", - }, - message_ts: { - type: Schema.types.string, - description: "Request Message TS", - }, - }, - required: ["approved", "reviewer", "message_ts"], - }, -}); -``` - -### Posting a message with block kit elements - -First, we need a message that has some [interactive components][interactivity] -from [Block Kit][block-kit] included! We can modify one of our app's -[functions][functions] to post a message that includes some interactive -components. Here's an example function (which we will assume exists in a -`mod.ts` file under the `functions/approval/` subdirectory in your app) that -posts a message with two buttons: an approval button, and a deny button: - -```typescript -import { SlackFunction } from "deno-slack-sdk/mod.ts"; -// ApprovalFunction is the function we defined in the previous section -import { ApprovalFunction } from "./definition.ts"; - -export default SlackFunction(ApprovalFunction, async ({ inputs, client }) => { - console.log("Incoming approval!"); - - await client.chat.postMessage({ - channel: inputs.approval_channel_id, - blocks: [{ - "type": "actions", - "block_id": "mah-buttons", - "elements": [{ - type: "button", - text: { - type: "plain_text", - text: "Approve", - }, - action_id: "approve_request", - style: "primary", - }, { - type: "button", - text: { - type: "plain_text", - text: "Deny", - }, - action_id: "deny_request", - style: "danger", - }], - }], - }); - // Important to set completed: false! We will set the function's complete - // status later - in our action handler - return { - completed: false, - }; -}); -``` - -The key bit of information we need to remember before moving on to adding an -action handler are the `action_id` and `block_id` properties defined in the -`blocks` payload. Using these IDs, we will be able to differentiate between the -different button components that users interacted with in this message. - -### Adding block action handlers - -The [Deno Slack SDK][sdk] - which comes bundled in your generated Run-on-Slack -application - provides a means for defining a handler to execute every time a -user interacts with an interactive Block Kit element created by your function. - -Continuing with our above example, we can now define a handler that will listen -for actions on one of the interactive components we attached to the message our -main function posted: either the approve button being clicked or the deny button -being clicked. The code to add a Block Kit action handler is "chained" off of -your top-level function, and would look like this: - -```typescript -export default SlackFunction(ApprovalFunction, async ({ inputs, client }) => { - // ... the rest of your ApprovalFunction logic here ... -}).addBlockActionsHandler( - ["approve_request", "deny_request"], // The first argument to addBlockActionsHandler can accept an array of action_id strings, among many other formats! - // Check the API reference at the end of this document for the full list of supported options - async ({ action, body, client }) => { // The second argument is the handler function itself - console.log("Incoming action handler invocation", action); - - const outputs = { - reviewer: body.user.id, - // Based on which button was pressed - determined via action_id - we can - // determine whether the request was approved or not. - approved: action.action_id === "approve_request", - message_ts: body.message.ts, - }; - - // Remove the button from the original message using the chat.update API - // and replace its contents with the result of the approval. - await client.chat.update({ - channel: body.function_data.inputs.approval_channel_id, - ts: outputs.message_ts, - blocks: [{ - type: "context", - elements: [ - { - type: "mrkdwn", - text: `${ - outputs.approved ? " :white_check_mark: Approved" : ":x: Denied" - } by <@${outputs.reviewer}>`, - }, - ], - }], - }); - - // And now we can mark the function as 'completed' - which is required as - // we explicitly marked it as incomplete in the main function handler. - await client.functions.completeSuccess({ - function_execution_id: body.function_data.execution_id, - outputs, - }); - }, -); -``` - -Now when you run your app and trigger your function, you have the basics in -place to provide interactivity between your application and users in Slack! - -### API reference - -#### `addBlockActionsHandler(constraint, handler)` - -```typescript -SlackFunction({ ... }).addBlockActionsHandler({ block_id: "mah-buttons", action_id: "approve_request"}, async (ctx) => { ... }); -``` - -`addHandler` registers a block action handler based on a `constraint` argument. -If any incoming actions match the `constraint`, then the specified `handler` -will be invoked with the action. This allows for authoring focussed, -single-purpose action handlers and provides a concise but flexible API for -registering handlers to specific actions. - -`constraint` is of type [`BlockActionConstraint`][constraint], which itself can -be either a [`BlockActionConstraintField`](#blockactionconstraintfield) or a -[`BlockActionConstraintObject`](#blockactionconstraintobject). - -If a [`BlockActionConstraintField`](#blockactionconstraintfield) is used as the -value for `constraint`, then this will be matched against the incoming action's -`action_id` property. - -[`BlockActionConstraintObject`](#blockactionconstraintobject) is a more complex -object used to match against actions. It contains nested `block_id` and -`action_id` properties - both optional - that are used to match against the -incoming action. - -##### `BlockActionConstraintField` - -```typescript -type BlockActionConstraintField = string | string[] | RegExp; -``` - -- when provided as a `string`, it must match the field exactly. -- when provided as an array of `string`s, it must match one of the array values - exactly. -- when provided as a `RegExp`, the regular expression must match. - -###### `BlockActionConstraintObject` - -```typescript -type BlockActionConstraintObject = { - block_id?: BlockActionConstraintField; - action_id?: BlockActionConstraintField; -}; -``` - -This object can contain two properties, both optional: `action_id` and/or -`block_id`. The type of each property is -[`BlockActionConstraintField`](#blockactionconstraintfield). - -If both `action_id` and `block_id` properties exist on the `constraint`, then -both `action_id` and `block_id` properties _must match_ any incoming action. If -only one of these properties is provided, then only the provided property must -match. - -[functions]: ./functions.md -[api]: https://api.slack.com/methods -[block-kit]: https://api.slack.com/block-kit -[interactivity]: https://api.slack.com/block-kit/interactivity -[sdk]: https://github.com/slackapi/deno-slack-sdk -[constraint]: ../src/functions/routers/types.ts#L53-L62 diff --git a/docs/functions-suggestion-handlers.md b/docs/functions-suggestion-handlers.md deleted file mode 100644 index 37b18090..00000000 --- a/docs/functions-suggestion-handlers.md +++ /dev/null @@ -1,244 +0,0 @@ -## Block Kit suggestion handlers - -Your application's [functions][functions] can do a wide variety of interesting -things: post messages, create channels, or anything available to developers via -the [Slack API][api]. One of the more compelling features available to app -developers is the ability to use [Block Kit][block-kit] to add richness and -depth to messages in Slack. Even better, [Block Kit][block-kit] supports a -variety of [interactive components][interactivity]! This document explores how -to provide dynamic menu options for -[external-data-sourced Block Kit drop-down menus](https://api.slack.com/reference/block-kit/block-elements#external_select). - -If you're already familiar with the main concepts underpinning Block Kit -Suggestion Handlers, then you may want to skip ahead to the -[`addBlockSuggestionHandler()` method API Reference](#api-reference). - -Worthwhile noting that while this document covers how to render custom menu -options for external-data-sourced Block Kit drop-down menus, it does _not_ cover -how to respond to a user selecting one of the custom menu options. Do not fear, -though! The same approach discussed in our -[Block Kit Action Handlers][action-handlers] document can be used to register an -action handler to respond to drop-down menu selections. - -- [Block Kit suggestion handlers](#block-kit-suggestion-handlers) - - [Requirements](#requirements) - - [Posting a message with block kit elements](#posting-a-message-with-block-kit-elements) - - [Adding Block Suggestion Handlers](#adding-block-suggestion-handlers) - - [API Reference](#api-reference) - - [`addBlockSuggestionHandler(constraint, handler)`](#addblocksuggestionhandlerconstraint-handler) - - [`BlockActionConstraintField`](#blockactionconstraintfield) - - [`BlockActionConstraintObject`](#blockactionconstraintobject) - -### Requirements - -Your app needs to have an existing [function][functions] defined, implemented -and working before you can add interactivity handlers like Block Kit Suggestion -Handlers to them. Make sure you have followed our -[functions documentation][functions] and have a function in your app ready that -we can expand with interactivity. Familiarity with the -[Block Kit Actions Handlers][action-handlers] would be a huge plus as the -handling Block Kit Actions and handling Block Kit Suggestions is practically -identical. - -As part of exploring how Block Kit Suggestion Handlers work, we'll walk through -an example that posts an inspirational quote. A user would trigger our app's -function, which would post a message with a drop down select menu and a button. -The options rendered in the select menu will be dynamically loaded from an -external API. Finally, when someone has selected a drop-down menu option and -clicked the button, our app can post the selection to the channel (note: for the -purposes of describing how to respond to the select menu interactions, we won't -cover handling the button click or posting the selection in this document). - -For the purposes of walking through this example, let us assume the following -[function][functions] definition (that we will store in a file called -`definition.ts` under the `functions/quote/` subdirectory inside your app): - -```typescript -import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; - -export const QuoteFunction = DefineFunction({ - callback_id: "quote", - title: "Inspire Me", - description: "Get an inspirational quote", - source_file: "functions/quote/mod.ts", // <-- important! Make sure this is where the logic for your function - which we will write in the next section - exists. - input_parameters: { - properties: { - requester_id: { - type: Schema.slack.types.user_id, - description: "Requester", - }, - channel_id: { - type: Schema.slack.types.channel_id, - description: "Channel", - }, - }, - required: [ - "requester_id", - "channel_id", - ], - }, - output_parameters: { - properties: { - quote: { - type: Schema.types.string, - description: "Quote", - }, - }, - required: ["quote"], - }, -}); -``` - -### Posting a message with block kit elements - -First, we need a message that has some [interactive components][interactivity] -from [Block Kit][block-kit] included! We can modify one of our app's -[functions][functions] to post a message that includes some interactive -components - including our external select drop down menu. Here's an example -function (which we will assume exists in a `mod.ts` file under the -`functions/quote/` subdirectory in your app) that posts a message with a -external data select drop down menu: - -```typescript -import { SlackFunction } from "deno-slack-sdk/mod.ts"; -// QuoteFunction is the function we defined in the previous section -import { QuoteFunction } from "./definition.ts"; - -export default SlackFunction(QuoteFunction, async ({ inputs, client }) => { - console.log("Incoming quote request!"); - - await client.chat.postMessage({ - channel: inputs.channel_id, - blocks: [{ - "type": "actions", - "block_id": "so-inspired", - "elements": [{ - type: "external_select", - placeholder: { - type: "plain_text", - text: "Inspire", - }, - action_id: "ext_select_input", - }, { - type: "button", - text: { - type: "plain_text", - text: "Post", - }, - action_id: "post_quote", - }], - }], - }); - // Important to set completed: false! We should set the function's complete - // status later - in the action handler responding to the button click - return { - completed: false, - }; -}); -``` - -The key bit of information we need to remember before moving on to adding a -suggestion handler are the `action_id` and `block_id` properties defined in the -`blocks` payload. Using these IDs, we will be able to differentiate between the -different Block Kit components that users interacted with in this message. - -### Adding Block Suggestion Handlers - -The [Deno Slack SDK][sdk] - which comes bundled in your generated Run-on-Slack -application - provides a means for defining a handler to execute every time a -user interacts with an interactive Block Kit element created by your function. - -Continuing with our above example, we can now define a handler that will listen -for interactions with the external data drop down menu. The code to add a Block -Kit suggestion handler is "chained" off of your top-level function, and would -look like this: - -```typescript -export default SlackFunction(QuoteFunction, async ({ inputs, client }) => { - // ... the rest of your QuoteFunction logic here ... -}).addBlockSuggestionHandler( - "ext_select_input", // The first argument to addBlockActionsHandler can accept an action_id string, among many other formats! - // Check the API reference at the end of this document for the full list of supported options - async ({ body, client }) => { // The second argument is the handler function itself - console.log("Incoming suggestion handler invocation", body); - // Fetch some inspirational quotes - const apiResp = await fetch( - "https://motivational-quote-api.herokuapp.com/quotes", - ); - const quotes = await apiResp.json(); - console.log("Returning", quotes.length, "quotes"); - const opts = { - "options": quotes.map((q) => ({ - value: `${q.id}`, - text: { type: "plain_text", text: q.quote.slice(0, 70) }, - })), - }; - return opts; - }, -); -``` - -### API Reference - -#### `addBlockSuggestionHandler(constraint, handler)` - -```typescript -SlackFunction({ ... }).addBlockSuggestionHandler({ block_id: "mah-buttons", action_id: "approve_request"}, async (ctx) => { ... }); -``` - -`addBlockSuggestionHandler` registers a block suggestion handler based on a -`constraint` argument. If any incoming suggestion events match the `constraint`, -then the specified `handler` will be invoked with the suggestion payload. This -allows for authoring focussed, single-purpose suggestion handlers and provides a -concise but flexible API for registering handlers to specific -external-data-sourced drop down menu. - -`constraint` is of type [`BlockActionConstraint`][constraint], which itself can -be either a [`BlockActionConstraintField`](#blockactionconstraintfield) or a -[`BlockActionConstraintObject`](#blockactionconstraintobject). - -If a [`BlockActionConstraintField`](#blockactionconstraintfield) is used as the -value for `constraint`, then this will be matched against the incoming action's -`action_id` property. - -[`BlockActionConstraintObject`](#blockactionconstraintobject) is a more complex -object used to match against actions. It contains nested `block_id` and -`action_id` properties - both optional - that are used to match against the -incoming suggestion. - -##### `BlockActionConstraintField` - -```typescript -type BlockActionConstraintField = string | string[] | RegExp; -``` - -- when provided as a `string`, it must match the field exactly. -- when provided as an array of `string`s, it must match one of the array values - exactly. -- when provided as a `RegExp`, the regular expression must match. - -###### `BlockActionConstraintObject` - -```typescript -type BlockActionConstraintObject = { - block_id?: BlockActionConstraintField; - action_id?: BlockActionConstraintField; -}; -``` - -This object can contain two properties, both optional: `action_id` and/or -`block_id`. The type of each property is -[`BlockActionConstraintField`](#blockactionconstraintfield). - -If both `action_id` and `block_id` properties exist on the `constraint`, then -both `action_id` and `block_id` properties _must match_ any incoming suggestion. -If only one of these properties is provided, then only the provided property -must match. - -[functions]: ./functions.md -[action-handlers]: ./functions-action-handlers.md -[api]: https://api.slack.com/methods -[block-kit]: https://api.slack.com/block-kit -[interactivity]: https://api.slack.com/block-kit/interactivity -[sdk]: https://github.com/slackapi/deno-slack-sdk -[constraint]: ../src/functions/routers/types.ts#L53-L62 diff --git a/docs/functions-view-handlers.md b/docs/functions-view-handlers.md deleted file mode 100644 index 21ab6239..00000000 --- a/docs/functions-view-handlers.md +++ /dev/null @@ -1,333 +0,0 @@ -## View Handlers - -Your application's [functions][functions] can do a wide variety of interesting -things: post messages, create channels, or anything available to developers via -the [Slack API][api]. They can even include -[interactive components][interactivity] or pop up a [Modal][modals]. -[Modals][modals] are composed of up to three [Views][views]. These -[Views][views] can contain form inputs or -[interactive components][interactivity]. [Views][views] themselves may also -[trigger events][view-events]. This document explores the APIs available to app -developers building Run-On-Slack applications to create [modals][modals] -composed of [views][views] and how applications can respond to the -[view submission and closed events][view-events] they can trigger. - -If you're already familiar with the main concepts underpinning View Handlers, -then you may want to skip ahead to the [API Reference](#api-reference). - -- [View Handlers](#view-handlers) - - [Requirements](#requirements) - - [Opening a view](#opening-a-view) - - [Opening a view from a custom function](#opening-a-view-from-a-custom-function) - - [Opening a view from a block action handler](#opening-a-view-from-a-block-action-handler) - - [Adding view handlers](#adding-view-handlers) - - [API reference](#api-reference) - - [`addViewSubmissionHandler(constraint, handler)`](#addviewsubmissionhandlerconstraint-handler) - - [`addViewClosedHandler(constraint, handler)`](#addviewclosedhandlerconstraint-handler) - -### Requirements - -This functionality requires at least version 0.2.0 of the -[`deno-slack-sdk`][sdk]. - -Your app needs to have an existing [function][functions] defined, implemented -and working before you can add interactivity handlers like View Handlers or -[Block Kit Action Handlers][action-handlers] to them. Make sure you have -followed our [functions documentation][functions] and have a function in your -app ready that we can expand with a View Handler. - -As part of exploring how View Handlers work, we'll walk through a simple diary -flow example. It is nothing more than a contrived example aimed at showing off -the APIs. A user would trigger our app's function, which would open a view with -a single text input. If the view is submitted with content, the application will -send the user a DM with their inputted content. If the view is closed, the -application will send the user a DM encouraging them not to give up on their -diarying habit. - -For the purposes of walking through this approval flow example, let us assume -the following [function][functions] definition (that we will store in a file -called `definition.ts` under the `functions/diary/` subdirectory inside your -app): - -```typescript -import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; - -export const DiaryFunction = DefineFunction({ - callback_id: "diary", - title: "Diary", - description: "Write a diary entry", - source_file: "functions/diary/mod.ts", // <-- important! Make sure this is where the logic for your function - which we will write in the next section - exists. - input_parameters: { - properties: { - interactivity: { // <-- important! This gives Slack a hint that your function will create interactive elements like views - type: Schema.slack.types.interactivity, - }, - channel_id: { - type: Schema.slack.types.channel_id, - }, - }, - required: ["interactivity"], - }, - output_parameters: { - properties: {}, - required: [], - }, -}); -``` - -### Opening a view - -[Opening a view via the `views.open` API][views-open] and -[pushing a new view onto the view stack via the `views.push` API][views-push] -both require the use of a [`trigger_id`][trigger-ids]. These are identifiers -representing specific user interactions. Slack uses these to prevent -applications from haphazardly opening modals in users' faces willy-nilly. -Without a `trigger_id`, your application can't create a modal and open a view. -FYI `trigger_id`s are also known as `interactivity_pointer`s. - -As such, there are two ways to open a view from inside a Run-On-Slack -application: doing so -[from a function directly](#opening-a-view-from-a-function) vs. doing so -[from a Block Action Handler](#opening-a-view-from-a-block-action-handler). The -sections covering each approach below discuss how to retrieve the `trigger_id` -in each scenario. - -We will explore implementing our contrived example above by opening a view from -a function. In a section further below, we will also cover -[opening a view from a Block Action Handler](#opening-a-view-from-a-block-action-handler). - -#### Opening a view from a custom function - -As mentioned in the previous section, we need to have a `trigger_id` handy in -order to open a view. This is why we defined an `interactivity` input in our -function definition earlier: this input will magically provide us with a -`trigger_id`. The property to use as a `trigger_id` exists on inputs with the -type `Schema.slack.types.interactivity` under the `interactivity_pointer` -property. Check out the code below for an example: - -```typescript -import { SlackFunction } from "deno-slack-sdk/mod.ts"; -// DiaryFunction is the function we defined in the previous section -import { DiaryFunction } from "./definition.ts"; - -export default SlackFunction(DiaryFunction, async ({ inputs, client }) => { - console.log('Someone might want to write a diary entry...'); - - await client.views.open({ - trigger_id: inputs.interactivity.interactivity_pointer, - view: { - "type": "modal", - "title": { - "type": "plain_text", - "text": "Modal title", - }, - "blocks": [ - { - "type": "input", - "block_id": "section1", - "element": { - "type": "plain_text_input", - "action_id": "diary_input", - "multiline": true, - "placeholder": { - "type": "plain_text", - "text": "What is on your mind today?", - }, - }, - "label": { - "type": "plain_text", - "text": "Diary Entry", - }, - "hint": { - "type": "plain_text", - "text": "Don't worry, no one but you will see this.", - }, - }, - ], - "close": { - "type": "plain_text", - "text": "Cancel", - }, - "submit": { - "type": "plain_text", - "text": "Save", - }, - "callback_id": "view_identifier_12", // <-- remember this ID, we will use it to route events to handlers! - "notify_on_close": true, // <-- this must be defined in order to trigger `view_closed` events! - }, - }); - // Important to set completed: false! We will set the function's complete - // status later - in our view submission handler - return { - completed: false, - }; -}; -``` - -#### Opening a view from a block action handler - -If [Block Kit Action Handlers][action-handlers] is a foreign concept to you, we -recommend first checking out [its documentation][action-handlers] before -venturing deeper into this section. - -Similarly to opening a view from a function, doing so from a -[Block Action Handler][action-handlers] is straightforward though slightly -different. It is important to remember that `trigger_id`s represent a unique -user interaction with a particular interactive component within Slack's UI. As -such, when responding to a Block Kit Action interactive component, we don't want -to use your function's `inputs` to retrieve the `interactivity_pointer`, as we -did in the previous section, but rather, we want to retrieve a `trigger_id` that -is unique to the Block Kit interactive component. - -Luckily for us, this is provided as a parameter to Block Kit Action Handlers! -You can use the value of `body.interactivity.interactivity_pointer` within an -action handler to open a view, like so: - -```typescript -export default SlackFunction(DiaryFunction, async ({ inputs, client }) => { - // ... the rest of your DiaryFunction logic here ... -}).addBlockActionsHandler( - "deny_request", - async ({ action, body, client }) => { - await client.views.open({ - trigger_id: body.interactivity.interactivity_pointer, - view: {/* your view object goes here */}, - }); - }, -); -``` - -### Adding view handlers - -The [Deno Slack SDK][sdk] - which comes bundled in your generated Run-on-Slack -application - provides a means for defining handlers to execute every time a -user interacts with a view. In this way you can route view-related events to -specific handlers inside your application. The key identifier that we'll need to -keep handy is the `callback_id` we assigned to any views we created. This ID -will be the property that determines which view event handler will respond to -incoming view events. - -Continuing with our above example, we can now define handlers that will listen -for view submission and closed events and respond accordingly. The code to add -view handlers is "chained" off of your top-level function, and would look like -this: - -```typescript -export default SlackFunction(DiaryFunction, async ({ inputs, client }) => { - // ... the rest of your DiaryFunction logic here ... -}).addViewSubmissionHandler( - /view/, // The first argument to any of the addView*Handler methods can accept a string, array of strings, or RegExp. - // This first argument will be used to match the view's `callback_id` - // Check the API reference at the end of this document for the full list of supported options - async ({ view, body, token }) => { // The second argument is the handler function itself - console.log("Incoming view submission handler invocation", body); - }, -) - .addViewClosedHandler( - /view/, - async ({ view, body, token }) => { - console.log("Incoming view closed handler invocation", body); - }, - ); -``` - -Importantly, more complex applications will likely be modifying views as users -interact with them: updating the view contents (to e.g. add new form fields), -perhaps pushing a new view onto the view stack to introduce a new UI to the -user, maybe reporting errors to the user for some manner of faulty interaction, -or even clearing the entire view stack altogether. All of these modal -interaction responses are -[covered in depth on our API documentation site][modifying] - make sure to spend -the time to understand the concepts presented there. - -In particular, modal interactions can be responded to by using the API, or by -returning particularly-crafted responses directly from inside the view handlers. -On our [API site detailing view modification][modifying], these returned view -handler responses are called `response_action`s. - -As an example, consider the following two code snippets. They yield identical -behavior! - -```typescript -export default SlackFunction(DiaryFunction, async ({ inputs, client }) => { - // ... the rest of your DiaryFunction logic here ... -}).addViewSubmissionHandler(/view/, async ({ client, body }) => { - // A view submission handler that pushes a new view using the API - await client.views.push({ - trigger_id: body.trigger_id, - view: {/* your view object goes here */}, - }); -}).addSubmissionHandler(/view/, async () => { - // A view submission handler that pushes a new view using the `response_action` - return { - response_action: "push", - view: {/* your view object goes here */}, - }; -}); -``` - -### API reference - -#### `addViewSubmissionHandler(constraint, handler)` - -```typescript -SlackFunction({ ... }).addViewSubmissionHandler("my_view_callback_id", async (ctx) => { ... }); -``` - -`addViewSubmissionHandler` registers a view handler based on a `constraint` -argument. If any incoming [`view_submission` event][view-events] matches the -`constraint`, then the specified `handler` will be invoked with the event -payload. This allows for authoring focussed, single-purpose view handlers and -provides a concise but flexible API for registering handlers to specific view -interactions. - -`constraint` can be either a string, an array of strings, or a regular -expression. - -- A simple string `constraint` must match a view's `callback_id` exactly. -- An array of strings `constraint` must match a view's `callback_id` to any of - the strings in the array. -- A regular expression `constraint` must match a view's `callback_id`. - -##### `addViewClosedHandler(constraint, handler)` - -```typescript -SlackFunction({ ... }).addViewClosedHandler("my_view_callback_id", async (ctx) => { ... }); -``` - -⚠️ IMPORTANT: you must set a view's `notify_on_close` property to `true` for the -`view_closed` event to trigger; by default this property is `false`. See the -[View reference documentation - in particular the Fields section][view-ref] for -more information. - -`addViewClosedHandler` registers a view handler based on a `constraint` -argument. If any incoming [`view_closed` event][view-events] matches the -`constraint`, then the specified `handler` will be invoked with the event -payload. This allows for authoring focussed, single-purpose view handlers and -provides a concise but flexible API for registering handlers to specific view -interactions. - -`constraint` can be either a string, an array of strings, or a regular -expression. - -- A simple string `constraint` must match a view's `callback_id` exactly. -- An array of strings `constraint` must match a view's `callback_id` to any of - the strings in the array. -- A regular expression `constraint` must match a view's `callback_id`. - -[functions]: ./functions.md -[action-handlers]: ./functions-action-handlers.md -[api]: https://api.slack.com/methods -[block-kit]: https://api.slack.com/block-kit -[interactivity]: https://api.slack.com/block-kit/interactivity -[sdk]: https://github.com/slackapi/deno-slack-sdk -[modals]: https://api.slack.com/surfaces/modals -[views]: https://api.slack.com/surfaces/modals/using -[modifying]: https://api.slack.com/surfaces/modals/using#modifying -[trigger-ids]: https://api.slack.com/interactivity/handling#modal_responses -[view-events]: https://api.slack.com/reference/interaction-payloads/views -[views-methods]: https://api.slack.com/methods?filter=views -[views-open]: https://api.slack.com/methods/views.open -[views-update]: https://api.slack.com/methods/views.update -[views-push]: https://api.slack.com/methods/views.push -[view-ref]: https://api.slack.com/reference/surfaces/views diff --git a/docs/functions.md b/docs/functions.md deleted file mode 100644 index 3898d51e..00000000 --- a/docs/functions.md +++ /dev/null @@ -1,169 +0,0 @@ -## Custom functions - -Functions are the core of your Slack app: they accept one or more input -parameters, execute some logic and return one or more output parameters. - -Functions can optionally define different kinds of Interactivity Handlers. If -your function creates messages or opens views, then it may need to define one or -more interactivity handlers to respond to user interactions with these -interactive components. Run-on-Slack applications support the following -interactivity handlers, follow the links to get more information about how each -of them work: - -- [Block Kit Action Handlers][action-handlers]: Handle events from interactive - [Block Kit][block-kit] components that you can use in messages like - [Buttons, Menus and Date/Time Pickers](https://api.slack.com/block-kit/interactivity) -- [View Handlers][view-handlers]: Handle events triggered from [Modals][modals], - which are composed of [Views][views]. -- [Block Kit Suggestion Handlers][suggest-handlers]: Handle events from - [external-data-sourced Block Kit select menus](https://api.slack.com/reference/block-kit/block-elements#external_select) - -### Defining a custom function - -Functions can be defined with the top level `DefineFunction` export. Below is an -example function that turns a `name` input parameter into a dinosaur name: - -```ts -import { DefineFunction, Schema } from "slack-cloud-sdk/mod.ts"; - -export const DinoFunction = DefineFunction({ - callback_id: "dino", - title: "Dino", - description: "Turns a name into a dinosaur name", - source_file: "functions/dino.ts", - input_parameters: { - name: { - type: Schema.types.string, - description: "The provided name", - }, - }, - output_parameters: { - dinoname: { - type: Schema.types.string, - description: "The new dinosaur name", - }, - }, -}); -``` - -Let's go over each of the arguments that must be provided to `DefineFunction`. - -#### Function definition - -The passed argument is the `definition` of the function, an object with a few -properties that help to describe and define the function in more detail. In -particular, the required properties of the object are: - -- `callback_id`: A unique string identifier representing the function (`"dino"` - in the above example). It must be unique in your application; no other - functions may be named identically. Changing a function's `callback_id` is not - recommended as it means that the function will be removed from the app and - created under the new `callback_id`, which will break any workflows - referencing the old function. -- `title`: A pretty string to nicely identify the function. -- `description`: A short-and-sweet string description of your function - succinctly summarizing what your function does. -- `source_file`: The relative path from the project root to the function - `handler` file. -- `input_parameters`: Itself an object which describes one or more input - parameters that will be available to your function. Each top-level property of - this object defines the name of one input parameter which will become - available to your function. The value for this property needs to be an object - with further sub-properties: - - `type`: The type of the input parameter. The supported types are `string`, - `integer`, `boolean`, `number`, `object` and `array`. - - `description`: A string description of the input parameter. -- `output_parameters`: Itself an object which describes one or more output - parameters that will be returned by your function. This object follows the - exact same pattern as `input_parameters`: top-level properties of the object - define output parameter names, with the property values being an object that - further describes the `type` and `description` of individual output - parameters. - -### Adding runtime logic to your custom function - -Now that you have defined your function's input and output parameters, it's time -to define the body of your function. - -First, create a new file at the location set on the `source_file` parameter of -your function definition. Next, let's add code for your function! You will want -to `export default` an instance of `SlackFunction`, like so: - -```typescript -import { SlackFunction } from "deno-slack-sdk/mod.ts"; - -export default SlackFunction( - // Pass along the function definition you created earlier using `DefineFunction` - DinoFunction, - ({ inputs }) => { // Provide any context properties, like `inputs`, `env`, or `token` - // Implement your function - const { name } = inputs; - const dinoname = `${name}asaurus`; - - // Don't forget any required output parameters - return { outputs: { dinoname } }; - }, -); -``` - -Key points: - -- The function takes a single argument, referred to as the - [function "context"](#function-handler-context). -- The function [returns an object](#function-return-object). - -#### Custom function handler context - -The single argument to your function is an object composed of several properties -that may be useful to leverage during your function's execution: - -- `env`: represents environment variables available to your function's execution - context. -- `inputs`: an object containing the input parameters you defined as part of - your function definition. In the example above, the `name` input parameter is - available on the `inputs` property of our function handler context. -- `client`: An API client ready for use in your function. An instance of the - `deno-slack-api` library. -- `token`: your application's access token. -- `team_id`: the encoded team (a.k.a. Slack workspace) ID, i.e. T12345. -- `enterprise_id`: the encoded enterprise ID, i.e. E12345. If the Slack - workspace the function executes in is not a part of an enterprise grid, then - this value will be the empty string (`""`). -- `event`: an object containing the full incoming event details. - -##### Custom function return object - -The object returned by your function that supports the following properties: - -- `error`: a string indicating the error that was encountered. If present, the - function will return an error regardless of what is passed to `outputs`. -- `outputs`: an object that exactly matches the structure of your function - definition's `output_parameters`. This is required unless an `error` is - returned. -- `completed`: a boolean indicating whether or not the function is completed. - This defaults to `true`. - -### Adding custom functions to the manifest - -Once you have defined a function, don't forget to include it in your -[`Manifest`][manifest] definition! - -```ts -import { ReverseString } from "./functions/reverse_definition.ts"; - -Manifest({ - name: "heuristic-tortoise", - description: "A demo showing how to use custom functions", - icon: "assets/icon.png", - botScopes: ["commands", "chat:write", "chat:write.public"], - functions: [ReverseString], // <-- don't forget this! -}); -``` - -[manifest]: ./manifest.md -[action-handlers]: ./functions-action-handlers.md -[view-handlers]: ./functions-view-handlers.md -[suggest-handlers]: ./functions-suggestion-handlers.md -[block-kit]: https://api.slack.com/block-kit -[modals]: https://api.slack.com/surfaces/modals -[views]: https://api.slack.com/surfaces/modals/using diff --git a/docs/guides/collaborating-with-teammates.md b/docs/guides/collaborating-with-teammates.md new file mode 100644 index 00000000..30d8a884 --- /dev/null +++ b/docs/guides/collaborating-with-teammates.md @@ -0,0 +1,51 @@ +--- +slug: /deno-slack-sdk/guides/collaborating-with-teammates +--- + +# Collaborating with teammates + + + +Have multiple developers working on an app? Never fear! Teams can collaborate when building and deploying workflow apps. + +## Deploy the app to make it available to collaborators {#deploy-to-collaborators} + +Let's say you're working along on an app and you realize it's going to need several types of triggers. Your teammate Luke is the resident trigger expert, so you ask if he'll jump in and help you out, to which he enthusiastically agrees—hooray! + +The first thing you'll do is to deploy your app using the `slack deploy` command. + +✨ **For directions on how to deploy your app**, refer to [deploy to Slack](/deno-slack-sdk/guides/deploying-to-slack). + +This will create an `apps.json` configuration file in your `.slack` folder (this folder may be hidden). This file contains information about your deployed apps, such as the installed workspace, the app name, the app ID, and the team ID. You'll want to check this file into version control in case you want to collaborate on the same deployed apps with others; if two or more people want to deploy or update the same app on the same workspace, the `apps.json` contents must be the same for everyone. + +## Add collaborators {#collaborators} + +Now, as long as Luke is in the same workspace as you, you can add Luke as a collaborator on your app right from the Slack CLI by entering the `slack collaborator add` command along with their email address or user ID. Choose your deployed environment, and voilà! Luke is now a collaborator on your app. To double-check, you can run the `slack collaborator list` command and choose your deployed environment—you should see yourself and Luke in the list. + +In the meantime, Luke can clone the GitHub repository containing the files that comprise your app, including the `apps.json` file, to their local machine. If Luke also wants to deploy the app to the same workspace as you, they will have to run the `slack login` command within that workspace. Once logged in, Luke will have the same access to run the `slack deploy` command (and other Slack CLI commands) as you. + +## Develop locally {#develop} + +Both you and Luke can now develop locally on your unique instances of the app. Use the `slack run` command while working to see your changes in real-time within your local environment. + +✨ **For information about developing locally**, refer to [local development](/deno-slack-sdk/guides/developing-locally). + +### App instances {#app-instances} + +Worthy of note is that there are now three instances of your app in existence: + +* The deployed version of the app that exists within your shared workspace +* Your local version of the app +* Luke's local version of the app + +:::info + +Both your local projects will include an `apps.dev.json` file in your respective `.slack` folders, which are unique to your app and your local development environments. These files are only for local development and **should not** be checked into version control. + +::: + +## Deploy updates to production {#deploy-to-production} + +Once you and Luke have completed development, either you or Luke can deploy the app to production by running the `slack deploy` command. + +However, be aware that this could lead to a situation in which Luke makes a change and runs the `slack deploy` command--and at the same time, you make a different, unrelated change and run the `slack deploy` command. Since you're both deploying the same app, the second deploy will overwrite the first. It is therefore important to either automate or coordinate your deployments with care. Teamwork makes the dream work! \ No newline at end of file diff --git a/docs/guides/creating-an-app.md b/docs/guides/creating-an-app.md new file mode 100644 index 00000000..c799c3c8 --- /dev/null +++ b/docs/guides/creating-an-app.md @@ -0,0 +1,150 @@ +--- +slug: /deno-slack-sdk/guides/creating-an-app +--- + +# Creating an app + + + +An app goes through stages of development, from creation to experimentation and development to production. Sometimes a [removal](/deno-slack-sdk/guides/removing-an-app) happens too. + +The [Slack CLI](/slack-cli) provides a set of commands to make managing these stages a bit easier with the following offerings: + +- `slack create` to [create a brand new app](#create-app) +- `slack app link` to [link an existing app to a project](#link-app) + +## Verify workspace authentication {#verify} + +Before you can create (or remove) an app, ensure your CLI is authenticated into the workspace you want to develop in. You can do so with the `slack auth list` command: + +```bash +$ slack auth list + +myworkspace (Team ID: T123456789) +User ID: U123456789 +Last update: 2022-03-24 18:20:47 -07:00 +Authorization Level: Workspace + +To change your active workspace authorization run slack login +``` + +## Create an app {#create-app} + +With your CLI authenticated into the workspace you want to develop in, the next step is to scaffold an app with the `slack create` command: + +```bash +$ slack create my-app +``` +The above command will scaffold a new app called `my-app` in a directory with the same name. If you don't pass an app name, `slack` will scaffold an app with a random alphanumeric name. + +You will be presented with three options to build from: +* The introductory [Issue Submission](https://github.com/slack-samples/deno-issue-submission) app +* The scaffolded [Deno Starter Template](https://github.com/slack-samples/deno-starter-template) +* The completely blank [Deno Blank Template](https://github.com/slack-samples/deno-blank-template) + +:::tip +If you'd like to build from a specific sample app, see [Create an app from a template](#templates) below. +::: + +```bash +? Select a template to build from: [Use arrows to move] + +> Issue submission (default sample) + Basic app that demonstrates an issue submission workflow + + Scaffolded project + Solid foundation that includes a Slack datastore + + Blank project + A, well.. blank project + + View more samples + + Guided tutorials can be found at api.slack.com/automation/samples + ``` + +Once you select an option the Slack CLI will get you set up for success. + +```bash +$ slack create my-app + +⚙️ Creating a new Slack app in ~/programming/my-app + +📦 Installed project dependencies + +✨ my-app successfully created + +🧭 Explore the documentation to learn more + Read the README.md or peruse the docs over at api.slack.com/automation + Find available commands and usage info with `slack help` + +📋 Follow the steps below to begin development + Change into your project directory with `cd my-app` + Develop locally and see changes in real-time with `slack run` + When you're ready to deploy for production use `slack deploy` + Create a trigger to invoke your workflows `slack trigger create` +``` + +After creating an app, don't forget to `cd` into your app project's directory. + +➡️ **To keep building your own app**, learn about your app's manifest in the [manifest](/deno-slack-sdk/guides/using-the-app-manifest) section. + +⤵️ **To use a sample app as a template instead**, read on! + +### Create an app from a template {#templates} + +:::warning[Evaluate third-party apps] + +Exercise caution before trusting third-party and open source applications and automations (those outside of [`slack-samples`](https://github.com/slack-samples)). Review all source code created by third-parties before running `slack create` or `slack deploy`. + +::: + +We have a [collection of sample apps](https://github.com/slack-samples) containing a bevy of use cases. Find one particularly suited for your needs? Great! You can use it as a template to build from. + +Create an app from a template by using the `create` command with the `--template` (or `-t`) flag and passing the link to the template's Github repo. + +For example, the following command creates an app using our [Welcome Bot](https://github.com/slack-samples/deno-welcome-bot) app as a template: + +```bash +slack create my-welcome-bot-app -t https://github.com/slack-samples/deno-welcome-bot +``` + +#### Create an app from a specific branch {#branch} + +Use a specific branch of a template repo by using the `--branch` (or `-b`) flag and passing the name of a branch: + +```bash +slack create my-welcome-bot-app -t https://github.com/slack-samples/deno-welcome-bot -b main +``` + +--- + +## Link an app {#link-app} + +Apps that were created without the CLI can still be used with the CLI for ease in app management. This requires the `app link` command to save information about the app with the project it exists on: + +```bash +slack app link --app A0123456789 --team T0123456789 --environment "deployed" +``` + +```bash +🏠 An existing app was added to the project + my-workspace: + App ID: A0123456789 + Team ID: T0123456789 + Status: Installed +``` + +This command can be used without flags, but the above example would use the app ID "A01234567890" from team "T01234567890" - the team the app was created on - as a "deployed" app. Following selections in commands will reveal this app as an option and it's all set for CLI use! + +Another `--environment` option is "local" and this saves apps to `.slack/apps.dev.json`. Local apps are intended for personal development and experimentation while "deployed" apps - saved to `.slack/apps.json` - serve the needs of production. + +Just one app ID can exist for each combination of team ID and environment to avoid accidental selections with duplicated possibilities. The provided app ID must also exist for this `link` command to complete with success. + +--- + +## Onward + +Your wish is our command! For information about other actions you can perform from the CLI, refer to the [Slack CLI commands guide](/slack-cli/guides/running-slack-cli-commands). + +Check out more about how to configure your app via the [manifest](/deno-slack-sdk/guides/using-the-app-manifest). You can also start building [functions](/deno-slack-sdk/guides/creating-slack-functions) to perform some logic, chain them together in [workflows](/deno-slack-sdk/guides/creating-workflows), and create [triggers](/deno-slack-sdk/guides/using-triggers) to invoke those workflows. \ No newline at end of file diff --git a/docs/guides/creating-workflows.md b/docs/guides/creating-workflows.md new file mode 100644 index 00000000..443b8ddb --- /dev/null +++ b/docs/guides/creating-workflows.md @@ -0,0 +1,318 @@ +--- +slug: /deno-slack-sdk/guides/creating-workflows +--- + +# Creating workflows + + + +Workflows are the combination of functions, executed in order. Remember: + +1. Both [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) and [custom functions](/deno-slack-sdk/guides/creating-custom-functions) define the actions of your app. +2. Workflows are a combination of functions, executed in order. (⬅️ you are here) +3. Triggers execute workflows. + +Depending on your use case, you'll want to acquaint yourself with either [Slack functions](/deno-slack-sdk/guides/creating-slack-functions), [custom functions](/deno-slack-sdk/guides/creating-custom-functions), or both. Then continue here to learn how to implement them in a workflow. + +We'll walk through defining a workflow, adding input and output parameters, adding both a Slack function and a custom function to the workflow, and declaring the workflow in your manifest. + +## Defining workflows {#defining-workflows} + +Workflows are defined in their own files within your app's `/workflows` directory and declared in your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). Listing workflows in your manifest tells the CLI that they are implemented in your app — more on that later. + +Before defining your workflow, import [`DefineWorkflow`](https://github.com/slackapi/deno-slack-sdk/blob/main/src/workflows/mod.ts) at the top of your workflow file: + +```javascript +// say_hello_workflow.ts +import { DefineWorkflow } from "deno-slack-sdk/mod.ts"; +``` + +Then, create a **workflow definition**. This is where you set, at a minimum, the workflow's title and its unique callback ID: + +```javascript +// say_hello_workflow.ts +export const SayHelloWorkflow = DefineWorkflow({ + callback_id: "say_hello_workflow", + title: "Say Hello", +}); +``` + +| Definition properties | Description | Required? | +| ------------------| ---------- | ------ | +| `callback_id` | A unique string that identifies this particular component of your app. | Required | +| `title` | The display name of the workflow that shows up in slugs, unfurl cards, and certain end-user modals. | Required | +| `description` | A string description of this workflow. | Optional | +| `input_parameters` | See [Defining input parameters](#defining-input-parameters). | Optional | + +In the next section, we'll look at `input_parameters` in more detail. +## Defining input parameters {#defining-input-parameters} + +Workflows can pass information into both functions and other workflows that are part of its workflow steps. To do this, we define what information we want to bring in to the workflow via its `input_parameters` property. + +A workflow's `input_parameters` property has two sub-properties: +* `required`, which is how you can ensure that a workflow only executes if specific input parameters are provided. +* `properties`, where you can list the specific parameters that your workflow accounts for. Any [built-in type](/deno-slack-sdk/reference/slack-types) or [custom type](/deno-slack-sdk/guides/creating-a-custom-type) can be used. + +Input parameters are listed in the `properties` sub-property. Each input parameter must include a `type` and a `description`, and can optionally include a `default` value. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +// Workflow definition +export const SomeWorkflow = DefineWorkflow({ + callback_id: "some_workflow", + title: "Some Workflow", + input_parameters: { + required: [], + properties: { + exampleString: { + type: Schema.types.string, + description: "Here's an example string.", + }, + exampleBoolean: { + type: Schema.types.boolean, + description: "An example boolean.", + default: true, + }, + exampleInteger: { + type: Schema.types.integer, + description: "An example integer.", + }, + exampleChannelId: { + type: Schema.slack.types.channel_id, + description: "Example channel ID.", + }, + exampleUserId: { + type: Schema.slack.types.user_id, + description: "Example user ID.", + }, + exampleUsergroupId: { + type: Schema.slack.types.usergroup_id, + description: "Example usergroup ID.", + }, + }, + }, +}); +``` + +Denote which properties are required by listing their names as strings in the `required` property of `input_parameters`. For example, here's how we can indicate that a parameter named `exampleUserId` is required: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +// Workflow definition +export const SomeWorkflow = DefineWorkflow({ + callback_id: "some_workflow", + title: "Some Workflow", + input_parameters: { + required: ["exampleUserId"], + properties: { + exampleUserId: { + type: Schema.slack.types.user_id, + description: "Example user ID.", + }, + }, + }, +}); +``` + +If a workflow is invoked and the required input parameters are not provided, the workflow will not execute. + +An important distinction: `input_parameters` are used when _defining_ a workflow, whereas _retrieving_ values will use `inputs`. `inputs` is also used when implementing the logic of a custom function. + +Once you've defined your workflow, you can then add functionality by calling Slack functions and custom functions. This is done with the `addStep` method, which takes two arguments: + +* the function you want to call +* the inputs (if any) you want to pass to that function. + +We'll see examples of how to call both types of functions in the following section. + +--- + +## Adding functions to workflows {#adding-functions} + +### Import Schema reference {#import-schema} + +The first step to adding a function to a workflow is to import `Schema` from the Slack SDK. + +```javascript +// /workflows/greeting_workflow.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +``` + +### Call a function with `addStep` {#call-function} + +#### Slack functions{#workflow-slack-functions} + +> [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) are essentially Slack-native actions, like creating a channel or sending a message. + +To use a Slack function, like [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message), let's look at an example from the [Deno Hello World](https://github.com/slack-samples/deno-hello-world) sample app. + +After defining the workflow, call the Slack function with your workflow's `addStep` method: + +```javascript +const GreetingWorkflow = DefineWorkflow({ + callback_id: "greeting_workflow", + title: "Send a greeting", + description: "Send a greeting to channel", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + required: ["interactivity"], + }, +}); + +const inputForm = GreetingWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Send a greeting", + interactivity: GreetingWorkflow.inputs.interactivity, + submit_label: "Send greeting", + fields: { + elements: [{ + name: "recipient", + title: "Recipient", + type: Schema.slack.types.user_id, + }, { + name: "channel", + title: "Channel to send message to", + type: Schema.slack.types.channel_id, + default: GreetingWorkflow.inputs.channel, + }, { + name: "message", + title: "Message to recipient", + type: Schema.types.string, + long: true, + }], + required: ["recipient", "channel", "message"], + }, + }, +); + +//...call GreetingFunctionDefinition in greetingFunctionStep + +// Example: taking the string output from the greetingFunctionStep function and passing it to SendMessage +GreetingWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: inputForm.outputs.fields.channel, + message: greetingFunctionStep.outputs.greeting, +}); +``` + +##### Using OpenForm in a workflow {#using-forms} + +The only Slack function that has an additional requirement is [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form). When creating a workflow that will have a step to open a form, your workflow needs to: + +* include the `interactivity` input parameter +* have the call to `OpenForm` be its **first** step _or_ ensure the preceding step is interactive. An interactive step will generate a fresh pointer to use for opening the form; for example, use the interactive button that can be added with the [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message) Slack function immediately before opening the form. + +Here's an example of a basic workflow definition using `interactivity`: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const SayHelloWorkflow = DefineWorkflow({ + callback_id: "say_hello_workflow", + title: "Say Hello to a user", + input_parameters: { + properties: { interactivity: { type: Schema.slack.types.interactivity } }, + required: ["interactivity"], + }, +}); +``` + +✨ Visit the [forms](/deno-slack-sdk/guides/creating-a-form) section for more details and code examples of using `OpenForm` in your app. + +#### Custom functions{#workflow-custom-functions} + +>[Custom functions](/deno-slack-sdk/guides/creating-custom-functions) are reusuable building blocks of automation of your own design. + +To use a [custom function](/deno-slack-sdk/guides/creating-custom-functions) that you have already defined: + +1. Import the function in your manifest, where you define the workflow: + +```javascript +import { SomeFunction } from "../functions/some_function.ts"; +``` + +2. Call your function, storing its output in a variable. Here you may also pass input parameters from the workflow into the function itself: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { SomeFunction } from "../functions/some_function.ts"; + +export const SomeWorkflow = DefineWorkflow({ + callback_id: "some_workflow", + title: "Some Workflow", + input_parameters: { + properties: { + someString: { + type: Schema.types.string, + description: "Some string", + }, + channelId: { + type: Schema.slack.types.channel_id, + description: "Target channel", + default: "C1234567", + }, + }, + required: [], + }, +}); + +const myFunctionResult = SomeWorkflow.addStep(SomeFunction, { + // ... Pass along workflow inputs via SomeWorkflow.inputs + // ... For example, SomeWorkflow.inputs.someString +}); +``` + +3. Use your function in follow-on steps. For example: + +```javascript +// Example: taking the string output from a function and passing it to SendMessage +SomeWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: SomeWorkflow.inputs.channelId, + message: SomeFunction.outputs.exampleOutput, // This comes from your function definition +}); +``` + +Once you've added all steps and functions to your workflow, there's one final stop to having a fully functioning workflow — adding it to the [app manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +--- +## Adding the workflow to the manifest {#add-workflow} + +The final step of using a workflow is adding it to your [manifest](/deno-slack-sdk/guides/using-the-app-manifest). Declare your workflow in your app's manifest definition of your manifest file like this: + +```javascript +// manifest.ts +import { Manifest } from "deno-slack-sdk/mod.ts"; +import { SayHelloWorkflow } from "./workflows/say_hello_workflow.ts"; + +export default Manifest({ + name: "sayhello", + description: "A deno app with an example workflow", + icon: "assets/icon.png", + workflows: [SayHelloWorkflow], // Add your workflow here + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +:::info + +The workflows guest or external users can run is based on whether those workflows run functions that are defined with certain scopes. Refer to [guests and external users](/deno-slack-sdk/guides/controlling-access-to-custom-functions#guests-external) for more details. + +::: + +--- +## Onward + + +➡️ **To keep building your app**, head to the [triggers](/deno-slack-sdk/guides/using-triggers) section to learn how to create a trigger that invokes a defined workflow. + +You can also learn about [creating a datastore](/deno-slack-sdk/guides/using-datastores) to store and retrieve information, or building [custom types](/deno-slack-sdk/guides/creating-a-custom-type) for your data. \ No newline at end of file diff --git a/docs/guides/datastores/adding-items-to-a-datastore.md b/docs/guides/datastores/adding-items-to-a-datastore.md new file mode 100644 index 00000000..ad1eeb68 --- /dev/null +++ b/docs/guides/datastores/adding-items-to-a-datastore.md @@ -0,0 +1,336 @@ +--- +slug: /deno-slack-sdk/guides/adding-items-to-a-datastore +--- + +# Adding items to a datastore + + + +There are a few ways you can add information to a datastore. You can: +- [Create or replace items with `put` and `bulkPut`](#create-replace) +- [Create or update items with `update`](#update) + +There's an important distinction between these methods! The `put` and `bulkPut` methods _replace_ entire existing items, while the `update` method will only _update_ the provided attributes for items. Be careful to not accidentally lose information when using the `put` and `bulkPut` methods. + +:::tip[Slack CLI commands] +You can also add items to a datastore with the [`datastore put`](/slack-cli/reference/commands/slack_datastore_put), [`datastore bulk-put`](/slack-cli/reference/commands/slack_datastore_bulk-put), and [`datastore update`](/slack-cli/reference/commands/slack_datastore_update) Slack CLI commands. The `datastore bulk-put` command even supports importing data from a [JSON Lines](https://jsonlines.org/) file. + +::: + +## Create or replace items with `put` and `bulkPut` {#create-replace} + +There are two methods for creating and replacing items in datastores: +- The [`apps.datastore.put`](https://api.slack.com/methods/apps.datastore.put) method is best for single items. +- The [`apps.datastore.bulkPut`](https://api.slack.com/methods/apps.datastore.bulkPut) method is best for multiple items. + +They work quite similarly. + +
+Example: Using the put method + +```js + const putResp = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + item: { + id: draftId, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + }); +``` +
+ +
+Example: Using the bulkPut method + +```js + const putResp = await client.apps.datastore.bulkPut< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + items: [ + { + id: draftId, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + { + id: draftId2, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + ] + }); +``` +
+ +That's the general format for each method - but let's look a full example to help connect the dots. + +In this example, we create a custom function that creates and sends an announcement draft to a channel. Values for each of the datastore's attributes are passed in to create that announcement draft. + +First is the custom function definition: + +```js +// /functions/create_draft/definition.ts +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const CREATE_DRAFT_FUNCTION_CALLBACK_ID = "create_draft"; +export const CreateDraftFunctionDefinition = DefineFunction({ + callback_id: CREATE_DRAFT_FUNCTION_CALLBACK_ID, + title: "Create a draft announcement", + description: + "Creates and sends an announcement draft to channel for review before sending", + source_file: "functions/create_draft/handler.ts", + input_parameters: { + properties: { + created_by: { + type: Schema.slack.types.user_id, + description: "The user that created the announcement draft", + }, + message: { + type: Schema.types.string, + description: "The text content of the announcement", + }, + channel: { + type: Schema.slack.types.channel_id, + description: "The channel where the announcement will be drafted", + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The channels where the announcement will be posted", + }, + icon: { + type: Schema.types.string, + description: "Optional custom bot icon to use display in announcements", + }, + username: { + type: Schema.types.string, + description: "Optional custom bot emoji avatar to use in announcements", + }, + }, + required: [ + "created_by", + "message", + "channel", + "channels", + ], + }, + output_parameters: { + properties: { + draft_id: { + type: Schema.types.string, + description: "Datastore identifier for the draft", + }, + message: { + type: Schema.types.string, + description: "The content of the announcement", + }, + message_ts: { + type: Schema.types.string, + description: "The timestamp of the draft message in the Slack channel", + }, + }, + required: ["draft_id", "message", "message_ts"], + }, +}); +``` + +Next we have the handler function for `CreateDraftFunction`. With it, we create a new datastore record using the `put` method: + +```js +// /functions/create_draft/handler.ts +import { SlackFunction } from "deno-slack-sdk/mod.ts"; + +import { CreateDraftFunctionDefinition } from "./definition.ts"; +import { buildDraftBlocks } from "./blocks.ts"; +import { + confirmAnnouncementForSend, + openDraftEditView, + prepareSendAnnouncement, + saveDraftEditSubmission, +} from "./interactivity_handler.ts"; +import { ChatPostMessageParams, DraftStatus } from "./types.ts"; + +import DraftDatastore from "../../datastores/drafts.ts"; + +/** + * This is the handling code for the CreateDraftFunction. It will: + * 1. Create a new datastore record with the draft + * 2. Build a Block Kit message with the draft and send it to input channel + * 3. Update the draft record with the successful sent drafts timestamp + * 4. Pause function completion until user interaction + */ +export default SlackFunction( + CreateDraftFunctionDefinition, + async ({ inputs, client }) => { + const draftId = crypto.randomUUID(); + + // 1. Create a new datastore record with the draft + const putResp = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + item: { + id: draftId, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + }); + + if (!putResp.ok) { + const draftSaveErrorMsg = + `Error saving draft announcement. Contact the app maintainers with the following information - (Error detail: ${putResp.error})`; + console.log(draftSaveErrorMsg); + + return { error: draftSaveErrorMsg }; + } + ... +``` + +If the call was successful, the payload's `ok` property will be `true`, and the `item` or `items` property will return a copy of the data you just inserted: + +```json +{ + "ok": true, + "datastore": "drafts", + "item": { + "id": "906dba92-44f5-4680-ada9-065149e4e930", + "created_by": "U045A5X302V", + "message": "This is a test message", + "channels": ["C039ARY976C"], + "channel": "C038M39A2TV", + "icon": "", + "username": "Slackbot", + "status": "draft", + } +} +``` + +If the call was not successful, the payload's `ok` property will be `false`, and you will have a error `code` and `message` property available: + +```json +{ + "ok": false, + "error": "datastore_error", + "errors": [ + { + "code": "some_error_code", + "message": "A description of the error", + "pointer": "/datastore/drafts" + } + ] +} +``` + +:::warning[Datastore bulk API methods may _partially_ fail] + +The `partial_failure` error message indicates that some items were successfully processed while others need to be retried. This is likely due to rate limits. Call the method again with only those failed items. + +You'll find a `failed_items` array within the API response. The array contains all the items that failed, in the same format they were passed in. Copy the `failed_items` array and use it in your request. + +::: + +If you're adding new data via the `put` or `bulkPut` method, provide each item with a new primary key value in the `id` property. If you're updating an existing items, provide the `id` of each item you wish to replace. Note that a `put` or `bulkPut` request replaces each entire specified object, if it exists. + +:::info["This datastore size is _just right_"] + +The total allowable size of an item (all fields in a record) must be less than 400 KB. + +::: + +## Create or update an item with `update` {#update} + +Updating only some of an item's attributes is done with the [`apps.datastore.update`](https://api.slack.com/methods/apps.datastore.update) API method. Let's see how that works by passing in values for only some of the datastore's attributes: + +```js +// /functions/create_draft_interactivity_handler.ts +... +export const saveDraftEditSubmission: ViewSubmissionHandler< + typeof CreateDraftFunction.definition +> = async ( + { inputs, view, client }, +) => { + // Get the datastore draft ID from the modal's private metadata + const { id, thread_ts } = JSON.parse(view.private_metadata || ""); + + const message = view.state.values.message_block.message_input.value; + + // Update the saved message + const updateResp = await client.apps.datastore.update({ + datastore: DraftDatastore.name, + item: { + id: id, + message: message, // This call will update only the message of the draft announcement + }, + }); + + if (!updateResp.ok) { + const updateDraftMessageErrorMsg = + `Error updating draft ${id} message. Contact the app maintainers with the following - (Error detail: ${putResp.error})`; + console.log(updateDraftMessageErrorMsg); + return; + } + ... +``` + +If the call was successful, the payload's `ok` property will be `true`, and the `item` property will return a copy of the updated data: + +```json +{ + "ok": true, + "datastore": "drafts", + "item": { + "id": "906dba92-44f5-4680-ada9-065149e4e930", + "created_by": "U045A5X302V", + "message": "This is a message that will be sent", + "channels": ["C039ARY976C"], + "channel": "C038M39A2TV", + "icon": "", + "username": "Slackbot", + "status": "draft", + } +} +``` + +If the call was not successful, the payload's `ok` property will be `false`, and you will have a error `code` and `message` property available: + +```json +{ + "ok": false, + "error": "datastore_error", + "errors": [ + { + "code": "some_error_code", + "message": "A description of the error", + "pointer": "/datastore/drafts" + } + ] +} +``` + +If an item with the provided `id` doesn't exist in the datastore, `update` will insert the item using the provided attributes. diff --git a/docs/guides/datastores/deleting-items-from-a-datastore.md b/docs/guides/datastores/deleting-items-from-a-datastore.md new file mode 100644 index 00000000..468ab0b0 --- /dev/null +++ b/docs/guides/datastores/deleting-items-from-a-datastore.md @@ -0,0 +1,213 @@ +--- +slug: /deno-slack-sdk/guides/deleting-items-from-a-datastore +--- + +# Deleting items from a datastore + + + +There are a couple ways you can delete items from a datastore. You can: +- [Delete items with `delete` and `bulkDelete`](#delete) +- [Delete items automatically](#delete-automatically) + +:::tip[Slack CLI commands] + +You can also delete items from a datastore with the [`datastore delete`](/slack-cli/reference/commands/slack_datastore_delete) and [`datastore bulk-delete`](/slack-cli/reference/commands/slack_datastore_bulk-delete) Slack CLI commands. + +::: + +## Delete items with `delete` and `bulkDelete` {#delete} + +There are two methods for deleting items in datastores: +- The [`apps.datastore.delete`](https://api.slack.com/methods/apps.datastore.delete) method is used for single items. +- The [`apps.datastore.bulkDelete`](https://api.slack.com/methods/apps.datastore.bulkDelete) method is used for multiple items. + +They work quite similarly. In the following examples we'll be deleting items from a datastore via their primary key. Regardless of what you named your `primary_key`, the query will always use the `id` key. + +
+Example: Using the delete method to delete an item by its primary_key + +```js +// Somewhere in your function: +const uuid = "6db46604-7910-4684-b706-ac5929dd16ef"; +const response = await client.apps.datastore.delete({ + datastore: "drafts", + id: uuid, +}); + +if (!response.ok) { + const error = `Failed to delete a row in datastore: ${response.error}`; + return { error }; +} +``` +
+ +
+Example: Using the bulkDelete method to delete an item by its primary_key + +```js +// Somewhere in your function: +const uuid = "6db46604-7910-4684-b706-ac5929dd16ef"; +const uuid2 = "1111111-1111-1111-1111-111111111111"; +const response = await client.apps.datastore.bulkDelete({ + datastore: "drafts", + ids: [uuid,uuid2] +}); + +if (!response.ok) { + const error = `Failed to delete a row in datastore: ${response.error}`; + return { error }; +} +``` +
+ +If the call was successful, the payload's `ok` property will be `true`. If it is not successful, it will be `false` and provide an error in the `errors` property. + +:::warning[Datastore bulk API methods may _partially_ fail] + +The `partial_failure` error message indicates that some items were successfully processed while others need to be retried. This is likely due to rate limits. Call the method again with only those failed items. + +You'll find a `failed_items` array within the API response. The array contains all the items that failed, in the same format they were passed in. Copy the `failed_items` array and use it in your request. + +::: + +## Delete items automatically {#delete-automatically} + +You can set up your datastore to automatically delete records which are old and no longer relevant. This is done with the Time To Live (TTL) feature offered by AWS DynamoDB. Use it to efficiently discard data your app no longer needs. + +For any item, define an expiration timestamp and the item will be automatically deleted once that expiration time has passed. + +Notice that we didn't say _immediately deleted_. AWS only guarantees deletion of expired items _48 hours past the expiration date_. If you query your table before 48 hours have passed, do not assume all expired items have been deleted. You can read more about this within the [AWS documentation](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/ttl-expired-items.html). + +See below for an example on querying a database while filtering out any remaining expired items. + +### Enable and utilize the TTL feature {#enable-ttl} + +##### Step 1. Select an attribute to use as the expiration timestamp + +You can use a pre-existing attribute or add a new attribute. + +The attribute's type _must_ be set as `Schema.slack.types.timestamp` in the datastore definition. + +In this example, we're using an attribute called `expire_ts`: + +```js +// /datastores/drafts.ts +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export default DefineDatastore({ + name: "drafts", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + expire_ts: { + type: Schema.slack.types.timestamp // This line! + } + }, + ... +}); +``` + +##### Step 2. Set `time_to_live_attribute` to the selected attribute in the datastore definition + +```js +// /datastores/drafts.ts +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export default DefineDatastore({ + name: "drafts", + time_to_live_attribute: "expire_ts" // This line! + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + expire_ts: { + type: Schema.slack.types.timestamp + }, + message: { + type: Schema.types.string, + } + }, + ... +}); +``` + +##### Step 3. Set `expire_ts` to a value, either programmatically or manually + +In this example we're adding an item containing the `expire_ts` key using the [`apps.datastore.put`](https://api.slack.com/methods/apps.datastore.put) method: + +```js + const expiration = + const putResp = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + item: { + id: draftId, + expire_ts: 23456432345, + message: "Congrats on the promotion Jesse!" + }, + }); +``` + +##### Step 4. Deploy your app + +Use the `slack deploy` command. See [Deploy to Slack](/deno-slack-sdk/guides/deploying-to-slack) for more information. + +##### Step 5. Properly query items + +As mentioned [above](#delete-automatically), expired items may not be deleted immediately. You'll likely want to filter out those expired items. + +Here is an example of a query that filters out any expired items that have not been automatically deleted yet: + +```js +const result = await client.apps.datastore.query({ + datastore: "DraftDatastore", + expression: "attribute_not_exists(#expire_ts) OR #expire_ts > :timestamp", + expression_attributes: { "#expire_ts": "expire_ts" }, + expression_values: { ":timestamp":1708448410 } //Timestamp should be the current time +}); +``` + +To see an example of filtering out expired items via the command line, see the documentation on the [`datastore query`](/slack-cli/reference/commands/slack_datastore_query) command. + +### Disable the TTL feature {#disable-ttl} + +##### Step 1. Remove `time_to_live_attribute` in the datastore definition + +We only commented it out here because showing the absence of something is a bit anticlimactic. + +```JS +export default DefineDatastore({ + name: "drafts", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + //expire_ts: { + // type: Schema.slack.types.timestamp + //} + }, + ... +}); +``` + +##### Step 2. Deploy your app. + +Use the `slack deploy` command. See [Deploy to Slack](/deno-slack-sdk/guides/deploying-to-slack) for more information. + +### Change the TTL attribute {#change-ttl} + +Due to AWS limitations, changing the TTL attribute is a bit clunky. + +Step 1. [Disable TTL](#disable-ttl) + +Step 2. Wait one hour. This wait is because AWS puts time limits on additional changes to the TTL feature. + +Step 3. [Enable TTL again with the new attribute](#enable-ttl) + +Don't forget to deploy both when disabling and enabling TTL! diff --git a/docs/guides/datastores/retrieving-items-from-a-datastore.md b/docs/guides/datastores/retrieving-items-from-a-datastore.md new file mode 100644 index 00000000..387ac687 --- /dev/null +++ b/docs/guides/datastores/retrieving-items-from-a-datastore.md @@ -0,0 +1,489 @@ +--- +slug: /deno-slack-sdk/guides/retrieving-items-from-a-datastore +--- + +# Retrieving items from a datastore + + + +:::tip[Slack CLI commands] +You can also retrieve items from a datastore with the [`datastore get`](/slack-cli/reference/commands/slack_datastore_query), [`datastore bulk-get`](/slack-cli/reference/commands/slack_datastore_bulk-get), and [`datastore query`](/slack-cli/reference/commands/slack_datastore_query) Slack CLI commands. The `datastore query` command even supports exporting data to a [JSON Lines](https://jsonlines.org/) file. + +::: + +## Retrieve items with `get` and `bulkGet` {#get} + +There are two methods for retrieving items in datastores: +- The [`apps.datastore.get`](https://api.slack.com/methods/apps.datastore.get) method is used for single items. +- The [`apps.datastore.bulkGet`](https://api.slack.com/methods/apps.datastore.bulkGet) method is used for multiple items. + +They work quite similarly. Regardless of what you named your `primary_key`, the query will always use the `id` key. + +
+Example: Using the get method to retrieve an item by its primary_key + +```js +// /functions/create_draft/interactivity_handler.ts +... +export const openDraftEditView: BlockActionHandler< + typeof CreateDraftFunction.definition +> = async ({ body, action, client }) => { + if (action.selected_option.value == "edit_message_overflow") { + const id = action.block_id; + + // Get the draft + const getResp = await client.apps.datastore.get < + typeof DraftDatastore.definition + > ( + { + datastore: DraftDatastore.name, + id: id, + }, + ); +... +``` +If the call was successful and data was found, the `item` property in the payload will include the attributes (and their values) from the datastore definition. + +```json +{ + "ok": true, + "datastore": "drafts", + "item": { + "id": "906dba92-44f5-4680-ada9-065149e4e930", + "created_by": "U045A5X302V", + "message": "This is a test message", + "channels": [ + "C039ARY976C" + ], + "channel": "C038M39A2TV", + "icon": "", + "username": "Slackbot", + "status": "draft", + } +} +``` + +If the call was successful but no data was found, the `item` property in the payload will be blank: + +```json +{ + "ok": true, + "datastore": "drafts", + "item": {} +} +``` + +
+ +
+Example: Using the bulkGet method to retrieve multiple items by their primary_key + +```js +// /functions/create_draft/interactivity_handler.ts +... +export const openDraftEditView: BlockActionHandler< + typeof CreateDraftFunction.definition +> = async ({ body, action, client }) => { + if (action.selected_option.value == "edit_message_overflow") { + const id = action.block_id; + + // Get the draft + const getResp = await client.apps.datastore.bulkGet < + typeof DraftDatastore.definition + > ( + { + datastore: DraftDatastore.name, + ids: [id, "41"] + }, + ); +... +``` +If multiple items are returned, the `item` properties will be contained in an `items` array + +```json +{ + "ok": true, + "datastore": "drafts", + "items": [ + { + "id": "906dba92-44f5-4680-ada9-065149e4e930", + "created_by": "U045A5X302V", + "message": "This is a test message", + "channels": [ + "C039ARY976C" + ], + "channel": "C038M39A2TV", + "icon": "", + "username": "Slackbot", + "status": "draft", + }, + { + "id": "906dba92-44f5-4680-ada9-065149e4e930", + "created_by": "U045A5X302V", + "message": "This is a test message", + "channels": [ + "C039ARY976C" + ], + "channel": "C038M39A2TV", + "icon": "", + "username": "Slackbot", + "status": "draft", + } + ] +} +``` + +If the call was successful but no data was found, the `items` property in the payload will be blank: + +```json +{ + "ok": true, + "datastore": "drafts", + "items": [] +} +``` +
+ +For both methods, if the call was unsuccessful, `ok` will be false and you'll see some information on the error. + +```json +{ + "ok": false, + "error": "datastore_error", + "errors": [ + { + "code": "datastore_config_not_found", + "message": "The datastore configuration could not be found", + "pointer": "/datastores" + } + ] +} +``` + +It is possible to have records with undefined values, and it's important to be proactive in expecting those situations in your code. Here are some examples of how to code around a potential undefined field while retrieving an item. + +This example snippet supports the case where the function returns an optional output: + +```js +const getResponse = await client.apps.datastore.get < typeof DraftsDatastore.definition > ({ ...}); +const announcementId = getResponse.item.id; // this is the primary key +const announcementIcon = getResponse.item.icon; // icon could be undefined + +return { + outputs: { + id: announcementId, // id is always defined + icon: announcementIcon, // icon must be an optional output of the function + } +} +``` + +This example snippet supports the case where the function assigns a default: + +```js +const getResponse = await client.apps.datastore.get < typeof DraftsDatastore.definition > ({ ...}); +const announcementId = getResponse.item.id; // this is the primary key + +// icon could be undefined, so use a fallback +const announcementIcon = getResponse.item.icon ?? "n/a"; + +return { + outputs: { + id: announcementId, // id is always defined + icon: announcementIcon, // email is always defined + } +} +``` + +And finally, this example snippet supports the case where the function should error: + +```js +const getResponse = await client.apps.datastore.get < typeof DraftsDatastore.definition > ({ ...}); +const announcementId = getResponse.item.id; // this is the primary key + +if (getResponse.item.icon) { + const announcementIcon = getResponse.item.icon; + return { + outputs: { + id: announcementId, + icon: announcementIcon + } + } +} else { + return { + error: "Announcement doesn't have an icon assigned" + } +} +``` + +:::warning[Datastore bulk API methods may _partially_ fail] + +The `partial_failure` error message indicates that some items were successfully processed while others need to be retried. This is likely due to rate limits. Call the method again with only those failed items. + +You'll find a `failed_items` array within the API response. The array contains all the items that failed, in the same format they were passed in. Copy the `failed_items` array and use it in your request. + +::: + +## Find items with `query` {#find} + +If you need to find data without already knowing the item's `id`, you'll want to run a query. Querying a datastore requires knowledge of a few different components. It's also helpful to brush up on how to use [pagination](#pagination) and [filter expressions](#filter-expressions). + +First, let's look at the fields of a datastore query and how they might look in code, then break down the details of each bit. + +A Slack datastore query includes the following arguments: + +| Parameter | Description | Required | +| --------- | ----------- | -------- | +| `datastore` | A string with the name of the datastore to read the data from | Required | +| `expression` | A DynamoDB filter expression, using DynamoDB's [filter expression syntax](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Scan.html#Scan.FilterExpression)| Optional | +| `expression_attributes` | A map of columns used by the `expression` | Optional | +| `expression_values` | A map of values used by the `expression` | Optional | +| `limit` | The maximum number of entries to return, 1-1000 (both inclusive); default is `100` | Optional | +| `cursor` | The string value to access the next page of results | Optional | + +Here's an example of how to query our `drafts` datastore using the [Slack CLI](/slack-cli/guides/installing-the-slack-cli-for-mac-and-linux) and retrieve a list of all the announcements with messages containing "timesheet": + +```javascript +const result = await client.apps.datastore.query({ + datastore: "drafts", + expression: "contains (#message_term, :message)", + expression_attributes: { "#message_term": "message" }, + expression_values: { ":message": "timesheet" }, +}); +``` + + If that example looks wonky to you; read on while we explain. Under the hood, the [`apps.datastore.query`](https://api.slack.com/methods/apps.datastore.query) API method is a DynamoDB scan, and thereby uses DynamoDB's [filter expression syntax](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Scan.html#Scan.FilterExpression). + +Let's break down that previous query example: + +The `expression` is the search criteria. The `expression_attributes` object is a map of the columns used for the comparison, and the `expression_values` object is a map of values. The `expression_attributes` property must always begin with a `#`, and the `expression_values` property must always begin with a `:`. + +To break that down further, `#message_term` seen here is a variable representing the `message` datastore attribute. So, why not just use `message` in the expression, such that it would be `expression: "message = :message"`? We do this to safeguard against anything that might break the search query, like double quotes or spaces in a name, or using DynamoDB's [reserved words](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/ReservedWords.html) as attribute names. The second such variable used in the `expression` is `:message`. We see that defined in `expression_values` as the hard-coded value of `"timesheet"`, but it's more likely that you'll use a variable here, perhaps a value obtained from a user interaction. + + In summary, this query searches for items in the `drafts` datastore that have a value of `"timesheet"` (represented by `:message`) in their `message` attribute (represented by `#message_term`). + +Let's take a look at another example, this one exploring searching a datastore by timestamp. Given this set of data in a datastore: +```json +{ + "id": "foo5", + "message": "bar5", + "timestamp": 1671752648 +} + +{ + "id": "foo4", + "message": "bar4", + "timestamp": 1670975048 +} + +{ + "id": "foo3", + "message": "bar3", + "timestamp": 1702511048 +} +``` + +If we run the [Slack CLI](/slack-cli/guides/installing-the-slack-cli-for-mac-and-linux) query: +```bash +slack datastore query '{ + "datastore": "messages", + "expression": "#timestamp between :time_start AND :time_end", + "expression_attributes": {"#timestamp":"timestamp"}, + "expression_values": {":time_start":1670975049,":time_end":1702511047} + }' +``` + +We will see this object as a result: +```json +{ + "id": "foo5", + "message": "bar5", + "timestamp": 1671752648 +} +``` +You can use filter expression operators with any of the date types ([`Schema.slack.types.date`](/deno-slack-sdk/reference/slack-types#date), [`Schema.slack.types.timestamp`](/deno-slack-sdk/reference/slack-types#timestamp), and [`Schema.slack.types.message_ts`](/deno-slack-sdk/reference/slack-types#message-ts)), so long as the values passed match the underlying type and format. + +Here is another example of a date query, this one using the [`Schema.slack.types.date`](/deno-slack-sdk/reference/slack-types#date) field. + +Given this set of data in a datastore: +```json +{ + "date": "2022-01-02", + "message": "First message", + "id": "1" +} + +{ + "date": "2023-04-11", + "message": "Second message", + "id": "2" +} + +{ + "date": "2024-01-01", + "message": "Third message", + "id": "3" +} +``` + +Running this query: +```bash +slack datastore query '{ + "datastore": "messages", + "expression": "#date < :date_end", + "expression_attributes": {"#date": "date"}, + "expression_values": {":date_end": "2023-01-01"} + }' +``` + +Will yield this result: +```json +{ + "date": "2022-01-02", + "message": "First message", + "id": "1" +} +``` + +## Pagination {#pagination} + +It is strongly recommended to always handle pagination when implementing a query so that you can easily view all of your query results. + +The following code snippet from the [Virtual Running Buddies sample app](https://github.com/slack-samples/deno-virtual-running-buddies) shows how to do this: + +```javascript +export async function queryRunningDatastore( + client: SlackAPIClient, + expressions?: object, +): Promise<{ + ok: boolean; + items: DatastoreItem[]; + error?: string; +}> { + const items: DatastoreItem[] = []; + let cursor = undefined; + + do { + const runs: DatastoreQueryResponse = + await client.apps.datastore.query < typeof RunningDatastore.definition > ({ + datastore: RUN_DATASTORE, + cursor, + ...expressions, + }); + + if (!runs.ok) { + return { ok: false, items, error: runs.error }; + } + + cursor = runs.response_metadata?.next_cursor; + items.push(...runs.items); + } while (cursor); + + return { ok: true, items }; +} +``` + +Essentially, you'll use the `cursor` parameter to retrieve the next page of your query results. + +:::tip + +If your initial query has another page of results, the `next_cursor` response parameter is the key returned that will unlock your next page of results. Use this key to query the datastore again and set `cursor` to the value of `next_cursor`. + +Remember that filters are applied post-hoc, so you should always be sure to check subsequent pages for results, even if the initial page has fewer results than expected. Continue to the [filter expressions](#filter-expressions) section for more context. + +::: + +## Filter expressions {#filter-expressions} + +Because datastore `query` is a DynamoDB [scan](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Scan.html), all query expressions are essentially filter expressions: it's what you put in the value of the `expression` argument. Filter expressions are applied post-hoc. This is important to understand because it can yield some confusing results; i.e. return fewer results than requested yet have additional pages of results to be queried and [paginated](#pagination). Each query can return a maximum of 1MB of data per page of results, and returns all results of the datastore _before_ applying any filter conditions. The filter conditions are applied to each page of results individually. This is how you could end up with the first page of zero results, yet still have a cursor for a following page of results. + +Here is the full list of comparison operators to use in a filter expression, followed by some examples: + +| Operator | Description | Example | +| ---------| ----------- | ------- | +| `=` | True if both values are equal | `a = b` | +| `<` | True if the left value is less than but not equal to the right | `a < b` | +| `<=` | True if the left value is less than or equal to the right | `a <= b` | +| `>` | True if the left value is greater than but not equal to the right | `a > b` | +| `>=` | True if the left value is greater than or equal to the right | `a >= b` | +| `BETWEEN ... AND ` | True if one value is greater than or equal to one and less than or equal to another | `#time_stamp BETWEEN :ts1 AND :ts2` | +| `begins_with(str, substr)` | True if a `string` begins with `substring` | `begins_with("#message_term", ":message")` | +| `contains (path, operand)` | True if attribute specified by `path` is a string that contains the `operand` string | `contains (#song, :inputsong)` + +:::warning[Expressions can only contain non-primary key attributes] + +If you try to write an expression that uses a primary key as its attribute (for example, to pull a single row from a datastore), you will receive a cryptic error. Please use [`apps.datastore.get`](#get) instead. We're hard at work on making these types of errors easier to understand! + +::: + +Revisiting our `drafts` datastore, here we retrieve all the announcements created by user `C123ABC456`: + +```javascript +const result = await client.apps.datastore.query({ + datastore: "drafts", + expression: "#announcement_creator = :user", + expression_attributes: { "#announcement_creator": "created_by" }, + expression_values: { ":user": "C123ABC456" }, +}); +``` + +If you wanted to verify the query before putting it in your app code, the CLI query for that same search would be: + +```bash +slack datastore query '{ + "datastore": "drafts", + "expression": "#announcement_creator = :user", + "expression_attributes": { "#announcement_creator": "created_by"}, + "expression_values": {":user": "C123ABC456"} +}' +``` + +Here's an example of a function that receives a string `message` via an `input` and queries for the announcement record that matches the provided message: + +```javascript +const result = await client.apps.datastore.query({ + datastore: "drafts", + expression: "contains (#message_term, :message)", + expression_attributes: { "#message_term": "message" }, + expression_values: { ":message": input.message }, +}); +``` + +You could also chain expressions together to narrow your results even further: + +```javascript +const result = await client.apps.datastore.query({ + datastore: "drafts", + expression: "contains (#message_term, :message) AND #announcement_creator = :creator", + expression_attributes: { "#message_term": "message", "#announcement_creator": "created_by" }, + expression_values: { ":message": input.message, ":creator": input.creator }, +}); +``` + +## Count items with `count` {#count} + +As mentioned above, querying a datastore uses a DynamoDB scan to return an array of matching items for your query results. We also mentioned that each query, i.e. each DynamoDB scan, can return a maximum of 1MB of data per page of results. For that reason, if you have over 1MB of data in your datastore, multiple scans are necessary to paginate through your entire datastore. + +DynamoDB accomplishes this by returning a cursor to start a new scan where you left off with your previous one. Therefore if you wanted to use the `query` method to count all of the matching items in your datastore, you would need to call the `query` command several times, then manually add together the sizes of each array of matching items returned. + +Instead, you can use the `count` method to paginate through your datastore and sum up the count of all the items matching your query. If a query is not provided, the count will be equal to the number of items in the entire datastore. + +Using the DynamoDB style of syntax, the following example would retrieve the number of records from a datastore called "good_tunes", where "You" is in the song title: + +```json +{ + "datastore": "good_tunes", + "expression": "contains (#song, :keyword)", + "expression_attributes": { "#song": "song" }, + "expression_values": { ":keyword": "You" } +} +``` + +The response to the request might look like the following: + +```json +{ + "ok": true, + "datastore": "good_tunes", + "count": 2 +} +``` diff --git a/docs/guides/datastores/using-datastores.md b/docs/guides/datastores/using-datastores.md new file mode 100644 index 00000000..0b084f96 --- /dev/null +++ b/docs/guides/datastores/using-datastores.md @@ -0,0 +1,311 @@ +--- +slug: /deno-slack-sdk/guides/using-datastores +--- + +# Using datastores + + + +Datastores are a Slack-hosted way to store data for your workflow apps. They are available for workflow apps only. + +Datastores are backed by [DynamoDB](https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/Introduction.html), a secure and performant NoSQL database. DynamoDB's data model uses three basic types of data units: tables, items, and attributes. Tables are collections of items, and items are collections of attributes. You will see how a collection of attributes comprises an item when we define a datastore later in this page. + +## Initializing a datastore {#create} + +To initialize a datastore: +1. [Define](#define) the datastore +2. [Add](#manifest) it to your manifest + +### 1. Define the datastore {#define} + +To keep your app tidy, datastores can be defined in their own source files just like [custom functions](/deno-slack-sdk/guides/creating-custom-functions). + +If you don't already have one, create a `datastores` directory in the root of your project, and inside, create a source file to define your datastore. + +Throughout this page, we'll use the example of the [Announcement bot sample app](https://github.com/slack-samples/deno-announcement-bot/tree/main). First, we'll create a datastore called `Drafts` and define it in a file named `drafts.ts`. It will hold information about an announcement the user drafts to send to a channel: + +```js +// /datastores/drafts.ts +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export default DefineDatastore({ + name: "drafts", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + created_by: { + type: Schema.slack.types.user_id, + }, + message: { + type: Schema.types.string, + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + icon: { + type: Schema.types.string, + }, + username: { + type: Schema.types.string, + }, + status: { + type: Schema.types.string, // possible statuses are draft, sent + }, + }, +}); +``` + +Datastores can contain three primary properties. The `primary_key` property is the only one that is required. When using additional optional properties, make sure to handle them properly to avoid running into any TypeScript errors in your code. + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `name` | String | A string to identify your datastore | Optional | +| `primary_key` | String | The attribute to be used as the datastore's unique key; ensure this is an actual attribute that you have defined | Required | +| `attributes` | Object (see below) | Properties to scaffold your datastore's columns | Optional | +| `time_to_live_attribute` | String | An optional attribute used as a Time To Live (TTL) feature to [delete datastore items automatically](/deno-slack-sdk/guides/deleting-items-from-a-datastore#delete-automatically) when set to a property of type `Schema.slack.types.timestamp`, which represents the item's expiration. | Optional | + +Attributes can be [custom types](/deno-slack-sdk/guides/creating-a-custom-type), [Slack types](/deno-slack-sdk/reference/slack-types), and the following basic schema types: +* array +* boolean +* int +* number +* object +* string + +:::warning[No nullable support] + +If you use a built-in Slack type for an attribute, there is no nullable support. For example, let's say you use `channel_id` for an attribute and at some point in your app, you'd like to clear out the `channel_id` for a given item. You cannot do this with a Slack built-in type. Change the data type to be a string if you'd like to support a null or empty value. + +::: + +### 2. Add the datastore to your app's manifest {#manifest} + +The last step in initializing your datastore is to add it to the `datastores` property in your manifest and include the required datastore bot scopes. + +To do that, first add the `datastores` property to your manifest if it does not exist, then list the datastores you have defined. Second, add the following datastore permission scopes to the `botScopes` property: + +* `datastore:read` +* `datastore:write` + +Here's an example manifest definition for the above `drafts` datastore in the [Announcement bot sample app](https://github.com/slack-samples/deno-announcement-bot/tree/main): + +```js +import { Manifest } from "deno-slack-sdk/mod.ts"; +// Import the datastore definition +import AnnouncementDatastore from "./datastores/announcements.ts"; +import DraftDatastore from "./datastores/drafts.ts"; +import { AnnouncementCustomType } from "./functions/post_summary/types.ts"; +import CreateAnnouncementWorkflow from "./workflows/create_announcement.ts"; + +export default Manifest({ + name: "Announcement Bot", + description: "Send an announcement to one or more channels", + icon: "assets/icon.png", + outgoingDomains: ["cdn.skypack.dev"], + datastores: [DraftDatastore, AnnouncementDatastore], // Add the datastore to this list + types: [AnnouncementCustomType], + workflows: [ + CreateAnnouncementWorkflow, + ], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "chat:write.customize", + "datastore:read", + "datastore:write", + ], +}); +``` + +Note that we've also added the required `datastore:read` and `datastore:write` bot scopes. + +:::info + +Updates to an existing datastore that could result in data loss (removal of an existing datastore or attribute from the app) may require the use of the force flag (`--force`) when re-deploying the app. See [schema_compatibility_error](/slack-cli/guides/troubleshooting-slack-cli-errors#schema_compatibility_error) for more information. + +::: + +--- + +## Importing data to a datastore {#import} + +You can import data from a [JSON Lines](https://jsonlines.org/) file to a datastore using the [`datastore bulk-put`](/slack-cli/reference/commands/slack_datastore_bulk-put) command with the `--from-file` flag. For example: + +``` +slack datastore bulk-put '{"datastore": "running_datastore"}' —-from-file /path/to/file.jsonl +``` + +See the [Add items to a datastore guide](/deno-slack-sdk/guides/adding-items-to-a-datastore) for more information on the API method underlying this command. + +--- + +## Exporting data from a datastore {#export} + +You can export data from a datastore to a [JSON Lines](https://jsonlines.org/) file using the [`datastore query`](/slack-cli/reference/commands/slack_datastore_query) command with the `--to-file` flag. For example: + +``` +slack datastore query '{"datastore": "running_datastore"}' —-to-file /path/to/file.jsonl +``` + +See the [Retrieve items from a datastore guide](/deno-slack-sdk/guides/retrieving-items-from-a-datastore) for more information on the API method underlying this command. + +--- + +## Counting items in a datastore {#count} + +You can count the number of items in a datastore that match a query by using the [`datastore count`](/slack-cli/reference/commands/slack_datastore_count) command. This command handles paginating through an entire datastore to return the number of matched items (rather than the items themselves, as with the `datastore query` command). + +See the [Count items in a datastore guide](/deno-slack-sdk/guides/retrieving-items-from-a-datastore#count) for more information on the API method underlying this command. + +--- + +## Interacting with a datastore {#interact} + +There are two ways to interact with your app's datastore. + +➡️ **To interact with your datastore through the command-line tool**, see the [datastore commands](/slack-cli/reference/commands/slack_datastore) section on the commands page. + +⤵️ **To interact with your datastore within a [custom function](/deno-slack-sdk/guides/creating-custom-functions)**, keep reading. + +Interacting with your app's datastore requires hitting the `SlackAPI`. To do this from within your code, we first need to import a mechanism that will allow us to call the `SlackAPI`. That mechanism is `SlackFunction`. First we import it into our function file from the `deno-slack-sdk` package, then we add a `SlackFunction` into our code. `SlackFunction` contains a property, `client`, which allows us to call the datastore. + +You can find examples of this in the [Slack API methods guide](/deno-slack-sdk/guides/calling-slack-api-methods) and [Add items to a datastore guide](/deno-slack-sdk/guides/adding-items-to-a-datastore). + +In all interactions with your datastore, double and triple-check the exact spelling of the fields in the datastore definition match your query, lest you should receive an error. + +When interacting with your datastore, it may be helpful to first visualize its structure. In our `drafts` example, let's say we have stored the following users and their drafted announcements: + +| id | created_by | message | channels | channel | message_ts | icon | username | status | +| -- | -------| -----| --- | --- | ---- | ----- | ------ | ---- | +| `906dba92-44f5-4680-ada9-065149e4e930` | `U045A5X302V` | `This is a test message` | `["C038M39A2TV"]` | `C039ARY976C`| `1691513323.119209` | | `Slackbot` | `sent` | +| `b8457c38-4401-4dd1-b979-a0e56f7c9a3d` | `BR75C7X4P90` | `Remember to submit your timesheets` | `["C038M39A2TV"]` | `C039ARY976C` | `1691520476.091369` | `:robot_face:` | `The Boss` | `draft` | +| `194a52d8-c75b-4eff-9f8f-4c40292cd9e7` | `G98I9345NI2` | `Happy Friday, team!` | `["D870D2223M23"]` | `D870D2223M23` | `2172813323.142610` | `:t-rex:`| `Slackasaurus Bot` | `sent` | + +:::warning[Beware of SQL injection] +Be sure to sanitize any strings received from a user and **never** use untrusted data in your query expressions. +::: + +--- + +## Generic types for datastores {#generic-types} + +You can provide your datastore's definition as a generic type, which will provide some automatic typing on the arguments and response: + +```js +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const DraftDatastore = DefineDatastore({ + name: "drafts", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + created_by: { + type: Schema.slack.types.user_id, + }, + message: { + type: Schema.types.string, + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + icon: { + type: Schema.types.string, + }, + username: { + type: Schema.types.string, + }, + status: { + type: Schema.types.string, + }, + }, +}); +``` + +You can use the result of your `DefineDatastore()` call as the type in a function by using its `definition` property: + +```js +import { DraftDatastore } from "../datastores/drafts.ts"; +... + const putResp = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + item: { + id: draftId, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + }); + ... +``` + +By using typed methods, the `datastore` property (e.g. `DraftDatastore.datastore`) will enforce that its value matches the datastore definition's `name` property across methods and the `item` matches the definition's `attributes` in arguments and responses. Also, for `get()` and `delete()`, a property matching the `primary_key` will be expected as an argument. + +--- + +## Onward {#onward} + +Ready to start manipulating data with your workflows? We've got a guide for each type of activity: + +* [Add items to a datastore](/deno-slack-sdk/guides/adding-items-to-a-datastore) +* [Retrieve items from a datastore](/deno-slack-sdk/guides/retrieving-items-from-a-datastore) +* [Delete items from a datastore](/deno-slack-sdk/guides/deleting-items-from-a-datastore) +* [Delete items from a datastore automatically](/deno-slack-sdk/guides/deleting-items-from-a-datastore#delete-automatically) +--- + +## Deleting a datastore {#delete-datastore} +If you need to delete a datastore completely, for instance you've changed the primary key, you have a couple of options. Datastores do support primary key changes, so first try using the `--force` flag on a [datastore CLI](/slack-cli/reference/commands/slack_datastore) operation if the Slack CLI informs you that the datastore has changed. Otherwise, do the following: + +Step 1. Remove the datastore definition from the app's manifest. + +Step 2. Run `slack deploy`. + +Step 3. Modify the datastore definition to your heart's content and add it back into the app's manifest. + +Step 4. Run `slack deploy` again. + +--- + +## Troubleshooting {#troubleshooting} + +If you're looking to audit or query your datastore from the terminal without having to go through code, see the [datastore commands](/slack-cli/reference/commands/slack_datastore). + +If you're getting errors, check the following: + +* The primary key is formatted as a string +* The datastore is included in the manifest's `datastores` property +* The datastore bot scopes are included in the manifest (`datastore:read` and `datastore:write`) +* The spelling of the fields in your query match exactly the spelling of the fields in the datastore's definition + +:::info +The information stored when initializing your datastore using `slack run` will be completely separate from the information stored in your datastore when using `slack deploy`. + +::: \ No newline at end of file diff --git a/docs/guides/datatypes/_choices-property-snippet.md b/docs/guides/datatypes/_choices-property-snippet.md new file mode 100644 index 00000000..aca225f1 --- /dev/null +++ b/docs/guides/datatypes/_choices-property-snippet.md @@ -0,0 +1,53 @@ +#### The `choices` property + +The `choices` property is an array of `EnumChoice` objects. Here is a closer look at the properties of the `EnumChoice` object: + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. For example, the default for a `boolean` would be `true` or `false`.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `value` | the type that the `EnumChoice` object corresponds to — in the example below, it is `string` | The value of the corresponding choice, which must map to the values present in the sibling `enum` property. | +| `title` | string | The label to display for this `EnumChoice`. | +| `description` | string | An optional description for this `EnumChoice`. | + +
+ A `choices` example using the [`string`](#string) Slack type, + +In the following example for the [`string`](#string) Slack type, defining the `choices` property allows us to have the label on the input form be capitalized, but the data we're going to save be lowercase. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +// ... + +const inputForm = LogFruitWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Tell us your favorite fruit", + interactivity: LogFruitWorkflow.inputs.interactivity, + submit_label: "Submit", + fields: { + elements: [{ + name: "Fruit", + title: "The three best fruits", + type: Schema.types.string, + enum: ['mango', 'strawberry', 'plum'], + choices: [ + {value: 'mango', title: 'Mango!', description: 'Wonderfully tropical'}, + {value: 'strawberry', title: 'Strawberry!', description: 'Absolutely fantastic'}, + {value: 'plum', title: 'Plum!', description: 'Tart, just the way I like em'}, + ] + }] + required: ["Fruit"], + }, + }, +); + +// ... +``` + +
\ No newline at end of file diff --git a/docs/guides/datatypes/creating-a-custom-type.md b/docs/guides/datatypes/creating-a-custom-type.md new file mode 100644 index 00000000..9489bab3 --- /dev/null +++ b/docs/guides/datatypes/creating-a-custom-type.md @@ -0,0 +1,336 @@ +--- +slug: /deno-slack-sdk/guides/creating-a-custom-type +--- + +# Creating a custom type + + + +Custom types provide a way to introduce reusable, sharable types to your workflow apps. Once registered in your manifest, you can use custom types as input or output parameters in any of your app's [functions](/deno-slack-sdk/guides/creating-slack-functions), [workflows](/deno-slack-sdk/guides/creating-workflows), or [datastores](/deno-slack-sdk/guides/using-datastores). The possibilities are endless! + +## Defining a type {#define-type} + +Types can be defined with a top level `DefineType` export. In the example below, a custom type object is defined for use in the [Deno Announcement Bot](https://github.com/slack-samples/deno-announcement-bot) sample app: + +```javascript +// types/incident.ts +import { DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const AnnouncementCustomType = DefineType({ + name: "Announcement", + type: Schema.types.object, + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + }, + success: { + type: Schema.types.boolean, + }, + permalink: { + type: Schema.types.string, + }, + error: { + type: Schema.types.string, + }, + }, + required: ["channel_id", "success"], +}); +``` + +Another way custom types can be defined is within a function, where they can be immediately used: + +```javascript +// Define the custom type +import { DefineFunction, DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const AnnouncementCustomType = DefineType({ + name: "Announcement", + type: Schema.types.object, + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + }, + success: { + type: Schema.types.boolean, + }, + permalink: { + type: Schema.types.string, + }, + error: { + type: Schema.types.string, + }, + }, + required: ["channel_id", "success"], +}); + +// Define your function, which uses the custom type we just defined +export const PrepareSendAnnouncementFunctionDefinition = DefineFunction({ + callback_id: "send_announcement", + title: "Send an announcement", + description: "Sends a message to one or more channels", + source_file: "functions/send_announcement/handler.ts", + input_parameters: { + properties: { + message: { + type: Schema.types.string, + description: "The content of the announcement", + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The destination channels of the announcement", + }, + icon: { + type: Schema.types.string, + description: "Optional custom bot icon to use display in announcements", + }, + username: { + type: Schema.types.string, + description: "Optional custom bot emoji avatar to use in announcements", + }, + draft_id: { + type: Schema.types.string, + description: "The datastore ID of the draft message if one was created", + }, + }, + required: [ + "message", + "channels", + ], + }, + output_parameters: { + properties: { + announcements: { + type: Schema.types.array, + items: { + type: AnnouncementCustomType, + }, + description: + "Array of objects that includes a channel ID and permalink for each announcement successfully sent", + }, + }, + required: ["announcements"], + }, +}); +// Finish implementing your function +export default SlackFunction(PrepareSendAnnouncementFunctionDefinition, async ({ inputs, client }) => { +// ... +``` + +If your custom type will be used in an array, create the array as a custom type too. For example, if we wanted an array of `AnnouncementCustomType`, it would look like this: +```javascript +// Define the custom type +import { DefineFunction, DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const AnnouncementCustomType = DefineType({ + name: "Announcement", + type: Schema.types.object, + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + }, + success: { + type: Schema.types.boolean, + }, + permalink: { + type: Schema.types.string, + }, + error: { + type: Schema.types.string, + }, + }, + required: ["channel_id", "success"], +}); + +// Define the array with the items as the custom type +export const AnnouncementArray = DefineType({ + name: "AnnouncementArray", + type: Schema.types.array, + items: { + type: AnnouncementCustomType + }, +}) + +``` + +:::info[Fully defined arrays] + +If a property on your custom type is an array, be sure to define its properties in the `items` field (refer to example [here](/deno-slack-sdk/reference/slack-types#array)). Untyped objects are not currently supported. + +::: + + +## Registering a type {#register-type} + +To register newly-defined types for use with your app, add them to the `types` array when defining your [manifest](/deno-slack-sdk/guides/using-the-app-manifest). Here's an example, again from the [Deno Announcement Bot](https://github.com/slack-samples/deno-announcement-bot) sample app. + +:::info + +All custom types must be registered in the [manifest](/deno-slack-sdk/guides/using-the-app-manifest) for them to be available for use. + +::: + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; +import AnnouncementDatastore from "./datastores/announcements.ts"; +import DraftDatastore from "./datastores/drafts.ts"; +import { AnnouncementCustomType } from "./functions/post_summary/types.ts"; +import CreateAnnouncementWorkflow from "./workflows/create_announcement.ts"; + +export default Manifest({ + name: "Announcement Bot", + description: "Send an announcement to one or more channels", + icon: "assets/icon.png", + outgoingDomains: ["cdn.skypack.dev"], + datastores: [DraftDatastore, AnnouncementDatastore], + types: [AnnouncementCustomType], + workflows: [ + CreateAnnouncementWorkflow, + ], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "chat:write.customize", + "datastore:read", + "datastore:write", + ], +}); + +``` + +## Referencing a type {#reference-type} + +To use a custom type as a [function](/deno-slack-sdk/guides/creating-slack-functions) parameter, import the type: + +```javascript +import { DefineFunction, DefineType, Schema } from "deno-slack-sdk/mod.ts"; +// ... +``` + +Then, set the parameter's `type` property to the custom type it should reference: + +```javascript +// ... +input_parameters: { + incident: { + title: "A String Cheese Incident", + type: IncidentType, + }, +}, +// ... +``` +In the example above, the `title` property from the custom type `IncidentType` is being overridden with the string "A String Cheese Incident". + +Let's look at using custom types in a bit more depth with another one of our sample apps, the [Virtual Running Buddies](https://github.com/slack-samples/deno-virtual-running-buddies) app. Taking a look at [`/types/runner_stats.ts`](https://github.com/slack-samples/deno-virtual-running-buddies/blob/main/types/runner_stats.ts) you'll find the definition for `RunnerStatsType`: + +```javascript +import { DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const RunnerStatsType = DefineType({ + title: "Runner Stats", + description: "Information about the recent runs for a runner", + name: "runner_stats", + type: Schema.types.object, + properties: { + runner: { type: Schema.slack.types.user_id }, + weekly_distance: { type: Schema.types.number }, + total_distance: { type: Schema.types.number }, + }, + required: ["runner", "weekly_distance", "total_distance"], +}); +``` + +Information about each runner is collected as a RunnerStatsType, which includes who the runner is (`user_id`), how far they ran each week (`weekly_distance`), and the total distance they've run so far (`total_distance`). This type is then used to describe the array output of the `CollectRunnerStatsFunction` which is called when the `DisplayLeaderBoardWorkflow` is started. Take a look at the `CollectRunnerStatsFunction` definition here to see how the custom type is returned as an output. + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import RunningDatastore, { RUN_DATASTORE } from "../datastores/run_data.ts"; +import { RunnerStatsType } from "../types/runner_stats.ts"; + +export const CollectRunnerStatsFunction = DefineFunction({ + callback_id: "collect_runner_stats", + title: "Collect runner stats", + description: "Gather statistics of past runs for all runners", + source_file: "functions/collect_runner_stats.ts", + input_parameters: { + properties: {}, + required: [], + }, + output_parameters: { + properties: { + runner_stats: { + type: Schema.types.array, + items: { type: RunnerStatsType }, + description: "Weekly and all-time total distances for runners", + }, + }, + required: ["runner_stats"], + }, +}); + +export default SlackFunction(CollectRunnerStatsFunction, async ({ client }) => { + // Query the datastore for all the data we collected + const runs = await client.apps.datastore.query< + typeof RunningDatastore.definition + >({ datastore: RUN_DATASTORE }); + + if (!runs.ok) { + return { error: `Failed to retrieve past runs: ${runs.error}` }; + } + + const runners = new Map(); + + // ... runners object is constructed + + // Return an array with runner stats + return { + outputs: { runner_stats: [...runners.entries()].map((r) => r[1]) }, + }; +}); +``` + +The `map` function you see here in the `outputs` is converting the entries in the `runners` map to an array, then mapping these entries to an array of `RunnerStatsType`. Another way of writing the map part of the function would be: + +```javascript +function (r) { + return r[1]; +} +``` + +`r[1]` is the value in the `runners` array, whereas `r[0]` would be the key, so the map function is essentially mapping an array of small arrays. `RunnerStatsType` here is returned as the output of the function, and the function sets the properties of the custom type before returning it. Pretty cool, huh? To see a full tutorial on this sample app, head over to [Create a social app to log runs with running buddies](/deno-slack-sdk/tutorials/virtual-running-buddies-app). + +## TypeScript-friendly type definitions {#define-property} + +:::warning[Object types are not supported within Workflow Builder at this time] + +If your function will be used within Workflow Builder, we suggest not using the Object types at this time. + +::: + +Use the `DefineProperty` helper function to get TypeScript-friendly type definitions for your input and output parameters. This is an optional helper utility that is highly recommended for TypeScript source code. While your code will still work without it, we recommend using `DefineProperty` when you have an object parameter with optional sub-properties so that your IDE autocomplete pop-ups will accurately respect the optional nature of properties. If all sub-properties are all required, you don’t need to use `DefineProperty`. Let's illustrate this with an example: + +```javascript +const messageAlertFunction = DefineFunction({ + ... + input_parameters: { + properties: { + msg_context: DefineProperty({ + type: Schema.types.object, + properties: { + message_ts: { type: Schema.types.string }, + channel_id: { type: Schema.types.string }, + user_id: { type: Schema.types.string }, + }, + required: ["message_ts"] + }) + } + }, + }); +``` diff --git a/docs/guides/datatypes/integrating-message-metadata-events.md b/docs/guides/datatypes/integrating-message-metadata-events.md new file mode 100644 index 00000000..287d5cdb --- /dev/null +++ b/docs/guides/datatypes/integrating-message-metadata-events.md @@ -0,0 +1,240 @@ +--- +slug: /deno-slack-sdk/guides/integrating-message-metadata-events +--- + +# Integrating message metadata events + + + +[Message metadata](https://api.slack.com/metadata) can connect your workflows to events happening within Slack. By doing so you can automate tasks within your workflows. + +Message metadata will take the form of the custom message metadata event types you create. There are three steps to integrate a custom event type: + +1. [Define](#define) the custom event type in its own file +1. [Register](#register) the custom event type in your app's manifest +1. [Use](#use) the custom event type in a trigger or function + +## 1. Define a custom event type {#define} + +When integrating message metadata into your app, you may want some additional type safety. [Custom message metadata event types](https://api.slack.com/reference/metadata) provide a way for apps to validate message metadata against a schema that you define. + +First, let's create a new file to store the custom event type's definition. In this example, we'll create `event_types/incident.ts`. + +The first thing we'll do in the file is to import `DefineEvent` and `Schema` from the SDK. Then, we'll use `DefineEvent` to create our event's definition. An example definition is as follows: + +```javascript +// event_types/incident.ts +import { DefineEvent, Schema } from "deno-slack-sdk/mod.ts"; + +const IncidentEvent = DefineEvent({ + name: "my_incident_event", + title: "Incident", + type: Schema.types.object, + properties: { + id: { type: Schema.types.string }, + title: { type: Schema.types.string }, + summary: { type: Schema.types.string }, + severity: { type: Schema.types.string }, + date_created: { type: Schema.types.number }, + }, + required: ["id", "title", "summary", "severity"], + // Set this to false to force the validation to catch any additional properties + additionalProperties: false, +}); + +export default IncidentEvent; +``` + +For the `type` property, events can be one of two kinds: the built-in `Schema.types.object` type, or a [custom type](/deno-slack-sdk/guides/creating-a-custom-type) that you define. If you go with a custom type, set the `type` property as your custom type — just don't forget to also import that custom type definition in your app's manifest. + +✨ **For more information about the app manifest**, refer to [app manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +## 2. Register a custom event type {#register} + +Before your app can use your custom event type, you'll need to register it with your app's manifest. To register the newly-defined custom event type, add it to the `events` array of your [manifest](/deno-slack-sdk/guides/using-the-app-manifest) definition: + +```javascript +// manifest.ts +import IncidentEvent from "./event_types/incident.ts"; + +export default Manifest({ + name: "my_incident_app", + description: "An app that uses a custom event type", + icon: "assets/default_new_app_icon.png", + workflows: [SampleWorkflow], + outgoingDomains: [], + datastores: [SampleDatastore], + events: [IncidentEvent], // Our custom event type + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "datastore:read", + "datastore:write", + "metadata.message:read", + ], +}); +``` + +## 3. Use a custom event type {#use} + +There are two ways you can use your custom event type: + +* [by posting a message to Slack](#posting) +* [by creating a message metadata trigger](#trigger) + +### Posting a message to Slack {#posting} + +When you post a message to Slack using the [`metadata` parameter](https://api.slack.com/methods/chat.postMessage#arg_metadata), if the `event_type` matches the `name` of a custom event type specified in your app's manifest, Slack's servers will validate that all of the required parameters are provided. If the required parameters are not provided, a warning will be returned in the response. The message will still be posted, but without the message metadata since it didn't pass validation. + +There are two ways to post a message in Slack: + +* [from within a custom function](#custom-function) +* [as one of your workflow's steps](#workflow-step) + +#### Posting a message from within a custom function {#custom-function} + +Here's an example of using your custom event type while calling `client.chat.postMessage()` from within a [custom function](/deno-slack-sdk/guides/creating-custom-functions): + +```javascript +// functions/my_incident_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import IncidentEvent from "../event_types/incident.ts"; + +export const MyFunctionDefinition = DefineFunction({ + callback_id: "my_incident_function", + title: "my incident function", + source_file: "functions/my_incident_function.ts", + input_parameters: { + properties: { + channel_id: { type: Schema.slack.types.channel_id }, + incident_id: { type: Schema.types.string }, + incident_title: { type: Schema.types.string }, + incident_summary: { type: Schema.types.string }, + incident_severity: { type: Schema.types.string }, + incident_date: { type: Schema.slack.types.timestamp }, + }, + required: ["channel_id"], + }, + output_parameters: { + properties: {}, + required: [] + }, +}); + +export default SlackFunction( + MyFunctionDefinition, + async ({ inputs, client }) => { + // This example assumes all required values are passed to the function's inputs + const response = await client.chat.postMessage({ + channel: inputs.channel_id, + text: "We have an incident!", + metadata: { + // Our custom event type + event_type: IncidentEvent, + event_payload: { + id: inputs.incident_id, + title: inputs.incident_title, + summary: inputs.incident_summary, + severity: inputs.incident_severity, + // This isn't required, so it doesn't need to exist to pass validation + date_created: inputs.incident_date, + }, + }, + }); + if (response.error) { + const error = `Failed to post a message with metadata: ${response.error}`; + return { error }; + } + // Do something meaningful here! + return { outputs: {} }; + }, +); +``` +#### Posting a message as one of your workflow's steps {#workflow-step} + +Here's an example of using your custom event type with the [Slack function `SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message) as one of your workflow's steps: + +```javascript +// workflows/my_workflow.ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import IncidentEvent from "../event_types/incident.ts"; + +export const MyWorkflow = DefineWorkflow({ + callback_id: "my_workflow", + title: "My workflow", + input_parameters: { + properties: { + channel_id: { type: Schema.slack.types.channel_id }, + incident_id: { type: Schema.types.string }, + incident_title: { type: Schema.types.string }, + incident_summary: { type: Schema.types.string }, + incident_severity: { type: Schema.types.string }, + incident_date: { type: Schema.slack.types.timestamp }, + }, + required: ["channel_id"], + }, +}); + +// This example assumes all required values are passed to the workflow's inputs +MyWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: MyWorkflow.inputs.channel_id, + message: "We have an incident!", + metadata: { + // Our custom event type + event_type: IncidentEvent, + event_payload: { + id: MyWorkflow.inputs.incident_id, + title: MyWorkflow.inputs.incident_title, + summary: MyWorkflow.inputs.incident_summary, + severity: MyWorkflow.inputs.incident_severity, + // This isn't required, so it doesn't need to exist to pass validation + date_created: MyWorkflow.inputs.incident_date, + }, + }, +}); +``` + +### Creating a message metadata trigger {#trigger} + +A trigger can be created to watch for any message posted with a metadata event type matching your custom event type. When a match is found, that trigger will execute its configured workflow as in the following example: + +```javascript +// triggers/incident_metadata_posted.ts +import { Trigger } from "deno-slack-api/types.ts"; +import MyWorkflow from "../workflows/my_workflow.ts"; +import IncidentEvent from "../event_types/incident.ts"; + +const trigger: Trigger = { + type: "event", + name: "Incident Metadata Posted", + inputs: { + incident_id: { value: "{{data.metadata.event_payload.id}}" }, + incident_title: { value: "{{data.metadata.event_payload.title}}" }, + incident_summary: { value: "{{data.metadata.event_payload.summary}}" }, + incident_severity: { value: "{{data.metadata.event_payload.severity}}" }, + incident_date: { value: "{{data.metadata.event_payload.incident_date}}" }, + }, + // This is the workflow that will be kicked off + workflow: `#/workflows/${MyWorkflow.definition.callback_id}`, + event: { + event_type: "slack#/events/message_metadata_posted", + // Our custom event type + metadata_event_type: IncidentEvent, + // The channel we're watching for message metadata being posted + channel_ids: ["C123ABC456"], + }, +}; + +export default trigger; +``` + +:::info + +[Event triggers](/deno-slack-sdk/guides/creating-event-triggers) such as the one above that listen for message metadata require the `metadata.message:read` scope to be added to the `botScopes` property of your [manifest definition](/deno-slack-sdk/guides/using-the-app-manifest). + +::: + +✨ **For more information about custom types**, refer to [custom types](/deno-slack-sdk/guides/creating-a-custom-type). + +✨ **For more information about event triggers**, refer to [event triggers](/deno-slack-sdk/guides/creating-event-triggers). \ No newline at end of file diff --git a/docs/guides/datatypes/utilizing-slack-and-custom-data-types.md b/docs/guides/datatypes/utilizing-slack-and-custom-data-types.md new file mode 100644 index 00000000..3dcb5410 --- /dev/null +++ b/docs/guides/datatypes/utilizing-slack-and-custom-data-types.md @@ -0,0 +1,23 @@ +--- +slug: /deno-slack-sdk/guides/utilizing-slack-and-custom-data-types +--- + +# Utilizing Slack & custom data types + + + +When building workflow apps, you can use a handful of Slack types. You can also [define your own custom type](/deno-slack-sdk/guides/creating-a-custom-type). + +## Slack types {#slack} + +Slack types are used in two ways: as input and output parameters of [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) & [custom functions](/deno-slack-sdk/guides/creating-custom-functions), as well as attributes of [datastores](/deno-slack-sdk/guides/using-datastores). + +All manifests can be written in JSON; however, declaring types in an app using the Deno Slack SDK is done differently, requiring a reference to the `Schema.slack` package for non-primitive types. The examples in the reference show both how would they appear in Typescript as they would appear in a Deno Slack SDK app and in JSON as they would be defined in a manifest. + +:arrow_right: [View the full Slack types reference catalog here](/deno-slack-sdk/reference/slack-types) + +## Custom types {#custom} + +Custom types provide a way to introduce reusable, sharable types to your workflow apps. Once registered in your manifest, you can use custom types as input or output parameters in any of your app's [functions](/deno-slack-sdk/guides/creating-slack-functions), [workflows](/deno-slack-sdk/guides/creating-workflows), or [datastores](/deno-slack-sdk/guides/using-datastores). The possibilities are endless! + +:arrow_right: To learn how to create your own custom type, read our [Creating a custom type](/deno-slack-sdk/guides/creating-a-custom-type) guide. \ No newline at end of file diff --git a/docs/guides/deno/developing-with-deno.md b/docs/guides/deno/developing-with-deno.md new file mode 100644 index 00000000..16d45dee --- /dev/null +++ b/docs/guides/deno/developing-with-deno.md @@ -0,0 +1,207 @@ +--- +slug: /deno-slack-sdk/guides/developing-with-deno +--- + +# Developing with Deno + + + +Now that we've [installed Deno](/deno-slack-sdk/guides/installing-deno), lets create our first app to get the hang of our new environment. + +## Running your first Deno app {#run} + +Using a plain text editor, create a new file called `app.js`. In that file, include the following: + +```javascript +console.log(`Hello, world!`); +``` + +In your terminal, change to the directory where you wrote that file. Then execute the following command: + +```bash +deno run app.js +``` + +It should respond with `Hello, world!` and then finish. Wohoo, you just wrote your first app and executed it using the Deno runtime! + +### Watching your app for changes {#run-watch} + +During development, you'll often find you are making lots of small changes in order to get everything in tip-top shape. It's annoying to get out of your developer flow because you have to constantly stop and restart your app, which is why it's great that Deno has a built in `watch` function that will constantly check for updates to a file and automatically reload it for you. + +First, tell Deno to watch your `app.js` file by appending the `--watch` flag to the `run` command as follows: + +```bash +deno run --watch app.js +``` + +Now, update `app.js` by adding a second line after the first `console.log`: + +```javascript +console.log(`What's up?`); +``` + +As soon as you save the file, you should see the output of **both** lines. + +### So fetch {#fetch} + +Just about any web app you could think to build is going to want to pull in data from somewhere else — it's pretty much how the whole digital economy works. Deno natively supports the [Fetch API that replaced `XMLHttpRequest`](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API) in modern browsers, so it's pretty easy to make fetch happen. + +The `fetch()` method is asynchronous and [Promises-based](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise). This is one of the key differences between Deno and Node.js. Since Promises allow asynchronous functions to behave like synchronous functions, this can make life easier for developers. Whereas synchronous functions run and return a value, *asynchronous* functions return a Promise to return a value (or an error) once it's finished calling the API, querying the database, or making a calculation. + +You don't need to worry about all of the particulars of the asynchronous function running in the background, you just need to write the code that gets executed once the Promise is fulfilled. Use the [`await`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/await) expression for this. When the Promise returns successfully, it resolves to a [Response object](https://developer.mozilla.org/en-US/docs/Web/API/Response). The Response comes as a stream of bytes, which another `await` function that calls the [`Response.text()` method](https://developer.mozilla.org/en-US/docs/Web/API/Response/text) can finally return as a string of the body of the HTTP response. + +Let's see how it works by replacing the code in your `app.js` file with the following: + +```javascript +const response = await fetch('https://example.com'); + +const body = await response.text(); +console.log(body); +``` + +If `deno run --watch app.js` is still running, it will detect the change and return a warning. This is part of [Deno's security model at work](https://deno.land/manual/getting_started/permissions), where by default, no app is allowed access to external resources like the filesystem, network access, or even sub-process and environment variables — access to these resources must be explicitly allowed when you run your app. + +This focus on security is one reason why [many projects](https://deno.land/showcase) are increasingly exploring Deno! + +You can answer `yes` to the prompt every time you run the app, or include the `--allow-net` flag when you execute the `deno run` command. If you want to limit the domains that the app has access to, include a comma-separated list of domains, such as `--allow-net=example.com` or `--allow-net=example.com,api.slack.com` as follows: + +```bash +deno run --watch --allow-net=example.com app.js +``` + +You should see the source code of `example.com` in your terminal. Parsing web page source code isn't very exciting, though. How about random photos of cats and dogs? Let's try replacing all the contents of our `app.js` with the following: + +```javascript +const cat_or_dog = Deno.args[0]; +let url = ""; + +switch (cat_or_dog) { + case 'cat': + console.log(`Meow, you're a kitty!`); + + const cat_response = await fetch('https://api.thecatapi.com/v1/images/search'); + const cat_json = await cat_response.json(); + url = cat_json[0].url; + + break; + case 'dog': + console.log(`Who's a good dog?`); + + const dog_response = await fetch('https://dog.ceo/api/breeds/image/random'); + const dog_json = await dog_response.json(); + url = dog_json.message; + + break; +} + +console.log(url); +``` + +This script takes a single argument at runtime, assigns it to the `cat_or_dog` variable, and then retrieves a random cat or dog picture. Run it with one of the following: + +```bash +deno run --allow-net=api.thecatapi.com,dog.ceo app.js cat +``` + +or + +```bash +deno run --allow-net=api.thecatapi.com,dog.ceo app.js dog +``` + +Notice that the `--allow-net` flag includes the domain names to both APIs, separated by a comma. + +## Third-party libraries {#third-party} + +Just like browsers, Deno can [import and execute scripts from remote locations](https://deno.land/manual/examples/manage_dependencies), making it possible to not only use third party libraries, but to load them from any URL. + +Let's say we wanted to actually load the random cat or dog pic in the user's default browser. The [Opener module](https://deno.land/x/opener/README.md) does exactly that, and it's cross-platform to boot. Import the Opener module's `open` function at the top of your script, then call it at the bottom: + +```javascript +import { open } from "https://deno.land/x/opener@v1.0.1/mod.ts"; + +// the rest of the logic of the cat/dog random imager + +await open(url); +``` + +When you run the script, you'll again see a warning that Deno needs permission to run the `open` command — that's because the [source code of the Opener module](https://deno.land/x/opener@v1.0.1/mod.ts) calls the `Deno.run()` method, which executes local commands on behalf of the user executing the script. Once again, Deno's security design requires explicit permission to run another command; passing the `--allow-run` flag will allow the user to run any sub-command, and passing a comma-separated list will only allow those specific commands to run. + +Deno will cache third party modules locally, but you aren't required to include a `package.json` file or the equivalent of a `node_modules` directory. In fact, your working directory is kept completely clean. + +## The standard library {#std-lib} + +Now that you've built an app and explored how to import modules, let's explore the ecosystem of third party modules and the [Deno standard library](#std-lib). Every programming language has a mechanism for allowing code to be easily shared and reused, Deno does this by leveraging [JavaScript's standard way of importing and exporting code](https://deno.land/manual/examples/manage_dependencies). + +The Deno project maintains a hosting service for open source modules at [deno.land/x/](https://deno.land/x). All of these modules are hosted on public GitHub repos and cached by the Deno project — in fact, every time a module is updated and tagged with a new version, that specific version is cached. This allows you to follow best practices for versioning the modules your application depends on. + +In addition to hosting a repository of open source modules, the Deno project also maintains a [standard library of common utilities](https://github.com/denoland/deno_std) that developers can use. Common programming tasks such as figuring out a date or time, running tests on code, writing to the filesystem, or launching an HTTP server are all part of the standard library, and these modules are audited by the Deno team to ensure they are up-to-date and do not require any other external dependencies. The standard library is located at `https://deno.land/std`, but you'll reference a specific version of the library in your apps, such as `https://deno.land/std@0.193.0` + +:::warning[Under development] + +The Deno standard library is still under development and parts are considered unstable. This means that if you use certain modules from the standard library, such as the [filesystem modules](https://deno.land/std/fs), you'll need to execute `deno run` with the `--unstable` flag. + +As the standard library matures, the plan is to version the modules alongside updates to the Deno runtime itself so it will be easier to know which version of a module to use with the version of the Deno runtime you are using. + +::: + +The standard library contains dozens of submodules, which are what you'll actually load for your app; you won't often import the entire standard library. For example, if your app needs to format dates, there's [the `format` submodule](https://deno.land/std/datetime#format), part of the `datetime` submodule, a part of the standard library. You would load it as follows: + +```javascript +import { format } from "https://deno.land/std@0.140.0/datetime/mod.ts"; +``` + +Then, you can call the `format` function: + +```javascript +// 🎈 February 12 Happy birthday, Slackbot! 🎈 +// That's not an error, the JavaScript Date constructor uses a zero-based `monthIndex` for months +// whereas days begin with 1. +// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/Date +format(new Date(2014, 01, 12), "yyyy-MM-dd"); // 2014-02-12 +``` + +## Managing versions {#versioning} + +The content at any URL on the web can change at any time; often, this is a powerful feature of the web and allows web sites to stay fresh. However, when you're loading, say, a block of code to be used as a dependency in an application, you want that code to be consistent and _immutable_. **Forever**. + +Pretty much every runtime that allows for external modules has run into this problem at some point. While there is great power in being able to load libraries, what happens if the developer makes a change that suddenly breaks your code? Or worse, what if a bad actor were to somehow gain access to where that module is hosted and [insert some malicious code](https://arstechnica.com/information-technology/2021/12/malicious-packages-sneaked-into-npm-repository-stole-discord-tokens/) that would allow the attacker access to your app? + +By caching every new version and making it immutable, Deno's hosted modules avoid this problem — but you have to make sure to include the version tag in the URL, like so: `https://deno.land/x/feathers@v5.0.0-pre.3/mod.ts`. If you omit the tag, you will automatically be redirected to whatever the latest version is: + +```javascript +// 🚯 will always import the latest version. AVOID. +import { feathers } from "https://deno.land/x/feathers/mod.ts"; + +// 😎 imports a specific version, DO THIS INSTEAD +import { feathers } from "https://deno.land/x/feathers@v5.0.0-pre.3/mod.ts"; +``` + +## Managing dependencies {#deps} +As your project grows in complexity, you may want to include a list of dependencies in a single place that can be tagged to a specific version. Deno uses the convention of a `deps.js` file to store this list. + +Let's say we want to take our dog/cat script to the next level, with internationalization and robust testing. We're going to use the [i18next library](https://deno.land/x/i18next@v21.8.1/index.js) for managing translations and the `asserts` functionality from the standard library. Our `deps.js` file might look like this: + +```javascript +export{ + assert, + assertEquals, +} from "https://deno.land/std@0.138.0/testing/asserts.js"; + +export { i18next } from "https://deno.land/x/i18next@v21.8.1/index.js"; +``` + +The URL includes a specific version number — this is the recommended way to import libraries, instead of pulling them from the main branch and hoping nothing breaks when the library gets updated. + +In our script, we import them from our local `deps.js` file as follows: + +```javascript +import {assertEquals, runTests, test } from "./deps.js" +import {i18next} from "./deps.js" +``` + +If we need to update the version or add additional libraries, we can do so from a single place. + +## Onward {#onward} + +Ready to dive into developing automations? Head over to our [getting started guide](/deno-slack-sdk/guides/getting-started) to start building a workflow app, or check out our [developing with TypeScript guide](/deno-slack-sdk/guides/developing-with-typescript) to learn more about the language you'll be developing with. \ No newline at end of file diff --git a/docs/guides/deno/developing-with-typescript.md b/docs/guides/deno/developing-with-typescript.md new file mode 100644 index 00000000..83d13620 --- /dev/null +++ b/docs/guides/deno/developing-with-typescript.md @@ -0,0 +1,54 @@ +--- +slug: /deno-slack-sdk/guides/developing-with-typescript +--- + +# Developing with TypeScript + + + +Now that we've covered the [Deno](/deno-slack-sdk/guides/installing-deno) runtime, let's talk about the language you'll use to develop: TypeScript. Just like how every chef is enamored by their own particular knife, we are enamored by TypeScript. We think you will be, too. + +Typescript is built upon JavaScript...and while JavaScript is arguably the most popular language in the world, it's not without its issues. It's a dynamically-typed language — meaning that any errors can sneak through until runtime. This can lead to some irksome situations, especially for more robust applications. + +TypeScript was created as a solution to these kinds of situations because it is a statically-typed language. There are more rules about what types are, and TypeScript will make you enforce those rules *before* you get to runtime. + +You could think of TypeScript as "JavaScript but with extra steps", but with the knowledge that those extra steps are actually pretty useful, and will save you from a lot of headaches down the road. + +## TypeScript for different devs {#different-devs-ts} + +We imagine it's likely that everyone here has previously dabbled in some other programming language. The following are a few resources to help you transition those skills to TypeScript. + +### You're a JavaScript dev {#js-ts} + +Great! JavaScript is quite similar to Typescript. Don't overthink things; the ways you perform actions in JavaScript are the same in TypeScript, you just need to be careful about those types. Much like transitioning from free-verse poetry to haikus, you'll add some rigidity to your art, but we believe your poetry will still shine through. + +* [TypeScript for JavaScript Programmers](https://www.typescriptlang.org/docs/handbook/typescript-in-5-minutes.html) +* [Migrating from JavaScript](https://www.typescriptlang.org/docs/handbook/migrating-from-javascript.html) +* [TypeScript for JavaScript Developers in 15 minutes](https://youtu.be/JUORwadOU7s) + +### You're a Java dev {#java-ts} + +Just like coffee, there are many types of statically-typed languages. Java is one, and TypeScript is but another. In Java, source code is compiled and then run by the Java Virtual Machine, whereas in TypeScript, source code is compiled into JavaScript code, which is then run by the JavaScript runtime. + +* [TypeScript for Java Programmers](https://www.typescriptlang.org/docs/handbook/typescript-in-5-minutes-oop.html) + +### You're a Python dev {#python-ts} + +This will feel pretty different! But worry not, you're still a programmer at heart, with the knowledge of programming principles within you. Python lets you play particularly fast and loose with typing, so you'll likely spend some time figuring out how to define certain types, and that's okay. Just keep a handy cheat-sheet nearby, and soon you'll laugh at the days you kept running into errors because a value you thought was an integer was actually an array. + +* [TypeScript for the New Programmer](https://www.typescriptlang.org/docs/handbook/typescript-from-scratch.html) + +### You're a wizard, Harry {#wizard-ts} + +Even wizards can brush up on their programming skills! + +* [TypeScript for the New Programmer](https://www.typescriptlang.org/docs/handbook/typescript-from-scratch.html) + +## TypeScript resources {#typescript-resources} + +While we feel practice is the best way to learn, a few supplementary readings can never hurt. + +* [The Official TypeScript Handbook](https://www.typescriptlang.org/docs/handbook/intro.html) +* [TypeScript Tooling in 5 minutes](https://www.typescriptlang.org/docs/handbook/typescript-tooling-in-5-minutes.html) +* [TypeScript blog](https://devblogs.microsoft.com/typescript/) +* [TypeScript Tips](https://www.totaltypescript.com/tips) \ No newline at end of file diff --git a/docs/guides/deno/installing-deno.md b/docs/guides/deno/installing-deno.md new file mode 100644 index 00000000..d31cefa3 --- /dev/null +++ b/docs/guides/deno/installing-deno.md @@ -0,0 +1,90 @@ +--- +slug: /deno-slack-sdk/guides/installing-deno +--- + +# Installing Deno + + + +Deno is a runtime that you'll need for developing automations apps. + +## First things first — what is Deno? {#what-deno} + +[Deno](https://deno.land) is a runtime that allows you to execute code written using [JavaScript](https://developer.mozilla.org/en-US/docs/Web/JavaScript), [TypeScript](https://www.typescriptlang.org/), and [WebAssembly](https://webassembly.org/). It uses the open source [V8 JavaScript and WebAssembly engine](https://v8.dev/), is written in a programming language called [Rust](https://www.rust-lang.org/), and is built on another runtime called [Tokio](https://tokio.rs/). So, if you know how to write apps in one of those languages (or would like to!) Deno is how you will execute your app. + +### Deno runtime {#deno-runtime} + +To understand what a runtime is, first let's take a look at the software programming lifecycle to get some context. At a basic level, the lifecycle is as follows: + +* **Develop**: source code is written and edited and an application or program is created. +* **Compile**: the source code is compiled into a machine code executable. +* **Link**: all of the machine code components of the application or program are connected together, including external libraries. +* **Distribute**: the application or program is copied to the computers of other users; for example, via an executable. +* **Install**: the user downloads the executable on their computer; their operating system places it in storage. +* **Load**: the user's operating system places the executable into active memory in order to begin execution of the application or program. +* **Run**: the distributed machine code is executed on the user's computer. + +We're interested in that last phase of this lifecycle, the _runtime_. However, it's important to note there are two concepts here that are related, but different: runtime as part of the lifecycle, and a runtime environment. Some confusion can occur since people sometimes shorten "runtime environment" to just "runtime" — but what we're talking about when we say _Deno is a runtime_ is really _Deno is a runtime environment._ + +So, what's a runtime environment? Essentially, it's a framework of all the hardware and software required to execute, or run, your code. A runtime environment accesses system resources, loads your application or program, and executes it, all of which is done independently of your operating system (which is also technically a runtime environment!). + +Why use a runtime environment? Well, because operating systems can differ significantly from one another, or even from one version to the next. Runtime environments enable cross-platform functionality for your applications or programs, allowing your code to run as smoothly as possible in a a variety of conditions. + +### Why Deno? {#why-deno} + +Where did Deno come from? Well, for a long time, JavaScript was used almost exclusively by web browsers to add interactivity to web pages. A clever programmer named Ryan Dahl [created a way to run JavaScript on servers](https://www.youtube.com/watch?v=ztspvPYybIY). He called it Node.js, and it was built on the JavaScript engine that powered Google's web browser. Because there are a lot of JavaScript programmers in the world, Node.js grew incredibly quickly, and soon added a way to package libraries of code called the [Node Package Manager](https://www.npmjs.com/) (npm for short). + +Dahl soon realized the original Node.js implementation had some problems. Security wasn't built-in, and the npm ecosystem that had grown so quickly introduced vulnerabilities that were affecting millions of developers. JavaScript continued to evolve and [Dahl decided he wanted to try again with a new runtime](https://www.youtube.com/watch?v=M3BM9TB-8yA) that was secure by default, adopted modern web standards for features like including libraries of code, and came with a standard library of functionality. Enter Deno. + +### Differences with Node.js {#node-differences} + +The biggest reason you may not want to move your existing Node.js-based applications to Deno is npm modules aren't yet fully supported. The vast ecosystem of modules that have been built over the years aren't guaranteed to work with your application by default. The Deno team is [working on some approaches to allowing npm modules](https://deno.land/manual@v1.12.2/npm_nodejs), including [the built-in standard library that should obviate the need for many npm modules](https://deno.land/manual@v1.12.2/npm_nodejs/std_node.md), and [CDNs that host npm modules](https://deno.land/manual@v1.12.2/npm_nodejs/cdns.md) in a way that Deno can use. That said, not every npm module will work with Deno automatically, particularly more complex libaries, so you may need to wait on a rewrite or roll your own. + +## Installing Deno {#install} + +If you've written JavaScript for web browsers before, everything will feel very familiar. If you've written JavaScript or TypeScript for web _servers_ before, probably using Node.js as your runtime, most things will feel familiar with a [few key differences](#node-differences). + +Deno ships as a single executable with no external dependencies. Versions are available for macOS (both Intel and Apple silicon architectures), Linux, and Windows (64-bit support only for Linux and Windows). + +The easiest way to install is to call the [`deno_install` script](https://github.com/denoland/deno_install) remotely. + +On macOS and Linux: + +```bash +curl -fsSL https://deno.land/x/install/install.sh | sh +``` + +On Windows: + +```bash +iwr https://deno.land/x/install/install.ps1 -useb | iex +``` + +[The official installation guide](https://deno.land/manual/getting_started/installation) provides details for various other platforms. As Deno is open source, you are free to [compile from source](https://deno.land/manual/contributing/building_from_source) as well. + +Once installed, run `deno --version` to verify the installation was successful. The output should look something like the following: + +```bash +$ deno --version +deno 1.37.0* (release, x86_64-apple-darwin) +v8 10.* +typescript 4.* +``` + +The minimum version of Deno runtime required for developing workflow apps is currently at version 1.37.0 + +## Installing the Deno extension for VSCode {#vscode} + +The installation script from our [Getting started](/deno-slack-sdk/guides/getting-started#install-cli) should cover everything for you, but if you want to manually install the `vscode_deno` extension, follow these steps: + +1. Within VSCode, select **Extensions** from the sidebar. +1. Enter **Deno** in the search bar. +1. Select **Deno** (a language server client for Deno, published by deno.land) and **Install**. + +Alternatively, you can download and install the extension from [here](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno). + +Once installed, you should see a splash screen welcoming you to the extension. For additional information about configuring the extension, refer to [Using Visual Studio Code](https://docs.deno.com/runtime/manual/references/vscode_deno). + +## Onward {#onward} + +Ready to dive into development? [Let's go](/deno-slack-sdk/guides/developing-with-deno)! \ No newline at end of file diff --git a/docs/guides/deployment/controlling-permissions-for-admins.md b/docs/guides/deployment/controlling-permissions-for-admins.md new file mode 100644 index 00000000..018cb162 --- /dev/null +++ b/docs/guides/deployment/controlling-permissions-for-admins.md @@ -0,0 +1,79 @@ +--- +slug: /deno-slack-sdk/guides/controlling-permissions-for-admins +--- + +# Controlling permissions for Admins + + + +As part of the [broader access controls available to administrators](https://slack.com/help/categories/200122103-Workspace-administration#workspace-settings-permissions), administrators can ensure only approved apps are installed and available to users. + +## Approval process for admins {#approval-admins} + +If a workspace has the [Admin-Approved Apps](https://slack.com/help/articles/222386767-Manage-app-installation-settings-for-your-workspace) feature enabled, apps must be approved by a Workspace Admin (as set in your workspace settings) before they can be deployed. + +However, even if a workspace has Admin-Approved Apps enabled, workspace owners can still run `slack deploy` to deploy apps or `slack run` to run apps locally without requesting Admin-Approved Apps permission. The Admin-Approved Apps approval process does not apply to standalone workspaces. + +When a developer deploys an app, administrators will receive a notification, either from Slackbot or using the [Admin-Approved Apps API workflow](https://api.slack.com/admins/approvals) as determined by the organization. The approval notification will include which [OAuth scopes](https://api.slack.com/tutorials/tracks/understanding-oauth-scopes-bot) the app is requesting, as well as any outgoing domains the app may want to access. + +Outgoing domains are a new concept, and apply only to apps deployed to Slack's managed infrastructure. These are domains the app may require access to — for example, if a developer writes a [function](/deno-slack-sdk/guides/creating-slack-functions) that makes a request to an external API, they will need to include that API in their outgoing domains. Outgoing domains do not constrain which ports on those domains a function can communicate with. Administrators can now approve or deny apps based on these defined outgoing domains, in the same way they would OAuth scopes. + +### Admin-Approved Apps and connector functions for admins {#admin-connectors} + +Developers can create apps that call connector functions. These connector functions are contained by another app; for example, if a developer wishes to add a row to a Google spreadsheet or to update that same row, they could call the respective [Google Sheets connector functions](/deno-slack-sdk/reference/connector-functions#google_sheets). + +In addition to the approval process for developer apps described above, you can also explicitly approve or deny apps that use connector functions for use in Enterprise Grid workspaces based on the specified connector function. For example, if a developer's app calls a connector function that has not yet been approved for your workspace, you will be notified for approval when the developer attempts to install their app. In this example, you would approve or deny the specific Google Sheets connector function for use in your workspace. + +If you deny the connector function, running `manifest validate` will inform the developer that the connector function is denied for use in the workspace. If you approve it, running `manifest validate` will install the specified connector function to the workspace. + +For more information and a list of connector functions and their containing apps, refer to [connector functions](/deno-slack-sdk/reference/connector-functions). + +### Changes to the APIs {#api-changes} + +If you are using the [Admin-Approved Apps APIs](https://api.slack.com/admins/approvals) to manage your app approval process, there will be some changes to the API responses you receive as well as some new parameters that you can send to account for the new concept of outgoing domains that applies to apps deployed to Slack's managed infrastructure. + +The following endpoints will now have a `domains` field next to the existing `scopes` field, as a string array: + +* [`admin.apps.approved.list`](https://api.slack.com/methods/admin.apps.approved.list) +* [`admin.apps.restricted.list`](https://api.slack.com/methods/admin.apps.restricted.list) +* [`admin.apps.requests.list`](https://api.slack.com/methods/admin.apps.requests.list) + +A response would look like this: + +```json +"scopes": [ + { + "name": "app_mentions:read", + "description": "View messages that directly mention @your_slack_app in conversations that the app is in", + "token_type": "bot" + } +], +"domains": ['slack.com'], +``` + +Additionally, the following endpoints will now have an optional `domains` string array field for including outgoing domains that should be included in the approve or deny request: + +* [`admin.apps.approve`](https://api.slack.com/methods/admin.apps.approve) +* [`admin.apps.restrict`](https://api.slack.com/methods/admin.apps.approve) + +If the `domains` array is left empty, the method will look up the domains specified by the app. + +## Approval process for developers {#approval-developers} + +For developers, the most important thing to know is that you may run into extra steps when deploying your apps. If the administrators of your workspace have enabled [Admin-Approved Apps](https://slack.com/help/articles/222386767-Manage-app-installation-settings-for-your-workspace), it means your app requires approval before it can be deployed. + +In this case, after you run `slack deploy`, a prompt will notify you via the CLI that admin approval is required on this workspace. You'll also be prompted to enter `y` or `n` to send a request to the workspace administrator for approval to install your app. + +Administrators will see which OAuth scopes your app is requesting, as well as which outgoing domains your app is requesting access to. Outgoing domains are specified in the `outgoingDomains` array of your apps `manifest.ts` file as comma-separated strings. Administrators may also ask for an additional description for your app. If this is enabled, you will be asked to provide that information when you deploy your app using the CLI. + +Once you have approval, you'll receive a notification from Slackbot, and you can then deploy your app. If you receive a Slackbot notification that your app was denied, reach out to your workspace administrator. + +Finally, if your app needs to request a new OAuth scope or outgoing domain, it will again trigger the approval process above. The existing app installation will continue to function, but the new scope or outgoing domain will not be functional until the app is reapproved and redeployed. + +### Admin-Approved Apps and connector functions for developers {#dev-connectors} + +To request approval for an app that uses a connector function requiring admin approval, perform the following steps (Enterprise Grid workspaces only): + +1. Install your app via the CLI. If the app uses a connector function that requires admin approval, you will be prompted with that information about the connector function and asked if you would like to submit a request for approval. +1. Select `Y`. The CLI wil prompt you for a reason; after entering one, an approval request will be created for the admin of the workspace. Running `manifest validate` will show that the connector app is now pending approval. +1. If you enter `N`, the CLI will do nothing, and your app will not be submitted for approval. \ No newline at end of file diff --git a/docs/guides/deployment/deploying-to-slack.md b/docs/guides/deployment/deploying-to-slack.md new file mode 100644 index 00000000..5fa35856 --- /dev/null +++ b/docs/guides/deployment/deploying-to-slack.md @@ -0,0 +1,51 @@ +--- +slug: /deno-slack-sdk/guides/deploying-to-slack +--- + +# Deploying to Slack + + + +Your apps can be deployed to Slack's managed infrastructure by running the `slack deploy` command at the root of your project. + +No local development server is started when running the `slack deploy` command, and your app will have a different ID than the one created for your app if you previously ran the `slack run` command. For more details about the differences between local and deployed apps, refer to [team collaboration](/deno-slack-sdk/guides/collaborating-with-teammates). + +:::info + +`npm` dependencies are supported but are still in beta, so ensure that you test any `npm:` specifiers when using the `slack deploy` command (the `slack run` command is not affected). + +::: + +## Using the `slack deploy` command {#slack-deploy} + +When you run the `slack deploy` command and you are logged into a Slack Enterprise Grid, you may also be asked to select a workspace within your organization to deploy your app to (default is all workspaces). + +You may also specify a workspace within your organization to grant your app access to with the `--org-workspace-grant` flag. + +The Slack CLI will package up your app and deploy it to the workspace you specify. At that point, anyone in your workspace will be able to find and add your app by navigating to **Apps > Manage > Browse apps**. + +:::info + +The `slack deploy` command behavior may vary slightly based on the operating system used. + +::: + +### Function access {#function-access} + +To make a function available so that another user (or group of users) can access workflows that reference your function after you deploy your app, you can execute the `slack function access` command. After choosing your workspace, you'll also be prompted to select which function you want to deploy, as well as who you would like to give access to your function -- app collaborators only, specific users, or everyone. Your function will then be accessible to those users the next time you deploy your app. + +✨ **For more information about function access**, refer to [custom function access](/deno-slack-sdk/guides/controlling-access-to-custom-functions). + +:::info + +Workflow apps are currently not eligible for distributing to the Slack Marketplace. + +::: + +## Redeploying your app {#re-deploy} + +If you need to make any changes to your app, you must redeploy it using the `slack deploy` command again. + +In addition, if administrators of your workspace have enabled [Admin-Approved Apps](https://slack.com/help/articles/222386767-Manage-app-installation-settings-for-your-workspace), it means your app will need approval before it can be deployed or redeployed to your workspace. + +✨ **For more information about getting your app approved**, check out [access controls for developers](/deno-slack-sdk/guides/controlling-permissions-for-admins#dev-connectors). diff --git a/docs/guides/deployment/developing-locally.md b/docs/guides/deployment/developing-locally.md new file mode 100644 index 00000000..9b9859a0 --- /dev/null +++ b/docs/guides/deployment/developing-locally.md @@ -0,0 +1,55 @@ +--- +slug: /deno-slack-sdk/guides/developing-locally +--- + +# Developing locally + + + +As you're developing your workflow app, you can see your changes propagated to your workspace in real-time by using the `slack run` command. We refer to the workspace you develop in as your _local_ environment, and the workspace you deploy your app to as your _deployed_ environment. + +You are not required to deploy your app. In fact, you might never need to use the `slack deploy` command — maybe there's something you want to do just one single time, or only when you need to — you can use `slack run` for that. + +Otherwise, you should think of your local environment as a development environment. We even append the string `(local)` to the end of your app's name when running in this context. + +## Using the `slack run` command {#slack-run} + +When you enter `slack run` from the root directory of a project and you are logged into a Slack Enterprise Grid, you may also be asked to select a workspace within your organization to grant your app access to. + +If administrators of your workspace have enabled [Admin-Approved Apps](/deno-slack-sdk/guides/controlling-permissions-for-admins), it means your app will need approval before it can be installed to your workspace. + +✨ **For information about getting your app approved**, refer to [access controls for developers](/deno-slack-sdk/guides/controlling-permissions-for-admins#dev-connectors). + +The Slack CLI will then start a local development server and establish a connection to the `(local)` version of your app. Check that your instance of the Slack CLI is logged in to the desired workspace by running `slack auth list`. + +To start the local development server, use the `slack run` command: + +``` +$ slack run +``` + +You'll know your development server is ready when your terminal says the following: + +`Connected, awaiting events` + +To turn off the development server, enter `Ctrl`+`c` in the command line. + +## Creating link triggers in local development {#local-triggers} + +Link triggers are unique to each installed version of your app. This means that their "shortcut URLs" will differ across each workspace, as well as between local and [deployed](/deno-slack-sdk/guides/deploying-to-slack) apps. + +When creating a trigger, you must select the workspace you'd like to create the trigger in. Each workspace has a development version (denoted by `(local)`), as well as a deployed version. + +✨ **For more information about link triggers**, refer to [access controls for developers](/deno-slack-sdk/guides/creating-link-triggers). + +If your app has any triggers created within that development environment, they'll be listed when you run the `slack run` command. If you only created triggers within a production environment using the `slack deploy` command, they will not appear. + +## Creating triggers with the `slack run` command {#cli-trigger-prompt} + +If you have not used the `slack triggers create` command to create a trigger prior to running the `slack run` command, you will receive a prompt in the Slack CLI to do so. + +Let's say you've created a Slack app and tried to run the `slack run` command without first creating a trigger. The Slack CLI will ask you which workspace you'd like to run your app in, and will then prompt you to choose a trigger definition file. If you choose a file, the trigger will be created and the app will run. If you do not choose a trigger definition file or if you do not yet have one created, a trigger will not be created. No worries either way, as your app will still continue with the run operation. + +## App visibility {#visibility} + +As discussed above, once you create an app and run it using the `slack run` command, a link trigger will be generated for your app. Once that link trigger is posted in a public channel within your workspace, other channel members can click it and interact with your app even though your app has not been deployed. diff --git a/docs/guides/deployment/following-security-best-practices.md b/docs/guides/deployment/following-security-best-practices.md new file mode 100644 index 00000000..df5db296 --- /dev/null +++ b/docs/guides/deployment/following-security-best-practices.md @@ -0,0 +1,62 @@ +--- +slug: /deno-slack-sdk/guides/following-security-best-practices +--- + +# Following security best practices + + + +Keeping your apps and functions secure is an important part of developing on the Slack platform. + +Slack’s managed hosted environment is built on the [Deno](https://deno.land/) runtime, a secure Javascript runtime. Learn more about Deno’s [permissions model](https://deno.land/manual@v1.32.1/basics/permissions). + +Here are some best practices to keep in mind when developing [custom functions](/deno-slack-sdk/guides/creating-custom-functions), [workflows](/deno-slack-sdk/guides/creating-workflows) and [triggers](/deno-slack-sdk/guides/using-triggers) for Slack automations. + +## Set appropriate access control levels (ACLs) {#ACL} +* [Limit access](https://api.slack.com/future/triggers/link#manage) to your [functions](/deno-slack-sdk/guides/creating-custom-functions) and [triggers](/deno-slack-sdk/guides/using-triggers) to only the intended audience. Use the `slack function access` or `slack trigger access` commands to control who can use your functions or trip your triggers. + +* Your app collaborators can deploy your functions and manage your workflows and triggers. Only add collaborators to your app that you trust. + +## Validate input {#validate-input} +* It’s always important to validate inputs to your functions. If you’ve changed the distribution of your function, keep in mind it may be used in other workflows in ways you may not anticipate. Your app collaborators may also create or update triggers to start your workflows. + +* When using Slack [datastores](/deno-slack-sdk/guides/using-datastores), avoid [injection attacks](https://owasp.org/Top10/A03_2021-Injection/) by properly sanitizing user input when building queries, and use the `expression_values` and `expression_attributes` fields when querying for data. Also ensure that the user has read access to data from the datastore before processing it. + +* Always confirm the user has access to perform whatever operation your function is being asked to perform. In complex workflows, several users may participate in the various steps, so ensure you’re checking the correct user’s permissions. For example, in a contract approval workflow, anyone may be able to request an approval, but only certain approvers may actually provide an approval. + +* When listening to [message metadata events](/deno-slack-sdk/guides/integrating-message-metadata-events), keep in mind that many apps may be posting messages with the same event types. If you’d like to listen to messages from only specific apps, use a filter on the `app_id` in the event trigger definition. + +## Handle sensitive data +* Secrets needed by your custom functions running on Slack-managed infrastructure should be shared with Slack using the `slack env add` command and never hard-coded in your functions. Examples are API keys, OAuth client IDs and secrets, certificates, and cryptography keys. If possible, use Slack’s [third party auth support](https://api.slack.com/faq#third-party) to manage OAuth-based credentials. + +* For local apps using `slack run`, ensure that the `.env` file containing local secrets does not end up in your source control system. + +* When using `slack env add`, ensure the secret does not end up in your shell’s history. This can be done using shell environment variables, or by running `slack env add` without any parameters. With no parameters, `slack env add` will prompt you for the secret’s name and value in the console, using a password display. + +* Be careful about collecting or logging sensitive information in workflows from users, especially passwords or personally identifiable information (PII). Data may be exposed to later workflow steps, in data exports, or in activity logs. + +## Secure credentials {#secure-credentials} +* The Slack CLI stores credentials in the `credentials.json` file in the `.slack` folder in your home directory. Slack will never ask you share the tokens contained in that file. + +* While these credentials expire and are regularly rotated, access to this file should be limited to only you. + +* Never share the tokens or challenge strings generated by the `slack login` flow. + +* Never paste a `/slackauthticket` command given to you by another user into Slack. + +## Use secure libraries {#secure-libraries} +* It is your responsibility to monitor and respond to security vulnerabilities in your custom function’s code and dependencies, and to deploy new versions to Slack-managed infrastructure as needed. + +* Keep your Slack CLI and SDKs up to date by upgrading when prompted. + +* Only use a Slack CLI download by following instructions on api.slack.com. + +## Handle network egress {#network-egress} +* Slack’s `outgoingDomains` configuration limits which domains your custom function code can use when making external network requests. Only list `outgoingDomains` if the domains are required by your functions. + +## Make scopes and tokens {#scopes-tokens} +* Functions are given a short-lived token that can be used to make [Slack API](/deno-slack-sdk/guides/calling-slack-api-methods) calls, which use the scopes requested in the app’s manifest. We recommend only sending these tokens to Slack API endpoints, and not logging them or sending them to external systems. + +* Only request the scopes your functions need to do their job. + +Following these guidelines will get you on your way to building secure workflow apps. \ No newline at end of file diff --git a/docs/guides/deployment/logging-function-and-app-behavior.md b/docs/guides/deployment/logging-function-and-app-behavior.md new file mode 100644 index 00000000..846c2a90 --- /dev/null +++ b/docs/guides/deployment/logging-function-and-app-behavior.md @@ -0,0 +1,103 @@ +--- +slug: /deno-slack-sdk/guides/logging-function-and-app-behavior +--- + +# Logging function and app behavior + +When building workflow apps, you can use both function-level and app-level logging to troubleshoot and debug your app. + +## Local app logs {#local-logging} + +When developing locally, you can log information to the console of the terminal window where you are running your development server with calls to `console.log`. Calls to `console.log` are processed while your local developer server is running. + +Let's use the [Virtual Running Buddies sample app](https://github.com/slack-samples/deno-virtual-running-buddies) as an example. Say you'd like to print out items that are getting stored in your datastore to verify that you're getting the type of data you expect. Within the `log_run.ts` file, we'll add a `console.log` call to the function that logs a run and stores it in the datastore as follows: + +```javascript +// log_run.ts +export default SlackFunction(LogRunFunction, async ({ inputs, client }) => { + const { distance, rundate, runner } = inputs; + const uuid = crypto.randomUUID(); + + const putResponse = await client.apps.datastore.put({ + datastore: RUN_DATASTORE, + item: { + id: uuid, + runner: runner, + distance: distance, + rundate: rundate, + }, + }); + + if (!putResponse.ok) { + return { error: `Failed to store run: ${putResponse.error}` }; + } + console.log(putResponse); // add this line + return { outputs: {} }; +}); +``` + +Now when you run your app locally, you'll see output similar to the following in your terminal window: + +```bash +✨ yourname123 of Your DevEnv +Connected, awaiting events +2023-03-29 07:51:07 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution started for workflow 'Log a run' +2023-03-29 07:51:07 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 1 of 3 +2023-03-29 07:51:07 [info] [Fn010N] (Trace=Tr050M6Y6QF8) Function execution started for builtin function 'Open a form' +2023-03-29 07:51:17 [info] [Fn010N] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Open a form' +2023-03-29 07:51:18 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Open a form' +2023-03-29 07:51:19 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 2 of 3 +{ + ok: true, + datastore: "running_datastore", + item: { + runner: "ABC123DEF45", + id: "abcd1234-ef56-gh78-ij91-klmnop234567", + distance: 4.5, + rundate: "2023-03-29" + } +} +2023-03-29 07:51:19 [info] [Fn050TKLQJ3V] (Trace=Tr050M6Y6QF8) Function execution started for app function 'Log a run' +2023-03-29 07:51:20 [info] [Fn050TKLQJ3V] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Log a run' +2023-03-29 07:51:21 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Log a run' +2023-03-29 07:51:22 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 3 of 3 +2023-03-29 07:51:22 [info] [Fn0102] (Trace=Tr050M6Y6QF8) Function execution started for builtin function 'Send a message to channel' +2023-03-29 07:51:24 [info] [Fn0102] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Send a message to channel' +2023-03-29 07:51:25 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Send a message to channel' +2023-03-29 07:51:25 [info] [Fn051HE94GSU] (Trace=Tr050TQV7ERG) Function execution completed for function 'Log a run' +2023-03-29 07:51:25 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow 'Log a run' +``` + +Running `console.log` emits information that _you_ provide for your app to emit. If you want to see logs for _all_ your app's activity, you'll need to install your app and run `slack activity`. + +Once you run `slack activity` and select your workspace and local app environment, you'll see output similar to the following in your terminal window: + +```bash +✨ yourname123 of Your DevEnv +2023-03-29 07:51:07 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution started for workflow 'Log a run' +2023-03-29 07:51:07 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 1 of 3 +2023-03-29 07:51:07 [info] [Fn010N] (Trace=Tr050M6Y6QF8) Function execution started for builtin function 'Open a form' +2023-03-29 07:51:17 [info] [Fn010N] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Open a form' +2023-03-29 07:51:18 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Open a form' +2023-03-29 07:51:19 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 2 of 3 +2023-03-29 07:51:19 [info] [Fn050TKLQJ3V] (Trace=Tr050M6Y6QF8) Function execution started for app function 'Log a run' +2023-03-29 07:51:20 [info] [Fn050TKLQJ3V] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Log a run' +2023-03-29 07:51:21 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Log a run' +2023-03-29 07:51:22 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Executing workflow step 3 of 3 +2023-03-29 07:51:22 [info] [Fn0102] (Trace=Tr050M6Y6QF8) Function execution started for builtin function 'Send a message to channel' +2023-03-29 07:51:24 [info] [Fn0102] (Trace=Tr050M6Y6QF8) Function execution completed for function 'Send a message to channel' +2023-03-29 07:51:25 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow step 'Send a message to channel' +2023-03-29 07:51:25 [info] [Fn051HE94GSU] (Trace=Tr050TQV7ERG) Function execution completed for function 'Log a run' +2023-03-29 07:51:25 [info] [Wf050D7QTLR5] (Trace=Tr050M6Y6QF8) Execution completed for workflow 'Log a run' +``` + +✨ **For more information about developing locally**, refer to [local development](/deno-slack-sdk/guides/developing-locally). + +## Deployed app logs {#deployed-logging} + +After your app is deployed, all calls to `console.log` will be captured remotely, and will be emitted along with the last seven days of your app's activity via the `slack activity` command **only**. + +Once deployed, invoke some of your app's workflows, run `slack activity`, then select your workspace and deployed app environment. In your output, you'll see all of your calls to `console.log` in addition to the workflow steps and function executions, [external auth information](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication), and any errors encountered when running your app. + +✨ **For more information about deploying your app**, refer to [deploy to Slack](/deno-slack-sdk/guides/deploying-to-slack). + diff --git a/docs/guides/functions/calling-slack-api-methods.md b/docs/guides/functions/calling-slack-api-methods.md new file mode 100644 index 00000000..79782994 --- /dev/null +++ b/docs/guides/functions/calling-slack-api-methods.md @@ -0,0 +1,132 @@ +--- +slug: /deno-slack-sdk/guides/calling-slack-api-methods +--- + +# Calling Slack API methods + + + +With workflow apps, you can interface with the Slack API we've come to know and love. While building, you can access and make calls to the Slack API via the `client` context property of a `SlackFunction`. Think of `SlackFunction` as a utility employed for implementing [custom functions](/deno-slack-sdk/guides/creating-custom-functions). It ensures input validity, provides auto-complete functionality, and manages the app's access token so you don't have to (you can thank us later). + +To use the `client` context property, import `SlackFunction` from `deno-slack-sdk` to your custom function file and add the `client` context property to your `SlackFunction`, as shown below. Only the `inputs` and `client` context properties are shown in this example, but other context properties you might want to include are outlined on the [custom functions](/deno-slack-sdk/guides/creating-custom-functions#context) page. + +```js +// functions/example_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const ExampleFunctionDefinition = DefineFunction({ + callback_id: "example_function_def", + title: "Example function", + source_file: "functions/example_function.ts", +}); + +export default SlackFunction( + ExampleFunctionDefinition, + ({ inputs, client }) => { // Add `client` here + + // ... +``` + +The `client` property allows you to access the Slack API in one of two ways: + +- `client..` (e.g., `client.chat.postMessage`) +- `client.apiCall('')` (e.g., `client.apiCall('chat.postMessage', {/* ... */});`) + +These API calls return a promise, so be sure to `await` their responses. A promise in TypeScript allows for asynchronous programming - handling multiple tasks at the same time. + +For example, let's look at our sample app, [Deno Request Time Off](https://github.com/slack-samples/deno-request-time-off), and send a message to a user's manager to request time off: + +```js +// functions/send_time_off_request_to_manager/definition.ts +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const SendTimeOffRequestToManagerFunction = DefineFunction({ + callback_id: "send_time_off_request_to_manager", + title: "Request Time Off", + description: "Sends your manager a time off request to approve or deny", + source_file: "functions/send_time_off_request_to_manager/mod.ts", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + employee: { + type: Schema.slack.types.user_id, + description: "The user requesting the time off", + }, + manager: { + type: Schema.slack.types.user_id, + description: "The manager approving the time off request", + }, + start_date: { + type: "slack#/types/date", + description: "Time off start date", + }, + end_date: { + type: "slack#/types/date", + description: "Time off end date", + }, + reason: { + type: Schema.types.string, + description: "The reason for the time off request", + }, + }, + required: [ + "employee", + "manager", + "start_date", + "end_date", + "interactivity", + ], + }, + output_parameters: { + properties: {}, + required: [], + }, +}); + +// functions/send_time_off_request_to_manager/mod.ts + +import { SendTimeOffRequestToManagerFunction } from "./definition.ts"; +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import BlockActionHandler from "./block_actions.ts"; +import { APPROVE_ID, DENY_ID } from "./constants.ts"; +import timeOffRequestHeaderBlocks from "./blocks.ts"; + +export default SlackFunction( + SendTimeOffRequestToManagerFunction, + async ({ inputs, client }) => { + console.log("Forwarding the following time off request:", inputs); + + // ... + + // Send the message to the manager + const msgResponse = await client.chat.postMessage({ + channel: inputs.manager, + blocks, + // Fallback text to use when rich media can't be displayed (i.e. notifications) as well as for screen readers + text: "A new time off request has been submitted", + }); + + if (!msgResponse.ok) { + console.log("Error during request chat.postMessage!", msgResponse.error); + } + + // IMPORTANT! Set `completed` to false in order to keep the interactivity + // points (the approve/deny buttons) "alive" + // We will set the function's complete state in the button handlers below. + return { + completed: false, + }; + } + + // ... +``` + +Most API endpoints require specific [permission scopes](https://api.slack.com/scopes). Add scopes to your app by listing them in the `botScopes` property of your [manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +## Slack API methods +You can call any Slack API method that is accessible via a [bot token](https://api.slack.com/concepts/token-types#bot) listed in the [method documentation](https://api.slack.com/methods). On those pages, you will discover more ways to access Slack API methods, but for use within workflow apps, we recommend using `SlackFunction` outlined here. + +## Next up +➡️ **To learn more about the custom functions** you can implement using `SlackFunction`, check out [custom functions](/deno-slack-sdk/guides/creating-custom-functions). diff --git a/docs/guides/functions/controlling-access-to-custom-functions.md b/docs/guides/functions/controlling-access-to-custom-functions.md new file mode 100644 index 00000000..43fc63e8 --- /dev/null +++ b/docs/guides/functions/controlling-access-to-custom-functions.md @@ -0,0 +1,132 @@ +--- +slug: /deno-slack-sdk/guides/controlling-access-to-custom-functions +--- + +# Controlling access to custom functions + + + +To make a function available so that another user (or many users) can access workflows that reference that function, you'll use the [`function access`](/slack-cli/reference/commands/slack_function) Slack CLI command. At this time, functions can be made available to: + +* _everyone_ in workspaces where the app has access, +* your app's _collaborators_, +* or _specific users_. + +In order to enable the [`function access`](/slack-cli/reference/commands/slack_function) command, your app must have been deployed _at least once before_ attempting to make your function available to others. + +You must also re-deploy your app after using `function access`. Anytime you make permission changes to your function using the `function access` command, your app must be redeployed, _each time after_, in order for the updates to be available in your app's workspace. + +## Grant access to one person {#single-access} + +Given: +- a function with a [callback ID](/deno-slack-sdk/guides/creating-custom-functions#fields) of `get_next_song` +- a user with ID `U1234567` + +You can make your `get_next_song` function available to the user `U1234567` like this: + +```cmd +$ slack function access --name get_next_song --users U1234567 --grant +``` + +_To revoke access, replace `--grant` with `--revoke`._ + +## Grant access to multiple people {#multi-access} + +Given: +- a function with a [callback ID](/deno-slack-sdk/guides/creating-custom-functions#fields) of `calculate_royalties` +- users with the following IDs: `U1111111`, `U2222222`, and `U3333333` + +You can make your function `calculate_royalties` available to the above users like this: + +```cmd +$ slack function access --name calculate_royalties --users U1111111,U2222222,U3333333 --grant +``` + +_To revoke access, replace `--grant` with `--revoke`._ + +## Grant access to all collaborators {#collaborator-access} + +Given: +- a function with a [callback ID](/deno-slack-sdk/guides/creating-custom-functions#fields) of `notify_escal_team` + +You can make your `notify_escal_team` function available to all of your app's [collaborators](/slack-cli/reference/commands/slack_collaborator) like this: + +```cmd +$ slack function access --name notify_escal_team --app_collaborators --grant +``` + +## Grant access to all workspace members {#all-access} + +Given: +- a function with a [callback ID](/deno-slack-sdk/guides/creating-custom-functions#fields) of `get_customer_profile` + +You can make your `get_customer_profile` function available to everyone in your workspace like this: + +```cmd +$ slack function access --name get_customer_profile --everyone --grant +``` + +## Grant access using the prompt-based approach {#distribute-prompt} + +The prompt-based approach allows you to distribute your function to one user, to multiple people, to collaborators, or to everyone in an interactive prompt. + +To activate the flow, use the following command in your terminal: + +```cmd +$ slack function access +``` + +Given: +- a function with a [callback ID](/deno-slack-sdk/guides/creating-custom-functions#fields) of `reverse` + +You will answer the first prompt in the following manner: + +#### Choose the name of the function you'd like to distribute + +```cmd +> reverse (Reverse) +``` +#### Choose who you'd like to to have access to your function + +If going from `everyone` or `app_collaborators` **to** specific users, you should be offered the option of adding collaborators to specific users. + +```cmd +> specific users (current) + app collaborators only + everyone +``` +#### Choose an action + +```cmd +> granting a user access + revoking a user's access +``` +#### Provide ID(s) of one or more user in your workspace + +Given: +- a user's ID in your workspace: `U0123456789` + +You will answer the following prompt below: + +```cmd +: U0123456789 +``` +You can add multiple users at the same time. To do this, separate the user IDs with a comma (e.g. `U0123456789`, `UA987654321`). + +After you've finished this flow, you'll receive a message indicating the type of distribution you chose. + +## Guests and external users {#guests-external} + +Guests and external users are limited in the workflows they may run based on the scopes defined for the functions in the workflows. There is a predefined set of scopes that are considered "risky", which guest users cannot run. These scopes include the following: + +* [`channels:manage`](https://api.slack.com/scopes/channels:manage) - create and manage public channels +* [`channels:write.invites`](https://api.slack.com/scopes/channels:write.invites) - invite members to public channels +* [`groups:write`](https://api.slack.com/scopes/groups:write) - create and manage private channels +* [`groups:write.invites`](https://api.slack.com/scopes/groups:write.invites) - invite members to private channels +* [`usergroups:write`](https://api.slack.com/scopes/usergroups:write) - create and manage usergroups + +If a guest or external user attempts to run a workflow containing a function with one of these scopes, they will receive an error. + +## More distribution options {#distribution-options} + +For more distributions options, including how to **revoke** access, head to the [distribute command reference](/slack-cli/reference/commands/slack_function). \ No newline at end of file diff --git a/docs/guides/functions/creating-connector-functions.md b/docs/guides/functions/creating-connector-functions.md new file mode 100644 index 00000000..ec5bba63 --- /dev/null +++ b/docs/guides/functions/creating-connector-functions.md @@ -0,0 +1,17 @@ +--- +slug: /deno-slack-sdk/guides/creating-connector-functions +--- + +# Creating connector functions + +Connectors are step functions for workflows that behave like [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) for services external to Slack. They take inputs and perform work for you when added as steps to your [workflows](/deno-slack-sdk/guides/creating-workflows). + +We recommend understanding the systems and APIs you're integrating with before setup. + +:::info + +To protect your organization, external users (those outside your organization connected through Slack Connect) cannot use a workflow that contains connector functions built by your organization. This may manifest in a `home_team_only` warning. Refer to [this help center article](https://slack.com/help/articles/14844871922195-Slack-administration--Manage-workflow-usage-in-Slack-Connect-conversations#enterprise-grid-1) for more details. + +::: + +[Browse all Connector functions here](/deno-slack-sdk/reference/connector-functions) diff --git a/docs/guides/functions/creating-custom-functions.md b/docs/guides/functions/creating-custom-functions.md new file mode 100644 index 00000000..dfbd67d0 --- /dev/null +++ b/docs/guides/functions/creating-custom-functions.md @@ -0,0 +1,485 @@ +--- +slug: /deno-slack-sdk/guides/creating-custom-functions +--- + +# Creating custom functions + + + +Custom functions are how you define custom workflow steps. + +They have three main components: + +* **Inputs**, which can come from a workflow's [trigger](/deno-slack-sdk/guides/using-triggers) or the outputs of a previous step +* **Logic**, which is your own code that carries out your instructions, +* **Outputs**, which allows your function to pass on the result of its computations to follow-on steps in Workflow Builder + +:::info + +To protect your organization, external users (those outside your organization connected through Slack Connect) cannot use a workflow that contains [connector steps](/deno-slack-sdk/reference/connector-functions) or [workflow steps](https://api.slack.com/concepts/workflow-steps) built by your organization. This may manifest in a `home_team_only` warning. Refer to [this help center article](https://slack.com/help/articles/14844871922195-Slack-administration--Manage-workflow-usage-in-Slack-Connect-conversations#enterprise-grid-1) for more details. + +::: + +## Define a custom function {#define} + +Functions are defined via the `DefineFunction` method, which is part of the [Slack SDK](https://github.com/slackapi/deno-slack-sdk) that is included with every newly-created project. Both the definition and implementation for your functions should live in the same file, so to keep your app organized, put all your function files in a `functions` folder in your app's root folder. + +Let's take a look at the [`greeting_function.ts`](https://github.com/slack-samples/deno-hello-world/blob/main/functions/greeting_function.ts) within the [Hello World](https://github.com/slack-samples/deno-hello-world) sample app: + +```javascript +// /slack-samples/deno-hello-world/functions/greeting_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const GreetingFunctionDefinition = DefineFunction({ + callback_id: "greeting_function", + title: "Generate a greeting", + description: "Generate a greeting", + source_file: "functions/greeting_function.ts", + input_parameters: { + properties: { + recipient: { + type: Schema.slack.types.user_id, + description: "Greeting recipient", + }, + message: { + type: Schema.types.string, + description: "Message to the recipient", + }, + }, + required: ["message"], + }, + output_parameters: { + properties: { + greeting: { + type: Schema.types.string, + description: "Greeting for the recipient", + }, + }, + required: ["greeting"], + }, +}); +``` + +Note that we import `DefineFunction`, which is used for defining our function, and also `SlackFunction`, which we'll use to implement our function in the [Implement a custom function](#implement) section. + + +### Custom function fields {#fields} + +| Field | Description | Required? | +| ---- | ------------------ | --- | +| `callback_id` | A unique string identifier representing the function; max 100 characters. No other functions in your application may share a callback ID. Changing a function's callback ID is not recommended, as the function will be removed from the app and created under the new callback ID, breaking any workflows referencing the old function. | Required | +| `title` | A string to nicely identify the function. Max 255 characters. | Required | +| `source_file` | The relative path from the project root to the function handler file (i.e., the source file). _Remember to update this if you start nesting your functions in folders._ | Required | +| `description` | A succinct summary of what your function does. | Optional | +| [`input_parameters`](#input-output) | An object which describes one or more input parameters that will be available to your function. Each top-level property of this object defines the name of one input parameter available to your function.| Optional | +| [`output_parameters`](#input-output) | An object which describes one or more output parameters that will be returned by your function. Each top-level property of this object defines the name of one output parameter your function makes available. | Optional | + +#### Input and output parameters {#input-output} + +Functions can (and generally should) declare inputs and outputs. Inputs are declared in the `input_parameters` property, and outputs are declared in the `output_parameters` property. + +A custom function's `input_parameters` and `output_parameters` properties have two sub-properties: +* `required`, which is how you can ensure that a function requires a specific parameter. +* `properties`, where you can list the specific parameters that your function accounts for. + +Parameters are listed in the `properties` sub-property. The value for a parameter needs to be an object with further sub-properties: + * `type`: The type of the input parameter. This can be a [built-in type](/deno-slack-sdk/reference/slack-types) or a [custom type](/deno-slack-sdk/guides/creating-a-custom-type) that you define. + * `description`: A string description of the parameter. + +For example, if you have an input parameter named `customer_id` that you want to be required, you can do so like this: + +```javascript +input_parameters: { + properties: { + customer_id: { + type: Schema.types.string, + description: "The customer's ID" + } + }, + required: ["customer_id"] +} +``` + +If your input or output parameter is a [custom type](/deno-slack-sdk/guides/creating-a-custom-type) with required sub-properties, use the `DefineProperty` function to to ensure that each sub-property's required status is respected. Let's look at an example. Given an `input_parameter` of `msg_context` with three sub-properties, `message_ts`, `channel_id`, and `user_id`, this is how we would ensure that `message_ts` is required: + +```javascript +const messageAlertFunction = DefineFunction({ + ... + input_parameters: { + properties: { + msg_context: DefineProperty({ + type: Schema.types.object, + properties: { + message_ts: { type: Schema.types.string }, + channel_id: { type: Schema.types.string }, + user_id: { type: Schema.types.string }, + }, + required: ["message_ts"] + }) + } + }, + }); +``` + +:::warning[Object types are not supported within Workflow Builder at this time] + +If your function will be used within Workflow Builder, we suggest not using the Object types at this time. Check out [Typescript-friendly type definitions](/deno-slack-sdk/guides/creating-a-custom-type#define-property) for more details. + +::: + +While, strictly speaking, input and output parameters are optional, they are a common and standard way to pass data between functions and nearly any function you write will expect at least one input and pass along an output. + +Functions are similar in philosophy to Unix system commands: they should be minimalist, modular, and reusable. Expect the output of one function to eventually become the input of another, with no other frame of reference. + +After defining your custom function, declare it in your app's manifest file: + +```javascript +// /manifest.ts + +// Import the function +import { GreetingFunctionDefinition } from "./functions/greeting_function.ts" + +// ... + +export default Manifest({ + //... + functions: [GreetingFunctionDefinition], + //... +}); +``` + +Once your function is defined in your app's manifest file, the next step is to implement the function in its respective source file. + +## Implement a custom function {#implement} + +To keep your project tidy, implement your functions in the same source file in which you defined them. + +Implementation involves creating a `SlackFunction` default export. This example is again from the [`greeting_function.ts`](https://github.com/slack-samples/deno-hello-world/blob/main/functions/greeting_function.ts) within the [Hello World](https://github.com/slack-samples/deno-hello-world) sample app: + +```javascript +// /slack-samples/deno-hello-world/functions/greeting_function.ts + +}); // end of DefineFunction + +export default SlackFunction( + // Pass along the function definition from earlier in the source file + GreetingFunctionDefinition, + ({ inputs }) => { // Provide any context properties, like `inputs`, `env`, or `token` + // Implement your function + const { recipient, message } = inputs; + const salutations = ["Hello", "Hi", "Howdy", "Hola", "Salut"]; + const salutation = + salutations[Math.floor(Math.random() * salutations.length)]; + const greeting = + `${salutation}, <@${recipient}>! :wave: Someone sent the following greeting: \n\n>${message}`; + + // Don't forget any required output parameters + return { outputs: { greeting } }; + }, +); +``` + +It is important to store your environment variables, as custom functions deployed to Slack will not run with the `--allow-env` permission. When locally running your app using `slack run`, the CLI will automatically load your local `.env` file and populate the `env` function input parameter. However, when deploying your app using `slack deploy`, the values you added using `slack env add` will be available in the `env` function input parameter. Refer to [environment variables](/slack-cli/guides/using-environment-variables-with-the-slack-cli) for more information. + +Similarly, when using a [locally running your app](/deno-slack-sdk/guides/developing-locally), you can use `console.log` to emit information to the console. However, when your app is [deployed to production](/deno-slack-sdk/guides/deploying-to-slack), any `console.log` commands are available via `slack activity`. Check out our [Logging](/deno-slack-sdk/guides/logging-function-and-app-behavior) page for more. + +When composing your functions, you can: + +* leverage external APIs, and even store API credentials, using the CLI's [`slack env add`](/slack-cli/reference/commands/slack_env_add) command +* [call Slack API methods](/deno-slack-sdk/guides/calling-slack-api-methods) or [third-party APIs](https://api.slack.com/faq#third-party) +* store and retrieve data from [datastores](/deno-slack-sdk/guides/using-datastores) + +You can also encapsulate your business logic separately from the function handler, then import what you need and build your functions that way. + + +:::info[Function timeouts] + +When building workflows using functions, there is a 60 second timeout for a deployed function and a 15 second timeout for a locally-run function. + +For deployed functions using a `block_suggestion`, `block_actions`, `view_submission`, or `view_closed` payload, there is a 10 second timeout. + +If a top-level custom function has not finished running within its respective time limit, you will see an error in your log. Refer to [logging](/deno-slack-sdk/guides/logging-function-and-app-behavior) for more details. This error may differ when running the function locally versus a deployed function. For example, a function that calls a third-party API could complete outside of the timeout and return after Slack has already marked the function as timed out. Locally, this may result in a `token_revoked` error. If deployed, it would return an error that the timeout was reached. + +If an [interactivity handler function](/deno-slack-sdk/guides/adding-interactivity#interactivity-handlers) times out, an error will render in the Slack client, but not in the logs. + +::: + +### Function context properties {#context} + +Your function handler's context supports several properties that you can use by declaring them. + +Here are all the context properties available: + +| Property | Kind | Description | +| ----- | ---- | ----------- | +| `env` | String | Represents environment variables available to your function's execution context. A locally running app gets its `env` properties populated via the local `.env` file. A deployed app gets its `env` properties populated via the CLI's [`slack env add`](/slack-cli/reference/commands/slack_env_add) command. | +| `inputs` | Object | Contains the input parameters you defined as part of your function definition. | +| `client` | Object | An API client ready for use in your function. Useful for [calling Slack API methods](/deno-slack-sdk/guides/calling-slack-api-methods). | +| `token` | String | Your application's access token.| +| `event` | Object | Contains the full incoming event details. | +| `team_id` | String | The ID of your Slack workspace, i.e. T123ABC456. | +| `enterprise_id` | String | The ID of the owning enterprise organization, i.e. "E123ABC456". Only applicable for [Slack Enterprise Grid](https://api.slack.com/enterprise/grid) customers, otherwise its value will be set to an empty string. | + +The object returned by your function supports the following properties: + +| Property | Kind | Description | +| ------ | ---- | ----------- | +| `error` | String | Indicates the error that was encountered. If present, the function will return an error regardless of what is passed to outputs. | +| `outputs` | Object | Exactly matches the structure of your function definition's output_parameters. This is required unless an error is returned. | +| `completed` | Boolean | Indicates whether or not the function is completed. This defaults to `true`. | + + +➡️ **To keep building your app**, head to the [workflows](/deno-slack-sdk/guides/creating-workflows) section to learn how to add a custom function to a workflow. + +➡️ **To learn how to distribute your custom function**, refer to the [custom function access guide](/deno-slack-sdk/guides/controlling-access-to-custom-functions)! + +--- + +## Graceful errors {#error-handling} + +To ensure that errors in your function are handled gracefully, consider wrapping your logic in a try-catch block, and ensure you're returning an empty `outputs` property along with an `error` property: + +```js +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import type { GetCustomerNameFunction } from "../manifest.ts"; +import { GetCustomerInfo } from "../mycorp/get_customer_info.ts"; + +export default SlackFunction( + GetCustomerNameFunction, + async ({inputs, client}) => { + console.log(`Getting profile for customer ID ${inputs.customer_id}...`); + let response; + try { + response = await GetCustomerInfo(inputs.customer_id); + } catch (error) { + if (error instanceof Deno.errors.NotFound) { + return { + error: `Could not find customer where ID == ${inputs.customer_id}!`, + outputs: {}, + }; + } + } + return { + outputs: { + first_name: response?.first_name, + last_name: response?.last_name, + }, + }; +}); + +``` + +```js +// mycorp/get_customer_info.ts +export interface Customer { + id: number; + first_name: string; + last_name: string; +} +export default function GetCustomerInfo(id: number): Customer { + if (id == 1) { + const customer: Customer = { + id: 1, + first_name: "Some", + last_name: "Person", + }; + + // Maybe here there's some third-party API we call + return customer; + } else { + throw new Deno.errors.NotFound(); + } +} +``` + +## Testing custom functions {#testing} + +During development, you may want to test your [custom functions](/deno-slack-sdk/guides/creating-custom-functions) before deploying them to production. You can do this by creating a unit test for each custom function you want to validate. Since we're developing in the [Deno](/deno-slack-sdk/guides/installing-deno) environment, we'll be working with the [`Deno.test` API](https://deno.land/manual/basics/testing#writing-tests). + +Let's go through a couple of examples from our sample apps. + +Using the `SlackFunctionTester`, we can specify the inputs to a function and then verify the outputs that function provides in order to ensure it is working properly. In other words, the `SlackFunctionTester` allows us to create the context for our function so that we can pass in the necessary parameters in order to test that function. Let's get started! + +--- + +The first thing we'll do is create a new test file named after our function. + +For example, in the [Hello World sample app](https://github.com/slack-samples/deno-hello-world), the file containing our function is called [`greeting_function.ts`](https://github.com/slack-samples/deno-hello-world/blob/main/functions/greeting_function.ts), and the file containing our test for the function is called [`greeting_function_test.ts`](https://github.com/slack-samples/deno-hello-world/blob/main/functions/greeting_function_test.ts). + +We'll import our function into the test file as follows: + +```javascript +import GreetingFunction from "./greeting_function.ts"; +``` + +Then, we'll import `SlackFunctionTester` into the test file: + +```javascript +import { SlackFunctionTester } from "deno-slack-sdk/mod.ts"; +``` + +And, one more import — the specific Deno assertion method that we'll be using from the `Deno.test` API. In this case, we'll need the `assertEquals` method: + +```javascript +import { assertEquals } from "https://deno.land/std@0.153.0/testing/asserts.ts"; +``` + +We can initialize an instance of the `SlackFunctionTester` we mentioned earlier to create a context for our function: + +```javascript +const { createContext } = SlackFunctionTester("greeting_function"); +``` + +To summarize our structure, here is the original file containing our function: + +```javascript +// greeting_function.ts + +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const GreetingFunctionDefinition = DefineFunction({ + callback_id: "greeting_function", + title: "Generate a greeting", + description: "Generate a greeting", + source_file: "functions/greeting_function.ts", + input_parameters: { + properties: { + recipient: { + type: Schema.slack.types.user_id, + description: "Greeting recipient", + }, + message: { + type: Schema.types.string, + description: "Message to the recipient", + }, + }, + required: ["message"], + }, + output_parameters: { + properties: { + greeting: { + type: Schema.types.string, + description: "Greeting for the recipient", + }, + }, + required: ["greeting"], + }, +}); + +export default SlackFunction( + GreetingFunctionDefinition, + ({ inputs }) => { + const { recipient, message } = inputs; + const salutations = ["Hello", "Hi", "Howdy", "Hola", "Salut"]; + const salutation = + salutations[Math.floor(Math.random() * salutations.length)]; + const greeting = + `${salutation}, <@${recipient}>! :wave: Someone sent the following greeting: \n\n>${message}`; + return { outputs: { greeting } }; + }, +); +``` + +And here we have our test file with the items we imported and our instance of the `SlackFunctionTester`: + +```javascript +// greeting_function_test.ts + +import GreetingFunction from "./greeting_function.ts"; +import { SlackFunctionTester } from "deno-slack-sdk/mod.ts"; +import { assertEquals } from "https://deno.land/std@0.153.0/testing/asserts.ts"; + +const { createContext } = SlackFunctionTester("greeting_function"); + +Deno.test("Greeting function test", async () => { + const inputs = { message: "Welcome to the team!" }; + const { outputs } = await GreetingFunction(createContext({ inputs })); + assertEquals( + outputs?.greeting.includes("Welcome to the team!"), + true, + ); +}); +``` + +Once we pass in the text we expect our function to output, we compare the two values, then check to see if the values are indeed a match. + +--- + +Let's look at another example, this time from the [GitHub Issue sample app](https://github.com/slack-samples/deno-github-functions). + +Similarly to the Hello World example, we have a file containing our function called [`create_issue.ts`](https://github.com/slack-samples/deno-github-functions/blob/main/functions/create_issue.ts), and a file containing our test for the function, which is called [`create_issue_test.ts`](https://github.com/slack-samples/deno-github-functions/blob/main/functions/create_issue_test.ts). Let's look at the test file below: + +```javascript +// create_issue_test.ts + +import * as mf from "https://deno.land/x/mock_fetch@0.3.0/mod.ts"; +import { assertEquals } from "https://deno.land/std@0.153.0/testing/asserts.ts"; +import { SlackFunctionTester } from "deno-slack-sdk/mod.ts"; +// import our original function as a handler +import handler from "./create_issue.ts"; + +mf.install(); + +mf.mock("POST@/api/apps.auth.external.get", () => { + return new Response(`{"ok": true, "external_token": "example-token"}`); +}); + +mf.mock("POST@/repos/slack-samples/deno-github-functions/issues", () => { + return new Response( + `{"number": 123, "html_url": "https://www.example.com/expected-html-url"}`, + { + status: 201, + }, + ); +}); + +const { createContext } = SlackFunctionTester("create_issue"); +const env = { logLevel: "CRITICAL" }; + +Deno.test("Create a GitHub issue with given inputs", async () => { + const inputs = { + githubAccessTokenId: {}, + url: "https://github.com/slack-samples/deno-github-functions", + githubIssue: { + title: "The issue title", + }, + }; + const { outputs } = await handler(createContext({ inputs, env })); + // Assert whether the collection of mocked URL responses we use as inputs matches the outputs from our function. + assertEquals(outputs?.GitHubIssueNumber, 123); + assertEquals( + outputs?.GitHubIssueLink, + "https://www.example.com/expected-html-url", + ); +}); +``` + +This sample makes API calls to both Slack and GitHub, and therefore requires special mocking in its test. In the test, we'll import a module called [mock fetch](https://deno.land/x/mock_fetch). This module mocks Deno's [`fetch`](https://deno.land/manual/examples/fetch_data) method, which is used to make HTTP requests. We will use `mock_fetch` to mock the responses of the Slack API. + +✨ **For more information about mocking responses**, refer to [mocking](https://deno.land/manual/basics/testing/mocking#mocking) and [mock_fetch](https://deno.land/x/mock_fetch). + +### Running a test {#run-test} + +From the command line, run [`deno test`](https://deno.land/manual/basics/testing#running-tests) and call the file that contains your test function, as in the following example: + +``` +$ deno test greeting_function_test.ts +``` + +If you're in the base directory for your project, run this command as follows: + +``` +$ deno test functions/greeting_function_test.ts +``` + +If you want to run all of your function tests, run this command without any file names as follows: + +``` +$ deno test +``` + +✨ **For more information about Deno's built-in test runner**, refer to [testing](https://deno.land/manual/basics/testing). + +### Integrating a test into your CI/CD pipeline {#cicd-test} + +For more information, refer to [Setting up CI/CD with the Slack CLI](/slack-cli/guides/setting-up-ci-cd-with-the-slack-cli). diff --git a/docs/guides/functions/creating-functions.md b/docs/guides/functions/creating-functions.md new file mode 100644 index 00000000..963ec579 --- /dev/null +++ b/docs/guides/functions/creating-functions.md @@ -0,0 +1,21 @@ +--- +slug: /deno-slack-sdk/guides/creating-functions +--- + +# Creating functions + + + +Functions are one of the three building blocks that make up workflow apps. You will encounter all three as you navigate the path of building your app: + +1. Functions define the actions of your app. (⬅️ you are here) +2. Workflows are a combination of functions, executed in order. +3. Triggers execute workflows. + +There are three types of functions: + +* **[Slack functions](/deno-slack-sdk/guides/creating-slack-functions)** enable Slack-native actions, like creating a channel or sending a message. +* **[Connector functions](/deno-slack-sdk/guides/creating-connector-functions)** enable actions native to services outside of Slack. Google Sheets, Dropbox and Microsoft Excel are just a few of the services with available connector functions. Connector functions cannot be used in a workflow intended for use in a Slack Connect channel. +* **[Custom functions](/deno-slack-sdk/guides/creating-custom-functions)** enable developer-specific actions. Pass in any desired inputs, perform any actions you can code up, and pass on outputs to other parts of your workflows. + +Custom functions also allow your app to create and process workflow steps that users can add in Workflow Builder. See the [Workflow Builder custom step tutorial](/deno-slack-sdk/tutorials/workflow-builder-custom-step/) for instruction. \ No newline at end of file diff --git a/docs/guides/functions/creating-slack-functions.md b/docs/guides/functions/creating-slack-functions.md new file mode 100644 index 00000000..e6c1da34 --- /dev/null +++ b/docs/guides/functions/creating-slack-functions.md @@ -0,0 +1,62 @@ +--- +slug: /deno-slack-sdk/guides/creating-slack-functions +--- + +# Creating Slack functions + + + +Slack functions are essentially Slack-native actions, like creating a channel or sending a message. Use them alongside your [custom functions](/deno-slack-sdk/guides/creating-custom-functions) in a [workflow](/deno-slack-sdk/guides/creating-workflows). [Browse our inventory of Slack functions](/deno-slack-sdk/reference/slack-functions). + +Slack functions need to be imported from the standard library built into the [Slack Deno SDK](https://github.com/slackapi/deno-slack-sdk)—all Slack functions are children of the `Schema.slack.functions` object. Just like custom functions, Slack functions can be added to steps in a workflow using the `addStep` method. + +Slack functions define their own inputs and outputs, as detailed for each Slack function in the catalog below. + +## Slack functions catalog {#catalog} + +The details for each Slack function can be found in our [reference documentation](/deno-slack-sdk/reference/slack-functions). + +## Slack function example {#example} + +Here's an example of a workflow that creates a new Slack channel using the [`CreateChannel`](/deno-slack-sdk/reference/slack-functions/create_channel) Slack function: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +// Define a workflow that can pass the parameters for the Slack function +const myWorkflow = DefineWorkflow({ + callback_id: "channel-creator", + title: "Channel Creator", + input_parameters: { + properties: { channel_name: { type: Schema.types.string } }, + required: ["channel_name"], + }, +}); + +const createChannelStep = myWorkflow.addStep( + Schema.slack.functions.CreateChannel, + { + channel_name: myWorkflow.inputs.channel_name, + is_private: false, + }, +); + +export default myWorkflow; +``` + +## Timeouts for functions + + When building workflows using functions, there is a 60 second timeout for a deployed function and a 15 second timeout for a locally-run function. + + For deployed functions using a `block_suggestion`, `block_actions`, `view_submission`, or `view_closed` payload, there is a 10 second timeout. + + If a function has not finished running within its respective time limit, you will see an error in your log. Refer to [logging](/deno-slack-sdk/guides/logging-function-and-app-behavior) for more details. + +--- + +➡️ **To learn how to add a Slack function to a workflow**, head to the [workflows](/deno-slack-sdk/guides/creating-workflows) section. + +➡️ **To browse all Slack functions**, head to the [Slack Function reference catalog](/deno-slack-sdk/reference/slack-functions) + +✨ **To learn how to create your own _custom_ functions**, head to the [custom functions](/deno-slack-sdk/guides/creating-custom-functions) section. + diff --git a/docs/guides/functions/integrating-with-services-requiring-external-authentication.md b/docs/guides/functions/integrating-with-services-requiring-external-authentication.md new file mode 100644 index 00000000..5c70a2e8 --- /dev/null +++ b/docs/guides/functions/integrating-with-services-requiring-external-authentication.md @@ -0,0 +1,416 @@ +--- +slug: /deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication +--- + +# Integrating with services requiring external authentication + + + +You can use the Slack CLI to encrypt and to store OAuth2 credentials. This enables your app to access information from another service without exchanging passwords, but rather, tokens. + +## What is OAuth2, and why should you use it? {#what-is-oauth} + +OAuth2 stands for Open Authorization 2.0. It is a standard protocol designed to allow apps to access resources hosted by other apps on behalf of a user. Unlike basic authorization, where you share a password with a user, OAuth2 uses access tokens to verify a user's identity. For providers that require it, Slack offers PKCE and HTTP Basic Auth support, as you will see in the [OAuth2 provider `options` properties](#options-properties) section below. + +The following steps guide you through integrating your app with an external service using [Google](https://developers.google.com/identity/protocols/oauth2) as an example. + +### 1. Obtain your OAuth2 credentials {#credentials} + +The first step is to obtain your OAuth2 credentials. To do that, create a new OAuth2 credential with the external service you'll be integrating with. + +For our example, navigate to the [Google API Console](https://console.cloud.google.com/projectselector2/apis/dashboard?supportedpurview=project) to obtain your OAuth2 **client ID** and **client secret**. + +If you're asked to provide a **redirect URL**, use the following: + +``` +https://oauth2.slack.com/external/auth/callback +``` + +When you're done creating your credential, copy the credential's **client ID** and **client secret**, then head to your [manifest](/deno-slack-sdk/guides/using-the-app-manifest) file (`manifest.ts`). + +### 2. Define your OAuth2 provider {#define} + +Next, tell your app about your OAuth2 provider by defining an **OAuth2 provider** within your app. Inside your app's manifest, import `DefineOAuth2Provider`. Then, create a new provider instance. + +An example provider instance for Google is below. You can define it right in your manifest, or put it in its own file and import it into the manifest. + +```js +// manifest.ts +import { DefineFunction, DefineOAuth2Provider, DefineWorkflow, Manifest, Schema,} from "deno-slack-sdk/mod.ts"; +// ... + +// Define a new OAuth2 provider +// Note: replace with your actual client ID +// if you're following along and creating an OAuth2 provider for +// Google. +const GoogleProvider = DefineOAuth2Provider({ + provider_key: "google", + provider_type: Schema.providers.oauth2.CUSTOM, + options: { + provider_name: "Google", + authorization_url: "https://accounts.google.com/o/oauth2/auth", + token_url: "https://oauth2.googleapis.com/token", + client_id: ".apps.googleusercontent.com", + scope: [ + "https://www.googleapis.com/auth/spreadsheets.readonly", + "https://www.googleapis.com/auth/userinfo.email", + "https://www.googleapis.com/auth/userinfo.profile", + ], + authorization_url_extras: { + prompt: "consent", + access_type: "offline", + }, + identity_config: { + url: "https://www.googleapis.com/oauth2/v1/userinfo", + account_identifier: "$.email", + http_method_type: "GET", + }, + use_pkce: false, + }, +}); +// ... +``` + +OAuth2 provider properties are described in the tables below. + +#### OAuth2 provider properties {#properties} + +| Field | Type | Description | Required? | +|-----------------|---------------------------|-------------|-----------| +| `provider_key` | `string` | The unique string identifier for a provider. An app cannot have two providers with the same unique identifier. Changing unique identifiers will be treated as the deletion of a provider. Providers with active tokens cannot be deleted. | Required | +| `provider_type` | `Schema.providers.oauth2` | The only supported provider type value at this time is `Schema.providers.oauth2.CUSTOM`. | Required | +| `options` | `object` | Object with further provider-specific details. See the [table below](#options-properties). | Required | + +##### OAuth2 provider `options` properties {#options-properties} + +| Field | Type | Description | Required? | +|----------------------------|----------|-------------|-----------| +| `provider_name` | `string` | The name of your provider. | Required | +| `client_id` | `string` | The client ID from your provider. | Required | +| `authorization_url` | `string` | An OAuth2 requirement to complete the OAuth2 flow and to direct the user to the provider consent screen. | Required | +| `scope` | `array` | An OAuth2 requirement to complete the OAuth2 flow and to grant only the scopes provided to the access token. | Required | +| `identity_config` | `object` | Used to obtain user identity by finding the account associated with the access token. See the [table below](#identity-properties) for details. | Required | +| `token_url` | `string` | An OAuth2 requirement to complete the OAuth2 flow and to exchange the code with an access token. | Required | +| `token_url_config` | `object` | An object that can further define a `use_basic_auth_scheme` object, which contains a solitary boolean property, `use_basic_auth_scheme`, that defines whether HTTP Basic Authentication should be used with the `token_url` field above. Defaults to `false`. | Optional | +| `authorization_url_extras` | `object` | HTTP request query parameters to attach to the `authorization_url`. Set object key names as query parameter names and object key values as query parameter values. | Optional | +| `use_pkce` | `boolean`| Specifies if the provider uses PKCE. Defaults to `false`. | Optional | + +#### OAuth2 provider `identity_config` properties {#identity-properties} + +| Field | Type | Description | Required? | +|----------------------|----------|-------------|-----------| +| `url` | `string` | The endpoint the provider exposes to fetch the user identity. It is used to identify the authenticated user. | Required | +| `account_identifier` | `string` | The field name in the response from the above `url` field representing the user identity. | Required | +| `headers` | `object` | Extra HTTP headers to attach to the request to the `url` field. Set object key names as header names and object key values as header values. Note: the `Authorization` header is automatically set by Slack. | Optional | +| `http_method_type` | `string` | The HTTP method to employ when sending a request to the identity `url`. Defaults to `GET`. Acceptable values include `GET` or `POST`. | Optional | +| `body` | `object` | HTTP body parameters that the identity `url` expects. Only used if `http_method_type` is set to `POST`. Set object key names as body parameter name and object key values as body parameter values. | Optional | + +:::info + +The `identity_config` field is used to extract an `external_user_id`; this value is then used to allow a single user to issue multiple tokens for multiple provider accounts. If a Slack user with multiple accounts extracts the same `external_user_id` from the provider for each of their accounts, the existing token will be overwritten, and they will not be able to use multiple accounts. + +::: + +### 3. Add your OAuth2 provider to your manifest {#adding-provider} + +In your manifest file, insert the newly-defined provider into the `externalAuthProviders` property (if that property doesn't exist yet, go ahead and create it): + +```js +export default Manifest({ + //... + // Tell your app about your OAuth2 providers here: + externalAuthProviders: [GoogleProvider], + //... +}); +``` + +Now, with your OAuth2 provider defined and your manifest configured to use it, you can encrypt and store your client secret so that your app's users can utilize the OAuth2 authorization flow. + +### 4. Encrypt and store your client secret {#client-secret} + +Your app needs to be deployed to Slack once in order to create a place to store your encrypted client secret. Run the `slack deploy` command in your terminal window: + +```zsh +slack deploy +``` + +This command will bring up a list of [currently authorized workspaces](/deno-slack-sdk/guides/getting-started#authorize-cli). Select the workspace where your app will exist, and wait for the CLI to finish deploying. + +When finished, stay in your terminal window to add your client secret for the newly-defined provider, ensuring that you wrap the secret string in double quotes as follows: + +```zsh +slack external-auth add-secret --provider google --secret "GOCSPX-abc123..." +``` + +Running the `add-secret` command will bring up a list of workspaces available to you. Find and select the workspace you recently deployed your app to; you'll know it's the workspace you recently installed the app in by locating the item in the list with your app's name and ID (e.g., `myapp A01BC...`) rather than "App is not installed to this workspace." + +:::info + +If you get a `provider_not_found` error, go back to your manifest file and check to make sure that you included your OAuth2 provider in the `externalAuthProviders` properties of your manifest definition. + +::: + +If everything was successful, the CLI will let you know: + +``` +✨ successfully added external auth client secret for google +``` + +Great! With your app configured to interact with your defined OAuth2 provider, we can now initialize the OAuth2 sign-in flow, connecting your external provider to your Slack app. + +### 5. Initialize the OAuth2 flow {#initialize-oauth-flow} + +Once your provider's client secret has been added, it's time to create a token for your app to interact with your OAuth2 provider with [`external-auth add`](/slack-cli/reference/commands/slack_external-auth_add). + +Run the following command: + +``` +slack external-auth add +``` + +This will display a list of workspaces your app is deployed to. Select the one you're currently working in. Upon selection, you'll be provided a list of all providers that have been defined for this app, along with whether there's a secret and token. + +``` +$ slack external-auth add + +? Select a provider [Use arrows to move, type to filter] +> Provider Key: google + Provider Name: Google + Client ID: .apps.googleusercontent.com + Client Secret Exists? Yes + Token Exists? No +``` + +If you have just created a provider, you'll notice that it reports no tokens existing. Let's +go ahead and create a token by initializing the OAuth2 sign-in flow. + +Select the provider you're working on, which will open a browser window for you to complete the OAuth2 sign-in flow according to your provider's requirements. You'll know you're successful when your browser sends you to a `oauth2.slack.com` page stating that your account was successfully connected. + +Verify that a valid token has been created by re-running the [`external-auth add`](/slack-cli/reference/commands/slack_external-auth_add) command: + +``` +slack external-auth add + +? Select a provider [Use arrows to move, type to filter] +> Provider Key: google + Provider Name: Google + Client ID: .apps.googleusercontent.com + Client Secret Exists? Yes + Token Exists? Yes +``` + +If you see `Token Exists? Yes`, that means a valid auth token has been created, and you're ready to use OAuth2 in your app! Exit out of this command flow by entering `Ctrl+C` in your terminal — otherwise you'll be guided through the OAuth2 sign-in flow again. + +### 6. Add OAuth2 to your function {#function} + +Your [custom functions](/deno-slack-sdk/guides/creating-custom-functions) can leverage your provider's token +by configuring it to receive a `Schema.slack.types.oauth2` type as an input parameter to your function's definition. + +Here's how that might look if we were to use the sample function from the [starter template](https://github.com/slack-samples/deno-starter-template): + +```js +// functions/sample_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const SampleFunctionDefinition = DefineFunction({ + callback_id: "sample_function", + title: "Sample function", + description: "A sample function", + source_file: "functions/sample_function.ts", + input_parameters: { + properties: { + message: { + type: Schema.types.string, + description: "Message to be posted", + }, + // Define token here + googleAccessTokenId: { + type: Schema.slack.types.oauth2, + oauth2_provider_key: "google", + }, + user: { + type: Schema.slack.types.user_id, + description: "The user invoking the workflow", + }, + }, + required: ["user"], + }, + output_parameters: { + properties: { + updatedMsg: { + type: Schema.types.string, + description: "Updated message to be posted", + }, + }, + required: ["updatedMsg"], + }, +}); + +export default SlackFunction( + SampleFunctionDefinition, // Define custom function + async ({ inputs, client }) => { + // Get the token: + const tokenResponse = await client.apps.auth.external.get({ + external_token_id: inputs.googleAccessTokenId, + }); + if (tokenResponse.error) { + const error = + `Failed to retrieve the external auth token due to ${tokenResponse.error}`; + return { error }; + } + + // If the token was retrieved successfully, use it: + const externalToken = tokenResponse.external_token; + // Make external API call with externalToken + const response = await fetch("https://somewhere.tld/myendpoint", { + headers: new Headers({ + "Authorization": `Bearer ${externalToken}`, + "Content-Type": "application/x-www-form-urlencoded", + }), + }); + if (response.status != 200) { + const body = await response.text(); + const error = + `Failed to call my endpoint! (status: ${response.status}, body: ${body})`; + return { error }; + } + + // Do something here + const myApiResponse = await response.json(); + const updatedMsg = + `:newspaper: Message for <@${inputs.user}>!\n\n>${myApiResponse}`; + + return { outputs: { updatedMsg } }; + }, +); +``` + +### 7. Include OAuth2 input in a workflow step {#workflow} + +Next, while configuring your workflow, choose the persona whose auth you want to use when the workflow runs. To do this, pass an object with a `credential_source` to the OAuth2 input in the step configuration. + +:::info + +Slack will automatically inject the correct token ID into the OAuth2 input property based on the selected `credential_source`; you **do not** need to provide a token ID here. + +::: + +#### Using end user tokens {#end-user-tokens} +If you would like the workflow to use the account of the end user running the workflow, use `credential_source: "END_USER"`. + +The end user will be asked to authenticate with the external service in order to connect and grant Slack access to their account before running the workflow. This workflow can only be started by a [Link Trigger](/deno-slack-sdk/guides/creating-link-triggers), as this is the only type of trigger guaranteed to originate directly from an end-user interaction. + +```js +// Somewhere in your workflow's implementation: +const sampleFunctionStep = SampleWorkflow.addStep(SampleFunctionDefinition, { + user: SampleWorkflow.inputs.user, + googleAccessTokenId: { + credential_source: "END_USER" + }, +}); +``` + +#### Using developer tokens {#developer-tokens} +If you would like the workflow to use the account of one of the app collaborators, use `credential_source: "DEVELOPER"`. + +```js +// Somewhere in your workflow's implementation: +const sampleFunctionStep = SampleWorkflow.addStep(SampleFunctionDefinition, { + user: SampleWorkflow.inputs.user, + googleAccessTokenId: { + credential_source: "DEVELOPER" + }, +}); +``` + +After deploying the manifest changes above, you have to select a specific account for each of your workflows in this app. Assuming that you had run `slack external-auth add` before to add an external account, use the command `slack external-auth select-auth` as shown below: + +``` +slack external-auth select-auth +? Select a workspace +? Choose an app environment Deployed +? Select a workflow Workflow: #/workflows/ + Providers: + Key: google, Name: Google, Selected Account: None + +? Select a provider Key: google, Name: Google, Selected Account: None +? Select an external account Account: @gmail.com, Last Updated: 2023-05-30 + +✨ Workflow #/workflows/ will use developer account @gmail.com when making calls to google APIs +``` + +Multiple collaborators can exist for the same app and each collaborator can create a token using the `slack external-auth add` command. To select the appropriate collaborator account to run a specific workflow, the same `slack external-auth select-auth` command can be used. However, a collaborator needs to set up their own account using `slack external-auth select-auth` command by invoking this command. i.e. a collaborator cannot use `slack external-auth select-auth` to select auth for a workflow on behalf of another collaborator for the same app. + +A collaborator can remove their account by running `slack external-auth remove` command. This would automatically delete the existing selected auths for each of the workflows that were using it. Therefore, in such a case, `slack external-auth select-auth` command would be needed to be invoked again before executing the relevant workflows successfully later. + +### 8. Force refreshing a token programmatically {#force-refresh} + +If you ever want to force a refresh of your external token as a part of error handling, retry mechanism, or something similar, you can use the sample code below: + +```js +// Somewhere in your functions error handling and retry logic: +const result = await client.apps.auth.external.get({ + external_token_id: inputs.googleAccessTokenId, + force_refresh: true // default force_refresh is false +}); +``` +### 9. Deleting a token programmatically {#delete-programmatically} + +If you ever want to delete your external token programmatically, you can use the sample code below. Bear in mind that once a token is deleted, all workflows that were previously using the token will no longer work. + +
+ +
+ +This will *not* revoke the token from the provider's system. It will only delete the reference to the token from Slack and prevent it from being used within the external authentication system. + +
+
+ +```js + // Somewhere in your function: + await client.apps.auth.external.delete({ + external_token_id: inputs.googleAccessTokenId, + }); +``` + +## Delete external auth tokens {#delete-tokens} + +If you'd like to delete your tokens and remove OAuth2 authentication from your Slack app, the following commands will allow you to do so: + +| Command | Description | +|----------------------------------------------------------------------|--------------------------------------------------------------------| +| `$ slack external-auth remove` | Choose a provider to delete tokens for from the list displayed. | +| `$ slack external-auth remove --all` | Delete all tokens for the app by specifying the `--all` flag. | +| `$ slack external-auth remove --provider provider_name --` | Delete all tokens for a provider by specifying the `--provider` flag. | + +:::info + +This will *not* revoke the token from the provider's system. It will only delete the reference to the token from Slack and prevent it from being used within the external authentication system. + +::: + +## Troubleshooting {#troubleshooting} + +You can view external authentication logs via `slack activity`. These logs contain information about errors encountered by users during the OAuth2 exchange and workflow execution. Below are some common errors: + +| Error | Description | +|----------|-------------| +| `access_token_exchange_failed` | An error was returned from the configured `token_url`. | +| `external_user_identity_not_found` | The configured `account_identifier` was not found in user identity response. | +| `internal_error` | An internal system error happened. Please reach out to Slack if this occurs consistently. | +| `invalid_identity_config_response` | `url` in the configured `identity_config` returned an invalid response. | +| `invalid_token_response` | `token_url` returned an invalid response. | +| `missing_client_secret` | Optional client secret was found for this provider. | +| `no_refresh_token` | Token to refresh the expired access token does not exist. | +| `oauth2_callback_error` | The OAuth2 provider returned an error. | +| `oauth2_exchange_error` | There was an error while obtaining the OAuth2 token from the configured provider. | +| `scope_mismatch_error` | Slack was not able to find an OAuth2 token that matched the `scope` configured on your provider. | +| `token_not_found` | Slack was not able to find an OAuth2 token for this user and provider. | + +## Next Steps {#next-steps} + +Check out the following sample projects to see how real-world workflow apps use OAuth: +* [Timesheet approval app](https://github.com/slack-samples/deno-timesheet-approval) uses Google Sheets to store information collected in a workflow form from Slack users +* [Simple survey app](https://github.com/slack-samples/deno-simple-survey) uses Google Sheets to store survey responses +* [GitHub functions repo](https://github.com/slack-samples/deno-github-functions) brings oft-used GitHub functionality - such as creating new issues - to Slack using functions and workflows diff --git a/docs/guides/getting-started.md b/docs/guides/getting-started.md new file mode 100644 index 00000000..f4f61d8a --- /dev/null +++ b/docs/guides/getting-started.md @@ -0,0 +1,430 @@ +--- +sidebar_label: Getting started +slug: /deno-slack-sdk/guides/getting-started +--- + +# Getting started with the Deno Slack SDK + + + +In the following guide, you'll install the Slack CLI and authorize it in your workspace. Then, you'll use the Slack CLI to scaffold a fully-functional workflow app and run it locally. + +Don't have a workspace yet? You can get up and running by provisioning a sandbox with an associated workspace by following [this guide](https://api.slack.com/docs/developer-sandbox). Come on back when you're ready! + +## Step 1: Install the Slack CLI {#install-cli} + + + + + **Run the automated installer from your terminal window:** + +```zsh +curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash +``` + +This will install the Slack CLI and all required dependencies, including [Deno](/deno-slack-sdk/guides/installing-deno), the +runtime environment for workflow apps. If you have VSCode installed, +the [VSCode Deno +extension](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno) +will be installed. + + +
+Optional: Use an alias for the Slack CLI binary + +If you have another CLI tool in your path called `slack`, you can rename the slack binary to a different name before you add it to your path. + +To do this, pass the `-s` argument to the installer script: + +```zsh +curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash -s +``` + +The alias you use should come after any flags used in the installation script. For example, if you use both flags noted below to pass a version and skip the Deno installation, your install script might look like this: + +``` +curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash -s -- -v 2.1.0 -d +``` + +You can also copy the Slack CLI into any folder that is already in your path (such as `/usr/local/bin`—you can use`echo $PATH` to find these), or add a new folder to your path by listing the folder you installed the Slack CLI to in `/etc/paths`. + +If you don't rename the slack binary to a different name, the installation script will detect existing binaries named `slack` and bail if it finds one—it will not overwrite your existing `slack` binary. + +
+ +
+Optional: customize installation using flags + +There are two optional flags available to customize the installation. + +1. Specify a version you'd like to install using the version flag, `-v`. The absence of this flag will ensure the latest Slack CLI version is installed. +``` +curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash -s -- -v 2.1.0 +``` + +2. Skip the Deno installation by using the `-d` flag, like this: +``` +curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash -s -- -d +``` +
+ +
+Troubleshooting + +#### Errors + +Error: _Failed to create a symbolic link! The installer doesn't have write access to /usr/local/bin. Please check permission and try again..._ + +Solution: Sudo actions within the scripts were removed so as not to create any security concerns. The `$HOME` env var is updated to `/root` — however, the installer is using `$HOME` for both Deno and the SDK install, which causes the whole install to be placed under `/root`, making both Deno and the SDK unusable for users without root permissions. +* For users who do not have root permissions, run the sudo actions manually as follows: `sudo mkdir -p -m 775 /usr/local/bin`, then `sudo ln -sf "$slack_cli_bin_path" "/usr/local/bin/$SLACK_CLI_NAME"` where `$slack_cli_bin_path` is typically `$HOME/.slack/bin/slack` and `$SLACK_CLI_NAME` is typically the alias (by default it’s `slack`). +* For users who do have root permissions, you can run the installation script as `sudo curl -fsSL https://downloads.slack-edge.com/slack-cli/install.sh | bash`. In this case, the script is executed as root. + +
+ +
+ + + + **Run the automated installer from Windows PowerShell:** + +```zsh +irm https://downloads.slack-edge.com/slack-cli/install-windows.ps1 | iex +``` + +:::warning + +PowerShell is required for installing the Slack CLI on Windows machines; an alternative shell will not work. + +::: + +This will install the Slack CLI and all required dependencies, including [Deno](/deno-slack-sdk/guides/installing-deno), the +runtime environment for workflow apps. If you have VSCode installed, +the [VSCode Deno +extension](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno) +will be installed. + +
+Optional: Use an alias for the Slack CLI binary + +If you have another CLI tool in your path called `slack`, you can rename the slack binary to a different name before you add it to your path. + +To do this, copy the Slack CLI into any folder that is already in your path, or add a new folder to your path by listing the folder you installed the Slack CLI to in your Environment Variables. You may not have access to edit System variables, so you might need to add it to your account's User variables. You can open the Environment Variables dialog by pressing the `Win`+`R` keys to open the Run window, and then entering the following command: + +```pwsh +rundll32.exe sysdm.cpl,EditEnvironmentVariables +``` + +You can also use the `-Alias` flag as described within **Optional: customize installation using flags**. + +
+ +
+Optional: customize the installation using flags + +There are several flags available to customize the installation. Since flags +cannot be passed to remote scripts, you must first download the installation +script to a local file: + +```zsh +irm https://downloads.slack-edge.com/slack-cli/install-windows.ps1 -outfile 'install-windows.ps1' +``` + +The available flags are: + +| Flag | What it does | Example | +| :-- | :-- | :-- | +| `-Alias` | Installs the Slack CLI as the provided alias | `-Alias slackcli` will create a binary named `slackcli.exe` and add it to your path | +| `-Version` | Installs a specific version of the Slack CLI | `-Version 2.1.0` installs version `2.1.0` of the Slack CLI | +| `-SkipGit` | If true, will not attempt to install Git when Git is not present | `-SkipGit $true` | +| `-SkipDeno` | If true, will not attempt to install Deno when Deno is not present | `-SkipDeno $true` | + +You can also see all available flags by passing `-?` to the installation script: + +```zsh +.\install-windows.ps1 -? +``` + +Here's an example invocation using every flag: + +```zsh +.\install-windows.ps1 -Version 2.1.0 -Alias slackcli -SkipGit $true -SkipDeno $true +``` + +
+ +
+Troubleshooting + +#### Errors + +Error: _Not working? You may need to update your session's Language Mode._ + +Solution: For the installer to work correctly, your PowerShell session's [language mode](https://learn.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_language_modes?view=powershell-7.3#what-is-a-language-mode) will need to be set to `FullLanguage`. To check your session's language mode, run the following in your PowerShell window: `ps $ExecutionContext.SessionState.LanguageMode`. To run the installer, your session's language mode will need to be `FullLanguage`. If it's not, you can set your session's language mode to `FullLanguage` with the following command: `ps $ExecutionContext.SessionState.LanguageMode = "FullLanguage"` + +
+ +
+ + + +**1. Download and install [Deno](https://deno.land).** Refer to [Install Deno](/deno-slack-sdk/guides/installing-deno) for more details. + +**2. Verify that Deno is installed and in your path.** + +```bash +$ deno --version +deno 1.31.1* (release, x86_64-apple-darwin) +v8 10.* +typescript 4.* +``` + +**3. Download and install [Git](https://git-scm.com/book/en/v2/Getting-Started-Installing-Git), a dependency of the** `slack` **CLI.** + +**4. Download the** `slack` **CLI installer for your environment.** + +   Windows (.zip) + +   Download for macOS (.tar.gz) + +   Download for Linux (.tar.gz) + +**5. Add the** `slack` **CLI to your path.** + +:::info + +Existing `slack` binary in path? + +If you have another CLI tool in your path called `slack`, we recommend renaming our slack binary to a different name before adding it to your path. See your OS-specific installation tab for more details. + +::: + + +**6. Verify that** `slack` **is installed and in your path.** +``` +$ slack version +Using slack v2.32.0 +``` + +**7. Verify that all dependencies have been installed.** + +Run the following command: + +``` +$ slack doctor +``` + +**A few notes about hooks** + +If you have upgraded your CLI version but your `deno-slack-hooks` version is less than `v1.3.0`, when running `slack doctor`, you will see the following near the end of the output: + + ✔ Configurations (your project's CLI settings) + Project ID: 1a2b3c4d-ef5g-67hi-8j9k1l2m3n4o + + ✘ Runtime (foundations for the application) + Error: The `doctor` hook was not found (sdk_hook_not_found) + Suggestion: Ensure this hook is implemented in your `slack.json` + + ✔ Dependencies (requisites for development) + deno_slack_hooks: 1.2.3 → 1.3.0 (supported version) + +In addition, if you attempt to run the `slack run` command without this dependency installed, you will see a similar error in your console: + + 🚫 The `start` script was not found (sdk_hook_not_found) + + 💡 Suggestion + Hook scripts are defined in the Slack configuration file ('slack.json'). + Every app requires a 'slack.json' file and you can find a working example at: + https://github.com/slack-samples/deno-starter-template/blob/main/slack.json + +Ensure that `deno-slack-hooks` is installed at the project level and that the version is not less than `v1.3.0`. + +**8. [Install the VSCode extension for + Deno](/deno-slack-sdk/guides/installing-deno#vscode) (recommended).** + + + +
+ +:::info +The minimum required Slack CLI version for Enterprise Grid as of September 19th, 2023 is `v2.9.0`. If you attempt to log in with an older version, you'll receive a `cli_update_required` error from the Slack API. Run `slack upgrade` to get the latest version. +::: + +## Step 2: Authorize the Slack CLI {#authorize-cli} + +With the Slack CLI installed, authorize the Slack CLI in your workspace with the following command: + +```zsh +slack login +``` + +In your terminal window, you should see an authorization ticket in the form of a +slash command, and a prompt to enter a challenge code: + +```zsh +$ slack login + +📋 Run the following slash command in any Slack channel or DM + This will open a modal with user permissions for you to approve + Once approved, a challenge code will be generated in Slack + +/slackauthticket ABC123defABC123defABC123defABC123defXYZ + +? Enter challenge code +``` + +Copy the slash command and paste it into any Slack conversation in the workspace you will be developing in. + +When you send the message containing the slash command, a modal will pop up, prompting you to grant certain permissions to the Slack CLI. Click the Confirm button in the modal to move to the next step. + +A new modal with a challenge code will appear. Copy that challenge code, and paste it back into your terminal: + + +```zsh +? Enter challenge code eXaMpLeCoDe + +✅ You've successfully authenticated! 🎉 + Authorization data was saved to ~/.slack/credentials.json + +💡 Get started by creating a new app with slack create my-app + Explore the details of available commands with slack help +``` + +Verify that your Slack CLI is set up by running `slack auth list` in your +terminal window: + + +```zsh +$ slack auth list + +myworkspace (Team ID: T123ABC456) +User ID: U123ABC456 +Last updated: 2023-01-01 12:00:00 -07:00 +Authorization Level: Workspace +``` + +You should see an entry for the workspace you just authorized. If you don't, get a new authorization ticket with `slack login` to try again. + +You're now ready to begin building workflow apps! In the next step, we'll +get started with a sample app. + +## Step 3: Create an app from a template {#create-app} + +:::info + +**Evaluate third-party apps** +Exercise caution when using third-party applications and automations (those outside of [`slack-samples`](https://github.com/slack-samples)). Review all source code created by third-parties before running `slack create` or `slack deploy`. + +::: + +The `create` command is how you create a workflow app. + +For this guide, we'll be creating a Slack app using the [Deno Starter Template](https://github.com/slack-samples/deno-starter-template) as a template: + +```zsh +slack create my-app --template https://github.com/slack-samples/deno-starter-template +``` + +The Slack CLI creates an app project folder and fills it with the sample app code. Once it has finished, `cd` into your new project directory: + +```zsh +cd my-app +``` + +Then continue to the next step. + +## Step 4: Run the app in local development mode {#local-development-mode} + +While building your app, you can see your changes propagated to your workspace +in real-time by running `slack run` within your app's directory. + +``` +slack run +``` + +When you execute `slack run`, you'll be asked to select a local environment: + +```zsh +? Choose a local environment +> Install to a new workspace or organization +``` + +Since you've not installed your app to any workspaces, select *Install to a new +workplace*. Then select the workspace you authenticated in. + +The Slack CLI will attempt to list any triggers, and in this case, will inform you there are no +existing triggers installed for the app. + +[Triggers](/deno-slack-sdk/guides/using-triggers) are what cause workflows to run. A [link +trigger](/deno-slack-sdk/guides/creating-link-triggers) generates a *Shortcut URL* which, when posted in +a channel or added as a bookmark, becomes a link. + +Triggers are created from trigger definition files. The Slack CLI will then look for any +trigger definition files and prompt you to select one. In this case, there is +only one trigger: `sample_trigger.ts`. Select it. + +```zsh +? Choose a trigger definition file: +> triggers/sample_trigger.ts + Do not create a trigger +``` + +Once your app's trigger is created, you will see the following output: + +``` +⚡ Trigger successfully created! + + Sample trigger (local) Ft0123ABC456 (shortcut) + Created: 2023-01-01 12:00:00 -07:00 (1 second ago) + Collaborators: + You! @You U123ABC456DE + Can be found and used by: + everyone in the workspace + https://slack.com/shortcuts/Ft0123ABC456/XYZ123 +``` + +The Slack CLI will also start a local development server, syncing changes to your +workspace's development version of your app. You'll know your local development +server is up and running when your terminal window tells you it's `Connected, +awaiting events`. + +## Step 5: Use your app {#use} + +Grab the `Shortcut URL` you generated in the previous step and paste it in a +public channel in your workspace. You will see the shortcut unfurl with a +"Start Workflow" button. Click the button to execute the shortcut. + +In the modal that appears, select a channel, and enter a message. When +you click the "Send message" button, you should see your message appear in the +channel you specified. + +When you want to turn _off_ the local development server, use `Ctrl+c` in the +command prompt. + +--- + +## Onward {#start-build} + +At this point your Slack CLI is fully authorized and ready to create new projects. +It's time to choose the next path of adventure. + +We have curated a collection of sample apps. Many have +tutorials. All highlight features of workflow apps. Learn how to: + +* design [datastores](/deno-slack-sdk/guides/using-datastores) to store data with the [Virtual + Running Buddies app](/deno-slack-sdk/tutorials/virtual-running-buddies-app). +* send [event-triggered](/deno-slack-sdk/guides/creating-event-triggers) automated + [messages](/deno-slack-sdk/reference/slack-functions/send_message) with the [Welcome Bot + app](/deno-slack-sdk/tutorials/welcome-bot). +* create [forms](/deno-slack-sdk/guides/creating-a-form) to receive user input with the [Give Kudos + app](/deno-slack-sdk/tutorials/give-kudos-app). + +Each tutorial will expose you to many aspects of the workflow automations. If +you'd rather explore the documentation on your own, here are a few places to +start. You can learn how to: + +* [deploy your app](/deno-slack-sdk/guides/deploying-to-slack) so you don't need to run it locally. +* [build an app from scratch](/deno-slack-sdk/guides/creating-an-app). +* use workflow apps in conjunction with other services, whether that's with + [third-party API calls](https://api.slack.com/faq#third-party) or [external + authentication](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication). +* use the [Deno Slack SDK](https://github.com/slackapi/deno-slack-sdk) in tandem with the Slack CLI to access the API, additional documentation, and code libraries. You'll first need to download and install [Deno](/deno-slack-sdk/guides/installing-deno). If you're using VSCode for development, make sure to also download the [Deno extension for VSCode](/deno-slack-sdk/guides/installing-deno#vscode). \ No newline at end of file diff --git a/docs/guides/interactivity/adding-interactivity.md b/docs/guides/interactivity/adding-interactivity.md new file mode 100644 index 00000000..5e975de8 --- /dev/null +++ b/docs/guides/interactivity/adding-interactivity.md @@ -0,0 +1,135 @@ +--- +slug: /deno-slack-sdk/guides/adding-interactivity +--- + +# Adding interactivity + + + +Adding interactivity to your app adds a dynamic experience that makes it more substantive than just a bot sending messages. There are a few different ways to achieve interactivity in a workflow app. + +1. Collect user input **[through a form](/deno-slack-sdk/guides/creating-a-form)** and use it later in the workflow. +2. Create an **[interactive message](/deno-slack-sdk/guides/creating-an-interactive-message)** with varying options of back-and-forth with the user. +3. Use **[interactive modals](/deno-slack-sdk/guides/creating-an-interactive-modal)** to ask your users questions, allow actions, and update based on the information they give you. + +## Basic elements of interactivity {#basic-elements} +All options share some basic elements in common. + +* Interactivity parameter +* Blocks with interactive parts +* Interactivity handlers + +### Interactivity parameter {#interactivity-parameter} + +In order to prevent inundating users with pop-ups they didn't ask for, all app interactivity requires an interactivity parameter. This is the user's consent to interact with the app; only a user's interaction can open a form or modal. + +Whether the user is opening a [form](/deno-slack-sdk/guides/creating-a-form#add-interactivity), [modal](/deno-slack-sdk/guides/creating-an-interactive-modal#add-interactivity), or sending an [interactive message](/deno-slack-sdk/guides/creating-an-interactive-message), this looks the same—including an `input_parameter` of the type `interactivity`. + +:::warning[Not supported in Workflow Builder] + +[Custom functions](/deno-slack-sdk/guides/creating-custom-functions) that require interactivity inputs are not currently supported in Workflow Builder. + +::: + +### Interactive blocks {#interactive-blocks} + +Both modals and interactive messages allow for [interactive blocks](https://api.slack.com/reference/block-kit/block-elements), a flexible and dynamic way to create visually appealing app interaction. Check out an example in step 2 of [Creating an interactive message](/deno-slack-sdk/guides/creating-an-interactive-message#add-block-kit), explore the [Block Kit reference](https://api.slack.com/reference/block-kit/block-elements) for a menu of interactive options, then try them out in [Block Kit Builder](https://app.slack.com/block-kit-builder/). + +### Interactivity handlers {#interactivity-handlers} + +An interactive form is a static mechanism that merely gathers information from the user, but modals and messages allow for more of a back-and-forth interaction. This means being able to respond to your user's actions and inputs dynamically. This is achieved by utilizing interactivity handlers. + +Handler name | Description | Where it can be used +------------ | ----------- | -------------------- +`BlockActionsHandler` | Used to respond to interactivity that happens within an [interactive block](https://api.slack.com/reference/block-kit/block-elements). | [`Messages`](https://api.slack.com/surfaces/messages) [`Modals`](https://api.slack.com/surfaces/modals) +`BlockSuggestionHandler` | Used alongside a [select menu of external data source](https://api.slack.com/reference/block-kit/block-elements#external_select) element. | [`Messages`](https://api.slack.com/surfaces/messages) [`Modals`](https://api.slack.com/surfaces/modals) +`ViewSubmissionHandler` | Used to update a modal view after it has been submitted. | [`Modals`](https://api.slack.com/surfaces/modals) +`ViewClosedHandler` | Used to update an app after a view has been closed. | [`Modals`](https://api.slack.com/surfaces/modals) +`UnhandledEventHandler` | Used as a catch-all for unhandled events. | [`Messages`](https://api.slack.com/surfaces/messages) [`Modals`](https://api.slack.com/surfaces/modals) + +:::tip + +It's best practice to properly handle a function's success or error when a modal is submitted or closed. Refer to [creating an interactive modal](/deno-slack-sdk/guides/creating-an-interactive-modal) for more details. + +::: + +View sample payloads for these handlers in the [Interaction payloads documentation](https://api.slack.com/reference/interaction-payloads). + +## Invoking interactivity handlers {#invoking} + +Each handler contains two arguments, a `constraint` and a `handler`, such that its invocation will look like this: +- `addBlockActionsHandler(constraint, handler)` +- `addBlockSuggestionHandler(constraint, handler)` +- `addViewSubmissionHandler(constraint, handler)` +- `addViewClosedHandler(constraint, handler)` +- `addUnhandledEventHandler(constraint, handler)` + +If any incoming event matches the `constraint`, the specified handler will be invoked with the event payload. The `handler` arugment is the handler function that you define—what you want to happen in this event. Every type of handler function has the same context properties available to it, which are the same as the [context properties](/deno-slack-sdk/guides/creating-custom-functions#context) available to custom functions. This allows for authoring focused, single-purpose handlers and provides a concise, yet flexible API for registering handlers to specific interactions. + +What the `constraint` field allows depends on the type of handler. + +### Block actions and block suggestions {#block-handlers} + +For the `BlockActionsHandler` and the `BlockSuggestionHandler`, the `constraint` can be either a `BlockActionConstraintField` or a `BlockActionConstraintObject`. + +`BlockActionConstraintField` can be one of three options. + +``` ts +type BlockActionConstraintField = string | string[] | RegExp; +``` + +- When provided as a `string`, it must match the field exactly. +- When provided as an array of `string`, it must match one of the array values exactly. +- When provided as a `RegExp`, the regular expression must match. + +The `BlockActionConstraintObject` contains two properties. + +```ts +type BlockActionConstraintObject = { + block_id?: BlockActionConstraintField; + action_id?: BlockActionConstraintField; +}; +``` + +When the `constraint` is provided as an object in the form of a `BlockActionConstraintObject`, it can contain either or both a `block_id` and an `action_id`. +- Each of these properties is a `BlockActionConstraintField` (see above). +- If both the `action_id` and `block_id` properties exist on the constraint, then both `action_id` and `block_id` properties must match any incoming action. +- If only one of these properties is provided, then only the provided property must match. + +See an example of the `BlockActionsHandler` and the `BlockSuggestionHandler` in action in the [Creating an interactive message](/deno-slack-sdk/guides/creating-an-interactive-message) guide. + +### View handlers {#view-handlers} + +The `constraint` field for the view handlers is a bit different than in the `BlockActionsHandler` and `BlockSuggestionHandler`. + +```ts +SlackFunction({ ... }).addViewSubmissionHandler("my_view_callback_id", async (ctx) => { ... }); +``` + +For view handlers, the `consraint` argument can be either a `string`, `string[]`, or a `RegExp`. +- A simple `string` constraint must match a view's `callback_id` exactly. +- A `string []` constraint must match a view's `callback_id` to any of the strings in the array. +- A regular expression constraint must match a view's `callback_id`. + +### Unhandled handlers {#unhandled-handlers} + +The `UnhandledEventHandler` handles everything unaccounted for. It then makes sense that this handler does not accept a `constraint` argument. It does, however, accept the same `handler` argument as the other handlers, which is the handler function you define. Remember, all custom function [context properties](/deno-slack-sdk/guides/creating-custom-functions#context) are availalbe for use here. + +```ts + +.addUnhandledEventHandler(({ body: _body }) => { + console.log("unhandled event happened"); + //add some other actions + }) + +``` + +## Next steps {#next-steps} + +Ready to get started? + +✨ Get started with collecting user input by **[creating a form](/deno-slack-sdk/guides/creating-a-form)**. + +✨ Dazzle your users by sending them an **[interactive message](/deno-slack-sdk/guides/creating-an-interactive-message)**. + +✨ Create some back-and-forth banter with **[interactive modals](/deno-slack-sdk/guides/creating-an-interactive-modal)**. \ No newline at end of file diff --git a/docs/guides/interactivity/creating-a-form.md b/docs/guides/interactivity/creating-a-form.md new file mode 100644 index 00000000..1d2328ca --- /dev/null +++ b/docs/guides/interactivity/creating-a-form.md @@ -0,0 +1,288 @@ +--- +slug: /deno-slack-sdk/guides/creating-a-form +--- + +# Creating a form + + + +Forms are a straight-forward way to collect user input and pass it onto to other parts of your workflow. Their interactivity is one way - users interact with a static form. You cannot update the form itself based on user input. + +For example, say you need to collect some information from a user, send it to your system, then update a Slack channel with a link to a summary. Each task can be configured as a step in your workflow, allowing for user interactivity data to be passed to each step sequentially until the process is complete. + +Forms are created with the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function. + +✨ **If you only need to update an already-created form**, refer to the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function reference page. + +## 1. Add interactivity to your workflow {#add-interactivity} + +First let's take a look at the "Send a Greeting" workflow from the [Hello World sample app](https://github.com/slack-samples/deno-hello-world). + +Making your app interactive is the key to collecting user data. To accomplish this, an [`interactivity`](/deno-slack-sdk/reference/slack-types#interactivity) input parameter must be included as a property in your workflow definition. The `interactivity` parameter is required to ensure users don't experience any unexpected or unwanted forms appearing - only their interaction can open a form. + +as in the following code snippet: + +```javascript +// workflows/greeting_workflow.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { GreetingFunctionDefinition } from "../functions/greeting_function.ts"; + +const GreetingWorkflow = DefineWorkflow({ + callback_id: "greeting_workflow", + title: "Send a greeting", + description: "Send a greeting to channel", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + required: ["interactivity"], + }, +}); +``` + +## 2. Add a form to your workflow {#add-form} + +Now that you've added the `interactivity` property into your workflow, it's time to add the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function to a step in your workflow. + +While some of the functions you add to your workflow will be [custom functions](/deno-slack-sdk/guides/creating-custom-functions), a variety of [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) that cover some of the most common tasks executed on our platform are also available. The [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function allows for the collection of user input. + +### Form element schema {#element-schema} + +The fields of a form are made up of different types of form elements. Form elements have several properties you can customize depending on the element type. + +Links using Markdown are supported in the top-level description, but not in individual element descriptions. + +| Property | Type | Description | Required? | +| :------- | :--- | :---------- | :-------- | +| `name` | String | The internal name of the element | Required | +| `title` | String | Title of the form shown to the user. Maximum length is 25 characters | Required | +| `type` | `Schema.slack.types.*` | The [type of form element](/deno-slack-sdk/guides/creating-a-form#type-parameters) to display | Required | +| `description` | String | Description of the form shown to the user | Optional | +| `default` | Same type as `type` | Default value for this field | Optional | + +The following parameters are available for each type when defining your form. For each parameter listed above, `type` is required. + +:::info + +Note the distinction that some element types are prefixed with `Schema.types`, while some are prefixed with `Schema.slack.types`. + +::: + +#### Form types and parameters {#type-parameters} + +| Type | Parameters | Optionaltes | +| :---------- | :--- | :--- | +| [`Schema.types.string`](/deno-slack-sdk/reference/slack-types#string) | `title`, `description`, `default`, `minLength`, `maxLength`, `format`, `enum`, `choices`, `long`, `type` | If the `long` parameter is provided and set to `true`, it will render as a multi-line text box. Otherwise, it renders as a single-line text input field. In addition, basic input validation can be done by setting `format` to either `email` or `url` | | +| [`Schema.types.boolean`](/deno-slack-sdk/reference/slack-types#boolean) | `title`, `description`, `default`, `type` | A boolean rendered as a radio button in the form | +| [`Schema.types.integer`](/deno-slack-sdk/reference/slack-types#integer) | `title`, `description`, `default`, `enum`, `choices`, `type`, `minimum`, `maximum` | A whole number, such as `-1`, `0`, or `31415926535` | +| [`Schema.types.number`](/deno-slack-sdk/reference/slack-types#number) | `title`, `description`, `default`, `enum`, `choices`, `type`, `minimum`, `maximum` | A number that allows decimal points, such as `13557523.0005` | +| [`Schema.types.array`](/deno-slack-sdk/reference/slack-types#array) | `title`, `description`, `default`, `type`, `items`, `maxItems`, `display_type` | The required `items` parameter is an object itself, which must have a `type` sub-property defined. It can accept multiple different kinds of sub-properties based on the type chosen. Can be [`Schema.types.string`](/deno-slack-sdk/reference/slack-types#string), [`Schema.slack.types.channel_id`](/deno-slack-sdk/reference/slack-types#channelid), [`Schema.slack.types.user_id`](/deno-slack-sdk/reference/slack-types#userid). The `display_type` parameter can be used if the `items` object has the `type` parameter set to `Schema.types.string` and contains an `enum` parameter. The `display_type` parameter can be then set to `multi_static_select` (default) or `checkboxes`. | +| [`Schema.slack.types.date`](/deno-slack-sdk/reference/slack-types#date) | `title`, `description`, `default`, `enum`, `choices`, `type` | A string containing a date, displayed in `YYYY-MM-DD` format | +| [`Schema.slack.types.timestamp`](/deno-slack-sdk/reference/slack-types#timestamp) | `title`, `description`, `default`, `enum`, `choices`, `type` | A Unix timestamp in seconds, rendered as a [date picker](https://api.slack.com/reference/block-kit/block-elements#datepicker) | +| [`Schema.slack.types.user_id`](/deno-slack-sdk/reference/slack-types#userid) | `title`, `description`, `default`, `enum`, `choices`, `type` | A user picker | +| [`Schema.slack.types.channel_id`](/deno-slack-sdk/reference/slack-types#channelid) | `title`, `description`, `default`, `enum`, `choices`, `type` | A channel picker | +| [`Schema.slack.types.rich_text`](/deno-slack-sdk/reference/slack-types#rich-text) | `title`, `description`, `default`, `type` | A way to nicely format messages in your app. Note that this type cannot be converted to other message types, such as a string | +| [`Schema.slack.types.file_id`](/deno-slack-sdk/reference/slack-types#fileid) | `title`, `description`, `type`, `allowed_filetypes_group`, `allowed_filetypes` | Needs the [`files:read`](https://api.slack.com/scopes/files:read) scope. | + + +
+ An additional example: a form element from the Give Kudos sample app. + +```javascript +// workflows/give_kudos.ts + +const kudo = GiveKudosWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Give someone kudos", + interactivity: GiveKudosWorkflow.inputs.interactivity, + submit_label: "Share", + description: "Continue the positive energy through your written word", + fields: { + elements: [{ + name: "doer_of_good_deeds", + title: "Whose deeds are deemed worthy of a kudo?", + description: "Recognizing such deeds is dazzlingly desirable of you!", + type: Schema.slack.types.user_id, + }, { + name: "kudo_channel", + title: "Where should this message be shared?", + type: Schema.slack.types.channel_id, + }, { + name: "kudo_message", + title: "What would you like to say?", + type: Schema.types.string, + long: true, + }, { + name: "kudo_vibe", + title: 'What is this kudo\'s "vibe"?', + description: "What sorts of energy is given off?", + type: Schema.types.string, + enum: [ + "Appreciation for someone 🫂", + "Celebrating a victory 🏆", + "Thankful for great teamwork ⚽️", + "Amazed at awesome work ☄️", + "Excited for the future 🎉", + "No vibes, just plants 🪴", + ], + }], + required: ["doer_of_good_deeds", "kudo_channel", "kudo_message"], + }, + }, +); +``` +
+ +Add the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function as a step in your workflow: + +```javascript +// workflows/greeting_workflow.ts + +const inputForm = GreetingWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Send a greeting", + interactivity: GreetingWorkflow.inputs.interactivity, + submit_label: "Send greeting", + fields: { + elements: [{ + name: "recipient", + title: "Recipient", + type: Schema.slack.types.user_id, + }, { + name: "channel", + title: "Channel to send message to", + type: Schema.slack.types.channel_id, + default: GreetingWorkflow.inputs.channel, + }, { + name: "message", + title: "Message to recipient", + type: Schema.types.string, + long: true, + }], + required: ["recipient", "channel", "message"], + }, + }, +); +``` + +Forms have two output parameters: +- `fields`: The same field names in the inputs, which are returned as outputs with the values entered by the user +- `interactivity`: The context about the form submit action interactive event + +Use these output parameters to pass the information you collected from the user to subsequent steps in a workflow. When using the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) Slack function, either add it as the first step in your workflow or ensure the preceding step is interactive, as an interactive step will generate a fresh pointer to use for opening the form. For example, use the interactive button in a later step in your workflow, which can be added with the [`Send a message`](/deno-slack-sdk/reference/slack-functions/send_message) Slack function immediately before opening the form. + + +It is important to validate the inputs you receive from the user: first, that the user is authorized to pass the input, and second, that the user is passing a value you expect to receive and nothing more. + +The example below passes the user's input data into the second step of the workflow, a [custom function](/deno-slack-sdk/guides/creating-custom-functions), by using the output parameter `fields` and selecting the desired output element by name (i.e. `recipient` and `message`) + +```javascript +// workflows/greeting_workflow.ts + +const greetingFunctionStep = GreetingWorkflow.addStep( + GreetingFunctionDefinition, + { + recipient: inputForm.outputs.fields.recipient, + message: inputForm.outputs.fields.message, + }, +); +``` + +User input data can also be passed to [Slack functions](/deno-slack-sdk/guides/creating-slack-functions). This example sends the user's message to a specific channel specified by the user. + +```javascript +// workflows/greeting_workflow.ts + +GreetingWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: inputForm.outputs.fields.channel, + message: greetingFunctionStep.outputs.greeting, +}); + +export default GreetingWorkflow; +``` + +Take note of the `title`, `description`, and `submit_label` fields. It is important be descriptive with these fields, as these are the first things the user will see once the workflow is started and your form is displayed to them: + +![form-metadata](form-metadata.png "Sample form metadata") + +## 3. Add your workflow to your manifest {#manifest-workflow} + +With a workflow defined and steps outlined, it's time to make this an official part of your app! Add the workflow definition to your manifest as in the following example: + +```javascript +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; +import GreetingWorkflow from "./workflows/greeting_workflow.ts"; + +export default Manifest({ + name: "deno-hello-world", + description: + "A sample that demonstrates using a function, workflow and trigger to send a greeting", + icon: "assets/default_new_app_icon.png", + workflows: [GreetingWorkflow], + outgoingDomains: [], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +✨ **To learn more about workflows**, check out the [workflows](/deno-slack-sdk/guides/creating-workflows) page. + +## 4. Add a trigger to kick off your workflow {#add-trigger} + +Let's add the needed momentum to your workflow and create a [link trigger](/deno-slack-sdk/guides/creating-link-triggers#create-cli__create-a-link-trigger-with-a-trigger-file). + +In the trigger definition, add `interactivity` as an input value. This value holds context about the user interactivity that invoked this trigger, and passes it along to your workflow. + +In a separate file, define your trigger in the following way: + +```javascript +// triggers/greeting_trigger.ts + +import { Trigger } from "deno-slack-sdk/types.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; +import GreetingWorkflow from "../workflows/greeting_workflow.ts"; + +const greetingTrigger: Trigger = { + type: TriggerTypes.Shortcut, + name: "Send a greeting", + description: "Send greeting to channel", + workflow: `#/workflows/${GreetingWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: TriggerContextData.Shortcut.interactivity, + }, + channel: { + value: TriggerContextData.Shortcut.channel_id, + }, + }, +}; + +export default greetingTrigger; +``` + +Run the following CLI command to create the link trigger: + +```sh +$ slack trigger create --trigger-def triggers/greeting_trigger.ts + +... + +⚡ Trigger created + Trigger ID: Ft0123ABC456 + Trigger Type: shortcut + Trigger Name: Send a greeting + URL: https://slack.com/shortcuts/Ft0123ABC456/c001a02b13c42de35f47b55a89aad33c +``` + +You now have a shortcut `URL` to share in a channel or save as a bookmark, which allows you to kick off your workflow and open your form. + +✨ **To learn more about starting workflows with triggers**, head to the [triggers overview](/deno-slack-sdk/guides/using-triggers) page. \ No newline at end of file diff --git a/docs/guides/interactivity/creating-an-interactive-message.md b/docs/guides/interactivity/creating-an-interactive-message.md new file mode 100644 index 00000000..5b71e646 --- /dev/null +++ b/docs/guides/interactivity/creating-an-interactive-message.md @@ -0,0 +1,381 @@ +--- +slug: /deno-slack-sdk/guides/creating-an-interactive-message +--- + +# Creating an interactive message + + + +Interactive messages are messages containing interactive Block Kit elements. Send interactive messages to users to collect dynamic input from users, and use that input to kick off other parts of your workflows. + +Interactive messages are created with Block Kit, and have their interactions reflected by Block Kit action events. + +This page will guide you through adding Block Kit interactivity to your app's message. + +✨ **To learn more about Block Kit**, refer to [Building with Block Kit](https://api.slack.com/block-kit/building) and [Interactivity in Block Kit](https://api.slack.com/block-kit/interactivity). + +## 1. Create the function {#create-function} + +Let's look at the example in the [Deno Request Time Off app](https://github.com/slack-samples/deno-request-time-off). It contains a workflow where one step is sending a message with two button options: **"Approve"** and **"Deny"**. When someone clicks either button, our app will handle these button interactions (which are composed in [Block Kit Actions](https://api.slack.com/reference/block-kit/blocks#actions)) and update the employee with notice that their request was either approved or denied. + +First, we'll look at the function definition for [SendTimeOffRequestToManagerFunction](https://github.com/slack-samples/deno-request-time-off/blob/main/functions/send_time_off_request_to_manager/definition.ts) that defines the inputs that will appear in the message, and the outputs from the approver's interaction with the message: + +```javascript +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; +/** + * Custom function that sends a message to the user's manager asking for approval + * for the time off request. The message includes some Block Kit with two interactive + * buttons: one to approve, and one to deny. + */ +export const SendTimeOffRequestToManagerFunction = DefineFunction({ + callback_id: "send_time_off_request_to_manager", + title: "Request Time Off", + description: "Sends your manager a time off request to approve or deny", + source_file: "functions/send_time_off_request_to_manager/mod.ts", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + employee: { + type: Schema.slack.types.user_id, + description: "The user requesting the time off", + }, + manager: { + type: Schema.slack.types.user_id, + description: "The manager approving the time off request", + }, + start_date: { + type: Schema.slack.types.date, + description: "Time off start date", + }, + end_date: { + type: Schema.slack.types.date, + description: "Time off end date", + }, + reason: { + type: Schema.types.string, + description: "The reason for the time off request", + }, + }, + required: [ + "employee", + "manager", + "start_date", + "end_date", + "interactivity", + ], + }, + output_parameters: { + properties: {}, + required: [], + }, +}); +``` + +## 2. Add interactive Block Kit elements {#add-block-kit} + +Using Block Kit, you can build a message layout that contains two button: **"Approve"** and **"Deny"**. To keep our app tidy, we have the implementation in a [separate file](https://github.com/slack-samples/deno-request-time-off/blob/main/functions/send_time_off_request_to_manager/mod.ts). + +Here is the first part that creates the blocks: + +```javascript + +import { SendTimeOffRequestToManagerFunction } from "./definition.ts"; +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import { APPROVE_ID, DENY_ID } from "./constants.ts"; +import timeOffRequestHeaderBlocks from "./blocks.ts"; + +// Custom function that sends a message to the user's manager asking +// for approval for the time off request. The message includes some Block Kit with two +// interactive buttons: one to approve, and one to deny. +export default SlackFunction( + SendTimeOffRequestToManagerFunction, + async ({ inputs, client }) => { + console.log("Forwarding the following time off request:", inputs); + + // Create a block of Block Kit elements composed of several header blocks + // plus the interactive approve/deny buttons at the end + const blocks = timeOffRequestHeaderBlocks(inputs).concat([{ + "type": "actions", // This is the type of layout block; learn more about other layout blocks types at https://api.slack.com/reference/block-kit/blocks + "block_id": "approve-deny-buttons", + "elements": [ + { + type: "button", + text: { + type: "plain_text", + text: "Approve", + }, + action_id: APPROVE_ID, // <-- important! we will differentiate between buttons using these IDs + style: "primary", + }, + { + type: "button", + text: { + type: "plain_text", + text: "Deny", + }, + action_id: DENY_ID, // <-- important! we will differentiate between buttons using these IDs + style: "danger", + }, + ], + }]); + // To be continued in the next step... +``` + +## 3. Add the message functionality {#post} + +There are two Block Kit parameters that your Block Kit element will use for interactivity with other aspects of your workflow: + +* The `action_id` property. This uniquely identifies a particular interactive component. This will be used to route the interactive callback to the correct handler when an interaction happens on that element. +* The `block_id` property. This uniquely identifies the entire Block Kit element. + +Then, we can use the provided Slack client in the function handler to call the [`chat.postMessage`](https://api.slack.com/methods/chat.postMessage) method directly to post our message. The message will contain two buttons the user can interact with: one for **"Approve"** and one for **"Deny"**. + +```javascript + + // Send the message to the manager with the Slack client + const msgResponse = await client.chat.postMessage({ + channel: inputs.manager, + blocks, + // Fallback text to use when rich media can't be displayed (i.e. notifications) as well as for screen readers + text: "A new time off request has been submitted", + }); + + if (!msgResponse.ok) { + console.log("Error during request chat.postMessage!", msgResponse.error); + } + + // IMPORTANT! Set `completed` to false in order to keep the interactivity + // points (the approve/deny buttons) "alive" + // We will set the function's complete state in the button handlers below. + return { + completed: false, + }; + }, + // Create an 'actions handler', which is a function that will be invoked + // when specific interactive Block Kit elements (like buttons!) are interacted + // with. +) +// To be completed in the next step... +``` + +We return `completed: false` here to ensure the function execution does not complete until the interactivity is complete. The function execution will be completed in the action handler in the next section. + +## 4. Add a Block Kit handler to respond to Block Kit element interactions {#blockkit} + +Now that we have some interactive components to listen for, let's define a handler to react to interactions with these components. There are two Block Kit handlers: +* the action handler +* the suggestions handler + +### Using the Block actions handler {#block-actions-handler} +When the interactive components are used in a function, we use `addBlockActionsHandler` chained onto the function to handle what happens after the interaction. + +In the same function source file (and "chaining" off our function implementation), we'll define a handler that will listen for actions performed on one of the two interactive components (`APPROVE_ID` and `DENY_ID`) that we'll attach to the message using the [`addBlockActionsHandler`](https://api.slack.com/reference/interaction-payloads/block-actions) helper method. + +```javascript +// ... continued from the step above +.addBlockActionsHandler( + // listen for interactions with components with the following action_ids + [APPROVE_ID, DENY_ID], + // interactions with the above two action_ids get handled by the function below + async function ({ action, body, client }) { + console.log("Incoming action handler invocation", action); + + const approved = action.action_id === APPROVE_ID; + + // Send manager's response as a message to employee + const msgResponse = await client.chat.postMessage({ + channel: body.function_data.inputs.employee, + blocks: [{ + type: "context", + elements: [ + { + type: "mrkdwn", + text: + `Your time off request from ${body.function_data.inputs.start_date} to ${body.function_data.inputs.end_date}` + + `${ + body.function_data.inputs.reason + ? ` for ${body.function_data.inputs.reason}` + : "" + } was ${ + approved ? " :white_check_mark: Approved" : ":x: Denied" + } by <@${body.user.id}>`, + }, + ], + }], + text: `Your time off request was ${approved ? "approved" : "denied"}!`, + }); + if (!msgResponse.ok) { + console.log( + "Error during requester update chat.postMessage!", + msgResponse.error, + ); + } +``` + +The final piece is to update the manager's message to remove the buttons and reflect the approval state: + +```javascript + // Update the manager's message to remove the buttons and reflect the approval + // state. Nice little touch to prevent further interactions with the buttons + // after one of them were clicked. + const msgUpdate = await client.chat.update({ + channel: body.container.channel_id, + ts: body.container.message_ts, + blocks: timeOffRequestHeaderBlocks(body.function_data.inputs).concat([ + { + type: "context", + elements: [ + { + type: "mrkdwn", + text: `${ + approved ? " :white_check_mark: Approved" : ":x: Denied" + }`, + }, + ], + }, + ]), + }); + if (!msgUpdate.ok) { + console.log("Error during manager chat.update!", msgUpdate.error); + } + + // And now we can mark the function as 'completed' - which is required as + // we explicitly marked it as incomplete in the main function handler. + await client.functions.completeSuccess({ + function_execution_id: body.function_data.execution_id, + outputs: {}, + }); + }, +); +``` + +Remember to mark the function as completed. This is required since we explicitly marked it as incomplete in the main function handler previously. + +### Using the Block suggestion handler {#block-suggestion-handler} + +Use `addBlockSuggestionHandler` to respond to events that are uniquely created by the [select menu of external data source](https://api.slack.com/reference/block-kit/block-elements#external_select) interactive Block element. Similarly implemented as the Block actions handler above, a user would create a block with the [select menu of external data source](https://api.slack.com/reference/block-kit/block-elements#external_select) element, then chain the handler onto their function. + +Let's take a look at an example; this one posts an inspirational quote. Once invoked, this function will post a message with a drop-down select menu and a button. The options rendered in the select menu will be dynamically loaded from an external API. Here is the function definition: + +```javascript +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const QuoteFunction = DefineFunction({ + callback_id: "quote", + title: "Inspire Me", + description: "Get an inspirational quote", + source_file: "functions/quote/mod.ts", // <-- important! Make sure this is where the logic for your function - which we will write in the next section - exists. + input_parameters: { + properties: { + requester_id: { + type: Schema.slack.types.user_id, + description: "Requester", + }, + channel_id: { + type: Schema.slack.types.channel_id, + description: "Channel", + }, + }, + required: [ + "requester_id", + "channel_id", + ], + }, + output_parameters: { + properties: { + quote: { + type: Schema.types.string, + description: "Quote", + }, + }, + required: ["quote"], + }, +}); +``` + +With `QuoteFunction` defined, we can add the interactive elements: + +```javascript +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +// QuoteFunction is the function we defined in the previous section +import { QuoteFunction } from "./definition.ts"; + +export default SlackFunction(QuoteFunction, async ({ inputs, client }) => { + console.log("Incoming quote request!"); + + await client.chat.postMessage({ + channel: inputs.channel_id, + blocks: [{ + "type": "actions", + "block_id": "so-inspired", + "elements": [{ + type: "external_select", + placeholder: { + type: "plain_text", + text: "Inspire", + }, + action_id: "ext_select_input", + }, { + type: "button", + text: { + type: "plain_text", + text: "Post", + }, + action_id: "post_quote", + }], + }], + }); + // Important to set completed: false! We should set the function's complete + // status later - in the action handler responding to the button click + return { + completed: false, + }; +}); +``` + +If this feels familiar to the Block actions handler example above, it's because it is! In the same way, we can then chain `addBlockSuggestionHandler` onto the function just as we did with `addBlockActionsHandler`: + +```javascript +export default SlackFunction(QuoteFunction, async ({ inputs, client }) => { + // ... the rest of your QuoteFunction logic here ... +}).addBlockSuggestionHandler( + "ext_select_input", // The first argument to addBlockActionsHandler can accept an action_id string, among many other formats! + // Check the API reference at the end of this document for the full list of supported options + async ({ body, client }) => { // The second argument is the handler function itself + console.log("Incoming suggestion handler invocation", body); + // Fetch some inspirational quotes + const apiResp = await fetch( + "https://motivational-quote-api.herokuapp.com/quotes", + ); + const quotes = await apiResp.json(); + console.log("Returning", quotes.length, "quotes"); + const opts = { + "options": quotes.map((q) => ({ + value: `${q.id}`, + text: { type: "plain_text", text: q.quote.slice(0, 70) }, + })), + }; + return opts; + }, +); +``` + +Using the example above, you could next code what happens after the button click, such as posting the selection to the channel. + +## Handling errors {#errors} + +It's important to validate the input data you receive from the user. + +1. First, validate that the user is authorized to pass the input. +2. Second, validate that the user is passing a value you expect to receive, and nothing more. + +## Onward +Now you have some interactivity weaved within your app, hooray! + +💻 **For an expanded version of the sample code provided above**, check out our [Request Time Off sample app](https://github.com/slack-samples/deno-request-time-off). + +✨ **To learn more about leveraging built-in powers or defining your own**, check out [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) and [custom functions](/deno-slack-sdk/guides/creating-custom-functions). + +✨ **For more details about handling events**, check out [creating an interactive modal](/deno-slack-sdk/guides/creating-an-interactive-modal). diff --git a/docs/guides/interactivity/creating-an-interactive-modal.md b/docs/guides/interactivity/creating-an-interactive-modal.md new file mode 100644 index 00000000..6d912b13 --- /dev/null +++ b/docs/guides/interactivity/creating-an-interactive-modal.md @@ -0,0 +1,369 @@ +--- +slug: /deno-slack-sdk/guides/creating-an-interactive-modal +--- + +# Creating an interactive modal + + + +A [modal](https://api.slack.com/surfaces/modals) is similar to an alert box, pop-up, or dialog box within Slack. Modals capture and maintain focus within Slack until the user submits or closes the modal. This makes them a powerful piece of app functionality for engaging with users. + +Interactive modals are modals containing interactive [Block Kit elements](https://api.slack.com/block-kit). Modals have a larger catalog of available interactive Block Kit elements than messages. + +Modals can be opened via a Block Kit interaction or a [link trigger](/deno-slack-sdk/guides/creating-link-triggers). A modal is updated by View events (close and submit) to reflect the user's inputs as they interact with the modal. + +This guide will use the an example file from our [deno-code-snippets](https://github.com/slack-samples/deno-code-snippets) repository. + +✨ **If you'd like a full sample app that uses modal interactivity**, check out the [Simple Survey sample app](https://github.com/slack-samples/deno-simple-survey). + +## Add `interactivity` to your function definition {#add-interactivity} + +A function needs to have an `interactivity` parameter added to have interactive functionality. The `interactivity` parameter is required to ensure users don't experience any unexpected or unwanted modals appearing—only their interaction can open a modal. The `interactivity` parameter is short-lived for this same reason, meaning as a developer you will need to keep grabbing a new one from the user as continued consent to modal views and updates. + +For modals, `interactivity` takes the form of the unique identifier `interactivity_pointer`. There are two different ways to retrieve and consume an `interactivity_pointer` when working with modals. + +1. When opening a modal view via [link trigger](/deno-slack-sdk/guides/creating-link-triggers), add a property with the type `Schema.slack.types.interactivity` to the `properties` object within a function's `input_parameters`. Your function can then access that interactivity event via your function argument's `inputs.interactivity.interactivity_pointer`. Note that in this example, the function argument is named `interactivity`, but you may choose to name it anything, so long as you use that name to access the `interactivity_pointer`. +2. When opening or updating a modal view from a block or view event, use the `interactivity_pointer` provided as part of the `body` of the block or view event payload, _not_ from the `inputs` parameter. Your function can access it via `body.interactivity.interactivity_pointer`. You will also see an example of this in the [Opening a modal based on a Block Kit action](#open-block-kit-action) section below. + +In our example file [`/Block_Kit_Modals/functions/demo.ts`](https://github.com/slack-samples/deno-code-snippets/blob/main/Block_Kit_Modals/functions/demo.ts), `interactivity` is added to the function's input parameters, since the modal is opened via link trigger: + +```javascript +// /functions.demo.ts + +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const def = DefineFunction({ + callback_id: "block-kit-modal-demo", + title: "Block Kit modal demo", + source_file: "Block_Kit_Modals/functions/demo.ts", + input_parameters: { + properties: { interactivity: { type: Schema.slack.types.interactivity } }, + required: ["interactivity"], + }, + output_parameters: { properties: {}, required: [] }, +}); +// To be continued ... +``` + +## Build a modal view {#build} + +Modal views are constructed partially using [Block Kit](https://api.slack.com/block-kit) pieces. That view will then be placed within an API call later on. + +Below is our example modal: + +```javascript +view: { + "type": "modal", + // Note that this ID can be used for dispatching view_submission and view_closed events. + "callback_id": "first-page", + // This option is required to be notified when this modal is closed by the user + "notify_on_close": true, + "title": { "type": "plain_text", "text": "My App" }, + // Not all modals need a submit button, but since we want to collect input, we do + "submit": { "type": "plain_text", "text": "Next" }, + "close": { "type": "plain_text", "text": "Close" }, + "blocks": [ + { + "type": "input", + "block_id": "first_text", + "element": { "type": "plain_text_input", "action_id": "action" }, + "label": { "type": "plain_text", "text": "First" }, + }, + ], +}, +``` + +Check out [Using the block suggestion handler](/deno-slack-sdk/guides/creating-an-interactive-message#block-suggestion-handler) on the [Interactive messages](/deno-slack-sdk/guides/creating-an-interactive-message) page to learn how to use the Block Kit element [select menu of external data source](https://api.slack.com/reference/block-kit/block-elements#external_select). Its use in modals and messages is similar. + +## Open a modal within your function {#open} + +With interactivity added to the function definition, we can open the interactive modal view. A view is opened using the [`views.open`](https://api.slack.com/methods/views.open) method. + +A modal view can be opened based on either of the following, causing your function to run: + +* a trigger (for example, clicking on a link trigger) +* a previous Block Kit action (for example, clicking a button in a message) + +Our example uses the first method. + +Some important considerations to note so that you can ensure your modal isn't left floating in the vast sea of suspended modals: + +1. Take note of the `callback_id`. We'll use it to define modal view handlers that react to `view_open` or `view_closed` events later. +1. Set `notify_on_close` to `true` in order to trigger a `view_closed` event. + +### Open a modal based on a trigger {#open-trigger} + +View this example in [demo.ts](https://github.com/slack-samples/deno-code-snippets/blob/cb432c83a539b9675e3dc7d9ec7c641c68a62a93/Block_Kit_Modals/functions/demo.ts): +```javascript +// demo.ts + +export default SlackFunction( + def, + // --------------------------- + // The first handler function that opens a modal. + // This function can be called when the workflow executes the function step. + // --------------------------- + async ({ inputs, client }) => { + // Open a new modal with the end-user who interacted with the link trigger + const response = await client.views.open({ + interactivity_pointer: inputs.interactivity.interactivity_pointer, + view: { + "type": "modal", + // Note that this ID can be used for dispatching view_submission and view_closed events. + "callback_id": "first-page", + // This option is required to be notified when this modal is closed by the user + "notify_on_close": true, + "title": { "type": "plain_text", "text": "My App" }, + "submit": { "type": "plain_text", "text": "Next" }, + "close": { "type": "plain_text", "text": "Close" }, + "blocks": [ + { + "type": "input", + "block_id": "first_text", + "element": { "type": "plain_text_input", "action_id": "action" }, + "label": { "type": "plain_text", "text": "First" }, + }, + ], + }, + }); + if (response.error) { + const error = + `Failed to open a modal in the demo workflow. Contact the app maintainers with the following information - (error: ${response.error})`; + return { error }; + } + return { + // To continue with this interaction, return false for the completion + completed: false, + }; + }, +) +``` + +Make sure to set the return to `completed: false`. You'll then set it to `true` later in your modal view event handler. + +### Open a modal based on a Block Kit action {#open-block-kit-action} + +Alternatively, a modal view can be opened using a Block Kit action handler. Below is the code structure for doing so: + +```javascript +export default SlackFunction(ConfigureEventsFunctionDefinition, async ({ inputs, client }) => { + +// "my_button" is the action_id of the Block element from which the action originated +).addBlockActionsHandler(["my_button"], async ({ body, client }) => { + const openingModal = await client.views.open({ + interactivity_pointer: body.interactivity.interactivity_pointer, + view, + }); + if (openingModal.error) { + return await client.functions.completeError({ function_execution_id: body.function_data.execution_id, error}); + } +}); +``` + +## Update the modal view {#update} + +With your defined modal view equipped with a `callback_id`, you can implement a modal view event handler to respond to interactions with your modal view. To respond to a `view_submission` event (the action of the user clicking the **Submit** button in your modal), use [`addViewSubmissionHandler`](https://github.com/slackapi/deno-slack-sdk/blob/main/docs/functions-view-handlers.md#addviewsubmissionhandlerconstraint-handler). + +The handler can update or push a view in two ways: +* by making a call to the [`views.update`](https://api.slack.com/methods/views.update) API method or the [`views.push`](https://api.slack.com/methods/views.push) API method. +* by setting the [`response_action`](https://api.slack.com/surfaces/modals/using#updating_response) property on the object returned by your interactivity handler. + +In addition to the `view_submission` and `view_closed` events, you can also update views using the `block_actions` and `options` events via the Block Kit action and suggestion handlers, respectively. Refer to [Add a Block Kit handler to respond to Block Kit element interactions](/deno-slack-sdk/guides/creating-an-interactive-message#blockkit) for more details. + +In the examples below, the [`addViewSubmissionHandler`](https://github.com/slackapi/deno-slack-sdk/blob/main/docs/functions-view-handlers.md#addviewsubmissionhandlerconstraint-handler) method registers a handler to push a new view on to the [view stack](https://api.slack.com/surfaces/modals#lifecycle). + +The first code snippet shows how to push a new view by calling `views.push`: + +```javascript +// ... +.addViewSubmissionHandler( + "first-page", // The callback_id of the modal + async ({ inputs, client, body }) => { + const response = await client.views.push({ + interactivity_pointer: body.interactivity.interactivity_pointer, + view, + }); + }, +) +// ... +``` + +The second code snippet shows how to use `response_action` to do the same thing. Both result in identical behavior! + +```javascript +// ... +.addViewSubmissionHandler( + "first-page", // The callback_id of the modal + async () => { + return { + response_action: "push", + view, + }; + }, +) +// ... +``` + +In our example, we'll be using the second way—updating `response_action`—to provide a second modal view when the first modal data is submitted. + +### Example: updating with a new interactive modal {#update-interactive} +In this example, notice how we extract the input values from the prior view using `view.state.values`. This is a property of the [view interaction payload](https://api.slack.com/reference/interaction-payloads/views). + +```javascript + // --------------------------- + // The handler that can be called when the above modal data is submitted. + // It saves the inputs from the first page as private_metadata, + // and then displays the second-page modal view. + // --------------------------- + .addViewSubmissionHandler(["first-page"], ({ view }) => { + // Extract the input values from the view data + const firstText = view.state.values.first_text.action.value; + // Input validations + if (firstText.length < 20) { + return { + response_action: "errors", + // The key must be a valid block_id in the blocks on a modal + errors: { first_text: "Must be 20 characters or longer" }, + }; + } + // Successful. Update the modal with the second page presentation + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "second-page", + // This option is required to be notified when this modal is closed by the user + "notify_on_close": true, + "title": { "type": "plain_text", "text": "My App" }, + "submit": { "type": "plain_text", "text": "Next" }, + "close": { "type": "plain_text", "text": "Close" }, + // Hidden string data, which is not visible to end-users + // You can use this property to transfer the state of interaction + // to the following event handlers. + // (Up to 3,000 characters allowed) + "private_metadata": JSON.stringify({ firstText }), + "blocks": [ + // Display the inputs from "first-page" modal view + { + "type": "section", + "text": { "type": "mrkdwn", "text": `First: ${firstText}` }, + }, + // New input block to receive text + { + "type": "input", + "block_id": "second_text", + "element": { "type": "plain_text_input", "action_id": "action" }, + "label": { "type": "plain_text", "text": "Second" }, + }, + ], + }, + }; + }) +``` + +### Example: updating with a static confirmation modal {#update-static} + +```javascript + // --------------------------- + // The handler that can be called when the second modal data is submitted. + // It displays the completion page view with the inputs from + // the first and second pages. + // --------------------------- + .addViewSubmissionHandler(["second-page"], ({ view }) => { + // Extract the first-page inputs from private_metadata + const { firstText } = JSON.parse(view.private_metadata!); + // Extract the second-page inputs from the view data + const secondText = view.state.values.second_text.action.value; + // Displays the third page, which tells the completion of the interaction + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "completion", + // This option is required to be notified when this modal is closed by the user + "notify_on_close": true, + "title": { "type": "plain_text", "text": "My App" }, + // This modal no longer accepts further inputs. + // So, the "Submit" button is intentionally removed from the view. + "close": { "type": "plain_text", "text": "Close" }, + // Display the two inputs + "blocks": [ + { + "type": "section", + "text": { "type": "mrkdwn", "text": `First: ${firstText}` }, + }, + { + "type": "section", + "text": { "type": "mrkdwn", "text": `Second: ${secondText}` }, + }, + ], + }, + }; + }) +``` + +## Success: closing a modal {#close-modal} + +To respond to a `view_closed` event (the action of the user clicking the **Close** button on your modal), use [`addViewClosedHandler`](https://api.slack.com/reference/interaction-payloads/views#view_closed) and add a call to the [`functions.completeSuccess`](https://api.slack.com/methods/functions.completeSuccess) method to explicitly mark the function as complete like this: + +```javascript + // --------------------------- + // The handler that can be called when the second modal data is closed. + // If your app runs some resource-intensive operations on the backend side, + // you can cancel the ongoing process and/or tell the end-user + // what to do next in DM and so on. + // --------------------------- + .addViewClosedHandler( + ["first-page", "second-page", "completion"], + ({ view }) => { + console.log(`view_closed handler called: ${JSON.stringify(view)}`); + + return await client.functions.completeSuccess({ + function_execution_id: body.function_data.execution_id, + outputs: {}, + }); + }, + ); +``` + +Remember, for an app to receive `view_closed` events, the view must set the `notify_on_close` option to `true` when it is initially opened or updated. + +## Error: handling an error {#errors} + + If the function execution was not successful, you can add a call to the [`functions.completeError`](https://api.slack.com/methods/functions.completeError) method to raise an error like so: + +```javascript +const response = await client.functions.completeError({ + function_execution_id: body.function_data.execution_id, + error: "Error completing function", +}); +``` + +Once you have opened a modal and handled your modal views, you may decide that you'd like to display any potential data validation error messages to your users. It is important to validate the inputs you receive from the user: first, that the user is authorized to pass the input, and second, that the user is passing a value you expect to receive and nothing more. + +As long as your submission handler returns an error object defined [on this page](https://api.slack.com/surfaces/modals/using#displaying_errors), the error messages you include in that object will be displayed right next to the relevant form fields based on their field IDs. + +## Stop a workflow {#stop-workflow} + +As discussed earlier, a function either completes successfully or fails with an error — and it's best practice to handle those events. However, there may be some cases in which you would like to stop a workflow early as a "quick fix" without necessarily calling [`functions.completeSuccess`](https://api.slack.com/methods/functions.completeSuccess) or [`functions.completeError`](https://api.slack.com/methods/functions.completeError). For example, when handling a modal view that the user closes prematurely: + +* the drawback with calling `functions.completeSuccess` in this scenario is that the rest of the functions in your workflow now require additional logic to handle undefined or null outputs. +* the drawback with calling `functions.completeError` in this scenario is that when the user closes the modal prematurely (for example, they realize they don't have time to enter all the required details for the modal inputs), then all your admins are pinged by SlackBot with the resulting error. + +So, what to do instead? Well, essentially, you can do nothing at all: + +```javascript +export default SlackFunction(..., ...) + .addViewClosedHandler("first-page", () => ({ client, body }) { + // clean up stuff + console.log('user closed modal view prematurely'); + // do nothing + }) + ``` + +With the above solution, the modal view closes, an entry in your activity log is made (when `console.log` is called), and the workflow simply doesn't continue on. That said, it's generally best practice to handle function successes and errors when possible to ensure things are tidied up and there are no functions left hanging in the ether. + +## Onward +You now have some shiny new modal views weaved within your app, and are on a course to providing a wonderful user experience. + +✨ **To learn more about other interactivity options**, refer to the [Interactivity overview](/deno-slack-sdk/guides/adding-interactivity). \ No newline at end of file diff --git a/docs/guides/interactivity/form-metadata.png b/docs/guides/interactivity/form-metadata.png new file mode 100644 index 00000000..8f97eb66 Binary files /dev/null and b/docs/guides/interactivity/form-metadata.png differ diff --git a/docs/guides/removing-an-app.md b/docs/guides/removing-an-app.md new file mode 100644 index 00000000..43789138 --- /dev/null +++ b/docs/guides/removing-an-app.md @@ -0,0 +1,65 @@ +--- +sidebar_label: Removing an app +slug: /deno-slack-sdk/guides/removing-an-app +--- + +# Removing an app + + + +All good things must come to an end. You can `uninstall` your app if you need to remove an app from a workspace, change app permissions, or `delete` the app in its entirety. + +## Uninstall an app from your team {#uninstall-app} + +Removing an app from a workspace doesn't have to be a permanent decision. Sometimes uninstalling the app to remove it's active presence in channels is sufficient! This option has the added benefit of reinstallation at a later time without recreating the entire app. + +To uninstall an app using the CLI, use the `slack uninstall` command. Then, choose the workspace you want to remove the app from: + +```bash +slack uninstall -a A123ABC456 -t T123ABC456 +``` + +```bash +⚠️ Warning + App (A123ABC456) will be uninstalled from my-workspace (T123ABC456) + All triggers, workflows, and functions will be deleted + Datastore records will be persisted + +❓ Are you sure you want to uninstall? Yes + +🏠 Workspace uninstall + Uninstalled the app "my-app" from workspace "my-workspace" +``` + +## Delete an app from your team {#delete-app} + +:::danger[Deleting your app _permanently_ deletes all of its data] + +Your app's related workflows, functions, and datastores will also be deleted. This decision is final and cannot be undone. + +::: + +To delete an app using the CLI, use the `slack delete` command: + +```bash +slack delete -a A123ABC456 -t T123ABC456 +``` + +```bash +⚠️ Danger zone + App (A123ABC456) will be permanently deleted + All triggers, workflows, and functions will be deleted + All datastores for this app will be deleted + Once you delete this app, there is no going back + +❓ Are you sure you want to delete the app? Yes + +🏠 App Uninstall + Uninstalled the app "my-app" from "my-workspace" + +📚 App Manifest + Deleted the app manifest for "my-app" from "my-workspace" + +🏘️ Apps + This project has no apps +``` \ No newline at end of file diff --git a/docs/guides/triggers/creating-event-triggers.md b/docs/guides/triggers/creating-event-triggers.md new file mode 100644 index 00000000..e1d9ee2b --- /dev/null +++ b/docs/guides/triggers/creating-event-triggers.md @@ -0,0 +1,1193 @@ +--- +slug: /deno-slack-sdk/guides/creating-event-triggers +--- + +# Creating event triggers + + + +>Invoke a workflow when a specific event happens in Slack + +Event triggers are a type of *automatic* trigger, as they don't require manual activation. Instead, they're automatically invoked when a certain event happens. + +## Supported events {#supported-events} + +A certain number of events have corresponding event triggers. + +Your app needs to have the proper scopes to use event triggers. Include these scopes within your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). Your app also needs to be a member of any channel where you want to listen for events. + +Events can be referenced in the form of `TriggerEventTypes.EVENTNAME`. + +| Event. Reference with `TriggerEventTypes.EVENTNAME` | Description | Required scopes | +|------------------|----------------------|------------------------| +|`AppMentioned`| Subscribe to only the message events that mention your app or bot. | [`app_mentions:read`](https://api.slack.com/scopes/app_mentions:read) | +| `ChannelArchived`| A channel was archived. | [`channels:read`](https://api.slack.com/scopes/channels:read) | +| `ChannelCreated`| A channel was created. | [`channels:read`](https://api.slack.com/scopes/channels:read) | +| `ChannelDeleted`| A channel was deleted. | [`channels:read`](https://api.slack.com/scopes/channels:read) | +| `ChannelRenamed`| A channel was renamed. | [`channels:read`](https://api.slack.com/scopes/channels:read) | +| `ChannelShared` | A channel has been shared with an external workspace. | [`channels:read`](https://api.slack.com/scopes/channels:read) [`groups:read`](https://api.slack.com/scopes/groups:read) | +| `ChannelUnarchived`| A channel was unarchived. | [`channels:read`](https://api.slack.com/scopes/channels:read) | +|`ChannelUnshared`| A channel has been unshared with an external workspace. | [`channels:read`](https://api.slack.com/scopes/channels:read) [`groups:read`](https://api.slack.com/scopes/groups:read) | +| `DndUpdated`| Do not Disturb settings changed for a member. | [`dnd:read`](https://api.slack.com/scopes/dnd:read) | +| `EmojiChanged` | A custom emoji has been added or changed. | [`emoji:read`](https://api.slack.com/scopes/emoji:read) | +|`MessagePosted`| A message was sent to a channel. _A [filter](#filters) is required to listen for this event._ | [`channels:history`](https://api.slack.com/scopes/channels:history) [`groups:history`](https://api.slack.com/scopes/groups:history) [`im:read`](https://api.slack.com/scopes/im:history) [`mpim:read`](https://api.slack.com/scopes/mpim:history) | +| `MessageMetadataPosted` | Message metadata was posted. | [`metadata.message:read`](https://api.slack.com/scopes/metadata.message:read) | +|`PinAdded`| A pin was added to a channel. | [`pins:read`](https://api.slack.com/scopes/pins:read) | +| `PinRemoved`| A pin was removed from a channel. | [`pins:read`](https://api.slack.com/scopes/pins:read) | +| `ReactionAdded`| A member has added an emoji reaction. | [`reactions:read`](https://api.slack.com/scopes/reactions:read) | +| `ReactionRemoved`| A member removed an emoji reaction. | [`reactions:read`](https://api.slack.com/scopes/reactions:read) | +| `SharedChannelInviteAccepted`| A shared channel invite was accepted. | [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) | +| `SharedChannelInviteApproved`| A shared channel invite was approved. | [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) | +| `SharedChannelInviteDeclined`| A shared channel invite was declined. | [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) | +| `SharedChannelInviteReceived`| A shared channel invite was sent to a Slack user. | [`conversations.connect:read`](https://api.slack.com/scopes/conversations.connect:read) | +| `SharedChannelInviteRequested` | A shared channel invite was requested to be sent. | [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) | +| `UserJoinedChannel`| A user joined a public or private channel. | [`channels:read`](https://api.slack.com/scopes/channels:read) [`groups:read`](https://api.slack.com/scopes/groups:read) | +| `UserJoinedTeam`| A new member has joined. | [`users:read`](https://api.slack.com/scopes/users:read) | +| `UserLeftChannel`| A user left a public or private channel. | [`channels:read`](https://api.slack.com/scopes/channels:read) [`groups:read`](https://api.slack.com/scopes/groups:read) + +### Events can activate multiple triggers {#multiple} + +When an event happens, all event triggers listening for that event will be invoked at roughly the same time. If you want to control which workflow runs first, you have two options: + +* Combine the functions of both workflows into a single workflow, invoked with a single event trigger. +* Have the second workflow be invoked by the first workflow, instead of the original event trigger. + + +### Avoid infinite loops {#loops} + +Your app can respond to events *and* be the cause of events. This can create situations where your app gets stuck in a loop. + +For example, if your app listens for all `message_posted` events in a channel and then posts its own message in response, it'll keep posting messages forever! That's why the `message_posted` event requires a filter. + +Carefully construct a [filter](#filters) to prevent boundless behavior. If your app does get stuck in an infinite loop, you can [delete the trigger](/deno-slack-sdk/guides/managing-triggers#delete) and the behavior will cease. + +## Create an event trigger {#create-trigger} + +Triggers can be added to workflows in two ways: + +* **You can add triggers with the CLI.** These static triggers are created only once. You create them with the Slack CLI, attach them to your app's workflow, and that's that. The trigger is defined within a trigger file. +* **You can add triggers at runtime.** These dynamic triggers are created at any step of a workflow so they can incorporate data acquired from other workflow steps. The trigger is defined within a function file. + + + + +:::info[Slack CLI built-in documentation] + +Use `slack trigger --help` to easily access information on the `trigger` command's flags and subcommands. + +::: + +The triggers you create when running locally (with the `slack run` command) will not work when you deploy your app in production (with the `slack deploy` command). You'll need to `create` any triggers again with the CLI. + +### Create the trigger file + +To create an event trigger with the CLI, you'll need to create a trigger file. The trigger file contains the payload you used to define your trigger. + +Create a TypeScript trigger file within your app's folder with the following form: + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { TriggerEventTypes, TriggerTypes, TriggerContextData } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + // your TypeScript payload +}; + +export default trigger; +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. The following is a trigger file with a payload creating an event trigger that listens for a `reaction_added` event in a specific channel: + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { TriggerEventTypes, TriggerTypes, TriggerContextData } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + type: TriggerTypes.Event, + name: "Reactji response", + description: "responds to a specific reactji", + workflow: "#/workflows/myWorkflow", + event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456"], + filter: { + version: 1, + root: { + statement: "{{data.reaction}} == sunglasses" + } + } + }, + inputs: { + stringtoSend: { + value: "how cool is that", + }, + channel: { + value: "C123ABC456", + }, + }, +}; + +export default trigger; +``` + +### Use the `trigger create` command + +Once you have created a trigger file, use the following command to create the event trigger: + +```bash +slack trigger create --trigger-def "path/to/trigger.ts" +``` +If you have not used the `slack triggers create` command to create a trigger prior to running the `slack run` command, you will receive a prompt in the Slack CLI to do so. + + + + +:::info + +Your app needs to have the [`triggers:write`](https://api.slack.com/scopes/triggers:write) scope to use a trigger at runtime. Include the scope within your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +::: + +The logic of a runtime trigger lies within a function's TypeScript code. Within your `functions` folder, you'll have the functions that are the steps making up your workflow. Within this folder is where you can create a trigger within the relevant `.ts` file. + +When you create a runtime trigger, you can leverage `inputs` acquired from functions within the workflow. Provide the workflow definition to get additional typing for the workflow and inputs fields. + +Create an event trigger at runtime using the `client.workflows.triggers.create` method within the relevant `function` file. + +```js +const triggerResponse = await client.workflows.triggers.create({ + // your TypeScript payload +}); +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. Below is a function file with an example TypeScript payload for an event trigger. This specific TypeScript payload is for creating an event trigger that listens for a `reaction_added` event in a specific channel: + +```js +// functions/example_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerEventTypes, TriggerTypes } from "deno-slack-api/mod.ts"; + +export const ExampleFunctionDefinition = DefineFunction({ + callback_id: "example_function_def", + title: "Example function", + source_file: "functions/example_function.ts", +}); + +export default SlackFunction( + ExampleFunctionDefinition, + ({ inputs, client }) => { + + const triggerResponse = await client.workflows.triggers.create({ + type: TriggerTypes.Event, + name: "Reactji response", + description: "responds to a specific reactji", + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456"], + filter: { + version: 1, + root: { + statement: "{{data.reaction}} == sunglasses" + } + } + }, + inputs: { + stringtoSend: { + value: "how cool is that", + }, + channel: { + value: "C123ABC456", + }, + } + }); + + // ... +``` + + + + +--- + +## Event trigger parameters {#parameters} + +| Field | Description | Required? | +|---------------|---------------------------------------------|----| +| `name` | The name of the trigger. | Required | +| `type` | The type of trigger: `TriggerTypes.Event`. | Required +| `workflow` | Path to workflow that the trigger initiates. | Required | +| `description` | The description of the trigger. | Optional | +| `event` | Contains [the `event` object](#event-object). | Optional | +| `inputs` | The inputs provided to the workflow. Can use with the [event response object](#response-object). | Optional | + +:::info + +Event triggers are not interactive. Use a [link trigger](/deno-slack-sdk/guides/creating-link-triggers) to take advantage of interactivity. + +::: + +### The `Event` object {#event-object} + +| Field | Description | Required? | +|---------------|---------------------------------------------|------------| +| `event_type` | The type of event; use one of the properties of `TriggerEventTypes`. | Required | +| `team_ids` | An array of event-related team ID strings. | Required for Enterprise Grid | +| `all_resources` | Trip the event trigger in all channels your app is present in. Defaults to `false`. Mutually exclusive with `channel_ids`. See [below](#scoping) for more details. | Dependent on the [event](#channel-based-event-triggers) | +| `channel_ids` | An array of channel IDs where the event trigger will trip. Mutually exclusive with `all_resources`. See [below](#scoping) for more details. | Dependent on the [event](#channel-based-event-triggers) | +| `filter` | See [trigger filters](#filters) for more details. | Optional | + +#### Scoping channel-based event triggers {#scoping} + +When writing a channel-based event trigger, you can pass the `channel_ids` field with a list of specific channels for the trigger to trip in. Example: + +```javascript +event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456", "C01234567", "C09876543"], +} +``` + +Alternatively, you can set `all_resources` to `true` The `channel_ids` field will no longer be required, and the event will now trigger in all channels in the workspace the app is a part of. Example: + +```javascript +event: { + event_type: TriggerEventTypes.ReactionAdded, + all_resources: true, +} +``` + +:::warning + +Setting `all_resources` to `true` could cause additional charges, as the event will trip in all channels the app is a member of in the workspace and may therefore lead to many workflow executions in workspaces with a large number of channels. + +::: + +#### Channel-based event triggers {#channel-based-event-triggers} + +The following channel-based event triggers require either the `channel_ids` or `all_resources` event object to be set: + +* `app_mentioned` +* `call_rejected` +* `channel_history_changed` +* `channel_id_changed` +* `channel_shared` +* `channel_unshared` +* `member_left_channel` +* `message_metadata_posted` +* `pin_added` +* `pin_removed` +* `reaction_added` +* `reaction_removed` +* `user_joined_channel` + +### The event response object {#response-object} + + An event's response object will contain additional information about that specific event instance. + +| Property | Description | +|-------------------|----------------------------------| +| `data` | Contains additional information dependent on event type. See [below](#data). | +| `enterprise_id` | A unique identifier for the enterprise where this event occurred. | +| `event_id` | A unique identifier for this specific event, globally unique across all workspaces. | +| `event_timestamp` | A Unix timestamp in seconds indicating when this event was dispatched. | +| `team_id` | A unique identifier for the workspace/team where this event occurred. | +| `type` | An identifier showing the `event` type | + +#### The `data` property {#data} + +Each type of event has unique sub-properties within the `data` property. You can pass these values on to your workflows. See the example below. + +##### Event types + +
+app_mentioned + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "app_id": "A1234ABC", + "channel_id": "C0123ABC", + "channel_name": "cool-channel", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/app_mentioned", + "message_ts": "164432432542.2353", + "text": "<@U0LAN0Z89> is it everything a river should be?", + "user_id:": "U0123ABC", + } +} +``` + +
+ +
+channel_archived* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "cool-channel", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/channel_archived", + "user_id": "U0123ABC", + } +} +``` +
+ +
+channel_created* + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "fun", + "channel_type": "public", + "created": 1360782804, + "creator_id": "U0123ABC", + "event_type": "slack#/events/channel_created", + } +} +``` +
+ +
+channel_deleted* + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "project_planning", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/channel_deleted", + "user_id": "U0123ABC", + } +} +``` +
+ +
+channel_renamed* + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "project_planning", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/channel_renamed", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+channel_shared + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "cool-channel", + "channel_type": "public/private/im/mpim", + "connected_team_id": "E0123ABC", + "event_type": "slack#/events/channel_shared", + } +} +``` + +
+ +
+channel_unarchived* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "cool-channel", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/channel_unarchived", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+channel_unshared + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "cool-channel", + "channel_type": "public/private/im/mpim", + "disconnected_team_id": "E0123ABC", + "event_type": "slack#/events/channel_unshared", + "is_ext_shared": false, + } +} +``` + +
+ +
+dnd_updated* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "dnd_status": { + "dnd_enabled": true, + }, + "event_type": "slack#/events/user_updated_dnd", + "user_id": "U0123ABC", + } +} +``` +
+ +
+emoji_changed* + + +##### Emoji added + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "event_type": "slack#/events/emoji_changed", + "name": "picard_facepalm", + "subtype": "add", + "value": "https://my.slack.com/emoji/picard_facepalm/abc123.gif" + } +} +``` + +##### Emoji removed + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "event_type": "slack#/events/emoji_changed", + "names": ["picard_facepalm"], + "subtype": "remove", + } +} +``` + +##### Emoji renamed + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "event_type": "slack#/events/emoji_changed", + "new_name": "captain_picard_facepalm", + "old_name": "picard_facepalm", + "subtype": "rename", + "value": "https://my.slack.com/emoji/picard_facepalm/abc123.gif" + } +} +``` + +
+ +
+user_joined_channel + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_type" : "public/private/im/mpim", + "event_type": "slack#/events/user_joined_channel", + "inviter_id": "U0123ABC", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+user_left_channel + + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_type" : "public/private/im/mpim", + "event_type": "slack#/events/user_left_channel", + "user_id": "W0123ABC", + } +} +``` + +
+ +
+message_posted + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/message_posted", + "message_ts": "1355517523.000005", + "text": "Hello world", + "thread_ts": "1355517523.000006", // Nullable + "user_id": "U0123ABC", + } +} +``` + +
+ +
+message_metadata_posted + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "app_id": "A0123ABC", + "channel_id": "C0123ABC", + "event_type": "slack#/events/message_metadata_posted", + "message_ts": "1630708981.000001", + "metadata": { + "event_type": "incident_created", + "event_payload": { + "incident": { + "id": 123, + "summary": "Someone tripped over", + "sev": 1 + } + } + }, + "user_id": "U0123ABC", + } +} +``` + +
+ +
+pin_added + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_type": "public/private/im/mpim", + "channel_name": "project_planning", + "event_type": "slack#/events/pin_added", + "message_ts": "1360782804.083113", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+pin_removed + + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "channel_name": "project_planning", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/pin_removed", + "message_ts": "1360782804.083113", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+reaction_added + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "event_type": "slack#/events/reaction_added", + "message_context": { + "message_ts": "1535430114.000100", + "channel_id": "C0123ABC", + }, + "message_ts": "1535430114.000100", + "message_link": "https:\/\/example.slack.com\/archives\/C0123ABC\/p1535430114000100", + "reaction": "joy", + "user_id": "U0123ABC", + "item_user": "U0123ABC", + "parent_message_link": "https:\/\/example.slack.com\/archives\/C0123ABC\/p1535430114000100", + } +} +``` + +
+ +
+reaction_removed + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 12345, + "type": "event", + "data": { + "channel_id": "C0123ABC", + "event_type": "slack#/events/reaction_removed", + "message_ts": "1535430114.000100", + "reaction": "thumbsup", + "user_id": "U0123ABC", + } +} +``` + +
+ +
+shared_channel_invite_accepted* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.088700, + "type": "event", + "data": { + "accepting_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "approval_required": false, + "channel_id": "C12345678", + "channel_name": "test-slack-connect", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/shared_channel_invite_accepted", + "invite": { + "date_created": 1626876000, + "date_invalid": 1628085600, + "id": "I0ABC123", + "inviting_team": { + "date_created": 1480946400, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + }, + "inviting_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "recipient_email": "golden@doodle.com", + "recipient_user_id": "U87654321", + }, + "teams_in_channel": [ + { + "date_created": 1626789600, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + } + ], + } +} +``` + +
+ +
+shared_channel_invite_approved* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.0887, + "type": "event", + "data": { + "approving_team_id": "T87654321", + "approving_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "team_id": "T123", + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T12345", + "timezone": "America/Los_Angeles", + }, + "channel_id": "C12345678", + "channel_name": "test-slack-connect", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/shared_channel_invite_approved", + "invite": { + "date_created": 1626876000, + "date_invalid": 1628085600, + "id": "I0123ABC", + "inviting_team": { + "date_created": 1480946400, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + }, + "inviting_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "recipient_email": "golden@doodle.com", + "recipient_user_id": "U87654321" + }, + "teams_in_channel": [ + { + "date_created": 1626789600, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + } + ], + } +} +``` + +
+ +
+shared_channel_invite_declined* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.0887, + "type": "event", + "data": { + "channel_id": "C12345678", + "channel_type": "public/private/im/mpim", + "channel_name": "test-slack-connect", + "declining_team_id": "T87654321", + "declining_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "event_type": "slack#/events/shared_channel_invite_declined", + "invite": { + "date_created": 1626876000, + "date_invalid": 1628085600, + "id": "I0123ABC", + "inviting_team": { + "date_created": 1480946400, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + }, + "inviting_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "recipient_email": "golden@doodle.com", + "recipient_user_id": "U3472391", + }, + "teams_in_channel": [ + { + "date_created": 1626789600, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + } + ], + } +} +``` + +
+ +
+shared_channel_invite_received* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1643810217.0887, + "type": "event", + "data": { + "channel_id": "C12345678", + "channel_name": "test-slack-connect", + "channel_type": "public/private/im/mpim", + "event_type": "slack#/events/shared_channel_invite_received", + "invite": { + "date_created": 1626876000, + "date_invalid": 1628085600, + "id": "I0123ABC", + "inviting_team": { + "date_created": 1480946400, + "domain": "corgis", + "icon": {...}, + "id": "T12345678", + "is_verified": false, + "name": "Corgis", + }, + "inviting_user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + }, + "recipient_email": "golden@doodle.com", + "recipient_user_id": "U87654321" + }, + + } +} +``` + +
+ +
+user_joined_team* + + +```json +{ + "team_id": "T0123ABC", + "enterprise_id": "E0123ABC", + "event_id": "Ev0123ABC", + "event_timestamp": 1630623713, + "type": "event", + "data": { + "event_type": "slack#/events/user_joined_team", + "user": { + "display_name": "John Doe", + "id": "U123", + "is_bot": false, + "name": "John Doe", + "real_name": "John Doe", + "team_id": "T123", + "timezone": "America/Los_Angeles", + } + } +} +``` + +
+ +\*When developing with these event types on Enterprise Grid, you must include the `team_ids` field when creating workspace-based event triggers. + +The data returned in the event response object can be passed along to workflows. In this example, we take the `user_id`, `channel_id`, and `message_ts` from the `reaction_added` event's response object and pass them along to the `joy_workflow` in the `inputs` field, referencing them by their respective enums. + +```javascript +{ + type: TriggerTypes.Event, + name: "Joy reactji event trigger", + description: "Joy reactji trigger", + workflow: "#/workflows/joy_workflow", + inputs: { + user: { + value: TriggerContextData.Event.ReactionAdded.user_id, // Pulled from event response body and passed into the workflow + }, + channel: { + value: TriggerContextData.Event.ReactionAdded.channel_id, // Pulled from event response body and passed into the workflow + }, + message_ts: { + value: TriggerContextData.Event.ReactionAdded.message_ts // Pulled from event response body and passed into the workflow + } + }, + event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456"] + } +} +``` + +## Event trigger filters {#filters} + +Trigger filters allow you to define a set of conditions for a trigger which must be true in order for the trigger to activate. + +Filters can also prevent your app from getting stuck in an infinite loop of responses triggered by responding to events it created. + +Trigger filters are implemented by inserting a filter payload within your `trigger` object. The payload takes the form of an object containing blocks of conditional logic. The logical condition within each block can be one of two types: + +* Conditional expressions (e.g. `x < y`) +* Boolean logic (e.g. `x AND y`) + +### Conditional expressions {#conditional-filters} +Conditional expression blocks need a single `statement` key with a string containing the comparison block. Values from the `inputs` payload can be referenced within the comparison block. + +Below is an example payload of a `reaction_added` event trigger that only invokes a workflow if the reaction was the `:eyes:` reaction. + +```js +{ + type: TriggerTypes.Event, + name: "Reactji response", + description: "responds to a specific reactji", + workflow: "#/workflows/myWorkflow", + event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456"], + filter: { + version: 1, + root: { + statement: "{{data.reaction}} == eyes" + } + } + }, + inputs: { + stringtoSend: { + value: "how cool is that", + }, + channel: { + value: "C123ABC456", + }, + }, +}; +``` + +The supported operand types are `integer`, `double`, `boolean`, `string`, and `null`. The following comparators are supported. + +| Comparator | Supported types | +| ---------- | -------------- | +| `==`| all types | +| `>` | `int`, `double` | +| `<` | `int`, `double` | +| `>=`| `int`, `double` | +| `<=`| `int`, `double` | +| `CONTAINS` | string; i.e. ``` {{data.text}} CONTAINS 'hello' ``` | + +### Boolean logic blocks {#boolean-filters} +Boolean logic blocks are made up of two key:value pairs: + +* An `operator` key with a string containing the comparison operator: `AND`, `OR`, or `NOT` +* An `inputs` key with the child blocks + +The child blocks then contain additional logic. The following example filters on the `reaction` value: + +```ts +{ + type: TriggerTypes.Event, + name: "Reactji response", + description: "responds to a specific reactji", + workflow: "#/workflows/myWorkflow", + event: { + event_type: TriggerEventTypes.ReactionAdded, + channel_ids: ["C123ABC456"], + filter: { + version: 1, + root: { + operator: "OR", + inputs: [{ + statement: "{{data.reaction}} == sunglasses" + }, + { + statement: "{{data.reaction}} == smile" + }], + } + } + }, + inputs: { + stringtoSend: { + value: "how cool is that", + }, + channel: { + value: "C123ABC456", + }, + }, +}; +``` + +:::info[Nested logic blocks] + +You can use the same boolean logic to create nested boolean logic blocks. It's boolean logic all the way down - up to a maximum of 5 nested blocks, that is. The `NOT` operator, however, must contain only one input. Also, at the moment the boolean logic block does _not_ support [short-circuit evaluation](https://en.wikipedia.org/wiki/Short-circuit_evaluation), so all arguments will be evaluated. + +::: + +### Trigger filters in sample apps + +✨ [The Simple Survey App](https://github.com/slack-samples/deno-simple-survey) lets users collect feedback on specific messages. The process begins when a user reacts to a message with the `:clipboard:` reaction. This is done with a `reaction_added` event trigger filtering for the `:clipboard:` reaction. + +✨ [The Daily Topic App](https://github.com/slack-samples/deno-daily-channel-topic) can reply to messages in a channel. The `message_posted` event filters out both messages by apps (to prevent recursive replies) and messages within a thread (to reply only once per thread). + +## Event trigger response {#response} + +The response will have a property called `ok`. If `true`, then the trigger was created, and the `trigger` property will be populated. + +Your response will include a `trigger.id`; be sure to store it! You use that to `update` or `delete` the trigger if need be. See [trigger management](/deno-slack-sdk/guides/managing-triggers). + +## Onward + +➡️ With your trigger created, you can now test your app by [running your app locally](/deno-slack-sdk/guides/developing-locally). + +✨ Once your app is active, see [trigger management](/deno-slack-sdk/guides/managing-triggers) for info on managing your triggers in your workspace. diff --git a/docs/guides/triggers/creating-link-triggers.md b/docs/guides/triggers/creating-link-triggers.md new file mode 100644 index 00000000..dda37c25 --- /dev/null +++ b/docs/guides/triggers/creating-link-triggers.md @@ -0,0 +1,507 @@ +--- +slug: /deno-slack-sdk/guides/creating-link-triggers +--- + +# Creating link triggers + + + +> Invoke a workflow from a channel in Slack + +Link triggers are an *interactive* type of trigger. You typically invoke them by clicking on the associated shortcut link. A link trigger will unfurl into a button when posted in a channel. + +They can be invoked in other ways as well. You can: +* add the link trigger as a bookmark in a channel, then select it +* invoke it with a slash command via the [Shortcut menu](https://api.slack.com/interactivity/shortcuts#global) +* create it as a [workflow button](#workflow_buttons), then click the button + +## Create a link trigger {#create-trigger} + +Triggers can be added to workflows in two ways: + +* **You can add triggers with the CLI.** These static triggers are created only once. You create them with the Slack CLI, attach them to your app's workflow, and that's that. The trigger is typically defined within a trigger file, although you can create a basic link trigger without one. + +* **You can add triggers at runtime.** These dynamic triggers are created at any step of a workflow so they can incorporate data acquired from other workflow steps. The trigger is defined within a function file. + + + + +:::info[Slack CLI built-in documentation] + +You can use `slack trigger --help` to easily access information on the `trigger` command's flags and subcommands. + +::: + +The triggers you create when running locally (with the `slack run` command) will not work when you deploy your app in production (with the `slack deploy` command). You'll need to `create` any triggers again with the CLI. + + +### Create a basic link trigger {#create-cli} + +If your workflow doesn't need any parameters mapped from the trigger, such as `interactivity`, then you can create a trigger using the `trigger create` command: + +``` +slack trigger create --workflow "#/workflows/your_workflow" +``` + +### Create a link trigger with interactivity {#create-cli-interactivity} + +If you need to use the `interactivity` parameter, append the `--interactivity` flag to that command: + +``` +slack trigger create --workflow "#/workflows/your_workflow" --interactivity +``` + +### Create a link trigger with additional parameters {#create-cli-additional-parameters} + +If you need to pass specific values, or use other parameters, you'll need to create a link trigger using a trigger file. The trigger file contains the payload you used to define your trigger. + +Create a TypeScript trigger file within your app's folder with the following form: + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + // your TypeScript payload +}; + +export default trigger; +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case; below is the trigger file from the [Deno Starter Template](https://github.com/slack-samples/deno-starter-template/blob/main/triggers/sample_trigger.ts): + +```js +import { Trigger } from "deno-slack-sdk/types.ts"; +import SampleWorkflow from "../workflows/sample_workflow.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; + +const sampleTrigger: Trigger = { + type: TriggerTypes.Shortcut, + name: "Sample trigger", + description: "A sample trigger", + workflow: "#/workflows/sample_workflow", + inputs: { + interactivity: { + value: TriggerContextData.Shortcut.interactivity, + }, + channel: { + value: TriggerContextData.Shortcut.channel_id, + }, + user: { + value: TriggerContextData.Shortcut.user_id, + }, + }, +}; + +export default sampleTrigger; +``` + +Once you have created a trigger file, use the `trigger create` command to create the link trigger by pointing to a trigger file: + +```bash +slack trigger create --trigger-def "path/to/trigger.ts" +``` + +If you have not used the `slack triggers create` command to create a trigger prior to running the `slack run` command, you will receive a prompt in the Slack CLI to do so. + +### The CLI response + +Once you've instructed Slack just how to create your link trigger using the CLI, it will respond with a shortcut link you can then copy and paste into a Slack channel or bookmarks bar. Use it to activate your workflow. Alternatively, you can use a [slash command](https://api.slack.com/interactivity/slash-commands). + +The triggers you create when running locally (with the `slack run` command) will not work when you deploy your app in production (with the `slack deploy` command). You'll need to `create` any triggers again with the CLI. + +When a `trigger create` command is successful, the CLI's response looks something like: + +```bash +⚡ Trigger successfully created! + + Train markovbot with words Ft0123ABC456 (shortcut) + Created: 2023-01-01 12:34:56 -07:00 (8 seconds ago) + Runnable by: everyone + https://slack.com/shortcuts/Ft0123ABC456/abc123... +``` + +This response includes the trigger name, ID, and type on the first line, followed by the time of creation and access list for the trigger. The last line includes the shortcut link you'll need for invoking the trigger in Slack. You can send this to a channel or add it as a bookmark, then click it to begin the workflow! + +When you need to [modify, remove, or otherwise maintain the triggers you've created](/deno-slack-sdk/guides/managing-triggers), the trigger ID will come in handy to specify which trigger you're updating. No need to memorize it though, since you can use `slack trigger list` to show all available triggers for your app. + + + + +Your app needs to have the [`triggers:write`](https://api.slack.com/scopes/triggers:write) scope to use a trigger at runtime. Include the scope within your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +The logic of a runtime trigger lies within a function's TypeScript code. Within your `functions` folder, you'll have the functions that are the steps making up your workflow. Within this folder is where you can create a trigger within the relevant `.ts` file. + +When you create a runtime trigger, you can leverage `inputs` acquired from functions within the workflow. Provide the workflow definition to get additional typing for the workflow and inputs fields. + +Create a link trigger at runtime using the `client.workflows.triggers.create` method within the relevant function file. + +```js +const triggerResponse = await client.workflows.triggers.create({ + // your TypeScript payload +}); +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. Here's a function file with an example TypeScript payload for a link trigger: + +```js +// functions/example_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +export const ExampleFunctionDefinition = DefineFunction({ + callback_id: "example_function_def", + title: "Example function", + source_file: "functions/example_function.ts", +}); + +export default SlackFunction( + ExampleFunctionDefinition, + ({ inputs, client }) => { + + const triggerResponse = await client.workflows.triggers.create({ + type: TriggerTypes.Shortcut, + name: "My Trigger", + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: { + input_name: { + value: "value", + } + } + }); + + // ... +``` + + + + +--- + +## Link trigger parameters {#parameters} + +| Field | Description | Required? | +|------------------------|---------------------------------------------| --------- | +| `type` | The type of trigger: `TriggerTypes.Shortcut` | Required | +| `name` | The name of the trigger | Required | +| `workflow` | Path to workflow that the trigger initiates | Required | +| `description` | The description of the trigger | Optional | +| `inputs` | The inputs provided to the workflow. See [the `inputs` object](#inputs) below | Optional | +| `shortcut` | Contains `button_text`, if desired | Optional | +| `shortcut.button_text` | The text of the shortcut button | Optional | + +### The `inputs` object {#inputs} + +The `inputs` of a trigger map to the inputs of a [workflow](/deno-slack-sdk/guides/creating-workflows). You can pass any value as an input. You should either provide a `value` for every input, or mark the input as `customizable: true` instead. See [workflow buttons](#workflow_buttons) for info on why you might want to use `customizable: true`. + +There are also a specific set of input values that contain information about the trigger. Pass any of these values to provide trigger information to your workflows! + +Fields that take the form of `data.VALUE` can be referenced in the form of `TriggerContextData.Shortcut.VALUE` + +| Referenced field | Type | Description | +|--------|------------------------|------| +| `TriggerContextData.Shortcut.action_id` | string | Only available when the trigger is invoked from a [Workflow Button](#workflow_buttons)! A unique identifier for the action that invoked the trigger. | +| `TriggerContextData.Shortcut.block_id` | string | Only available when the trigger is invoked from a [Workflow Button](#workflow_buttons)! A unique identifier for the block where the trigger was invoked. | +| `TriggerContextData.Shortcut.bookmark_id` | string | Only available when the trigger is invoked from a channel's bookmarks bar! A unique identifier for the bookmark where the trigger was invoked. | +| `TriggerContextData.Shortcut.channel_id` | string | Only available when the trigger is invoked from a channel, DM or MPDM! A unique identifier for the channel where the trigger was invoked. | +| `TriggerContextData.Shortcut.interactivity` | object | A temporary token for use in building interactive UIs in the Slack client. | +| `TriggerContextData.Shortcut.location` | string | Where the trigger was invoked. Can be `message`, `bookmark` or `button`. | +| `TriggerContextData.Shortcut.message_ts` | string | Only available when the trigger is invoked from a channel, DM or MPDM! A unique Unix timestamp in seconds indicating when the trigger-invoking message was sent. | +| `TriggerContextData.Shortcut.user_id` | string | A unique identifier for the Slack user who invoked the trigger. | +| `TriggerContextData.Shortcut.event_timestamp` | timestamp | A Unix timestamp in seconds indicating when this event was dispatched. | + +The following snippet shows a `channel_id` input being set with a value of `TriggerContextData.Shortcut.channel_id`, which is a unique identifier for the channel where the trigger was invoked. + +```js +... +inputs: { + channel_id: { + value: TriggerContextData.Shortcut.channel_id + } +}, +... +``` + +## Link trigger response {#trigger-response} + +The response will have a property called `ok`. If `true`, then the trigger was created, and the `trigger` property will be populated. + +Your response will include a `trigger.id`; be sure to store it! You use that to `update` or `delete` the trigger if need be. See [trigger management](/deno-slack-sdk/guides/managing-triggers). + + +
+ An example response of a created link trigger + +```json +{ + // If ok == true, the trigger was created + ok: true, + + // The newly created trigger's details are here + trigger: { + // Your trigger's unique ID + id: "Ft12345", + + // inputs will contain a summary of your inputs as defined in the trigger file + inputs: {}, + + // since this is a link trigger, `outputs` will automatically contain: + // {{event_timestamp}}: time when the workflow started + // {{data.user_id}}: The user ID of the person who invoked the trigger + // (by clicking the shortcut link or run button in Slack) + // {{data.channel_id}}: The channel where the shortcut was run + // {{data.interactivity}}: The trigger's interactivity context + outputs: { + "{{event_timestamp}}": { + type: "string", + name: "event_timestamp", + title: "Time when workflow started", + is_required: false, + description: "Time when workflow started" + }, + "{{data.user_id}}": { + type: "slack#/types/user_id", + name: "user_id", + title: "Person who ran this shortcut", + is_required: true, + description: "Person who clicked the shortcut link or run button in Slack" + }, + "{{data.channel_id}}": { + type: "slack#/types/channel_id", + name: "channel_id", + title: "Channel where the shortcut was run", + is_required: false, + description: "Channel where the shortcut was run, if available" + }, + "{{data.interactivity}}": { + type: "slack#/types/interactivity", + name: "interactivity", + title: "Interactivity context", + is_required: true, + description: "Interactivity context", + is_hidden: true + } + }, + + // Trigger-specific information + date_created: 1661894315, + date_updated: 1661894315, + type: "shortcut", + name: "Submit a ticket to our work management system", + description: "", + + // The shortcut URL that will activate this trigger and invoke the underlying workflow + shortcut_url: "https://slack.com/shortcuts/Ft12345/caef7d773d611ddd1da81fd85de08a78", + + // Details about the workflow associated with this trigger + workflow: { + id: "Fn1234567890", + callback_id: "handle_new_tickets_workflow", + title: "Handle new tickets", + description: "Handles a new ticket and updates the submitting user", + type: "workflow", + + // Any workflow inputs will be included here + input_parameters: [], + + // Any of the workflow's outputs will be included here + output_parameters: [], + + app_id: "A1234567890", + + // App-specific details + app: { + id: "A1234567890", + name: "ticket-management-app", + icons: [Object], + is_workflow_app: false + }, + date_created: 1661889787, + date_updated: 1661894304, + date_deleted: 0, + workflow_id: "Wf01234567890" + } + } +} +``` +
+ + +## Workflow buttons {#workflow_buttons} + +You can also create link triggers in the form of workflow buttons! [Workflow buttons](https://api.slack.com/reference/block-kit/block-elements/#workflow_button) are Block Kit elements that allow you to use link triggers with _customizable_ inputs. These customizable inputs are provided by the workflow button itself when it is used, as opposed to the inputs being provided when the link trigger is created. You can create a link trigger once, and then use that same link trigger multiple times with different values. + +You should use workflow buttons when your function's logic execution is about to complete but you want to provide users with follow-up actions in a message. These follow-up actions likely fit better in a separate workflow than within your interactivity handler. Distributing actions for multiple users (like a poll), or updating an object created by your function (like an incident), are common examples of when one would use workflow buttons. + +### Creating the link trigger for a button {#workflow_buttons_create} + +Creating a link trigger to use as a workflow button is much like how you would normally [create a link trigger with the CLI](#create-cli). The only addition is that you need to specify which of the parameters are customizable. Customizable parameters can have their values provided by a workflow button that wraps a link trigger. You can set a maximum of 10 input parameters as customizable. + +The below example shows a sample payload used to create a link trigger that has one customizable parameter: `some_customizable_parameter`, and one non-customizable parameter: `channel_id`. + +```js +// In a file: ./triggers/some_workflow_trigger.ts + +import { Trigger } from "deno-slack-sdk/types.ts"; +import SomeWorkflow from "../workflows/some_workflow.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; + +export const someWorkflowTrigger: Trigger = { + type: TriggerTypes.Shortcut, + name: "Some Workflow trigger", + description: "A trigger for SomeWorkflow that allows for workflow_buttons to customize the value of some_customizable_parameter", + workflow: "#/workflows/some_workflow", + inputs: { + channel_id: { + value: TriggerContextData.Shortcut.channel_id, + }, + some_customizable_parameter: { + customizable: true, + }, + }, +}; +``` + +The trigger definition above is for a workflow defined like so: +```js +// In a file: ./workflows/some_workflow.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const SomeWorkflow = DefineWorkflow({ + callback_id: "some_workflow", + title: "Some Workflow", + input_parameters: { + required: [], + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + }, + some_customizable_parameter: { + type: Schema.types.string, + }, + }, + }, +}); +``` + +Running `slack triggers create --trigger-def="./triggers/some_workflow_trigger.ts"` would output a link trigger URL (e.g. `https://slack.com/shortcuts/Ft0123ABC456/123XYZ`), which will be referenced elsewhere inside the actual `workflow_button` elements. See [below](#workflow_buttons_use) for more details. + +You can have both customizable and non-customizable input parameters, but only the customizable input parameters can be provided elsewhere by a [`workflow_button`](https://api.slack.com/reference/block-kit/block-elements/#workflow_button) element. A non-customizable input parameter does not have a `value`, since it will be provided elsewhere by a `workflow_button` element. + +:::info + +The values used for input parameters set as `customizable: true` may be visible client-side to end users. You should not share sensitive information or secrets via these input parameters. + +::: + +Input parameters marked as `customizable: true` are restricted to input parameters of types `string`, `Schema.slack.types.channel_id`, or `Schema.slack.types.user_id`. + +:::info + +Remember, link triggers are specific to an environment and workspace. You will need to create the link trigger that uses customizable inputs in each environment and workspace you want to use workflow buttons in. + +::: + +### Using the link trigger with a button {#workflow_buttons_use} + +After you have created a link trigger with customizable input parameters, you can use it in a [`workflow_button`](https://api.slack.com/reference/block-kit/block-elements/#workflow_button), within which you will actually provide the values for the customizable input parameters. + +Workflow buttons are only supported in messages and message attachments, and not on views. + +You can use [`workflow_buttons`](https://api.slack.com/reference/block-kit/block-elements/#workflow_button) with any Slack function that accepts `interactive_blocks`, like [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message) and [`SendDm`](/deno-slack-sdk/reference/slack-functions/send_dm). + +```js +MyWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: MyWorkflow.inputs.channel, + message: `Click on this workflow button to run SomeWorkflow (not MyWorkflow!).`, + interactive_blocks: [ + { + type: "actions", + elements: [ + { + type: "workflow_button", + text: { + type: "plain_text", + text: "Run Me", + }, + workflow: { + trigger: { + url: "https://slack.com/shortcuts/Ft0123ABC456/123XYZ", + customizable_input_parameters: [ + { + name: "some_customizable_parameter", + value: MyWorkflow.inputs.input_coming_from_elsewhere, + }, + ], + }, + }, + }, + ], + }, + ], +}); +``` + +The above example is a workflow step for `MyWorkflow`, which has a button that will run a *different* workflow (`SomeWorkflow`) via the link trigger URL `https://slack.com/shortcuts/Ft0123ABC456/123XYZ`. When `SomeWorkflow` gets run through this workflow button, the value of `MyWorkflow.inputs.input_coming_from_elsewhere` will be passed in to `SomeWorkflow`. + +You can also use [`workflow_buttons`](https://api.slack.com/reference/block-kit/block-elements/#workflow_button) when you call [`chat.postMessage`](https://api.slack.com/methods/chat.postMessage), [`chat.scheduleMessage`](https://api.slack.com/methods/chat.scheduleMessage), or [`chat.postEphemeral`](https://api.slack.com/methods/chat.postEphemeral) inside the `blocks` directly. + +```js +export default SlackFunction(MyFunction, async ({ inputs, token, env, client }) => { + const resp = await client.chat.postMessage({ + channel: inputs.channel_id, + blocks: [ + { + type: "header", + text: { + type: "plain_text", + text: "A header above the workflow button.", + }, + }, + { + type: "section", + text: { + type: "mrkdwn", + text: "A section block with a workflow button accessory: ", + }, + accessory: { + type: "workflow_button", + text: { + type: "plain_text", + text: "Run Me", + }, + workflow: { + trigger: { + url: "https://slack.com/shortcuts/Ft0123ABC456/123XYZ", + customizable_input_parameters: [ + { + name: "some_customizable_parameter", + value: inputs.some_value, + }, + ], + }, + }, + }, + }, + ], + }); + ... +}); +``` + +### Data validation {#workflow_buttons_data_validation} +Defining customizable input parameters for your link trigger means that the values for these inputs will be provided elsewhere (where the link trigger is used in a workflow button). + +In your workflow, validate the inputs you receive by ensuring that any provided users are authorized to pass the input, and that the values received are ones you expect to receive and nothing more. + +For example: You're building a workflow that grants salary increases for individuals. Your workflow has three parameters: `approver_user`, `target_user`, `amount`. You'll want to make sure to validate inside +the workflow itself that the provided `approver_user` is authorized to grant a salary increase of `amount` for the `target_user`. + +## Onward + +➡️ With your trigger created, you can now test your app by [running your app locally](/deno-slack-sdk/guides/developing-locally). + +✨ Once your app is active, see [trigger management](/deno-slack-sdk/guides/managing-triggers) for info on managing your triggers in your workspace. \ No newline at end of file diff --git a/docs/guides/triggers/creating-scheduled-triggers.md b/docs/guides/triggers/creating-scheduled-triggers.md new file mode 100644 index 00000000..bb2ddbd2 --- /dev/null +++ b/docs/guides/triggers/creating-scheduled-triggers.md @@ -0,0 +1,473 @@ +--- +slug: /deno-slack-sdk/guides/creating-scheduled-triggers +--- + +# Creating scheduled triggers + + + +> Invoke a workflow at specific time intervals + +Scheduled triggers are an *automatic* type of trigger. This means that once the trigger is created, they do not require any user input. + +Use a scheduled trigger if you need a workflow to kick off after a delay or on an hourly, daily, weekly, or annual cadence. + +## Create a scheduled trigger {#create-trigger} + +Triggers can be added to workflows in two ways: + +* **You can add triggers with the CLI.** These static triggers are created only once. You create them with the Slack CLI, attach them to your app's workflow, and that's that. The trigger is defined within a trigger file. + +* **You can add triggers at runtime.** These dynamic triggers are created at any step of a workflow so they can incorporate data acquired from other workflow steps. The trigger is defined within a function file. + + + + +:::info[Slack CLI built-in documentation] + +Use `slack trigger --help` to easily access information on the `trigger` command's flags and subcommands. + +::: + +The triggers you create when running locally (with the `slack run` command) will not work when you deploy your app in production (with the `slack deploy` command). You'll need to `create` any triggers again with the CLI. + +### Create the trigger file + +To create a scheduled trigger with the CLI, you'll need to create a trigger file. The trigger file contains the payload you used to define your trigger. + +Create a TypeScript trigger file within your app's folder with the following form: + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + // your TypeScript payload +}; + +export default trigger; +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. Below is the trigger file from the [Message Translator](https://github.com/slack-samples/deno-message-translator) app: + +```js +// triggers/daily_maintenance_job.ts +import { Trigger } from "deno-slack-sdk/types.ts"; +import workflowDef from "../workflows/maintenance_job.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +/** + * A trigger that periodically starts the "maintenance-job" workflow. + */ +const trigger: Trigger = { + type: TriggerTypes.Scheduled, + name: "Trigger a scheduled maintenance job", + workflow: `#/workflows/${workflowDef.definition.callback_id}`, + inputs: {}, + schedule: { + // Schedule the first execution 60 seconds from when the trigger is created + start_time: new Date(new Date().getTime() + 60000).toISOString(), + end_time: "2037-12-31T23:59:59Z", + frequency: { type: "daily", repeats_every: 1 }, + }, +}; + +export default trigger; +``` + +### Use the `trigger create` command + +Once you have created a trigger file, use the following command to create the scheduled trigger: + +```bash +slack trigger create --trigger-def "path/to/trigger.ts" +``` + +If you have not used the `slack triggers create` command to create a trigger prior to running the `slack run` command, you will receive a prompt in the Slack CLI to do so. + + + + + +:::info + +Your app needs to have the [`triggers:write`](https://api.slack.com/scopes/triggers:write) scope to use a trigger at runtime. Include the scope within your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +::: + +The logic of a runtime trigger lies within a function's TypeScript code. Within your `functions` folder, you'll have the functions that are the steps making up your workflow. Within this folder is where you can create a trigger within the relevant `.ts` file. + +When you create a runtime trigger, you can leverage `inputs` acquired from functions within the workflow. Provide the workflow definition to get additional typing for the workflow and inputs fields. + +Create a scheduled trigger at runtime using the `client.workflows.triggers.create` method within the relevant `function` file. + +```js +const triggerResponse = await client.workflows.triggers.create({ + // your TypeScript payload +}); +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. Below is the function file with a TypeScript payload for a scheduled trigger from the [Daily Channel Topic](https://github.com/slack-samples/deno-daily-channel-topic) app: + +```js +// functions/create_scheduled_trigger.ts +import { SlackAPI } from "deno-slack-api/mod.ts"; +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +export const CreateScheduledTrigger = DefineFunction({ + title: "Create a scheduled trigger", + callback_id: "create_scheduled_trigger", + source_file: "functions/create_scheduled_trigger.ts", + input_parameters: { + properties: { + channel_id: { + description: "The ID of the Channel to create a schedule for", + type: Schema.slack.types.channel_id, + }, + }, + required: ["channel_id"], + }, + output_parameters: { + properties: { + trigger_id: { + description: "The ID of the trigger created by the Slack API", + type: Schema.types.string, + }, + }, + required: ["trigger_id"], + }, +}); + +export default SlackFunction( + CreateScheduledTrigger, + async ({ inputs, token }) => { + console.log(`Creating scheduled trigger to update daily topic`); + + const client = SlackAPI(token, {}); + const scheduleDate = new Date(); + // Start schedule 1 minute in the future. Start_time must always be in the future. + scheduleDate.setMinutes(scheduleDate.getMinutes() + 1); + + // triggers/sample_scheduled_update_topic.txt has a JSON example of the payload + const scheduledTrigger = await client.workflows.triggers.create({ + name: `Channel ${inputs.channel_id} Schedule`, + workflow: "#/workflows/scheduled_update_topic", + type: TriggerTypes.Scheduled, + inputs: { + channel_id: { value: inputs.channel_id }, + }, + schedule: { + start_time: scheduleDate.toUTCString(), + frequency: { + type: "daily", + repeats_every: 1, + }, + }, + }); + + if (!scheduledTrigger.trigger) { + return { + error: "Trigger could not be created", + }; + } + + console.log("scheduledTrigger has been created"); + + return { + outputs: { trigger_id: scheduledTrigger.trigger.id }, + }; + }, +); +``` + + + + +## Scheduled trigger parameters {#parameters} + +| Field | Description | Required? | +|---------------|---------------------------------------------------------| --------- | +| `type` | The type of trigger: `TriggerTypes.Scheduled` | Required | +| `name` | The name of the trigger | Required | +| `workflow` | Path to workflow that the trigger initiates | Required | +| `schedule` | When and how often the trigger will activate. See the [ `schedule`](#schedule) object below | Required | +| `description` | The description of the trigger | Optional | +| `inputs` | The inputs provided to the workflow. See the [ `inputs`](#inputs) object below | Optional | + +:::info + +Scheduled triggers are not interactive. Use a [link trigger](/deno-slack-sdk/guides/creating-link-triggers) to take advantage of interactivity. + +::: + +### The `inputs` object {#inputs} + +The `inputs` of a trigger map to the inputs of a [workflow](/deno-slack-sdk/guides/creating-workflows). You can pass any value as an input. + +There is also a specific input value that contains information about the trigger. Pass this value to provide trigger information to your workflows! + +Fields that take the form of `data.VALUE` can be referenced in the form of `TriggerContextData.Shortcut.VALUE` + +| Referenced field | Type | Description | +|--------|---------------|-------------| +| `TriggerContextData.Scheduled.user_id` | string | A unique identifier for the user who created the trigger. | + `TriggerContextData.Scheduled.event_timestamp` | timestamp | A Unix timestamp in seconds indicating when this event was dispatched. | + +The following snippet shows a `user_id` input being set with a value of `TriggerContextData.Scheduled.user_id`, representing the user who created the trigger. + +```js +... +inputs: { + user_id: { + value: TriggerContextData.Scheduled.user_id + } +}, +... +``` + +### The `schedule` object {#schedule} + +| Field | Description | Required? | +|--------------------|-----------------------------------------------------------------|-----------| +| `start_time` | ISO date string of the first scheduled trigger | Required | +| `timezone` | Timezone string to use for scheduling | Optional | +| `frequency` | Details on what cadence trigger will activate. See the [`frequency`](#frequency) object below | Optional | +| `occurrence_count` | The maximum number of times trigger will run | Optional | +| `end_time` | If set, this trigger will not run past the provided ISO date string | Optional | + +### The `frequency` object {#frequency} + + +
+ One-time triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `once` | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Optional | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + + +#### Example one-time trigger {#once-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + timezone: "asia/kolkata", + frequency: { + type: "once", + }, + }, +}; +export default schedule; +``` + +
+ + +
+ Hourly triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `hourly` | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Required | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + +#### Example hourly trigger {#hourly-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + end_time: "2040-05-01T14:00:00Z", + frequency: { + type: "hourly", + repeats_every: 2, + }, + }, +}; +export default schedule; +``` + +
+ + +
+ Daily triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `daily` | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Required | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + +#### Example daily trigger {#daily-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + end_time: "2040-05-01T14:00:00Z", + occurrence_count: 3, + frequency: { type: "daily" }, + }, +}; +export default schedule; +``` +
+ + +
+ Weekly triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `weekly` | Required | +| `on_days` | The days of the week the trigger should activate on | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Required | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + +#### Example weekly trigger {#weekly-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + frequency: { + type: "weekly", + repeats_every: 3, + on_days: ["Friday", "Monday"], + }, + }, +}; +export default schedule; +``` + +
+ + +
+ Monthly triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `monthly` | Required | +| `on_days` | The day of the week the trigger should activate on. Provide the `on_week_num` value along with this field. | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Required | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + +#### Example monthly trigger {#monthly-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + frequency: { + type: "monthly", + repeats_every: 3, + on_days: ["Friday"], + on_week_num: 1, + }, + }, +}; +export default schedule; +``` + +
+ + +
+ Yearly triggers + +| Field | Description | Required? | +|-----------------|-------------------------------------------------------------------|-----------| +| `type` | How often the trigger will activate: `yearly` | Required | +| `repeats_every` | How often the trigger will repeat, respective to `frequency.type` | Required | +| `on_week_num` | The nth week of the month the trigger will repeat | Optional | + +#### Example yearly trigger {#yearly-example} + +```js +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import { ScheduledTrigger } from "deno-slack-api/typed-method-types/workflows/triggers/scheduled.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; + +const schedule: ScheduledTrigger = { + name: "Sample", + type: TriggerTypes.Scheduled, + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: {}, + schedule: { + // Starts 60 seconds after creation + start_time: new Date(new Date().getTime() + 60000).toISOString(), + frequency: { + type: "yearly", + repeats_every: 2, + }, + }, +}; +export default schedule; +``` + +
+ +## Scheduled trigger response {#response} + +The response will have a property called `ok`. If `true`, then the trigger was created, and the `trigger` property will be populated. + +Your response will include a `trigger.id`; be sure to store it! You use that to `update` or `delete` the trigger if need be. See [trigger management](/deno-slack-sdk/guides/managing-triggers). + +## Onward + +➡️ With your trigger created, you can now test your app by [running your app locally](/deno-slack-sdk/guides/developing-locally). + +✨ Once your app is active, see [trigger management](/deno-slack-sdk/guides/managing-triggers) for info on managing your triggers in your workspace. diff --git a/docs/guides/triggers/creating-webhook-triggers.md b/docs/guides/triggers/creating-webhook-triggers.md new file mode 100644 index 00000000..372288c3 --- /dev/null +++ b/docs/guides/triggers/creating-webhook-triggers.md @@ -0,0 +1,223 @@ +--- +slug: /deno-slack-sdk/guides/creating-webhook-triggers +--- + +# Creating webhook triggers + + + +> Invoke a workflow when a specific URL receives a POST request + +Webhook triggers are an *automatic* type of trigger that listens for a certain type of data, much like event triggers. + +While event triggers are used for activating a trigger based on *internal* activity, webhooks are instead used when activating a trigger based on *external* activity. In other words, webhook triggers are useful when tying Slack functionality together with non-Slack services. + +There are two steps to using a webhook trigger: + +1. [Create a trigger, either via the CLI or at runtime](#create-trigger) +2. [Invoke the trigger with a POST Request](#invoke-trigger) + +## Create a webhook trigger {#create-trigger} + +Triggers can be added to workflows in two ways: + +* **You can add triggers with the CLI.** These static triggers are created only once. You create them with the Slack CLI, attach them to your app's workflow, and that's that. The trigger is defined within a trigger file. + +* **You can add triggers at runtime.** These dynamic triggers are created at any step of a workflow so they can incorporate data acquired from other workflow steps. The trigger is defined within a function file. + + + + +:::info[Slack CLI built-in documentation] + +Use `slack trigger --help` to easily access information on the `trigger` command's flags and subcommands. + +::: + +The triggers you create when running locally (with the `slack run` command) will not work when you deploy your app in production (with the `slack deploy` command). You'll need to `create` any triggers again with the CLI. + +### Create the trigger file + +To create a webhook trigger with the CLI, you'll need to create a trigger file. The trigger file contains the payload you used to define your trigger. + +Create a TypeScript trigger file within your app's folder with the following form: + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + // your TypeScript payload +}; + +export default trigger; +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. The following is a TypeScript payload for creating a webhook trigger: + + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +const trigger: Trigger = { + type: TriggerTypes.Webhook, + name: "sends 'how cool is that' to my fav channel", + description: "runs the example workflow", + // "myWorkflow" must be a valid callback_id of a workflow + workflow: "#/workflows/myWorkflow", + inputs: { + stringToReverse: { + value: "how cool is that", + }, + channel: { + value: "{{data.channel}}", + }, + }, +}; + +export default trigger; +``` + +### Use the `trigger create` command + +Once you have created a trigger file, use the following command to create the webhook trigger: + +```bash +slack trigger create --trigger-def "path/to/trigger.ts" +``` + +If you have not used the `slack triggers create` command to create a trigger prior to running the `slack run` command, you will receive a prompt in the Slack CLI to do so. + + + + +:::info + +Your app needs to have the [`triggers:write`](https://api.slack.com/scopes/triggers:write) scope to use a trigger at runtime. Include the scope within your app's [manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +::: + +The logic of a runtime trigger lies within a function's TypeScript code. Within your `functions` folder, you'll have the functions that are the steps making up your workflow. Within this folder is where you can create a trigger within the relevant `.ts` file. + +When you create a runtime trigger, you can leverage `inputs` acquired from functions within the workflow. Provide the workflow definition to get additional typing for the workflow and inputs fields. + +Create a webhook trigger at runtime using the `client.workflows.triggers.create` method within the relevant `function` file. + +```js +const triggerResponse = await client.workflows.triggers.create({ + // your TypeScript payload +); +``` + +Your TypeScript payload consists of the [parameters](#parameters) needed for your own use case. Below is a function file with an example TypeScript payload for a webhook trigger. + +```js +// functions/example_function.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { ExampleWorkflow } from "../workflows/example_workflow.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; + +export const ExampleFunctionDefinition = DefineFunction({ + callback_id: "example_function_def", + title: "Example function", + source_file: "functions/example_function.ts", +}); + +export default SlackFunction( + ExampleFunctionDefinition, + ({ inputs, client }) => { + + const triggerResponse = await client.workflows.triggers.create({ + type: TriggerTypes.Webhook, + name: "sends 'how cool is that' to my fav channel", + description: "runs the example workflow", + workflow: `#/workflows/${ExampleWorkflow.definition.callback_id}`, + inputs: { + stringToReverse: { + value: "how cool is that", + }, + channel: { + value: "{{data.channel}}", + }, + } + }); + + // ... +``` + + + + + +--- + +## Webhook trigger parameters {#parameters} +| Field | Description | Required? | +|------------------|---------------------------------------------|-----------| +| `type` | The type of trigger: `TriggerTypes.Webhook` | Required | +| `name` | The name of the trigger | Required | +| `workflow` | Path to workflow that the trigger initiates | Required | +| `description` | The description of the trigger | Optional | +| `inputs` | The inputs provided to the workflow | Optional | +| `webhook` | Contains `filter`, if desired | Optional | +| `webhook.filter` | See [trigger filters](/deno-slack-sdk/guides/creating-event-triggers/#filters) | Optional | + +:::info + +Webhook triggers are not interactive. Use a [link trigger](/deno-slack-sdk/guides/creating-link-triggers) to take advantage of interactivity. + +::: + +## Webhook trigger response {#response} + +The response will have a property called `ok`. If `true`, then the trigger was created, and the `trigger` property will be populated. + +Your response will include a `trigger.id`; be sure to store it! You use that to `update` or `delete` the trigger if need be. See [trigger management](/deno-slack-sdk/guides/managing-triggers). + +--- + +## Invoke the trigger {#invoke-trigger} + +Send a POST request to invoke the trigger. Within that POST request you can send values for specific inputs. + +All JSON objects sent in the POST request need to be flat. Nested JSON objects will return a `parameter_validation_failed` error. + +**Good *flattened* JSON object:** +``` +{"channel":"C123ABC456","user":"U123ABC456"} +``` + +**No good, very bad *nested* JSON object:** + +```js +// JSON does not support comments but we really don't want you using this code +{"channel":"C123ABC456","user":{"first_name":"Jesse","last_name":"Slacksalot"}} +``` + +Now let's look at an entire example. + +### Example POST request {#example} + +``` +curl \ + -X POST "https://hooks.slack.com/triggers/T123ABC456/.../..." \ + --header "Content-Type: application/json; charset=utf-8" \ + --data '{"channel":"C123ABC456"}' +``` + +If the webhook was received and successfully handled, you'll get the following response: + +``` +{ + "ok":true +} +``` + +## Onward {#onward} + +➡️ With your trigger created, you can now test your app by [running your app locally](/deno-slack-sdk/guides/developing-locally). + +✨ Once your app is active, see [trigger management](/deno-slack-sdk/guides/managing-triggers) for info on managing your triggers in your workspace. \ No newline at end of file diff --git a/docs/guides/triggers/managing-triggers.md b/docs/guides/triggers/managing-triggers.md new file mode 100644 index 00000000..d541074a --- /dev/null +++ b/docs/guides/triggers/managing-triggers.md @@ -0,0 +1,134 @@ +--- +slug: /deno-slack-sdk/guides/managing-triggers +--- + +# Managing triggers + + + +All triggers can be updated, deleted, viewed, and have their access restricted in the same way. + +## Update a trigger {#update} + +### Update a trigger with the CLI {#update_cli} + +Make an update to a pre-existing trigger with the CLI by using the `slack trigger update` command. Provide the same payload you used to create the trigger *in its entirety*, in addition to the trigger ID. + +```bash +slack trigger update --trigger-id Ft123ABC --trigger-def "path/to/trigger.ts" +``` + +### Update a trigger at runtime {#update_runtime} + +You can update a runtime trigger, but the trigger must be updated in its entirety. Use the same structure as `client.workflows.triggers.create()` but for `client.workflows.triggers.update` with the additional `trigger_id` parameter. + +```js +const triggerId = "FtABC123"; +const response = await client.workflows.triggers.update({ + trigger_id: triggerId, + type: "", + name: "My trigger", + workflow: "#/workflows/myworkflow", + inputs: { + input_name: { + value: "value", + } + } +}); +// Error handling example in your custom function +if (response.error) { + const error = `Failed to update a trigger (id: ${triggerId}) due to ${repsonse.error}`; + return { error }; +} +``` + +## Delete a trigger {#delete} + +### Delete a trigger with the CLI {#delete_cli} + +You can delete a trigger with the `slack trigger delete` command. + +```bash +slack trigger delete --trigger-id FtABC123 +``` + + +### Delete a trigger at runtime {#delete_runtime} + +Deleting a runtime trigger deletes that *specific* trigger created in one instance of the workflow. This means that you'll need to have stored the `trigger_id` created for that instance. Your app will continue to be able to create triggers until you remove the relevant code. + +You can delete a runtime trigger by using `client.workflows.triggers.delete()`. + +```js +const response = await client.workflows.triggers.delete({ + trigger_id: "FtABC123" +}); +// Error handling example in your custom function +if (response.error) { + const error = `Failed to delete a trigger due to ${response.error}`; + return { error }; +} +``` + +## List a trigger {#list} + +Triggers created in your local development environment will only work if your application is still running locally. You can view triggers created in your local development environment with the `slack run --show-triggers` command. Triggers created in a deployed environment will not be returned. + +You can use the `slack triggers list` command to view information about your app's triggers, including the trigger ID, name, type, creation, and last updated time. When you use that command, you'll be prompted to select the workspace and then the environment (either local or deployed) for the triggers to list. + +## Manage access to a trigger {#manage} + +A newly-created trigger is accessible to anyone inside the workspace by default. You can manage who can access the trigger using the `access` Slack CLI command. + +### Grant access {#grant-access} + +| Required Flag | Description | Example Argument | +|----------------|------------------------------------------|------------------| +| `--grant` | A switch to grant access | | +| `--trigger-id` | The `trigger_id` of the desired trigger | `Ft123ABC` | + +Set one of the following flags to grant access to different groups. If no flag is selected you will be prompted to select a group within the Slack CLI. + +| Flag | Description | Example Argument | +|-----------------------|----------------------------------------------------------|--------------------| +| `--app-collaborators` | A switch to grant access to all app collaborators | | +| `--channels` | The channel IDs of channels to be granted access | `C123ABC, C456DEF` | +| `--everyone` | A switch to grant access to all workspace members | | +| `--organizations` | The enterprise IDs of organizations to be granted access | `E123ABC, E456DEF` | +| `--users` | The user ID of users to be granted access | `U123ABC, U456DEF` | +| `--workspaces` | The team IDs of workspaces to be granted access | `T123ABC, T456DEF` | + +You can combine types of named entities (channels, organizations, users, and workspaces) in a single command. For example, the following command grants access to the trigger `FtABC123` for channel `C123ABC`, organization `E123ABC`, user `U123ABC` and workspace `T123ABC`: + +```bash +slack trigger access --trigger-id Ft123ABC --channels C123ABC --organizations E123ABC --users U123ABC --workspaces T123ABC --grant +``` +### Revoke access {#revoke-access} + +| Required Flag | Description | Example Argument | +|----------------|------------------------------------------|------------------| +| `--revoke` | A switch to revoke access | | +| `--trigger-id` | The `trigger_id` of the desired trigger | `Ft123ABC` | + +Set one of the following flags to revoke access to different groups. If no flag is selected you will be prompted to select a group within the Slack CLI. + +| Flag | Description | Example Argument | +|-------------------|-------------------------------------------------------------------|--------------------| +| `--channels` | The channel IDs of channels whose access will be revoked | `C123ABC, C456DEF` | +| `--organizations` | The enterprise IDs of organizations whose access will be revoked | `E123ABC, E456DEF` | +| `--users` | The user IDs of users whose access will be revoked | `U123ABC, U456DEF` | +| `--workspaces` | The team IDs of workspaces whose access will be revoked | `T123ABC, T456DEF` | + +The following example command revokes access to the trigger `FtABC123` for users `U123ABC` and `U456DEF` and channels `C123ABC` and `C456DEF`. + +```bash +slack trigger access --trigger-id FtABC123 --users U123ABC, U456DEF --channels C123ABC, C456DEF --revoke +``` + +## Onward + +Remember, you won't be able to use any triggers unless your app is active. + +✨ **If you're still testing out your app** you'll want to [run your app locally](/deno-slack-sdk/guides/developing-locally). + +✨ **If your app is ready to go** you'll want to [deploy it to Slack](/deno-slack-sdk/guides/deploying-to-slack). diff --git a/docs/guides/triggers/using-triggers.md b/docs/guides/triggers/using-triggers.md new file mode 100644 index 00000000..e4e7a2b5 --- /dev/null +++ b/docs/guides/triggers/using-triggers.md @@ -0,0 +1,75 @@ +--- +slug: /deno-slack-sdk/guides/using-triggers +--- + +# Using triggers + + + +> Triggers are wonderful things! + +Triggers are one of the three building blocks that make up workflow apps. You will encounter all three as you navigate the path of building your workflow app: + +1. Functions define the actions of your app. +2. Workflows are a combination of functions, executed in order. +3. Triggers execute workflows (⬅️ you are here) + +Since triggers are what kick off your workflows, you need to have a workflow before you can create a trigger. Acquaint yourself with the [documentation on workflows](/deno-slack-sdk/guides/creating-workflows), then head back here. We'll wait! + +With the knowledge of workflows within your noggin, let's take a look at how you can implement triggers into your new app. + +## Understanding triggers {#understanding} + +You will come to many forks in this metaphorical road that is trigger implementation. There are no wrong choices; all roads lead to your own wonderful workflow. + +Triggers can be added to workflows in two ways: + +* **You can add triggers with the CLI.** These static triggers are created only once. You attach them to your app's workflow, create them with the Slack CLI, and that's that. + +* **You can add triggers at runtime.** These dynamic triggers are created at any step of a workflow so they can incorporate data acquired from other workflow steps. + +Triggers created for a locally-running app (with the `slack run` command) are distinct from triggers created for an app in a production environment (with the `slack deploy` command). + +## Custom trigger paths {#custom-path} + +While Slack assumes your triggers will be located in the default `/triggers` directory in the root of your project, it also allows the flexibility to define them elsewhere. In order for Slack to find them, be sure to declare the alternate path in `slack.json` with the `config.trigger-paths` property. It might look like this: + +```json +{ + "hooks": { + ... + }, + "config": { + "trigger-paths": ["my-triggers/*.ts"] + } +} +``` + +## Trigger types {#types} + +There are four types of triggers, each one having its own specific implementation. + +:::info + +In a hurry? You can create a basic [link trigger](/deno-slack-sdk/guides/creating-link-triggers) with a single Slack CLI command! + +::: + +| Trigger type | Use case | +|----------------------------------|---------------------------------------------------------------| +| [Link triggers](/deno-slack-sdk/guides/creating-link-triggers) | Invoke a workflow from a public channel in Slack | +| [Scheduled triggers](/deno-slack-sdk/guides/creating-event-triggers) | Invoke a workflow at specific time intervals | +| [Event triggers](/deno-slack-sdk/guides/creating-event-triggers) | Invoke a workflow when a specific event happens in Slack | +| [Webhook triggers](/deno-slack-sdk/guides/creating-webhook-triggers) | Invoke a workflow when a specific URL receives a POST request | + +## Onward + +Each type of trigger has a guide where you will learn how to create that type of trigger. Choose one to move forward: + +➡️ **To learn more about link triggers,** read the [link triggers](/deno-slack-sdk/guides/creating-link-triggers) documentation or explore the [Give Kudos](/deno-slack-sdk/tutorials/give-kudos-app) sample app. This sample app uses a link trigger to allow users to open up a form to give a compliment to a user. + +✨**To learn more about scheduled triggers,** read the [scheduled triggers](/deno-slack-sdk/guides/creating-event-triggers) documentation or explore the [Virtual Running Buddies](/deno-slack-sdk/tutorials/virtual-running-buddies-app) sample app. This app uses a scheduled trigger to post a weekly message to a channel about people's running activity. + +✨**To learn more about event triggers,** read the [event triggers](/deno-slack-sdk/guides/creating-event-triggers) documentation or explore the [Welcome Bot](/deno-slack-sdk/tutorials/welcome-bot) sample app. This app uses an event trigger to send a message to a user when they join a channel. + +✨**To learn more about webhook triggers,** read the [webhook triggers](/deno-slack-sdk/guides/creating-webhook-triggers) documentation. \ No newline at end of file diff --git a/docs/guides/using-the-app-manifest.md b/docs/guides/using-the-app-manifest.md new file mode 100644 index 00000000..b1e6cf84 --- /dev/null +++ b/docs/guides/using-the-app-manifest.md @@ -0,0 +1,107 @@ +--- +slug: /deno-slack-sdk/guides/using-the-app-manifest +--- + +# Using the app manifest + + + +An app's [manifest](https://api.slack.com/concepts/manifests) is where you can configure its name and scopes, declare the functions your app will use, and [more](#manifest-properties). + +The manifest file, named `manifest.ts` is located within the root of your directory. Inside the manifest file, you will find an `export default Manifest` block that defines the app's configuration. + +For an example, below is an annotated version of the manifest for our [Hello World app](https://github.com/slack-samples/deno-hello-world): + +```javascript +// manifest.ts +import { Manifest } from "deno-slack-sdk/mod.ts"; +// Import your workflow +import GreetingWorkflow from "./workflows/greeting_workflow.ts"; + +export default Manifest({ + + // This is the internal name for your app. + // It can contain spaces (e.g., "My App") + name: "deno-hello-world", + + // A description of your app that will help users decide whether to use it. + description: "A sample that demonstrates using a function, workflow and trigger to send a greeting", + + // Your app's profile picture that will appear in the Slack client. + icon: "assets/default_new_app_icon.png", + + // A list of all workflows your app will use. + workflows: [GreetingWorkflow], + + // If your app communicates to any external domains, list them here. + outgoingDomains: [], // e.g., myapp.tld + + // Bot scopes can be declared here. + // For the beta, you can keep these as-is. + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +## Manifest properties {#manifest-properties} + +|Property|Type|Description| Required? | +|---|---|---|-----| +| `name` | String | The internal name for your app. It can contain spaces (e.g., "My App") | Required | +| `description` |String| A short sentence describing your application. A description of your app that will help users decide whether to use it | Required | +| `icon` | String | A relative path to an image asset to use for the app's icon. Your app's profile picture that will appear in the Slack client. Note this icon is only used if the app is deployed with `slack deploy`. | Required | +| `botScopes` | Array of Strings | A list of bot [scopes](https://api.slack.com/scopes?filter=granular_bot), or permissions, the app's functions require | Required | +| `displayName` | String | A custom name for the app to be displayed that's different from the `name` | Optional | +| `longDescription` | String| A more detailed description of your application | Optional | +| `backgroundColor` | String | A six digit combination of numbers and letters (the hexadecimal color code) that make up the color of your app background e.g., "#000000" is the color black | Optional | +| `functions` | Array | A list of all functions your app will use | Optional | +| `workflows` | Array | A list of all workflows your app will use | Optional | +| `outgoingDomains` | Array of Strings | As of [`v1.15.0`]( https://api.slack.com/changelog#entry-november_2022_0): if your app communicates to any external domains, list them here. If you make API calls to `slack.com`, it does not need to be explicitly listed. e.g., myapp.tld | Optional | +| `events` | Array | A list of all event structures that the app is expecting to be passed via [Message Metadata](https://api.slack.com/metadata/using)) | Optional | +| `types` | Array | A list of all [custom types](/deno-slack-sdk/guides/creating-a-custom-type) your app will use | Optional | +| `datastores` | Array | A list of all [Datastores](/deno-slack-sdk/guides/using-datastores) your app will use | Optional | +| `features` | Object | A configuration object of your app features | Optional | + +### More on outgoing domains {#outgoing-domains} + +[Deno](/deno-slack-sdk/guides/installing-deno) requires explicit permission to access external resources. Therefore, to make HTTP requests to external domains, you'll need to add the domain to your app manifest as an outgoing domain. Otherwise, you may run into a `PermissionDenied: Detected missing network permissions` error. + +## Function definitions in the manifest {#functions} + +You may also see some function definitions in the manifest file. While [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) _can_ be defined here, to keep your code tidy, we recommend defining your functions in their own respective source files in your app's `/functions` directory. + +Regardless of where you define them, each function your app uses must be declared in the manifest. + +## The Messages tab {#messages-tab} + +By default, apps created with `slack create` will include both a read-only Messages tab and an About tab within Slack. + +You can use the [Slack function](/deno-slack-sdk/guides/creating-slack-functions) [`SendDm`](/deno-slack-sdk/reference/slack-functions/send_dm) to send users direct messages from your app—which will appear for them in the app's Messages tab. + +Your app's Messages tab will be enabled and read-only by default. If you'd like to disable read-only mode and/or disable the Messages tab completely, add the optional `features` property to your manifest definition like this: + +```javascript +// manifest.ts +import { Manifest } from "deno-slack-sdk/mod.ts"; +import GreetingWorkflow from "./workflows/greeting_workflow.ts"; + +export default Manifest({ + name: "deno-hello-world", + description: + "A sample that demonstrates using a function, workflow and trigger to send a greeting", + icon: "assets/default_new_app_icon.png", + workflows: [GreetingWorkflow], + outgoingDomains: [], + // Add this ------ + features: { + appHome: { + messagesTabEnabled: false, + messagesTabReadOnlyEnabled: false, + }, + }, + // --------------- + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` +--- + +➡️ **To keep building your new app**, head to the [Slack functions](/deno-slack-sdk/guides/creating-slack-functions) section. \ No newline at end of file diff --git a/docs/index.md b/docs/index.md new file mode 100644 index 00000000..41cce612 --- /dev/null +++ b/docs/index.md @@ -0,0 +1,37 @@ +# Deno Slack SDK + + + +You can create Slack-hosted workflows written in TypeScript using the [Deno Slack SDK](https://github.com/slackapi/deno-slack-sdk). + +Workflows are a combination of functions, executed in order. + +There are a three types of functions: +- **Slack functions** enable Slack-native actions, like creating a channel or sending a message. +- **Connector functions** enable actions native to services _outside_ of Slack. Google Sheets, Dropbox and Microsoft Excel are a few of the services with available connector functions. +- **Custom functions** enable developer-specific actions. Pass in any desired inputs, perform any actions you can code up, and pass on outputs to other parts of your workflows. + +Workflows are invoked via triggers. You can invoke workflows: +- via a link within Slack, +- on a schedule, +- when specified events occurs, +- or via webhooks. + +Workflows make use of specifically-designed features of the Slack platform such as [datastores](/deno-slack-sdk/guides/using-datastores), a Slack-hosted way to store data. + +While in development, you can keep your project mostly to yourself, or share it with a close collaborator. If your Slack admin requires approval of app installations, they’ll need to approve what you’re creating first. + +:::warning + +The app management UI on `api.slack.com/apps` doesn’t support configuring workflow apps. Also, workflow apps are currently not eligible for listing in the Slack Marketplace. + +:::: + +## Getting help + +These docs have lots of information on the Deno Slack SDK. Please explore! + +If you otherwise 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/deno-slack-sdk/issues) for questions, bug reports, feature requests, and general discussion related to Bolt for JavaScript. Try searching for an existing issue before creating a new one. +* [Email](mailto:support@slack.com) our developer support team: `support@slack.com`. \ No newline at end of file diff --git a/docs/manifest.md b/docs/manifest.md deleted file mode 100644 index e2248f12..00000000 --- a/docs/manifest.md +++ /dev/null @@ -1,46 +0,0 @@ -## Manifest - -A Manifest defines your entire Slack application, from its core properties like -its name and description to its behavioural aspects like what -[functions][functions] it contains. - -### Defining a manifest - -A Manifest can be defined with the top level `Manifest` export. Below is an -example, taken from the template application: - -```ts -import { Manifest } from "slack-cloud-sdk/mod.ts"; -import { ReverseString } from "./functions/reverse_definition.ts"; - -export default Manifest({ - name: "heuristic-tortoise-312", - description: "A demo showing how to use Slack functions", - icon: "assets/icon.png", - botScopes: ["commands", "chat:write", "chat:write.public"], - functions: [ReverseString], - datastores: [], - outgoing_domains: [], -}); -``` - -The object passed into the `Manifest` method is the type -[`SlackManifestType`][manifest-type]. Check out [its definition][manifest-type] -for the full list of attributes it supports, but the minimum required properties -are listed in the table below: - -| Property | Type | Description | -| ------------- | ------------- | ------------------------------------------------------------------------- | -| `name` | string | Your Slack application name. | -| `description` | string | A short sentence describing your application. | -| `icon` | string | A relative path to an image asset to use for the application's icon. | -| `botScopes` | Array | A list of [scopes][scopes], or permissions, the bot requires to function. | - -Furthermore, to set up how your application works, you would create -[functions][functions], and register them in the Manifest using the `functions` -property of [`SlackManifestType`][manifest-type] argument used when creating a -new `Manifest`. - -[functions]: ./functions.md -[manifest-type]: ../src/types.ts#L13 -[scopes]: https://api.slack.com/scopes diff --git a/docs/reference/connector-functions.md b/docs/reference/connector-functions.md new file mode 100644 index 00000000..833521aa --- /dev/null +++ b/docs/reference/connector-functions.md @@ -0,0 +1,161 @@ +# Connectors reference catalog + +| Service | Connector | Connector description +|:---|:---|:---| +Adobe Sign | [`send_agreement`](/deno-slack-sdk/reference/connector-functions/adobe.sign/send_agreement) | Send an agreement | +Airtable | [`add_record`](/deno-slack-sdk/reference/connector-functions/airtable/add_record) | Add a record | +Airtable| [`delete_record`](/deno-slack-sdk/reference/connector-functions/airtable/delete_record) | Delete a record | +Airtable| [`select_record`](/deno-slack-sdk/reference/connector-functions/airtable/select_record) | Select a record | +Airtable| [`update_record`](/deno-slack-sdk/reference/connector-functions/airtable/update_record) | Update a record | +Asana | [`add_task_to_section`](/deno-slack-sdk/reference/connector-functions/asana/add_task_to_section) | Add task to a section | +Asana | [`create_project`](/deno-slack-sdk/reference/connector-functions/asana/create_project) | Create a project | +Asana | [`create_comment`](/deno-slack-sdk/reference/connector-functions/asana/create_comment) | Comment on a task | +Asana | [`create_task`](/deno-slack-sdk/reference/connector-functions/asana/create_task) | Create a task | +Asana | [`update_task`](/deno-slack-sdk/reference/connector-functions/asana/update_task) | Update a task | +Atlassian Bitbucket |[`create_issue`](/deno-slack-sdk/reference/connector-functions/atlassian.bitbucket/create_issue) | Create an issue | +Atlassian Bitbucket | [`merge_pull_request`](/deno-slack-sdk/reference/connector-functions/atlassian.bitbucket/merge_pull_request) | Merge pull request | +Box | [`create_folder`](/deno-slack-sdk/reference/connector-functions/box.core/create_folder) | Create a folder | +Box | [`copy_file`](/deno-slack-sdk/reference/connector-functions/box.core/copy_file) | Copy a file | +Box Sign | [`create_sign_request`](/deno-slack-sdk/reference/connector-functions/box.sign/create_sign_request) | Create a sign request with a template | +Basecamp | [`create_project`](/deno-slack-sdk/reference/connector-functions/basecamp/create_project) | Create a project | +Basecamp | [`create_todo`](/deno-slack-sdk/reference/connector-functions/basecamp/create_todo) | Create a to-do | +Basecamp | [`create_todo_list`](/deno-slack-sdk/reference/connector-functions/basecamp/create_todo_list) | Create a to-do list | +Basecamp | [`mark_todo_complete`](/deno-slack-sdk/reference/connector-functions/basecamp/mark_todo_complete) | Mark a to-do complete | +Basecamp | [`mark_todo_pending`](/deno-slack-sdk/reference/connector-functions/basecamp/mark_todo_pending) | Mark a to-do pending | +ClickUp | [`create_task`](/deno-slack-sdk/reference/connector-functions/clickup/create_task) | Create a task in a folder | +Calendly | [`get_meeting_link`](/deno-slack-sdk/reference/connector-functions/calendly/get_meeting_link) | Get meeting link | +Deel | [`add_time_off_request`](/deno-slack-sdk/reference/connector-functions/deel/add_time_off_request) | Add a time off request | +Deel | [`create_contract`](/deno-slack-sdk/reference/connector-functions/deel/create_contract) | Create a new contract | +Dialpad | [`send_sms`](/deno-slack-sdk/reference/connector-functions/dialpad/send_sms) | Send an SMS | +DocuSign | [`create_envelope`](/deno-slack-sdk/reference/connector-functions/docusign/create_envelope) | Create an envelope | +DocuSign | [`send_envelope`](/deno-slack-sdk/reference/connector-functions/docusign/send_envelope) | Send an envelope | +Dropbox | [`copy_document`](/deno-slack-sdk/reference/connector-functions/dropbox.core/copy_document) | Copy a document | +Dropbox | [`delete_document`](/deno-slack-sdk/reference/connector-functions/dropbox.core/delete_document) | Delete a document | +Dropbox | [`move_document`](/deno-slack-sdk/reference/connector-functions/dropbox.core/move_document) | Move a document | +Dropbox | [`create_folder`](/deno-slack-sdk/reference/connector-functions/dropbox.core/create_folder) | Create a folder | +Dropbox | [`create_share_link`](/deno-slack-sdk/reference/connector-functions/dropbox.core/create_share_link) | Create a shared link | +Dropbox Sign | [`send_signature_request_with_template`](/deno-slack-sdk/reference/connector-functions/dropbox.sign/send_signature_request_with_template) | Send signature request using a template | +FireHydrant | [`create_incident`](/deno-slack-sdk/reference/connector-functions/firehydrant/create_incident) | Create an incident | +FireHydrant | [`create_task`](/deno-slack-sdk/reference/connector-functions/firehydrant/create_task) | Create a task | +FireHydrant | [`update_task`](/deno-slack-sdk/reference/connector-functions/firehydrant/update_task) | Update a task | +FireHydrant | [`update_incident`](/deno-slack-sdk/reference/connector-functions/firehydrant/update_incident) | Update an incident | +Giphy | [`get_translated_gif`](/deno-slack-sdk/reference/connector-functions/giphy/get_translated_gif) | Search for a GIF | +Giphy | [`get_random_gif`](/deno-slack-sdk/reference/connector-functions/giphy/get_random_gif) | Random GIF | +GitHub | [`create_issue`](/deno-slack-sdk/reference/connector-functions/github.cloud/create_issue) | Create an issue | +GitHub Enterprise Server | [`create_issue`](/deno-slack-sdk/reference/connector-functions/github.enterprise_server/create_issue) | Create an issue | +GitLab | [`create_issue`](/deno-slack-sdk/reference/connector-functions/gitlab/create_issue) | Create an issue | +Google Calendar | [`add_to_event`](/deno-slack-sdk/reference/connector-functions/google.calendar/add_to_event) | Add attendee to an event | +Google Calendar| [`create_event`](/deno-slack-sdk/reference/connector-functions/google.calendar/create_event) | Create a calendar event | +Google Mail | [`send_email`](/deno-slack-sdk/reference/connector-functions/google.mail/send_email) | Send an email | +Google Meet | [`start_meeting`](/deno-slack-sdk/reference/connector-functions/google.meet/start_meeting) | Start a meeting | +Google Sheets | [`add_spreadsheet_row`](/deno-slack-sdk/reference/connector-functions/google.sheets/add_spreadsheet_row) | Add to spreadsheet | +Google Sheets | [`delete_spreadsheet_row`](/deno-slack-sdk/reference/connector-functions/google.sheets/delete_spreadsheet_row) | Delete from a spreadsheet | +Google Sheets | [`update_spreadsheet_row`](/deno-slack-sdk/reference/connector-functions/google.sheets/update_spreadsheet_row) | Update a spreadsheet | +Google Sheets | [`select_spreadsheet_row`](/deno-slack-sdk/reference/connector-functions/google.sheets/select_spreadsheet_row) | Select a spreadsheet row | +Google Tasks | [`create_tasklist`](/deno-slack-sdk/reference/connector-functions/google.tasks/create_tasklist) | Create a task list | +Google Tasks| [`insert_task`](/deno-slack-sdk/reference/connector-functions/google.tasks/insert_task) | Insert a task | +Greenhouse Onboarding | [`fetch_employees`](/deno-slack-sdk/reference/connector-functions/greenhouse.onboarding/fetch_employees) | Fetch employees | +Greenhouse Onboarding| [`create_pending_hire`](/deno-slack-sdk/reference/connector-functions/greenhouse.onboarding/create_pending_hire) | Create pending hire | +Greenhouse Recruiting | [`hire_application`](/deno-slack-sdk/reference/connector-functions/greenhouse.recruiting/hire_application) | Hire Application | +Greenhouse Recruiting| [`list_candidate_activity`](/deno-slack-sdk/reference/connector-functions/greenhouse.recruiting/list_candidate_activity) | Candidate activity | +Greenhouse Recruiting| [`reject_application`](/deno-slack-sdk/reference/connector-functions/greenhouse.recruiting/reject_application) | Reject Application | +Greenhouse Recruiting| [`list_job_candidates`](/deno-slack-sdk/reference/connector-functions/greenhouse.recruiting/list_job_candidates) | List job candidates | +Guru | [`create_card`](/deno-slack-sdk/reference/connector-functions/guru/create_card) | Create a card | +Guru| [`delete_card`](/deno-slack-sdk/reference/connector-functions/guru/delete_card) | Delete a card | +Guru| [`unverify_card`](/deno-slack-sdk/reference/connector-functions/guru/unverify_card) | Unverify a card | +Guru| [`update_card`](/deno-slack-sdk/reference/connector-functions/guru/update_card) | Update a card | +Guru| [`verify_card`](/deno-slack-sdk/reference/connector-functions/guru/verify_card) | Verify a card | +Guru| [`add_comment`](/deno-slack-sdk/reference/connector-functions/guru/add_comment) | Add a comment | +Intercom | [`create_article`](/deno-slack-sdk/reference/connector-functions/intercom/create_article) | Create an article | +Intercom| [`create_ticket`](/deno-slack-sdk/reference/connector-functions/intercom/create_ticket) | Create a ticket | +JIRA Cloud | [`edit_issue`](/deno-slack-sdk/reference/connector-functions/jira.cloud/edit_issue) | Edit an issue | +JIRA Cloud | [`create_issue`](/deno-slack-sdk/reference/connector-functions/jira.cloud/create_issue) | Create an issue | +launchdarkly | [`create_approval_request_update_flag`](/deno-slack-sdk/reference/connector-functions/launchdarkly/create_approval_request_update_flag) | Create approval request to update a feature flag's state | +launchdarkly| [`create_feature_flag`](/deno-slack-sdk/reference/connector-functions/launchdarkly/create_feature_flag) | Create a boolean feature flag | +LaunchDarkly| [`update_feature_flag_state`](/deno-slack-sdk/reference/connector-functions/launchdarkly/update_feature_flag_state) | Update a feature flag's state | +LaunchDarkly| [`update_target_feature_flag`](/deno-slack-sdk/reference/connector-functions/launchdarkly/update_target_feature_flag) | Update a target in a feature flag | +LaunchDarkly| [`update_target_segment`](/deno-slack-sdk/reference/connector-functions/launchdarkly/update_target_segment) | Update a target in a segment | +Lever | [`create_interview`](/deno-slack-sdk/reference/connector-functions/lever/create_interview) | Create an interview | +Lever| [`create_opportunity`](/deno-slack-sdk/reference/connector-functions/lever/create_opportunity) | Create an opportunity | +Lever| [`create_panel`](/deno-slack-sdk/reference/connector-functions/lever/create_panel) | Create a panel | +Linear | [`add_comment`](/deno-slack-sdk/reference/connector-functions/linear/add_comment) | Add a comment | +Linear| [`create_issue`](/deno-slack-sdk/reference/connector-functions/linear/create_issue) | Create an issue | +Linear| [`create_project`](/deno-slack-sdk/reference/connector-functions/linear/create_project) | Create a project | +Linear| [`update_issue`](/deno-slack-sdk/reference/connector-functions/linear/update_issue) | Update an issue | +Loopio | [`create_project`](/deno-slack-sdk/reference/connector-functions/loopio/create_project) | Create a project | +Lucid | [`create_document`](/deno-slack-sdk/reference/connector-functions/lucid/create_document) | Create a document | +Mailchimp | [`create_campaign`](/deno-slack-sdk/reference/connector-functions/mailchimp/create_campaign) | Create an email campaign | +Mailchimp| [`add_contact`](/deno-slack-sdk/reference/connector-functions/mailchimp/add_contact) | Add a contact to audience | +Mailchimp| [`get_campaign_report`](/deno-slack-sdk/reference/connector-functions/mailchimp/get_campaign_report) | Get campaign report | +Mailchimp| [`send_campaign`](/deno-slack-sdk/reference/connector-functions/mailchimp/send_campaign) | Send a Campaign | +Microsoft Excel | [`add_worksheet_row`](/deno-slack-sdk/reference/connector-functions/microsoft.excel/add_worksheet_row) | Add to worksheet | +Microsoft Excel | [`delete_worksheet_row`](/deno-slack-sdk/reference/connector-functions/microsoft.excel/delete_worksheet_row) | Delete from a worksheet | +Microsoft Excel | [`select_worksheet_row`](/deno-slack-sdk/reference/connector-functions/microsoft.excel/select_worksheet_row) | Select a worksheet row | +Microsoft Excel | [`update_worksheet_row`](/deno-slack-sdk/reference/connector-functions/microsoft.excel/update_worksheet_row) | Update a worksheet | +Microsoft OneDrive | [`copy_file`](/deno-slack-sdk/reference/connector-functions/microsoft.onedrive/copy_file) | Copy a file | +Microsoft OneDrive | [`create_file`](/deno-slack-sdk/reference/connector-functions/microsoft.onedrive/create_file) | Create a file | +Microsoft OneNote | [`update_page`](/deno-slack-sdk/reference/connector-functions/microsoft.onenote/update_page) | Update a page | +Microsoft OneNote | [`create_page`](/deno-slack-sdk/reference/connector-functions/microsoft.onenote/create_page) | Create a page | +Microsoft Outlook Calendar | [`create_event`](/deno-slack-sdk/reference/connector-functions/microsoft.outlook.calendar/create_event) | Create a calendar event | +Microsoft Outlook Calendar | [`send_email`](/deno-slack-sdk/reference/connector-functions/microsoft.outlook.email/send_email) | Send an email | +Microsoft Teams Calls | [`create_meeting`](/deno-slack-sdk/reference/connector-functions/microsoft.teams/create_meeting) | Create a meeting | +Miro | [`create_board`](/deno-slack-sdk/reference/connector-functions/miro/create_board) | Create board | +Miro| [`copy_board`](/deno-slack-sdk/reference/connector-functions/miro/copy_board) | Copy board | +Monday | [`create_board`](/deno-slack-sdk/reference/connector-functions/monday/create_board) | Create a board | +Monday| [`create_group`](/deno-slack-sdk/reference/connector-functions/monday/create_group) | Create a group | +Monday| [`create_item`](/deno-slack-sdk/reference/connector-functions/monday/create_item) | Create an item | +Monday| [`archive_board`](/deno-slack-sdk/reference/connector-functions/monday/archive_board) | Archive a board | +Notion | [`archive_page`](/deno-slack-sdk/reference/connector-functions/notion/archive_page) | Archive a page | +Notion| [`create_page`](/deno-slack-sdk/reference/connector-functions/notion/create_page) | Create a page | +PagerDuty | [`add_a_note`](/deno-slack-sdk/reference/connector-functions/pagerduty/add_a_note) | Add a note | +PagerDuty | [`create_incident`](/deno-slack-sdk/reference/connector-functions/pagerduty/create_incident) | Trigger an incident | +PagerDuty | [`escalate_incident`](/deno-slack-sdk/reference/connector-functions/pagerduty/escalate_incident) | Change escalation level | +PagerDuty | [`resolve_incident`](/deno-slack-sdk/reference/connector-functions/pagerduty/resolve_incident) | Resolve an incident | +PagerDuty | [`send_status_update`](/deno-slack-sdk/reference/connector-functions/pagerduty/send_status_update) | Create a status update | +Ramp | [`create_physical_card`](/deno-slack-sdk/reference/connector-functions/ramp/create_physical_card) | Create new physical card | +Ramp | [`get_spend_request`](/deno-slack-sdk/reference/connector-functions/ramp/get_spend_request) | Get a spend request | +Ramp | [`create_virtual_card`](/deno-slack-sdk/reference/connector-functions/ramp/create_virtual_card) | Create new virtual card | +Ramp | [`unlock_card`](/deno-slack-sdk/reference/connector-functions/ramp/unlock_card) | Unlock a card | +Ramp | [`terminate_card`](/deno-slack-sdk/reference/connector-functions/ramp/terminate_card) | Terminate a card | +Ramp | [`create_spend_request`](/deno-slack-sdk/reference/connector-functions/ramp/create_spend_request) | Create spend request | +Ramp | [`suspend_card`](/deno-slack-sdk/reference/connector-functions/ramp/suspend_card) | Suspend a card | +RingCentral | [`send_sms`](/deno-slack-sdk/reference/connector-functions/ringcentral/send_sms) | Send a SMS | +Rootly | [`create_cause`](/deno-slack-sdk/reference/connector-functions/rootly/create_cause) | Create a cause | +Rootly | [`create_alert`](/deno-slack-sdk/reference/connector-functions/rootly/create_alert) | Create an alert | +Rootly | [`update_cause`](/deno-slack-sdk/reference/connector-functions/rootly/update_cause) | Update a cause | +Salesforce | [`create_record`](/deno-slack-sdk/reference/connector-functions/salesforce/create_record) | Create a record | +Salesforce| [`delete_record`](/deno-slack-sdk/reference/connector-functions/salesforce/delete_record) | Delete a record | +Salesforce| [`read_record`](/deno-slack-sdk/reference/connector-functions/salesforce/read_record) | Read a record | +Salesforce| [`run_flow`](/deno-slack-sdk/reference/connector-functions/salesforce/run_flow) | Run a Flow | +Salesforce| [`update_record`](/deno-slack-sdk/reference/connector-functions/salesforce/update_record) | Update a record | +ServiceNow | [`create_incident`](/deno-slack-sdk/reference/connector-functions/servicenow/create_incident) | Create an incident | +ServiceNow| [`get_incident`](/deno-slack-sdk/reference/connector-functions/servicenow/get_incident) | Get an incident | +SmartRecruiters | [`create_candidate`](/deno-slack-sdk/reference/connector-functions/smartrecruiters/create_candidate) | Create a candidate | +SmartRecruiters | [`create_candidate_and_assign_to_job`](/deno-slack-sdk/reference/connector-functions/smartrecruiters/create_candidate_and_assign_to_job) | Create a candidate and assign to job | +SmartRecruiters| [`give_candidate_review`](/deno-slack-sdk/reference/connector-functions/smartrecruiters/give_candidate_review) | Provide feedback for a candidate | +Smartsheet | [`add_row`](/deno-slack-sdk/reference/connector-functions/smartsheet/add_row) | Add a row to a Smartsheet | +Smartsheet| [`delete_row`](/deno-slack-sdk/reference/connector-functions/smartsheet/delete_row) | Delete a row from Smartsheet | +Smartsheet| [`select_row`](/deno-slack-sdk/reference/connector-functions/smartsheet/select_row) | Select a Smartsheet row | +Smartsheet| [`update_row`](/deno-slack-sdk/reference/connector-functions/smartsheet/update_row) | Update a row to Smartsheet | +Snyk | [`create_ignore`](/deno-slack-sdk/reference/connector-functions/snyk/create_ignore) | Ignore an issue | +SurveyMonkey | [`copy_survey`](/deno-slack-sdk/reference/connector-functions/surveymonkey/copy_survey) | Copy a survey | +SurveyMonkey| [`copy_survey_from_template`](/deno-slack-sdk/reference/connector-functions/surveymonkey/copy_survey_from_template) | Copy a survey from a template | +Travis CI | [`restart_build`](/deno-slack-sdk/reference/connector-functions/travisci/restart_build) | Restart build | +Travis CI | [`trigger_build`](/deno-slack-sdk/reference/connector-functions/travisci/trigger_build) | Trigger build | +Travis CI | [`cancel_build`](/deno-slack-sdk/reference/connector-functions/travisci/cancel_build) | Cancel build | +Twilio | [`send_sms`](/deno-slack-sdk/reference/connector-functions/twilio/send_sms) | Send SMS | +Typeform | [`duplicate_form`](/deno-slack-sdk/reference/connector-functions/typeform/duplicate_form) | Duplicate an existing form | +Typeform | [`create_workspace`](/deno-slack-sdk/reference/connector-functions/typeform/create_workspace) | Create a workspace | +Typeform | [`get_form`](/deno-slack-sdk/reference/connector-functions/typeform/get_form) | Get a form | +Typeform | [`get_form_insights`](/deno-slack-sdk/reference/connector-functions/typeform/get_form_insights) | Get form insights | +Webex | [`create_meeting`](/deno-slack-sdk/reference/connector-functions/webex/create_meeting) | Create a meeting | +Workast | [`create_task`](/deno-slack-sdk/reference/connector-functions/workast/create_task) | Create a task | +Wrike | [`comment_on_a_folder`](/deno-slack-sdk/reference/connector-functions/wrike/comment_on_a_folder) | Comment on a folder | +Wrike| [`comment_on_a_task`](/deno-slack-sdk/reference/connector-functions/wrike/comment_on_a_task) | Comment on a task | +Wrike| [`create_a_folder`](/deno-slack-sdk/reference/connector-functions/wrike/create_a_folder) | Create a folder | +Wrike| [`create_a_task`](/deno-slack-sdk/reference/connector-functions/wrike/create_a_task) | Create a task | +Wrike| [`update_a_task`](/deno-slack-sdk/reference/connector-functions/wrike/update_a_task) | Update a task | +Zendesk | [`add_tags`](/deno-slack-sdk/reference/connector-functions/zendesk/add_tags) | Add tags | +Zendesk| [`create_ticket`](/deno-slack-sdk/reference/connector-functions/zendesk/create_ticket) | Create a ticket | +Zendesk| [`update_ticket`](/deno-slack-sdk/reference/connector-functions/zendesk/update_ticket) | Update a ticket | +Zoom | [`create_meeting`](/deno-slack-sdk/reference/connector-functions/zoom/create_meeting) | Create a meeting | \ No newline at end of file diff --git a/docs/reference/connector-functions/adobe.sign/send_agreement.md b/docs/reference/connector-functions/adobe.sign/send_agreement.md new file mode 100644 index 00000000..427cba20 --- /dev/null +++ b/docs/reference/connector-functions/adobe.sign/send_agreement.md @@ -0,0 +1,13 @@ + +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/airtable/add_record.md b/docs/reference/connector-functions/airtable/add_record.md new file mode 100644 index 00000000..92e0ba4a --- /dev/null +++ b/docs/reference/connector-functions/airtable/add_record.md @@ -0,0 +1,3 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/airtable/delete_record.md b/docs/reference/connector-functions/airtable/delete_record.md new file mode 100644 index 00000000..3e98ddef --- /dev/null +++ b/docs/reference/connector-functions/airtable/delete_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/airtable/select_record.md b/docs/reference/connector-functions/airtable/select_record.md new file mode 100644 index 00000000..fba492fb --- /dev/null +++ b/docs/reference/connector-functions/airtable/select_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/airtable/update_record.md b/docs/reference/connector-functions/airtable/update_record.md new file mode 100644 index 00000000..894b0045 --- /dev/null +++ b/docs/reference/connector-functions/airtable/update_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/asana/add_task_to_section.md b/docs/reference/connector-functions/asana/add_task_to_section.md new file mode 100644 index 00000000..3ea9b586 --- /dev/null +++ b/docs/reference/connector-functions/asana/add_task_to_section.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/asana/create_comment.md b/docs/reference/connector-functions/asana/create_comment.md new file mode 100644 index 00000000..9492d3bb --- /dev/null +++ b/docs/reference/connector-functions/asana/create_comment.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/asana/create_project.md b/docs/reference/connector-functions/asana/create_project.md new file mode 100644 index 00000000..2498cf46 --- /dev/null +++ b/docs/reference/connector-functions/asana/create_project.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/asana/create_task.md b/docs/reference/connector-functions/asana/create_task.md new file mode 100644 index 00000000..f2f173f4 --- /dev/null +++ b/docs/reference/connector-functions/asana/create_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/asana/update_task.md b/docs/reference/connector-functions/asana/update_task.md new file mode 100644 index 00000000..74d97bbb --- /dev/null +++ b/docs/reference/connector-functions/asana/update_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/atlassian.bitbucket/create_issue.md b/docs/reference/connector-functions/atlassian.bitbucket/create_issue.md new file mode 100644 index 00000000..e08d18e5 --- /dev/null +++ b/docs/reference/connector-functions/atlassian.bitbucket/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/atlassian.bitbucket/merge_pull_request.md b/docs/reference/connector-functions/atlassian.bitbucket/merge_pull_request.md new file mode 100644 index 00000000..2a1ead3d --- /dev/null +++ b/docs/reference/connector-functions/atlassian.bitbucket/merge_pull_request.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/basecamp/create_project.md b/docs/reference/connector-functions/basecamp/create_project.md new file mode 100644 index 00000000..4ba0d488 --- /dev/null +++ b/docs/reference/connector-functions/basecamp/create_project.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/basecamp/create_todo.mdx b/docs/reference/connector-functions/basecamp/create_todo.mdx new file mode 100644 index 00000000..496b51f8 --- /dev/null +++ b/docs/reference/connector-functions/basecamp/create_todo.mdx @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/basecamp/create_todo_list.md b/docs/reference/connector-functions/basecamp/create_todo_list.md new file mode 100644 index 00000000..4757cc66 --- /dev/null +++ b/docs/reference/connector-functions/basecamp/create_todo_list.md @@ -0,0 +1,11 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/basecamp/mark_todo_complete.md b/docs/reference/connector-functions/basecamp/mark_todo_complete.md new file mode 100644 index 00000000..8d6b86f6 --- /dev/null +++ b/docs/reference/connector-functions/basecamp/mark_todo_complete.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/basecamp/mark_todo_pending.md b/docs/reference/connector-functions/basecamp/mark_todo_pending.md new file mode 100644 index 00000000..6ca2b06a --- /dev/null +++ b/docs/reference/connector-functions/basecamp/mark_todo_pending.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/box.core/copy_file.md b/docs/reference/connector-functions/box.core/copy_file.md new file mode 100644 index 00000000..58d3054e --- /dev/null +++ b/docs/reference/connector-functions/box.core/copy_file.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/box.core/create_folder.md b/docs/reference/connector-functions/box.core/create_folder.md new file mode 100644 index 00000000..891fd434 --- /dev/null +++ b/docs/reference/connector-functions/box.core/create_folder.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/box.sign/create_sign_request.md b/docs/reference/connector-functions/box.sign/create_sign_request.md new file mode 100644 index 00000000..b038ecd1 --- /dev/null +++ b/docs/reference/connector-functions/box.sign/create_sign_request.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/calendly/get_meeting_link.md b/docs/reference/connector-functions/calendly/get_meeting_link.md new file mode 100644 index 00000000..d6380e0d --- /dev/null +++ b/docs/reference/connector-functions/calendly/get_meeting_link.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/clickup/create_task.md b/docs/reference/connector-functions/clickup/create_task.md new file mode 100644 index 00000000..cb32bea0 --- /dev/null +++ b/docs/reference/connector-functions/clickup/create_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/deel/add_time_off_request.md b/docs/reference/connector-functions/deel/add_time_off_request.md new file mode 100644 index 00000000..8b54d7d1 --- /dev/null +++ b/docs/reference/connector-functions/deel/add_time_off_request.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/deel/create_contract.md b/docs/reference/connector-functions/deel/create_contract.md new file mode 100644 index 00000000..20b7610d --- /dev/null +++ b/docs/reference/connector-functions/deel/create_contract.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dialpad/send_sms.md b/docs/reference/connector-functions/dialpad/send_sms.md new file mode 100644 index 00000000..669914c5 --- /dev/null +++ b/docs/reference/connector-functions/dialpad/send_sms.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/docusign/create_envelope.md b/docs/reference/connector-functions/docusign/create_envelope.md new file mode 100644 index 00000000..07819025 --- /dev/null +++ b/docs/reference/connector-functions/docusign/create_envelope.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/docusign/send_envelope.md b/docs/reference/connector-functions/docusign/send_envelope.md new file mode 100644 index 00000000..97bfe90b --- /dev/null +++ b/docs/reference/connector-functions/docusign/send_envelope.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.core/copy_document.md b/docs/reference/connector-functions/dropbox.core/copy_document.md new file mode 100644 index 00000000..7477a250 --- /dev/null +++ b/docs/reference/connector-functions/dropbox.core/copy_document.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.core/create_folder.md b/docs/reference/connector-functions/dropbox.core/create_folder.md new file mode 100644 index 00000000..2a68937c --- /dev/null +++ b/docs/reference/connector-functions/dropbox.core/create_folder.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.core/create_share_link.md b/docs/reference/connector-functions/dropbox.core/create_share_link.md new file mode 100644 index 00000000..97e80213 --- /dev/null +++ b/docs/reference/connector-functions/dropbox.core/create_share_link.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.core/delete_document.md b/docs/reference/connector-functions/dropbox.core/delete_document.md new file mode 100644 index 00000000..2943999c --- /dev/null +++ b/docs/reference/connector-functions/dropbox.core/delete_document.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.core/move_document.md b/docs/reference/connector-functions/dropbox.core/move_document.md new file mode 100644 index 00000000..9285e455 --- /dev/null +++ b/docs/reference/connector-functions/dropbox.core/move_document.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/dropbox.sign/send_signature_request_with_template.md b/docs/reference/connector-functions/dropbox.sign/send_signature_request_with_template.md new file mode 100644 index 00000000..1dcb5bb4 --- /dev/null +++ b/docs/reference/connector-functions/dropbox.sign/send_signature_request_with_template.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/firehydrant/create_incident.md b/docs/reference/connector-functions/firehydrant/create_incident.md new file mode 100644 index 00000000..33bbe84d --- /dev/null +++ b/docs/reference/connector-functions/firehydrant/create_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/firehydrant/create_task.md b/docs/reference/connector-functions/firehydrant/create_task.md new file mode 100644 index 00000000..53d09f8d --- /dev/null +++ b/docs/reference/connector-functions/firehydrant/create_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/firehydrant/update_incident.md b/docs/reference/connector-functions/firehydrant/update_incident.md new file mode 100644 index 00000000..8fc1a9dd --- /dev/null +++ b/docs/reference/connector-functions/firehydrant/update_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/firehydrant/update_task.md b/docs/reference/connector-functions/firehydrant/update_task.md new file mode 100644 index 00000000..6a568a08 --- /dev/null +++ b/docs/reference/connector-functions/firehydrant/update_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/giphy/get_random_gif.md b/docs/reference/connector-functions/giphy/get_random_gif.md new file mode 100644 index 00000000..c04917b0 --- /dev/null +++ b/docs/reference/connector-functions/giphy/get_random_gif.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/giphy/get_translated_gif.md b/docs/reference/connector-functions/giphy/get_translated_gif.md new file mode 100644 index 00000000..873e4410 --- /dev/null +++ b/docs/reference/connector-functions/giphy/get_translated_gif.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/github.cloud/create_issue.md b/docs/reference/connector-functions/github.cloud/create_issue.md new file mode 100644 index 00000000..b8609e6b --- /dev/null +++ b/docs/reference/connector-functions/github.cloud/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/github.enterprise_server/create_issue.md b/docs/reference/connector-functions/github.enterprise_server/create_issue.md new file mode 100644 index 00000000..bd54582e --- /dev/null +++ b/docs/reference/connector-functions/github.enterprise_server/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/gitlab/create_issue.md b/docs/reference/connector-functions/gitlab/create_issue.md new file mode 100644 index 00000000..b1fec954 --- /dev/null +++ b/docs/reference/connector-functions/gitlab/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.calendar/add_to_event.md b/docs/reference/connector-functions/google.calendar/add_to_event.md new file mode 100644 index 00000000..404058e0 --- /dev/null +++ b/docs/reference/connector-functions/google.calendar/add_to_event.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.calendar/create_event.md b/docs/reference/connector-functions/google.calendar/create_event.md new file mode 100644 index 00000000..5768836c --- /dev/null +++ b/docs/reference/connector-functions/google.calendar/create_event.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.mail/send_email.md b/docs/reference/connector-functions/google.mail/send_email.md new file mode 100644 index 00000000..e37b1709 --- /dev/null +++ b/docs/reference/connector-functions/google.mail/send_email.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.meet/start_meeting.md b/docs/reference/connector-functions/google.meet/start_meeting.md new file mode 100644 index 00000000..a06dbb1a --- /dev/null +++ b/docs/reference/connector-functions/google.meet/start_meeting.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.sheets/add_spreadsheet_row.md b/docs/reference/connector-functions/google.sheets/add_spreadsheet_row.md new file mode 100644 index 00000000..ad79baec --- /dev/null +++ b/docs/reference/connector-functions/google.sheets/add_spreadsheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.sheets/delete_spreadsheet_row.md b/docs/reference/connector-functions/google.sheets/delete_spreadsheet_row.md new file mode 100644 index 00000000..1a682958 --- /dev/null +++ b/docs/reference/connector-functions/google.sheets/delete_spreadsheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.sheets/select_spreadsheet_row.md b/docs/reference/connector-functions/google.sheets/select_spreadsheet_row.md new file mode 100644 index 00000000..3312e6c5 --- /dev/null +++ b/docs/reference/connector-functions/google.sheets/select_spreadsheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.sheets/update_spreadsheet_row.md b/docs/reference/connector-functions/google.sheets/update_spreadsheet_row.md new file mode 100644 index 00000000..d4c3a21d --- /dev/null +++ b/docs/reference/connector-functions/google.sheets/update_spreadsheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.tasks/create_tasklist.md b/docs/reference/connector-functions/google.tasks/create_tasklist.md new file mode 100644 index 00000000..43b856b1 --- /dev/null +++ b/docs/reference/connector-functions/google.tasks/create_tasklist.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/google.tasks/insert_task.md b/docs/reference/connector-functions/google.tasks/insert_task.md new file mode 100644 index 00000000..bf0f4207 --- /dev/null +++ b/docs/reference/connector-functions/google.tasks/insert_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.onboarding/create_pending_hire.md b/docs/reference/connector-functions/greenhouse.onboarding/create_pending_hire.md new file mode 100644 index 00000000..feb5d2ad --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.onboarding/create_pending_hire.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.onboarding/fetch_employees.md b/docs/reference/connector-functions/greenhouse.onboarding/fetch_employees.md new file mode 100644 index 00000000..5e19a08a --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.onboarding/fetch_employees.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.recruiting/hire_application.md b/docs/reference/connector-functions/greenhouse.recruiting/hire_application.md new file mode 100644 index 00000000..6f68075e --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.recruiting/hire_application.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.recruiting/list_candidate_activity.md b/docs/reference/connector-functions/greenhouse.recruiting/list_candidate_activity.md new file mode 100644 index 00000000..c691bb12 --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.recruiting/list_candidate_activity.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.recruiting/list_job_candidates.md b/docs/reference/connector-functions/greenhouse.recruiting/list_job_candidates.md new file mode 100644 index 00000000..9e4d7c7c --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.recruiting/list_job_candidates.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/greenhouse.recruiting/reject_application.md b/docs/reference/connector-functions/greenhouse.recruiting/reject_application.md new file mode 100644 index 00000000..bf9215b4 --- /dev/null +++ b/docs/reference/connector-functions/greenhouse.recruiting/reject_application.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/add_comment.md b/docs/reference/connector-functions/guru/add_comment.md new file mode 100644 index 00000000..60658fd7 --- /dev/null +++ b/docs/reference/connector-functions/guru/add_comment.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/create_card.md b/docs/reference/connector-functions/guru/create_card.md new file mode 100644 index 00000000..39ddf9b9 --- /dev/null +++ b/docs/reference/connector-functions/guru/create_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/delete_card.md b/docs/reference/connector-functions/guru/delete_card.md new file mode 100644 index 00000000..95b14029 --- /dev/null +++ b/docs/reference/connector-functions/guru/delete_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/unverify_card.md b/docs/reference/connector-functions/guru/unverify_card.md new file mode 100644 index 00000000..55e4b1f8 --- /dev/null +++ b/docs/reference/connector-functions/guru/unverify_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/update_card.md b/docs/reference/connector-functions/guru/update_card.md new file mode 100644 index 00000000..87961867 --- /dev/null +++ b/docs/reference/connector-functions/guru/update_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/guru/verify_card.md b/docs/reference/connector-functions/guru/verify_card.md new file mode 100644 index 00000000..7f2a42b5 --- /dev/null +++ b/docs/reference/connector-functions/guru/verify_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/intercom/create_article.md b/docs/reference/connector-functions/intercom/create_article.md new file mode 100644 index 00000000..9dbd983f --- /dev/null +++ b/docs/reference/connector-functions/intercom/create_article.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/intercom/create_ticket.md b/docs/reference/connector-functions/intercom/create_ticket.md new file mode 100644 index 00000000..f539789e --- /dev/null +++ b/docs/reference/connector-functions/intercom/create_ticket.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/jira.cloud/create_issue.md b/docs/reference/connector-functions/jira.cloud/create_issue.md new file mode 100644 index 00000000..492fa1bc --- /dev/null +++ b/docs/reference/connector-functions/jira.cloud/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/jira.cloud/edit_issue.md b/docs/reference/connector-functions/jira.cloud/edit_issue.md new file mode 100644 index 00000000..d99cb4a4 --- /dev/null +++ b/docs/reference/connector-functions/jira.cloud/edit_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/launchdarkly/create_approval_request_update_flag.md b/docs/reference/connector-functions/launchdarkly/create_approval_request_update_flag.md new file mode 100644 index 00000000..92213c9f --- /dev/null +++ b/docs/reference/connector-functions/launchdarkly/create_approval_request_update_flag.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/launchdarkly/create_feature_flag.md b/docs/reference/connector-functions/launchdarkly/create_feature_flag.md new file mode 100644 index 00000000..c101634d --- /dev/null +++ b/docs/reference/connector-functions/launchdarkly/create_feature_flag.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/launchdarkly/update_feature_flag_state.md b/docs/reference/connector-functions/launchdarkly/update_feature_flag_state.md new file mode 100644 index 00000000..08ce72e3 --- /dev/null +++ b/docs/reference/connector-functions/launchdarkly/update_feature_flag_state.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/launchdarkly/update_target_feature_flag.md b/docs/reference/connector-functions/launchdarkly/update_target_feature_flag.md new file mode 100644 index 00000000..3b63af04 --- /dev/null +++ b/docs/reference/connector-functions/launchdarkly/update_target_feature_flag.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/launchdarkly/update_target_segment.md b/docs/reference/connector-functions/launchdarkly/update_target_segment.md new file mode 100644 index 00000000..d9aa1dbc --- /dev/null +++ b/docs/reference/connector-functions/launchdarkly/update_target_segment.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/lever/create_interview.md b/docs/reference/connector-functions/lever/create_interview.md new file mode 100644 index 00000000..731f9995 --- /dev/null +++ b/docs/reference/connector-functions/lever/create_interview.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/lever/create_opportunity.md b/docs/reference/connector-functions/lever/create_opportunity.md new file mode 100644 index 00000000..290b8354 --- /dev/null +++ b/docs/reference/connector-functions/lever/create_opportunity.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/lever/create_panel.md b/docs/reference/connector-functions/lever/create_panel.md new file mode 100644 index 00000000..40fb2317 --- /dev/null +++ b/docs/reference/connector-functions/lever/create_panel.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/linear/add_comment.md b/docs/reference/connector-functions/linear/add_comment.md new file mode 100644 index 00000000..ba4a9ec0 --- /dev/null +++ b/docs/reference/connector-functions/linear/add_comment.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/linear/create_issue.md b/docs/reference/connector-functions/linear/create_issue.md new file mode 100644 index 00000000..ad3eef0a --- /dev/null +++ b/docs/reference/connector-functions/linear/create_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/linear/create_project.md b/docs/reference/connector-functions/linear/create_project.md new file mode 100644 index 00000000..4d4600ed --- /dev/null +++ b/docs/reference/connector-functions/linear/create_project.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/linear/update_issue.md b/docs/reference/connector-functions/linear/update_issue.md new file mode 100644 index 00000000..5d03fe11 --- /dev/null +++ b/docs/reference/connector-functions/linear/update_issue.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/loopio/create_project.md b/docs/reference/connector-functions/loopio/create_project.md new file mode 100644 index 00000000..22da9060 --- /dev/null +++ b/docs/reference/connector-functions/loopio/create_project.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/lucid/create_document.md b/docs/reference/connector-functions/lucid/create_document.md new file mode 100644 index 00000000..4ec337b0 --- /dev/null +++ b/docs/reference/connector-functions/lucid/create_document.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/mailchimp/add_contact.md b/docs/reference/connector-functions/mailchimp/add_contact.md new file mode 100644 index 00000000..0768f6d2 --- /dev/null +++ b/docs/reference/connector-functions/mailchimp/add_contact.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/mailchimp/create_campaign.md b/docs/reference/connector-functions/mailchimp/create_campaign.md new file mode 100644 index 00000000..7d5b73c6 --- /dev/null +++ b/docs/reference/connector-functions/mailchimp/create_campaign.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/mailchimp/get_campaign_report.md b/docs/reference/connector-functions/mailchimp/get_campaign_report.md new file mode 100644 index 00000000..e64e6186 --- /dev/null +++ b/docs/reference/connector-functions/mailchimp/get_campaign_report.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/mailchimp/send_campaign.md b/docs/reference/connector-functions/mailchimp/send_campaign.md new file mode 100644 index 00000000..81a128ce --- /dev/null +++ b/docs/reference/connector-functions/mailchimp/send_campaign.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.excel/add_worksheet_row.md b/docs/reference/connector-functions/microsoft.excel/add_worksheet_row.md new file mode 100644 index 00000000..d20b4027 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.excel/add_worksheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.excel/delete_worksheet_row.md b/docs/reference/connector-functions/microsoft.excel/delete_worksheet_row.md new file mode 100644 index 00000000..51d5a8de --- /dev/null +++ b/docs/reference/connector-functions/microsoft.excel/delete_worksheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.excel/select_worksheet_row.md b/docs/reference/connector-functions/microsoft.excel/select_worksheet_row.md new file mode 100644 index 00000000..054e0fbc --- /dev/null +++ b/docs/reference/connector-functions/microsoft.excel/select_worksheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.excel/update_worksheet_row.md b/docs/reference/connector-functions/microsoft.excel/update_worksheet_row.md new file mode 100644 index 00000000..e572d9b5 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.excel/update_worksheet_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.onedrive/copy_file.md b/docs/reference/connector-functions/microsoft.onedrive/copy_file.md new file mode 100644 index 00000000..101cef8a --- /dev/null +++ b/docs/reference/connector-functions/microsoft.onedrive/copy_file.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.onedrive/create_file.md b/docs/reference/connector-functions/microsoft.onedrive/create_file.md new file mode 100644 index 00000000..e6a15a87 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.onedrive/create_file.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.onenote/create_page.md b/docs/reference/connector-functions/microsoft.onenote/create_page.md new file mode 100644 index 00000000..ed5413ff --- /dev/null +++ b/docs/reference/connector-functions/microsoft.onenote/create_page.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.onenote/update_page.md b/docs/reference/connector-functions/microsoft.onenote/update_page.md new file mode 100644 index 00000000..8f27c178 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.onenote/update_page.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.outlook.calendar/create_event.md b/docs/reference/connector-functions/microsoft.outlook.calendar/create_event.md new file mode 100644 index 00000000..81636018 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.outlook.calendar/create_event.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.outlook.email/send_email.md b/docs/reference/connector-functions/microsoft.outlook.email/send_email.md new file mode 100644 index 00000000..cae04dfe --- /dev/null +++ b/docs/reference/connector-functions/microsoft.outlook.email/send_email.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/microsoft.teams/create_meeting.md b/docs/reference/connector-functions/microsoft.teams/create_meeting.md new file mode 100644 index 00000000..4e1ad445 --- /dev/null +++ b/docs/reference/connector-functions/microsoft.teams/create_meeting.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/miro/copy_board.md b/docs/reference/connector-functions/miro/copy_board.md new file mode 100644 index 00000000..3733900f --- /dev/null +++ b/docs/reference/connector-functions/miro/copy_board.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/miro/create_board.md b/docs/reference/connector-functions/miro/create_board.md new file mode 100644 index 00000000..332259d8 --- /dev/null +++ b/docs/reference/connector-functions/miro/create_board.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/monday/archive_board.md b/docs/reference/connector-functions/monday/archive_board.md new file mode 100644 index 00000000..b3c9b7db --- /dev/null +++ b/docs/reference/connector-functions/monday/archive_board.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/monday/create_board.md b/docs/reference/connector-functions/monday/create_board.md new file mode 100644 index 00000000..15ff11b6 --- /dev/null +++ b/docs/reference/connector-functions/monday/create_board.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/monday/create_group.md b/docs/reference/connector-functions/monday/create_group.md new file mode 100644 index 00000000..d63bf126 --- /dev/null +++ b/docs/reference/connector-functions/monday/create_group.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/monday/create_item.md b/docs/reference/connector-functions/monday/create_item.md new file mode 100644 index 00000000..6a1bb9c6 --- /dev/null +++ b/docs/reference/connector-functions/monday/create_item.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/notion/archive_page.md b/docs/reference/connector-functions/notion/archive_page.md new file mode 100644 index 00000000..294b9ecf --- /dev/null +++ b/docs/reference/connector-functions/notion/archive_page.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/notion/create_page.md b/docs/reference/connector-functions/notion/create_page.md new file mode 100644 index 00000000..7a190d31 --- /dev/null +++ b/docs/reference/connector-functions/notion/create_page.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/pagerduty/add_a_note.md b/docs/reference/connector-functions/pagerduty/add_a_note.md new file mode 100644 index 00000000..20b947ca --- /dev/null +++ b/docs/reference/connector-functions/pagerduty/add_a_note.md @@ -0,0 +1,13 @@ + +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/pagerduty/create_incident.md b/docs/reference/connector-functions/pagerduty/create_incident.md new file mode 100644 index 00000000..ab5f4c9b --- /dev/null +++ b/docs/reference/connector-functions/pagerduty/create_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/pagerduty/escalate_incident.md b/docs/reference/connector-functions/pagerduty/escalate_incident.md new file mode 100644 index 00000000..b84a36d2 --- /dev/null +++ b/docs/reference/connector-functions/pagerduty/escalate_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/pagerduty/resolve_incident.md b/docs/reference/connector-functions/pagerduty/resolve_incident.md new file mode 100644 index 00000000..f8df8f71 --- /dev/null +++ b/docs/reference/connector-functions/pagerduty/resolve_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/pagerduty/send_status_update.md b/docs/reference/connector-functions/pagerduty/send_status_update.md new file mode 100644 index 00000000..6570b973 --- /dev/null +++ b/docs/reference/connector-functions/pagerduty/send_status_update.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/create_physical_card.md b/docs/reference/connector-functions/ramp/create_physical_card.md new file mode 100644 index 00000000..d67ce91a --- /dev/null +++ b/docs/reference/connector-functions/ramp/create_physical_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/create_spend_request.md b/docs/reference/connector-functions/ramp/create_spend_request.md new file mode 100644 index 00000000..dd25d474 --- /dev/null +++ b/docs/reference/connector-functions/ramp/create_spend_request.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/create_virtual_card.md b/docs/reference/connector-functions/ramp/create_virtual_card.md new file mode 100644 index 00000000..5d6a5439 --- /dev/null +++ b/docs/reference/connector-functions/ramp/create_virtual_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/get_spend_request.md b/docs/reference/connector-functions/ramp/get_spend_request.md new file mode 100644 index 00000000..f83d7508 --- /dev/null +++ b/docs/reference/connector-functions/ramp/get_spend_request.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/suspend_card.md b/docs/reference/connector-functions/ramp/suspend_card.md new file mode 100644 index 00000000..95106eaf --- /dev/null +++ b/docs/reference/connector-functions/ramp/suspend_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/terminate_card.md b/docs/reference/connector-functions/ramp/terminate_card.md new file mode 100644 index 00000000..0596c0c1 --- /dev/null +++ b/docs/reference/connector-functions/ramp/terminate_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ramp/unlock_card.md b/docs/reference/connector-functions/ramp/unlock_card.md new file mode 100644 index 00000000..20fb815e --- /dev/null +++ b/docs/reference/connector-functions/ramp/unlock_card.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/ringcentral/send_sms.md b/docs/reference/connector-functions/ringcentral/send_sms.md new file mode 100644 index 00000000..3e4ed975 --- /dev/null +++ b/docs/reference/connector-functions/ringcentral/send_sms.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/rootly/create_alert.md b/docs/reference/connector-functions/rootly/create_alert.md new file mode 100644 index 00000000..280466cc --- /dev/null +++ b/docs/reference/connector-functions/rootly/create_alert.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/rootly/create_cause.md b/docs/reference/connector-functions/rootly/create_cause.md new file mode 100644 index 00000000..00d66182 --- /dev/null +++ b/docs/reference/connector-functions/rootly/create_cause.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/rootly/update_cause.md b/docs/reference/connector-functions/rootly/update_cause.md new file mode 100644 index 00000000..638d928b --- /dev/null +++ b/docs/reference/connector-functions/rootly/update_cause.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/salesforce/create_record.md b/docs/reference/connector-functions/salesforce/create_record.md new file mode 100644 index 00000000..b9169fa1 --- /dev/null +++ b/docs/reference/connector-functions/salesforce/create_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/salesforce/delete_record.md b/docs/reference/connector-functions/salesforce/delete_record.md new file mode 100644 index 00000000..e44fee0c --- /dev/null +++ b/docs/reference/connector-functions/salesforce/delete_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/salesforce/read_record.md b/docs/reference/connector-functions/salesforce/read_record.md new file mode 100644 index 00000000..36b37299 --- /dev/null +++ b/docs/reference/connector-functions/salesforce/read_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/salesforce/run_flow.md b/docs/reference/connector-functions/salesforce/run_flow.md new file mode 100644 index 00000000..32fb1d13 --- /dev/null +++ b/docs/reference/connector-functions/salesforce/run_flow.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/salesforce/update_record.md b/docs/reference/connector-functions/salesforce/update_record.md new file mode 100644 index 00000000..64ab1937 --- /dev/null +++ b/docs/reference/connector-functions/salesforce/update_record.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/servicenow/create_incident.md b/docs/reference/connector-functions/servicenow/create_incident.md new file mode 100644 index 00000000..711ea7de --- /dev/null +++ b/docs/reference/connector-functions/servicenow/create_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/servicenow/get_incident.md b/docs/reference/connector-functions/servicenow/get_incident.md new file mode 100644 index 00000000..ca869a5e --- /dev/null +++ b/docs/reference/connector-functions/servicenow/get_incident.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartrecruiters/create_candidate.md b/docs/reference/connector-functions/smartrecruiters/create_candidate.md new file mode 100644 index 00000000..61f2eaa5 --- /dev/null +++ b/docs/reference/connector-functions/smartrecruiters/create_candidate.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartrecruiters/create_candidate_and_assign_to_job.md b/docs/reference/connector-functions/smartrecruiters/create_candidate_and_assign_to_job.md new file mode 100644 index 00000000..ddedb77b --- /dev/null +++ b/docs/reference/connector-functions/smartrecruiters/create_candidate_and_assign_to_job.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartrecruiters/give_candidate_review.md b/docs/reference/connector-functions/smartrecruiters/give_candidate_review.md new file mode 100644 index 00000000..d0c23352 --- /dev/null +++ b/docs/reference/connector-functions/smartrecruiters/give_candidate_review.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartsheet/add_row.md b/docs/reference/connector-functions/smartsheet/add_row.md new file mode 100644 index 00000000..c6df863c --- /dev/null +++ b/docs/reference/connector-functions/smartsheet/add_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartsheet/delete_row.md b/docs/reference/connector-functions/smartsheet/delete_row.md new file mode 100644 index 00000000..5c58f5bd --- /dev/null +++ b/docs/reference/connector-functions/smartsheet/delete_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartsheet/select_row.md b/docs/reference/connector-functions/smartsheet/select_row.md new file mode 100644 index 00000000..c31db305 --- /dev/null +++ b/docs/reference/connector-functions/smartsheet/select_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/smartsheet/update_row.md b/docs/reference/connector-functions/smartsheet/update_row.md new file mode 100644 index 00000000..714bb6e5 --- /dev/null +++ b/docs/reference/connector-functions/smartsheet/update_row.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/snyk/create_ignore.md b/docs/reference/connector-functions/snyk/create_ignore.md new file mode 100644 index 00000000..8298c5ed --- /dev/null +++ b/docs/reference/connector-functions/snyk/create_ignore.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/surveymonkey/copy_survey.md b/docs/reference/connector-functions/surveymonkey/copy_survey.md new file mode 100644 index 00000000..ea2b420b --- /dev/null +++ b/docs/reference/connector-functions/surveymonkey/copy_survey.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/surveymonkey/copy_survey_from_template.md b/docs/reference/connector-functions/surveymonkey/copy_survey_from_template.md new file mode 100644 index 00000000..87a824ad --- /dev/null +++ b/docs/reference/connector-functions/surveymonkey/copy_survey_from_template.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/travisci/cancel_build.md b/docs/reference/connector-functions/travisci/cancel_build.md new file mode 100644 index 00000000..9350fe91 --- /dev/null +++ b/docs/reference/connector-functions/travisci/cancel_build.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/travisci/restart_build.md b/docs/reference/connector-functions/travisci/restart_build.md new file mode 100644 index 00000000..dd8f7651 --- /dev/null +++ b/docs/reference/connector-functions/travisci/restart_build.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/travisci/trigger_build.md b/docs/reference/connector-functions/travisci/trigger_build.md new file mode 100644 index 00000000..f76338a8 --- /dev/null +++ b/docs/reference/connector-functions/travisci/trigger_build.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/twilio/send_sms.md b/docs/reference/connector-functions/twilio/send_sms.md new file mode 100644 index 00000000..f8449480 --- /dev/null +++ b/docs/reference/connector-functions/twilio/send_sms.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/typeform/create_workspace.md b/docs/reference/connector-functions/typeform/create_workspace.md new file mode 100644 index 00000000..f14b0055 --- /dev/null +++ b/docs/reference/connector-functions/typeform/create_workspace.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/typeform/duplicate_form.md b/docs/reference/connector-functions/typeform/duplicate_form.md new file mode 100644 index 00000000..8d58b6c7 --- /dev/null +++ b/docs/reference/connector-functions/typeform/duplicate_form.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/typeform/get_form.md b/docs/reference/connector-functions/typeform/get_form.md new file mode 100644 index 00000000..8a5f961d --- /dev/null +++ b/docs/reference/connector-functions/typeform/get_form.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/typeform/get_form_insights.md b/docs/reference/connector-functions/typeform/get_form_insights.md new file mode 100644 index 00000000..e5b9e656 --- /dev/null +++ b/docs/reference/connector-functions/typeform/get_form_insights.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/webex/create_meeting.md b/docs/reference/connector-functions/webex/create_meeting.md new file mode 100644 index 00000000..d299f08d --- /dev/null +++ b/docs/reference/connector-functions/webex/create_meeting.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/workast/create_task.md b/docs/reference/connector-functions/workast/create_task.md new file mode 100644 index 00000000..8198b476 --- /dev/null +++ b/docs/reference/connector-functions/workast/create_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/wrike/comment_on_a_folder.md b/docs/reference/connector-functions/wrike/comment_on_a_folder.md new file mode 100644 index 00000000..d05d8dbd --- /dev/null +++ b/docs/reference/connector-functions/wrike/comment_on_a_folder.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/wrike/comment_on_a_task.md b/docs/reference/connector-functions/wrike/comment_on_a_task.md new file mode 100644 index 00000000..b276e2e5 --- /dev/null +++ b/docs/reference/connector-functions/wrike/comment_on_a_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/wrike/create_a_folder.md b/docs/reference/connector-functions/wrike/create_a_folder.md new file mode 100644 index 00000000..ea3f5ca9 --- /dev/null +++ b/docs/reference/connector-functions/wrike/create_a_folder.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/wrike/create_a_task.md b/docs/reference/connector-functions/wrike/create_a_task.md new file mode 100644 index 00000000..9eb46d18 --- /dev/null +++ b/docs/reference/connector-functions/wrike/create_a_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/wrike/update_a_task.md b/docs/reference/connector-functions/wrike/update_a_task.md new file mode 100644 index 00000000..39cde780 --- /dev/null +++ b/docs/reference/connector-functions/wrike/update_a_task.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/zendesk/add_tags.md b/docs/reference/connector-functions/zendesk/add_tags.md new file mode 100644 index 00000000..7ea89c94 --- /dev/null +++ b/docs/reference/connector-functions/zendesk/add_tags.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/zendesk/create_ticket.md b/docs/reference/connector-functions/zendesk/create_ticket.md new file mode 100644 index 00000000..448bd53f --- /dev/null +++ b/docs/reference/connector-functions/zendesk/create_ticket.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/zendesk/remove_tags.md b/docs/reference/connector-functions/zendesk/remove_tags.md new file mode 100644 index 00000000..56157ee9 --- /dev/null +++ b/docs/reference/connector-functions/zendesk/remove_tags.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/zendesk/update_ticket.md b/docs/reference/connector-functions/zendesk/update_ticket.md new file mode 100644 index 00000000..5d7f0fc9 --- /dev/null +++ b/docs/reference/connector-functions/zendesk/update_ticket.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/connector-functions/zoom/create_meeting.md b/docs/reference/connector-functions/zoom/create_meeting.md new file mode 100644 index 00000000..3bb380b7 --- /dev/null +++ b/docs/reference/connector-functions/zoom/create_meeting.md @@ -0,0 +1,12 @@ +import ConnectorFunctionPage from '@site/src/components/ConnectorFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Authentication', id: 'authentication', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, + +]; + + \ No newline at end of file diff --git a/docs/reference/slack-functions.md b/docs/reference/slack-functions.md new file mode 100644 index 00000000..84e8a9f6 --- /dev/null +++ b/docs/reference/slack-functions.md @@ -0,0 +1,25 @@ +# Slack functions catalog + +| Slack function | Description +|:---|:---| +[`add_bookmark`](/deno-slack-sdk/reference/slack-functions/add_bookmark) | Add a bookmark to a channel +[`add_pin`](/deno-slack-sdk/reference/slack-functions/add_pin) | Pin a message to a channel +[`add_user_to_usergroup`](/deno-slack-sdk/reference/slack-functions/add_user_to_usergroup) | Add a user to a user group +[`archive_channel`](/deno-slack-sdk/reference/slack-functions/archive_channel) | Archive a channel +[`canvas_copy`](/deno-slack-sdk/reference/slack-functions/canvas_copy) | Copy a canvas +[`canvas_create`](/deno-slack-sdk/reference/slack-functions/canvas_create) | Create a canvas +[`canvas_update_content`](/deno-slack-sdk/reference/slack-functions/canvas_update_content) | Update a canvas +[`channel_canvas_create`](/deno-slack-sdk/reference/slack-functions/channel_canvas_create) | Create channel canvas +[`create_channel`](/deno-slack-sdk/reference/slack-functions/create_channel) | Create a new channel +[`create_usergroup`](/deno-slack-sdk/reference/slack-functions/create_usergroup) | Create a new user group +[`delay`](/deno-slack-sdk/reference/slack-functions/delay) | Pause a workflow for a specified amount of time +[`invite_user_to_channel`](/deno-slack-sdk/reference/slack-functions/invite_user_to_channel) | Invite a user to a channel +[`open_form`](/deno-slack-sdk/reference/slack-functions/open_form) | Open an interactive form +[`remove_user_from_usergroup`](/deno-slack-sdk/reference/slack-functions/remove_user_from_usergroup) | Remove a user from a user group +[`reply_in_thread`](/deno-slack-sdk/reference/slack-functions/reply_in_thread) | Reply to a message by creating or adding to a thread +[`send_dm`](/deno-slack-sdk/reference/slack-functions/send_dm) | Send a direct message +[`send_ephemeral_message`](/deno-slack-sdk/reference/slack-functions/send_ephemeral_message) | Send an ephemeral message (one only the recipient can see in channel) +[`send_message`](/deno-slack-sdk/reference/slack-functions/send_message) | Send a message in a channel +[`share_canvas`](/deno-slack-sdk/reference/slack-functions/share_canvas) | Share a canvas +[`share_canvas_in_thread`](/deno-slack-sdk/reference/slack-functions/share_canvas_in_thread) | Share a canvas in thread +[`update_channel_topic`](/deno-slack-sdk/reference/slack-functions/update_channel_topic) | Update a channel's topic \ No newline at end of file diff --git a/docs/reference/slack-functions/add_bookmark.md b/docs/reference/slack-functions/add_bookmark.md new file mode 100644 index 00000000..c89ff10a --- /dev/null +++ b/docs/reference/slack-functions/add_bookmark.md @@ -0,0 +1,26 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +### Example workflow step + +```ts +const addBookmarkStep = ExampleWorkflow.addStep( + Schema.slack.functions.AddBookmark, + { + channel_id: "C0123ABC456", + name: "Great Wisconsin Cheese Festival", + link: "https://cheesefest.org/", + }, +); +``` + + + diff --git a/docs/reference/slack-functions/add_pin.md b/docs/reference/slack-functions/add_pin.md new file mode 100644 index 00000000..bec1c1b0 --- /dev/null +++ b/docs/reference/slack-functions/add_pin.md @@ -0,0 +1,24 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +### Example workflow step + +```ts +const addPinStep = ExampleWorkflow.addStep( + Schema.slack.functions.AddPin, + { + channel_id: "C123ABC456", + message: "1645554142.024680", + }, +); +``` + + diff --git a/docs/reference/slack-functions/add_user_to_usergroup.md b/docs/reference/slack-functions/add_user_to_usergroup.md new file mode 100644 index 00000000..11836baf --- /dev/null +++ b/docs/reference/slack-functions/add_user_to_usergroup.md @@ -0,0 +1,27 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Your workspace has user groups and you want to automate adding users to them in workflows? This function is here to make your user groups grow! + +### Example workflow step +Here is an example of how to use this function in a workflow step. + +```ts +const addUserToUsergroupStep = ExampleWorkflow.addStep( + Schema.slack.functions.AddUserToUsergroup, + { + usergroup_id: "S0123ABC456", + user_ids: ["U111AAA111", "U999ZZZ999"], + }, +); +``` + + diff --git a/docs/reference/slack-functions/archive_channel.md b/docs/reference/slack-functions/archive_channel.md new file mode 100644 index 00000000..7bbfadf8 --- /dev/null +++ b/docs/reference/slack-functions/archive_channel.md @@ -0,0 +1,25 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Not all conversations need to continue forever, so when the project is done, this function archives the channel, preventing new messages from being posted while preserving its history. + +### Example workflow step + +```ts +const archiveChannelStep = ExampleWorkflow.addStep( + Schema.slack.functions.ArchiveChannel, + { + channel_id: "C0123ABC456", + }, +); +``` + + \ No newline at end of file diff --git a/docs/reference/slack-functions/canvas_copy.md b/docs/reference/slack-functions/canvas_copy.md new file mode 100644 index 00000000..138e2adf --- /dev/null +++ b/docs/reference/slack-functions/canvas_copy.md @@ -0,0 +1,38 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function copies a canvas. The app using this function will need the following App Home features set in the app manifest file: + +```yaml +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +For information about the expanded_rich_text type that you can use to update your canvases, refer to [expanded_rich_text](/deno-slack-sdk/reference/slack-types#expandedrichtext). + +### Example workflow step + +```ts +const copyCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.CanvasCopy, + { + canvas_id: "CAN87654", + title: "My new canvas", + owner_id: "ABC123456" + }, +); +``` + + \ No newline at end of file diff --git a/docs/reference/slack-functions/canvas_create.md b/docs/reference/slack-functions/canvas_create.md new file mode 100644 index 00000000..fa65ebfa --- /dev/null +++ b/docs/reference/slack-functions/canvas_create.md @@ -0,0 +1,41 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function creates a canvas. If `canvas_create_type` is not passed, it is set as blank. + +The app using this function will need the following App Home features set in the app manifest file: + +```yaml +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +For information about the expanded_rich_text type that you can use to create your canvases, refer to [expanded_rich_text](/deno-slack-sdk/reference/slack-types#expandedrichtext). + +### Example workflow step + +```ts +const createCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.CanvasCreate, + { + title: "My new canvas", + owner_id: "PERSON12345", + canvas_create_type: "blank", + content: { inputs.content } + }, +); +``` + + diff --git a/docs/reference/slack-functions/canvas_update_content.md b/docs/reference/slack-functions/canvas_update_content.md new file mode 100644 index 00000000..f2ce9127 --- /dev/null +++ b/docs/reference/slack-functions/canvas_update_content.md @@ -0,0 +1,42 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function updates the content of a canvas. Either `canvas_id` or `channel_id` must be provided, but not both. Which one is needed is based on the `canvas_update_type`. If the `canvas_update_type` is set to standalone, then a `canvas_id` should be provided. If `canvas_update_type` is set to `channel_canvas`, then a `channel_id` should be provided. The default value of `canvas_update_type` is standalone. + +The app using this function will need the following App Home features set in the app manifest file: + +```yaml +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +For information about the expanded_rich_text type that you can use to update your canvases, refer to [expanded_rich_text](/deno-slack-sdk/reference/slack-types#expandedrichtext). + +### Example workflow step + +```ts +const updateCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.CanvasUpdateContent, + { + action: "append", + content: { inputs.content }, + canvas_update_type: "standalone", + canvas_id: "CAN1234ABC" + }, +); +``` + + + diff --git a/docs/reference/slack-functions/channel_canvas_create.md b/docs/reference/slack-functions/channel_canvas_create.md new file mode 100644 index 00000000..8c8a3082 --- /dev/null +++ b/docs/reference/slack-functions/channel_canvas_create.md @@ -0,0 +1,51 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function creates a channel canvas. The `canvas_create_type` will default to blank if not provided. + +In order to use this Slack function in a coded workflow, you must do the following: + +1. The channel in which you want to create a canvas by using this function must be created by your coded workflow as a step before this function is called. You can use the [`create_channel`](/deno-slack-sdk/reference/slack-functions/create_channel) Slack function to do this. + +2. The invoking user (i.e., the end-user triggering the workflow) must be a member of the newly-created channel before this function executes. There are two recommended ways of doing this: + + * The [`create_channel`](/deno-slack-sdk/reference/slack-functions/create_channel) Slack function accepts a `manager_ids` parameter where you can assign a user as a channel manager. This allows you to automatically add a user to a channel when creating it. + + * You could call the [`invite_user_to_channel`](/deno-slack-sdk/reference/slack-functions/invite_user_to_channel) Slack function after the [`create_channel`](/deno-slack-sdk/reference/slack-functions/create_channel) Slack function completes to invite the invoking user to the channel. + +In addition, the app calling this function will need the following App Home features configured in its manifest file: + +```ts +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +For information about the expanded_rich_text type that you can use to update your canvases, refer to [expanded_rich_text](/deno-slack-sdk/reference/slack-types#expandedrichtext). + +### Example workflow step + +```ts +const createChannelCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.ChannelCanvasCreate, + { + channel_id: "CHAN123456", + canvas_create_type: "template", + canvas_template_id: "TEM123456", + content: { inputs.content } + }, +); +``` + + diff --git a/docs/reference/slack-functions/create_channel.md b/docs/reference/slack-functions/create_channel.md new file mode 100644 index 00000000..7c65f6c3 --- /dev/null +++ b/docs/reference/slack-functions/create_channel.md @@ -0,0 +1,30 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Your automations may create channels as part of a workflow using this Slack function. + +Consider the `channel_name` you specify as merely a suggestion. If it's taken, Slack might append characters to it. Or if you provide some kind of characters Slack doesn't use for channel names, they might be munged. + +To set the channel managers as part of the newly created channel, specify one or more IDs in an array to `manager_ids`. + +### Example workflow step + +```ts +const createChannelStep = ExampleWorkflow.addStep( + Schema.slack.functions.CreateChannel, + { + channel_name: "broadcast-jesse-promotion", + is_private: false, + }, +); +``` + + diff --git a/docs/reference/slack-functions/create_usergroup.md b/docs/reference/slack-functions/create_usergroup.md new file mode 100644 index 00000000..663584ae --- /dev/null +++ b/docs/reference/slack-functions/create_usergroup.md @@ -0,0 +1,28 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + + +Create a new user group. You can add members to it later. + +### Example workflow step + +```ts +const createUsergroupStep = ExampleWorkflow.addStep( + Schema.slack.functions.CreateUsergroup, + { + usergroup_name: "Baking enthusiasts", + usergroup_handle: "cookies", + }, +); +``` + + + diff --git a/docs/reference/slack-functions/delay.md b/docs/reference/slack-functions/delay.md new file mode 100644 index 00000000..66c5d3ae --- /dev/null +++ b/docs/reference/slack-functions/delay.md @@ -0,0 +1,26 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Delay a workflow for some amount of time. Good things come to those who wait. + +## Example workflow step + +```ts +const delayStep = ExampleWorkflow.addStep( + Schema.slack.functions.Delay, + { + minutes_to_delay: 12, + }, +); +``` + + + diff --git a/docs/reference/slack-functions/invite_user_to_channel.md b/docs/reference/slack-functions/invite_user_to_channel.md new file mode 100644 index 00000000..b22bff9d --- /dev/null +++ b/docs/reference/slack-functions/invite_user_to_channel.md @@ -0,0 +1,28 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function allows your workflow to add users to a channel. It only works with channels your workflow created. + +You can provide the usergroup_ids or user_ids parameters. + +## Example workflow step + +```ts +const inviteUserToChannelStep = ExampleWorkflow.addStep( + Schema.slack.functions.InviteUserToChannel, + { + channel_ids: ["C0123ABC456"], + user_ids: ["U111AAA111", "UZZZ999ZZZ"], + }, +); +``` + + diff --git a/docs/reference/slack-functions/open_form.md b/docs/reference/slack-functions/open_form.md new file mode 100644 index 00000000..b2adaeef --- /dev/null +++ b/docs/reference/slack-functions/open_form.md @@ -0,0 +1,92 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Forms are a straight-forward way to collect user input and pass it onto to other parts of your workflow. Their interactivity is one way - users interact with a static form. You cannot update the form itself based on user input. + +Refer to [Creating a form](/deno-slack-sdk/guides/creating-a-form) for guidance. + +### Form element schema + +Form elements have several properties you can customize depending on the element type. + +Links using Markdown are supported in the top-level description, but not in individual form element descriptions. See the [Announcement Bot tutorial](https://api.slack.com/tutorials/announcement-bot) for an example. + +| Property | Type | Description | Required? +| --- | --- | ---|---| +| `name` | `string` | Internal name of the element | Required +| `title` | `string` | Title of the form shown to the user. Max length is 25 characters. | Required +| `type` | `Schema.slack.types.*` | The [type of form element](/deno-slack-sdk/guides/creating-a-form#type-parameters) to display. | Required +| `description` | `string` | Description of the form shown to the user | Optional +| `default` | same as `type` property | Default value for this field | Optional + +#### Form element `type` parameters + +The following parameters are available for each type when defining your form. For each parameter listed, `type` is required. + +Pay careful attention: some element types are prefixed with `Schema.types`, while some are prefixed with `Schema.slack.types`. + +TODO: Table to go here once i have internet again :TODO + +### Example workflow step + +``` +const openFormStep = SayHelloWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Send a greeting", + interactivity: SayHelloWorkflow.inputs.interactivity, + submit_label: "Send", + fields: { + elements: [{ + name: "recipient", + title: "Recipient", + type: Schema.slack.types.user_id, + }, { + name: "channel", + title: "Channel to send message to", + type: Schema.slack.types.channel_id, + default: SayHelloWorkflow.inputs.channel, + }, { + name: "message", + title: "Message to recipient", + type: Schema.types.string, + long: true, + }], + required: ["recipient", "channel", "message"], + }, + }, +); +``` + +### Additional requirements + +When creating a workflow that will have a step to open a form, your workflow must have the call to `OpenForm` be its **first** step or ensure the preceding step is interactive. An interactive step will generate a fresh pointer to use for opening the form. + +Here's an example of a basic workflow definition using `interactivity`: + +```ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const SayHelloWorkflow = DefineWorkflow({ + callback_id: "say_hello_workflow", + title: "Say Hello to another user", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + }, + required: ["interactivity"], + }, +}); +``` + + diff --git a/docs/reference/slack-functions/remove_user_from_usergroup.md b/docs/reference/slack-functions/remove_user_from_usergroup.md new file mode 100644 index 00000000..ecb39d4b --- /dev/null +++ b/docs/reference/slack-functions/remove_user_from_usergroup.md @@ -0,0 +1,26 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Users have different needs and different teams over time. Sometimes you want to automate removing users from user groups. + +## Example workflow step + +``` +const removeUserFromUsergroupStep = ExampleWorkflow.addStep( + Schema.slack.functions.RemoveUserFromUsergroup, + { + usergroup_id: "S0123ABC456", + user_ids: ["U111AAA111", "UZZZ999ZZZ"], + }, +); +``` + + diff --git a/docs/reference/slack-functions/reply_in_thread.md b/docs/reference/slack-functions/reply_in_thread.md new file mode 100644 index 00000000..b3e779ae --- /dev/null +++ b/docs/reference/slack-functions/reply_in_thread.md @@ -0,0 +1,65 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Based on certain conditions, you can reply to a message in-thread. + +## Example workflow step + +``` +const replyInThreadStep = ExampleWorkflow.addStep( + Schema.slack.functions.ReplyInThread, + { + message_context: sendMessageStep.outputs.message_context, + reply_broadcast: false, + message: "Thank you for submitting a message to #sos, Gilligan.", + }, +); +``` + +To reply in-thread with buttons, they must be wrapped in a block (example: [section block](https://api.slack.com/reference/block-kit/blocks#section)). Then, use the [`workflow_button`](https://api.slack.com/reference/block-kit/block-elements#workflow_button) element. + +### Example workflow step with interactive blocks + +```ts +const replyInThreadButton = ExampleWorkflow.addStep( + Schema.slack.functions.ReplyInThread, + { + message_context: sendMessageStep.outputs.message_context, + reply_broadcast: false, + message: "Please confirm your request for help in #sos.", + interactive_blocks: [ + { + "type": "section", + "text": { + "type": "plain_text", + "text": "Click to confirm", + }, + "accessory": { + "type": "workflow_button", + "text": { + "type": "plain_text", + "text": "Confirm", + }, + "action_id": "button-action-id-custom", + "workflow": { + "trigger": { + "url": + "https://slack.com/shortcuts/Ft0123ABC456/xyz...zyx", + }, + }, + }, + }, + ] + }, +); +``` + + diff --git a/docs/reference/slack-functions/send_dm.md b/docs/reference/slack-functions/send_dm.md new file mode 100644 index 00000000..7016d3b1 --- /dev/null +++ b/docs/reference/slack-functions/send_dm.md @@ -0,0 +1,46 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Sends a message directly to a user, from your workflow. + +If you include a button in the direct message, the function execution will not continue until an end user clicks on that button. + +### Example workflow step + +``` +const sendDmStep = ExampleWorkflow.addStep( + Schema.slack.functions.SendDm, + { + user_id: "U123ABC456", + message: "Don't give up. Never surrender. Except the cookies. Surrender the cookies.", + }, +); +``` + +### Example action payload + +``` +{ + "action_id": "WaXA", + "block_id": "=qXel", + "text": { + "type": "plain_text", + "text": "View", + "emoji": true + }, + "value": "click_me_123", + "type": "button", + "action_ts": "1548426417.840180" +} +``` + + + diff --git a/docs/reference/slack-functions/send_ephemeral_message.md b/docs/reference/slack-functions/send_ephemeral_message.md new file mode 100644 index 00000000..cfc105b2 --- /dev/null +++ b/docs/reference/slack-functions/send_ephemeral_message.md @@ -0,0 +1,27 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Sends an [ephemeral message](https://api.slack.comhttps://api.slack.com/surfaces/messages#ephemeral) to a specific channel. This lets a user see what your workflow has to say without everyone in the conversation having to see it or it becoming part of the conversation's record. + +### Example workflow step + +```ts +const sendEphemeralMessageStep = ExampleWorkflow.addStep( + Schema.slack.functions.SendEphemeralMessage, + { + channel_id: "C082T4F6S1N", + user_id: "U0J46F228L0", + message: "Someone in this conversation is not accurately representing reality. Converse further with care.", + }, +); +``` + + diff --git a/docs/reference/slack-functions/send_message.md b/docs/reference/slack-functions/send_message.md new file mode 100644 index 00000000..28411795 --- /dev/null +++ b/docs/reference/slack-functions/send_message.md @@ -0,0 +1,129 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Sends a message to a specific channel. The `message` input only supports non-interactive [`rich_text`](/deno-slack-sdk/reference/slack-types#rich-text). + +This function returns a timestamp of the new message, which also serves as a confirmation that the message was sent. + +:::info[Direct messages] + +The `send_message` function does not allow for direct messages to users — use the [`send_dm`](/deno-slack-sdk/reference/slack-functions/send_dm) function instead. + +::: + +### Adding interactivity with buttons {#buttons} + +The `interactive_blocks` input only supports the [`button`](https://api.slack.com/reference/block-kit/block-elements#button) and [`workflow_button`](https://api.slack.com/reference/block-kit/block-elements#workflow_button) [interactive blocks](https://api.slack.com/reference/block-kit/block-elements). + +Ensure that you do not use non-interactive elements via the `interactive_blocks` input, as this could cause unintended behavior. + +If you include a button in the message, the function execution will not continue until a user clicks on that button. + +### Example + +To collect a formatted message from your end users and send it as-is, you can use a form to collect the formatted text, and a Slack function to send it: + +```ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const RichTextWorkflow = DefineWorkflow({ + callback_id: "rich_text_workflow", + title: "rich-text input workflow", + input_parameters: { + properties: { + interactivity: { type: Schema.slack.types.interactivity }, + channel: { type: Schema.slack.types.channel_id }, + }, + required: ["interactivity", "channel"], + }, +}); + +const inputForm = RichTextWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Send formatted message", + interactivity: RichTextWorkflow.inputs.interactivity, + submit_label: "Send formatted message", + fields: { + elements: [ + { + name: "formattedInput", + title: "Formatted input", + type: Schema.slack.types.rich_text, + }, + { + name: "channel", + title: "Post in:", + type: Schema.slack.types.channel_id, + default: RichTextWorkflow.inputs.channel, + }, + ], + required: ["channel", "formattedInput"], + }, + }, +); + +RichTextWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: inputForm.outputs.fields.channel, + message: inputForm.outputs.fields.formattedInput, + interactive_blocks: [{ + "type": "actions", + "elements": [ + { + "type": "button", + "text": { "type": "plain_text", "text": "Approve" }, + "style": "primary", + "value": "approve", + "action_id": "approve_button", + }, + ], + }], +}); +``` + +Here's an example with just regular text: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const createAnnouncement = DefineWorkflow({ + callback_id: "create-announcement", + title: "Workflow for creating announcements", + input_parameters: { properties: {}, required: [] }, +}); + +const sendPreview = createAnnouncement.addStep( + Schema.slack.functions.SendMessage, + { + channel_id: "CTLC2K3JS", + message: "Good to see you here!", + }, +); +``` + +### Example action payload {#action-payload} + +```js +{ + "action_id": "WaXA", + "block_id": "=qXel", + "text": { + "type": "plain_text", + "text": "View", + "emoji": true + }, + "value": "click_me_123", + "type": "button", + "action_ts": "1548426417.840180" +} +``` + + diff --git a/docs/reference/slack-functions/share_canvas.md b/docs/reference/slack-functions/share_canvas.md new file mode 100644 index 00000000..2e6af2bf --- /dev/null +++ b/docs/reference/slack-functions/share_canvas.md @@ -0,0 +1,57 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function allows you to share a canvas to users and channels. + +The app using this function will need the following App Home features set in the app manifest file: + +```yaml +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +For information about the `expanded_rich_text` type that you can use to update your canvases, refer to [expanded_rich_text](/deno-slack-sdk/reference/slack-types#expandedrichtext). + +## Example workflow step + +```ts +const shareCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.ShareCanvas, + { + canvas_id: "CAN123456", + channel_ids: ["C111AAA111","C222BBB222"], + user_ids: ["U333DDD333","U444EEE444"], + access_level: "edit", + message: [ + { + "type": "rich_text", + "elements": [ + { + "type": "rich_text_section", + "elements": [{ + "type": "text", + "text": "Sharing the create canvas", + }], + }, + ], + }, + ], + }, +); +``` + + + + diff --git a/docs/reference/slack-functions/share_canvas_in_thread.md b/docs/reference/slack-functions/share_canvas_in_thread.md new file mode 100644 index 00000000..2926fe34 --- /dev/null +++ b/docs/reference/slack-functions/share_canvas_in_thread.md @@ -0,0 +1,39 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +This function allows you to share a canvas in a thread. + +The app using this function will need the following App Home features set in the app manifest file: + +``` +features: { + appHome: { + messagesTabEnabled: true, + messagesTabReadOnlyEnabled: false, + }, +}, +``` + +### Example workflow step: + +``` +const shareCanvasStep = ExampleWorkflow.addStep( + Schema.slack.functions.ShareCanvasInThread, + { + canvas_id: "CAN123456", + access_level: "edit", + message_context: inputs.messageContext, + message: { inputs.message } + }, +); +``` + + diff --git a/docs/reference/slack-functions/update_channel_topic.md b/docs/reference/slack-functions/update_channel_topic.md new file mode 100644 index 00000000..d6626ff5 --- /dev/null +++ b/docs/reference/slack-functions/update_channel_topic.md @@ -0,0 +1,27 @@ +import SlackFunctionPage from '@site/src/components/SlackFunctionPage'; + +export const toc = [ +{ value: 'Facts', id: 'facts', level: 2 }, +{ value: 'Input parameters', id: 'input-parameters', level: 2 }, +{ value: 'Output parameters', id: 'output-parameters', level: 2 }, +{ value: 'Usage info', id: 'usage-info', level: 2 }, +]; + + + +Change the topic to what's top of mind. Folks like predictable, informative topics. Your workflow must be a member of the conversation. + +### Example workflow step + +```ts +const updateChannelTopicStep = ExampleWorkflow.addStep( + Schema.slack.functions.UpdateChannelTopic, + { + channel_id: "C0123ABC456", + topic: "The main idea in mind", + }, +); +``` + + + diff --git a/docs/reference/slack-types.md b/docs/reference/slack-types.md new file mode 100644 index 00000000..18d23f35 --- /dev/null +++ b/docs/reference/slack-types.md @@ -0,0 +1,2320 @@ +--- +slug: /deno-slack-sdk/reference/slack-types +--- + +import ChoicesProperty from '/content/deno-slack-sdk/guides/datatypes/_choices-property-snippet.md'; + +# Slack types + +The examples of declaring a type are shown below in both TypeScript, as they would appear in an app built using the Deno SDK, and in JSON, as they would be defined in a manifest. All manifests can be written in JSON; however, declaring types in an app using the Deno SDK is done differently, requiring a reference to the `Schema.slack` package for non-primitive types. + +| Name | Type | Description | +| :--- | :--- | :--- | +| [`array`](#array) | Array | An array of items (based on a type that you specify). +| [`blocks`](#blocks) | Array of Slack Blocks | An array of objects that contain layout and style information about your message. | +| [`boolean`](#boolean) | Boolean | A logical value, must be either `true` or `false`. | +| [`canvas_id`](#canvasid) | String | A Slack canvas ID, such as `F123456AB`. | +| [`canvas_template_id`](#canvastemplateid) | String | A Slack canvas template ID, such as `T5678ABC`. | +| [`channel_id`](#channelid) | String | A Slack channel ID, such as `C123ABC456` or `D123ABC456`. | +| [`date`](#date) | String | A string containing a date, format is displayed as `YYYY-MM-DD`. | +| [`expanded_rich_text`](#expandedrichtext) | Object | A way to nicely format messages in your app. This type cannot convert other message types, e.g. blocks or strings, and is explicitly for use with canvases. | +| [`file_id`](#fileid) | Object | A file ID, such as `F123ABC456`. | +| [`integer`](#integer) | Integer | A whole number, such as `-1`, `0`, or `31415926535`. | +| [`interactivity`](#interactivity) | Object | An object that contains context about the interactive event that led to opening of the form. | +| [`list_id`](#list-id) | String | A Slack list ID, such as `F123456ABC`. | +| [`message_context`](#message-context) | Object | An individual instance of a message. | +| [`message_ts`](#message-ts) | String | A Slack-specific hash/timestamp necessary for referencing events like messages in Slack. | +| [`number`](#number) | Number | A number that allows decimal points such as `13557523.0005`. | +| [`oauth2`](#oauth2) | Object | The OAuth2 context created after authenticating with [external auth](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication). | +| [`object`](#object) | Object | A custom Javascript object, like `{"site": "slack.com"}`. | +| [`rich_text`](#rich-text) | Object | A way to nicely format messages in your app. This type cannot convert other message types e.g. blocks, strings. | +| [`string`](#string) | String | UTF-8 encoded string, up to 4000 bytes. | +| [`team_id`](#team_id) | String | A Slack team ID, such as `T1234567890`. | +| [`timestamp`](#timestamp) | Integer | A Unix timestamp in seconds. Not compatible with Slack message timestamps - use [string](#string) instead. | +| [`user_context`](#usercontext) | Object | Represents a user who interacted with a workflow at runtime. | +| [`user_id`](#userid) | String | A Slack user ID, such as `U123ABC456` or `W123ABC456`. | +| [`usergroup_id`](#usergroupid) | String | A Slack usergroup ID, such as `S123ABC456`. | + + +## Slack types for datastores {#datastores} +When defining a [datastore](/deno-slack-sdk/guides/using-datastores), you can use certain Slack types for its attributes. Attributes accept only a single `type` property in the definition, instead of all the common properties listed above. The following is a list of the Slack types supported for use with datastores: + +* [`channel_id`](#channelid) +* [`date`](#date) +* [`message_ts`](#message-ts) +* [`rich_text`](#rich-text) +* [`timestamp`](#timestamp) +* [`user_id`](#userid) +* [`usergroup_id`](#usergroupid) + +Here's a sample datastore definition using Slack types for attributes: + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const MyDatastore = DefineDatastore({ + name: "my_datastore", + primary_key: "id", + attributes: { + id: { type: Schema.types.string }, + channel: { type: Schema.slack.types.channel_id }, + message: { type: Schema.types.string }, + author: { type: Schema.slack.types.user_id }, + isMember: { type: Schema.types.boolean }, + }, +}); +``` + +--- + +## Array {#array} + +Type: `array` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `items` | object | The type of items in the array. Can be one of the following types: [`channel_id`](#channelid), [`user_id`](#userid), [`usergroup_id`](#usergroupid), [`timestamp`](#timestamp), [`string`](#string), [`integer`](#integer), [`number`](#number), [`boolean`](#boolean), [`list_id`](#list-id), [`canvas_id`](#canvasid), [`canvas_template_id`](#canvastemplateid), `channel_canvas_id`, [`team_id`](#team_id), [`file_id`](#fileid). | +| `minItems` | integer | Minimum number of items allowed. | +| `maxItems` | integer | Maximum number of items allowed. | + + +Declare an `array` of types: + + + + +```javascript +// ... +{ + name: "departments", + title: "Your department", + type: Schema.types.array, + items: { + type: Schema.types.string, + enum: ["marketing", "design", "sales", "engineering"], + }, + default: ["sales", "engineering"], +} +// ... +``` + + + + +```json +// ... +"departments": { + "title": "Your department", + "type": "array", + "items": { + "type": "string", + "enum": [ + "marketing", "design", "sales", "engineering" + ] + } +} +// ... +``` + + + + +:::warning[Arrays and object types] + +Be sure to define the array's properties in the `items` object. Untyped objects are not currently supported. In addition, you can only use an object as the item type of array if it's a custom object. Otherwise, you may receive the following error: + +`Unexpected schema encountered for array type: failed to match exactly one allowed schema for items - {"type":"one_of"} (failed_constraint).` + +::: + +
+Array example + +In this example function, we have an array of the [custom type](/deno-slack-sdk/guides/creating-a-custom-type) `ChannelType` as both an `input_parameter` and `output_parameter`. See this custom type and array in action in the [Deno Archive Channel](https://github.com/slack-samples/deno-archive-channel) sample app. + +```javascript +const ChannelType = DefineType(...) + +export const FilterStaleChannelsDefinition = DefineFunction({ + callback_id: "filter_stale_channels", + title: "Filter Stale Channels", + description: + "Filter out any channels that have received messages within the last 6 months", + source_file: "functions/filter_stale_channels.ts", + input_parameters: { + properties: { + channels: { + type: Schema.types.array, + description: "The list of Channel IDs to filter", + items: { + type: ChannelType, + }, + }, + }, + required: ["channels"], + }, + output_parameters: { + properties: { + filtered_channels: { + type: Schema.types.array, + description: "The list of stale Channel IDs", + items: { + type: ChannelType, + }, + }, + }, + required: [], + }, +}); +``` +
+ +--- + +## Blocks {#blocks} +Type: `slack#/types/blocks` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare an array of [Block Kit](https://api.slack.com/block-kit) JSON objects. + + + + +```javascript +// ... +properties: { + forecast: { + type: Schema.slack.types.blocks, + }, +}, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "forecast": { + "type": "slack#/types/blocks" + } +} +// ... +``` + + + + + +If you use [Block Kit builder](https://api.slack.com/tools/block-kit-builder) to build your Block Kit objects, be sure to _only_ grab the `blocks` array. For example: + +```json +[ + { + "type": "section", + "text": { + "type": "plain_text", + "text": "This is a plain text section block.", + "emoji": true + } + }, + { + "type": "image", + "image_url": "example.com/png" + "alt_text": "inspiration" + } +] +``` + + + +
+Blocks example + +In this example function, we get the current weather forecast. + +```javascript +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const ForecastFunctionDefinition = DefineFunction({ + callback_id: "get_forecast", + title: "Weather forecast", + description: "A function to get the weather forecast", + source_file: "functions/weather_forecast.ts", + input_parameters: { + properties: { + city: { + type: Schema.types.string, + description: "City", + }, + country: { + type: Schema.types.string, + description: "Country", + }, + state: { + type: Schema.types.string, + description: "State", + }, + }, + required: ["city"], + }, + output_parameters: { + properties: { + forecast: { + type: Schema.slack.types.blocks, + description: "Weather forecast", + }, + }, + required: ["forecast"], + }, +}); +``` +
+ + +--- + +## Boolean {#boolean} +Type: `boolean` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. For a `boolean` it would be `true` or `false`.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare a `boolean` type: + + + + +```javascript +// ... +isMember: { + type: Schema.types.boolean, +} +// ... +``` + + + + +```json +// ... +"isMember": { + "type": "boolean" +} +// ... +``` + + + + +
+Boolean example + +In this example datastore definition, we use a `boolean` type to capture whether the message author holds membership in our club. + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const MyDatastore = DefineDatastore({ + name: "my_datastore", + primary_key: "id", + attributes: { + id: { type: Schema.types.string }, + channel: { type: Schema.slack.types.channel_id }, + message: { type: Schema.types.string }, + author: { type: Schema.slack.types.user_id }, + isMember: { type: Schema.types.boolean }, + }, +}); +``` +
+ + + +--- + +## Channel ID {#channelid} +Type: `slack#/types/channel_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + + +Declare a `channel_id` type: + + + + +```javascript +// ... + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + }, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "channel": { + "type": "slack#/types/channel_id" + } +} +// ... +``` + + + + +--- + +## Canvas ID {#canvasid} +Type: `slack#/types/canvas_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below | +| `render_condition` | Object | The `render_condition` property contains three properties of its own: the `operator` property is a string logical operator which acts on the conditions; the `is_required` property is a boolean indicating if the property is required, and the `conditions` property is an array of object conditions which specify if the field should be rendered. + + + +Declare a `canvas_id` type: + + + + +```javascript +// ... +{ + name: "project_canvas", + title: "Project Canvas", + type: Schema.slack.types.canvas_id +} +// ... +``` + + + + +```json +// ... +"project_canvas": { + "title": "Project Canvas", + "type": "slack#/types/canvas_id" +} +// ... +``` + + + + +
+Canvas ID example + +In this example workflow, we get a canvas ID and information to update it. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { CanvasWorkflow } from "../workflows/canvas.ts"; + +const inputForm = CanvasWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Provide info to update a canvas", + interactivity: CanvasWorkflow.inputs.interactivity, + submit_label: "Submit", + fields: { + elements: [ + { + name: "canvas", + title: "Canvas to update", + type: Schema.slack.types.canvas_id + }, + { + name: "content", + title: "Content", + type: Schema.slack.types.expanded_rich_text, + } + ], + required: ["canvas", "content"], + }, + }, +); +``` +
+ + + +--- + +## Canvas Template ID {#canvastemplateid} +Type: `slack#/types/canvas_template_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below | + + + +Declare a `canvas_template_id` type: + + + + +```javascript +// ... +{ + name: "onboarding_template", + title: "Onboarding Canvas Template", + type: Schema.slack.types.canvas_template_id +} +// ... +``` + + + +```json +// ... +"onboarding_template": { + "title": "Onboarding Canvas Template", + "type": "slack#/types/canvas_template_id" +} +// ... +``` + + + +
+Canvas Template ID example + +In this example workflow, we receive a `canvas_template_id` for creating a new canvas. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { CanvasWorkflow } from "../workflows/canvas.ts"; + +const inputForm = CanvasWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Provide a template your canvas from", + interactivity: CanvasWorkflow.inputs.interactivity, + submit_label: "Submit", + fields: { + elements: [ + { + name: "template", + title: "Canvas template", + type: Schema.slack.types.canvas_template_id + }, + { + name: "title", + title: "Canvas title", + type: Schema.types.string, + }, + { + name: "owner_id", + title: "Owner ID", + type: Schema.slack.types.user_id + } + ], + required: ["template", "title", "owner_id"], + }, + }, +); +``` +
+ + + +--- + +## Date {#date} +Type: `slack#/types/date` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +Declare a `date` type: + + + + +```javascript +// ... +fields: { + elements: [ + { + name: "date", + title: "Date Posted", + type: Schema.slack.types.date, + }, + ], +}, +// ... +``` + + + + +```json +// ... +"fields": { + "elements": [ + { + "date_posted": { + "type": "slack#/types/date" + } + } + ] +} +// ... +``` + + + +
+Date example + +In this example workflow, a form requires a `date` as input, which is printed along with the message after the form is submitted. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const TestReverseWorkflow = DefineWorkflow({ + callback_id: "test_reverse", + title: "Test reverse", + input_parameters: { + properties: { + channel: { type: Schema.slack.types.channel_id }, + interactivity: { type: Schema.slack.types.interactivity }, + }, + required: ["interactivity"], + }, +}); + +const formData = TestReverseWorkflow.addStep(Schema.slack.functions.OpenForm, { + title: "Reverse string form", + submit_label: "Submit form", + description: "Submit a string to reverse", + interactivity: TestReverseWorkflow.inputs.interactivity, + fields: { + required: ["channel", "stringInput", "date"], + elements: [ + { + name: "stringInput", + title: "String input", + type: Schema.types.string, + }, + { + name: "date", + title: "Date Posted", + type: Schema.slack.types.date, + }, + { + name: "channel", + title: "Post in", + type: Schema.slack.types.channel_id, + default: TestReverseWorkflow.inputs.channel, + }, + ], + }, +}); + +import { ReverseFunction } from "../functions/reverse.ts"; +const reverseStep = TestReverseWorkflow.addStep(ReverseFunction, { + input: formData.outputs.fields.stringInput, +}); + +// Add the date parameter as a step in your workflow. The message and date will be printed side by side. +TestReverseWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: formData.outputs.fields.channel, + message: reverseStep.outputs.reverseString + " " + + formData.outputs.fields.date, +}); +``` +
+ + + +--- + +## Expanded Rich Text {#expandedrichtext} +Type: `slack#/types/expanded_rich_text` + +The `expanded_rich_text` type is a superset of the [`rich_text`](#rich-text) type, and is explicitly for use with canvases. It accepts all elements that `rich_text` provides and behaves in the same way as `rich_text`, except that it also accepts the following additional sub-elements: + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `rich_text_header` | object | The text for the header, in the form of a `plain_text` [text object](https://api.slack.com/reference/block-kit/composition-objects#text). | +| `rich_text_divider` | object | Creates a divider to place between text. | +| `rich_text_list` | object | This is an expanded version of the [`rich_text_list`](https://api.slack.com/reference/block-kit/blocks#rich_text_list) element used in `rich_text` blocks. It behaves the same, except that it accepts two new style fields: `checked` and `unchecked`. This allows for the creation of checklists. | + +Declare an `expanded_rich_text` type: + + + + +```javascript +// ... +{ + name: "canvas_content", + title: "Canvas Content", + type: Schema.slack.types.expanded_rich_text +} +// ... +``` + + + + +```json +// ... +"canvas_content": { + "title": "Canvas Content", + "type": "slack#/types/expanded_rich_text" +} +// ... +``` + + + + +
+Expanded rich text example + +Here is an example payload that shows the `expanded_rich_text` type and all of its sub-elements: + +```javascript +[ + { + "type": "expanded_rich_text", + "elements": [ + { + "type": "rich_text_header", + "elements": [ + { + "type": "text", + "text": "Hello world" + } + ], + "level": 1 + }, + { + "type": "rich_text_list", + "style": "unchecked", + "indent": 0, + "elements": [ + { + "type": "rich_text_section", + "elements": [ + { + "type": "text", + "text": "one" + } + ] + } + ], + "border": 0 + }, + { + "type": "rich_text_list", + "style": "checked", + "indent": 1, + "elements": [ + { + "type": "rich_text_section", + "elements": [ + { + "type": "text", + "text": "two" + } + ] + } + ], + "border": 0 + }, + { + "type": "rich_text_divider" + } + ] + } +] +``` + +
+ + + +--- + +## File ID {#fileid} +Type: `slack#/types/file_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `allowed_filetypes_group` | string | If provided, a predefined subset of filetypes will be restricted for file upload when this type is used in an OpenForm function. Can either be `ALL` or `IMAGES_ONLY`. | +| `allowed_filetypes` | string[] | If provided, only these filetypes are allowed for file upload in an `OpenForm` function. Empty arrays are not allowed. Filetypes defined here will override any restrictions set in `allowed_filetypes_group`. | + +Declare a `file_id` type: + + + + +```javascript +// ... +fields: { + elements: [ + { + title: "Enter a file", + name: "image-123", + type: Schema.types.array, + maxItems: 1, + description: "", + items: { + type: Schema.slack.types.file_id, + allowed_filetypes_group: "ALL" + } + }, + ], +}, +// ... +``` + + + + +```json +// ... +"file": { + "type": "slack#/types/file_id", + "allowed_filetypes_group": "ALL" +} +// ... +``` + + + + +### OpenForm parameters {#openform-parameters} + +When using the `file_id` type in an OpenForm function, there are two additional parameters that can be utilized. + +| Parameter | Type | Description | +|---|---|---| +| `allowed_filetypes_group` | `string` | Can be either `ALL` or `IMAGES_ONLY`. If provided, specifies allowed predefined subset of filetypes for file in an OpenForm function. | +| `allowed_filetypes` | `array` of `strings` | If provided, specifies allowed filetypes for file upload in an OpenForm function. Overrides any restrictions set in `allowed_filetypes_group`. | + +
+File ID example + +In this example workflow, we collect a file from the user. + +```js +import { Schema } from "deno-slack-sdk/mod.ts"; +import { ImageWorkflow } from "../workflows/ImageWorkflow.ts"; + +const getImageStep = ImageWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Submit this form", + interactivity: ImageWorkflow.inputs.interactivity, + fields:{ + elements: [ + { + title: "Enter a file", + name: "image-123", + type: Schema.types.array, + maxItems: 1, + description: "", + items: { + type: Schema.slack.types.file_id, + allowed_filetypes_group: "ALL" + }, + } + ], + required: ["image-123"], + } + }, +); +``` +
+ + + +--- + +## Integer {#integer} +Type: `integer` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `minimum` | number | Absolute minimum acceptable value for the integer. | +| `maximum` | number | Absolute maximum acceptable value for the integer. | +| `enum` | number[] | Constrain the available integer options to just the list of integers denoted in the `enum` property. Usage of `enum` also instructs any UI that collects a value for this parameter to render a dropdown select input rather than a free-form text input.| +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + + + + +Declare an `integer` type: + +```javascript +// ... + name: "meetings", + title: "Number of meetings", + type: Schema.types.integer, +// ... +``` + + + + +```json +// ... +"meetings": { + "title": "Number of meetings", + "type": "integer" +} +// ... +``` + + + + +
+Integer example + +In this example workflow, we check the number of meetings we have scheduled for the day. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { MeetingsWorkflow } from "../workflows/meetings.ts"; + +const inputForm = MeetingsWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Number of meetings", + interactivity: MeetingsWorkflow.inputs.interactivity, + submit_label: "Check meetings", + fields: { + elements: [ + { + name: "channel", + title: "Channel to send results to", + type: Schema.slack.types.channel_id, + default: MeetingsWorkflow.inputs.channel, + }, + { + name: "meetings", + title: "Number of meetings", + description: "meetings", + type: Schema.types.integer, + minimum: -1, + maximum: 5, + }, + { + name: "meetingdate", + title: "Meeting date", + type: Schema.slack.types.date, + }, + ], + required: ["channel", "meetings", "meetingdate"], + }, + }, +); +``` +
+ + + +--- + +## Interactivity {#interactivity} +Type: `slack#/types/interactivity` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `interactivity_pointer` | string | A pointer used to confirm user-initiated interactivity in a function. | +| `interactor` | `user_context` | Context information of the user who initiated the interactivity. | + +Declare the `interactivity` type: + + + + +```javascript +// ... +properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, +}, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "interactivity": { + "type": "slack#/types/interactivity" + } +} +// ... +``` + + + +
+Interactivity example + +In this example workflow, we specify that it is an interactive workflow. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const GreetingWorkflow = DefineWorkflow({ + callback_id: "greeting_workflow", + title: "Send a greeting", + description: "Send a greeting to channel", + input_parameters: { + properties: { + interactivity: { type: Schema.slack.types.interactivity }, + channel: { type: Schema.slack.types.channel_id }, + }, + required: ["interactivity"], + }, +}); +``` +
+ + + +--- + +## List ID {#list-id} + +Type: `slack#/types/list_id` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare the `list_id` type: + + + + +```javascript +// ... + name: "current_bugs", + title: "Current Bug List", + type: Schema.slack.types.list_id, +// ... +``` + + + + +```json +// ... +"current_bugs": { + "title": "Current Bug List", + "type": "slack#/types/list_id" +} +// ... +``` + + + +
+List ID example + +In this example workflow, we disseminate meeting information. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { MeetingsWorkflow } from "../workflows/meetings.ts"; + +const inputForm = MeetingsWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Meeting follow-up", + interactivity: MeetingsWorkflow.inputs.interactivity, + submit_label: "Meeting follow-up", + fields: { + elements: [ + { + name: "channel", + title: "Channel to send info to", + type: Schema.slack.types.channel_id, + default: MeetingsWorkflow.inputs.channel, + }, + { + name: "meetingdate", + title: "Meeting date", + type: Schema.slack.types.date, + }, + { + name: "notes", + title: "Meeting notes", + type: Schema.slack.types.canvas_id + }, + { + name: "actions", + title: "Meeting to-dos", + description: "Action items from the meeting", + type: Schema.slack.types.list_id + } + ], + required: ["channel", "meetingdate","notes", "actions"], + }, + }, +); +``` +
+ + +--- + +## Message Context {#message-context} +Type: `slack#/types/message_context` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +The `message_context` type is used in the [`ReplyInThread`](/deno-slack-sdk/reference/slack-functions/reply_in_thread) Slack function as the target message you want to reply to. + +For example, let's say you have a workflow step that uses the [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message) function. If you want to send a reply to that message in a follow-on +step that calls the [`ReplyInThread`](/deno-slack-sdk/reference/slack-functions/reply_in_thread) function, pass +the return value from the first step into the `message_context` parameter of `ReplyInThread`. + +Here's a brief example: + + + + +```javascript +// Send a message to channel with ID C123456 +const msgStep = GreetingWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: "C123456", + message: "This is a message to the channel.", +}); + +// Send a message as an in-thread reply to the above message by passing +// the outputs' message_context property +GreetingWorkflow.addStep(Schema.slack.functions.ReplyInThread, { + message_context: msgStep.outputs.message_context, + message: "This is a threaded reply to the above message.", +}); +``` + + + + +```json +// ... +"message_context": { + "type": "slack#/types/message_context" +} +// ... +``` + + + + +You can also construct and deconstruct the `message_context` as you see fit in your app. Here is what comprises `message_context`: + +| Property | Type | Description | Required | +|----------|------|-------------|----------| +| `message_ts` | [Schema.slack.types.message_ts](#message-ts) | A Slack-specific hash/timestamp necessary for referencing events like messages in Slack. | Required | +| `channel_id` | [Schema.slack.types.channel_id](#channelid) | The ID of the channel where the message is posted. | Optional | + +Any individual property on message_context could be referenced too. See the below example where we pass `message_context.message_ts` to the `trigger_ts` property: + +```javascript +//... + +const message = CreateSurveyWorkflow.addStep( + Schema.slack.functions.ReplyInThread, + { + message_context: { + channel_id: CreateSurveyWorkflow.inputs.channel_id, + message_ts: CreateSurveyWorkflow.inputs.parent_ts, + }, + message: + `Your feedback is requested – <${trigger.outputs.trigger_url}|survey now>!`, + }, +); + +CreateSurveyWorkflow.addStep(SaveSurveyFunctionDefinition, { + channel_id: CreateSurveyWorkflow.inputs.channel_id, + parent_ts: CreateSurveyWorkflow.inputs.parent_ts, + reactor_id: CreateSurveyWorkflow.inputs.reactor_id, + trigger_ts: message.outputs.message_context.message_ts, //Here we reference message_ts from message_context + trigger_id: trigger.outputs.trigger_id, + survey_stage: "SURVEY", +}); + +//... +``` + + + +--- + +## Message Timestamp {#message-ts} +Type: `slack#/types/message_ts` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare a `message_ts` type: + + + + +```javascript +//... +input_parameters: { + properties: { + message_ts : { + type: Schema.slack.types.message_ts, + description: "The ts value of a message" + }, + }, +}, +//... +``` + + + + +```json +// ... +"input_parameters": { + "message_ts": { + "type": "slack#/types/message_ts", + "description": "The ts value of a message" + } +} +// ... +``` + + + + +
+Message Timestamp example + +In this example workflow from the [Simple Survey sample app](https://github.com/slack-samples/deno-simple-survey), a `message_ts` is used as an input parameter in two functions. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +import { CreateGoogleSheetFunctionDefinition } from "../functions/create_google_sheet.ts"; +import { CreateTriggerFunctionDefinition } from "../functions/create_survey_trigger.ts"; +import { SaveSurveyFunctionDefinition } from "../functions/save_survey.ts"; +import { RemoveThreadTriggerFunctionDefintion } from "../functions/remove_thread_trigger.ts"; + +const CreateSurveyWorkflow = DefineWorkflow({ + callback_id: "create_survey", + title: "Create a survey", + description: "Add a request for feedback to a message", + input_parameters: { + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + description: "The channel containing the reacted message", + }, + parent_ts: { + type: Schema.types.string, + description: "Message timestamp of the reacted message", + }, + parent_url: { + type: Schema.types.string, + description: "Permalink to the reacted message", + }, + reactor_id: { + type: Schema.slack.types.user_id, + description: "User that added the reacji", + }, + }, + required: ["channel_id", "parent_ts", "parent_url", "reactor_id"], + }, +}); + +// Step 1: Create a new Google spreadsheet +const sheet = CreateSurveyWorkflow.addStep( + CreateGoogleSheetFunctionDefinition, + { + google_access_token_id: {}, + title: CreateSurveyWorkflow.inputs.parent_ts, + }, +); + +// Step 2: Create a link trigger for the survey +const trigger = CreateSurveyWorkflow.addStep(CreateTriggerFunctionDefinition, { + google_spreadsheet_id: sheet.outputs.google_spreadsheet_id, + reactor_access_token_id: sheet.outputs.reactor_access_token_id, +}); + +// Step 3: Delete the prompt message and metadata +CreateSurveyWorkflow.addStep(RemoveThreadTriggerFunctionDefintion, { + channel_id: CreateSurveyWorkflow.inputs.channel_id, + parent_ts: CreateSurveyWorkflow.inputs.parent_ts, + reactor_id: CreateSurveyWorkflow.inputs.reactor_id, +}); + +// Step 4: Notify the reactor of the survey spreadsheet +CreateSurveyWorkflow.addStep(Schema.slack.functions.SendDm, { + user_id: CreateSurveyWorkflow.inputs.reactor_id, + message: + `Feedback for <${CreateSurveyWorkflow.inputs.parent_url}|this message> is being <${sheet.outputs.google_spreadsheet_url}|collected here>!`, +}); + +// Step 5: Send the survey into the reacted thread +const message = CreateSurveyWorkflow.addStep( + Schema.slack.functions.ReplyInThread, + { + message_context: { + channel_id: CreateSurveyWorkflow.inputs.channel_id, + message_ts: CreateSurveyWorkflow.inputs.parent_ts, //used here as part of the message_context object + }, + message: + `Your feedback is requested – <${trigger.outputs.trigger_url}|survey now>!`, + }, +); + +// Step 6: Store new survey metadata +CreateSurveyWorkflow.addStep(SaveSurveyFunctionDefinition, { + channel_id: CreateSurveyWorkflow.inputs.channel_id, + parent_ts: CreateSurveyWorkflow.inputs.parent_ts, + reactor_id: CreateSurveyWorkflow.inputs.reactor_id, + trigger_ts: message.outputs.message_context.message_ts, //Referenced here individually + trigger_id: trigger.outputs.trigger_id, + survey_stage: "SURVEY", +}); + +export default CreateSurveyWorkflow; + + +``` +
+ + + +--- + +## Number {#number} +Type: `number` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `minimum` | number | Absolute minimum acceptable value for the number.| +| `maximum` | number | Absolute maximum acceptable value for the number. | +| `enum` | number[] | Constrain the available number options to just the list of numbers denoted in the `enum` property. Usage of `enum` also instructs any UI that collects a value for this parameter to render a dropdown select input rather than a free-form text input.| +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +Declare a `number` type: + + + + +```javascript +// ... +{ + name: "distance", + title: "race distance", + type: Schema.types.number, +} +// ... +``` + + + + +```json +// ... +"distance": { + "title": "race distance", + "type": "number" +} +// ... +``` + + + + +
+Number example + +In this example workflow, we collect a runner's distance and date of their last run. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { LogRunWorkflow } from "../workflows/log_run.ts"; + +const inputForm = LogRunWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Log a run", + interactivity: LogRunWorkflow.inputs.interactivity, + submit_label: "Log run", + fields: { + elements: [ + { + name: "channel", + title: "Channel to send logged run to", + type: Schema.slack.types.channel_id, + default: LogRunWorkflow.inputs.channel, + }, + { + name: "distance", + title: "Distance (in miles)", + type: Schema.types.number, + description: "race distance (in miles)", + minimum: 0, + maximum: 26.2, + }, + { + name: "rundate", + title: "Run date", + type: Schema.slack.types.date, + }, + ], + required: ["channel", "distance", "rundate"], + }, + }, +); +``` +
+ + + +--- + +## OAuth2 {#oauth2} +Type: `slack#/types/credential/oauth2` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare an `oauth2` type: + + + + +```javascript +// ... + githubAccessTokenId: { + type: Schema.slack.types.oauth2, + oauth2_provider_key: "github", + }, +// ... +``` + + + + +```json +// ... +"github_access_token_id": { + "type": "slack#/types/credential/oauth2", + "oauth2_provider_key": "github" +} +// ... +``` + + + + +
+OAuth2 example + +In this example, we use the `oauth2` type for an input parameter in a [custom function](/deno-slack-sdk/guides/creating-custom-functions). To read more about a full implementation of `oauth2`, check out [External authentication](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication). + +```javascript +export const CreateIssueDefinition = DefineFunction({ + callback_id: "create_issue", + title: "Create GitHub issue", + description: "Create a new GitHub issue in a repository", + source_file: "functions/create_issue.ts", + input_parameters: { + properties: { + githubAccessTokenId: { + type: Schema.slack.types.oauth2, + oauth2_provider_key: "github", + }, + url: { + type: Schema.types.string, + description: "Repository URL", + }, + +// ... + + }, + output_parameters: { + properties: { + GitHubIssueNumber: { + type: Schema.types.number, + description: "Issue number", + }, + GitHubIssueLink: { + type: Schema.types.string, + description: "Issue link", + }, + }, + required: ["GitHubIssueNumber", "GitHubIssueLink"], + }, +}); +``` +
+ + + +--- + +## Object {#object} +Type: `object` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +:::warning[Object types are not supported within Workflow Builder at this time] + +If your function will be used within Workflow Builder, we suggest not using the Object types at this time. + +::: + + +Objects can be typed or untyped. Here we have examples of both. + +### Typed Object {#typed-object} + +Refer to [custom types](/deno-slack-sdk/guides/creating-a-custom-type) for more information about typed objects, including properties and how to use [`DefineProperty`](/deno-slack-sdk/guides/creating-a-custom-type#define-property) to enforce required properties. + +Declare a custom `object` type: + + + + +```javascript +// ... +properties: { + reviewer: DefineProperty({ + type: Schema.types.object, + properties: { + login: { type: "string" }, + }, + }), +}, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "reviewer": { + "type": "object", + "properties": { + "login": { "type": "string" } + } + } +} + +//... +``` + + + + +
+Object example + +In this example workflow, we notify authors about updates to their file review status. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +export const ReviewStatusWorkflow = DefineWorkflow({ + callback_id: "review_status_workflow", + title: "Review status", + description: "Review status", + input_parameters: { + properties: { + action: { + type: Schema.types.string, + }, + review_request: { + type: Schema.types.object, + properties: { + number: { type: "integer" }, + title: { type: "string" }, + body: { type: "string" }, + changed_files: { type: "integer" }, + }, + }, + author: { + type: Schema.types.object, + properties: { + login: { type: "string" }, + }, + }, + reviewer: { + type: Schema.types.object, + properties: { + login: { type: "string" }, + }, + }, + }, + required: ["action", "review_request", "author", "reviewer"], + }, +}); +``` +
+ +### Untyped Object {#untyped-object} +Untyped objects do not have properties defined on them. They are malleable; you can assign any kind of properties to them. In TypeScript lingo, these objects are typed as [`any`](https://www.typescriptlang.org/docs/handbook/2/everyday-types.html#any). + +Declare an untyped `object` type: + +```javascript +properties: { + flexibleObject: { + type: Schema.types.object, + } +}, +``` + + +--- + +## Rich text {#rich-text} +Type: `slack#/types/rich_text` + +| Property | Type | Description | +| -------- | --------- | ----------- | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | + +Declare a `rich_text` type: + + + + +```javascript +// ... +elements: [ + { + name: "formattedStringInput", + title: "String input", + type: Schema.slack.types.rich_text, + }, +], +// ... +``` + + + + +```json +// ... +"elements": { + "formattedStringInput": { + "title": "String input", + "type": "slack#/types/rich_text" +} +} +// ... +``` + + + + +
+Rich text example + +In this example workflow, we collect a formatted message from the user using the `rich_text` type. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const TestWorkflow = DefineWorkflow({ + callback_id: "test", + title: "Test", + input_parameters: { + properties: { + channel: { type: Schema.slack.types.channel_id }, + interactivity: { type: Schema.slack.types.interactivity }, + }, + required: ["interactivity"], + }, +}); + +const formData = TestWorkflow.addStep(Schema.slack.functions.OpenForm, { + title: "Send Message Form", + submit_label: "Send Message form", + interactivity: TestWorkflow.inputs.interactivity, + fields: { + required: ["channel", "formattedStringInput"], + elements: [ + { + name: "formattedStringInput", + title: "String input", + type: Schema.slack.types.rich_text, + }, + { + name: "channel", + title: "Post in", + type: Schema.slack.types.channel_id, + default: TestWorkflow.inputs.channel, + }, + ], + }, +}); + +// To share this message object with other users, embed it into a Slack function such as SendMessage. +TestWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: formData.outputs.fields.channel, + message: formData.outputs.fields.formattedStringInput, +}); +``` +
+ + + +--- + +## String {#string} +Type: `string` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `minLength` | number | Minimum number of characters comprising the string. | +| `maxLength` | number | Maximum number of characters comprising the string. | +| `enum` | string[] | Constrain the available string options to just the list of strings denoted in the `enum` property. Usage of `enum` also instructs any UI that collects a value for this parameter to render a dropdown select input rather than a free-form text input.| +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | +| `format` | string | Define accepted format of the string. Valid options include `url` or `email`. | + + + +Declare a `string` type: + + + + +```javascript +// ... +{ + name: "notes", + title: "Notes", + type: Schema.types.string, +}, +// ... +``` + + + + +```json +// ... +"notes": { + "type": "string", + "title": "notes" +} +// ... +``` + + + + +
+String example + +In this example workflow, we use a `string` type to allow a user to add notes about their time off request. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { CreateFTOWorkflow } from "../workflows/create_fto_workflow.ts"; + +const ftoRequestData = CreateFTOWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Request dates off", + description: "Hooray for vacay!", + interactivity: CreateFTOWorkflow.inputs.interactivity, + submit_label: "Submit request", + fields: { + elements: [ + { + name: "start_date", + title: "Start date", + type: Schema.slack.types.date, + }, + { + name: "end_date", + title: "End date", + type: Schema.slack.types.date, + }, + { + name: "notes", + title: "Notes", + description: "Anything to note?", + type: Schema.types.string, + long: true, // renders the input box as a multi-line text box on the form + }, + ], + required: ["start_date", "end_date"], + }, + }, +); +``` +
+ + + +--- + +## Team ID {#team_id} +Type: `slack#/types/team_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +:::warning + +This type is not supported for use in the [OpenForm](/deno-slack-sdk/reference/slack-functions/open_form) Slack function. + +::: + + +Declare a `team_id` type: + + + + +```javascript +// ... +attributes: { + team_id: { + type: Schema.slack.types.team_id, + }, +}, +// ... +``` + + + + +```json +// ... +"team_id": { + "type": "slack#/types/team_id" +} +// ... +``` + + + + +--- + +## Timestamp {#timestamp} +Type: `slack#/types/timestamp` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +Declare a `timestamp` type: + + + + +```javascript +// ... +inputs: { + currentTime: { + value: "{{data.trigger_ts}}", + type: Schema.slack.types.timestamp, + }, +}, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "current_time": { + "type": "slack#/types/timestamp" + } +} + +// ... +``` + + + +
+Timestamp example + +In this example trigger, we call a workflow that logs an incident and the time it occurred. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { Trigger } from "deno-slack-api/types.ts"; + +export const MyWorkflow = DefineWorkflow({ + callback_id: "my_workflow", + title: "My workflow", + input_parameters: { + properties: { + currentTime: { type: Schema.slack.types.timestamp }, + interactivity: { type: Schema.slack.types.interactivity }, + }, + required: [], + }, +}); + +export const incidentTrigger: Trigger = { + type: "shortcut", + name: "Log an incident", + workflow: `#/workflows/${MyWorkflow.definition.callback_id}`, + inputs: { + currentTime: { value: "{{data.trigger_ts}}" }, + interactivity: { value: "{{data.interactivity}}" }, + }, +}; +``` +
+ +--- + +## User context {#usercontext} + +Type: `slack#/types/user_context` + +
+Using user_context in Workflow Builder + +In Workflow Builder, this input type will not have a visible input field and cannot be set manually by a builder + +Instead, the way the value is set is dependent on the situation: + +* **If the workflow starts from an explicit user action (with a link trigger, for example),** then the `user_context` will be passed from the trigger to the function input. If the workflow contains a step that alters the `user_context` value (like a message with a button), then the altered `user_context` value is passed to the function input. + +* **If the workflow starts from something _other_ than an explicit user action (from a scheduled trigger, for example),** then the builder of the workflow must place a step that sets the `user_context` value (like a message with a button). This value will then be passed to the input of the function. + +If a workflow step requires `user_context` and there is no way to ascertain the value within Workflow Builder, the workflow cannot be published. + +
+ +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `id` | string | The `user_id` of the person to which the `user_context` belongs. | +| `secret` | string | A hash used internally by Slack to validate the authenticity of the `id` in the `user_context`. This can be safely ignored, since it's only used by us at Slack to avert malicious actors! | + +Declare the `user_context` type: + + + + +```javascript +// ... +input_parameters: { + properties: { + person_reporting_bug: { + type: Schema.slack.types.user_context, + description: "Which user?", + }, + }, +}, +// ... +``` + + + + +```json +// ... +"input_parameters": { + "person_reporting_bug": { + "type": "slack#/types/user_context", + "description": "Which user?" + } +} +// ... +``` + + + + +
+User context example + +In this example workflow, we use the `Schema.slack.types.user_context` type to report a bug in a system and to collect the reporter's information. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const ReportBugWorkflow = DefineWorkflow({ + callback_id: "report_bug", + title: "Report a Bug", + description: "Report a bug", + input_parameters: { + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + description: "Which channel?", + }, + person_reporting_bug: { + type: Schema.slack.types.user_context, + description: "Which user?", + }, + }, + required: ["person_reporting_bug"], + }, +}); + +import { CreateBugFunction } from "../functions/create_bug.ts"; + +ReportBugWorkflow.addStep( + CreateBugFunction, + { + title: "title", + summary: "summary", + urgency: "S0", + channel_id: ReportBugWorkflow.inputs.channel_id, + creator: ReportBugWorkflow.inputs.person_reporting_bug, + }, +); +``` +
+ + + +--- + +## User ID {#userid} +Type: `slack#/types/user_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described.| An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +Declare a `user_id` type: + + + + +```javascript +// ... +{ + name: "runner", + title: "Runner", + type: Schema.slack.types.user_id, +} +// ... +``` + + + + +```json +// ... +"runner": { + "title": "Runner", + "type": "slack#/types/user_id" +} +// ... +``` + + + +
+User ID example + +In this example workflow, we get a runner's ID and the distance of their logged run. + +```javascript +import { Schema } from "deno-slack-sdk/mod.ts"; +import { RunWorkflow } from "../workflows/run.ts"; + +const inputForm = RunWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Log your run", + interactivity: RunWorkflow.inputs.interactivity, + submit_label: "Submit", + fields: { + elements: [ + { + name: "channel", + title: "Channel to send entry to", + type: Schema.slack.types.channel_id, + default: RunWorkflow.inputs.channel, + }, + { + name: "runner", + title: "Runner", + type: Schema.slack.types.user_id, + }, + { + name: "distance", + title: "Distance (in miles)", + type: Schema.types.number, + }, + ], + required: ["channel", "runner", "distance"], + }, + }, +); +``` +
+ + + +--- + +## Usergroup ID {#usergroupid} +Type: `slack#/types/usergroup_id` + +| Property | Type | Description | +| ---- | --------- | ------------------ | +| `default` | The type that is being described. | An optional parameter default value. | +| `description` | string |An optional parameter description. | +| `examples` | An array of the type being described. | An optional list of examples. +| `hint` | string | An optional parameter hint. | +| `title` | string |An optional parameter title. | +| `type` | string |String that defines the parameter type. | +| `choices` | EnumChoice[] | Defines labels that correspond to the `enum` values. See below. | + + + +Declare a `usergroup_id` type: + + + + +```javascript +// ... +attributes: { + usergroup_id: { + type: Schema.slack.types.usergroup_id, + }, +}, +// ... +``` + + + + +```json +// ... +"usergroup_id": { + "type": "slack#/types/usergroup_id" +} +// ... +``` + + + + +
+Usergroup ID example + +In this example datastore definition, we store work shift details for a team. + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const MyShifts = DefineDatastore({ + name: "shifts", + primary_key: "id", + attributes: { + id: { type: Schema.types.string }, + team_id: { type: Schema.types.team_id }, + channel: { type: Schema.slack.types.channel_id }, + usergroup_id: { type: Schema.slack.types.usergroup_id }, + shiftRotation: { type: Schema.types.string }, + }, +}); +``` +
\ No newline at end of file diff --git a/docs/tutorials/announcement-bot.md b/docs/tutorials/announcement-bot.md new file mode 100644 index 00000000..cb4646c0 --- /dev/null +++ b/docs/tutorials/announcement-bot.md @@ -0,0 +1,1074 @@ +# Announcement bot + + + +Hear ye, hear ye! + +In this tutorial, you will learn how to create an app for an announcement bot that helps users draft, edit, and post an announcement to a channel (or channels) in a user's workspace, all while exploring the following workflow app concepts: + +* [Custom](/deno-slack-sdk/guides/creating-custom-functions) and [built-in](/deno-slack-sdk/guides/creating-slack-functions) functions +* [Datastores](/deno-slack-sdk/guides/using-datastores) +* [Workflows](/deno-slack-sdk/guides/creating-workflows) +* [Custom types](/deno-slack-sdk/guides/creating-a-custom-type) +* [Triggers](/deno-slack-sdk/guides/creating-link-triggers) + +For an overview of how the final product will look and function, check out the demo video in the `README.md` of the [GitHub repo](https://github.com/slack-samples/deno-announcement-bot) for this project. + +Each Slack app built using the CLI begins with the same steps. Make sure you have everything you need before you call the attention of the masses to deliver your announcement. + +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started guide](/deno-slack-sdk/guides/getting-started), then come on back. + + +## Choose your adventure + +Once those items are complete, you have two possible ways to proceed. + +### Use a blank app +You can create a blank app with the Slack CLI using the following command: + +``` +slack create announcement-bot-app --template https://github.com/slack-samples/deno-blank-template +``` + +### Use a pre-built app +Or, you can use the pre-built [Announcement Bot app](https://github.com/slack-samples/deno-announcement-bot): + +``` +slack create announcement-bot-app --template https://github.com/slack-samples/deno-announcement-bot +``` + +Whichever option you choose, be sure to have the sample app repo open for reference, since we won't cover every file here, for brevity's sake. + +Once your new project is ready to go, navigate to your project directory and let's get this show on the road. + +## Plan your app + +The best way to go about creating a workflow app is to take a bird’s eye view of it and determine: +* What would you like this workflow to accomplish? +* What is your goal? +* How can you break that down into smaller steps and actions? + +This is how we will go about showing you this app’s creation, and we think it’s the best way to create workflow apps. To begin, the idea: an app that assists users in sending an announcement to a number of channels. But wait! Just in case they prematurely send it (if that’s you, check out [this help article](https://slack.com/help/articles/115005523006-Set-your-Enter-key-preference) or perhaps [this one](https://slack.com/help/articles/202395258-Edit-or-delete-messages#unsend-a-message)), let’s allow the user to preview and edit the announcement before making it final. + +### How can we break this down into smaller steps? +If we think about the flow of the app, here's what happens. The user: +* initiates a workflow +* fills out a form and submits it +* sees a preview where they can edit the message they drafted +* sends the message +* sees a summary posted by the app + +The first action will be handled by a [trigger](/deno-slack-sdk/guides/creating-link-triggers) and the rest will be handled by [functions](/deno-slack-sdk/guides/creating-custom-functions). The execution of the functions will be chained together in a [workflow](/deno-slack-sdk/guides/creating-workflows), which essentially dictates which actions happen in which order. Along the way, we'll also add some visual sprinkles to sweeten the app's appearance in the form of [blocks](https://api.slack.com/reference/block-kit) and talk about the [app manifest](/deno-slack-sdk/guides/using-the-app-manifest). + +Ready to unroll your proverbial scroll, gather the masses, and announce your next big message? Buckle up and let’s dive in. + + + +## Define and implement the workflow + +The [`create_announcement.ts`](https://github.com/slack-samples/deno-announcement-bot/blob/main/workflows/create_announcement.ts) workflow file will give us an idea of the flow of actions. + +The first step to creating a workflow is to define it. +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { CreateDraftFunctionDefinition } from "../functions/create_draft/definition.ts"; +import { PostSummaryFunctionDefinition } from "../functions/post_summary/definition.ts"; +import { PrepareSendAnnouncementFunctionDefinition } from "../functions/send_announcement/definition.ts"; + +const CreateAnnouncementWorkflow = DefineWorkflow({ + callback_id: "create_announcement", + title: "Create an announcement", + description: + "Create and send an announcement to one or more channels in your workspace.", + input_parameters: { + properties: { + created_by: { + type: Schema.slack.types.user_id, + }, + interactivity: { + type: Schema.slack.types.interactivity, + }, + }, + required: ["created_by", "interactivity"], + }, +}); +``` + +This definition tells us that we will use an `interactivity` input parameter. This means that we will be requiring interaction from the user, in this case, to find out information about their announcement. Continuing on: + +```javascript +// Step 1: Open a form to create an announcement using Slack function, OpenForm +// For more on Slack functions +// https://api.slack.com/reference/slack-functions +const formStep = CreateAnnouncementWorkflow + .addStep(Schema.slack.functions.OpenForm, { + title: "Create an announcement", + description: + "Create a draft announcement. You will have the opportunity to preview & edit it in channel before sending.\n\n_Want to create a richer announcement? Use and paste the full payload into the message input below._", + interactivity: CreateAnnouncementWorkflow.inputs.interactivity, + submit_label: "Preview", + fields: { + elements: [{ + name: "message", + title: "Message", + type: Schema.types.string, + description: "Compose your message using plain text, mrkdwn, or blocks", + long: true, + }, { + name: "channels", + title: "Destination channel(s)", + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The channels where your announcement will be posted", + }, { + name: "channel", + title: "Draft channel", + type: Schema.slack.types.channel_id, + description: + "The channel where you and your team can preview & edit the announcement before sending", + }, { + name: "icon", + title: "Custom emoji icon", + type: Schema.types.string, + description: + "Emoji to override the default app icon. Must use the format :robot_face: to be applied correctly.", + }, { + name: "username", + title: "Custom username", + type: Schema.types.string, + description: "Name to override the default app name", + }], + required: ["message", "channels", "channel"], + }, + }); +``` + +The first step you see here is the `OpenForm` [Slack function](/deno-slack-sdk/reference/slack-functions/open_form), which handles collecting input from the user. You might be wondering why we defined this step as a variable called `formStep` instead of adding the step to the workflow, which is also a viable option. If you want to use any information collected in this function, you will need to store it in a variable to retrieve it later. We'll see this in the next step, where we'll add a step to handle drafting the announcement. + +```javascript +const draftStep = CreateAnnouncementWorkflow.addStep( + CreateDraftFunctionDefinition, + { + created_by: CreateAnnouncementWorkflow.inputs.created_by, + message: formStep.outputs.fields.message, + channels: formStep.outputs.fields.channels, + channel: formStep.outputs.fields.channel, + icon: formStep.outputs.fields.icon, + username: formStep.outputs.fields.username, + }, +); +``` + + Notice how we're now accessing the information stored in the previous step's function via the `formStep.outputs.fields` property. This is how you pass data between functions. Don't worry about the particulars of `CreateDraftFunctionDefinition` just yet; we'll get to that in a bit. For now, we're just mapping out the flow of the entire app. Next, we'll send the announcement. + +```javascript +// Step 3: Send announcement(s) +const sendStep = CreateAnnouncementWorkflow.addStep( + PrepareSendAnnouncementFunctionDefinition, + { + message: draftStep.outputs.message, + channels: formStep.outputs.fields.channels, + icon: formStep.outputs.fields.icon, + username: formStep.outputs.fields.username, + draft_id: draftStep.outputs.draft_id, + }, +); +``` + +As you can see, we've added another step to the workflow; this one is called `PrepareSendAnnouncementFunctionDefinition`, in which we're using data collected from both the function stored in `formStep` and `draftStep`. One final step: + +```javascript +// Step 4: Post message summary of announcement +CreateAnnouncementWorkflow.addStep(PostSummaryFunctionDefinition, { + announcements: sendStep.outputs.announcements, + channel: formStep.outputs.fields.channel, + message_ts: draftStep.outputs.message_ts, +}); +``` + +This step sends a summary of the posted announcement using data from all three prior steps. Looking back on our code, we've defined a workflow and added steps to gather data from the user, draft the announcement, send it, and post a summary of it. Satisfied with this flow, we have one more line to add to our workflow file: + +```javascript +export default CreateAnnouncementWorkflow; +``` + +Awesome. Let's check out the particulars of these functions in the next section. + + +## Define and implement functions + +Now that we know the goal of what we're building and how we'll break it down, let's take a closer look at those different functions we identified in the workflow: +* `OpenForm` +* `CreateDraftFunctionDefinition` +* `PrepareSendAnnouncementFunctionDefinition` +* `PostSummaryFunctionDefinition` + +### Creating a form + +The `OpenForm` function is a [Slack function](https://api.slack.com/reference/slack-functions/open_form) that allows us to collect information from the user. We saw its definition in the workflow, but let's look at it again here: + +```javascript +// This function exists in /workflows/create_announcement.ts +const formStep = CreateAnnouncementWorkflow + .addStep(Schema.slack.functions.OpenForm, { + title: "Create an announcement", + description: + "Create a draft announcement. You will have the opportunity to preview & edit it in channel before sending.\n\n_Want to create a richer announcement? Use and paste the full payload into the message input below._", + interactivity: CreateAnnouncementWorkflow.inputs.interactivity, + submit_label: "Preview", + fields: { + elements: [{ + name: "message", + title: "Message", + type: Schema.types.string, + description: "Compose your message using plain text, mrkdwn, or blocks", + long: true, + }, { + name: "channels", + title: "Destination channel(s)", + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The channels where your announcement will be posted", + }, { + name: "channel", + title: "Draft channel", + type: Schema.slack.types.channel_id, + description: + "The channel where you and your team can preview & edit the announcement before sending", + }, { + name: "icon", + title: "Custom emoji icon", + type: Schema.types.string, + description: + "Emoji to override the default app icon. Must use the format :robot_face: to be applied correctly.", + }, { + name: "username", + title: "Custom username", + type: Schema.types.string, + description: "Name to override the default app name", + }], + required: ["message", "channels", "channel"], + }, + }); +``` + +We've defined the necessary inputs - `title`, `description`, `interactivity`, `submit_label`, and `fields`. The `fields` property represents what information we want to collect from the user. As you can see, we've required the user to minimally include `message`, `channels` to which to send the message, and a `channel` where the draft should post. + +### Create a draft + +Our next function, `CreateDraftFunctionDefinition`, is a [custom function](/deno-slack-sdk/guides/creating-custom-functions). If you're following along in the sample code, navigate to the `/functions/create_draft` directory. For easier reading, we've split the function into three files - `definition.ts`, `handler.ts`, and `blocks.ts`. Technically speaking, you could combine the `definition.ts` and `handler.ts` files, but we like the visual separation in order to keep files shorter and easier to digest. Let's dive into the definition. + +```javascript +// /functions/create_draft/definition.ts + +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const CREATE_DRAFT_FUNCTION_CALLBACK_ID = "create_draft"; +export const CreateDraftFunctionDefinition = DefineFunction({ + callback_id: CREATE_DRAFT_FUNCTION_CALLBACK_ID, + title: "Create a draft announcement", + description: + "Creates and sends an announcement draft to channel for review before sending", + source_file: "functions/create_draft/handler.ts", + input_parameters: { + properties: { + created_by: { + type: Schema.slack.types.user_id, + description: "The user that created the announcement draft", + }, + message: { + type: Schema.types.string, + description: "The text content of the announcement", + }, + channel: { + type: Schema.slack.types.channel_id, + description: "The channel where the announcement will be drafted", + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The channels where the announcement will be posted", + }, + icon: { + type: Schema.types.string, + description: "Optional custom bot icon to use display in announcements", + }, + username: { + type: Schema.types.string, + description: "Optional custom bot emoji avatar to use in announcements", + }, + }, + required: [ + "created_by", + "message", + "channel", + "channels", + ], + }, + output_parameters: { + properties: { + draft_id: { + type: Schema.types.string, + description: "Datastore identifier for the draft", + }, + message: { + type: Schema.types.string, + description: "The content of the announcement", + }, + message_ts: { + type: Schema.types.string, + description: "The timestamp of the draft message in the Slack channel", + }, + }, + required: ["draft_id", "message", "message_ts"], + }, +}); +``` + +This file defines the function's six `input_parameters` and their types (four of which are required), as well as its three `output_parameters`, all of which are required. Let's check out what this function does in `handler.ts`, starting with just the first part: + +```javascript +// /functions/create_draft/handler.ts + +import { SlackFunction } from "deno-slack-sdk/mod.ts"; + +import { CreateDraftFunctionDefinition } from "./definition.ts"; +import { buildDraftBlocks } from "./blocks.ts"; +import { + confirmAnnouncementForSend, + openDraftEditView, + prepareSendAnnouncement, + saveDraftEditSubmission, +} from "./interactivity_handler.ts"; +import { ChatPostMessageParams, DraftStatus } from "./types.ts"; + +import DraftDatastore from "../../datastores/drafts.ts"; + +/** + * This is the handling code for the CreateDraftFunction. It will: + * 1. Create a new datastore record with the draft + * 2. Build a Block Kit message with the draft and send it to input channel + * 3. Update the draft record with the successful sent drafts timestamp + * 4. Pause function completion until user interaction + */ +export default SlackFunction( + CreateDraftFunctionDefinition, + async ({ inputs, client }) => { + const draftId = crypto.randomUUID(); + + // 1. Create a new datastore record with the draft + const putResp = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + // @ts-ignore expected fix in future release - otherwise missing non-required items throw type error + item: { + id: draftId, + created_by: inputs.created_by, + message: inputs.message, + channels: inputs.channels, + channel: inputs.channel, + icon: inputs.icon, + username: inputs.username, + status: DraftStatus.Draft, + }, + }); + + if (!putResp.ok) { + const draftSaveErrorMsg = + `Error saving draft announcement. Contact the app maintainers with the following information - (Error detail: ${putResp.error})`; + console.log(draftSaveErrorMsg); + + return { error: draftSaveErrorMsg }; + } +``` + +Here we see the `SlackFunction` defined with the `CreateDraftFunctionDefinition` we previously explored. `SlackFunction` is the necessary mechanism we need to use in order to interact with the [SlackAPI](/deno-slack-sdk/guides/calling-slack-api-methods), via the `client` property. + +We haven't yet covered the datastore setup, so file that away in your brain for now; just know that it's a place where we'll store and retrieve data for this app's use. Take a minute to notice those properties on `item`. Hey those look familiar! `message`, `channels`, `channel`, `icon`, and `username` were all the inputs we collected in the `OpenForm` function step of the workflow. Next up: Build a Block Kit message with draft announcement, and send it to the input channel. + +```javascript + // 2. Build a Block Kit message with draft announcement and send it to input channel + const blocks = buildDraftBlocks( + draftId, + inputs.created_by, + inputs.message, + inputs.channels, + ); + + const params: ChatPostMessageParams = { + channel: inputs.channel, + blocks: blocks, + text: `An announcement draft was posted`, + }; + + if (inputs.icon) { + params.icon_emoji = inputs.icon; + } + + if (inputs.username) { + params.username = inputs.username; + } + + const postDraftResp = await client.chat.postMessage(params); + if (!postDraftResp.ok) { + const draftPostErrorMsg = + `Error posting draft announcement to ${params.channel}. Contact the app maintainers with the following information - (Error detail: ${postDraftResp.error})`; + console.log(draftPostErrorMsg); + + return { error: draftPostErrorMsg }; + } +``` + +This step handles posting the draft announcement given the message we collected from the user to the draft channel they requested, through the [`postMessage`](https://api.slack.com/methods/chat.postMessage) API method. Read more about how blocks work [over here](https://api.slack.com/block-kit). Next up, let's update that draft record: + +```javascript + + // 3. Update the draft record with the successful sent drafts timestamp + const putResp2 = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + // @ts-expect-error expecting fix in future SDK release + item: { + id: draftId, + message_ts: postDraftResp.ts, + }, + }); + + if (!putResp2.ok) { + const draftUpdateErrorMsg = + `Error updating draft announcement timestamp for ${draftId}. Contact the app maintainers with the following information - (Error detail: ${putResp2.error})`; + console.log(draftUpdateErrorMsg); + + return { error: draftUpdateErrorMsg }; + } + + /** + * IMPORTANT! Set `completed` to false in order to pause function's complete state + * since we will wait for user interaction in the button handlers below. + * Steps after this step in the workflow will not execute until we + * complete our function. + */ + return { completed: false }; + }, +).addBlockActionsHandler( + /** + * These are additional interactivity handlers for events triggered + * by a users interaction with Block Kit elements: + */ + "preview_overflow", + openDraftEditView, +).addViewSubmissionHandler( + "edit_message_modal", + saveDraftEditSubmission, +).addBlockActionsHandler( + "send_button", + confirmAnnouncementForSend, +).addViewSubmissionHandler( + "confirm_send_modal", + prepareSendAnnouncement, +); +``` + +That last bit is important - we're pausing the function's completion in order to wait for an edit or confirmation to send. + +### Send the announcement + +Let's take a look at the next function, located in `/functions/send_announcement`. Here's its definition: + +```javascript +// /functions/send_announcement/definition.ts + +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; +import { AnnouncementCustomType } from "../post_summary/types.ts"; + +export const SEND_ANNOUNCEMENT_FUNCTION_CALLBACK_ID = "send_announcement"; +export const PrepareSendAnnouncementFunctionDefinition = DefineFunction({ + callback_id: SEND_ANNOUNCEMENT_FUNCTION_CALLBACK_ID, + title: "Send an announcement", + description: "Sends a message to one or more channels", + source_file: "functions/send_announcement/handler.ts", + input_parameters: { + properties: { + message: { + type: Schema.types.string, + description: "The content of the announcement", + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + description: "The destination channels of the announcement", + }, + icon: { + type: Schema.types.string, + description: "Optional custom bot icon to use display in announcements", + }, + username: { + type: Schema.types.string, + description: "Optional custom bot emoji avatar to use in announcements", + }, + draft_id: { + type: Schema.types.string, + description: "The datastore ID of the draft message if one was created", + }, + }, + required: [ + "message", + "channels", + ], + }, + output_parameters: { + properties: { + announcements: { + type: Schema.types.array, + items: { + type: AnnouncementCustomType, + }, + description: + "Array of objects that includes a channel ID and permalink for each announcement successfully sent", + }, + }, + required: ["announcements"], + }, +}); +``` + +Whoa, what's that `AnnouncementCustomType`?! Nothing to worry about, my dearest Slack dev. That's a [custom type](/deno-slack-sdk/guides/creating-a-custom-type) that workflow apps allow us to define to suit our specialized needs. We can see its definition over in `/functions/post_summary/types.ts`: + +```javascript +// /functions/post_summary/types.ts + +import { DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const AnnouncementCustomType = DefineType({ + name: "Announcement", + type: Schema.types.object, + properties: { + channel_id: { + type: Schema.slack.types.channel_id, + }, + success: { + type: Schema.types.boolean, + }, + permalink: { + type: Schema.types.string, + }, + error: { + type: Schema.types.string, + }, + }, + required: ["channel_id", "success"], +}); + +/** + * Corresponding TS typing for use elsewhere + */ +export type AnnouncementType = { + channel_id: string; + success: boolean; + permalink?: string; + error?: string; +}; +``` + +Let's check out the implementation of that function to see how it sends the announcement: + +```javascript +// /functions/send_announcement/handler.ts + +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import { SlackAPIClient } from "deno-slack-api/types.ts"; + +import { PrepareSendAnnouncementFunctionDefinition } from "./definition.ts"; +import { buildAnnouncementBlocks, buildSentBlocks } from "./blocks.ts"; + +import { AnnouncementType } from "../post_summary/types.ts"; +import { ChatPostMessageParams, DraftStatus } from "../create_draft/types.ts"; + +import DraftDatastore from "../../datastores/drafts.ts"; +import AnnouncementsDatastore from "../../datastores/announcements.ts"; + +/** + * This is the handling code for PrepareSendAnnouncementFunction. It will: + * 1. Send announcement to each channel supplied + * 2. Updates the status of the announcement in the + */ + +export default SlackFunction( + PrepareSendAnnouncementFunctionDefinition, + async ({ inputs, client }) => { + // Array to gather chat.postMessage responses + // deno-lint-ignore no-explicit-any + const chatPostMessagePromises: Promise[] = []; + + // Incoming draft_id to link all announcements that are + // part of the same draft. If a draft_id was not provided, + // create a new identifier for this announcements. + const draft_id = inputs.draft_id || crypto.randomUUID(); + + const blocks = buildAnnouncementBlocks(inputs.message); + + for (const channel of inputs.channels) { + const params: ChatPostMessageParams = { + channel: channel, + blocks: blocks, + text: `An announcement was posted`, + }; + + if (inputs.icon) { + params.icon_emoji = inputs.icon; + } + + if (inputs.username) { + params.username = inputs.username; + } + + const announcementRes = sendAndSaveAnnouncement(params, draft_id, client); + chatPostMessagePromises.push(announcementRes); + } + + const announcements = await Promise.all(chatPostMessagePromises); + + // Update draft if one was created + if (inputs.draft_id) { + const { item } = await client.apps.datastore.put< + typeof DraftDatastore.definition + >({ + datastore: DraftDatastore.name, + // @ts-ignore expected fix in future release - otherwise missing non-required items throw type error + item: { + id: inputs.draft_id, + status: DraftStatus.Sent, + }, + }); + + const blocks = buildSentBlocks( + item.created_by, + inputs.message, + inputs.channels, + ); + + await client.chat.update({ + channel: item.channel, + ts: item.message_ts, + blocks: blocks, + }); + } + + return { outputs: { announcements: announcements } }; + }, +); +``` + +Here we see the message being posted as well as updating the draft in the datastore. Continuing on: + +```javascript +/** + * This method send an announcement to a channel, gets its permalink, and stores the details in the datastore + * @param params parameters used in the chat.postMessage request + * @param draft_id ID of the draft announcement that is being posted + * @returns promise with summary + */ + +async function sendAndSaveAnnouncement( + params: ChatPostMessageParams, + draft_id: string, + client: SlackAPIClient, +): Promise { + let announcement: AnnouncementType; + + // Send it + const post = await client.chat.postMessage(params); + + if (post.ok) { + console.log(`Sent to ${params.channel}`); + + // Get permalink to message for use in summary + const { permalink } = await client.chat.getPermalink({ + channel: params.channel, + message_ts: post.ts, + }); + + announcement = { + channel_id: params.channel, + success: true, + permalink: permalink, + }; + } // There was an error sending the announcement + else { + console.log(`Error sending to ${params.channel}: ${post.error}`); + announcement = { + channel_id: params.channel, + success: false, + error: post.error, + }; + } + + // Save each announcement to DB even if there was an error posting + await client.apps.datastore.put({ + datastore: AnnouncementsDatastore.name, + item: { + id: crypto.randomUUID(), + draft_id: draft_id, + success: post.ok, + error_message: post.error, + channel: post.channel, + message_ts: post.ts, + }, + }); + + return announcement; +} +``` + +You might be wondering why this function returns the announcement if it's already been posted and updated in the datastore. Ah, but remember back to the app workflow when we had one final step of the flow? We post the announcement and then we post a summary of the announcement to the user who initiated the workflow. We'll use that output in our final function. Read on to see how. + +### Post summary +Now let's head on over to `/functions/post_summary` to check out the function's definition in `definition.ts`: + +```javascript +// /functions/post_summary/definition.ts + +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; +import { AnnouncementCustomType } from "./types.ts"; + +export const POST_ANNOUNCEMENT_FUNCTION_CALLBACK_ID = "post_summary"; + +export const PostSummaryFunctionDefinition = DefineFunction({ + callback_id: POST_ANNOUNCEMENT_FUNCTION_CALLBACK_ID, + title: "Post announcement summary", + description: "Post a summary of all sent announcements ", + source_file: "functions/post_summary/handler.ts", + input_parameters: { + properties: { + announcements: { + type: Schema.types.array, + items: { + type: AnnouncementCustomType, + }, + description: + "Array of objects that includes a channel ID and permalink for each announcement successfully sent", + }, + channel: { + type: Schema.slack.types.channel_id, + description: "The channel where the summary should be posted", + }, + message_ts: { + type: Schema.types.string, + description: + "Options message timestamp where the summary should be threaded", + }, + }, + required: [ + "announcements", + "channel", + ], + }, + output_parameters: { + properties: { + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + }, + required: ["channel", "message_ts"], + }, +}); +``` + +Notice the input parameters include an array of `announcements`. That is sourced from the `output_parameters` of the function executed prior to this one in the workflow - `PrepareSendAnnouncementFunctionDefinition`. Now that we've drafted, edited, and sent the announcement to the requested channels, we can use the last function to post a summary. Here's what that definition looks like: + +```javascript +// /functions/post_summary/definition.ts +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; +import { AnnouncementCustomType } from "./types.ts"; + +export const POST_ANNOUNCEMENT_FUNCTION_CALLBACK_ID = "post_summary"; + +export const PostSummaryFunctionDefinition = DefineFunction({ + callback_id: POST_ANNOUNCEMENT_FUNCTION_CALLBACK_ID, + title: "Post announcement summary", + description: "Post a summary of all sent announcements ", + source_file: "functions/post_summary/handler.ts", + input_parameters: { + properties: { + announcements: { + type: Schema.types.array, + items: { + type: AnnouncementCustomType, + }, + description: + "Array of objects that includes a channel ID and permalink for each announcement successfully sent", + }, + channel: { + type: Schema.slack.types.channel_id, + description: "The channel where the summary should be posted", + }, + message_ts: { + type: Schema.types.string, + description: + "Options message timestamp where the summary should be threaded", + }, + }, + required: [ + "announcements", + "channel", + ], + }, + output_parameters: { + properties: { + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + }, + required: ["channel", "message_ts"], + }, +}); +``` + +Refer back to the workflow definition at any point to see where this function, or any given function, is receiving its input parameters, as well as where its output parameters go. Let's continue on to the handler of this function: + +```javascript +// /functions/post_summary/handler.ts + +import { SlackFunction } from "deno-slack-sdk/mod.ts"; + +import { buildSummaryBlocks } from "./blocks.ts"; +import { PostSummaryFunctionDefinition } from "./definition.ts"; + +/** + * This is the handling code for PostSummaryFunction. It will: + * 1. Post a message in thread to the draft announcement message + * with a summary of announcement's sent + * 2. Complete this function with either required outputs or an error + */ +export default SlackFunction( + PostSummaryFunctionDefinition, + async ({ inputs, client }) => { + const blocks = buildSummaryBlocks(inputs.announcements); + + // 1. Post a message in thread to the draft announcement message + const postResp = await client.chat.postMessage({ + channel: inputs.channel, + thread_ts: inputs.message_ts || "", + blocks: blocks, + unfurl_links: false, + }); + if (!postResp.ok) { + const summaryTS = postResp ? postResp.ts : "n/a"; + const postSummaryErrorMsg = + `Error posting announcement send summary: ${summaryTS} to channel: ${inputs.channel}. Contact the app maintainers with the following - (Error detail: ${postResp.error})`; + console.log(postSummaryErrorMsg); + + // 2. Complete function with an error message + return { error: postSummaryErrorMsg }; + } + + const outputs = { + channel: inputs.channel, + message_ts: postResp.ts, + }; + + // 2. Complete function with outputs + return { outputs: outputs }; + }, +); +``` + +This function handles sending a summary to the user of the announcement they drafted and sent to their selected channels. This concludes our dive into the functions of this app. Let's take a deeper look at the datastores holding our announcement data in the next section. + + +## Define datastores + +Two different datastores are needed for this application - one for drafts and one for announcements. + +### Drafts datastore + +Let's first navigate to `/datastores` to check out the contents of `drafts.ts`: + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export default DefineDatastore({ + name: "drafts", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + created_by: { + type: Schema.slack.types.user_id, + }, + message: { + type: Schema.types.string, + }, + channels: { + type: Schema.types.array, + items: { + type: Schema.slack.types.channel_id, + }, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + icon: { + type: Schema.types.string, + }, + username: { + type: Schema.types.string, + }, + status: { + type: Schema.types.string, // possible statuses are draft, sent + }, + }, +}); +``` + +You can read more about interacting with datastores on the [datastores page](/deno-slack-sdk/guides/using-datastores), but the gist of it is that it's a Slack-hosted place to store data. In this definition, we see that this particular datastore has nine attributes - what you might consider fields in a database. Each attribute's type is listed along with its definition above. + +### Announcements datastore + +The second datastore - `announcements` - is defined in `/datastores/announcements.ts` and looks like this: + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export default DefineDatastore({ + name: "announcements", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + draft_id: { + type: Schema.types.string, + }, + success: { + type: Schema.types.boolean, + }, + error_message: { + type: Schema.types.string, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + message_ts: { + type: Schema.types.string, + }, + }, +}); +``` + +If you refer back to `/functions/send_announcement/handler.ts` you'll see that this datastore is used to keep a record of all announcements, regardless of their success status. To play around more with datastores outside of your code, check out the datastore [commands](/slack-cli/reference/commands/slack_datastore). + + +## Kick things off with a trigger + +Triggers invoke workflows, so they're pretty important. In this app we'll be using a [link trigger](/deno-slack-sdk/guides/creating-link-triggers), but you should know there are four types of [triggers](/deno-slack-sdk/guides/using-triggers) available. Navigate to `/triggers/create_announcement.ts` and let's check it out. + +```javascript +import { Trigger } from "deno-slack-api/types.ts"; +import CreateAnnouncementWorkflow from "../workflows/create_announcement.ts"; + +const trigger: Trigger< + typeof CreateAnnouncementWorkflow.definition +> = { + type: "shortcut", + name: "Create an announcement", + description: + "Create and send an announcement to one or more channels in your workspace.", + workflow: "#/workflows/create_announcement", + inputs: { + created_by: { + value: "{{data.user_id}}", + }, + interactivity: { + value: "{{data.interactivity}}", + }, + }, +}; + +export default trigger; +``` + +Next, run the trigger command in the terminal: + +```bash +slack trigger create --trigger-def triggers/create_announcement.ts +``` + +After executing the command, select your app and workspace. The terminal will output a link called a "Shortcut URL", also known as your link trigger. Save that URL; we'll use it later. If you ever lose track of that URL, you can always run the command `slack triggers -info` and select your workspace to find it again. + + +## Report app contents in the app manifest + +We've got one last stop to highlight before running this application, and that is the [app manifest](/deno-slack-sdk/guides/using-the-app-manifest). The manifest is located in the root directory of your project. Navigating to it in the sample project, its contents look like this: + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; +import AnnouncementDatastore from "./datastores/announcements.ts"; +import DraftDatastore from "./datastores/drafts.ts"; +import { AnnouncementCustomType } from "./functions/post_summary/types.ts"; +import CreateAnnouncementWorkflow from "./workflows/create_announcement.ts"; + +export default Manifest({ + name: "Announcement Bot", + description: "Send an announcement to one or more channels", + icon: "assets/icon.png", + outgoingDomains: ["cdn.skypack.dev"], + datastores: [DraftDatastore, AnnouncementDatastore], + types: [AnnouncementCustomType], + workflows: [ + CreateAnnouncementWorkflow, + ], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "chat:write.customize", + "datastore:read", + "datastore:write", + ], +}); +``` + +The app manifest is the app's configuration. It is very important that this file is structured correctly in order for your app to run smoothly. Each function, custom type, and datastore defined in an app must be declared in the manifest file. + + +## Deploy your app + +Ready to see this thing in action? Let's use development mode to run this workflow in Slack. Start it off with this command in your terminal: + +```bash +slack run +``` + +After you've chosen your app and assigned it to your workspace, you can switch over to the app in Slack and test it out. Remember the link trigger you created earlier? Copy and paste that URL in a message to yourself in Slack. It will unfurl into a button that you can click to initiate the workflow. + + +## You did it! + +Awww yea, you did it! You made it through the tutorial and are successfully sending announcements (but, you know, not too many...) to your workspace channels. Great job! + +## Next steps + +For your next challenge, perhaps consider creating [a bot to welcome users to your workspace](/deno-slack-sdk/tutorials/welcome-bot)! \ No newline at end of file diff --git a/docs/tutorials/definition-bot.md b/docs/tutorials/definition-bot.md new file mode 100644 index 00000000..492a9382 --- /dev/null +++ b/docs/tutorials/definition-bot.md @@ -0,0 +1,568 @@ +# Definition bot + + + +Have you ever found yourself in a company or position where it feels like everyone around you is speaking in a language you don’t understand? Where the use of so many acronyms has you drowning in an alphabet soup of obscured meaning? In this tutorial, we’ll walk you through using a [trigger](/deno-slack-sdk/guides/creating-link-triggers), [workflow](/deno-slack-sdk/guides/creating-workflows), [custom function](/deno-slack-sdk/guides/creating-custom-functions), [datastore](/deno-slack-sdk/guides/using-datastores), and [modal view interactivity](/deno-slack-sdk/guides/creating-an-interactive-modal) to create a workflow app that serves as a crowdsourced glossary of acronyms and team vernacular to help you talk the talk while you walk the walk. + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + + +## Get started + +Let’s get things started by creating a blank app via the CLI. Run the following command in your terminal. + +```bash +slack create define-app --template https://github.com/slack-samples/deno-blank-template +``` + +Next, navigate to the project directory and open it in the code editor of your choice; we like Visual Studio Code. + + +## Plan your app + +Let’s think about the flow of logic in our app. + +We'll need a [trigger](/deno-slack-sdk/guides/using-triggers) to set things in motion, and a [modal](/deno-slack-sdk/guides/creating-an-interactive-modal) to open and ask for a term that the user would like defined. From there, we'll take that term and search for it in a [datastore](/deno-slack-sdk/guides/using-datastores). If it is found, we'll deliver that definition to the user in an updated modal. If it is not found, we'll ask the user if they would like to submit a definition for it. + +Because all of these actions will be done with view updates in the same modal, we’ll use one [function](/deno-slack-sdk/guides/creating-custom-functions) with a few view handlers. We’ll also need one [workflow](/deno-slack-sdk/guides/creating-workflows) and one [trigger](/deno-slack-sdk/guides/creating-link-triggers). Let’s start this out by creating the [function](/deno-slack-sdk/guides/creating-custom-functions), the main meat of the app. + + +## Write the custom function + +Our function will handle the bulk of the logic in this app. Because all of the interaction will be within one modal pop-up, we’ll keep all the logic in one function (as opposed to breaking it out into separate functions strung together by the workflow). Create a folder called `functions` and a file within it, `term_lookup_function.ts`. First we define the function, laying out the expected inputs and outputs. + +```javascript +// term_lookup_function.ts + +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { // don’t worry about these for now, we’ll talk about them in a later step + showConfirmationView, + showDefinitionSubmissionView, + showDefinitionView, +} from "./interactivity_handler.ts"; + +export const TermLookupFunction = DefineFunction({ + callback_id: "term_lookup_function", + title: "Define a term", + source_file: "functions/term_lookup_function.ts", + input_parameters: { + properties: { interactivity: { type: Schema.slack.types.interactivity } }, + required: ["interactivity"], + }, + output_parameters: { properties: {}, required: [] }, +}); +``` + +The `interactivity` input parameter is essential for allowing the modal to first appear, as well as the subsequent user interactions to happen. `interactivity` gives the app permission to do these actions because the user initiated it. Without this parameter, modal interaction cannot take place. No output parameters are needed because all actions will take place within this function; we will not be passing data to another function. Now, the function implementation. Place this in the same file, after the function definition: + +```javascript +export default SlackFunction( + TermLookupFunction, + async ({ inputs, client }) => { + const response = await client.views.open({ + interactivity_pointer: inputs.interactivity.interactivity_pointer, + view: { + "type": "modal", + "callback_id": "first-page", + "notify_on_close": false, + "title": { "type": "plain_text", "text": "Search for a definition" }, + "submit": { "type": "plain_text", "text": "Search" }, + "close": { "type": "plain_text", "text": "Close" }, + "blocks": [ + { + "type": "input", + "block_id": "term", + "element": { "type": "plain_text_input", "action_id": "action" }, + "label": { "type": "plain_text", "text": "Term" }, + }, + ], + }, + }); + if (response.error) { + const error = + `Failed to open a modal in the term lookup workflow. Contact the app maintainers with the following information - (error: ${response.error})`; + return { error }; + } + return { + completed: false, + }; + }, +) +``` + +This implementation will create the first modal with a title, input block, submit button, and close button. Once the user enters a term in the input field and clicks “submit”, we have to handle that action in a view submission handler, which will use the `callback_id` of the modal to react. We’ll take a look at that in the next step. + + + +## Create show definition submission view + +For ease of readability, we’ll put all of our interactivity handlers in a file separate from the main function file. Create a new file in the `functions` folder and call it `interactivity_handler.ts`. But before we get ahead of ourselves, we’ll need to look up the term the user submitted in a [datastore](/deno-slack-sdk/guides/using-datastores). Let’s define that now. Back up to the root directory of your project and create a new folder called `datastores`. Add a file to it called `terms.ts`. This datastore will hold the crowdsourced terms in our app. When a user submits a term, it will be saved in the datastore, and when a user looks for a term, it will be retrieved from the datastore. Define it here: + +```javascript +// terms.ts +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const TermsDatastore = DefineDatastore({ + name: "terms", + primary_key: "id", + attributes: { + id: { type: Schema.types.string }, + term: { type: Schema.types.string }, + definition: { type: Schema.types.string }, + }, +}); +``` + +Now we’re ready to go back to `interactivity_handler.ts` and define a view submission handler to handle what happens after a user enters a term to be defined and clicks "submit". Let’s call that `showDefinitionView`. The first step we'll need to take in this handler is look up the submitted term in our newly-defined datastore like this: + +```javascript +// interactivity_handler.ts + +import { ViewSubmissionHandler } from "deno-slack-sdk/functions/interactivity/types.ts"; +import { TermLookupFunction } from "./term_lookup_function.ts"; +import { TermsDatastore } from "../datastores/terms.ts"; + +// This handler is invoked after a user submits a term to be defined +export const showDefinitionView: ViewSubmissionHandler< + typeof TermLookupFunction.definition +> = async ({ view, client }) => { + const termEntered = view.state.values.term.action.value; + + if (termEntered.length < 1) { + return { + response_action: "errors", + errors: { term_entered: "Must be 1 character or longer" }, + }; + } + + const queryResult = await client.apps.datastore.query({ + datastore: TermsDatastore.name, + expression: "#term = :term", + expression_attributes: { "#term": "term" }, + expression_values: { ":term": termEntered }, + }); + +``` +For some helpful guidance on how this query was constructed, check out the [Datastores](/deno-slack-sdk/guides/using-datastores) page. Once the query is run, we have two possible outcomes: the term is found and we return it to the user, or the term is not found and we ask the user if they’d like to submit a definition for it. Here’s the logic for the former: + +```javascript +// interactivity_handler.ts + + // If the term is found, display the associated definition + if (queryResult.items.length >= 1) { + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "second-page", + "notify_on_close": false, + "title": { "type": "plain_text", "text": termEntered }, + "close": { "type": "plain_text", "text": "Close" }, + "private_metadata": JSON.stringify({ termEntered }), + "blocks": [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": queryResult.items[0].definition, + }, + }, + ], + }, + }; + } +``` +This modal will present the user with the definition and a close button only. We don’t provide a submit button here because the modal is only informative; there is no new data to submit. Alternatively, if the term is not found, we’ll present the user with the option to submit a definition for it: + +```javascript +// interactivity_handler.ts + + // If the term is not found in the datastore, ask if they'd like to add a definition + if (queryResult.items.length < 1) { + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "add-definition", + "notify_on_close": false, + "title": { "type": "plain_text", "text": termEntered }, + "close": { "type": "plain_text", "text": "Close" }, + "submit": { "type": "plain_text", "text": "Click here to add one" }, + "private_metadata": JSON.stringify({ termEntered }), + "blocks": [ + { + "type": "section", + "text": { + "type": "plain_text", + "text": `There is currently no definition for ${termEntered}`, + }, + }, + ], + }, + }; + } +}; +``` +Here, we’ve changed the text of the submit button to indicate that clicking it will allow the user to submit a definition of their own. So what happens when they click it? We’ll create another view submission handler for that. Something to make note of: notice how we carry forward the term itself in `private_metadata`. Without this, we would not have access to what term we are defining, since that data was submitted in a prior modal. Also make note of the `callback_id` of the modal; we’ll use that later to call the next handler. + + +## Create show definition submission view + +Once a user elects to submit a new definition for a term that does not have one, we need a new view to handle the input of that data. This is done through another view submission handler. Let’s call this one `showDefinitionSubmissionView` and add it to the same `interactivity_handler.ts` file that we put our first handler in. + +```javascript +// interactivity_handler.ts + +// This handler is invoked after a user elects to add a new definition +export const showDefinitionSubmissionView: ViewSubmissionHandler< + typeof TermLookupFunction.definition +> = ({ view }) => { + const { termEntered } = JSON.parse(view.private_metadata!); + + if (termEntered.length < 1) { + return { + response_action: "errors", + errors: { term_entered: "Must be 1 character or longer" }, + }; + } + + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "definition-submission", + "notify_on_close": false, + "title": { "type": "plain_text", "text": termEntered }, + "submit": { "type": "plain_text", "text": "Submit" }, + "close": { "type": "plain_text", "text": "Close" }, + "private_metadata": JSON.stringify({ termEntered }), + "blocks": [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": `Add a definition for ${termEntered}`, + }, + }, + { + "type": "input", + "block_id": "definition", + "element": { + "type": "plain_text_input", + "action_id": "action", + "multiline": true, + }, + "label": { "type": "plain_text", "text": "Definition" }, + }, + { + "type": "context", + "elements": [ + { + "type": "mrkdwn", + "text": + "You can use Slack markdown for this field, like `*bold*` and `_italics_`.", + }, + ], + }, + ], + }, + }; +}; +``` + +Once the user submits the button to add a new definition, we present this modal, which provides an input block for their definition, as well as submit and close buttons. Remember the term we stored away in `private_metadata`? We can now retrieve it to use as the title for this modal. We’ll again store it in `private_metadata` so that we can use it in the subsequent modal too. Again, take note of the `callback_id`, we’ll use this later. + + + +## Create a confirmation view + +The final view submission handler to write occurs once the user submits a new definition for the term. First let’s save the submitted definition to the datastore. + +```javascript +// interactivity_handler.ts + +// This handler is invoked after a new definition is submitted +export const showConfirmationView: ViewSubmissionHandler< + typeof TermLookupFunction.definition +> = async ({ view, client }) => { + const { termEntered } = JSON.parse(view.private_metadata!); + const definition = view.state.values.definition.action.value; + + let saveSuccess: boolean; + + const uuid = crypto.randomUUID(); + + const putResponse = await client.apps.datastore.put({ + datastore: TermsDatastore.name, + item: { + id: uuid, + term: termEntered, + definition: definition, + }, + }); + + if (!putResponse.ok) { + console.log("Error calling apps.datastore.put:"); + saveSuccess = false; + return { + error: putResponse.error, + }; + } else { + saveSuccess = true; + } +``` + +This means we two different possible outcomes: the save is successful and the user is on their way, or the save is not successful. Here is the former: + +```javascript + if (saveSuccess == true) { + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "completion_successful", + "notify_on_close": false, + "title": { "type": "plain_text", "text": `${termEntered} added` }, + "close": { "type": "plain_text", "text": "Close" }, + "blocks": [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": `We've added ${termEntered} to your company definitions.`, + }, + }, + { + "type": "divider", + }, + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": `*${termEntered}*\n${definition}`, + }, + }, + ], + }, + }; + } +``` + +And the latter: + +```javascript +else { + return { + response_action: "update", + view: { + "type": "modal", + "callback_id": "completion_not_successful", + "notify_on_close": false, + "title": { "type": "plain_text", "text": "Add definition" }, + "close": { "type": "plain_text", "text": "Close" }, + "blocks": [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": "Something went wrong and the save was not successful.", + }, + }, + ], + }, + }; + } +}; +``` + +This concludes the logic for the `interactivity_handler.ts` file. Next, let’s see how these handlers are wired up. + + + +## Write a view closed handler + +Back in `functions/term_lookup_function.ts`, we need to add the handler functions we just wrote in `interactivity_handler.ts`. Here’s how that’s done: + +```javascript +// term_lookup_function.ts + + .addViewSubmissionHandler( + ["first-page"], + showDefinitionView, + ) + .addViewSubmissionHandler( + ["add-definition"], + showDefinitionSubmissionView, + ) + .addViewSubmissionHandler( + ["definition-submission"], + showConfirmationView, + ) +``` +The first parameter of each function is the `callback_id` of the modal they respond to. Because these are view submission handlers, when the user clicks the submit button on the modal with the `callback_id` of “first-page”, the `showDefinitionView` submission handler will be called. When the modal with the `callback_id` of “add-definition” is submitted, `showDefinitionSubmissionView` is the handler that is called, and when the modal with the `callback_id` of “definition-submission” is submitted, `showConfirmationView` is the handler that is called. Finally, we’ll add a handler for when a view is closed, a view closed handler. This one is short; add it into the same file right after the functions above. + +```javascript +// term_lookup_function.ts + + .addViewClosedHandler( + ["first-page", "add-definition", "definition-submission"], + ({ view }) => { + console.log(`view_closed handler called: ${JSON.stringify(view)}`); + return { completed: true }; + }, + ); +``` + +This handler takes care of what happens when the view is closed from any of the three handlers we noted in the parameters. + + + +## Implement a workflow + +We are now ready to create a workflow as an entry point to our function. Create a new folder at the root of the project called `workflows` and add a file named `definition_workflow.ts`. + +```javascript +// definition_workflow.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { TermLookupFunction } from "../functions/term_lookup_function.ts"; + +export const DefinitionWorkflow = DefineWorkflow({ + callback_id: "definition_workflow", + title: "Definition workflow", + description: + "A workflow to show you definitions and add them if they don't exist.", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + }, + required: ["interactivity"], + }, +}); + +DefinitionWorkflow.addStep(TermLookupFunction, { + interactivity: DefinitionWorkflow.inputs.interactivity, +}); + +export default DefinitionWorkflow; +``` + +This is a workflow with only one step. We need to collect `interactivity` as an input parameter to pass along to the function and require no outputs. Next, we’ll update our manifest to declare all that we've created thus far. + + + +## Update the manifest + +When we created the app via the CLI initially, a bare bones `manifest.ts` file was created that looks like this: + +```javascript +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; + +/** + * The app manifest contains the app's configuration. This + * file defines attributes like app name and description. + * tools.slack.dev/deno-slack-sdk/guides/using-the-app-manifest + */ +export default Manifest({ + name: "define-app", + description: "A blank template for building Slack apps with Deno", + icon: "assets/default_new_app_icon.png", + functions: [], + workflows: [], + outgoingDomains: [], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); + +``` + +We’ll add to it now by reporting our function, workflow, datastore, and necessary scopes that the datastore requires. While we’re here, let’s update the description too. + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; +import { DefinitionWorkflow } from "./workflows/definition_workflow.ts"; +import { TermsDatastore } from "./datastores/terms.ts"; +import { TermLookupFunction } from "./functions/term_lookup_function.ts"; + +export default Manifest({ + name: "define-app", + description: + "This project allows users to look up and add new definitions of company acronyms and terms.", + icon: "assets/default_new_app_icon.png", + functions: [TermLookupFunction], + workflows: [DefinitionWorkflow], + datastores: [TermsDatastore], + outgoingDomains: [], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "datastore:read", + "datastore:write", + ], +}); + +``` +The app manifest is the app's configuration. It is very important that this file is structured correctly in order for your app to run smoothly. Each function, workflow, custom type, and datastore defined in an app must be declared in the manifest file. + + + +## Create a trigger + +This is our final step before we are able to run our app! We need to add a trigger to kick off the workflow and collect that `interactivity` parameter needed to initiate a modal’s interactivity. Create one more folder at the root of the project and call it `triggers`. Add a file to it and name it `term_definition_trigger`. In it, place the following code: + +```javascript +// term_definition_trigger.ts + +import { Trigger } from "deno-slack-sdk/types.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; +import { DefinitionWorkflow } from "../workflows/definition_workflow.ts"; + +const termDefinitionTrigger: Trigger = { + type: TriggerTypes.Shortcut, + name: "Term Definition Trigger", + description: + "A trigger that starts the workflow to define a user-entered term", + workflow: `#/workflows/${DefinitionWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: TriggerContextData.Shortcut.interactivity, + }, + }, +}; + +export default termDefinitionTrigger; +``` + +This is a link trigger that, upon clicking, will initiate the workflow, which will call the function, which will allow the user to search for and add company definitions to our app. To get the link for that link trigger, run the following in your terminal: + +```bash +slack trigger create —trigger-def triggers/term_definition_trigger.ts +``` + +After executing the command, select your app and workspace. The terminal will output a link called a "Shortcut URL", also known as your link trigger. Save that URL; we'll use it later. If you ever lose track of that URL, you can always run the command `slack triggers -info` and select your workspace to find it again. + + +## Run your app + +While in your project’s root directory, run this command in your terminal: + +```bash +slack run +``` + +Choose your app and assign it to your workspace. Then, switch over to the app in Slack and test it out. Remember the link trigger you created earlier? Copy and paste that URL in a message to yourself in Slack. It will unfurl into a button that you can click to initiate the workflow. + + +## Share your app + +Because nobody knows everything, including company jargon, this would be a great app to share with your team. Check out [Deploy to Slack](/deno-slack-sdk/guides/deploying-to-slack) to discover how to share this app with your team. + +## Next steps +For your next challenge, perhaps consider creating an app to [create an issue in GitHub](/deno-slack-sdk(/tutorials/github-issues-app)! \ No newline at end of file diff --git a/docs/tutorials/github-issues-app.md b/docs/tutorials/github-issues-app.md new file mode 100644 index 00000000..235a1009 --- /dev/null +++ b/docs/tutorials/github-issues-app.md @@ -0,0 +1,411 @@ +# GitHub issues app + + + +In this tutorial, you'll learn how to build a workflow app to create an issue in Github. We'll walk you through sequencing the right workflow steps together and building a function to call GitHub's APIs. Even if you're not looking to create issues on Github, you'll learn how you might invoke any third-party API from a function. + +If creating issues on GitHub is actually what you want to do, then all you'll need to do is deploy your app when you're done customizing it. By following a form with a custom function that calls [an eminent endpoint by GitHub](https://docs.github.com/en/rest/issues/issues#create-an-issue), we can create an issue and close it ourselves! + +Some features you’ll acquaint yourself with while building this app include: + +* [Functions](/deno-slack-sdk/guides/creating-slack-functions): the building blocks of common Slack functionality. +* [Workflows](/deno-slack-sdk/guides/creating-workflows): a set of steps for calling your functions that are executed in order. +* [Custom functions](/deno-slack-sdk/guides/creating-custom-functions): building blocks that _you_ define! +* [Triggers](/deno-slack-sdk/guides/using-triggers): for kicking off your workflows. + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + + +## Choose your adventure + +After you've [installed the command-line interface](/deno-slack-sdk/guides/getting-started), you have two ways you can get started: + +### Use a blank app + +You can create a blank app with the Slack CLI using the following command: + +```bash +slack create github-functions-app --template https://github.com/slack-samples/deno-blank-template +``` + +### Use a pre-built app + +Or, you can use the pre-built [GitHub Functions app](https://github.com/slack-samples/deno-github-functions): + +``` +slack create github-functions-app --template https://github.com/slack-samples/deno-github-functions +``` + +### Change your directory + +Once you have your new project ready to go, change into your project directory. + +## Create a GitHub personal access token + +A personal access token is required when calling the GitHub API. Create a new token for this tutorial by visiting your developer settings on [GitHub](https://github.com/settings/tokens). + +Since your personal access token will be used in this tutorial, all issues created from the workflow will appear to have been created by your account. + +### Select required scopes + +To access public repositories, create a new personal token [on GitHub](https://github.com/settings/tokens) with the following scopes: + ++ `public_repo`, `repo:invite` ++ `read:org` ++ `read:user`, `user:email` ++ `read:enterprise` + +To prevent `404: Not Found` errors when attempting to access private repositories, the `repo` scope must also be selected. + +## Add the token to your environment variables + +When developing locally, you can store your API credentials by adding your GitHub token to your app's environment variables. To do this, create a file called `.env` in the root directory of your project, and add your token to the file as follows: + +```bash +GITHUB_TOKEN=ghp_1234AbCd5678 +``` + +For more information, refer to [using environment variables with the Slack CLI](/slack-cli/guides/using-environment-variables-with-the-slack-cli). + +## Define the custom function + +Defining the definitions and manifest of our app gives us a birds-eye view before we dive into building. Open your text editor (we recommend VSCode with the Deno plugin) and point to the directory we created earlier. + +Since the workflow we're creating revolves around creating a new GitHub issue, we'll begin by defining a [custom function](/deno-slack-sdk/guides/creating-custom-functions) with the inputs we know (the repository and information about the issue) and the outputs we expect (the issue number and link). + +The `DefineFunction` method will allow us to define the attributes that comprise this function. Here, we'll describe the attributes seen by other people and used by workflows, as well as the input and output types. + +```javascript +// functions/create_issue/definition.ts + +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +const CreateIssueDefinition = DefineFunction({ + callback_id: "create_issue", + title: "Create GitHub issue", + description: "Create a new GitHub issue in a repository", + source_file: "functions/create_issue/mod.ts", + input_parameters: { + properties: { + url: { + type: Schema.types.string, + description: "Repository URL", + }, + title: { + type: Schema.types.string, + description: "Issue Title", + }, + description: { + type: Schema.types.string, + description: "Issue Description", + }, + assignees: { + type: Schema.types.string, + description: "Assignees", + }, + }, + required: ["url", "title"], + }, + output_parameters: { + properties: { + GitHubIssueNumber: { + type: Schema.types.number, + description: "Issue number", + }, + GitHubIssueLink: { + type: Schema.types.string, + description: "Issue link", + }, + }, + required: ["GitHubIssueNumber", "GitHubIssueLink"], + }, +}); + +export default CreateIssueDefinition; +``` + +The source code for `functions/create_issue/mod.ts` is shared in Step 4, but keep this definition in mind until then! Or rush ahead and write the function! We won't mind. 😉 + + +## Scaffold your workflow + +Start by defining the workflow and outlining the steps. We'll add functions and inputs to these steps later. + +```javascript +// workflows/create_new_issue.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const CreateNewIssueWorkflow = DefineWorkflow({ + callback_id: "create_new_issue_workflow", + title: "Create new issue", + description: "Create a new GitHub issue", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + required: ["interactivity", "channel"], + }, +}); + +/* Step 1 - Open a form */ +// const issueFormData = CreateNewIssueWorkflow.addStep( ... ); + +/* Step 2 - Create a new issue */ +// const issue = CreateNewIssueWorkflow.addStep( ... ); + +/* Step 3 - Post the new issue to channel */ +// CreateNewIssueWorkflow.addStep( ... ); + +export default CreateNewIssueWorkflow; +``` + + +## Make your manifest + +Import and add these definitions to your app's manifest. + +```javascript +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; +import CreateIssueDefinition from "./functions/create_issue/definition.ts"; +import CreateNewIssueWorkflow from "./workflows/create_new_issue.ts"; + +export default Manifest({ + name: "Workflows for GitHub", + description: "Bringing oft-used GitHub functionality into Slack", + icon: "assets/icon.png", + functions: [CreateIssueDefinition], + workflows: [CreateNewIssueWorkflow], + outgoingDomains: [], + // If your organization uses a separate Github enterprise domain, add that domain to this list + // so that functions can make API calls to it. + outgoingDomains: ["api.github.com"], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +## Collect user input + +You can call functions in an ordered sequence by adding them to your workflow. + +The Slack function [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) can be used to collect input data that is used by later steps in the workflow. + +```javascript +// workflows/create_new_issue.ts + +... + +/* Step 1 - Open a form */ +const issueFormData = CreateNewIssueWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Create an issue", + interactivity: CreateNewIssueWorkflow.inputs.interactivity, + submit_label: "Create", + description: "Create a new issue inside of a GitHub repository", + fields: { + elements: [{ + name: "url", + title: "Repository URL", + description: "The GitHub URL of the repository", + type: Schema.types.string, + }, { + name: "title", + title: "Issue title", + type: Schema.types.string, + }, { + name: "description", + title: "Issue description", + type: Schema.types.string, + }, { + name: "assignees", + title: "Issue assignees", + description: + "GitHub username(s) of the user(s) to assign the issue to (separated by commas)", + type: Schema.types.string, + }], + required: ["url", "title"], + }, + }, +); +``` + +## Call your custom function + +The second step of the workflow calls our custom function to create an issue on GitHub. Similar to other steps, the definition of this function is provided, along with the inputs to the function. + +```javascript +// workflows/create_new_issue.ts +import CreateIssueDefinition from "../functions/create_issue/definition.ts"; + +... + +/* Step 2 - Create a new issue */ +const issue = CreateNewIssueWorkflow.addStep(CreateIssueDefinition, { + url: issueFormData.outputs.fields.url, + title: issueFormData.outputs.fields.title, + description: issueFormData.outputs.fields.description, + assignees: issueFormData.outputs.fields.assignees, +}); +``` + +Notice how the input of this function uses the output from the form in our previous step! The output of this step — the values to be returned from our custom function — will be used to construct and post a message with the basic details of a newly-created issue. + + +## Post the GitHub response + +The Slack function [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message) can be used to post details about the newly-created issue in the channel. + +```javascript +// workflows/create_new_issue.ts + +... + +/* Step 3 - Post the new issue to channel */ +CreateNewIssueWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: CreateNewIssueWorkflow.inputs.channel, + message: + `Issue #${issue.outputs.GitHubIssueNumber} has been successfully created\n` + + `Link to issue: ${issue.outputs.GitHubIssueLink}`, +}); +``` + +## Craft the custom function + +Here's where you can take input from Slack, apply custom code to it, and return it back to a workflow. + +Copy and paste the following code into `functions/create_issue/mod.ts`. The `mod.ts` filename is a convention to declare the entry point to your function in `functions/create_issue`. + +```javascript +// functions/create_issue/mod.ts + +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import CreateIssueDefinition from "./definition.ts"; + +// https://docs.github.com/en/rest/issues/issues#create-an-issue +export default SlackFunction( + CreateIssueDefinition, + async ({ inputs, env }) => { + const headers = { + Accept: "application/vnd.github+json", + Authorization: "Bearer " + env.GITHUB_TOKEN, + "Content-Type": "application/json", + }; + + const { url, title, description, assignees } = inputs; + + try { + const { hostname, pathname } = new URL(url); + const [_, owner, repo] = pathname.split("/"); + + // https://docs.github.com/en/enterprise-server@3.3/rest/guides/getting-started-with-the-rest-api + const apiURL = hostname === "github.com" + ? "api.github.com" + : `${hostname}/api/v3`; + const issueEndpoint = `https://${apiURL}/repos/${owner}/${repo}/issues`; + + const body = JSON.stringify({ + title, + body: description, + assignees: assignees?.split(",").map((assignee: string) => { + return assignee.trim(); + }), + }); + + const issue = await fetch(issueEndpoint, { + method: "POST", + headers, + body, + }).then((res: Response) => { + if (res.status === 201) return res.json(); + else throw new Error(`${res.status}: ${res.statusText}`); + }); + + return { + outputs: { + GitHubIssueNumber: issue.number, + GitHubIssueLink: issue.html_url, + }, + }; + } catch (err) { + console.error(err); + return { + error: + `An error was encountered during issue creation: \`${err.message}\``, + }; + } + }, +); +``` + +For the curious, this function dissects input from the workflow's form, then makes a POST API request to the ["Create an issue" GitHub API endpoint](https://docs.github.com/en/rest/issues/issues#create-an-issue). The result of this API call is then returned as output as defined in the function definition (`functions/creation_issue/definition.ts`); otherwise an error is returned. + +## Create a trigger + +Triggers are how workflows are invoked. Each workflow can have multiple triggers. + +There are four types of triggers: [link triggers](/deno-slack-sdk/guides/creating-link-triggers), [scheduled triggers](/deno-slack-sdk/guides/creating-event-triggers), [event triggers](/deno-slack-sdk/guides/creating-event-triggers), and [webhook triggers](/deno-slack-sdk/guides/creating-webhook-triggers). A link trigger is what we'll be using. + +Link triggers are an interactive type, which means they require a user to manually start them. Define your link trigger in a separate file in a `triggers` folder called `create_new_issue_shortcut.ts`: + +```javascript +// triggers/create_new_issue_shortcut.ts + +import { Trigger } from "deno-slack-api/types.ts"; +import CreateNewIssueWorkflow from "../workflows/create_new_issue.ts"; + +const createNewIssueShortcut: Trigger< + typeof CreateNewIssueWorkflow.definition +> = { + type: "shortcut", + name: "Create GitHub issue", + description: "Create a new GitHub issue in a repository", + workflow: "#/workflows/create_new_issue_workflow", + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + channel: { + value: "{{data.channel_id}}", + }, + }, +}; + +export default createNewIssueShortcut; +``` + +Run the `trigger create` command in your terminal: + +```cmd +slack trigger create --trigger-def "triggers/create_new_issue_shortcut.ts" +``` + +After executing this command, select your workspace and choose the _Local_ app environment. When the process completes, you'll be given a link called "shortcut URL." This is your _link trigger_ for this workflow on this workspace. Save that URL for when you start testing. + +## Run your code to test and tweak + +Here's the step we're going to leave you, but this is where your development experience will begin as you alter, test, falter, alter, and test again. + +You're building along and your workflow should be, too. You can use development mode to run this workflow in Slack directly from the machine you're reading this from now: + +```bash +slack run +``` + +After you've chosen your development app and assigned it to your workspace, you can switch over to your Slack app and try out your new workflow. + +In Slack, you'll want to use the _link trigger_ you created earlier. Once you paste its URL into the message box and post it, it'll unfurl and give you a button to invoke the workflow. + +## Next steps + +For your next challenge, perhaps consider creating an app your users can use to [request time off](/deno-slack-sdk/tutorials/request-time-off-app)! \ No newline at end of file diff --git a/docs/tutorials/give-kudos-app.md b/docs/tutorials/give-kudos-app.md new file mode 100644 index 00000000..4385bc2c --- /dev/null +++ b/docs/tutorials/give-kudos-app.md @@ -0,0 +1,427 @@ +# Give kudos app + + + +import SlackMessage from '@site/src/components/SlackMessage'; + + + +This app will allow users to give kudos and share kind words with anyone in your workspace. + +We're setting sail on our maiden voyage – creating and deploying a workflow app – and would love to show you the ropes. By the end of the expedition, we'll have an app to parrot personal "kudos" throughout a workspace. Nothing is more important than a crew's morale, after all! + +A curiosity of the waters is not the only thing you need to sail the open seas, so ensure you have the following prerequisites completed, then climb aboard to discover what it takes to be captain of your own ship – that is, developing your own Slack app! + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + +## Choose your heading + +How much guidance would you like on this journey? + +## Use a blank app + +If you're ready to take the helm yourself, you can create a blank app with the Slack CLI. Don't worry, we'll be right beside you. + +``` +slack create give-kudos-app --template https://github.com/slack-samples/deno-blank-template +``` +## Use a pre-built app + +If you'd like to follow along without steering the ship, use the pre-built [Give Kudos app](https://github.com/slack-samples/deno-give-kudos): + +``` +slack create give-kudos-app --template https://github.com/slack-samples/deno-give-kudos +``` + +Once you have your new project ready to go, change into your project directory. + +## Comprehend the manifest + +Before setting sail, we'll need to take inventory of our ship's cargo. The [manifest](/deno-slack-sdk/guides/using-the-app-manifest) is essentially an overview summary in a file at the root of your project. It is created automatically when you create a project using the Slack CLI. An inspection of the `manifest.ts` of the [sample app](https://github.com/slack-samples/deno-give-kudos) reveals a collection of workflows, functions, and other app-related attributes. + +```javascript +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; +import { FindGIFFunction } from "./functions/find_gif.ts"; +import { GiveKudosWorkflow } from "./workflows/give_kudos.ts"; + +export default Manifest({ + name: "Kudo", + description: "Brighten someone's day with a heartfelt thank you", + icon: "assets/icon.png", + functions: [FindGIFFunction], + workflows: [GiveKudosWorkflow], + outgoingDomains: [], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +The details of this expedition’s manifest are soon to follow, but first let's shine a light through the fog and get a bearing on the three fundamentals of a Slack app: [workflows](/deno-slack-sdk/guides/creating-workflows), [functions](/deno-slack-sdk/guides/creating-slack-functions), and [triggers](/deno-slack-sdk/guides/using-triggers). + + +## Define the app building blocks + +A [workflow](/deno-slack-sdk/guides/creating-workflows) is a collection of processes, executed in response to certain events. In nautical terms, repairing the hull is a common workflow that happens when yours truly runs the ship aground. + +The processes that make up a workflow are known as [functions](/deno-slack-sdk/guides/creating-slack-functions). Functions can be built-in, everyday actions such as sending a message or opening a form - or made [custom](/deno-slack-sdk/guides/creating-custom-functions), defined by your own logic with various inputs and outputs. + +On the sea, raising the sails and scrubbing the decks are oft run functions. + +[Triggers](/deno-slack-sdk/guides/using-triggers) act as inspiration for the actions of a workflow, defining when a workflow is invoked. Certain events in Slack (such as a clicked link or reaction added) or a schedule of specified time intervals can serve as triggers for a workflow. + +After taking in the view from the mast, we’re ready to set sail! + +## Weave a workflow + +The goal of our expedition is to build an app that can perform the task of parroting a personalized message. To do so, a workflow called "Give kudos" is created in `workflows/give_kudos.ts`. This workflow will contain all the actions our app needs to complete that task. + + +```javascript +// workflows/give_kudos.ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; + +const GiveKudosWorkflow = DefineWorkflow({ + callback_id: "give_kudos_workflow", + title: "Give kudos", + description: "Acknowledge the impact someone had on you", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + }, + required: ["interactivity"], + }, +}); + +// your steps go here + +export { GiveKudosWorkflow }; +``` + +This workflow doesn't do anything - yet. For now, just note that it'll contain steps that require user interactivity. + + +## Sketch the steps + +Our task of sharing kudos can be accomplished with three actions: + +* collecting information about a message, +* finding the right GIF, +* then sharing the love in the form of a message (bottle not included). + +Let's look at how these actions are added as steps to a workflow. Each step is composed of a function definition as well as the input object. The input object allows the outputs of one step to become inputs to another in a chain of functions. + +The first step is composed of a [Slack function](/deno-slack-sdk/guides/creating-slack-functions), [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form), that will, as hinted, open a form for the user. This lets our app collect the kudos users want to give. + + +```javascript +// workflows/give_kudos.ts + +... + +/* Step 1. Collect message information */ +const kudo = GiveKudosWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Give someone kudos", + interactivity: GiveKudosWorkflow.inputs.interactivity, + submit_label: "Share", + description: "Continue the positive energy through your written word", + fields: { + elements: [{ + name: "doer_of_good_deeds", + title: "Whose deeds are deemed worthy of a kudo?", + description: "Recognizing such deeds is dazzlingly desirable of you!", + type: Schema.slack.types.user_id, + }, { + name: "kudo_channel", + title: "Where should this message be shared?", + type: Schema.slack.types.channel_id, + }, { + name: "kudo_message", + title: "What would you like to say?", + type: Schema.types.string, + long: true, + }, { + name: "kudo_vibe", + title: 'What is this kudo\'s "vibe"?', + description: "What sorts of energy is given off?", + type: Schema.types.string, + enum: [ + "Appreciation for someone 🫂", + "Celebrating a victory 🏆", + "Thankful for great teamwork ⚽️", + "Amazed at awesome work ☄️", + "Excited for the future 🎉", + "No vibes, just plants 🪴", + ], + }], + required: ["doer_of_good_deeds", "kudo_channel", "kudo_message"], + }, + }, +); + +... +``` + +The form will look like this for the user wanting to give kudos. + +![The form within slack](/img/give-kudos/give-kudos-open-form.png) + +The second step is composed of a [custom function](/deno-slack-sdk/guides/creating-custom-functions), `FindGIFFunction`. This is a function built specifically for this app. It'll find a GIF related to the "vibe" someone gives a message. + +```javascript +// workflows/give_kudos.ts +import { FindGIFFunction } from "../functions/find_gif.ts"; + +... + +/* Step 2. Find the right GIF */ +const gif = GiveKudosWorkflow.addStep(FindGIFFunction, { + vibe: kudo.outputs.fields.kudo_vibe, +}); + +... +``` + +Notice how the `OpenForm` step contains the `interactivity` context of the workflow and `FindGIFFunction` uses `kudo_vibe`, an output parameter of the `OpenForm` step. + +The third step is composed of a Slack function, [`SendMessage`](/deno-slack-sdk/reference/slack-functions/send_message), that will send a message to a specific channel. It'll combine the user's words and the chosen GIF into one spectacular message fit for even the most seasoned sailors. That message will be sent to the channel the user specified within the form from step one. + +```javascript +// workflows/give_kudos.ts + +... + +/* Step 3. Share the love */ +GiveKudosWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: kudo.outputs.fields.kudo_channel, + message: + `*Hey <@${kudo.outputs.fields.doer_of_good_deeds}>!* Someone wanted to share some kind words with you :otter:\n` + + `> ${kudo.outputs.fields.kudo_message}\n` + + `${gif.outputs.URL}`, +}); + +export { GiveKudosWorkflow }; +... +``` + +And that's all you need to do for Slack functions! `FindGIFFunction` is a custom function, however. We'll have to roll up our sleeves and build that out ourselves. + + +## Define the custom function + +The `FindGIFFunction` function, unique to our app, is defined in `functions/find_gif.ts`, where inputs, outputs, and other attributes are described. This definition allows `FindGIFFunction` to be used within workflows, as when sharing a kudo! + +```javascript +// functions/find_gif.ts +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; + +export const FindGIFFunction = DefineFunction({ + callback_id: "find_gif", + title: "Find a GIF", + description: "Search for a GIF that matches the vibe", + source_file: "functions/find_gif.ts", + input_parameters: { + properties: { + vibe: { + type: Schema.types.string, + description: "The energy for the GIF to match", + }, + }, + required: [], + }, + output_parameters: { + properties: { + URL: { + type: Schema.types.string, + description: "GIF URL", + }, + alt_text: { + type: Schema.types.string, + description: "description of the GIF", + }, + }, + required: ["URL"], + }, +}); +``` + + +## Find a GIF + +With the `FindGIFFunction` function defined, it can then be implemented, where input from the form is parsed and converted into a random GIF that's obtained from the GIF catalog. + +```javascript +// functions/find_gif.ts +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import gifs from "../assets/gifs.json" assert { type: "json" }; + +... + +const getEnergy = (vibe: string): string => { + if (vibe === "Appreciation for someone 🫂") return "appreciation"; + if (vibe === "Celebrating a victory 🏆") return "celebration"; + if (vibe === "Thankful for great teamwork ⚽️") return "thankful"; + if (vibe === "Amazed at awesome work ☄️") return "amazed"; + if (vibe === "Excited for the future 🎉") return "excited"; + if (vibe === "No vibes, just plants 🪴") return "plants"; + return "otter"; // 🦦 +}; + +interface GIF { + URL: string; + alt_text?: string; + tags: string[]; +} + +const matchVibe = (vibe: string): GIF => { + const energy = getEnergy(vibe); + const matches = gifs.filter((g: GIF) => g.tags.includes(energy)); + const randomGIF = Math.floor(Math.random() * matches.length); + return matches[randomGIF]; +}; + +export default SlackFunction(FindGIFFunction, ({ inputs }) => { + const { vibe } = inputs; + const gif = matchVibe(vibe ?? ""); + return { outputs: gif }; +}); +``` + + +## A collection of GIFs + +The GIF catalog – the treasure of this expedition – is listed in `assets/gifs.json`, but could safely be stored in a [datastore](/deno-slack-sdk/guides/using-datastores). To do that, you would create a [datastore](/deno-slack-sdk/guides/using-datastores) and use the [CLI command](/slack-cli/reference/commands/slack_datastore_put) to save all of the GIFs to it. Think of datastores as your very own treasure chests. + +For readability, we've only included a portion of the GIFs in the snippet below. If you want the full treasure trove, [grab it from the GitHub repo](https://github.com/slack-samples/deno-give-kudos/blob/main/assets/gifs.json). + +```javascript +// assets/gifs.json +[ + { + "URL": "https://media2.giphy.com/media/3oEjHWXddcCOGZNmFO/giphy.gif", + "alt_text": "A person wearing a banana hat says thanks a bunch", + "tags": ["thankful"] + }, + { + "URL": "https://media.giphy.com/media/3fBVaRM2c79TtXbyi6/giphy.gif", + "alt_text": "The future king of the pirates smiles at you", + "tags": ["amazed"] + }, + { + "URL": "https://media.giphy.com/media/WKdPOVCG5LPaM/giphy.gif", + "alt_text": "A cheerful high-five from the newsroom", + "tags": ["celebration", "excited"] + }, + { + "URL": "https://media1.giphy.com/media/Lcn0yF1RcLANG/giphy-downsized.gif", + "alt_text": "Wow! A feeling of wild disbelief overwhelms the senses", + "tags": ["amazed"] + }, + { + "URL": "https://media0.giphy.com/media/rgIdiNjWC933y/giphy.gif", + "alt_text": "A kingly racoon nodding over many subjects", + "tags": ["excited", "amazed"] + }, + { + "URL": "https://media0.giphy.com/media/kyLYXonQYYfwYDIeZl/giphy.gif", + "alt_text": "Elmo dances in celebration", + "tags": ["celebration"] + }, + { + "URL": "https://media2.giphy.com/media/3ohs7NuHL3gjbe2uGI/giphy-downsized.gif", + "alt_text": "You're noticed and appreciated <3", + "tags": ["appreciation"] + }, + { + "URL": "https://media1.giphy.com/media/xUA7aOIFDR4ZgqLy8w/giphy.gif", + "alt_text": "Fern having a messy hair day", + "tags": ["plants"] + }, + "URL": "https://media1.giphy.com/media/MbAlP79yMRysHKUyHV/giphy-downsized.gif", + "alt_text": "Sleepy otter rubs checks and yawns", + "tags": ["otter"] + } +] +``` + +## Invoke with a trigger + +The workflows and functions discussed above cannot be invoked until a trigger is created, similar to how a sailor needs an order from their captain to carry out tasks. + +To create an entry point into our app's "Give kudos" workflow, a [link trigger](/deno-slack-sdk/guides/creating-link-triggers) is defined in `triggers/give_kudos.ts`. + +```javascript +// triggers/give_kudos.ts +import { Trigger } from "deno-slack-api/types.ts"; + +const trigger: Trigger = { + type: "shortcut", + name: "Give some kudos", + description: "Broadcast your appreciation with kind words and a GIF", + workflow: "#/workflows/give_kudos_workflow", + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + }, +}; + +export default trigger; +``` + +The trigger is then created using the Slack CLI with the [`slack trigger create`](/slack-cli/reference/commands/slack_trigger_create) command, generating a Shortcut URL that can be clicked from a channel: + +```bash +slack trigger create --trigger-def triggers/give_kudos.ts +``` + +You'll be prompted to install your app to a workspace. Select your desired workspace, and then select a `Local` app environment. When you want to [deploy](/deno-slack-sdk/guides/deploying-to-slack) your app later on, you would repeat this step and select a `Deployed` app environment. + +## Run your app + +With the Shortcut URL in hand, our destination is near. Run the app in development mode to test it out on your local machine. + +```bash +slack run +``` + +Use the Shortcut URL within Slack to kick off the workflow. Play around and test the waters yourself. Once you're ready to deploy to other sailors, read on! + + +## Deploy your app + +When you're ready to welcome others aboard, you'll `deploy` your app instead of `run` it. Run the following command in your terminal: + +```bash +slack deploy +``` + +And then create the trigger again, but choose the _Deployed_ option this time: + +```bash +slack trigger create --trigger-def triggers/give_kudos.ts +``` + +There's no X, but rest assured you've found the ultimate treasure - the experience of building your very own _Give Kudos_ app. + + + +## End your expedition + +Returning back to harbor, we can reminisce about the journey of sharing a kudo, recalling that functions compose workflows and workflows are invoked by triggers. + +With these ropes, you're ready to take the seas yourself! There are many more knots to learn while on these waters, but you'll stay afloat just fine with what you now know. + +### The next adventure + +The next time you set sail, perhaps consider creating [a bot to welcome users to your workspace](/deno-slack-sdk/tutorials/welcome-bot). Bon voyage, captain! + + + diff --git a/docs/tutorials/governing-slack-connect-invites.md b/docs/tutorials/governing-slack-connect-invites.md new file mode 100644 index 00000000..618b6ca2 --- /dev/null +++ b/docs/tutorials/governing-slack-connect-invites.md @@ -0,0 +1,365 @@ +--- +slug: /deno-slack-sdk/tutorials/governing-slack-connect-invites +--- + +# Governing Slack Connect invites + + + +This guide will show you how to create a Slack workflow to automate the process of governing [Slack Connect](https://api.slack.com/apis/connect) invitations. + +## Prerequisites + +You'll need to have admin access to your Slack workspace to continue! Complete the following steps to allow workflow automations to interact with your Slack Connect invitations. + +1. In the Admin Dashboard, under **Slack Connect Settings**, enable the “Apply automation rules before channel invitations are sent” preference within **Slack Connect Settings** in the Admin Dashboard. +2. To allow users to request invites, in the Admin Dashboard, under **Slack Connect Settings**, then **Channels**, enable the "Sending Invitations with Permission to Post Only" or "Sending Invitations with permission to post, invite and more" preference. + +In addition, please consider the following before proceeding: + +* Only external invites sent by members of your organization to channels owned by your organization will be governed by these automation tools. +* MPDM to Private Channel conversions are not considered invitations and will not be governed by automation rules. We recommend you review your policy around MPDM to Private Channel changes (under **Slack Connect Settings**). +* Admins and owners and those who have the permission to approve Slack Connect invitations have implicit approval to send invitation requests, meaning their invitations will not be held and subject to automation rules. +* Requests to Invite from Bots that have the [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) scope will implicitly be sent and will not be held and subject to automation rules. +* Admin request messages will be directed to the same channels or individuals as specified under “Who can approve requests and manage channels?” and “Send requests to…” under **Approving Channel Invitations**. + +## Setting up your workflow app {#setting-up} + +### 1. Create a Deno Slack SDK app from the Slack CLI {#create} + +If you haven't yet, install and authorize the [Slack CLI](/deno-slack-sdk/guides/getting-started). + +Then use the `create` Slack CLI command to create a workflow app. + +``` +slack create my-app +``` + +### 2. Add the `conversations.connect:manage` scope {#add-scope} + +Scopes are added to workflows via the manifest. Modify the `botScopes` array to include the [`conversations.connect:manage`](https://api.slack.com/scopes/conversations.connect:manage) scope. + +```js + botScopes: ["conversations.connect:manage"], +``` + + +## Creating the workflow {#workflow} + +### 1. Define a function with input and output parameters for handling invite requests {#define-function} + +[Functions](/deno-slack-sdk/guides/creating-slack-functions) are the building blocks of workflow apps. We'll be using a [custom function](/deno-slack-sdk/guides/creating-custom-functions) to utilize the Slack Connect API methods. Let's define the function now. + +```js +import { DefineFunction, Schema } from "deno-slack-sdk/mod.ts"; +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import { SlackAPI } from "deno-slack-api/mod.ts"; + +export const RequestInfoFunction = DefineFunction({ + callback_id: "request_info_function", + title: "Request to Invite", + description: "Handle the requested invites", + source_file: "functions/request_info.ts", + input_parameters: { + properties: { + invite_request: { + type: Schema.types.object, + }, + }, + required: ["invite_request"], + }, + output_parameters: { + properties: { + result: { + type: Schema.types.string, + }, + }, + required: ["result"], + }, +}); +``` + +### 2. Define the main logic to handle Slack Connect Invitation Requests {#define-logic} + +You can filter invitees, automatically approving or denying certain requests. + +This is done using two Web API methods: + +* [`conversations.requestSharedInvite.approve`](https://api.slack.com/methods/conversations.requestSharedInvite.approve) +* [`conversations.requestSharedInvite.deny`](https://api.slack.com/methods/conversations.requestSharedInvite.deny) + +In this example the filtering is based on their email domains. + +```js +// Define some constants to be used for domain filtering logic. +const ALLOWED_EMAIL_DOMAINS = ["@slack-corp.com", "@approved-vendor.com"]; +const BLOCKED_EMAIL_DOMAINS = ["@danger.com"]; +const REQUEST_REASON_EMAIL_DOMAINS = ["@gmail.com"]; + +const filterInvites = (invitees: any[], domains: string[]) => + invitees.filter((invite: any) => + domains.some((domain) => invite.email.endsWith(domain)) + ); + +const handleInvites = async (client: any, invites: any[], action: string, message?: string) => { + for (const invite of invites) { + const response = await client.apiCall(`conversations.requestSharedInvite.${action}`, { invite_id: invite.invite_id, + message, +}); +}}; +``` + +You also have the ability to prompt the user for additional information via a form link when needed. + +```js +const postReasonRequestMessage = async (client: any, userId: string) => { + // This code snippet sends a message to the user with a link to a form + // prompting for more information. + const response = await client.chat.postMessage({ + channel: userId, // Use the userId of the requesting user + text: "Provide more info about SC channel request", + blocks: [ + { "type": "header", + "text": { + "type": "plain_text", + "text": "Slack Connect Request Details", + }, + }, + { "type": "context", + "elements": [ + { + "type": "mrkdwn", + "text": "Provide the following details about the requested invitees.", + } + ], + }, + { "type": "actions", + "elements": [ + { + "type": "button", + "text": { + type: "plain_text", + text: "Start", + emoji: true, + }, + action_id: "open_form", + value: "value1_approve", + style: "primary", + } + ], + }, + ], + }); + + return { + completed: false, + }; +}; +``` + +Let's use these features now in a custom function. + +### 3. Define the main function {#define-main-function} + +This function uses the filters previously created to automatically sort invites. + +The `BLOCKED_EMAIL_DOMAINS` values are denied, the `ALLOWED_EMAIL_DOMAINS` values are approved, and the the `REQUEST_REASON_EMAIL_DOMAINS` values are followed up with a prompt asking for additional information with the info set up in the previous instruction step using a [Block Kit actions handler](/deno-slack-sdk/guides/creating-an-interactive-modal#open-block-kit-action). + +```js +export default SlackFunction(RequestInfoFunction, + async ({ inputs, tokens, env }) => { + const invitees = inputs.invite_request.target_users; + const client = SlackAPI(token, { slackApiUrl: env.SLACK_API_URL }); + + // Filter invitees based on email domain. + const denyInvites = filterInvites(invitees, BLOCKED_EMAIL_DOMAINS); + const approveInvites = filterInvites(invitees, ALLOWED_EMAIL_DOMAINS); + const requiresReasonInvites = filterInvites(invitees, REQUEST_REASON_EMAIL_DOMAINS); + + await handleInvites(client, approveInvites, "approve"); + await handleInvites(client, denyInvites, "deny", "The recipients are not part of a pre-approved organization to work with."); + + // Auto-approve or deny based on blocked or allow domain criteria. + // Prompt for more information if meets needs more info status. + if (requiresReasonInvites.length) { + const userId = inputs.invite_request.actor.id; + await postReasonRequestMessage(client, inputs.invite_request.actor.id); + } + + return { completed: false }; + }, +) + .addBlockActionsHandler( + "open_form", + async ({ inputs: _inputs, body, token, env }) => { + const formMetadata = { + ts: body?.message?.ts ?? "", + message_channel_id: body?.channel?.id ?? "", + }; + + // Handle button click to open a form for additional information. + const client = SlackAPI(token, { slackApiUrl: env.SLACK_API_URL }); + + // Build form that requests a reason for the invite + const view_payload = { + interactivity_pointer: body.interactivity.interactivity_pointer, + view: { + private_metadata: JSON.stringify(formMetadata), + type: "modal", + title: { + type: "plain_text", + text: "Details about the Connecting Teams", + }, + submit: { + type: "plain_text", + text: "Submit", + }, + blocks: [ + { + "type": "section", + "text": { + "type": "mrkdwn", + "text": "Additional information", + }, + }, + { + "type": "divider", + }, + { + "type": "input", + "block_id": "reason_block", + "element": { + "type": "plain_text", + "action_id": "reason_input", + "multiline": true, + "placeholder": { + "type": "plain_text", + "text": "Please provide a reason for this invitation.", + }, + }, + "label": { + "type": "plain_text", + "text": "Reason", + }, + }, + ], + } + }; + }, + ) + .addViewSubmissionHandler( + "approval_info_modal", + async ({ body, view, token, env, inputs }) => { + // Handle View Submission + // Update the Slackbot message to end workflow submissions + const privateMetadata = JSON.parse(view.private_metadata ?? ""); + + const client = SlackAPI(token, { + slackApiUrl: env.SLACK_API_URL, + }); + + + // Update message to user to show that submission was made with the reason + const resp = await client.chat.update({ + channel: privateMetadata.message_channel_id, + ts: privateMetadata.ts, + as_user: true, + text: "Provide more info about SC channel request", + blocks: [ + { + "type": "header", + "text": { + "type": "plain_text", + "text": "Slack Connect Approval Details", + }, + }, + { + "type": "context", + "elements": [ +{ + "type": "mrkdwn", + "text": "Thank you for your submission :white_check_mark:", + }, + ], + }, + ], + }); + + // TODO: Handle user input in the way you want + const reasonForInviteRequest = view.state.values.reason_block.reason_input.value; + }, +); +``` + +### 4. Define the workflow {#define-workflow} + +With the desired functionality achieved in the custom function, now it needs to be added to a workflow. The following defines a workflow and adds the `RequestInfoFunction` as a step. + +```js +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { RequestInfoFunction } from "../functions/request_info.ts"; + +const InviteRequested = DefineWorkflow({ + callback_id: "invite_requested_workflow", + title: "Invite Requested Workflow", + description: "", + input_parameters: { + properties: { + invite_request: { + type: Schema.types.object, + description: "A requested invite and its metadata", + }, + }, + required: [], + }, +}); + +// This is the step that makes the workflow process that invite request as defined in the // main function. +InviteRequested.addStep(RequestInfoFunction, { + invite_request: InviteRequested.inputs.invite_request, +}); + +export default InviteRequested; +``` + +### 5. Create the event trigger {#event-trigger} + +Workflows are only invoked by [triggers](/deno-slack-sdk/guides/using-triggers). We'll be using an [event trigger](/deno-slack-sdk/guides/creating-event-triggers) to listen for the [`shared_channel_invite_requested`](https://api.slack.com/events/shared_channel_invite_requested) event to invoke the created workflow. + +```js +import { Trigger } from "deno-slack-api/types.ts"; +import { TriggerTypes } from "deno-slack-api/mod.ts"; +import InviteRequested from "../workflows/invite_requested.ts"; + +const inviteRequestedTrigger: Trigger = { + type: TriggerTypes.Event, + name: "InviteRequested", + description: "Gather details about the requested invite", + workflow: "#/workflows/invite_requested_workflow", + event: { + event_type: "slack#/events/shared_channel_invite_requested", + team_ids: [""], +}, +inputs: { + invite_request: { + value: "{{data}}", + }, +}, +}; + +export default inviteRequestedTrigger; +``` + +✅ Function + +✅ Workflow + +✅ Trigger + +And look at that, you've assembled all the parts of a workflow app! Now you'll just need to decide how to use it. + +## Onward {#onward} + +➡️ **To learn how to deploy your workflow app**, head over to the [Deploy your app](/deno-slack-sdk/guides/deploying-to-slack) page. + +✨ **To learn more about using Slack Web API methods in your workflows**, head over to the [Slack API calls](/deno-slack-sdk/guides/calling-slack-api-methods) page. \ No newline at end of file diff --git a/docs/tutorials/hello-world-app.md b/docs/tutorials/hello-world-app.md new file mode 100644 index 00000000..1f9e7c93 --- /dev/null +++ b/docs/tutorials/hello-world-app.md @@ -0,0 +1,351 @@ +# Hello world app + + + +In this tutorial, we're going to create an app based on the pre-built Hello World app. This is an app that will send a greeting to a channel. + +We'll create an app, interact with it in our workspace, then review the components that made that interaction possible. + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + +## Choose your adventure + +We can create our "Hello World" app in one of two ways: + +### Use a blank app + +You can create a blank app with the Slack CLI using the following command: + +``` +slack create hello-world-app --template https://github.com/slack-samples/deno-blank-template +``` + +### Use a pre-built app + +Or, you can use the pre-built [Hello World app](https://github.com/slack-samples/deno-hello-world): + +``` +slack create hello-world-app --template https://github.com/slack-samples/deno-hello-world +``` + +For this tutorial, we'll use the pre-built app. Once you have your new project ready to go, change into your project directory. + +## Explore the app structure + +Let's take a look at what's inside our new "Hello World" project directory: + +``` +LICENSE +README.md +assets/ +deno.jsonc +functions/ +import_map.json +manifest.ts +slack.json +triggers/ +workflows/ +``` + +The first place to direct your attention are the `functions`, `triggers`, and `workflows` folders. These are where the definitions and implementations for the inner workings of your app live. + +The next place to look is the `manifest.ts` file. This contains your app's manifest, which is where we can configure things like bot scopes and tell our app about our workflows. + +Other items in the project include: + +- `.slack/`: a home for internal configuration files, scripts hooks, and the app SDK. _This directory must be checked into your version control._ You'll also notice a `.slack/apps.dev.json` once you begin building: this file is in `.gitignore` and **should not** be checked in to version control. + +- `import_map.json`: a helper file for Deno that specifies where modules should be imported from. + +- `assets/`: a place to store assets related with the project. This is a great place to store the icon that your app will display when users interact with it. + +With our project ready, it's time to take it for a spin — but before we do, we have _one more_ thing to do, which is to create the trigger that we'll use to kick off our workflow. We'll talk about triggers and the specific kind we're going to create in the next section. Onward! + +## Trigger time + +Inside the `triggers` folder, there's a file called `greeting_trigger.ts`. + +This is a trigger configuration file. It's used by the CLI to create a type of [trigger](/deno-slack-sdk/guides/using-triggers) called a "Link trigger." + +Since this is a working sample app, it comes pre-baked with working code. The only thing we need to do so that the app will work correctly is to create the one trigger it uses to kick things off. + +To create the trigger, use the `trigger create` command: + +```zsh +$ slack trigger create --trigger-def "triggers/greeting_trigger.ts" +``` + +Since you haven't installed this trigger to a workspace yet, you'll be prompted to install the trigger to a new workspace. Select an authorized workspace in which to install the app. + +When you select your workspace, you will be prompted to choose an app environment for the trigger. Choose the _Local_ option so you can interact with your app while developing locally. The CLI will then finish installing your trigger. + +Once your app's trigger is finished being installed, you'll see the following output: + +```zsh +📚 App Manifest + Created app manifest for "hello-world (local)" in "myworkspace" workspace + +🏠 Workspace Install + Installed "hello-world (local)" app to "myworkspace" workspace + Finished in 1.7s + +⚡ Trigger created + Trigger ID: ABCD1234EFGH + Trigger Type: shortcut + Trigger Name: Send a greeting + Trigger Created Time: 2023-03-31 10:02:15 -04:00 + Trigger Updated Time: 2023-03-31 10:02:15 -04:00 + URL: https://slack.com/shortcuts/ABCD1234EFGH/01d8db3db6ea1a9e05012a90028ed678 +``` + +See that "URL" in the output? Copy it from the terminal output — that's going to be how we start our workflow — and head to the next section to try it out! + + + +## Starting a local development server + +With our Shortcut URL in hand (or, rather, in our clipboard), paste it into any public channel in your workspace. This will unfurl into a card with a **Run** button. You will also see your shortcut in the bookmarks bar in the `workflows` folder. + +If you try to interact with your app right now, nothing will happen since our local development server isn't running yet. So let's get our local server running with the `run` command: + +```zsh +$ slack run +``` + +Once your development server is running, click the **Run** button on the unfurled card, or select your shortcut's name from the `workflows` folder in the bookmark bar to start the workflow assigned to that trigger. + +In the window that pops up, fill out the form and click the **Send greeting** button. + +In the channel from which you executed the workflow, you'll see a new message for the user you selected in the form. + +--- + +So far we have: +- created an app based on the "Hello World" app template +- created a **trigger** to interact with our app, which produced a **Shortcut URL** +- started a **local development server** + +But we've only scratched the surface. The trigger you created is configured to call a [workflow](/deno-slack-sdk/guides/creating-workflows), and each workflow is configured to call one or more [functions](/deno-slack-sdk/guides/creating-slack-functions). + +In the next section, let's dive in to see how everything is wired together! + +## Take a look around: workflows + +Let's open up the trigger file `triggers/greeting_trigger.ts`, and see how it relates to the rest of the app: + +```js +// greeting_trigger.ts + +import { Trigger } from "deno-slack-api/types.ts"; +import GreetingWorkflow from "../workflows/greeting_workflow.ts"; + +const greetingTrigger: Trigger = { + type: "shortcut", + name: "Send a greeting", + description: "Send greeting to channel", + workflow: "#/workflows/greeting_workflow", + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + channel: { + value: "{{data.channel_id}}", + }, + }, +}; + +export default greetingTrigger; +``` + +Triggers take inputs and pass them along to an assigned workflow. Our trigger above is configured to invoke the `greeting_workflow`; notice the special string formatting for calling the workflow's name. + +When you create a trigger using a trigger definition like this one, your app will look for that workflow in all the workflows that you have registered in your manifest. + +Let's go back to the parent folder of our project and open up the `manifest.ts` file next: + +```js +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; +import GreetingWorkflow from "./workflows/greeting_workflow.ts"; + +export default Manifest({ + name: "deno-hello-world", + description: + "A sample that demonstrates using a function, workflow and trigger to send a greeting", + icon: "assets/default_new_app_icon.png", + workflows: [GreetingWorkflow], + outgoingDomains: [], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +Here you can see the `workflows` property in your app's manifest. This is where you will list out all of your workflows. + +Notice at the top there's something that we saw in the trigger file, too: an import call to `GreetingWorkflow`. + +The manifest registers the workflow, and then the trigger is configured to invoke it. With that in mind, let's open up that workflow to see what's going on: + +```js +// workflows/greeting_workflow.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { GreetingFunctionDefinition } from "../functions/greeting_function.ts"; + +// Here we define a new workflow called GreetingWorkflow, configuring its +// required input parameters. Note how one of the input parameters is of type +// `Schema.slack.types.interactivity`: +const GreetingWorkflow = DefineWorkflow({ + callback_id: "greeting_workflow", + title: "Send a greeting", + description: "Send a greeting to channel", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + required: ["interactivity"], + }, +}); + +// Once the workflow is defined, we can "add steps" to the workflow with the +// titular method `addStep`. In this case, we're using the Slack function +// `OpenForm` to leverage that interactivity input parameter in order to +// interact with the user (with a form): +const inputForm = GreetingWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Send a greeting", + interactivity: GreetingWorkflow.inputs.interactivity, + submit_label: "Send greeting", + fields: { + elements: [{ + name: "recipient", + title: "Recipient", + type: Schema.slack.types.user_id, + }, { + name: "channel", + title: "Channel to send message to", + type: Schema.slack.types.channel_id, + default: GreetingWorkflow.inputs.channel, + }, { + name: "message", + title: "Message to recipient", + type: Schema.types.string, + long: true, + }], + required: ["recipient", "channel", "message"], + }, + }, +); + +// After the first step, which is to send the form, we use the form data +// in subsequent steps. Here, we are passing it along as inputs to +// a custom function defined by `GreetingFunctionDefinition`. You'll note that +// we also imported this into our workflow file. +const greetingFunctionStep = GreetingWorkflow.addStep( + GreetingFunctionDefinition, + { + recipient: inputForm.outputs.fields.recipient, + message: inputForm.outputs.fields.message, + }, +); + +// Finally, we're using another Slack function called `SendMessage` to +// send the results of our custom function to a channel specified by the +// user filling out the form: +GreetingWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: inputForm.outputs.fields.channel, + message: greetingFunctionStep.outputs.greeting, +}); + +export default GreetingWorkflow; +``` + +The trigger invokes the workflow, and the workflow invokes one or more [custom](/deno-slack-sdk/guides/creating-custom-functions) or [built-in](/deno-slack-sdk/guides/creating-slack-functions) functions. The workflow is also registered in the app's manifest. + +Adding custom functions to your app is very similar to adding workflows, except you don't have to register them in the manifest; any functions that your workflows use are automatically registered with your app. + +In our next section, let's take a look at the custom function that our workflow uses. + +## Take a look around: functions + +Inside the `functions` folder we'll find the star of the show, our "Greeting" function, +in `greeting_function.ts`. + +This file contains both the function definition and its implementation. + +At the top, just after the imports, is the definition: + +```js +// greeting_function.ts + +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +export const GreetingFunctionDefinition = DefineFunction({ + callback_id: "greeting_function", + title: "Generate a greeting", + description: "Generate a greeting", + source_file: "functions/greeting_function.ts", + input_parameters: { + properties: { + recipient: { + type: Schema.slack.types.user_id, + description: "Greeting recipient", + }, + message: { + type: Schema.types.string, + description: "Message to the recipient", + }, + }, + required: ["message"], + }, + output_parameters: { + properties: { + greeting: { + type: Schema.types.string, + description: "Greeting for the recipient", + }, + }, + required: ["greeting"], + }, +}); +``` + +Notice how it looks very similar to our workflow definition; we have inputs, outputs, and the option to mark parameters required. + +Below that is its implementation: + +```js +export default SlackFunction( + GreetingFunctionDefinition, + ({ inputs }) => { + const { recipient, message } = inputs; + const salutations = ["Hello", "Hi", "Howdy", "Hola", "Salut"]; + const salutation = + salutations[Math.floor(Math.random() * salutations.length)]; + const greeting = + `${salutation}, <@${recipient}>! :wave: Someone sent the following greeting: \n\n>${message}`; + return { outputs: { greeting } }; + }, +); +``` + +If you go back to the workflow file, you'll see that when this function is added as a step to the workflow, the first context property we pass along is its definition. This gives us strong typing right out of the box for our custom functions. + +With our trigger calling our workflow, our workflow calling our functions, and our functions automating things in our workspace, we've now seen a very small sampling of what workflow apps can do! + +## Wrap it up + +In this tutorial we've taken a tour of the "Hello World" sample app. This example only shows one [trigger](/deno-slack-sdk/guides/using-triggers), [workflow](/deno-slack-sdk/guides/creating-workflows), and [custom function](/deno-slack-sdk/guides/creating-custom-functions), and is limited to essentially passing a string around. + +### Next steps + +For your next challenge, perhaps consider creating [a bot to welcome users to your workspace](/deno-slack-sdk/tutorials/welcome-bot)! \ No newline at end of file diff --git a/docs/tutorials/open-authorization.md b/docs/tutorials/open-authorization.md new file mode 100644 index 00000000..e15c3bc8 --- /dev/null +++ b/docs/tutorials/open-authorization.md @@ -0,0 +1,506 @@ +# Open authorization + + + +This tutorial guides you through setting up [external authentication](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication) in an app via OAuth2. The sample app for this tutorial demonstrates multi-stage workflows for requesting and collecting feedback on messages that starts with the press of a reaction, with responses stored dynamically in a Google Sheet. Because the focus of this tutorial is setting up OAuth2, we won't be going through every workflow and function, but you can check them out in the [sample app repo on GitHub](https://github.com/slack-samples/deno-simple-survey). + +✨ **First time creating a workflow app?** Try a basic app to build your confidence, such as [Hello World](/deno-slack-sdk/tutorials/hello-world-app)! + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + +## Choose your adventure + +After you've [installed the command-line interface](/deno-slack-sdk/guides/getting-started) you have two ways you can get started: + +### Use a blank app + +You can create a blank app with the Slack CLI using the following command: + +```bash +slack create simple-survey-app --template https://github.com/slack-samples/deno-blank-template +``` + +### Use a pre-built app + +Or, you can use the pre-built [Simple Survey app](https://github.com/slack-samples/deno-simple-survey): + +```bash +slack create simple-survey-app --template https://github.com/slack-samples/deno-simple-survey +``` + +Because this tutorial will not go through creating all of the files, a pre-built app might be easiest to follow along with. + +Once you have your new project ready to go, change into your project directory. + +## Explore the app structure + +Let's take a look at what's inside our new "Simple Survey" project directory: + +``` +assets/ +datastores/ +deno.jsonc +deno.lock +external_auth/ +functions/ +import_map.json +LICENSE +manifest.ts +README.md +slack.json +triggers/ +workflows/ +``` + +The first place to direct your attention are the `datastores`, `functions`, `triggers`, and `workflows` folders. These are where the definitions and implementations for the inner workings of your app live. As you might have guessed, the `external_auth` folder is where we'll define our external authentication. + +The next place to look is the `manifest.ts` file. This contains your app's manifest, which is where we can configure things like bot scopes and tell our app about our workflows. We'll return to the manifest a bit later. + +Other items in the project include: +* `.slack/`: a home for internal configuration files, scripts hooks, and the app SDK. This directory must be checked into your version control. You'll also notice a `.slack/apps.dev.json` once you begin building: this file is in `.gitignore` and should not be checked in to version control. +* `import_map.json`: a helper file for Deno that specifies where modules should be imported from. +* `assets/`: a place to store assets related to the project. This is a great place to store the icon that your app will display when users interact with it. + +## Explore the supported workflows + +This app's functionality is centered around six workflows. While we won't go into detail on all of them, it's important to know how they all fit together so that we can understand the flow of data later on. Here are the workflows: +* **Prompt survey creation**: Ask if a user wants to create a survey when a 📋 reaction is added to a message. +* **Create a survey**: Respond to the reacted message with a feedback form and make a new spreadsheet to store responses. +* **Respond to a survey**: Open the feedback form and store responses in the spreadsheet. +* **Remove a survey**: Delete messages with survey and surveying users for reaction events. +* **Event configurator**: Update the channels to survey and surveying users for reaction events. +* **Maintenance job**: A daily run workflow that ensures bot user membership in channels specified for event reaction triggers. Recommended for production-grade operations. + +Now that we've covered the app's basic structure, let's look at hooking up OAuth2 by first preparing our Google services. + +## Get your Google credentials + +With [external authentication](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication), you can programmatically interact with Google services and APIs from your app. + +The client credentials needed for these interactions can be collected from a Google Cloud project with OAuth enabled and with access to the appropriate services. + +### Create a Google Cloud Project + +Begin by creating a new project from the [Google Cloud resource manager](https://console.cloud.google.com/cloud-resource-manager), then enable the Google Sheets API for this project. + +Next, create an OAuth consent screen for your app. The "User Type" and other required app information can be configured as you wish. No additional scopes need to be added here, and you can add test users for development if you want. + +Client credentials can be collected by creating an OAuth client ID with an application type of "Web application". Under the "Authorized redirect URIs" section, add https://oauth2.slack.com/external/auth/callback, then click "Create". + +You'll use these newly-created client credentials in the next steps. + +## Define your OAuth2 provider + +Next up, we'll define the OAuth2 provider. Open `/external_auth/google_provider.ts` to see how that's done. + +Take your client ID and add it as the value for `client_id` where it's marked below. + +```javascript +// google_provider.ts + +import { DefineOAuth2Provider, Schema } from "deno-slack-sdk/mod.ts"; + +/** + * External authentication uses the OAuth 2.0 protocol to connect with + * accounts across various services. Once authenticated, an access token + * can be used to interact with the service on behalf of the user. + */ +const GoogleProvider = DefineOAuth2Provider({ + provider_key: "google", + provider_type: Schema.providers.oauth2.CUSTOM, + options: { + "provider_name": "Google", + "authorization_url": "https://accounts.google.com/o/oauth2/auth", + "token_url": "https://oauth2.googleapis.com/token", + "client_id": "", // TODO: Add your Client ID here! + "scope": [ + "https://www.googleapis.com/auth/spreadsheets", + "https://www.googleapis.com/auth/userinfo.email", + ], + "authorization_url_extras": { + "prompt": "consent", + "access_type": "offline", + }, + "identity_config": { + "url": "https://www.googleapis.com/oauth2/v1/userinfo", + "account_identifier": "$.email", + }, + }, +}); + +export default GoogleProvider; +``` +Once complete, your app needs to be deployed to Slack in order to create an environment for storing your external authentication client secret and access token. Run the following in your terminal: + +```bash +slack deploy +``` + +Running these commands will warn you that a client secret must be added for your OAuth2 provider. Don't worry about that for now; we'll take care of this in a future step! + +## Add the provider to your app manifest + +At the root of every app, there exists an app manifest, which defines how an app presents itself. + +The app [manifest](/deno-slack-sdk/guides/using-the-app-manifest) is where we configure the app's name and scopes, and declare which workflows our app uses. We will also need to add our new provider as an `externalAuthProvider`. + +```javascript +// manifest.ts + +import { Manifest } from "deno-slack-sdk/mod.ts"; + +import GoogleProvider from "./external_auth/google_provider.ts"; +import SurveyDatastore from "./datastores/survey_datastore.ts"; + +import ConfiguratorWorkflow from "./workflows/configurator.ts"; +import MaintenanceJobWorkflow from "./workflows/maintenance_job.ts"; + +import AnswerSurveyWorkflow from "./workflows/answer_survey.ts"; +import CreateSurveyWorkflow from "./workflows/create_survey.ts"; +import RemoveSurveyWorkflow from "./workflows/remove_survey.ts"; +import PromptSurveyWorkflow from "./workflows/prompt_survey.ts"; + +export default Manifest({ + name: "Simple Survey", + description: "Gather input and ideas at the press of a reacji", + icon: "assets/default_new_app_icon.png", + externalAuthProviders: [GoogleProvider], //Here is where we tell our app about the Google provider. + datastores: [SurveyDatastore], + workflows: [ + ConfiguratorWorkflow, + MaintenanceJobWorkflow, + AnswerSurveyWorkflow, + CreateSurveyWorkflow, + PromptSurveyWorkflow, + RemoveSurveyWorkflow, + ], + outgoingDomains: ["sheets.googleapis.com"], + botScopes: [ + "channels:join", + "chat:write", + "chat:write.public", + "commands", + "datastore:read", + "datastore:write", + "reactions:read", + "triggers:read", + "triggers:write", + ], +}); +``` + +Now that the provider is created and added to the manifest, we can encrypt and store the client secret. + +## Encrypt and store the client secret + +With your client secret ready, run the following command in your terminal, replacing GOOGLE_CLIENT_SECRET with your own secret: + +```bash +slack external-auth add-secret --provider google --secret GOOGLE_CLIENT_SECRET +``` + +When prompted to select an app, choose the `dev` app only if you are running locally. + +If everything was successful, the CLI will let you know: +``` +✨ successfully added external auth client secret for google +``` + +## Initiate OAuth2 flow + +With your Google project created and the Client ID and secret set, you're ready to initiate the OAuth flow! + +If all the right values are in place, the following command will prompt you to choose an app, select a provider (hint: choose the Google one), then pick the Google account you want to authenticate with. This will open a browser window for you to complete the OAuth2 sign-in flow according to your provider's requirements. You'll know you're successful when your browser sends you to the oauth2.slack.com page stating that your account was successfully connected. + +```bash +slack external-auth add +``` + +Verify that a token has been created by re-running the `external-auth add` command. If you see `Token Exists? Yes`, then token creation was successful. + +You're nearly ready to create surveys at the press of a reaction! + +## Use in a custom function + +If you refer back to the workflow overview we explored in Step 1 above, you'll recall that after the survey creation is prompted, the `create_survey` workflow gets kicked off. We'll now look at how that workflow gets the token and passes it through the app to be used for communicating with Google services, demonstrating how it is used in both a custom function and a workflow. + +Turning your attention to `create_survey.ts`, you will see that the first step in the workflow is calling the `CreateGoogleSheetFunctionDefinition`. + +```javascript +// /workflows/create_survey.ts + +... + +// Step 1: Create a new Google spreadsheet +const sheet = CreateSurveyWorkflow.addStep( + CreateGoogleSheetFunctionDefinition, + { + google_access_token_id: {}, + title: CreateSurveyWorkflow.inputs.parent_ts, + }, +); + +... +``` + +That function's definition looks like this: + +```javascript +// /functions/create_google_sheet.ts + +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; + +/** + * Custom functions can gather OAuth access tokens from external + * authentication to perform individualized actions on external APIs. + */ +export const CreateGoogleSheetFunctionDefinition = DefineFunction({ + callback_id: "create_google_sheet", + title: "Create spreadsheet", + description: "Create a new Google Sheet", + source_file: "functions/create_google_sheet.ts", + input_parameters: { + properties: { + google_access_token_id: { + type: Schema.slack.types.oauth2, + oauth2_provider_key: "google", + }, + title: { + type: Schema.types.string, + description: "The title of the spreadsheet", + }, + }, + required: ["google_access_token_id", "title"], + }, + output_parameters: { + properties: { + google_spreadsheet_id: { + type: Schema.types.string, + description: "Newly created spreadsheet ID", + }, + google_spreadsheet_url: { + type: Schema.types.string, + description: "Newly created spreadsheet URL", + }, + reactor_access_token_id: { + type: Schema.types.string, + description: "The Google access token ID of the reactor", + }, + }, + required: [ + "google_spreadsheet_id", + "google_spreadsheet_url", + "reactor_access_token_id", + ], + }, +}); + +... + +``` + +This function takes an input parameter of `google_access_token_id`, which is the `Schema.slack.types.oauth2` type. Using this type indicates to the app that there must be OAuth2 provider defined inside this application and set up for this parameter. The value of the `oauth2_provider_key` property on this parameter must match the `provider_key` for an OAuth2 provider. For this app, that is "google." + +When the function receives the desired `oauth2` input, it can use the API client's provided `apps.auth.external.get` method to retrieve any stored third party token or credential secret, like this: + +```javascript +// create_google_sheet.ts + +... + +export default SlackFunction( + CreateGoogleSheetFunctionDefinition, + async ({ inputs, client }) => { + // Collect Google access token + const auth = await client.apiCall("apps.auth.external.get", { + external_token_id: inputs.google_access_token_id, + }); + + if (!auth.ok) { + return { error: `Failed to collect Google auth token: ${auth.error}` }; + } + + // Create spreadsheet + const url = "https://sheets.googleapis.com/v4/spreadsheets"; + const sheets = await fetch(url, { + method: "POST", + headers: { + "Authorization": `Bearer ${auth.external_token}`, + }, + body: JSON.stringify({ + properties: { title: `Slack Survey - ${inputs.title}` }, + sheets: [{ + properties: { title: "Responses" }, + data: [{ + rowData: [{ + values: [ + { userEnteredValue: { stringValue: "Submitted" } }, + { userEnteredValue: { stringValue: "Impression" } }, + { userEnteredValue: { stringValue: "Comments" } }, + ], + }], + }], + }], + }), + }); + + const body = await sheets.json(); + if (body.error) { + return { + error: `Failed to create the survey spreadsheet: ${body.error.message}`, + }; + } + + return { + outputs: { + google_spreadsheet_id: body.spreadsheetId, + google_spreadsheet_url: body.spreadsheetUrl, + reactor_access_token_id: inputs.google_access_token_id, + }, + }; + }, +); +``` + +Note how this function returns the `google_access_token_id` as an output, called `reactor_access_token_id`. This will be important in the next step, where we'll see how the token is used as an input to a workflow. + +## Use the auth token in a workflow + +The `CreateGoogleSheetFunctionDefinition` we explored above is called from `CreateSurveyWorkflow`. Looking back at that workflow, we see that its second step - calling the `CreateTriggerFunctionDefinition` - takes the `reactor_access_token_id` as a parameter, which we now know was an output of `CreateGoogleSheetFunctionDefinition`(also known as the OAuth2 token ID). If we dive into how `CreateTriggerFunctionDefinition` uses that, we'll see how to use the credential as a workflow input. + +```javascript +// /functions/create_survey_trigger.ts + +... + +export default SlackFunction( + CreateTriggerFunctionDefinition, + async ({ inputs, client }) => { + const { google_spreadsheet_id, reactor_access_token_id } = inputs; + + const trigger = await client.workflows.triggers.create< + typeof AnswerSurveyWorkflow.definition + >({ + type: "shortcut", + name: "Survey your thoughts", + description: "Share your thoughts about this post", + workflow: `#/workflows/${AnswerSurveyWorkflow.definition.callback_id}`, + inputs: { + interactivity: { value: "{{data.interactivity}}" }, + google_spreadsheet_id: { value: google_spreadsheet_id }, + reactor_access_token_id: { value: reactor_access_token_id }, //Here it is used in calling a workflow + }, + }); + + if (!trigger.ok || !trigger.trigger.shortcut_url) { + return { + error: `Failed to create link trigger for the survey: ${trigger.error}`, + }; + } + + return { + outputs: { + trigger_id: trigger.trigger.id, + trigger_url: trigger.trigger.shortcut_url, + }, + }; + }, +); + +``` + +After the definition of this function, we see how the `reactor_access_token_id` is used, and that is as an input to a workflow! Excellent. Turn your attention to the `/workflows/answer_survey.ts` file to see how that token ID is used in a workflow definition: + +```javascript +// answer_survey.ts + +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { SaveResponseFunctionDefinition } from "../functions/save_response.ts"; + +/** + * A workflow is a set of steps that are executed in order. + * Each step in a workflow is a function. + * This workflow uses interactivity. + */ +const AnswerSurveyWorkflow = DefineWorkflow({ + callback_id: "answer_survey", + title: "Respond to a survey", + description: "Add comments and feedback to a survey", + input_parameters: { + properties: { + interactivity: { type: Schema.slack.types.interactivity }, + google_spreadsheet_id: { + type: Schema.types.string, + description: "Spreadsheet ID for storing survey results", + }, + reactor_access_token_id: { + type: Schema.types.string, + description: "External authentication access token for the reactor", + }, + }, + required: [ + "interactivity", + "google_spreadsheet_id", + "reactor_access_token_id", + ], + }, +}); + +... + +``` + +The `reactor_access_token_id` is then used in the implementation of the workflow by passing it along, just like any other input parameter, to a step in the workflow: + +```javascript +// answer_survey.ts + +... + +// Step 2: Append responses to the spreadsheet +AnswerSurveyWorkflow.addStep(SaveResponseFunctionDefinition, { + reactor_access_token_id: AnswerSurveyWorkflow.inputs.reactor_access_token_id, + google_spreadsheet_id: AnswerSurveyWorkflow.inputs.google_spreadsheet_id, + impression: response.outputs.fields.impression, + comments: response.outputs.fields.comments, +}); + +export default AnswerSurveyWorkflow; + +``` + +Now you have seen how a Google token ID can be obtained from the app, used in a custom function, and used in a workflow. + +To complete the connection process, you need to let your app know what authenticated account you'll be using for specific workflows. + +For this app, only `CreateSurveyWorkflow` requires a configured external Google account, so we can set that up with our freshly-authed account. To do so, run the following: + +```sh +slack external-auth select-auth +``` + +Select the workspace and app environment for your app, then select the `#/workflows/create_survey` workflow to give it access to your Google account. Select the same provider and the external account that you authenticated with above. + +At last — you're all set to survey. Let's run our app! + +## Run your app + +If you embarked on this tutorial with a blank app, complete your app files using the [sample app](https://github.com/slack-samples/deno-simple-survey) as your guide. If you used a pre-built app, we're ready to run it! + +To see this app in action, run the following command in your terminal: +```bash +slack run +``` + +After you've chosen your app and assigned it to your workspace, you can switch over to the app in Slack and try it out. This app is triggered by reactions in Slack, so it should be ready to use once deployed (as opposed to apps that need [link triggers](/deno-slack-sdk/guides/creating-link-triggers) to run). + +## Next steps + +Congratulations, you've added OAuth2 to a workflow app! Where do you go from here? + +Consider taking a deep dive into our [External authorization](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication) documentation. + +Perhaps try another tutorial, like creating a [social app to log runs with virtual running buddies](/deno-slack-sdk/tutorials/virtual-running-buddies-app). \ No newline at end of file diff --git a/docs/tutorials/request-time-off-app.md b/docs/tutorials/request-time-off-app.md new file mode 100644 index 00000000..37cc9579 --- /dev/null +++ b/docs/tutorials/request-time-off-app.md @@ -0,0 +1,296 @@ +# Request time off app + + + +This tutorial will guide you in creating, running, and deploying a workflow app. The Request Time Off App models how to collect user inputs as well as how to send those inputs to other users in Slack. More specifically, this app showcases one way [user interactivity](/deno-slack-sdk/guides/creating-a-form) is implemented within an app. By the end, you will have a working app that can post [Block Kit](https://api.slack.com/block-kit/interactivity) messages, handle user interactions, and update messages in real time. + +✨ **First time creating a workflow app?** Try an app to build your confidence, such as [Hello World](/deno-slack-sdk/tutorials/hello-world-app)! + +We can break this app into 3 major parts that work together to create a symphonic harmony: +1. Functions +2. Workflows +3. Triggers + +Each segment will give an explanation of the components, along with some tips & tricks for orchestrating a successful path forward. + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + +## Choose your adventure + +After you've [installed the command-line interface](/deno-slack-sdk/guides/getting-started) you have two ways you can get started: + +### Use a blank app + +You can create a blank app with the Slack CLI using the following command: + +```bash +slack create request-time-off-app --template https://github.com/slack-samples/deno-blank-template +``` + +### Use a pre-built app + +Or, you can use the pre-built [Request Time Off app](https://github.com/slack-samples/deno-request-time-off): + +``` +slack create request-time-off-app --template https://github.com/slack-samples/deno-request-time-off +``` + +Once you have your new project ready to go, change into your project directory. + +## Compose the manifest + +The app manifest is where we define the intricacies of an app. Below is the manifest that powers the Request Time Off app: + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; +import { CreateTimeOffRequestWorkflow } from "./workflows/CreateTimeOffRequestWorkflow.ts"; +import { SendTimeOffRequestToManagerFunction } from "./functions/send_time_off_request_to_manager/definition.ts"; + +export default Manifest({ + name: "Request Time Off", + description: "Ask your manager for some time off", + icon: "assets/default_new_app_icon.png", + workflows: [CreateTimeOffRequestWorkflow], + functions: [SendTimeOffRequestToManagerFunction], + outgoingDomains: [], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "datastore:read", + "datastore:write", + ], +}); + +``` +The manifest of an app describes the most important application information, such as its `name`, `description`, `icon`, the list of [workflows](/deno-slack-sdk/guides/creating-workflows) and [functions](/deno-slack-sdk/guides/creating-custom-functions), and more. Read through the full [manifest documentation](/deno-slack-sdk/guides/using-the-app-manifest) to learn more. + +## Create a function + +First we will define and implement our function. [Functions](/deno-slack-sdk/guides/creating-custom-functions) are reusable building blocks that accept inputs, perform calculations, and provide outputs. + +The code behind the app's function is stored under the `./functions/send_time_off_request_to_manager/` directory. We're working with five files inside (not including test files): + +1. `block_actions.ts`: An action handler for our interactive blocks. +2. `blocks.ts`: A layout of visual blocks that is easy on the eyes. +3. `constants.ts`: Constant variables referenced throughout the app. +4. `definition.ts`: Our function definition, which houses the function's `input_parameters`, `output_parameters`, `title`, `description` and implementation source file. This is a [custom function](/deno-slack-sdk/guides/creating-custom-functions) as opposed to [Slack function](/deno-slack-sdk/guides/creating-slack-functions), meaning the function implementation is up to you! _Notice the `interactivity` parameter of type `Schema.slack.types.interactivity` -- one of the many built-in [Slack types](/deno-slack-sdk/reference/slack-types#interactivity) available to allow your function to utilize user interaction._ +5. `mod.ts`: Our function implementation. + +### Implement a function + +Once you define your custom function, we'll bring it to life by completing the `mod.ts` file with various [API calls](/deno-slack-sdk/guides/calling-slack-api-methods) and Block Kit [blocks](https://api.slack.com/reference/block-kit/blocks). + +Remember, the Request Time Off app collects time off start and end dates, and sends that request to a manager for approval. We can utilize [Block Kit](https://api.slack.com/reference/block-kit/blocks) buttons to help facilitate the decision process and to create a rich user experience. + +```javascript +import { SendTimeOffRequestToManagerFunction } from "./definition.ts"; +import { SlackFunction } from "deno-slack-sdk/mod.ts"; +import BlockActionHandler from "./block_actions.ts"; +import { APPROVE_ID, DENY_ID } from "./constants.ts"; +import timeOffRequestHeaderBlocks from "./blocks.ts"; + +// Custom function that sends a message to the user's manager asking +// for approval for the time off request. The message includes some Block Kit with two +// interactive buttons: one to approve, and one to deny. +export default SlackFunction( + SendTimeOffRequestToManagerFunction, + async ({ inputs, client }) => { + console.log("Forwarding the following time off request:", inputs); + + // Create a block of Block Kit elements composed of several header blocks + // plus the interactive approve/deny buttons at the end + const blocks = timeOffRequestHeaderBlocks(inputs).concat([{ + "type": "actions", + "block_id": "approve-deny-buttons", + "elements": [ + { + type: "button", + text: { + type: "plain_text", + text: "Approve", + }, + action_id: APPROVE_ID, // <-- important! we will differentiate between buttons using these IDs + style: "primary", + }, + { + type: "button", + text: { + type: "plain_text", + text: "Deny", + }, + action_id: DENY_ID, // <-- important! we will differentiate between buttons using these IDs + style: "danger", + }, + ], + }]); + // ...continued in the next snippet +``` + +Now we have a message with two buttons, each using a unique `ACTION_ID` to differentiate between an approval or denial. In order to properly utilize the Block Kit buttons, we'll rely on the [`BlockActionsHandler`](https://api.slack.com/reference/interaction-payloads/block-actions) to route the button actions. Check it out below: + + +```javascript +// ...continued from the snippet above + // Send the message to the manager + const msgResponse = await client.chat.postMessage({ + channel: inputs.manager, + blocks, + // Fallback text to use when rich media can't be displayed (i.e. notifications) as well as for screen readers + text: "A new time off request has been submitted", + }); + + if (!msgResponse.ok) { + console.log("Error during request chat.postMessage!", msgResponse.error); + } + + // IMPORTANT! Set `completed` to false in order to keep the interactivity + // points (the approve/deny buttons) "alive" + // We will set the function's complete state in the button handlers below. + return { + completed: false, + }; + }, + // Create an 'actions router' which is a helper utility to route interactions + // with different interactive Block Kit elements (like buttons!) +).addBlockActionsHandler( + // listen for interactions with components with the following action_ids + [APPROVE_ID, DENY_ID], + // interactions with the above components get handled by the function below + BlockActionHandler, +); +``` + +This `mods.ts` function is responsible for building a message, sending it to the selected manager, and replying with a response that is triggered by the decision of that manager. How do we connect these function steps, you may ask? Not to worry, our next step covers how to bring together the functions using a [workflow](/deno-slack-sdk/guides/creating-workflows)! + +## Define a workflow + +A [workflow](/deno-slack-sdk/guides/creating-workflows) is a set of steps that are executed in order. Each step in a workflow can be a [function](/deno-slack-sdk/guides/creating-custom-functions). Similar to functions, workflows can also optionally accept inputs and pass them further along to other functions that comprise the workflow. + +This app contains a single workflow stored within the `workflows/` folder. + +This app's workflow is composed of two functions chained sequentially as steps: + +1. The workflow uses the [OpenForm Slack function](/deno-slack-sdk/reference/slack-functions/open_form) to collect data from the user that started the workflow. +2. Form data is then passed to your app's [custom function](/deno-slack-sdk/guides/creating-custom-functions), which is called `SendTimeOffRequestToManagerFunction`. This function is stored within the `functions/` folder. + +First let's define the workflow with the `DefineWorkflow` method. Make sure to set a custom `callback_id` that you can reference later on. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { SendTimeOffRequestToManagerFunction } from "../functions/send_time_off_request_to_manager/definition.ts"; + +/** + * A Workflow composed of two steps: asking for time off details from the user + * that started the workflow, and then forwarding the details along with two + * buttons (approve and deny) to the user's manager. + */ +export const CreateTimeOffRequestWorkflow = DefineWorkflow({ + callback_id: "create_time_off", + title: "Request Time Off", + description: + "Create a time off request and send it for approval to your manager", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + }, + required: ["interactivity"], + }, +}); +``` + +Then, place the functions in order of execution. In this case, use the Slack [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) function to open a modal form to collect the time off request data; then use the [custom function](/deno-slack-sdk/guides/creating-custom-functions) you built to send the request for approval. + + +```javascript +// Step 1: opening a form for the user to input their time off details. +const formData = CreateTimeOffRequestWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Time Off Details", + interactivity: CreateTimeOffRequestWorkflow.inputs.interactivity, + submit_label: "Submit", + description: "Enter your time off request details", + fields: { + required: ["manager", "start_date", "end_date"], + elements: [ + { + name: "manager", + title: "Manager", + type: Schema.slack.types.user_id, + }, + { + name: "start_date", + title: "Start Date", + type: "slack#/types/date", + }, + { + name: "end_date", + title: "End Date", + type: "slack#/types/date", + }, + { + name: "reason", + title: "Reason", + type: Schema.types.string, + }, + ], + }, + }, +); + +// Step 2: send time off request details along with approve/deny buttons to manager +CreateTimeOffRequestWorkflow.addStep(SendTimeOffRequestToManagerFunction, { + interactivity: formData.outputs.interactivity, + employee: CreateTimeOffRequestWorkflow.inputs.interactivity.interactor.id, + manager: formData.outputs.fields.manager, + start_date: formData.outputs.fields.start_date, + end_date: formData.outputs.fields.end_date, + reason: formData.outputs.fields.reason, +}); + +``` +Voilà! Next, let's define a trigger to get the wheels in motion! + +## Create a trigger + +A [trigger](/deno-slack-sdk/guides/using-triggers) is a crucial finishing piece of your app. Creating a trigger sets the steps of your workflow in motion, which runs your custom & Slack functions, allowing your app to provide a pleasant experience. + +These triggers can be invoked by a user, or automatically as a response to an event within Slack. + +A [link trigger](/deno-slack-sdk/guides/creating-link-triggers) is a type of trigger that generates a shortcut URL which, when posted in a channel or added as a bookmark, becomes a link. When clicked, the link trigger will run the associated workflow. + +To create a link trigger for our workflow, run the following command: + +```bash +$ slack trigger create --trigger-def triggers/trigger.ts +``` + +After selecting a workspace and an app environment, the output provided will include the URL. Copy and paste this URL into a channel as a message, or add it as a bookmark in a channel of the workspace you selected. + +_Note: this link won't run the workflow until the app is either running locally or deployed! Read on to learn how to run your app locally and eventually deploy it to Slack hosting._ + +## Run your app + +You're almost to the end! Let's use development mode to run this workflow in Slack directly from the machine you're reading this from now: + +```bash +$ slack run +``` +After you've chosen your app and assigned it to your workspace, you can switch over to the app in Slack and try it out. Use the link trigger you created previously; when you paste the shortcut URL into the message box and post them, it'll unfurl and give you a button for invoking your workflow. + +## Great work! + +Congratulations! You've successfully built an approval workflow app, providing fancy buttons to all who request time off. Now that we've posted a message using Block Kit, handled the user interaction of buttons, and updated a message — you have the capability to either extend this app or to create a new one from scratch. + +### Next steps + +For your next challenge, perhaps consider creating [a social app to log runs with virtual running buddies](/deno-slack-sdk/tutorials/virtual-running-buddies-app)! + + + diff --git a/docs/tutorials/virtual-running-buddies-app.md b/docs/tutorials/virtual-running-buddies-app.md new file mode 100644 index 00000000..508ecc7c --- /dev/null +++ b/docs/tutorials/virtual-running-buddies-app.md @@ -0,0 +1,779 @@ +# Virtual running buddies app + + + +In this tutorial, you'll learn how to create a social app to log runs with your virtual running buddies. + +We'll pace you along the path to design a [datastore](/deno-slack-sdk/guides/using-datastores), craft a [custom type](/deno-slack-sdk/guides/creating-a-custom-type), tailor [triggers](/deno-slack-sdk/guides/using-triggers), wire [workflows](/deno-slack-sdk/guides/creating-workflows), and form [functions](/deno-slack-sdk/guides/creating-slack-functions). Now that you're familiar with the course map, let's head to the start line. + +✨ **First time creating a workflow app?** Try a basic app to build your confidence, such as [Hello World](/deno-slack-sdk/tutorials/hello-world-app)! + +Before we begin, ensure you have the following prerequisites completed: + +* Install the Slack CLI. +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the Getting started, then come on back. + +--- + +## Step 1: Create an app to complete your warmup {#warmup} + +Each Slack app built using the CLI begins with the same steps. Make sure you have everything you need before you show up at the start line. + +After you've [installed the command-line interface](/deno-slack-sdk/guides/getting-started), you have two ways you can get started. + + + + +You can create a blank app with the Slack CLI using the following command: + +```bash +slack create virtual-running-buddies-app --template https://github.com/slack-samples/deno-blank-template +``` + + + + + +Or, you can use the pre-built [Virtual Running Buddies app](https://github.com/slack-samples/deno-virtual-running-buddies): + +``` +slack create virtual-running-buddies-app --template https://github.com/slack-samples/deno-virtual-running-buddies +``` + + + + +Once you have your new project ready to go, change into your project directory and get to the start line. + +--- + +## Step 2: Map your course with your manifest {#map-course} + +Determining the definitions and manifest of your app allows you to create a course map of where you want to go. Open your text editor (we recommend VSCode with the [Deno plugin](https://marketplace.visualstudio.com/items?itemName=denoland.vscode-deno)) and point to the directory you created with the `slack` command. + +Import and add the following definitions to your [app's manifest](/deno-slack-sdk/guides/using-the-app-manifest): + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; +import RunningDatastore from "./datastores/run_data.ts"; +import LogRunWorkflow from "./workflows/log_run_workflow.ts"; +import DisplayLeaderboardWorkflow from "./workflows/display_leaderboard_workflow.ts"; +import { RunnerStatsType } from "./types/runner_stats.ts"; + +export default Manifest({ + name: "my-run-app", + description: "Log runs with virtual running buddies!", + icon: "assets/icon.png", + workflows: [LogRunWorkflow, DisplayLeaderboardWorkflow], + outgoingDomains: [], + datastores: [RunningDatastore], + types: [RunnerStatsType], + botScopes: [ + "commands", + "chat:write", + "chat:write.public", + "datastore:read", + "datastore:write", + "channels:read", + "triggers:write", + ], +}); + +``` + +You'll notice a lot of stuff we haven't talked about yet, but not to worry! We'll cover everything in the following steps. + +--- + +## Step 3: Define a datastore {#datastore} +We need a way to store all the information that our running buddies log, including their user ID (`runner`), how long they ran (`distance`), and the date they ran (`rundate`). Enter [datastores](/deno-slack-sdk/guides/using-datastores)! + +Datastores have three main properties: + +* `name`: to identify your datastore. +* `primary_key`: the attribute to be used as the datastore's primary key (ensure this is an actual attribute that you have defined), which we'll use for querying information later. For more information, refer to [querying the datastore](/deno-slack-sdk/guides/retrieving-items-from-a-datastore). +* `attributes`: to scaffold your datastore's columns. + +For what we need, the following datastore will do the trick. Let's create a `datastores` folder and add a file called `run_data.ts` with our datastore definition: + +```javascript +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const RUN_DATASTORE = "running_datastore"; + +const RunningDatastore = DefineDatastore({ + name: RUN_DATASTORE, + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + runner: { + type: Schema.slack.types.user_id, + }, + distance: { + type: Schema.types.number, + }, + rundate: { + type: Schema.slack.types.date, + }, + }, +}); + +export default RunningDatastore; + +``` + +We already did this earlier when we defined our manifest — but if you hadn't yet, you would need to import your datastore within the manifest file: + +`import RunningDatastore from "./datastores/run_data.ts";` + +And then, register it: + +`datastores: [RunningDatastore],` + +Additionally, for any datastore, you'll need to add the following bot scopes to your manifest: + +* `datastore:read` +* `datastore:write` + +--- + +## Step 4: Craft a custom type {#custom-type} + +The next thing we'll do is define a custom type for our runners' recent runs. Since we're going to be passing this information to our datastore, workflows, and functions, having our own custom reusable type will make our lives a little easier. + +✨ **For more information about custom types and how to define them**, refer to [custom types](/deno-slack-sdk/guides/creating-a-custom-type). + +We'll create a new `types` folder with a file called `runner_stats.ts` with our custom type definition: + +```javascript +import { DefineType, Schema } from "deno-slack-sdk/mod.ts"; + +export const RunnerStatsType = DefineType({ + title: "Runner Stats", + description: "Information about the recent runs for a runner", + name: "runner_stats", + type: Schema.types.object, + properties: { + runner: { type: Schema.slack.types.user_id }, + weekly_distance: { type: Schema.types.number }, + total_distance: { type: Schema.types.number }, + }, + required: ["runner", "weekly_distance", "total_distance"], +}); + +``` + +Just like with our datastore, we'll also verify that we imported our custom type into our manifest earlier: + +`import { RunnerStatsType } from "./types/runner_stats.ts";` + +And then, make sure it's registered: + +`types: [RunnerStatsType],` + +--- + +## Step 5: Wire your workflows {#workflows} + +Next, let's define our workflows — we'll have two of them. Don't worry about functions yet; we'll get to those next. For now, you can just import and call them as shown in our workflows. + +For our first workflow, we need three steps to accomplish the following: + +1. Allow a runner on our team to log their run details. The Slack function [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) is used to collect data. +2. Save those details to the datastore we defined earlier. +3. Post a message of encouragement to the runner. Every runner loves a good cheering section! + +Let's create a `workflows` folder and add a file called `log_run_workflow.ts` with the following workflow definition: + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { LogRunFunction } from "../functions/log_run.ts"; + +const LogRunWorkflow = DefineWorkflow({ + callback_id: "log_run_workflow", + title: "Log a run", + description: "Collect and store info about a recent run", + input_parameters: { + properties: { + interactivity: { type: Schema.slack.types.interactivity }, + channel: { type: Schema.slack.types.channel_id }, + user_id: { type: Schema.slack.types.user_id }, + }, + required: ["interactivity", "channel", "user_id"], + }, +}); + +// Step 1: Collect run information with a form +const inputForm = LogRunWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Log your run", + interactivity: LogRunWorkflow.inputs.interactivity, + submit_label: "Submit run", + fields: { + elements: [{ + name: "runner", + title: "Runner", + type: Schema.slack.types.user_id, + default: LogRunWorkflow.inputs.user_id, + }, { + name: "distance", + title: "Distance (in miles)", + type: Schema.types.number, + minimum: 0, + }, { + name: "rundate", + title: "Run date", + type: Schema.slack.types.date, + default: new Date().toISOString().split("T")[0], // YYYY-MM-DD + }, { + name: "channel", + title: "Channel to send entry to", + type: Schema.slack.types.channel_id, + default: LogRunWorkflow.inputs.channel, + }], + required: ["channel", "runner", "distance", "rundate"], + }, + }, +); + +// Step 2: Save run info to the datastore +LogRunWorkflow.addStep(LogRunFunction, { + runner: inputForm.outputs.fields.runner, + distance: inputForm.outputs.fields.distance, + rundate: inputForm.outputs.fields.rundate, +}); + +// Step 3: Post a message about the run +LogRunWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: inputForm.outputs.fields.channel, + message: + `:athletic_shoe: <@${inputForm.outputs.fields.runner}> submitted ${inputForm.outputs.fields.distance} mile(s) on ${inputForm.outputs.fields.rundate}. Keep up the great work!`, +}); + +export default LogRunWorkflow; +``` + +--- + +For our second workflow, we need to generate our leaderboard. To do this, our workflow contains the following steps: + +1. Gather our team's runs. +2. Gather each individual's runs. +3. Format our leaderboard. +4. Post the leaderboard to our channel. + +```javascript +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { CollectTeamStatsFunction } from "../functions/collect_team_stats.ts"; +import { CollectRunnerStatsFunction } from "../functions/collect_runner_stats.ts"; +import { FormatLeaderboardFunction } from "../functions/format_leaderboard.ts"; + +const DisplayLeaderboardWorkflow = DefineWorkflow({ + callback_id: "display_leaderboard_workflow", + title: "Display the leaderboard", + description: + "Show team statistics and highlight the top runners from the past week", + input_parameters: { + properties: { + channel: { type: Schema.slack.types.channel_id }, + interactivity: { type: Schema.slack.types.interactivity }, + }, + required: ["channel", "interactivity"], + }, +}); + +// Step 1: Gather team stats from the past week +const teamStats = DisplayLeaderboardWorkflow.addStep( + CollectTeamStatsFunction, + {}, +); + +// Step 2: Collect individual runner stats +const runnerStats = DisplayLeaderboardWorkflow.addStep( + CollectRunnerStatsFunction, + {}, +); + +// Step 3: Format the leaderboard message +const leaderboard = DisplayLeaderboardWorkflow.addStep( + FormatLeaderboardFunction, + { + team_distance: teamStats.outputs.weekly_distance, + percent_change: teamStats.outputs.percent_change, + runner_stats: runnerStats.outputs.runner_stats, + }, +); + +// Step 4: Post the leaderboard message to channel +DisplayLeaderboardWorkflow.addStep(Schema.slack.functions.SendMessage, { + channel_id: DisplayLeaderboardWorkflow.inputs.channel, + message: + `${leaderboard.outputs.teamStatsFormatted}\n\n${leaderboard.outputs.runnerStatsFormatted}`, +}); + +export default DisplayLeaderboardWorkflow; +``` + +Our workflows also need to be imported into our manifest, so let's just double-check the following lines are there: +`import LogRunWorkflow from "./workflows/log_run_workflow.ts";` +`import DisplayLeaderboardWorkflow from "./workflows/display_leaderboard_workflow.ts";` + +And that they are registered: + +`workflows: [LogRunWorkflow, DisplayLeaderboardWorkflow],` + +--- + +## Step 6: Form your functions {#functions} + +Following fast, functions: we'll fashion four. + +The first function will store our collected run data in our datastore (so don't forget to import it first): + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { RUN_DATASTORE } from "../datastores/run_data.ts"; + +export const LogRunFunction = DefineFunction({ + callback_id: "log_run", + title: "Log a run", + description: "Record a run in the datastore", + source_file: "functions/log_run.ts", + input_parameters: { + properties: { + runner: { + type: Schema.slack.types.user_id, + description: "Runner", + }, + distance: { + type: Schema.types.number, + description: "Distance", + }, + rundate: { + type: Schema.slack.types.date, + description: "Run date", + }, + }, + required: ["runner", "distance", "rundate"], + }, + output_parameters: { + properties: {}, + required: [], + }, +}); + +export default SlackFunction(LogRunFunction, async ({ inputs, client }) => { + const { distance, rundate, runner } = inputs; + const uuid = crypto.randomUUID(); + + const putResponse = await client.apps.datastore.put({ + datastore: RUN_DATASTORE, + item: { + id: uuid, + runner: runner, + distance: distance, + rundate: rundate, + }, + }); + + if (!putResponse.ok) { + return { error: `Failed to store run: ${putResponse.error}` }; + } + return { outputs: {} }; +}); + +``` +✨ **For more information about how data is stored and successful vs. unsuccessful payloads**, refer to [creating or updating an item](/deno-slack-sdk/guides/adding-items-to-a-datastore). + +--- + +Our second function calculates weekly and all-time total distance statistics for an individual runner. + +We'll query the datastore to get a runner's logged run details, calculate some statistics for that runner, and then return an array containing all of that information: + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import RunningDatastore, { RUN_DATASTORE } from "../datastores/run_data.ts"; +import { RunnerStatsType } from "../types/runner_stats.ts"; + +export const CollectRunnerStatsFunction = DefineFunction({ + callback_id: "collect_runner_stats", + title: "Collect runner stats", + description: "Gather statistics of past runs for all runners", + source_file: "functions/collect_runner_stats.ts", + input_parameters: { + properties: {}, + required: [], + }, + output_parameters: { + properties: { + runner_stats: { + type: Schema.types.array, + items: { type: RunnerStatsType }, + description: "Weekly and all-time total distances for runners", + }, + }, + required: ["runner_stats"], + }, +}); + +export default SlackFunction(CollectRunnerStatsFunction, async ({ client }) => { + // Query the datastore for all the data we collected + const runs = await client.apps.datastore.query< + typeof RunningDatastore.definition + >({ datastore: RUN_DATASTORE }); + + if (!runs.ok) { + return { error: `Failed to retrieve past runs: ${runs.error}` }; + } + + const runners = new Map(); + + const startOfLastWeek = new Date(); + startOfLastWeek.setDate(startOfLastWeek.getDate() - 6); + + // Add run statistics to the associated runner + runs.items.forEach((run) => { + const isRecentRun = run.rundate >= + startOfLastWeek.toLocaleDateString("en-CA", { timeZone: "UTC" }); + + // Find existing runner record or create new one + const runner = runners.get(run.runner) || + { runner: run.runner, total_distance: 0, weekly_distance: 0 }; + + // Add run distance to the runner's totals + runners.set(run.runner, { + runner: run.runner, + total_distance: runner.total_distance + run.distance, + weekly_distance: runner.weekly_distance + (isRecentRun && run.distance), + }); + }); + + // Return an array with runner stats + return { + outputs: { runner_stats: [...runners.entries()].map((r) => r[1]) }, + }; +}); + +``` + +✨ **For more information about how data is retrieved and successful vs. unsuccessful payloads**, refer to [retrieving a single item](/deno-slack-sdk/guides/retrieving-items-from-a-datastore#get) and [querying the datastore](/deno-slack-sdk/guides/retrieving-items-from-a-datastore#query). + +--- + +Our third function calculates the weekly and all-time total distance for the whole team, as well as the percentage difference between this week's runs and the previous week's runs. + +Similar to the query for individual runners, we'll query the datastore to get the team's logged run details, calculate statistics for the team, and then return all of that information: + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { SlackAPIClient } from "deno-slack-api/types.ts"; +import RunningDatastore, { RUN_DATASTORE } from "../datastores/run_data.ts"; + +export const CollectTeamStatsFunction = DefineFunction({ + callback_id: "collect_team_stats", + title: "Collect team stats", + description: "Gather and compare run data from the last week", + source_file: "functions/collect_team_stats.ts", + input_parameters: { + properties: {}, + required: [], + }, + output_parameters: { + properties: { + weekly_distance: { + type: Schema.types.number, + description: "Total number of miles ran last week", + }, + percent_change: { + type: Schema.types.number, + description: "Percent change of miles ran compared to the prior week", + }, + }, + required: ["weekly_distance", "percent_change"], + }, +}); + +export default SlackFunction(CollectTeamStatsFunction, async ({ client }) => { + const today = new Date(); + + // Collect runs from the past week (days 0-6) + const lastWeekStartDate = new Date(new Date().setDate(today.getDate() - 6)); + const lastWeekDistance = await distanceInWeek(client, lastWeekStartDate); + if (lastWeekDistance.error) { + return { error: lastWeekDistance.error }; + } + + // Collect runs from the prior week (days 7-13) + const priorWeekStartDate = new Date(new Date().setDate(today.getDate() - 13)); + const priorWeekDistance = await distanceInWeek(client, priorWeekStartDate); + if (priorWeekDistance.error) { + return { error: priorWeekDistance.error }; + } + + // Calculate percent difference between totals of last week and the prior week + const weeklyDiff = lastWeekDistance.total - priorWeekDistance.total; + let percentageDiff = 0; + if (priorWeekDistance.total != 0) { + percentageDiff = 100 * weeklyDiff / priorWeekDistance.total; + } + + return { + outputs: { + weekly_distance: Number(lastWeekDistance.total.toFixed(2)), + percent_change: Number(percentageDiff.toFixed(2)), + }, + }; +}); + +// Sum all logged runs in the seven days following startDate +async function distanceInWeek( + client: SlackAPIClient, + startDate: Date, +): Promise<{ total: number; error?: string }> { + const endDate = new Date(startDate); + endDate.setDate(endDate.getDate() + 6); + + const runs = await client.apps.datastore.query< + typeof RunningDatastore.definition + >({ + datastore: RUN_DATASTORE, + expression: "#date BETWEEN :start_date AND :end_date", + expression_attributes: { "#date": "rundate" }, + expression_values: { + ":start_date": startDate.toLocaleDateString("en-CA", { timeZone: "UTC" }), + ":end_date": endDate.toLocaleDateString("en-CA", { timeZone: "UTC" }), + }, + }); + + if (!runs.ok) { + return { total: 0, error: `Failed to retrieve past runs: ${runs.error}` }; + } + + const total = runs.items.reduce((sum, entry) => (sum + entry.distance), 0); + return { total }; +} +``` + +--- + +Our final function generates our ordered leaderboard, as well as a formatted message with all of our queried data and calculated statistics: + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { RunnerStatsType } from "../types/runner_stats.ts"; + +export const FormatLeaderboardFunction = DefineFunction({ + callback_id: "format_leaderboard", + title: "Format leaderboard message", + description: "Format team and runner stats for a sharable message", + source_file: "functions/format_leaderboard.ts", + input_parameters: { + properties: { + team_distance: { + type: Schema.types.number, + description: "Total number of miles ran last week for the team", + }, + percent_change: { + type: Schema.types.number, + description: + "Percent change of miles ran compared to the prior week for the team", + }, + runner_stats: { + type: Schema.types.array, + items: { type: RunnerStatsType }, + description: "Weekly and all-time total distances for runners", + }, + }, + required: ["team_distance", "percent_change", "runner_stats"], + }, + output_parameters: { + properties: { + teamStatsFormatted: { + type: Schema.types.string, + description: "A formatted message with team stats", + }, + runnerStatsFormatted: { + type: Schema.types.string, + description: "An ordered leaderboard of runner stats", + }, + }, + required: ["teamStatsFormatted", "runnerStatsFormatted"], + }, +}); + +export default SlackFunction(FormatLeaderboardFunction, ({ inputs }) => { + const teamStatsFormatted = + `Your team ran *${inputs.team_distance} miles* this past week: a ${inputs.percent_change}% difference from the prior week.`; + + const runnerStatsFormatted = inputs.runner_stats.sort((a, b) => + b.weekly_distance - a.weekly_distance + ).map((runner) => + ` - <@${runner.runner}> ran ${runner.weekly_distance} miles last week (${runner.total_distance} total)` + ).join("\n"); + + return { + outputs: { teamStatsFormatted, runnerStatsFormatted }, + }; +}); +``` + +Whew! Don't slow down now...we're almost there! + +--- + +## Step 7: Tailor your triggers {#triggers} + +Triggers invoke workflows. There are four types of available triggers, but we'll only be using two: [link triggers](/deno-slack-sdk/guides/creating-link-triggers) and [scheduled triggers](/deno-slack-sdk/guides/creating-scheduled-triggers). For this app, we'll need three triggers, two of which will be link triggers. This means that they require a user to manually trigger them. + +First, we'll create a `triggers` folder and define a link trigger for collecting our team's runs called `log_run_trigger.ts`: + +```javascript +import { Trigger } from "deno-slack-api/types.ts"; +import LogRunWorkflow from "../workflows/log_run_workflow.ts"; + +const LogRunTrigger: Trigger = { + type: "shortcut", + name: "Log a run", + description: "Save the details of a recent run", + workflow: `#/workflows/${LogRunWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + channel: { + value: "{{data.channel_id}}", + }, + user_id: { + value: "{{data.user_id}}", + }, + }, +}; + +export default LogRunTrigger; + +``` + +Run the `trigger create` command in terminal: + +```cmd +slack trigger create --trigger-def triggers/log_run_trigger.ts +``` + +After executing this command, select your app and workspace. Once completed, you'll be given a link called "Shortcut URL." This is your link trigger for this workflow on this workspace. + +Save that URL for when you start testing, since that's how you'll invoke this particular trigger. You can also use the `slack triggers -info` command and select your workspace to grab that URL again later, or click the `/` icon within Slack to open the `Run workflow` menu and select your trigger. + +--- + +Second, we'll need a trigger to display our leaderboard. Create another link trigger in that same `triggers` folder called `display_leaderboard_trigger.ts` and define it as follows: + +```javascript +import { Trigger } from "deno-slack-api/types.ts"; +import DisplayLeaderboardWorkflow from "../workflows/display_leaderboard_workflow.ts"; + +const DisplayLeaderboardTrigger: Trigger< + typeof DisplayLeaderboardWorkflow.definition +> = { + type: "shortcut", + name: "Display the leaderboard", + description: "Show stats for the team and individual runners", + workflow: `#/workflows/${DisplayLeaderboardWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + channel: { + value: "{{data.channel_id}}", + }, + }, +}; + +export default DisplayLeaderboardTrigger; + +``` + +Run the `trigger create` command in the terminal again and save the Shortcut URL for our second link trigger: + +```cmd +slack trigger create --trigger-def triggers/display_leaderboard_trigger.ts +``` + +--- + +Finally, we'll create a scheduled trigger in our `triggers` folder to post a message to a channel with our stats on a weekly basis. We'll call this one `display_weekly_stats.ts`: + +```javascript +import { Trigger } from "deno-slack-api/types.ts"; +import DisplayLeaderboardWorkflow from "../workflows/display_leaderboard_workflow.ts"; + +const DisplayWeeklyStats: Trigger< + typeof DisplayLeaderboardWorkflow.definition +> = { + type: "scheduled", + name: "Display weekly stats", + description: "Display weekly running stats on a schedule", + workflow: `#/workflows/${DisplayLeaderboardWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: "{{data.interactivity}}", + }, + channel: { + value: "{{data.channel_id}}", + }, + }, + schedule: { + start_time: new Date(new Date().getTime() + 60000).toISOString(), + timezone: "EDT", + frequency: { + type: "weekly", + on_days: ["Thursday"], + repeats_every: 1, + }, + }, +}; + +export default DisplayWeeklyStats; + +``` + +Since this is a scheduled trigger, we won't have a Shortcut URL for this one since there's nothing to invoke manually. Use the following command to create the [scheduled trigger](/deno-slack-sdk/guides/creating-event-triggers): + +`slack trigger create --trigger-def triggers/display_weekly_stats.ts` + +Make sure that the app has the `triggers:write` scope added to the manifest! + +--- + +## Step 8: Cross the finish line {#finish} + +You're almost to the finish line! Let's use development mode to run this workflow in Slack directly from the machine you're reading this from now: + +```bash +slack run +``` + +After you've chosen your app and assigned it to your workspace, you can switch over to the app in Slack and give it a spin. Use the link triggers you created previously; when you paste the Shortcut URLs into the box and post them as messages, they'll unfurl and give you buttons for invoking our workflows. + +Here is an example of the message displayed after logging a run: + +![Log a run](/img/social-app/log_run.png) + + +Here is an example of a message displayed after generating the leaderboard: + +![Display leaderboard](/img/social-app/display_leaderboard.png) + +## Next steps + +Congratulations, you made it! 🎉 + +Thinking about signing up for your next race? For your next challenge, perhaps consider creating an app your users can use to [request time off](/deno-slack-sdk/tutorials/request-time-off-app)! \ No newline at end of file diff --git a/docs/tutorials/welcome-bot.md b/docs/tutorials/welcome-bot.md new file mode 100644 index 00000000..295e56d8 --- /dev/null +++ b/docs/tutorials/welcome-bot.md @@ -0,0 +1,694 @@ +# Welcome bot + + + +import SlackMessage from '@site/src/components/SlackMessage'; + + + +What a nice message to read! It sure would be nice if everyone joining a Slack channel received such a message! + +In this tutorial you'll learn how to create a Slack app that sends a friendly welcome message, similar to the one at the top of this page, to a user when they join a channel. A user in the channel will be able to create the custom message from a form. + +Before we begin, ensure you have the following prerequisites completed: +* Install the [Slack CLI](/deno-slack-sdk/guides/getting-started). +* Run `slack auth list` and ensure your workspace is listed. +* If your workspace is not listed, address any issues by following along with the [Getting started](/deno-slack-sdk/guides/getting-started), then come on back. + +## Create a blank project + +Create a blank app with the Slack CLI using the following command: + +``` +slack create welcome-bot-app --template https://github.com/slack-samples/deno-blank-template +``` + +A new app folder will be created. Once you have your new project ready to go, change into your project directory. You'll be bouncing between a few folders, so we recommend using an editor that streamlines switching between files. + +Welcome to your Slack app! There may not be a welcoming message, but do not fret, you can make yourself at home here. Slack apps are built around their flexibility; don't be afraid to run wild! + +For now though, just make three folders within your app folder. Each folder will contain a fundamental building block of a Slack app: +* `functions` +* `workflows` +* `triggers` + +With the setup complete, you can get building! + +## Alternatively, create an app from the template + +If you want to follow along without placing the code yourself, use the pre-built [Welcome Bot app](https://github.com/slack-samples/deno-welcome-bot): + +``` +slack create welcome-bot-app --template https://github.com/slack-samples/deno-welcome-bot +``` + +Once you have your new project ready to go, change into your project directory. + +## Create the app manifest + +The app manifest provides a sneak peak at what you'll be building throughout the rest of this tutorial. The recipe for the Welcome Bot app calls for: +* two workflows, imported from their files: + * `MessageSetupWorkflow` + * `SendWelcomeMessageWorkflow` +* one datastore imported from its file: + * `WelcomeMessageDatastore` +* and six scopes: + * [`chat:write`](https://api.slack.com/scopes/chat:write) + * [`chat:write.public`](https://api.slack.com/scopes/chat:write.public) + * [`datastore:read`](https://api.slack.com/scopes/datastore:read) + * [`datastore:write`](https://api.slack.com/scopes/datastore:write) + * [`channels:read`](https://api.slack.com/scopes/channels:read) + * [`triggers:write`](https://api.slack.com/scopes/triggers:write) + * [`triggers:read`](https://api.slack.com/scopes/triggers:read) + +Put that all together and your `manifest.ts` file will look like: + +```javascript +// /manifest.ts +import { Manifest } from "deno-slack-sdk/mod.ts"; +import { WelcomeMessageDatastore } from "./datastores/messages.ts"; +import { MessageSetupWorkflow } from "./workflows/create_welcome_message.ts"; +import { SendWelcomeMessageWorkflow } from "./workflows/send_welcome_message.ts"; + +export default Manifest({ + name: "Welcome Message Bot", + description: + "Quick way to setup automated welcome messages for channels in your workspace.", + icon: "assets/default_new_app_icon.png", + workflows: [MessageSetupWorkflow, SendWelcomeMessageWorkflow], + outgoingDomains: [], + datastores: [WelcomeMessageDatastore], + botScopes: [ + "chat:write", + "chat:write.public", + "datastore:read", + "datastore:write", + "channels:read", + "triggers:write", + "triggers:read", + ], +}); +``` + +We've provided you all this upfront to streamline the tutorial, but you would likely build up your manifest as you add workflows and datastores to your app. + +## Define a workflow for setting up the welcome message + +In this step we'll be creating a [workflow](/deno-slack-sdk/guides/creating-workflows) named `MessageSetupWorkflow`. This workflow will contain the functions needed for someone in the channel to create a welcome message with a form. + +Create a file named `create_welcome_message.ts` within the `workflows` folder. There you'll add the following workflow definition: + +```javascript +// /workflows/create_welcome_message.ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { WelcomeMessageSetupFunction } from "../functions/create_welcome_message.ts"; + +/** + * The MessageSetupWorkflow opens a form where the user creates a + * welcome message. The trigger for this workflow is found in + * `/triggers/welcome_message_trigger.ts` + */ +export const MessageSetupWorkflow = DefineWorkflow({ + callback_id: "message_setup_workflow", + title: "Create Welcome Message", + description: " Creates a message to welcome new users into the channel.", + input_parameters: { + properties: { + interactivity: { + type: Schema.slack.types.interactivity, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + }, + required: ["interactivity"], + }, +}); +``` + +The `input_parameters` you need are `interactivity` and `channel`. The `interactivity` parameter enables interactive elements, like the form you'll set up next. + +## Add a form for the user to specify the welcome message + +You add functions to workflows by using the `addStep` method. In this case, you'll be adding the form the user will interact with. + +This is done using a [Slack function](/deno-slack-sdk/guides/creating-slack-functions). Slack functions give you the ability to add common Slack functionality without the need to do so from scratch. + +The Slack function to use here is the [`OpenForm`](/deno-slack-sdk/reference/slack-functions/open_form) function. Add it to your `create_welcome_message.ts` workflow like so: + +```javascript +// /workflows/create_welcome_message.ts +/** + * This step uses the OpenForm Slack function. The form has two + * inputs -- a welcome message and a channel id for that message to + * be posted in. + */ +const SetupWorkflowForm = MessageSetupWorkflow.addStep( + Schema.slack.functions.OpenForm, + { + title: "Welcome Message Form", + submit_label: "Submit", + description: ":wave: Create a welcome message for a channel!", + interactivity: MessageSetupWorkflow.inputs.interactivity, + fields: { + required: ["channel", "messageInput"], + elements: [ + { + name: "messageInput", + title: "Your welcome message", + type: Schema.types.string, + long: true, + }, + { + name: "channel", + title: "Select a channel to post this message in", + type: Schema.slack.types.channel_id, + default: MessageSetupWorkflow.inputs.channel, + }, + ], + }, + }, +); +``` + +This creates a form that will show the following fields: +* "Your welcome message", where the user provides the message as a string of text +* "Select a channel to post this message in", where the user provides the channel for the desired channel. + +The user can then submit the form. + +## Add a confirmation ephemeral message when submitting the form + +When the user submits the form, they'll want confirmation that it is submitted. + +You can do this by using the Slack [`SendEphemeralMessage`](/deno-slack-sdk/reference/slack-functions/send_ephemeral_message) function. Add the following step to your `create_welcome_message.ts` workflow: + +```javascript +// /workflows/create_welcome_message.ts +/** + * This step takes the form output and passes it along to a custom + * function which sets the welcome message up. + * See `/functions/setup_function.ts` for more information. + */ +MessageSetupWorkflow.addStep(WelcomeMessageSetupFunction, { + message: SetupWorkflowForm.outputs.fields.messageInput, + channel: SetupWorkflowForm.outputs.fields.channel, + author: MessageSetupWorkflow.inputs.interactivity.interactor.id, +}); + +/** + * This step uses the SendEphemeralMessage Slack function. + * An ephemeral confirmation message will be sent to the user + * creating the welcome message, after the user submits the above + * form. + */ +MessageSetupWorkflow.addStep(Schema.slack.functions.SendEphemeralMessage, { + channel_id: SetupWorkflowForm.outputs.fields.channel, + user_id: MessageSetupWorkflow.inputs.interactivity.interactor.id, + message: + `Your welcome message for this channel was successfully created! :white_check_mark:`, +}); + +export default MessageSetupWorkflow; +``` + +This function takes the provided `message` text and sends it to the specified user and channel, both pulled from the `OpenForm` function step above. + +Wonderful! Now let's build functionality to handle that welcome message once its submitted by a user. + +## Create a datastore to store the welcome message + +The message data needs to be accessible at a later time (when a user joins the channel), so it needs to be stored somewhere, like a [datastore](/deno-slack-sdk/guides/using-datastores). + +Within your `datastores` folder, create a file named `messages.ts`. Within it, define the datastore: + +```javascript +// /datastores/messages.ts +import { DefineDatastore, Schema } from "deno-slack-sdk/mod.ts"; + +export const WelcomeMessageDatastore = DefineDatastore({ + name: "messages", + primary_key: "id", + attributes: { + id: { + type: Schema.types.string, + }, + channel: { + type: Schema.slack.types.channel_id, + }, + message: { + type: Schema.types.string, + }, + author: { + type: Schema.slack.types.user_id, + }, + }, +}); +``` + +Each `attribute` is a type of information you want to store. In this case, it's the information from the form submission. Next, you'll fill the datastore with that information. + +## Create a custom function to send the message to the datastore + +Within your `functions` folder, create a file named `create_welcome_message.ts`. This is where you'll define this [custom function](/deno-slack-sdk/guides/creating-custom-functions). + +The custom function you'll add here will take the form input the user provided and store that information in the created datastore. + +Add the function definition to the `create_welcome_message.ts` file: + +```javascript +// /functions/create_welcome_message.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { SlackAPIClient } from "deno-slack-sdk/types.ts"; + +import { SendWelcomeMessageWorkflow } from "../workflows/send_welcome_message.ts"; +import { WelcomeMessageDatastore } from "../datastores/messages.ts"; + +/** + * This custom function will take the initial form input, store it + * in the datastore and create an event trigger to listen for + * user_joined_channel events in the specified channel. + */ +export const WelcomeMessageSetupFunction = DefineFunction({ + callback_id: "welcome_message_setup_function", + title: "Welcome Message Setup", + description: "Takes a welcome message and stores it in the datastore", + source_file: "functions/create_welcome_message.ts", + input_parameters: { + properties: { + message: { + type: Schema.types.string, + description: "The welcome message", + }, + channel: { + type: Schema.slack.types.channel_id, + description: "Channel to post in", + }, + author: { + type: Schema.slack.types.user_id, + description: + "The user ID of the person who created the welcome message", + }, + }, + required: ["message", "channel"], + }, +}); +``` + +This function provides three `properties` as `input_parameters`. These are the three pieces of information you want to pass to the datastore: the welcome message, the channel to post in, and the user ID of the person who created the message. + +## Add the custom function's functionality + +The actual functionality involves taking those input parameters and putting them into a datastore. Put this right below your function definition within `create_welcome_message.ts`: + +```javascript +// /functions/create_welcome_message.ts +export default SlackFunction( + WelcomeMessageSetupFunction, + async ({ inputs, client }) => { + const { channel, message, author } = inputs; + const uuid = crypto.randomUUID(); + + // Save information about the welcome message to the datastore + const putResponse = await client.apps.datastore.put< + typeof WelcomeMessageDatastore.definition + >({ + datastore: WelcomeMessageDatastore.name, + item: { id: uuid, channel, message, author }, + }); + + if (!putResponse.ok) { + return { error: `Failed to save welcome message: ${putResponse.error}` }; + } + + // Search for any existing triggers for the welcome workflow + const triggers = await findUserJoinedChannelTrigger(client, channel); + if (triggers.error) { + return { error: `Failed to lookup existing triggers: ${triggers.error}` }; + } + + // Create a new user_joined_channel trigger if none exist + if (!triggers.exists) { + const newTrigger = await saveUserJoinedChannelTrigger(client, channel); + if (!newTrigger.ok) { + return { + error: `Failed to create welcome trigger: ${newTrigger.error}`, + }; + } + } + + return { outputs: {} }; + }, +); +``` + +## Add the custom function to the workflow + +Add the custom function you created as a step in the workflow. This connection allows you to use inputs and outputs from previous steps, which is how you'll get the specific pieces of information. + +Pivot back to your `create_welcome_message.ts` workflow file. Add the following step: + +```javascript +// /workflows/create_welcome_message.ts +/** + * This step takes the form output and passes it along to a custom + * function which sets the welcome message up. + * See `/functions/setup_function.ts` for more information. + */ +MessageSetupWorkflow.addStep(WelcomeMessageSetupFunction, { + message: SetupWorkflowForm.outputs.fields.messageInput, + channel: SetupWorkflowForm.outputs.fields.channel, + author: MessageSetupWorkflow.inputs.interactivity.interactor.id, +}); + +export default MessageSetupWorkflow; +``` + +Now you've created a workflow that will: +* let a user fill out a form with information for a welcome message +* store the welcome message information in a datastore + +## Create the link trigger + +You need to create a [trigger](/deno-slack-sdk/guides/using-triggers) that will start the workflow, which provides a user the form to fill out. + +This app will use a specific type of trigger called a [link trigger](/deno-slack-sdk/guides/creating-link-triggers). Link triggers kick off workflows when a user clicks on their link. + +Within your triggers folder, create a file named `create_welcome_message_shortcut.ts`. Place this trigger definition within that file: + +```javascript +// triggers/create_welcome_message_shortcut.ts +import { Trigger } from "deno-slack-api/types.ts"; +import MessageSetupWorkflow from "../workflows/create_welcome_message.ts"; +import { TriggerContextData, TriggerTypes } from "deno-slack-api/mod.ts"; + +/** + * This link trigger prompts the MessageSetupWorkflow workflow. + */ +const welcomeMessageTrigger: Trigger = { + type: TriggerTypes.Shortcut, + name: "Setup a Welcome Message", + description: "Creates an automated welcome message for a given channel.", + workflow: `#/workflows/${MessageSetupWorkflow.definition.callback_id}`, + inputs: { + interactivity: { + value: TriggerContextData.Shortcut.interactivity, + }, + channel: { + value: TriggerContextData.Shortcut.channel_id, + }, + }, +}; + +export default welcomeMessageTrigger; +``` + +This defines a trigger that will kick off the provided workflow, `message_setup_workflow`, along with an added bonus: it'll pass along the channel ID of the channel it was started in. + +## Create the event trigger to start a second workflow + +The workflow to send a message to a user needs to be invoked _after_ the message is created in the workflow. It also needs to be invoked whenever a new user joins the channel. + +This calls for using a different type of trigger: an [event trigger](/deno-slack-sdk/guides/creating-event-triggers). Event triggers are only invoked when a certain event happens. In this case, our event is `user_joined_channel`. + +Think of your `setup` function as priming everything needed for that message to send. The final piece to set up is this trigger. + +Since it runs at a certain point in a workflow, you'll actually place it within a function file. Place it within the `/functions/create_welcome_message.ts` file: + +```javascript +// /functions/create_welcome_message.ts +/** + * findUserJoinedChannelTrigger returns if the user_joined_channel trigger + * exists for the "Send Welcome Message" workflow in a channel. + */ +export async function findUserJoinedChannelTrigger( + client: SlackAPIClient, + channel: string, +): Promise<{ error?: string; exists?: boolean }> { + // Collect all existing triggers created by the app + const allTriggers = await client.workflows.triggers.list({ is_owner: true }); + if (!allTriggers.ok) { + return { error: allTriggers.error }; + } + + // Find user_joined_channel triggers for the "Send Welcome Message" + // workflow in the specified channel + const joinedTriggers = allTriggers.triggers.filter((trigger) => ( + trigger.workflow.callback_id === + SendWelcomeMessageWorkflow.definition.callback_id && + trigger.event_type === "slack#/events/user_joined_channel" && + trigger.channel_ids.includes(channel) + )); + + // Return if any matching triggers were found + const exists = joinedTriggers.length > 0; + return { exists }; +} + +/** + * saveUserJoinedChannelTrigger creates a new user_joined_channel trigger + * for the "Send Welcome Message" workflow in a channel. + */ +export async function saveUserJoinedChannelTrigger( + client: SlackAPIClient, + channel: string, +): Promise<{ ok: boolean; error?: string }> { + const triggerResponse = await client.workflows.triggers.create< + typeof SendWelcomeMessageWorkflow.definition + >({ + type: "event", + name: "User joined channel", + description: "Send a message when a user joins the channel", + workflow: + `#/workflows/${SendWelcomeMessageWorkflow.definition.callback_id}`, + event: { + event_type: "slack#/events/user_joined_channel", + channel_ids: [channel], + }, + inputs: { + channel: { value: channel }, + triggered_user: { value: "{{data.user_id}}" }, + }, + }); + + if (!triggerResponse.ok) { + return { ok: false, error: triggerResponse.error }; + } + return { ok: true }; +} +``` + +This trigger passes the event-related `channel` and `triggered_user` values on to your soon-to-be workflow. With those accessible, you can now build out your next workflow. + +## Create a workflow for sending the welcome message + +This second workflow will retrieve the message from the datastore and send it to the channel when a new user joins that channel. + +Navigate back to your `workflows` folder, and create a new file `send_welcome_message.ts`. + +Within that file place the workflow definition: + +```javascript +// /workflows/send_welcome_message.ts +import { DefineWorkflow, Schema } from "deno-slack-sdk/mod.ts"; +import { SendWelcomeMessageFunction } from "../functions/send_welcome_message.ts"; + +/** + * The SendWelcomeMessageWorkFlow will retrieve the welcome message + * from the datastore and send it to the specified channel, when + * a new user joins the channel. + */ +export const SendWelcomeMessageWorkflow = DefineWorkflow({ + callback_id: "send_welcome_message", + title: "Send Welcome Message", + description: + "Posts an ephemeral welcome message when a new user joins a channel.", + input_parameters: { + properties: { + channel: { + type: Schema.slack.types.channel_id, + }, + triggered_user: { + type: Schema.slack.types.user_id, + }, + }, + required: ["channel", "triggered_user"], + }, +}); +``` + +This workflow will have two inputs: `channel` and `triggered_user`, both acquired from the trigger invocation. + +## Create a custom function that sends the welcome message + +Navigate to the `functions` folder, and create a new file called `send_welcome_message.ts`. + +Within that file add the definition for a function that uses the inputs `channel` and `triggered_user`: + +```javascript +// /functions/send_welcome_message.ts +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +import { WelcomeMessageDatastore } from "../datastores/messages.ts"; + +/** + * This custom function will pull the stored message from the datastore + * and send it to the joining user as an ephemeral message in the + * specified channel. + */ +export const SendWelcomeMessageFunction = DefineFunction({ + callback_id: "send_welcome_message_function", + title: "Sending the Welcome Message", + description: "Pull the welcome messages and sends it to the new user", + source_file: "functions/send_welcome_message.ts", + input_parameters: { + properties: { + channel: { + type: Schema.slack.types.channel_id, + description: "Channel where the event was triggered", + }, + triggered_user: { + type: Schema.slack.types.user_id, + description: "User that triggered the event", + }, + }, + required: ["channel", "triggered_user"], + }, +}); +``` + +## Add the custom function's functionality + +With the function defined, add the actual functionality right after: + + +```javascript +// /functions/send_welcome_message.ts +export default SlackFunction(SendWelcomeMessageFunction, async ( + { inputs, client }, +) => { + // Querying datastore for stored messages + const messages = await client.apps.datastore.query< + typeof WelcomeMessageDatastore.definition + >({ + datastore: WelcomeMessageDatastore.name, + expression: "#channel = :mychannel", + expression_attributes: { "#channel": "channel" }, + expression_values: { ":mychannel": inputs.channel }, + }); + + if (!messages.ok) { + return { error: `Failed to gather welcome messages: ${messages.error}` }; + } + + // Send the stored messages ephemerally + for (const item of messages["items"]) { + const message = await client.chat.postEphemeral({ + channel: item["channel"], + text: item["message"], + user: inputs.triggered_user, + }); + + if (!message.ok) { + return { error: `Failed to send welcome message: ${message.error}` }; + } + } + + return { + outputs: {}, + }; +}); +``` + +This creates a function that: +* queries the datastore for stored messages +* posts an ephemeral message using the `message` item from the datastore with a matching `channel` channel ID value to the user with the `triggered_user` user ID. + +## Add the custom function to the workflow + +With the custom function built, add it to your `send_welcome_message.ts` workflow as a step: + +```javascript +// /workflows/send_welcome_message.ts +SendWelcomeMessageWorkflow.addStep(SendWelcomeMessageFunction, { + channel: SendWelcomeMessageWorkflow.inputs.channel, + triggered_user: SendWelcomeMessageWorkflow.inputs.triggered_user, +}); +``` + +And with that, you have created the two workflows that contain all the functionality you need to send a custom ephemeral message to a user joining a new channel. + +## Run your Slack app + +For now, you'll want to [locally install the app](/deno-slack-sdk/guides/developing-locally) to the workspace. From the command line, within your app's root folder, run the following command: + +``` +slack run +``` + +Proceed through the prompts until you have a local server running in that terminal instance. + +It's installed! You can't use it quite yet though. + +## Invoke the link trigger + +Within a terminal located within that folder, you'll need to create that initial link trigger. You can open a new terminal tab or cancel your running server and restart later if you'd like. + +You can do that with the `slack trigger create` command. Make it so. + +``` +slack trigger create --trigger-def triggers/create_welcome_message_shortcut.ts +``` +Since you haven't installed this trigger to a workspace yet, you'll be prompted to install the trigger to a new workspace. Then select an authorized workspace in which to install the app. + +When you select your workspace, you will be prompted to choose an app environment for the trigger. Choose the _Local_ option so you can interact with your app while developing locally. The CLI will then finish installing your trigger. + +Once your app's trigger is finished being installed, you will see the following output: + +```zsh +📚 App Manifest + Created app manifest for "welcomebot (local)" in "myworkspace" workspace + +⚠️ Outgoing domains + No allowed outgoing domains are configured + If your function makes network requests, you will need to allow the outgoing domains + Learn more about upcoming changes to outgoing domains: https://api.slack.com/changelog + +🏠 Workspace Install + Installed "welcomebot (local)" app to "myworkspace" workspace + Finished in 1.5s + +⚡ Trigger created + Trigger ID: Ft0123ABC456 + Trigger Type: shortcut + Trigger Name: Setup a Welcome Message + Shortcut URL: +https://slack.com/shortcuts/Ft0123ABC456/XYZ123 +... +``` + +Copy the URL, paste, and post it in a channel to kick off the first workflow and create a message. + + +## Deploy your Slack app + +When you're ready to make the app accessible to others, you'll want to [deploy it](/deno-slack-sdk/guides/deploying-to-slack) instead of running it: + +``` +slack deploy +``` + +And then create the trigger again, but choosing the _Deployed_ option this time: + +``` +slack trigger create --trigger-def triggers/create_welcome_message_shortcut.ts +``` + +Other than that, the steps are the same. + +## Pause and reflect + +Congratulations! You've successfully built your friendly neighborhood welcome bot, providing a cozy presence to all who enter your desired channel. + +### Next steps + +For your next challenge, perhaps consider creating [an app that creates an issue in GitHub](/deno-slack-sdk(/tutorials/github-issues-app)! \ No newline at end of file diff --git a/docs/tutorials/workflow-builder-custom-step/1.png b/docs/tutorials/workflow-builder-custom-step/1.png new file mode 100644 index 00000000..6d89bfa5 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/1.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/10.png b/docs/tutorials/workflow-builder-custom-step/10.png new file mode 100644 index 00000000..8e300dc1 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/10.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/11.png b/docs/tutorials/workflow-builder-custom-step/11.png new file mode 100644 index 00000000..30c316b4 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/11.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/12.png b/docs/tutorials/workflow-builder-custom-step/12.png new file mode 100644 index 00000000..edb92f7c Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/12.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/13.png b/docs/tutorials/workflow-builder-custom-step/13.png new file mode 100644 index 00000000..e2da28b2 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/13.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/14.png b/docs/tutorials/workflow-builder-custom-step/14.png new file mode 100644 index 00000000..871a089a Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/14.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/15.png b/docs/tutorials/workflow-builder-custom-step/15.png new file mode 100644 index 00000000..056ccaf5 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/15.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/16.png b/docs/tutorials/workflow-builder-custom-step/16.png new file mode 100644 index 00000000..1a548e17 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/16.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/17.png b/docs/tutorials/workflow-builder-custom-step/17.png new file mode 100644 index 00000000..d2d4d5a9 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/17.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/18.png b/docs/tutorials/workflow-builder-custom-step/18.png new file mode 100644 index 00000000..0d62bf59 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/18.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/19.png b/docs/tutorials/workflow-builder-custom-step/19.png new file mode 100644 index 00000000..936b4ec0 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/19.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/2.png b/docs/tutorials/workflow-builder-custom-step/2.png new file mode 100644 index 00000000..e7b228c1 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/2.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/20.png b/docs/tutorials/workflow-builder-custom-step/20.png new file mode 100644 index 00000000..34beeb85 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/20.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/21.png b/docs/tutorials/workflow-builder-custom-step/21.png new file mode 100644 index 00000000..33e27be7 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/21.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/22.png b/docs/tutorials/workflow-builder-custom-step/22.png new file mode 100644 index 00000000..6f70753d Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/22.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/23.png b/docs/tutorials/workflow-builder-custom-step/23.png new file mode 100644 index 00000000..e54adb70 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/23.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/24.png b/docs/tutorials/workflow-builder-custom-step/24.png new file mode 100644 index 00000000..16a3eb4f Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/24.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/25.png b/docs/tutorials/workflow-builder-custom-step/25.png new file mode 100644 index 00000000..350282d4 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/25.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/26.png b/docs/tutorials/workflow-builder-custom-step/26.png new file mode 100644 index 00000000..be639a24 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/26.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/27.png b/docs/tutorials/workflow-builder-custom-step/27.png new file mode 100644 index 00000000..8c1731e7 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/27.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/28.png b/docs/tutorials/workflow-builder-custom-step/28.png new file mode 100644 index 00000000..bcfb3881 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/28.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/29.png b/docs/tutorials/workflow-builder-custom-step/29.png new file mode 100644 index 00000000..f8068be1 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/29.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/3.png b/docs/tutorials/workflow-builder-custom-step/3.png new file mode 100644 index 00000000..1238381d Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/3.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/30.png b/docs/tutorials/workflow-builder-custom-step/30.png new file mode 100644 index 00000000..526d8c26 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/30.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/31.png b/docs/tutorials/workflow-builder-custom-step/31.png new file mode 100644 index 00000000..b4b86b06 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/31.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/4.png b/docs/tutorials/workflow-builder-custom-step/4.png new file mode 100644 index 00000000..9a6b1eaf Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/4.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/5.png b/docs/tutorials/workflow-builder-custom-step/5.png new file mode 100644 index 00000000..10b718f1 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/5.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/6.png b/docs/tutorials/workflow-builder-custom-step/6.png new file mode 100644 index 00000000..508ee482 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/6.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/7.png b/docs/tutorials/workflow-builder-custom-step/7.png new file mode 100644 index 00000000..9e521c50 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/7.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/8.png b/docs/tutorials/workflow-builder-custom-step/8.png new file mode 100644 index 00000000..a43d6bf9 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/8.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/9.png b/docs/tutorials/workflow-builder-custom-step/9.png new file mode 100644 index 00000000..985cc503 Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/9.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/B615131356AA4389BDDCC4131E995B27.png b/docs/tutorials/workflow-builder-custom-step/B615131356AA4389BDDCC4131E995B27.png new file mode 100644 index 00000000..2c82fcef Binary files /dev/null and b/docs/tutorials/workflow-builder-custom-step/B615131356AA4389BDDCC4131E995B27.png differ diff --git a/docs/tutorials/workflow-builder-custom-step/index.md b/docs/tutorials/workflow-builder-custom-step/index.md new file mode 100644 index 00000000..44b92dab --- /dev/null +++ b/docs/tutorials/workflow-builder-custom-step/index.md @@ -0,0 +1,415 @@ +--- +sidebar_label: Workflow Builder custom step +--- + +# Create a custom step for Workflow Builder with the Deno Slack SDK + + + +Custom functions in apps can be added as workflow steps in Workflow Builder. + +In this tutorial, you will define and implement a [custom function](/deno-slack-sdk/guides/creating-custom-functions), then wire it up as a workflow step in our no-code automation platform [Workflow Builder](https://slack.com/help/articles/360035692513-Guide-to-Workflow-Builder). + +When finished, you'll be ready to build custom, scalable functions for anyone using Workflow Builder in your workspace. + +You'll build three things in this tutorial: + +- A custom function +- A workflow app +- A workflow in Workflow Builder + +The custom function will take a user-supplied string—the name of a composite color—and use a switch statement to return a new string—the meaning of the color—based on its value. The function will collect the input string from the user once the workflow is started, then return the result to the user in the form of an ephemeral message. An _ephemeral_ message in Slack is one that is only visible to the user. + +:::info + +To protect your organization, external users (those outside your organization connected through Slack Connect) cannot use a workflow that contains connector functions built by your organization. This may manifest in a `home_team_only` warning. Refer to [this help center article](https://slack.com/help/articles/14844871922195-Slack-administration--Manage-workflow-usage-in-Slack-Connect-conversations#enterprise-grid-1) for more details. + +::: + + +:::tip[Skip to the code] + +If you'd rather skip the tutorial and just head straight to the code, create a new app and use our [function sample](https://github.com/slack-samples/deno-function-template) as a template. The sample custom function provided in the template will be a good place to start exploring! + +::: + +### The road ahead + +1. We'll sketch out what we want the function to do and how users will integrate it in their workflows. +2. You'll write the custom function and deploy the app so you can use the step in Workflow Builder. +3. You'll find the workflow step in Workflow Builder and use it as a step in a workflow that you can run from inside Slack. + +Ready? Let's get started! + +## Install & authorize the Slack CLI + +>Have you installed the CLI? Are you authorized in a workspace? If you answered 'yes' to both questions, skip this step! + +You'll need to have the Slack CLI **installed** and **authorized** to begin this tutorial. If you need help, follow the [Quickstart](/deno-slack-sdk/guides/getting-started) guide and you'll be ready to build. + +You'll also need a development workspace where you have permission to install apps. Please note that the features in this tutorial require that the workspace be part of a [paid Slack plan](https://slack.com/pricing). + +## Create a new app + +When you deploy custom functions for Workflow Builder, users will be able to search for your deployed app and then include any steps you've provided for them. + +Let's get our new app project started so we can define and then implement our custom function. + +With your CLI authorized, go to your terminal and create a new app with the blank template: + +1. Run the command `slack create meaning-of-color`. This will tell the CLI you want to create a new workflow app named `meaning-of-color`. + +2. When prompted to select a template to build from, select the **Blank template**. + +3. When the CLI is finished setting up your project, follow the instructions in your terminal to `cd` into your project's directory. + +:::tip[VS Code] + +If you're using VS Code, once you `cd` into your project's +directory, open it up with VS code by executing `code .`. + +::: + +## Create a new custom function + +Create a new folder called `functions`. + +In the `functions` folder, create a new file called `interpret_color.ts` where you'll will define and implement a custom Slack function. + +Start the file off by importing the necessary modules: + +```javascript +import { DefineFunction, Schema, SlackFunction } from "deno-slack-sdk/mod.ts"; +``` + +Next, define a new custom function named `InterpretColorFunction`. It should have a single string as both an input and output parameter: + +```javascript +export const InterpretColorFunction = DefineFunction({ + callback_id: "interpret_color_function", + source_file: "functions/interpret_color.ts", + title: "Interpret Color", + input_parameters: { + properties: { + input_string: { + type: Schema.types.string, + }, + }, + required: ["input_string"], + }, + output_parameters: { + properties: { + result: { + type: Schema.types.string, + }, + }, + required: ["result"], + }, +}); +``` + +The `title` property will be used to identify this function for users in Workflow Builder. + +With the function defined, the next step is to implement it. This function will take the `input_string` and run it through a switch statement, then return a new string based on the matching case: + +```javascript +export default SlackFunction( + InterpretColorFunction, + ({ inputs }) => { + const input_string = inputs.input_string; + + switch (input_string) { + case "orange": + return { + outputs: { + input_string, + result: "Orange is the color of ambition", + }, + }; + case "green": + return { + outputs: { + input_string, + result: "Green is the color of collaboration", + }, + }; + case "purple": + return { + outputs: { + input_string, + result: "Purple is the color of harmony", + }, + }; + default: + return { + outputs: { + input_string, + result: "That's not a color I recognize", + }, + }; + } + }, +); +``` + +The function is aware of three composite colors: orange, green, and purple. If a user supplies one of those composite colors as input, the function will return a fun meaning of that color. If the user supplies a color that is not known (i.e., not orange, green, or purple), then they'll get the default case message. + +In every return statement, the function includes both the required output (the `result` string) as well as the original input (the `input_string` string), which is not a required output parameter. This is a design decision that you can make depending on your use case; include the inputs in the return value if you want them available as inputs in follow-on workflow steps. + +## Configure the app's manifest + +Your app's manifest is where you configure which functions your app should care about, among other things. To do that, import your custom function and add it to your manifest's `functions` property. + +Start by opening `manifest.ts` and importing `InterpretColorFunction`: + +```javascript +import { Manifest } from "deno-slack-sdk/mod.ts"; + +// Add this: +import { InterpretColorFunction } from "./functions/interpret_color.ts"; +``` + +Next, configure your app's manifest. For this tutorial, update the `name`, `description`, and `functions` property: + +```javascript +export default Manifest({ + name: "Meaning of Color App", + description: "The meaning of colors", + icon: "assets/default_new_app_icon.png", + functions: [InterpretColorFunction], + workflows: [], + outgoingDomains: [], + botScopes: ["commands", "chat:write", "chat:write.public"], +}); +``` + +This app is now finished! + +In the next section, you'll start a local development server and test drive the app before deploying it for your users. + +## Start the local development server + +Start a local development server with `slack run`. + +Since this is the first time starting the local development server for this project, you'll be prompted to choose a local environment. The local environment is the Slack workspace you will be using to interact with the app you're currently developing. + +Select the option to install to a new workspace, then select the workspace you want to use for development. + +Once your local development server has started, you'll see it says `Connected, awaiting events` in your terminal window. + +At this point, you're ready to build a workflow in Workflow Builder to test your custom function. + +**Keep your local development server running while building the workflow!** + +## Use the workflow step in Workflow Builder + +In the workspace you just installed your app, open up Workflow Builder and create a new workflow: + +1. From your desktop, click your workspace name in the top left. +2. Select **Tools** from the menu, then click **Workflow Builder**. This will open a new window titled "Workflow Builder." +3. In the "Workflow Builder" window that just opened, click the **Create Workflow** button in the top right. + +You can also find this screen by navigating to the left-side nav menu in Slack and clicking the elipses **...More** tile, then selecting **Automations**. + +There are many ways to start a workflow in Workflow Builder. For this tutorial, we will use a link trigger. Under **Start the workflow...**, select **Choose an event**, then select **From a link in Slack**: + +![Creating a new workflow in Workflow Builder](1.png) + +A modal will appear that show you an example **workflow link** (more on that soon) along with a scrollable container that reveals a **Custom inputs** section. Click the **Add Input** button. + +:::info + +Depending on which Slack plan you are on, the latest version of Workflow Builder may still be rolling out and your UI may appear differently. If you do not see an **Add Input** button, we recommend skipping to the code and using our [function sample](https://github.com/slack-samples/deno-function-template) as a template. + +::: + +![Adding custom inputs to a new workflow in Workflow Builder](2.png) + +After clicking the **Add Input** button, a form will appear. Use this form to describe the required input parameter for your custom function—i.e., the `color` to interpret. When you're done, click the **Done** button: + +![Configuring custom inputs to a new workflow in Workflow Builder](3.png) + +After clicking **Done**, the form will disappear and you'll see the custom input you just defined listed above the **Add Input** button. Since your custom function only has one required input, you can now click the **Continue** button to start building the workflow: + +![Finishing creating a new workflow in Workflow Builder](4.png) + +You'll now be editing your workflow. The only thing configured right now is the trigger that will start your workflow. Recall that this workflow ought to do two things: run the custom function to get the result of the color interpretation, then send an ephemeral message to the user who ran the workflow. + +To add your custom function as a step in this workflow, search for the name of the function ("Interpret Color") in the right-hand **Steps** sidebar: + +![Searching for a custom function in Workflow Builder](5.png) + +In the search results that appear in the right-hand sidebar, you should see the name of your app ("Meaning of Color App") along with "(local)", which indicates that this function is provided via your local development server at this time, followed by your "Interpret Color" function. Click on the **Interpret Color** function to begin adding it to your workflow: + +![Configuring inputs for custom function in Workflow Builder](6.png) + +As soon as you click on your function to add it to your workflow, you'll be presented with a modal prompting you for an **Input String**. This is your custom function's required input parameter, so pass in the custom input you configured earlier when first creating this workflow. To do that, click on the curly braces icon to the right of the text box labelled "Input String", then select **Color** from the dropdown. + +When you're finished, click **Save** to go back to editing your workflow: + +![Selecting custom input for function input in Workflow Builder](7.png) + +Your function—along with the input configuration you set up—is now the first step in the workflow following the trigger. The last thing this workflow needs to do is send an ephemeral message to the user that started it. + +In the **Steps** sidebar on the right, click **Messages**: + +![Locating Messages functions in Workflow Builder](8.png) + +This brings up built-in functions for common Slack messaging that you can use as steps in your workflow. + +Select **Send an "only visible to you" message**, which sends an ephemeral message to a user that only they can see: + +![Selecting the ephemeral message function in Workflow Builder](9.png) + +After you select **Send an "only visible to you" message**, a modal will appear where you can configure this step. + +For **Select a channel**, choose the option **Channel where the workflow was used**. + +For **Select a member of the channel**, choose the option **Person who used this workflow**. + +For the **Add a message** field, select the **Insert a variable** link below the textbox: + +![Configuring ephemeral message step in Workflow Builder](10.png) + +In the **Insert a variable** menu that appears, select the **Result** option located in your custom function's output variable group. + +![Selecting custom function output for ephemeral message step in Workflow Builder](11.png) + +When you're done, **Save** this step to return to the workflow editor: + +![Saving ephemeral message step in Workflow Builder](12.png) + +The workflow is almost ready. The last thing to do before publishing it is configure its name. Click on **Untitled Workflow** at the top to open the **Workflow Details** modal: + +![Editing details in Workflow Builder](13.png) + +In the **Workflow Details** modal, edit the **Name** and **Description** of your workflow. These are what the user will see when the workflow is shared with them. When you're done, click the **Save** button: + +![Editing name and description in in Workflow Builder](14.png) + +Begin publishing your workflow by clicking the **Finish Up** button at the top: + +![Publishing a workflow in Workflow Builder](15.png) + +In the **Finish Up** modal, confirm your workflow's name and description, then scroll down to see **Workflow managers**. If you want to add a workflow collaborator, you can do this here. Scroll down to click **Show more permissions**. This is where you may edit your workflow's permissions: + +![Confirming details in Workflow Builder](16.png) + +Let's leave everything as is, and finally, click **Publish**: + +![Final publishing checks in Workflow Builder](17.png) + +**Your workflow has been published!** + +## Run your workflow in Slack + +Your workflow is now ready for you to try out. Since you configured this workflow to start from a link in Slack, you'll need to copy the workflow link and then share it in any channel in your workspace. + +To copy the workflow link, click the **Copy Link** button: + +![Copying workflow link in Workflow Builder](18.png) + +
+Where else can I copy the workflow link? + +When you open up your workflow, hover over the first step, +**Starts from a link in Slack**, and click the **Copy Link** icon +that appears: + +![Copying workflow link in Workflow Builder](21.png) + +
+ +With the workflow link copied, leave Workflow Builder and go to Slack. + +In any channel, paste the workflow link and send it in a new message. Once the message is sent, Slack will recognize it as a workflow link and it will unfurl with a button: + +![An unfurled workflow link](19.png) + +Try out your workflow by clicking the **Start Workflow** button. + +Enter `purple` for the Color input, then run the workflow by clicking the **Start Workflow** button in the modal. + +You should see an ephemeral message with the appropriate return value based on how you implemented the custom function: + +![Am ephemeral message from a workflow step](20.png) + +Congratulations! You just successfully built a function, wired it up in Workflow Builder, and executed it inside Slack. + +Run your workflow a few more times, trying the following inputs to see what happens: +- `purple` +- `green` +- `blue` +- `orange` + +You can also update your function implementation, then re-run the workflow to see your changes propagated to Slack in realtime. + +## Deploy your function + +Your local development server is great for rapidly testing out your custom functions while building them. + +When you're ready to deploy your functions in workflows that you intend to be used by people in their day-to-day jobs, you'll need to deploy your function to Slack and then swap out the local function step with the newly-deployed function step. + +To do this, first go to your terminal where your local development server is running and enter `Ctrl`+`C`. This stops your server. + +Next, deploy your app by running `slack deploy`. Since this is your first time _deploying_ your app, you'll go through the same steps you did when you first _installed_ your app: select **Install to a new workspace**, then select the workspace you intend to use this function as a part of a workflow in. + +Once your app has finished deploying, go back into Workflow Builder and select your workflow: + +![Selecting your workflow in Workflow Builder](22.png) + +To swap out the local version of your app for the deployed version of your function, you'll first create a new step that calls the deployed function, then modify the ephemeral message step to use the deployed function instead of the local one. + +First, find the deployed version of your function by typing its name in the sidebar's search box. Notice how you have two options now—one for the local version, and one for the newly-deployed version. Click on the deployed version to add it into your workflow: + +![Selecting a function from your deployed app in Workflow Builder](23.png) + +When the modal to configure the **Input String** appears, do the same as before: click on the curly brace icon to the right of the input field, select the **Color** option from the **From a link in Slack** group, then click the **Save** button to return to the editor: + +![Configuring function inputs in Workflow Builder](24.png) + +You'll see two separate **Interpret Color** steps now. Delete the top one, which is the old _local_ version, by hovering over the step and clicking the **Delete** icon: + +![Deleting the old function in Workflow Builder](25.png) + +Confirm deletion in the modal that appears: + +![Confirming deletion in Workflow Builder](26.png) + +The **Needs attention** warning is expected, since the step that provided input into the ephemeral message step was just removed. + +Move the deployed **Interpret Color** function so that it is above the ephemeral message step, putting it where the deleted local function was earlier, by selecting the **Move Up** button in step controls: + +![Moving the deployed function step up in Workflow Builder](27.png) + +Click on the **Edit** icon in the ephemeral message step to fix the missing data issue: + +![Editing the ephemeral message step in Workflow Builder](28.png) + +Delete the `Missing Data` element from the **Add a message** field: + +![Removing the missing data in Workflow Builder](29.png) + +Then, click on the **Insert a variable** link underneath the field and select the **Result** option from the **Interpret Color** function group. + +When finished, click the **Save** button: + +![Saving changes in Workflow Builder](30.png) + +**Congratulations! Your workflow is done!** + +All that's left to do now is **Publish changes**: + +![Publishing changes in Workflow Builder](31.png) + +## Onward + +In this tutorial, you built a custom function, wired it up in Workflow Builder, and fully deployed your app so that it's ready to be used in other workflows. + +Here are some areas to explore now that you've come this far: + +* Want to integrate with a third-party? Augment your custom function by leveraging [external authentication](/deno-slack-sdk/guides/integrating-with-services-requiring-external-authentication)! + +* Instead of using Workflow Builder, add your custom function as steps in [coded workflows](/deno-slack-sdk/guides/creating-workflows)! + +* Check out our other tutorials for more ideas about what you can do with the workflow automations! diff --git a/docs/types.md b/docs/types.md deleted file mode 100644 index 504664c2..00000000 --- a/docs/types.md +++ /dev/null @@ -1,75 +0,0 @@ -## Types - -Custom Types provide a way to introduce reusable, sharable types to Apps. - -### Defining a type - -Types can be defined with the top level `DefineType` export. Below is an example -of setting up a custom Type used for incident management. - -```ts -const IncidentType = DefineType({ - name: "incident", - title: "Incident Ticket", - description: "Use this to enter an Incident Ticket", - type: Schema.types.object, - properties: { - id: { - type: Schema.types.string, - minLength: 3, - }, - title: { - type: Schema.types.string, - }, - summary: { - type: Schema.types.string, - }, - severity: { - type: Schema.types.string, - }, - date_created: { - type: Schema.types.number, - }, - }, - required: [], -}); -``` - -### Registering a type with the App - -To register the newly defined type, add it to the array assigned to the `types` -parameter while defining the [`Manifest`][manifest]. - -Note: All Custom Types **must** be registered to the [Manifest][manifest] in -order for them to be used, but any types referenced by existing -[`functions`][functions], [`workflows`][workflows], [`datastores`][datastores], -or other types will be registered automatically. - -```ts -Manifest({ - ... - types: [IncidentType], -}); -``` - -### Referencing types - -To use a type as a [function][functions] parameter, set the parameter's `type` -property to the Type it should reference. - -```js -input_parameters: { - incident: : { - title: 'A Special Incident', - type: IncidentType - } - ... -} -``` - -_In the provided example the title from the Custom Type is being overridden_ - -[functions]: ./functions.md -[manifest]: ./manifest.md -[datastores]: ./datastores.md -[workflows]: ./workflows.md diff --git a/docs/workflows.md b/docs/workflows.md deleted file mode 100644 index a28fbcd1..00000000 --- a/docs/workflows.md +++ /dev/null @@ -1,102 +0,0 @@ -## Workflows - -Workflows can be defined and included in your [manifest][manifest]. A workflow -itself has several pieces of metadata, such as a unique `callback_id`, a `title` -and a `description`. It can also include `input_parameters` just like a -[function][function]. Key to a workflow is a series of steps, each of which are -a function that can be passed dynamic data to their inputs through referencing -workflow inputs, or outputs from previous steps. Let's take a look at an -example. - -```ts -import { DefineWorkflow, Manifest, Schema } from "slack-cloud-sdk/mod.ts"; - -const workflow = DefineWorkflow({ - callback_id: "my_workflow", - title: "My Workflow", - description: "A sample workflow", - input_parameters: { - properties: { - a_string: { - type: Schema.types.string, - }, - a_channel: { - type: Schema.slack.types.channel_id, - } - }, - required: ["a_string", "a_channel"], - }, -}); - -// register your workflow in your manifest -export default Manifest({ - ..., - workflows: [ - workflow, - ] -}); -``` - -A workflow by itself isn't of much use, and isn't valid, until you add some -steps. Let's use the `DinoFunction` we've defined over in the -[functions][function] example as one of our steps. The `DinoFunction` has a -single `input_parameter` of `name` that we'll need to pass it. We'll use our -`a_string` workflow `input_parameter` as the value for this, but you could just -as easily pass a hard-coded value to any step input parameter as well. - -```ts -import { DefineWorkflow } from "slack-cloud-sdk/mod.ts"; -import { DinoFunction } from '../functions/dino.ts'; - -const workflow = DefineWorkflow({...}); - -const step1 = workflow.addStep(DinoFunction, { - name: workflow.inputs.a_string, -}); -``` - -Great, we've got a single step workflow that takes a string, and turns it into a -dinosaur name via our `DinoFunction`. It would be nice to see what that looks -like, so lets add another step that sends that value as a message somewhere. For -this, we can use one of Slack's functions. Notice how we can also use our -reference to `step1` to access an output called `dinoname` that the -`DinoFunction` produces. - -```ts -const step1 = workflow.addStep(...); - -workflow.addStep("slack#/functions/send_message", { - channel: workflow.inputs.a_channel, - message: `A dinosaur name: ${step1.outputs.dinoname}`, -}); -``` - -You'll notice the first parameter to `addStep()` here is a string, instead of -something like our `DinoFunction`. This is because we're referencing a step -produced outside of our app, in this case by `slack`. We're using the string -reference of `"slack#/functions/send_message"` to identify the function we're -adding as a step. In fact, you can do the same thing with your own functions by -creating what's called a local reference string to your own app's function. This -uses your `callback_id`, and would look like `"#/functions/my_workflow"`. If we -added our function as a step that way, it would look like this: - -```ts -const step1 = workflow.addStep("#/functions/my_workflow", { - name: workflow.inputs.a_string, -}); -``` - -The big difference here is you won't get some of the automatic typing of -`inputs` and `outputs` for that step, but you can still reference them as long -as you follow the definition of that function. - -### Auto-Registration of workflow dependencies - -When a workflow is registered on your `Manifest()` any `functions` it uses as -steps, or custom `types` used as `input_parameters` to the workflow or functions -it references are automatically registered in your manifest. This can save you -from having to register each function and type that a workflow might use, and -just register the workflow. - -[manifest]: ./manifest.md -[function]: ./functions.md