Spring BootアプリケーションでSMSを送信する方法
本稿では、Spring BootでWebSocket APIを使用し、シンプルなSMS送信アプリケーションを構築する方法について説明します。
WebSocketは、サーバーとクライアント間の双方向通信チャネルを確立する通信プロトコルです。WebSocketは、現在一般的に使用されているブラウザのほとんどにおいてサポートされています。
プロジェクトの設定
まず、Twilioアカウントの登録と電話番号を取得します。
Twilioのアカウントを登録します。Twilioホームページをブラウザで開き、[今すぐ無料サインアップ]ボタンをクリックするか、Twilioアカウントの作成リンクからサインアップします。このリンクを使用するとアカウントのアップグレード時に$10(米国ドル)相当分のクレジットが追加で付与されます。
次に、SMSの送信に使用する電話番号を購入します。
TwilioコンソールのBuy a Numberにアクセスします。
Countryで、電話番号が属する国を選択します。現在、TwilioではSMS送信機能付きの日本の電話番号は販売していませんので、米国等を選択してください。
各国の電話番号で使用できる機能について詳しくは、TwilioヘルプセンターのTwilioの国際電話番号の利用と機能(英語)を参照してください。
CapabilitiesでSMSがチェックされていることを確認してください。
購入した電話番号をコピーし、記録しておきます。
Spring Initializrを使いプロジェクトを生成します。手順は、次のとおりです。
- http://start.spring.io/にアクセスします。
- [Artifact]の値に「websocket-callback」と入力します。
- [Dependencies]セクションに「Websocket」を追加します。
- [Generate]をクリックし、プロジェクトをダウンロードします。
- ダウンロードしたZIPファイルを展開します。
WebSocketの設定
まずは、WebSocketエンドポイントとメッセージブローカーを設定します。WebSocketConfig
クラスをcom.twilio.websocketcallback
パッケージ内に作成し、以下の内容を入力します。
@EnableWebSocketMessageBroker
は、WebSocketサーバーの有効化に使用します。WebSocketMessageBrokerConfigurer
インタフェースを組み込み、WebSocket接続を設定するメソッドを指定します。
最初のメソッドのconfigureMessageBroker
では、クライアント間のメッセージルーティングに使用するメッセージブローカーを設定しています。
15行目では、宛先が/topic
で始まるメッセージをメッセージブローカーにルーティングするよう定義しています。メッセージブローカーは、特定のトピックを購読しているすべての接続クライアントにメッセージを配信します。
16行目では、宛先が/app
で始まるメッセージを定義しています。メッセージの処理後、コントローラーはメッセージをブローカーチャネルに送信します。
2番目のメソッドregisterStompEndpoints
では、クライアントがWebSocketサーバーへの接続に使用するWebSocketエンドポイントを登録しています。
エンドポイント設定にwithSockJS()
を使用しています。SockJSは、WebSocketをサポートしていないブラウザのフォールバックオプションを有効化するのに使用します。
メソッド名の「Stomp」は、SpringフレームワークのSTOMP実装が由来です。STOMPは、Simple Text Oriented Messaging Protocolの頭字語です。これは、データ交換の形式とルールを定義するメッセージングプロトコルです。
以下の図は、全二重通信チャネルを活用したWebSocketを表したものです。
Gradleを使い構築する
build.gradleファイルを作成し、以下のようにTwilioとWebSocketの依存ライブラリを含めます。
リソース表現クラスを作成する
メッセージ送信SMSクラスをモデル化するには、to
とmessage
のプロパティと、関連するメソッドを持つPlain Old Java Object(POJO)を作成します。
メッセージ処理コントローラーを作成する
STOMPメッセージングを使用するSpringのアプローチでは、STOMPメッセージを@Controller
クラスにルーティングできます。以下のようにSMSController
をマッピングして/sms
へのメッセージを処理します。
このコントローラーは簡潔ですが、内部で実行されるプロセスが多数あります。順に詳しく見てみましょう。
@RequestMapping
アノテーションは、メッセージがPOSTメソッドを介して/sms
に送信されるようにします。コンシューマーが受け入れるcontent_typeは、consumes = MediaType.APPLICATION_JSON_VALUE
に設定されます。
produces = MediaType.APPLICATION_JSON_VALUE
は、マッピングされたハンドラーが生成する値です。produces
は、1つ以上のメディアタイプで構成されます。そのうちの1つは、リクエストの「受け入れ可能な」メディアタイプに対するコンテンツ交渉を介して選択する必要があります。通常、メディアタイプはAccept
ヘッダーから抽出されますが、クエリパラメーターや別の場所から抽出される場合もあります。
メッセージのペイロードは、@RequestBody SMS
オブジェクトにバインドされ、SMSSubmit()
に渡されます。
/topic/sms
の購読者全員への配信は、webSocket.convertAndSend()
メソッドにて指定しています。このメソッドは、アプリケーションからメッセージを、接続されたクライアントに送信するために使用されます。どのアプリケーションコンポーネントもメッセージを「brokerChannel」に送信できます。メッセージの送信は、SimpMessagingTemplate
注入により行われます。
message
フィールドとto
フィールドをUIから受け取ると、フィールド値が解析され、send
メソッドで定義するSMSサービスに送信されます。Twilio電話番号からUIに入力した電話番号に、SMSメッセージが送信されます。
smsCallback
メソッドは、Twilioエンドポイントからのコールバックを受信するコールバックハンドラーメソッドです。このメソッドは、MediaType.APPLICATION_FORM_URLENCODED_VALUE
タイプのリクエストcontent-type
ヘッダーを利用し、MediaType.APPLICATION_JSON_VALUE
タイプのネゴシエーション可能なmediatype
を生成します。このmediatype
を、フロントエンドのJavaScriptエンジンにより処理します。Twilioは、URL_ENCODEDフォーム値でコールバックを送信するため、アプリケーションで簡単に利用できるタイプの形式にしておく必要があります。
メッセージングサービスを作成する
コンポーネントは、単独で交換やアップグレードができるソフトウェアの単位です。
マイクロサービスアーキテクチャでは、ライブラリを使用しますが、ソフトウェアをコンポーネント化する主な方法は、サービスに分解することです。ライブラリはプログラムに関連付けられ、メモリ内の関数呼び出しで呼び出されるコンポーネントとして定義されます。また、サービスはWebサービスリクエストやリモートプロシージャコールなどの方法で通信するプロセス外のコンポーネントとして定義されます。サービスを(ライブラリではなく)コンポーネントとして使用する主な理由は、サービスが単独で展開できるからです。本稿で作成するアプリケーションでは、Twilioエンドポイントと直接通信するサービスを使用し、コールバック情報を送受信しています。
Twilioは、認証情報を開発環境やプロジェクト内に保存しないことを推奨しています。代わりに、開発マシンや、オペレーティングシステムに応じた別の場所にある./bash_profile
(環境)ファイルに認証情報を保存し、そこから取り出すようにしてください。Springでは、外部化設定を介してデータを抽出できます。
環境変数と認証情報を関連付ける方法については、以下のドキュメントを参照してください。
Store Your Twilio Credentials Securely (英語)
以下のコードで、メッセージングサービスを作成します。
ブラウザクライアントを作成する
すべてのサーバーコンポーネントの構築と接続が完了すると、JavaScriptクライアントとHTMLページを使用してUIを実装できます。src/main/resources/static
に、次のようなindex.htmlファイルを作成します。
このHTMLファイルでは、SockJS
とSTOMP
のJavaScriptライブラリをインポートします。これらのライブラリは、WebSocket経由でSTOMPを使用したサーバーとの通信に使用されます。ここでは、クライアントアプリケーションのロジックが含まれたapp.js
ファイルもインポートしています。
そのファイルを作成しましょう。
Gradleを使用している場合は、./gradlew bootRun
を使用し、アプリケーションを実行できます。
サービスをテストする
実行中のサービスをテストします。ブラウザでhttp://localhost:8080を開き、[Send]ボタンをクリックします。
接続が開始されると、宛先の電話番号とメッセージを入力するよう要求されます。情報を入力し、[Send]をクリックします。データは、STOMPを介してJSONとしてサーバーに送信されます。1秒間の遅延をシミュレーションした後、サーバーはTwilioから受信したコールバックステータス情報を含むメッセージを送り返します。
Spring BootとWebSocketでTwilioアプリケーションを構築する
すべてのテストが正常に完了すると、Spring BootとWebSocketで構築したSMSアプリケーションが完成します。本稿でご紹介したコードの全体を確認したい場合は、GitHubリポジトリを確認してください。
ぜひ、ご感想をお寄せください。コミュニケーションの未来を構築しましょう。
その他の参考資料(すべて英語):
Pooja Srinathは、Twilioのシニアソリューションエンジニアです。新しいことを学び、新進気鋭の開発者の助けとなる便利なアプリを構築することに力を入れています。ご不明な点は、psrinath@twilio.comまでお問い合わせください。