Browse Source

Remove builtin support for storage

develop
andrei 8 years ago
parent
commit
3b90db2c2f
  1. 11
      disco/bot/bot.py
  2. 1
      disco/bot/plugin.py
  3. 89
      disco/bot/storage.py
  4. 21
      examples/basic_plugin.py
  5. 33
      examples/storage.py
  6. 1
      tests/imports.py

11
disco/bot/bot.py

@ -12,7 +12,6 @@ from gevent.wsgi import WSGIServer
from disco.types.guild import GuildMember
from disco.bot.plugin import Plugin
from disco.bot.command import CommandEvent, CommandLevels
from disco.bot.storage import Storage
from disco.util.config import Config
from disco.util.logging import LoggingClass
from disco.util.serializer import Serializer
@ -94,11 +93,6 @@ class BotConfig(Config):
plugin_config_format = 'json'
plugin_config_dir = 'config'
storage_enabled = True
storage_fsync = True
storage_serializer = 'json'
storage_path = 'storage.json'
http_enabled = False
http_host = '0.0.0.0'
http_port = 7575
@ -136,11 +130,6 @@ class Bot(LoggingClass):
# The context carries information about events in a threadlocal storage
self.ctx = ThreadLocal()
# The storage object acts as a dynamic contextual aware store
self.storage = None
if self.config.storage_enabled:
self.storage = Storage(self.ctx, self.config.from_prefix('storage'))
# If the manhole is enabled, add this bot as a local
if self.client.config.manhole_enable:
self.client.manhole_locals['bot'] = self

1
disco/bot/plugin.py

@ -179,7 +179,6 @@ class Plugin(LoggingClass, PluginDeco):
self.client = bot.client
self.state = bot.client.state
self.ctx = bot.ctx
self.storage = bot.storage
self.config = config
# General declartions

89
disco/bot/storage.py

@ -1,89 +0,0 @@
import os
from six.moves import UserDict
from disco.util.hashmap import HashMap
from disco.util.serializer import Serializer
class StorageHashMap(HashMap):
def __init__(self, data):
self.data = data
class ContextAwareProxy(UserDict):
def __init__(self, ctx):
self.ctx = ctx
@property
def data(self):
return self.ctx()
class StorageDict(UserDict):
def __init__(self, parent, data):
self._parent = parent
self.data = data
def update(self, other):
self.data.update(other)
self._parent._update()
def __setitem__(self, key, value):
self.data[key] = value
self._parent._update()
def __delitem__(self, key):
del self.data[key]
self._parent._update()
class Storage(object):
def __init__(self, ctx, config):
self._ctx = ctx
self._path = config.path
self._serializer = config.serializer
self._fsync = config.fsync
self._data = {}
if os.path.exists(self._path):
with open(self._path, 'r') as f:
self._data = Serializer.loads(self._serializer, f.read())
if not self._data:
self._data = {}
def __getitem__(self, key):
if key not in self._data:
self._data[key] = {}
return StorageHashMap(StorageDict(self, self._data[key]))
def _update(self):
if self._fsync:
self.save()
def save(self):
if not self._path:
return
with open(self._path, 'w') as f:
f.write(Serializer.dumps(self._serializer, self._data))
def guild(self, key):
return ContextAwareProxy(
lambda: self['_g{}:{}'.format(self._ctx['guild'].id, key)],
)
def channel(self, key):
return ContextAwareProxy(
lambda: self['_c{}:{}'.format(self._ctx['channel'].id, key)],
)
def plugin(self, key):
return ContextAwareProxy(
lambda: self['_p{}:{}'.format(self._ctx['plugin'].name, key)],
)
def user(self, key):
return ContextAwareProxy(
lambda: self['_u{}:{}'.format(self._ctx['user'].id, key)],
)

21
examples/basic_plugin.py

@ -32,6 +32,13 @@ class BasicPlugin(Plugin):
requests.rate_limited_duration(),
))
@Plugin.command('safe', '<user_input:str...>')
def on_safe(self, event, user_input):
# Disco exposes a function that can sanitize user input
event.msg.reply(u'No at-everyones here! {}'.format(
S(user_input),
))
@Plugin.command('ban', '<user:snowflake> <reason:str...>')
def on_ban(self, event, user, reason):
event.guild.create_ban(user, reason=reason + u'\U0001F4BF')
@ -64,20 +71,6 @@ class BasicPlugin(Plugin):
def on_math_sub_command(self, event, a, b):
event.msg.reply('{}'.format(a - b))
@Plugin.command('tag', '<name:str> [value:str...]')
def on_tag(self, event, name, value=None):
# Plugins can easily store data locally using Disco's built in storage
tags = self.storage.guild.ensure('tags')
if value:
tags[name] = value
event.msg.reply(u':ok_hand: created tag `{}`'.format(S(name)))
else:
if name in tags:
return event.msg.reply(tags[name])
else:
return event.msg.reply(u'Unknown tag: `{}`'.format(S(name)))
@Plugin.command('test', parser=True)
@Plugin.parser.add_argument('-a', '--asdf', help='wow')
@Plugin.parser.add_argument('--help', action='store_true')

33
examples/storage.py

@ -1,33 +0,0 @@
from disco.bot import Plugin
class BasicPlugin(Plugin):
def load(self, ctx):
super(BasicPlugin, self).load(ctx)
self.tags = self.storage.guild('tags')
@Plugin.command('add', '<name:str> <value:str...>', group='tags')
def on_tags_add(self, event, name, value):
if name in self.tags:
return event.msg.reply('That tag already exists!')
self.tags[name] = value
return event.msg.reply(u':ok_hand: created the tag {}'.format(name), sanitize=True)
@Plugin.command('get', '<name:str>', group='tags')
def on_tags_get(self, event, name):
if name not in self.tags:
return event.msg.reply('That tag does not exist!')
return event.msg.reply(self.tags[name], sanitize=True)
@Plugin.command('delete', '<name:str>', group='tags', aliases=['del', 'rmv', 'remove'])
def on_tags_delete(self, event, name):
if name not in self.tags:
return event.msg.reply('That tag does not exist!')
del self.tags[name]
return event.msg.reply(u':ok_hand: I deleted the {} tag for you'.format(
name
), sanitize=True)

1
tests/imports.py

@ -9,7 +9,6 @@ from disco.bot.bot import *
from disco.bot.command import *
from disco.bot.parser import *
from disco.bot.plugin import *
from disco.bot.storage import *
from disco.gateway.client import *
from disco.gateway.events import *
from disco.gateway.ipc import *

Loading…
Cancel
Save