4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

SlackBotへのメンションをトリガーにメール送信する方法

Last updated at Posted at 2021-07-04

概要

Slackのスレッドへ勤怠連絡するのにメール報告も行わなければならないのが面倒だったので、Slackへの投稿をトリガーにメール送信する機能を実装しました。

また、メール送信成功時にはSlackBotから特定のチャンネルに投稿させるようにしました。

Slack Events API

Slackへの投稿やBotへのメンション等の様々なイベントをトリガーに設定して、サーバーにHTTPリクエストを送って様々な機能を実装可能です。

例えば、ほぼリアルタイムにSlackへ投稿したり、Slackの機能に限らずにサーバー上で実現できることは何でも実現させることができます。

HTTPリクエストを受け取るサーバー

今回は個人的に気になっていたGoogle Apps Script(GAS)を使って実装します。

実装手順

GASでサーバーを用意する

  1. Googleドライブにログイン後、新規→その他→Google Apps Scriptの順にクリックする。
    1.png
  2. Slackのリクエスト先の認証を成功させるために表示されたエディタに下記のコードを記述して保存する(ctl + s)。
function doPost(e) {
  const params = JSON.parse(e.postData.getDataAsString());
  return ContentService.createTextOutput(params.challenge);
}

        
3. デプロイ→新しいデプロイの順にクリックする。
2.png
4. 歯車マーク→ウェブアプリの順にクリックする。
3.png
5. 説明を簡単に記述し、アクセスできるユーザーを「全員」に設定する(アクセス制限をかけるとSlackからアクセス出来ないため、後ほどSlackが生成するトークンを利用した実行制限を実装する)。
4.png
6. デプロイ後に表示されるウェブアプリURLはすぐに使うのでコピーしておく。

slack apiを作成する

  1. slack apiのトップページから「Create an app」をクリックする
    5.png
  2. From scratchを選択
    6.png
  3. App Nameを適当に設定し、アプリケーションを利用するSlackのワークスペースを選択して、「Create App」をクリックする。
    7.png

イベントを有効にする

  1. Event Subscriptionsをクリックする。
    8.png
  2. Enable Eventsを「On」にして、先ほどGASで作成したウェブアプリケーションのURLをRequest URLに貼り付ける。
    9.png
  3. 「Verified」と表示されれば、ひとまず成功です。「Your URL didn't respond with the value of the challenge parameter.」と表示された場合は今までの工程で間違っている可能性があります。
  4. Subscribe to bot events→Add Bot User Eventの順にクリックする。
    10.png
  5. 今回はイベントトリガーはBotのメンションを設定したいので、「app_mention」を選択して「Save Changes」をクリックする。

権限を設定する

  1. メニューからOAuth & Permissionsを開く。
  2. ScopesのBot Token ScopesのAdd an OAuth Scopeを都度クリックして以下の権限を付与する。
    11.png
    • app_mentions:read
    • channels:history
    • channels:join
    • chat:write
    • users:read

Workspaceにインストールする

  1. メニューからBasic Informationを開く。
  2. Install your appの「Install to Workspace」をクリックする。
    12.png
  3. 表示されたアクセス権限が間違いなければ、「許可する」をクリックする。
    13.png

アクセス制御に利用するトークンをGASのプロパティに設定する

function setValues() {
  PropertiesService.getScriptProperties().setProperties({SLACK_ACCESS_TOKEN: 'accessToken',SLACK_VERIFICATION_TOKEN: 'verifyToken'});
};

  1. 上記のコードをGASのエディタに追加する。
  2. slack apiでOAuth & Permissionsを開く。
  3. OAuth Tokens for Your WorkspaceのBot User OAuth TokenをaccessTokenの箇所にコピペする。
    15.png
  4. slack apiでBasic Informationを開く。
  5. App CredentialsのVerification Tokenを上記コードのverifyTokenの箇所にコピペする。
    14.png
  6. GASのエディタ上で実行する関数を「setValues」に設定して、実行をクリックする。
    16.png
  7. エラーが発生せず、実行完了した場合は全てのコードを削除して構いません。

アクセス制御を含むメール送信アプリケーションを作成する

  1. GASのエディタに下記のコードをコピペして、自分の利用方法に合わせて設定値や文章を変更する。
function doPost(e) {
  try{
    if(typeof e.postData === 'undefined') {
      return createTextOutput('invalid request');
    }

    const accessToken = PropertiesService.getScriptProperties().getProperty('SLACK_ACCESS_TOKEN');
    const verificationToken = PropertiesService.getScriptProperties().getProperty("SLACK_VERIFICATION_TOKEN");

    if(e.postData.type === 'application/json') {
      const params = JSON.parse(e.postData.getDataAsString());
      // リクエストURL認証用の処理
      if (typeof params.challenge !== 'undefined') {
        return createTextOutput(params.challenge);
      }
      // 検証用トークンの検証
      if (params.token !== verificationToken) {
        return createTextOutput('invalid request');
      }
      // 特定のチャンネルのみのメンションを受け付ける場合はチャンネルIDを検証
      if (params.event.channel === "ChannelId") {
        const userData = JSON.parse(UrlFetchApp.fetch('https://slack.com/api/users.info?user=' + params.event.user, createRequestOptions('GET', accessToken)).getContentText());

        sendMail(userData.user, params);
        postChatMessage(createRequestOptions('POST', accessToken), userData.user.real_name + 'さんの勤怠連絡をメールしました。');
      }
    }
  }catch(err) {
    return createTextOutput(err);
  }
}

function createTextOutput(params) {
  if (typeof params === 'string') {
    return ContentService.createTextOutput(params);
  } else {
    return ContentService.createTextOutput(JSON.stringify(params));
  }
}

function createRequestOptions(method, accessToken) {
  return {'method': method,
    'headers': {
      'Authorization': 'Bearer ' + accessToken
    }
  }
}

function postChatMessage (options, message) {
  const payload = {
    'channel': '投稿するチャンネル名',
    'text': message,
    'username': '作成したBot名'
  };

  UrlFetchApp.fetch('https://slack.com/api/chat.postMessage', {...options, payload});
}

function sendMail(user, params) {
    const toAdr = "";         //送り先アドレス
    const ccAdr = "";         //Ccアドレス
    const bccAdr = "";        //bccアドレス
    const subject = "";       //メールの題目
    const name = "";          //送り主の名前
    const files = new Array();//添付ファイル
    
    const replaceText = params.event.text.replace(`<@${params.authorizations[0].user_id}>`, '');

    const body =
      'お疲れ様です。' + user.real_name + 'です。\n\n' +
      replaceText + '\n\n' +
      'よろしくお願いいたします。';

    MailApp.sendEmail({to:toAdr, cc:ccAdr, bcc:bccAdr, subject:subject, name:name, body:body, attachments:files});
}

※最低限、toAdr及びsubject、channelId、投稿するチャンネル名、作成したBot名は変更してください。
        
2. デプロイ→デプロイを管理の順にクリックする。
17.png
3. 鉛筆アイコン(編集ボタン)をクリックした後、バージョンを新バージョンに設定して、デプロイをクリックしてアプリケーションを更新する。
18.png
4. 実際にBotに対してメンションしてメール送信テストを実行する。

最後に

今回はGASのメソッドを利用してメール送信を実装しましたが、メールサーバーを別途用意したり、SendGrid等を使えば、メールの送信元を指定することも可能です。

GASのメール送信は1日100通の制限があるので、利用頻度や用途に合わせてカスタマイズしていただければ幸いです。

4
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?