2
2

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 5 years have passed since last update.

ErrbotのBackendプラグインを作ろう

Last updated at Posted at 2016-12-03

はじめに

PythonのChatBotツールであるErrbotのBackendプラグインを作成したので、その作り方を書き残します。

最新版のErrbotはPython3系のみ対応なのでこのドキュメントもPython3系を利用します。

Backendプラグインとは

Errbotとチャットシステム(SlackとかHipChatとか)をつなぐプラグインのこと。
SlackとかHipChatとかメジャーなチャットシステムは公式のBackendプラグインが用意されています。
(なので、よほどのことがない限りこのドキュメントは読まれることは無いと思います…)

Botにこういうことをしゃべらせたい!などの通常のプラグインの作り方は http://errbot.io/en/latest/user_guide/plugin_development/index.html をみてください。

つくりかた

まずは[Advanced] Backend developmentのページを穴のあくほど読みましょう。
そんなに長いドキュメントではないのであとは英語力、英語力。

以下簡単にだーっと説明。

基本構造

すべてのBackendプラグインはErrBotクラスを継承します。
そしてこのErrBotクラスの中に存在するメソッドを必要な分だけオーバーライドして実装を進めます。ErrBotクラスはさらにBackendクラスを継承していたりするので意外に実装が必要なメソッドは多め。

継承元のクラスを眺めているとNotImplementedErrorが記載されているメソッドが多数存在していて、そこを開発者は自分が対応したいチャットシステムの機能に合わせて実装をしていきます。

識別子

すべてのBackendプラグインは ユーザーの識別子チャットルームにいるユーザーの識別子 を管理する必要があります。特に build_identifierメソッドは重要なもので、特定の文字列からこれらの識別子インスタンスを返す必要があります。

細かいことを気にせず以下のひな型で識別子の使われ方をfeelしてください。

ひな型

hogeというチャットシステムに対応したBackendプラグインを作りましょう。

Qiitaで読むのは長い!という方にはGithubもどうぞ。config.pyの設定例も入っています。
https://github.com/deko2369/errbot-backend-plugin-template

# -*- coding: utf-8 -*-
from errbot.backends.base import Message, ONLINE, Room, \
                                    RoomOccupant, Identifier
from errbot.core import ErrBot

# Can't use __name__ because of Yapsy
log = logging.getLogger('errbot.backends.hoge')

# Hogeチャットシステムのライブラリを読み込む(ライブラリは実在しません)
try:
    import HogeChatClient
except ImportError:
    log.exception("Could not start the hoge backend")
    log.fatal(
        "You need to install the HogeChatClient package in order "
        "to use the Hoge backend. "
        "You should be able to install this package using: "
        "pip install HogeChatClient"
    )
    sys.exit(1)

class HogeUser(Identifier):
    """
    チャットシステムのユーザーを表現するクラス

    直接このクラスをインスタンス化せず、Backendクラスのbuild_identifierメソッドの中で
    オブジェクトを生成するようにする

    ref. http://errbot.io/en/latest/errbot.backends.base.html#errbot.backends.base.Identifier
    """
    def __init__(self, username, bot):
        """
        ユーザーを初期化する

        usernameはチャットシステムで利用されている名前を指定
        """
        self._username = username
        self._bot = bot

    @property
    def username(self):
        """
        ユーザーの名前を返す
        """
        self return._username

class HogeRoomOccupant(RoomOccupant, HogeUser):
    """
    チャットシステムのチャットルームにいるユーザーを表現するクラス

    ref. http://errbot.io/en/latest/errbot.backends.base.html#errbot.backends.base.RoomOccupant
    """
    def __init__(self, username, roomname, bot):
        """
        特定のチャットルームにいるユーザーを初期化する
        """
        super().__init__(username, bot)
        self._room = HogeRoom(roomname, bot)

    @property
    def room(self):
        """
        ルーム情報を返す
        """
        return self._room

