core/services/stat/reacted.py

124 lines
4.4 KiB
Python
Raw Normal View History

2022-08-11 09:09:57 +00:00
import asyncio
from datetime import datetime
from sqlalchemy.types import Enum
2022-08-11 09:09:57 +00:00
from sqlalchemy import Column, DateTime, ForeignKey, Integer
2022-08-13 16:19:16 +00:00
# from sqlalchemy.orm.attributes import flag_modified
2022-08-11 09:09:57 +00:00
from base.orm import Base, local_session
2022-08-13 16:19:16 +00:00
from orm.reaction import Reaction, ReactionKind, kind_to_rate
from orm.topic import ShoutTopic
2022-08-11 09:09:57 +00:00
class ReactedByDay(Base):
__tablename__ = "reacted_by_day"
id = None
reaction = Column(ForeignKey("reaction.id"), primary_key = True)
shout = Column(ForeignKey('shout.slug'), primary_key=True)
2022-08-14 03:07:11 +00:00
replyTo = Column(ForeignKey('reaction.id'), nullable=True)
kind: int = Column(Enum(ReactionKind), nullable=False, comment="Reaction kind")
2022-08-11 09:09:57 +00:00
day = Column(DateTime, primary_key=True, default=datetime.now)
class ReactedStorage:
reacted = {
'shouts': {},
'topics': {},
'reactions': {}
2022-08-11 09:09:57 +00:00
}
rating = {
'shouts': {},
'topics': {},
'reactions': {}
}
reactions = []
2022-08-11 09:09:57 +00:00
to_flush = []
period = 30*60 # sec
lock = asyncio.Lock()
@staticmethod
2022-08-13 10:05:46 +00:00
def init(session):
2022-08-11 09:09:57 +00:00
self = ReactedStorage
all_reactions = session.query(ReactedByDay).all()
2022-08-13 16:19:16 +00:00
all_reactions2 = session.query(Reaction).filter(Reaction.deletedAt == None).all()
print('[stat.reacted] %d reactions total' % len(all_reactions or all_reactions2))
rrr = (all_reactions or all_reactions2)
create = False
if not all_reactions: create = True
for reaction in rrr:
2022-08-11 09:09:57 +00:00
shout = reaction.shout
topics = session.query(ShoutTopic.topic).where(ShoutTopic.shout == shout).all()
kind = reaction.kind
2022-08-13 16:19:16 +00:00
self.reacted['shouts'][shout] = self.reacted['shouts'].get(shout, [])
self.reacted['shouts'][shout].append(reaction)
self.rating['shouts'][shout] = self.rating['shouts'].get(shout, 0) + kind_to_rate(kind)
for t in topics:
2022-08-13 16:19:16 +00:00
self.reacted['topics'][t] = self.reacted['topics'].get(t, [])
self.reacted['topics'][t].append(reaction)
self.rating['topics'][t] = self.rating['topics'].get(t, 0) + kind_to_rate(kind) # rating
2022-08-13 16:19:16 +00:00
if reaction.replyTo:
self.reacted['reactions'][reaction.replyTo] = self.reacted['reactions'].get(reaction.replyTo, [])
self.reacted['reactions'][reaction.replyTo].append(reaction)
self.rating['reactions'][reaction.replyTo] = self.rating['reactions'].get(reaction.replyTo, 0) + kind_to_rate(reaction.kind)
ttt = self.reacted['topics'].values()
print('[stat.reacted] %d topics reacted' % len(ttt))
print('[stat.reacted] %d shouts reacted' % len(self.reacted['shouts']))
print('[stat.reacted] %d reactions reacted' % len(self.reacted['reactions']))
2022-08-13 16:19:16 +00:00
if len(all_reactions) == 0 and len(all_reactions2) != 0:
with local_session() as session:
for r in all_reactions2:
session.add(ReactedByDay(reaction=r.id, shout=r.shout, reply=r.replyTo, kind=r.kind, day=r.createdAt.replace(hour=0, minute=0, second=0)))
session.commit()
2022-08-11 09:09:57 +00:00
@staticmethod
async def get_shout(shout_slug):
self = ReactedStorage
async with self.lock:
2022-08-13 16:19:16 +00:00
return self.reacted['shouts'].get(shout_slug, [])
2022-08-11 09:09:57 +00:00
@staticmethod
async def get_topic(topic_slug):
2022-08-11 09:09:57 +00:00
self = ReactedStorage
async with self.lock:
2022-08-13 16:19:16 +00:00
return self.reacted['topics'].get(topic_slug, [])
2022-08-11 09:09:57 +00:00
@staticmethod
2022-08-13 16:19:16 +00:00
async def get_reaction(reaction_id):
2022-08-11 09:09:57 +00:00
self = ReactedStorage
async with self.lock:
2022-08-13 16:19:16 +00:00
return self.reacted['reactions'].get(reaction_id, [])
2022-08-11 09:09:57 +00:00
@staticmethod
2022-08-13 16:19:16 +00:00
async def get_rating(shout_slug):
2022-08-11 09:09:57 +00:00
self = ReactedStorage
async with self.lock:
2022-08-13 16:19:16 +00:00
return self.reacted['shouts'].get(shout_slug, 0)
2022-08-11 09:09:57 +00:00
@staticmethod
2022-08-13 16:19:16 +00:00
async def get_topic_rating(topic_slug):
2022-08-11 09:09:57 +00:00
self = ReactedStorage
async with self.lock:
2022-08-13 16:19:16 +00:00
return self.rating['topics'].get(topic_slug, 0)
@staticmethod
async def get_reaction_rating(reaction_id):
self = ReactedStorage
async with self.lock:
return self.rating['reactions'].get(reaction_id, 0)
@staticmethod
async def increment(shout_slug, kind, reply_id = None):
self = ReactedStorage
reaction: ReactedByDay = None
async with self.lock:
2022-08-13 10:05:46 +00:00
with local_session() as session:
reaction = ReactedByDay.create(shout=shout_slug, kind=kind, reply=reply_id)
self.reacted['shouts'][shout_slug] = self.reacted['shouts'].get(shout_slug, [])
self.reacted['shouts'][shout_slug].append(reaction)
if reply_id:
self.reacted['reaction'][reply_id] = self.reacted['reactions'].get(shout_slug, [])
2022-08-13 16:19:16 +00:00
self.reacted['reaction'][reply_id].append(reaction)
self.rating['reactions'][reply_id] = self.rating['reactions'].get(reply_id, 0) + kind_to_rate(kind)
else:
self.rating['shouts'][shout_slug] = self.rating['shouts'].get(shout_slug, 0) + kind_to_rate(kind)