0
0

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 1 year has passed since last update.

medibaAdvent Calendar 2023

Day 14

やべぇ!? クレカの口座引き落とし忘れてた...からのGAS+Amazon Connect自動電話通知システム構築

Last updated at Posted at 2023-12-13

本記事について

  • これはmediba Adventカレンダーの14日目の記事です。テクノロジーセンター Eng5Gのずーんが担当いたします。インフラエンジニアのマネージャーを普段やってます。
  • Amazon Connect + Google Apps Scriptの連携をやろうと思っている方向けの事例として記載しています。
  • あくまでも個人的に作っているものなので、不具合や💩等があると思いますのでご承知おき下さい。

ダメージの大きなやらかし

突然ですが皆さん。クレジットカード等の口座引き落としをうっかり忘れてたなんてことありませんか?私はつい先日やらかしました。Googleカレンダーにも引き落としの日を入力していたのですが、カレンダーからの通知を見逃してお金を移動することを忘れました。支払いが遅れるとクレジットヒストリーにも傷がついてしまうのでかなり落ち込みました。

カレンダーの通知って目立ちにくい

再発防止をするために見逃してしまう原因を考えたのですが、カレンダーの通知って押しが弱いのでスルーしがちだと思いました。ちょろっと音が鳴って予定が表示されるだけなので仕事で忙しい時などは後で見よとなってそのまま忘れることがしばしばあります。

そこで考えた方法

普段インフラエンジニアをやっている関係でTwilioなどを利用したアラート電話などを受け取ることが多いのですが、電話の着信は何かしらのアクションを行わない限り止まらないのと作業を中断して見なければならない気持ちがただの通知よりも強く働くことに気が付きました。

そこでGoogleカレンダーに登録されている引き落としイベントの前日に電話で自動的に通知するシステムを構築しようと考えました。

アーキテクチャ

自動電話通知システムの構成は次のようにしてみました。

ブログ.drawio.png

  • 採用理由
    • Amazon Connect
      • 電話と連携できるAPIかつ個人でも利用できるサービスはかなり限られています。Twilio等を検討しましたが電話フローをコードレスで構築できてかつ、AWSサービスとの連携が容易であるため採用しました。
    • Google Apps Script (GAS)
      • 今回はGoogleカレンダーからのデータを取得するということでGASを採用しました。いままで触ったことがほとんどなかったため触りたかったというのも理由になります。
      • 基本JavaScriptで処理を書くことになります。
    • AWS Lambda
      • サーバレスでお手軽に処理を書くと言ったらAWS LambdaやCloud Funcionsが思い浮かびますが、今回はAmazon Connectとの連携が必要なためAWS Lambdaを採用しました。
      • 言語はPythonでPython3.9を採用しました。

