質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.49%

discord.pyで一定時間おきに繰り返し発言させたい

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 3,193

score 188

前提・実現したいこと

Discordに緊急地震速報を送るBotを作りたく、毎秒リクエスト可能な緊急地震速報のAPIをDiscord.pyで毎秒リクエストし、変化があった場合にDiscordのテキストチャンネルに送信したいと思っています。

そのために、Discord.pyで一定時間おきに、実行させ発言させるプログラムを作ろうと思っています。

初めて質問します。何か不備があればよろしくお願いします。

発生している問題

エラーは発生しませんが、Discordでは一度しか発言されません。
Discordのテキストチャンネル
※画像は編集しています。

該当のソースコード

import discord
import asyncio

client = discord.Client()

token = "DiscordBotのトークン"

@client.event
async def on_ready():
    asyncio.ensure_future(greeting_gm())

async def greeting_gm():
    channel = client.get_channel('チャンネルID')
    await client.send_message(channel, 'おはよう')
    await asyncio.sleep(10)

client.run(token)

試したこと

ソースコード

@client.event
async def on_ready():
    loop = asyncio.get_event_loop()
    loop.run_until_complete(greeting_gm())

async def greeting_gm():
    channel = client.get_channel('チャンネルID')
    await client.send_message(channel, 'おはよう')
    await asyncio.sleep(10)

発生している問題・エラーメッセージ

Ignoring exception in on_ready
Traceback (most recent call last):
  File "/*/python3.5/site-packages/discord/client.py", line 307, in _run_event
    yield from getattr(self, event)(*args, **kwargs)
  File "testbot.py", line 12, in on_ready
    loop.run_until_complete(greeting_gm())
  File "/*/python3.5/asyncio/base_events.py", line 325, in run_until_complete
    self.run_forever()
  File "/*/python3.5/asyncio/base_events.py", line 290, in run_forever
    raise RuntimeError('Event loop is running.')
RuntimeError: Event loop is running.
Task exception was never retrieved
future: <Task finished coro=<greeting_gm() done, defined at testbot.py:14> exception=InvalidArgument('Destination must be Channel, PrivateChannel, User, or Object. Received NoneType',)>
Traceback (most recent call last):
  File "/*/python3.5/asyncio/tasks.py", line 239, in _step
    result = coro.send(None)
  File "testbot.py", line 17, in greeting_gm
    await client.send_message(channel, 'おはよう')
  File "/*/python3.5/site-packages/discord/client.py", line 1145, in send_message
    channel_id, guild_id = yield from self._resolve_destination(destination)
  File "/*/python3.5/site-packages/discord/client.py", line 289, in _resolve_destination
    raise InvalidArgument(fmt.format(destination))
discord.errors.InvalidArgument: Destination must be Channel, PrivateChannel, User, or Object. Received NoneType
Traceback (most recent call last):
  File "testbot.py", line 20, in <module>
    client.run(token)
  File "/*/python3.5/site-packages/discord/client.py", line 519, in run
    self.loop.run_until_complete(self.start(*args, **kwargs))
  File "/*/python3.5/asyncio/base_events.py", line 335, in run_until_complete
    raise RuntimeError('Event loop stopped before Future completed.')
RuntimeError: Event loop stopped before Future completed.

おそらくですが、asyncioのループとdiscord.pyのループが重複していることが原因だと思います。

補足情報(FW/ツールのバージョンなど)

  • Python 3.5.1
  • discord.py==0.16.12(async版)

可能であれば、Pythonの最新バージョンを使用し、最新のdiscord.py(rewrite版)を使用したいのですが、アップロードする予定のサーバーが、Python2.7.12とPython3.5.1しか対応していないため仕方なくバージョンを下げています。

ソースコードはこちらのサイトのものを使用させていただいています。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+1

@client.event
async def on_ready():
    asyncio.ensure_future(greeting_gm())

async def greeting_gm():
    channel = client.get_channel('チャンネルID')
    while True:
        await client.send_message(channel, 'おはよう')
        await asyncio.sleep(10)

手元のライブラリのバージョンが違ったので試せていませんが、
これでどうでしょう。変更点は while True: ~

エラーは発生しませんが、Discordでは一度しか発言されません。

greeting_gm でループが必要です。

おそらくですが、asyncioのループとdiscord.pyのループが重複していることが原因だと思います。

その通り、client.run で同等の操作(asyncioのループ)を行っています。


非同期プログラミングでは、メインでのイベントループ1か所以外に
ブロッキングを伴う操作はしてはいけません。
(後ろにprint文を挟む等して、コードが実行されているかどうかで判別できます)

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/04/30 19:46

    回答していただきありがとうございました。
    とても参考になりました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.49%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る