2
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.

Cloud Functionsでリマインダーを実装してみた

Last updated at Posted at 2021-12-10

はじめに

Qiita初投稿です!
現在就活の予定や企業の管理を行う就活管理アプリをしている中で、予定に対して通知機能を実装することになったので、その知見を共有できたらと考えています。

通知用のデータ構造

コレクションのデータ構造は以下のようになっています。

  • users
    • calendars(usersのサブコレクション)
      • notifications(calendarsのサブコレクション)
        • id
        • isDone(通知完了フラグ)
        • time(通知時間)

このようにnotificationsをcalendarsのサブコレクションにしたのはGoogle Calendarのように通知を複数登録したかったためです。

実装について

実装の流れは以下のようになります

  1. Cloud Functionsのスケジュール関数を利用して毎分関数を実行する
  2. その際に通知時間(time)が一致しているかつ、 通知が完了していないもの(isDoneがfalse)であるデータを取得
  3. 通知用データに紐づけられているプッシュ通知に必要なデータ(予定の時間や、FCM用のトークン)を取得
  4. プッシュ通知を実行

今回はこちらの記事を参考にさせていただきました。

1. Cloud Functionsの毎分実行

詳細は公式ドキュメントに記載してますので、説明は省かせていただきます。

コードに関しては以下のようになります。

push-notification.ts
exports.sendReminder = functions.region('asia-northeast1')
  .runWith({ memory: '512MB' })
  .pubsub.schedule('every 1 minutes')
  .timeZone('Asia/Tokyo')
  .onRun(async (context) => {})

ここで注意する必要があるのはリージョンとタイムゾーンを設定する必要があるということです。
もし設定していない場合には、プッシュ通知を行う時間がズレたりします。

補足
ローカルからfunctionsをデプロイした際にタイムゾーンの設定が外れることがあるそうです。以下のサイトを参考にしてみてください。

2. 通知用データの取得

以下のようなコードになります

push-notification.ts
exports.sendReminder = functions.region('asia-northeast1')
  .runWith({ memory: '512MB' })
  .pubsub.schedule('every 1 minutes')
  .timeZone('Asia/Tokyo')
  .onRun(async (context) => {
    // 秒を切り捨てた現在時刻
    const now = (() => {
      let s = admin.firestore.Timestamp.now().seconds
      s = s - s % 60
      return new admin.firestore.Timestamp(s, 0)
    })()
    // notificationsコレクションから現在時刻に該当するデータを取得
    const remindersRef = await admin.firestore().collectionGroup('notifications')
      .where('time', '==', now)
      .where('isDone', '==', false)
      .get()
})

時間が一致しているかを検索する際に、秒数が邪魔になるので切り捨てています。

3. 通知に必要なデータの取得

この辺りは一般的なfirestoreの処理になるので紹介のみにさせていただきます。

push-notification.ts
exports.sendReminder = functions
  .region('asia-northeast1')
  .runWith({ memory: '512MB' })
  .pubsub.schedule('every 1 minutes')
  .timeZone('Asia/Tokyo')
  .onRun(async (context) => {
    // 秒を切り捨てた現在時刻
    const now = (() => {
      let s = admin.firestore.Timestamp.now().seconds
      s = s - s % 60
      return new admin.firestore.Timestamp(s, 0)
    })()
    // notificationsコレクションから現在時刻に該当するデータを取得
    const remindersRef = await admin.firestore().collectionGroup('notifications')
      .where('time', '==', now)
      .where('isDone', '==', false)
      .get()

    // カレンダーのデータとユーザーデータを取得する
    const notCompletedRemindTask: CalendarReminder[] = [];
    await Promise.all(remindersRef.docs.map(async (doc) => {
            /* 省略 */
            notCompletedRemindTask.push(new CalendarReminder(fcmToken, title, content, calendarData.id, userId))
        }).catch((err) => {
          functions.logger.error(`calendarDataError: ${err}`);
        });
      }
    }))
    if (notCompletedRemindTask.length !== 0) {
      // プッシュ通知の送信
      await sendPushNotificationsForReminder(notCompletedRemindTask)
    }
  })

4. プッシュ通知の送信

push-notification.ts
async function sendPushNotificationsForReminder(reminders: CalendarReminder[]) {
  const messages: admin.messaging.Message[] = []
  for (const reminder of reminders) {
    // プッシュ通知の設定
    const notification: admin.messaging.Notification = {
      title: reminder.title,
      body: reminder.content
    }
    const message: admin.messaging.Message = {
      token: reminder.fcmToken,
      // 通知を押したときの遷移先用にdataを入れています
      data: {
        userId: reminder.userId,
        calendarId: reminder.calendarId
      },
      notification: notification
    }
    messages.push(message)
  }
  if (messages.length === 0) { return }
  await sendAll(messages)
}

async function sendAll(messages: admin.messaging.Message[]) {
  // 500件ずつに分割する(500件以上は、sendAllメソッド側でエラーになるため)
  const msgs = messages.reduce((acc: admin.messaging.Message[][], val) => {
    const MAX_BATCH_SIZE = 500
    const last = acc[acc.length - 1]
    if (last.length === MAX_BATCH_SIZE) {
      acc.push([val])
      return acc
    }
    last.push(val)
    return acc
  }, [[]])

  // Promise.allで分割して配信
  const promises = msgs
    .map((ms) => admin.messaging()
      .sendAll(ms)
      .then((batchResponse) => {
        batchResponse.responses.forEach((res) => {
          if (res.error) {
            functions.logger.error(res.error)
          }
        })
        functions.logger.log(`send messages: ${ms.length} count`)
      })
      .catch((e) => {
        functions.logger.error(e)
      })
    )
  await Promise.all(promises)
}

参考にさせていただいた記事にもありますが、Promise.allを利用して並列処理を実装しないで、単純に直列処理にしてもいいと思いました。
特に自分のサービスはそこまでデータ量も多いわけではないので今後のためにという形で実装しておきました。

最後に

ユーザー単体に送る通知やトピックを設定する通知は実装したことがありましたが、今回のような少し複雑なのものは初めて実装したので勉強になりました。

最後にはなりますが、現在開発しているアプリに関してご紹介させていただきます。

就活の管理を行うアプリJobhunです!
Jobhunは「就活状況を一括管理したい...」 「自分が就活の時、予定を組むのが大変だった...」という就活生の声から生まれたアプリです。そのため、実際に就活を経験してきた方の意見を参考に、「本当に必要な機能」をまとめました!

以下のような機能があります。

  • 企業管理機能
  • カレンダー機能
  • ES管理機能 etc...

気軽に就活管理ができるようになっているので、是非ダウンロードしてみてください!!
LPIOSAndroid
1639120308483.jpg

2
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
2
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?