LoginSignup
3
0

【続】LINEメッセージをAWS Lambdaで送受信してみたお話

Posted at

はじめに

業務にてLINE MessagingAPIを利用したLINEでのコミュニケーションツールを作成したので、技術共有と自分の備忘録もかねて書いていきます。
今回は前回の投稿の追加機能実装となります。以前までのテキストメッセージ以外にも画像の送受信実装を行います。

前回の振り返り

前回はLINE公式アカウントを作成し、テキストメッセージの送受信を行いました。
いわゆるこんな感じ。
まずはLINEからユーザーがメッセージを行うと、同じテキストを返信する、というもの。

そしてAPIを叩くことで、テキストメッセージを送信するです。

ざっとの説明のため、詳しく知りたい、また今回LINE公式アカウント設定などは省略するため、よければぜひ前回の投稿をご覧ください。

LINE構成

今回作成する構成は以下のようになります。

前回からの追加はS3バケットを追加したことです。
ユーザーが送信する画像の情報をS3バケットに格納し、中身が確認できるようにする。
そしてAPIからは送信したい画像の情報をS3バケットに格納し、それをトリガーにLINEでユーザーへ送信します。

事前準備

前回構成に追加でS3バケットを作成します。

お好きなバケット名でバケットを作成します。

バケット名はURLの一部にもなるので、AWS内で一意である必要があります。同じ名前で複製等できないのでご注意を!

作成したバケットのセキュリティを変更します。
デフォルトでパブリックアクセスをすべてブロックしているため、そちらを一度解除し、アクセスコントロールリストからのアクセスをブロックにチェックを入れます。

次にS3バケットポリシーを以下に設定します。
許可するアクションは、作成したバケットに対し
・”PutObject”…バケットにオブジェクトをアップロードを許可
・”GetObject”…バケット内のオブジェクトをダウンロードを許可
です。

policy.json
{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "Statement1",
            "Effect": "Allow",
            "Principal": "*",
            "Action": [
                "s3:PutObject",
                "s3:GetObject"
            ],
            "Resource": "arn:aws:s3:::MY_BUCKET_NAME/*"
        }
    ]
}

S3にフォルダを2つ作成します。
・LINEから取得した画像情報を格納するフォルダ(今回は「line/」)
・LINE公式アカウントから送信したい画像情報を格納するフォルダ(今回は「temp/」)

作成した2つ目のS3フォルダにはLambdaのトリガーとして追加します。
接続したいLambda関数を選択し、”トリガーを追加”を選択します。

トリガーの設定では先ほど作成したS3と選択します。
今回手動でS3へ格納するため、イベントタイプはPUTを選択。オプションで格納先フォルダを指定しておきます。

設定時にもチェックが必要な項目ですが、PUTをトリガーとしLambda関数が起動するため、異なったフォルダを設定をしたり、Lambda関数内でPUT処理を記述すると処理がループする可能性がありますのでご注意を!

LINE公式アカウントへメッセージ送信

早速Lambda関数を作成していきます!前回投稿のコード分岐内を一部修正した形での実装です。

Receive.py
import json
import urllib

import boto3
from linebot import (
    LineBotApi,
)
from linebot.models import TextSendMessage,ImageSendMessage

# S3接続情報    
s3_resource = boto3.client('s3')

def lambda_handler(event, context):
    
    # paramater取得
    body = json.loads(event["body"])
    event_type = body["events"][0]["type"]
    user_id = body['events'][0]['source']['userId']
    print(f'body: {body}')
    
    # 環境値(ご自身の環境値に置き換えてください。)
    line_channel_access_token = 'LINE_CHANNEL_ACCESS_TOKEN'

    # メッセージ送信BOT作成
    line_bot_api = LineBotApi(line_channel_access_token)

    # Type分岐
    # メッセージ送信の場合の処理
    if event_type == 'message':
        message_type = body['events'][0]['message']['type']
        reply_token = body['events'][0]['replyToken']

        # 送信されたメッセージがtext(テキスト)の場合の処理
        if message_type == 'text':
            message = body['events'][0]['message']['text']
            line_bot_api.reply_message(reply_token, TextSendMessage(message))
            
        # 送信されたメッセージがimage(画像)の場合の処理
        elif message_type =='image':
        ##---- 追加 ---- 
            message_id = body['events'][0]['message']['id']

            # 環境値(ご自身の環境値に置き換えてください。)
            bucket_name = 'S3_BUCKET_IMAGES'

            # リクエストURLを構築
            url = f"https://api-data.line.me/v2/bot/message/{message_id}/content"
            # リクエストヘッダー
            headers = {
                'Authorization': f'Bearer {line_channel_access_token}'
            }
        
            # HTTP GETリクエストの送信
            try:        
                request = urllib.request.Request(
                    url, headers=headers)
                with urllib.request.urlopen(request) as f:
                    response = f.read()
        
            except Exception:
                # エラーハンドリング
                return {
                    'statusCode': 200,
                    'body': json.dumps({'message': 'システムエラーが発生しました。(LINE通信エラー)'}, ensure_ascii=False)
                }
            
            file_path = 'line/' + message_id + '.jpg'
            
            try:
                # s3格納
                s3_resource.put_object(Bucket=bucket_name, Key=file_path, Body=response)
        
            except Exception:
                # エラーハンドリング
                return {
                    'statusCode': 200,
                    'body': json.dumps({'message': 'システムエラーが発生しました。(バケットエラー)'}, ensure_ascii=False)
                }

            line_bot_api.reply_message(reply_token, ImageSendMessage(original_content_url='https://s3-ap-northeast-1.amazonaws.com/' + bucket_name + '/' + file_path, preview_image_url='https://s3-ap-northeast-1.amazonaws.com/' + bucket_name + '/' + file_path))
            
        # 上記以外のmessage_typeの場合の処理(今回の対象外)
        else:
            pass
               
    # メッセージ送信取消の場合の処理
    elif event_type == 'unsend':
        line_bot_api.push_message(user_id, TextSendMessage("送信取消されました。"))
    
    # 上記以外のevent_typeの場合の処理(今回の対象外)
    else:
        pass
    
    return {
        'statusCode': 200
    }


