All files / src/domain EventProseccor.ts

0% Statements 0/148
0% Branches 0/1
0% Functions 0/1
0% Lines 0/148

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149                                                                                                                                                                                                                                                                                                         
import Logger from "bunyan";
import {
  LineChannel,
  LineChannelRepository,
  LineUserRepository,
  LineWebhookMessage,
  LineWebhookMessageRepository,
} from "./types/Line";
import {
  Event,
  FollowEvent,
  UnfollowEvent,
} from "@line/bot-sdk/dist/webhook/api";

export class EventProcessor {
  constructor(
    private repo: LineChannelRepository &
      LineUserRepository &
      LineWebhookMessageRepository
  ) {}

  async process(logger: Logger): Promise<void> {
    const lineChannels = await this.repo.fetchAllLineChannels();

    for (const lineChannel of lineChannels) {
      const childLogger = logger.child({ line_channel_id: lineChannel.id });
      await this.processByLineChannel(childLogger, lineChannel);
    }
  }

  private async processByLineChannel(
    logger: Logger,
    lineChannel: LineChannel
  ): Promise<void> {
    logger.info(
      `process by LINE Channel ${lineChannel.memo}[${lineChannel.id}]`
    );

    const messages = await this.repo.fetchWebhookMessages(lineChannel.id);
    logger.info(`fetched message count is ${messages.length}`);

    for (const message of messages) {
      const childLogger = logger.child({ message_id: message.messageId });
      await this.handleMessage(childLogger, lineChannel, message);
    }
  }

  private async handleMessage(
    logger: Logger,
    lineChannel: LineChannel,
    message: LineWebhookMessage
  ): Promise<void> {
    try {
      logger.info(
        `handle message [${message.messageId}], event count is ${message.events.length}`
      );

      for (const event of message.events) {
        const childLogger = logger.child({
          event_type: event.type,
          webhook_event_id: event.webhookEventId,
        });
        await this.handleEvent(childLogger, lineChannel, event);
      }

      await this.repo.deleteWebhookMessage(lineChannel.id, message.messageId);
    } catch (err) {
      logger.error({ err }, `failed handle message [${message.messageId}]`);
    }
  }

  private async handleEvent(
    logger: Logger,
    lineChannel: LineChannel,
    event: Event
  ): Promise<void> {
    try {
      logger.info(`handle event [${event.type}]`);
      switch (event.type) {
        case "follow":
          this.handleFollowEvent(logger, lineChannel, event);
          break;
        case "unfollow":
          this.handleUnfollowEvent(logger, lineChannel, event);
          break;
        default:
          logger.warn(`event type [${event.type}] is not supported`);
      }
    } catch (err) {
      logger.error({ err }, `failed handle event [${event.type}]`);
    }
  }

  private async handleFollowEvent(
    logger: Logger,
    lineChannel: LineChannel,
    event: FollowEvent
  ): Promise<void> {
    if (!event.source) {
      logger.warn(`skip event, source is not found`);
      return;
    }
    if (event.source.type != "user") {
      logger.warn(`skip event, source type [${event.source.type}] is not user`);
      return;
    }
    if (!event.source.userId) {
      logger.warn(`skip event, source userId is not found`);
      return;
    }

    const userId = event.source.userId;
    await this.repo.addLineUserIfNotExists(userId);
    await this.repo.upsertElectricityNotifySetting(
      lineChannel.id,
      userId,
      true
    );
    logger.info(`update notify setting to enable for user[${userId}]`);
  }

  private async handleUnfollowEvent(
    logger: Logger,
    lineChannel: LineChannel,
    event: UnfollowEvent
  ): Promise<void> {
    if (!event.source) {
      logger.warn(`skip event, source is not found`);
      return;
    }
    if (event.source.type != "user") {
      logger.warn(`skip event, source type [${event.source.type}] is not user`);
      return;
    }
    if (!event.source.userId) {
      logger.warn(`skip event, source userId is not found`);
      return;
    }

    const userId = event.source.userId;
    await this.repo.upsertElectricityNotifySetting(
      lineChannel.id,
      userId,
      false
    );
    logger.info(`update notify setting to disable for user[${userId}]`);
  }
}