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

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

ただいまの
回答率

89.13%

Pythonでのdiscord bot作成、ユーザーリストの自動取得

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 481

moorumoru

score 10

前提・実現したいこと

C言語やJavaScriptなどをすこし理解できているくらいの人間なのですが、
discordでBOTを作りたくて初めてPythonを触り始めました。
グループ内でお昼ご飯を注文するために以下の条件で絞り込み、お知らせしたいと考えています。

①指定した時間に
②特定のチャンネル内にいる
③お昼ご飯を注文したい人で(ロールを付与しておく)
④オンラインの人から(オフラインの人は除外)
⑤ランダムで一人代表者を決める
⑥メンションを付けてお知らせメッセージを飛ばす

試したこと・発生している問題

①と⑤はできそうなのですが、②~⑤の条件をかなえるための手法をご提案いただきたいです。
調べたところ、コマンド入力をしてメッセージを受け取ったチャンネルにおいてメンバーの取得はできましたが、
時刻をトリガーとして自動で選び、お知らせしてほしいです。

該当のソースコード

from datetime import datetime
from discord.ext import tasks
import random
import asyncio
import discord

client = discord.Client()

BOT_TOKEN = "hogehoge"
CHANNEL_ID = hogehogenumber


@client.event #ログインの確認
async def on_ready():
    print('Logged in as')
    print(client.user.name)
    print(client.user.id)
    print('------')
    #ループ処理実行
    loops.start()

# 60秒に一回ループ
@tasks.loop(seconds=60)
async def loops():
    # 現在の時刻
    now = datetime.now().strftime('%H:%M')
    print(lunchtime)
    channel_sent = client.get_channel(CHANNEL_ID)

    #お知らせの設定時刻
    if now == '11:00': 
        m = "今日のお昼ご飯当番を選ばなきゃ!!\n`/members` と入力してね!"
    await channel_sent.send(m)  


@client.event
async def on_message(message):

    # メンバーのリストを取得して表示
    if message.content == '/members':
        s = message.channel.members
        m = "今日のお昼ご飯当番は" + s[0].mention + "さんです!!"  
        await message.channel.send(m)





# Botの起動とDiscordサーバーへの接続
client.run(BOT_TOKEN)

わからないこと

discordにおける
・channel と guild
・member と user
など、区別がついてないなど初歩的な躓きも多いので、
もし合わせてご教授いただけますと幸いです。

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

数日前にインストールしたばかりのため
・Python3.8.2
・discord.py 1.3 (rewriteのほう)
だと思います。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+1

2~5の条件を一つずつ解決していくと良いでしょう。

2. 特定のチャンネル内にいるメンバーの取得

on_message 内の message.channel.members と同様に、
discord.TextChannel.members を使用すると良いでしょう。(ドキュメント

loops内では channel_sent の内容が discord.TextChannel インスタンスですね。
つまり、channel_sent.members がそのメンバーのリストになります。

3. 特定のロールを付与されているメンバーの取得

discord.Role.members を使用します。(ドキュメント
なお、2.で取得したメンバーリストと重複するメンバーを抜き出すなら、一度それらを集合型に変換して、その積集合を取ると良いでしょう。

4. オンラインのメンバーの取得

既にメンバーのリストを取得しているので、その中からオンラインのメンバーを抽出します。
メンバーのステータスは discord.Member.status で取得することができます。
これは、discord.Status クラスで定義されているいずれかの値を取ります。
オンラインの場合、discord.Status.online になります。

実際の抽出方法についてですが、様々な実装方法があると思います。
内包表記を使ったり、filter() 関数を利用するとスマートです。

以下はメンバーのリストからオンラインのものを抜き出す例です(membersを抽出前のそれと定義します)

onlines = []
for member in members:
    if members.status == discord.Status.online:
        onlines.append(member)

onlines = [member for member in members if member.status == discord.Status.online]
onlines = list(filter(lambda member:member.status == discord.Status.online,members))

5. ランダムで一人代表者を決める

リストから特定の要素をランダムに選択する場合、random.choice() を使用するのが良いでしょう。
random.choice はランダムに選択した要素をそのまま返します。

実装例

このように実装できるのでは、と思います。
(動作確認は行っていません、ご了承ください)

基本的には上記の内容をそのまま実装しているつもりです。
質問内容との齟齬があれば、すみません。

ROLE_ID = 123456789123456789 # ここにはロールのIDを記述することを想定しています

# 60秒に一回ループ
@tasks.loop(seconds=60)
async def loops():
    # 現在の時刻
    now = datetime.now().strftime('%H:%M')
    print(lunchtime)
    channel_sent = client.get_channel(CHANNEL_ID)
    lunch_role = channel_sent.guild.get_role(ROLE_ID) # ROLE_ID のロールを取得

    tmp = list(set(channel_sent.members) & set(lunch_role.members)) 
    # 特定のチャンネル内にいる、お昼ご飯を注文したい人のリスト。積集合を取っています

    online_members = [member for member in tmp if member.status == discord.Status.online]
    # 先のリストから、ステータスがオンラインのメンバーのみのリストを作成。


    #お知らせの設定時刻
    if now == '11:00': 
        m = f"今日のお昼ご飯当番は {random.choice(online_members).mention} さんです!!"
    await channel_sent.send(m)

わからないこと について

discord.User はDiscordのユーザーそのものを指し示すクラスです。
つまり、Discord全体から見た一個人です。

discord.Member は、あるサーバーのメンバーを指し示すクラスです。
これは、一つのサーバーから見た一個人です。

discord.User にサーバー特有の情報(持っている役職など)を付け加えたものが discord.Member 、という感じでしょうか?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2020/05/16 13:16

    ご丁寧にありがとうございます!!!

    キャンセル

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

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