[errors, perdu, storage] From Pickle to JSON

This commit is contained in:
Suwako Moriya 2020-03-21 21:15:02 +01:00
parent bccf545911
commit 9beffc9e39
Signed by: SuwakoMmh
GPG Key ID: A27482B806F13CD5
4 changed files with 50 additions and 33 deletions

View File

@ -1,5 +1,4 @@
import asyncio import asyncio
import collections
import random import random
import traceback import traceback
@ -26,24 +25,24 @@ class MainClass(BaseClassPython):
def __init__(self, client): def __init__(self, client):
super().__init__(client) super().__init__(client)
self.config.init({"dev_chan": [], "memes": [""], "icon": ""}) self.config.init({"dev_chan": [], "memes": [""], "icon": ""})
self.errorsDeque = None self.errorsList = None
async def on_load(self): async def on_load(self):
if await self.objects.save_exists('errorsDeque'): if await self.objects.save_exists('errorsList'):
self.errorsDeque = await self.objects.load_object('errorsDeque') self.errorsList = await self.objects.load_object('errorsList')
else: else:
self.errorsDeque = collections.deque() self.errorsList = []
async def on_ready(self): async def on_ready(self):
for i in range(len(self.errorsDeque)): for i in range(len(self.errorsList)):
try: try:
msg_id = self.errorsDeque.popleft() msg_id = self.errorsList.pop(0)
channel = self.client.get_channel(msg_id["channel_id"]) channel = self.client.get_channel(msg_id["channel_id"])
to_delete = await channel.fetch_message(msg_id["msg_id"]) to_delete = await channel.fetch_message(msg_id["msg_id"])
await to_delete.delete() await to_delete.delete()
except: except:
raise raise
await self.objects.save_object('errorsDeque', self.errorsDeque) await self.objects.save_object('errorsList', self.errorsList)
async def command(self, message, args, kwargs): async def command(self, message, args, kwargs):
raise Exception("KERNEL PANIC!!!") raise Exception("KERNEL PANIC!!!")
@ -86,9 +85,9 @@ class MainClass(BaseClassPython):
message = await channel.send(embed=embed.set_footer(text="Ce message va s'autodétruire dans une minute", message = await channel.send(embed=embed.set_footer(text="Ce message va s'autodétruire dans une minute",
icon_url=self.config.icon)) icon_url=self.config.icon))
msg_id = {"channel_id": message.channel.id, "msg_id": message.id} msg_id = {"channel_id": message.channel.id, "msg_id": message.id}
self.errorsDeque.append(msg_id) self.errorsList.append(msg_id)
# Save message in errorsDeque now to keep them if a reboot happend during next 60 seconds # Save message in errorsList now to keep them if a reboot happend during next 60 seconds
self.objects.save_object('errorsDeque', self.errorsDeque) await self.objects.save_object('errorsList', self.errorsList)
# Wait 60 seconds and delete message # Wait 60 seconds and delete message
await asyncio.sleep(60) await asyncio.sleep(60)
@ -100,8 +99,8 @@ class MainClass(BaseClassPython):
raise raise
finally: finally:
try: try:
self.errorsDeque.remove(msg_id) self.errorsList.remove(msg_id)
except ValueError: except ValueError:
pass pass
# Save now to avoid deleting unkown message # Save now to avoid deleting unkown message
self.objects.save_object('errorsDeque', self.errorsDeque) await self.objects.save_object('errorsList', self.errorsList)

View File