class HogeBackend(ErrBot):
    """
    Backendクラス本体

    ここにチャットシステムの実際のやりとりを書いていく
    """
    def __init__(self, config):
        """
        初期設定
        """
        super().__init__(config)
        identity = config.BOT_IDENTIY
        self.token = identity.get('token', None)
        if not self.token:
            log.fatal(
                'You need to set your token in the BOT_IDENTITY setting '
                'in your config.py .'
            )
            sys.exit(1)

        # tokenを指定してクライアントを初期化
        self.client = HogeChatClient.Client(self.token)

        # bot自身のidentifierを作成
        self.bot_identifier = HogeUser('BOT NAME', self)

    def build_reply(self, mess, text=None, private=False):
        """
        返信メッセージを作成する
        """
        # メッセージを構築
        response = self.build_message(text)
        # 返信元のIdentifier
        response.frm = self.bot_identifier
        # 返信先のIdentifier
        response.to = mess.frm

        return response

    def prefix_groupchat_reply(self, message, identifier):
        """
        グループチャットへの返信メッセージのテンプレ
        """
        message.body = '@%s %s' % (identifier.username, message.text)

    def build_message(self, text):
        """
        メッセージオブジェクトの作成
        """
        return super().build_message(text)

    def build_identifier(self, text_repf):
        """
        Identifierオブジェクトの作成

        Hogeチャットシステムは以下の書式でIdentifierを構築する
        ユーザー: @<username>
        チャットルームにいるユーザー: @<username>#<roomname>
        """
        text = text_repr.strip()

        if text.startswith('@') and '#' not in text:
            return HogeUser(text.split('@')[1], self)
        elif '#' in text:
            username, roomname = text.split('#')
            return HogeRoomOccupant(username.split('@')[1], roomname, self)

        raise RuntimeError('Unrecognized identifier: %s' % text)

    def serve_once(self):
        """
        チャットシステムから新着メッセージを受け取り処理を行うメソッド

        このserve_onceメソッドはErrBotから定期的にコールされる
        似たようなオーバーライド対象メソッドにserve_foreverもある

        ref. http://errbot.io/en/latest/errbot.backends.base.html#errbot.backends.base.Backend.serve_forever
        """
        # 新着メッセージを取得
        mess = self.client.new_messages()

        # 取得したメッセージを順番に処理
        # メッセージにはユーザー名と発言されたルーム名も格納されている
        for msg in mess:
            # Messageオブジェクトを構築
            m = Message(msg)
            m.frm = HogeRoomOccupant(msg.username, msg.roomname, self)
            m.to = HogeRoom(msg.roomname, self)

            # メッセージのコールバックを呼び出す
            self.callback_message(m)

            # @<username>が含まれている場合はcallback_mentionも呼び出す
            # 詳細な実装を省いているので注意
            if '@' in msg:
                mentions = [HogeUser(username, self), ...]
                self.callback_mention(m, mentions)

    def send_message(self, mess):
        """
        チャットシステムへの送信部分の実装
        """
        self.client.send(mess.body)

    def connect(self):
        """
        チャットシステムのライブラリのコネクションを返却
        """
        return self.client

    def query_room(self, room)
        """
        roomの文字列からHogeRoomのオブジェクトを返却する処理
        """
        r = self.client.room_info(room)
        return HogeRoom(r.name, self)

    @property
    def mode(self):
        """
        現在のバックエンドを示すユニークな文字列
        """
        return 'hoge'

    @property
    def rooms(self):
        """
        botの入室しているRoomインスタンスを返却
        """
        return []

    def change_presense(self, status=ONLINE, message=''):
        """
        botの入室ステータスが変化するときに呼び出される処理
        """
        super().change_presence(status=status, message=message)

class HogeRoom(Room):
    """
    Hogeチャットシステムのチャットルームについての定義

    チャットルームに対してbotが参加したり、作成したりする機能を実装する
    チャットシステムのクライアントがそういった機能を提供していない場合は実装不可能...

    ref. http://errbot.io/en/latest/errbot.backends.base.html#errbot.backends.base.Room
    """
    def __init__(self, name, bot):
        self._name = name
        self._bot = bot

    def join(self, username=None, password=None):
        """
        botが部屋にjoinする
        """
        self._bot.client.join_room(self._name)

    def leave(self, reason=None):
        """
        botが部屋からleaveする
        """
        self._bot.client.leave_room(self._name)

    def create(self):
        """
        botが部屋をcreateする
        """
        self._bot.client.create_room(self._name)

    def destroy(self):
        """
        botが部屋をdestroyする
        """
        self._bot.client.destroy_room(self._name)

    @property
    def exists(self):
        """
        部屋が存在するかどうか
        """
        return self._bot.client.exist_room(self._name)

    @property
    def joined(self):
        """
        botが部屋にjoinしているかどうか
        """
        return self._bot.client.is_joined(self._name)

    @property
    def topic(self):
        """
        部屋のトピックを取得
        """
        return self._bot.client.room_info(self._name).topic

    @topic.setter
    def topic(self, topic):
        """
        部屋のトピックを設定
        """
        return self._bot.client.set_room_topic(self._name, topic)

    @property
    def occupants(self):
        """
        部屋に存在するユーザーのIdentifierを取得
        """
        return [HogeUser(name, self._bot) \
                for name in self._bot.client.get_room_usernames(self._name)]

    def invite(self, *args):
        """
        部屋にユーザーを招待
        """
        for ident in args:
            self._bot.client.invite(self._name, ident.username)

最後に

Errbotを使う側は楽ですが、各チャットシステムとの架け橋(Backend)を作る開発者は辛い思いをしますねw

参考

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?