Running Flask & a Discord bot in the same application
Asked Answered
I

6

15

I am building a Discord bot in Python and would like to receive HTTP requests from Twitch.tv's API (See Webhooks Guide & Webhooks Reference) (To subscribe to events like; X streamer has gone live) and based on the content of the HTTP (POST or GET) request received from Twitch, do something on the Discord bot, e.g: Output a message on a text channel.

I am using the discord.py Python Discord API/Library.

I've looked into the matter and found that Flask seemed like a good minimalist choice for a webserver to receive these requests on.

I should preface this by saying I'm very new to Python and I've never used Flask before.

Now. The problem is I can't seem to figure out a way to run the Flask server inside of my discord bot.

I've tried adding this simple code into my discord.py script:

from flask import Flask, request
app = Flask(__name__)
@app.route('/posts', methods=['POST'])
def result():
    print(request.form['sched'])
    # Send a message to a discord text channel etc...
    return 'Received !'

When I run my discord.py script which looks something like this: (Stripped away some commands and features for the sake of keeping this shorter)

import discord
import asyncio

from flask import Flask, request
app = Flask(__name__)
@app.route('/posts', methods=['POST'])
def result():
    print(request.form['sched'])
    # Send a message to a discord text channel etc...
    return 'Received !'

client = discord.Client()

@client.event
async def on_ready():
    print('Logged in as')
    print(client.user.name)
    print(client.user.id)
    print('------')

@client.event
async def on_message(message):

    if message.author == client.user:
        return

    content = message.content
    fullUser = message.author.name+'#'+message.author.discriminator
    print(str(message.timestamp)+" #"+message.channel.name+" "+fullUser+": "+str(content.encode('ascii', 'ignore').decode('ascii')))
    if content.startswith('!'):

        content = content[1:]
        if content.startswith('test'):
            counter = 0
            tmp = await client.send_message(message.channel, 'Calculating messages...')
            async for log in client.logs_from(message.channel, limit=100):
                if log.author == message.author:
                    counter += 1

            await client.edit_message(tmp, 'You have {} messages.'.format(counter))

client.run('MyTokenHere')

It seems like if I point flask to discord.py (the above) and run it, it'll start the code, get to the "client.run('MyTokenHere')" part for discord, and just stop at that and run the discord bot. It's not until I exit out of the bot by doing Ctrl+C that the actual Flask server starts, but now the discord bot is disconnected and no longer does any processing.

The same problem persists if I were to for example add "app.run()" somewhere in my code (before calling "client.run()" which starts the Discord bot part) to launch the Flask server; It'll just run the flask, get stuck on that until I Ctrl+C out of the Flask server, then it'll proceed to start the Discord bot. Ultimately, I need to use the Discord API and I need to be connected to the Discord API gateway and all that good jazz to actually send messages to a channel with the bot, so I don't really know what to do here.

So. I think I've tried my best to explain what I'm ultimately trying to achieve here, and hopefully someone can help me find a way to either make this work with Flask, or if there's a better and easier way, provide a different solution.

Imperturbation answered 8/2, 2018 at 19:9 Comment(2)
You'll probably have to run them in different threads.Cleistogamy
You'll need to create a child thread for your discord bot. Try using the threading module.Nuris
I
8

As the kind commenters informed me; threading seems like the way to go. Thanks guys!

Imperturbation answered 9/2, 2018 at 13:56 Comment(2)
Did you do threading on flask or the bot?Kucik
The Discord Bot REQUIRES being in the main thread from what it looked like. So, there are certainly some gaps with implementationEliseoelish
A
10

This is cog example in discord.py

I made this thing for dbl (Discord Bot Lists) you can implement, the thing you need.

Note: run your heroku app with webprocess

example :
web: python main.py

Then go on https://uptimerobot.com/ and setup to ping your web app after every 5 minutes

from aiohttp import web
from discord.ext import commands, tasks
import discord
import os
import aiohttp

app = web.Application()
routes = web.RouteTableDef()


def setup(bot):
    bot.add_cog(Webserver(bot))


