from sqlalchemy import and_, func, case, true, select from sqlalchemy.orm import aliased from orm.author import AuthorRating, Author from orm.reaction import Reaction, ReactionKind from orm.shout import Shout from services.auth import login_required from services.db import local_session from services.schema import mutation @mutation.field('rate_author') @login_required async def rate_author(_, info, rated_slug, value): user_id = info.context['user_id'] with local_session() as session: rated_author = session.query(Author).filter(Author.slug == rated_slug).first() rater = session.query(Author).filter(Author.slug == user_id).first() if rater and rated_author: rating: AuthorRating = ( session.query(AuthorRating) .filter( and_( AuthorRating.rater == rater.id, AuthorRating.author == rated_author.id, ) ) .first() ) if rating: rating.plus = value > 0 session.add(rating) session.commit() return {} else: try: rating = AuthorRating( rater=rater.id, author=rated_author.id, plus=value > 0 ) session.add(rating) session.commit() except Exception as err: return {'error': err} return {} def count_author_comments_rating(session, author_id) -> int: replied_alias = aliased(Reaction) replies_likes = ( session.query(replied_alias) .join(Reaction, replied_alias.id == Reaction.reply_to) .where( and_( replied_alias.created_by == author_id, replied_alias.kind == ReactionKind.COMMENT.value, ) ) .filter(replied_alias.kind == ReactionKind.LIKE.value) .count() ) or 0 replies_dislikes = ( session.query(replied_alias) .join(Reaction, replied_alias.id == Reaction.reply_to) .where( and_( replied_alias.created_by == author_id, replied_alias.kind == ReactionKind.COMMENT.value, ) ) .filter(replied_alias.kind == ReactionKind.DISLIKE.value) .count() ) or 0 return replies_likes - replies_dislikes def count_author_shouts_rating(session, author_id) -> int: shouts_likes = ( session.query(Reaction, Shout) .join(Shout, Shout.id == Reaction.shout) .filter( and_( Shout.authors.any(id=author_id), Reaction.kind == ReactionKind.LIKE.value, ) ) .count() or 0 ) shouts_dislikes = ( session.query(Reaction, Shout) .join(Shout, Shout.id == Reaction.shout) .filter( and_( Shout.authors.any(id=author_id), Reaction.kind == ReactionKind.DISLIKE.value, ) ) .count() or 0 ) return shouts_likes - shouts_dislikes def load_author_ratings(session, author: Author): comments_count = ( session.query(Reaction) .filter( and_( Reaction.created_by == author.id, Reaction.kind == ReactionKind.COMMENT.value, Reaction.deleted_at.is_(None), ) ) .count() ) likes_count = ( session.query(AuthorRating) .filter(and_(AuthorRating.author == author.id, AuthorRating.plus.is_(True))) .count() ) dislikes_count = ( session.query(AuthorRating) .filter(and_(AuthorRating.author == author.id, AuthorRating.plus.is_not(True))) .count() ) author.stat = author.stat if isinstance(author.stat, dict) else {} author.stat['rating'] = likes_count - dislikes_count author.stat['rating_shouts'] = count_author_shouts_rating(session, author.id) author.stat['rating_comments'] = count_author_comments_rating(session, author.id) author.stat['commented'] = comments_count return author def add_rating_columns(q, group_list): # old karma q = q.outerjoin(AuthorRating, AuthorRating.author == Author.id) q = q.add_columns( func.sum(case((AuthorRating.plus == true(), 1), else_=0)).label('likes_count'), func.sum(case((AuthorRating.plus != true(), 1), else_=0)).label('dislikes_count'), ) # by shouts rating shout_reaction = aliased(Reaction) shouts_rating_subq = ( select( Author.id, func.coalesce(func.sum( case( (shout_reaction.kind == ReactionKind.LIKE.value, 1), (shout_reaction.kind == ReactionKind.DISLIKE.value, -1), else_=0 ) )).label('shouts_rating') ) .select_from(shout_reaction) .outerjoin( Shout, Shout.authors.any(id=Author.id) ) .outerjoin( shout_reaction, and_( shout_reaction.reply_to.is_(None), shout_reaction.shout == Shout.id, shout_reaction.deleted_at.is_(None), ), ) .group_by(Author.id) .subquery() ) q = q.outerjoin(shouts_rating_subq, Author.id == shouts_rating_subq.c.id) q = q.add_columns(shouts_rating_subq.c.shouts_rating) group_list = [shouts_rating_subq.c.shouts_rating] # by comments replied_comment = aliased(Reaction) reaction_2 = aliased(Reaction) comments_subq = select( Author.id, func.coalesce(func.sum( case( (reaction_2.kind == ReactionKind.LIKE.value, 1), (reaction_2.kind == ReactionKind.DISLIKE.value, -1), else_=0 ) )).label('comments_rating'), ).select_from(reaction_2).outerjoin( replied_comment, and_( replied_comment.kind == ReactionKind.COMMENT.value, replied_comment.created_by == Author.id, reaction_2.kind.in_([ReactionKind.LIKE.value, ReactionKind.DISLIKE.value]), reaction_2.reply_to == replied_comment.id, reaction_2.deleted_at.is_(None) ) ).group_by(Author.id).subquery() q = q.outerjoin(comments_subq, Author.id == comments_subq.c.id) q = q.add_columns(comments_subq.c.comments_rating) group_list.extend([comments_subq.c.comments_rating]) return q, group_list