実際の動きはこのような感じです。

テキストメッセージは前回同様におうむ返しですが、画像を送信してみると、、?

送信した画像もおうむ返しされました!(オイシソウ)
またS3のバケットを確認してみると、指定フォルダ配下にオブジェクトが保存されているのも確認できました!


具体的なコードの説明を行なっていきます。今回のポイントは以下の箇所。

# リクエストURLを構築
url = f"https://api-data.line.me/v2/bot/message/{message_id}/content"
# リクエストヘッダー
headers = {
    'Authorization': f'Bearer {line_channel_access_token}'
}

LINEのMessaging APIを用いてコンテンツを取得します。パスパラメーターにはWebhookで受信したメッセージIDを使っています。こちらは画像以外にも、動画や音声も取得可能です。
正常なメッセージIDの場合200、
存在しないメッセージIDの場合404、
すでに送信取り消しをされたメッセージIDの場合410、のレスポンスとなります。

取得できるのはLINEメッセージ送信から30日以内です。

参照:LINE公式ドキュメント コンテンツ取得


LINEから送信できるメディアには制限があります。

画像

制限項目
画像のURL 最大文字数、2000文字
プロトコル HTTPS
画像フォーマット JPEGまたはPNG
画像最大ファイルサイズ 10MB(プレビューは1MB)

動画

制限項目
動画のURL 最大文字数、2000文字
プロトコル HTTPS
動画フォーマット MP4
動画最大ファイルサイズ 200MB(プレビューは1MB)

画像動画以外のメディアもLINEから送信が可能です。詳しい制限については以下を参照ください。
参照:LINE公式ドキュメント 送信メッセージ条件

LINE公式アカウントからメッセージ受信

次にLINE公式アカウントからS3バケットに格納した画像情報をユーザーへ送信します。
(今回送信する画像情報は上記記載のLINEメッセージ送信制限を満たしたものとします)

Send.py
from linebot import (
    LineBotApi,
)
from linebot.models import ImageSendMessage


def lambda_handler(event, context):

    # paramater取得
    bucket_name = event['Records'][0]['s3']['bucket']['name']
    file_path = event['Records'][0]['s3']['object']['key']

    # 環境値(ご自身の環境値に置き換えてください。)
    line_channel_access_token = 'LINE_CHANNEL_ACCESS_TOKEN'   
    user_id = 'LINE_UID'
    
    content_url = 'https://s3-ap-northeast-1.amazonaws.com/' + bucket_name + '/' + file_path
    
    # メッセージ送信BOT作成
    line_bot_api = LineBotApi(line_channel_access_token)
    line_bot_api.push_message(user_id, ImageSendMessage(original_content_url= content_url, preview_image_url = content_url))
      

実際の動きはこのような感じです。
S3バケットにJPEGファイルをアップロードします。

そうすると…?

S3バケットに格納した画像が受信できました!(メシテロ)

LINE Webhook受信についての注意

今回の実装はLINEのWebhookをAPI Gatewayを通じ、Lamdba関数を実行します。LINEには2秒以内にレスポンスを返す必要があるため、Lambda関数での処理は非同期処理で実装することを推奨しています。
(記載している実装は署名処理を省いていること、処理実行時間が2秒かからないことから非同期処理実装をしておりません。)

引用:LINE公式ドキュメント Webhookリクエスト受信時の処理フロー

まとめ

今回はLINE公式アカウントを通じ、テキストメッセージ以外のメッセージをS3バケットを通じて送受信を行いました。例では画像情報しか実装していませんが、動画や音声などさまざまなメディア形式を送受信可能です。ぜひ試してみてくださいね!
引き続き、AWSおよびLINEについての技術吸収、発信ができればと思います。よろしくお願いします〜!

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