class Webserver(commands.Cog):
    def __init__(self, bot):
        self.bot = bot
        self.web_server.start()

        @routes.get('/')
        async def welcome(request):
            return web.Response(text="Hello, world")

        @routes.post('/dbl')
        async def dblwebhook(request):
            if request.headers.get('authorization') == '3mErTJMYFt':
                data = await request.json()
                user = self.bot.get_user(data['user']) or await self.bot.fetch_user(data['user'])
                if user is None:
                    return
                _type = f'Tested!' if data['type'] == 'test' else f'Voted!'
                upvoted_bot = f'<@{data["bot"]}>'
                embed = discord.Embed(title=_type, colour=discord.Color.blurple())
                embed.description = f'**Upvoter :** {user.mention} Just {_type}' + f'\n**Upvoted Bot :** {upvoted_bot}'
                embed.set_thumbnail(url=user.avatar_url)
                channel = self.bot.get_channel(5645646545142312312)
                await channel.send(embed=embed)
            return 200

        self.webserver_port = os.environ.get('PORT', 5000)
        app.add_routes(routes)

    @tasks.loop()
    async def web_server(self):
        runner = web.AppRunner(app)
        await runner.setup()
        site = web.TCPSite(runner, host='0.0.0.0', port=self.webserver_port)
        await site.start()

    @web_server.before_loop
    async def web_server_before_loop(self):
        await self.bot.wait_until_ready()
Assimilation answered 20/6, 2020 at 3:32 Comment(0)
I
8

As the kind commenters informed me; threading seems like the way to go. Thanks guys!

Imperturbation answered 9/2, 2018 at 13:56 Comment(2)
Did you do threading on flask or the bot?Kucik
The Discord Bot REQUIRES being in the main thread from what it looked like. So, there are certainly some gaps with implementationEliseoelish
A
3

Another cool way if you aren't going to use flask extensions, use quart instead of flask, then it will super easy for you.

Note: Run your heroku app with webprocess

example : web: python main.py

Then go on https://uptimerobot.com/ and setup to ping your web app after every 5 minutes

# Code Example
from discord.ext import commands
from quart import Quart
import os
app = Quart(__name__)

bot = commands.Bot('!')

@bot.command()
async def something(ctx):
    ...

"""
Note: On Heroku you cant bind your webserver with 5000 port as they aren't static.

To fix above problem you will have to get dynamic port from the environment variable and you are good to go.

"""

PORT = os.environ.get('PORT')

bot.loop.create_task(app.run_task('0.0.0.0', PORT))

bot.run('Token')
Assimilation answered 26/8, 2020 at 8:5 Comment(0)
G
0

Or you can use the Terminal Multiplexer, tmux to run them independently!. If you are running on a Linux platform, tmux python3 flaskapp.py would run the flask app, while you can independently run the discord bot.

Gmur answered 23/11, 2018 at 18:43 Comment(1)
You mean running both discord.py and flask seperately in different files. I don't think thats what he meant. I think he wants to run discordpy in flask so he can get data from discordpyWorkaday
M
0

You can use asyncio with flask to perform this

import discord
import asyncio

class PrintDiscordChannelsClient(discord.Client):
    def __init__(self, *args, **kwargs):
        self.guild_id = kwargs.pop('guild_id')

        super().__init__(*args, **kwargs)

    async def on_ready(self):
        try:
            await self.wait_until_ready()
            guild = self.get_guild(int(self.guild_id))
            print(f'{guild.name} is ready!')
            channels = guild.text_channels
            print(f'{(channels)} channels found')
            await self.close()
        except Exception as e:
            print(e)
            await self.close()

async def print_discord_channels(guild_id):
    client = PrintDiscordChannelsClient(guild_id=guild_id)
    await client.start('DISCORD_BOT_TOKEN')

@application.route("/discord-api", methods=["GET"])
def discord_api():
    guild_id = request.args.get('guild_id')
    asyncio.run(print_discord_channels(guild_id=guild_id))
    return make_response("Success", 200)
Mallee answered 16/4, 2022 at 21:43 Comment(1)
Your answer could be improved with additional supporting information. Please edit to add further details, such as citations or documentation, so that others can confirm that your answer is correct. You can find more information on how to write good answers in the help center.Radiocarbon
L
0

You can run them in separate terminals by doing this:

if __name__ == "__main__":
   client.run(TOKEN)
Libbi answered 29/3, 2024 at 18:11 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.