@ -23,11 +23,11 @@ class MainClass(BaseClassPython):
def __init__(self, client): def __init__(self, client):
super().__init__(client) super().__init__(client)
self.config.init({"channel": 0, "lost_role": 0, "min_delta": datetime.timedelta(minutes=30).total_seconds()}) self.config.init({"channel": 0, "lost_role": 0, "min_delta": datetime.timedelta(minutes=26).total_seconds()})
async def on_load(self): async def on_load(self):
if await self.objects.save_exists('history'): if await self.objects.save_exists('history'):
self.history = self.objects.load_object('history') self.history = await self.objects.load_object('history')
else: else:
self.history = {} self.history = {}
@ -44,7 +44,7 @@ class MainClass(BaseClassPython):
delta = message.created_at - self.history[message.author.id][-1][0] delta = message.created_at - self.history[message.author.id][-1][0]
if delta.total_seconds() >= self.config.min_delta: if delta.total_seconds() >= self.config.min_delta:
self.history[message.author.id].append((message.created_at, delta)) self.history[message.author.id].append((message.created_at, delta))
self.objects.save_object("history", self.history) await self.objects.save_object("history", self.history)
await self.parse_command(message) await self.parse_command(message)
async def fill_history(self): async def fill_history(self):
@ -58,7 +58,7 @@ class MainClass(BaseClassPython):
delta = self.history[message.author.id][-1][0] - message.created_at delta = self.history[message.author.id][-1][0] - message.created_at
if delta.total_seconds() >= self.config.min_delta: if delta.total_seconds() >= self.config.min_delta:
self.history[message.author.id].append((message.created_at, delta)) self.history[message.author.id].append((message.created_at, delta))
self.objects.save_object("history", self.history) await self.objects.save_object("history", self.history)
def get_top(self, top=10, since=datetime.datetime(year=1, month=1, day=1)): def get_top(self, top=10, since=datetime.datetime(year=1, month=1, day=1)):
"""Return [(userid, [(date, delta), (date,delta), ...]), ... ]""" """Return [(userid, [(date, delta), (date,delta), ...]), ... ]"""

View File

@ -1,5 +1,5 @@
import pickle import json
import storage.jsonenc
from storage import path as pth from storage import path as pth
@ -125,19 +125,17 @@ class Objects:
self.storage = storage self.storage = storage
self.storage.makedirs("objects", exist_ok=True) self.storage.makedirs("objects", exist_ok=True)
def save_object(self, object_name, object_instance): async def save_object(self, object_name, object_instance):
"""Save object into pickle file""" """Save object into json file"""
with self.storage.open(pth.join("objects", object_name), "wb") as f: with self.storage.open(pth.join("objects", object_name + ".json"), "w") as f:
pickler = pickle.Pickler(f) json.dump([object_instance], f, cls=storage.jsonenc.Encoder)
pickler.dump(object_instance)
def load_object(self, object_name): async def load_object(self, object_name):
"""Load object from pickle file""" """Load object from json file"""
if self.save_exists(object_name): if await self.save_exists(object_name):
with self.storage.open(pth.join("objects", object_name), "rb") as f: with self.storage.open(pth.join("objects", object_name + ".json"), "r") as f:
unpickler = pickle.Unpickler(f) return json.load(f, object_hook=storage.jsonenc.hook)[0]
return unpickler.load()
def save_exists(self, object_name): async def save_exists(self, object_name):
"""Check if pickle file exists""" """Check if json file exists"""
return self.storage.exists(pth.join("objects", object_name)) return self.storage.exists(pth.join("objects", object_name + ".json"))

20
storage/jsonenc.py Normal file
View File

@ -0,0 +1,20 @@
import json
import datetime
class Encoder(json.JSONEncoder):
def default(self, obj):
if isinstance(obj, (datetime.datetime)):
return {'data_type':'datetime.datetime', 'iso':obj.isoformat()}
if isinstance(obj, (datetime.timedelta)):
return {'data_type':'datetime.timedelta', 'totalseconds':obj.total_seconds()}
return json.JSONEncoder.default(self, obj)
def hook(dct):
if 'data_type' in dct:
if dct['data_type'] == "datetime.datetime":
return datetime.datetime.fromisoformat(dct['iso'])
elif dct['data_type'] == "datetime.timedelta":
return datetime.timedelta(seconds=dct['totalseconds'])
return dct