Browse Source

Add support for built-in HTTP/Flask server (#34)

* Add support for built-in HTTP/Flask server

* Don't default http_enabled to true
pull/38/head
Andrei Zbikowski 8 years ago
committed by GitHub
parent
commit
f48671d9bb
  1. 20
      disco/bot/bot.py
  2. 14
      disco/bot/plugin.py
  3. 1
      docs/installation.md
  4. 5
      examples/basic_plugin.py
  5. 1
      setup.py

20
disco/bot/bot.py

@ -1,11 +1,13 @@
import re import re
import os import os
import six import six
import gevent
import inspect import inspect
import importlib import importlib
from six.moves import reload_module from six.moves import reload_module
from holster.threadlocal import ThreadLocal from holster.threadlocal import ThreadLocal
from gevent.wsgi import WSGIServer
from disco.types.guild import GuildMember from disco.types.guild import GuildMember
from disco.bot.plugin import Plugin from disco.bot.plugin import Plugin
@ -63,6 +65,13 @@ class BotConfig(Config):
The serialization format plugin configuration files are in. The serialization format plugin configuration files are in.
plugin_config_dir : str plugin_config_dir : str
The directory plugin configuration is located within. The directory plugin configuration is located within.
http_enabled : bool
Whether to enable the built-in Flask server which allows plugins to handle
and route HTTP requests.
http_host : str
The host string for the HTTP Flask server (if enabled)
http_port : int
The port for the HTTP Flask server (if enabled)
""" """
levels = {} levels = {}
plugins = [] plugins = []
@ -90,6 +99,10 @@ class BotConfig(Config):
storage_serializer = 'json' storage_serializer = 'json'
storage_path = 'storage.json' storage_path = 'storage.json'
http_enabled = False
http_host = '0.0.0.0'
http_port = 7575
class Bot(LoggingClass): class Bot(LoggingClass):
""" """
@ -132,6 +145,13 @@ class Bot(LoggingClass):
if self.client.config.manhole_enable: if self.client.config.manhole_enable:
self.client.manhole_locals['bot'] = self self.client.manhole_locals['bot'] = self
if self.config.http_enabled:
from flask import Flask
self.log.info('Starting HTTP server bound to %s:%s', self.config.http_host, self.config.http_port)
self.http = Flask('disco')
self.http_server = WSGIServer((self.config.http_host, self.config.http_port), self.http)
self.http_server_greenlet = gevent.spawn(self.http_server.serve_forever)
self.plugins = {} self.plugins = {}
self.group_abbrev = {} self.group_abbrev = {}

14
disco/bot/plugin.py

@ -131,6 +131,17 @@ class BasePluginDeco(object):
'kwargs': kwargs, 'kwargs': kwargs,
}) })
@classmethod
def route(cls, *args, **kwargs):
"""
Adds an HTTP route.
"""
return cls.add_meta_deco({
'type': 'http.add_route',
'args': args,
'kwargs': kwargs,
})
class PluginDeco(BasePluginDeco): class PluginDeco(BasePluginDeco):
""" """
@ -227,6 +238,9 @@ class Plugin(LoggingClass, PluginDeco):
getattr(command.parser, meta['type'].split('.', 1)[-1])( getattr(command.parser, meta['type'].split('.', 1)[-1])(
*meta['args'], *meta['args'],
**meta['kwargs']) **meta['kwargs'])
elif meta['type'] == 'http.add_route':
meta['kwargs']['view_func'] = member
self.bot.http.add_url_rule(*meta['args'], **meta['kwargs'])
else: else:
raise Exception('unhandled meta type {}'.format(meta)) raise Exception('unhandled meta type {}'.format(meta))

1
docs/installation.md

@ -21,6 +21,7 @@ pip install disco[performance]
| Name | Explanation | Versions | | Name | Explanation | Versions |
|------|-------------|----------| |------|-------------|----------|
| voice | Adds functionality required to connect and use voice | Both | | voice | Adds functionality required to connect and use voice | Both |
| http | Adds a built-in HTTP server w/ Flask, allowing plugins to handle HTTP requests | Both |
| music | Adds the ability to stream and play music from various third party sites | Both | | music | Adds the ability to stream and play music from various third party sites | Both |
| performance | Adds a faster JSON parser (ujson) and an ETF encoding parser | 2.x Only | | performance | Adds a faster JSON parser (ujson) and an ETF encoding parser | 2.x Only |
| sharding | Adds a library which is required to enable auto-sharding | 2.x Only | | sharding | Adds a library which is required to enable auto-sharding | 2.x Only |

5
examples/basic_plugin.py

@ -74,3 +74,8 @@ class BasicPlugin(Plugin):
if args.help: if args.help:
return event.msg.reply(event.parser.format_help()) return event.msg.reply(event.parser.format_help())
event.msg.reply(args.asdf) event.msg.reply(args.asdf)
@Plugin.route('/test')
def on_test_route(self):
print 'WOW!'
return 'Hi!'

1
setup.py

@ -18,6 +18,7 @@ with open('README.md') as f:
extras_require = { extras_require = {
'voice': ['pynacl==1.1.2'], 'voice': ['pynacl==1.1.2'],
'http': ['flask==0.12.2'],
'music': ['youtube_dl==2017.4.26'], 'music': ['youtube_dl==2017.4.26'],
'performance': ['erlpack==0.3.2', 'ujson==1.35'], 'performance': ['erlpack==0.3.2', 'ujson==1.35'],
'sharding': ['gipc==0.6.0'], 'sharding': ['gipc==0.6.0'],

Loading…
Cancel
Save