構築方法

  1. Amazon Connectの構築

    1. 東京リージョンでAmazon Connectのコンソールを開きます。

    2. インスタンスを追加するボタンをクリックして、インスタンスを作成します。パラメータは下記をご参考ください。

      • IDを管理
        • Amazon Connectにユーザを保存
      • アクセス URL
        • 任意の名前を入力
          • インスタンスにアクセスするときのURLになります。
      • 管理者を指定
        • 自分のアカウント情報を適宜入力
          • IAMとは全く別なので自由に設定してもよい
      • テレフォニーオプション
        • 着信を許可するのみにチェック
      • データストレージ
        • デフォルトのまま
        • 利用しないのでCustomer Profilesを有効化のチェックは解除する
    3. インスタンスの作成ボタンを押下するとインスタンスの構築が実行されます。しばらくするとインスタンス作成が完了するのでアクセスURLからAmazon Connectの管理画面にアクセスする。

    4. 先ほど管理者に指定したアカウント情報を入力し、ログインする。

    5. ログインが完了するとこのような画面になります。

      • image.png
    6. 左のメニューから下記のアイコンのメニューを展開し、電話番号を選択します。

      • image.png
    7. 電話番号の取得ボタンをクリックして、電話番号取得の手続きを行います。

      • image.png
    8. 電話番号取得のためにチャンネルと国を選択します。

      • 注意1. 日本の電話番号は取得に際して、法人登記等の証明書が必要になるため個人では取得不可能です。日本の法律で厳しい規則があるのでしょうがないです。。。
      • 注意2. Amazon Connectでは日本の携帯電話のプレフィックスナンバーにあたる+8109, +8108, +8107に対して初期状態ではコールできません。サポート窓口から緩和申請を行う必要があります。SESと似たようなものですね。
      • 注意3. 電話番号を取得して保持している間は1日単位で課金が発生します。どのタイプの電話番号を持つかによって単価が変動します。オーストラリアの電話番号はDIDだと1日0.1USD掛かります。
      • このため今回はオーストラリアの電話番号を選択し、取得します。
      • image.png
    9. 電話番号を取得出来たらフローメニューを選択して、フローを作成をクリックする。処理を作成して、保存の上公開ボタンをクリックする。

      • image.png
    10. 左のブロックメニューからアクションを選択して、繋げていくことで簡単に電話でやりたいことを指示できます。

      • ここで利用しているブロック
        • 音声の設定
          • 合成音声を作成するための初期設定。言語や音声の種類を選択できます。ここでは日本語・Kazuhaにしています。
        • プロンプトの再生
          • 合成音声を利用してしゃべらせる内容を定義しています。ただ単に固定のテキストをしゃべらせるだけではなく、動的に定義される属性を利用して内容を差し替えることが可能です。ユーザに関する属性なども使えるようになっています。
          • ここでは、どの引き落としイベントがあるのかをAPIから差し込みできるようにwithdrawという属性を定義して読み上げる処理を入れています。
        • 切断
          • 処理の終端になります。ここでは、エラー等が起きても切断するように設定しています。
      • image.png
    11. ここまで出来たら次の情報をメモしておきます。

  2. AWS Lambdaの構築

    1. 今回はSAM CLIを利用して構築します。長くなるので細かな内容は省きます。
      parameter_overrides = [
          "DestinationPhoneNumber=<電話をかける先の番号>",
          "ContactFlowId=<Contact Flow ID>",
          "AmazonConnectInstanceId=<Amazon Connect Instance ID>",
          "SourcePhoneNumber=<今回取得した電話番号>"
      ]
      
      • その他、Profileなどを適宜指定して、sam deployでデプロイします。
    2. 肝心のPythonのコードではboto3を利用して、Amazon Connectのstart_outbound_voice_contactのAPIを叩いています。APIを実行する際の引数にAttributesという項目がありますが、こちらでコンタクトフローに渡したいパラメータを入力することができます。今回はwithdrawという属性にGASから受け取ったイベント内容を突っ込んで渡しています。(サクッと作っているのでコードが汚いのは許してください)
    3. また、template.yamlで作成しているリソースはLambdaとAmazon Connectを操作するためのロールです。今回はAPI Gatewayを利用せずに関数URLを利用するためにURLの発行設定も記述しています。
    4. SAMのデプロイができたら関数URLをLambda Functionのコンソールの設定画面より取得してメモしておきます。
  3. GASの構築

    1. GASへのアクセスはGoogleスプレッドシートから可能です。
      1. 拡張機能を選択し、Apps Scriptを選択します。

        • image.png
      2. ファイルの+ボタンを押して、スクリプトを追加してコードを記述します。記述が終わったら保存ボタンを押下します。

        • image.png
        const LAMBDA_URL = PropertiesService.getScriptProperties().getProperty("LAMBDA_URL") // AWS Lambdaの呼び出しURL
        const regx = /引き落とし/g // イベントのタイトルで引き落としのイベントに含まれるワード
        
        
        const main = function() {
          var calendars = CalendarApp.getAllOwnedCalendars();
        
          // 引き落としがあるイベント
          let withdraw_event = []
        
          // スキャン対象日付
          const now = new Date()
          console.log(now)
          const nextday_begin = new Date(now.getFullYear(), now.getMonth(), now.getDate() + 1, 0, 0, 0, 0)
          const nextday_end = new Date(now.getFullYear(), now.getMonth(), now.getDate() + 1, 23, 59, 0, 0)
          console.log(nextday_begin, nextday_end)
        
          // 対象期間のイベント取得
          const events = CalendarApp.getEvents(nextday_begin, nextday_end)
          console.log(events.map((x) => x.getTitle()))
        
          // 対象期間中のイベントをスキャンし、regxに当てはまるイベントがあればtrueを配列に格納する。なければFalseを配列に格納する。
          const withdraw_eventflags = events.map((x) => {
            const title = x.getTitle()
            const result = title.match(regx)
            if (result === null) {
              return false
            }
            withdraw_event.push(title)
            return true
          })
        
          // 最終的にregxに含まれるイベントがあったかを判定
          const withdraw_flag = withdraw_eventflags.reduce((x, y) => x || y, false)
          console.log(withdraw_flag)
        
          // regxが含まれるイベントがあった場合
          if (withdraw_flag) {
            console.log(withdraw_event)
            const event_name = withdraw_event.join('')
            const base_url = LAMBDA_URL
        
        
            // AWS Lambdaに対してHTTPリクエスト
            // クエリパラメータであるwithdrawに引き落としイベント名を包含して引き渡す。
            const url = base_url + '?withdraw=' + event_name
            const options = {
              method: "GET",
            }
            UrlFetchApp.fetch(url, options)
          }
        
          return
        }
        
      3. 左メニューからプロジェクトの設定をクリックして、スクリプトプロパティとして、LAMBDA_URLを設定します。値には先ほどメモしたLambdaのURLを入力し保存します。

        • image.png
        • image.png
      4. エディタ画面に戻り、一度実行ボタンをクリックします。そうすると自分のGoogleアカウントへのアクセスをGASから許可するかを尋ねるプロンプトが出てくるため許可を押下します。

      5. ここまで来たらあとは、実行トリガーを設定します。左メニューのトリガーを押下します。

        • image.png
      6. 右下のトリガーを追加ボタンをクリックして、画像のように設定し、保存します。

        • ここでは毎晩7-8時の間に実行される設定になっています。
        • image.png

実際の挙動

実際にこのシステムが動くとこのような電話が掛かってきます。

感想

GASを今回初めて触りましたが想像以上に簡単にGoogleサービスとの連携ができたので、今後会社の業務でもより活用していきたいモチベーションが上がりました。また、Amazon Connectとの連携はLambdaを嚙ませれば簡単にできるので2日程度で構築ができました。これで二度と口座引き落としを忘れることはないでしょう。

おわりに

現在medibaでは、メンバーを大募集しています。

募集・応募ページ
medibaってどんな会社だろう? と興味を持っていただいた方は、
カジュアル面談もやっておりますので、お気軽にお申込み頂ければと思います。

特にAWS周りのインフラ強い人待ってます!

0
0
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
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?