From c936c603ca9e17591fe79936ca9714d5c05d78b2 Mon Sep 17 00:00:00 2001
From: mtrela <mtrela@syncad.com>
Date: Thu, 10 Sep 2020 10:45:50 +0200
Subject: [PATCH] Refactoring + `list_votes` sql function changed into
 `list_votes_by_voter_comment`/`list_votes_by_comment_voter`

---
 hive/db/db_state.py                  |   4 +-
 hive/db/schema.py                    | 178 ++++++++++++++++-----------
 hive/server/bridge_api/methods.py    |  10 +-
 hive/server/bridge_api/objects.py    |   4 +-
 hive/server/bridge_api/thread.py     |   6 +-
 hive/server/condenser_api/methods.py |  12 +-
 hive/server/condenser_api/objects.py |   6 +-
 hive/server/database_api/methods.py  |  18 ++-
 hive/server/tags_api/methods.py      |   2 +-
 9 files changed, 140 insertions(+), 100 deletions(-)

diff --git a/hive/db/db_state.py b/hive/db/db_state.py
index 1d3d56937..7c698c656 100644
--- a/hive/db/db_state.py
+++ b/hive/db/db_state.py
@@ -114,8 +114,8 @@ class DbState:
 
             'hive_post_tags_tag_id_idx',
 
-            'hive_votes_voter_id_permlink_id_idx',
-            'hive_votes_permlink_id_voter_id_idx'
+            'hive_votes_voter_id_post_id_idx',
+            'hive_votes_post_id_voter_id_idx'
         ]
 
         to_return = []
diff --git a/hive/db/schema.py b/hive/db/schema.py
index 302c788ee..205c7f8a9 100644
--- a/hive/db/schema.py
+++ b/hive/db/schema.py
@@ -211,8 +211,8 @@ def build_metadata():
 
         sa.Index('hive_votes_post_id_idx', 'post_id'),
         sa.Index('hive_votes_voter_id_idx', 'voter_id'),
-        sa.Index('hive_votes_voter_id_permlink_id_idx', 'voter_id', 'permlink_id'),
-        sa.Index('hive_votes_permlink_id_voter_id_idx', 'permlink_id', 'voter_id'),
+        sa.Index('hive_votes_voter_id_post_id_idx', 'voter_id', 'post_id'),
+        sa.Index('hive_votes_post_id_voter_id_idx', 'post_id', 'voter_id'),
         sa.Index('hive_votes_block_num_idx', 'block_num')
     )
 
@@ -844,9 +844,9 @@ def setup(db):
     db.query_no_return(sql)
 
     sql = """
-        DROP VIEW IF EXISTS hive_votes_accounts_permlinks_view
+        DROP VIEW IF EXISTS hive_votes_view
         ;
-        CREATE OR REPLACE VIEW hive_votes_accounts_permlinks_view
+        CREATE OR REPLACE VIEW hive_votes_view
         AS
         SELECT
             hv.voter_id as voter_id,
@@ -872,33 +872,35 @@ def setup(db):
     db.query_no_return(sql)
 
     sql = """
-        DROP FUNCTION IF EXISTS list_votes( character varying, character varying, character varying, int, bool );
-
-        CREATE OR REPLACE FUNCTION public.list_votes
+        DROP TYPE IF EXISTS database_api_vote CASCADE;
+
+        CREATE TYPE database_api_vote AS (
+          voter VARCHAR(16),
+          author VARCHAR(16),
+          permlink VARCHAR(255),
+          weight NUMERIC,
+          rshares BIGINT,
+          percent INT,
+          last_update TIMESTAMP,
+          num_changes INT,
+          reputation FLOAT4
+        );
+
+        DROP FUNCTION IF EXISTS list_votes_by_voter_comment( character varying, character varying, character varying, int );
+
+        CREATE OR REPLACE FUNCTION public.list_votes_by_voter_comment
         (
           in _VOTER hive_accounts.name%TYPE,
           in _AUTHOR hive_accounts.name%TYPE,
           in _PERMLINK hive_permlink_data.permlink%TYPE,
-          in _LIMIT INT,
-          in _IS_VOTER_COMMENT_SORT BOOLEAN
-        )
-        RETURNS TABLE
-        (
-          voter hive_accounts.name%TYPE,
-          author hive_accounts.name%TYPE,
-          permlink hive_permlink_data.permlink%TYPE,
-          weight hive_votes.weight%TYPE,
-          rshares hive_votes.rshares%TYPE,
-          percent hive_votes.vote_percent%TYPE,
-          last_update hive_votes.last_update%TYPE,
-          num_changes hive_votes.num_changes%TYPE,
-          reputation hive_accounts.reputation%TYPE
+          in _LIMIT INT
         )
+        RETURNS SETOF database_api_vote
         LANGUAGE 'plpgsql'
-        VOLATILE
-        AS $BODY$
+        AS
+        $function$
         DECLARE _VOTER_ID INT;
-        DECLARE _PERMLINK_ID INT;
+        DECLARE _POST_ID INT;
         BEGIN
 
         IF _VOTER = '' THEN
@@ -911,60 +913,90 @@ def setup(db):
                       );
         END IF;
 
-        _PERMLINK_ID = find_comment_id( _AUTHOR, _PERMLINK, True);
+        _POST_ID = find_comment_id( _AUTHOR, _PERMLINK, True);
 
-        IF _IS_VOTER_COMMENT_SORT = True THEN
-          RETURN QUERY
-          (
-                  SELECT
-                      v.voter,
-                      v.author,
-                      v.permlink,
-                      v.weight,
-                      v.rshares,
-                      v.percent,
-                      v.last_update,
-                      v.num_changes,
-                      v.reputation
-                  FROM
-                      hive_votes_accounts_permlinks_view v
-                      WHERE
-                          ( v.voter_id = _VOTER_ID and v.permlink_id >= _PERMLINK_ID )
-                          OR
-                          ( v.voter_id > _VOTER_ID )
-                      ORDER BY
-                        voter_id,
-                        permlink_id
-                  LIMIT _LIMIT
-          );
+        RETURN QUERY
+        (
+                SELECT
+                    v.voter,
+                    v.author,
+                    v.permlink,
+                    v.weight,
+                    v.rshares,
+                    v.percent,
+                    v.last_update,
+                    v.num_changes,
+                    v.reputation
+                FROM
+                    hive_votes_view v
+                    WHERE
+                        ( v.voter_id = _VOTER_ID and v.post_id >= _POST_ID )
+                        OR
+                        ( v.voter_id > _VOTER_ID )
+                    ORDER BY
+                      voter_id,
+                      post_id
+                LIMIT _LIMIT
+        );
+
+        END
+        $function$;
+
+        DROP FUNCTION IF EXISTS list_votes_by_comment_voter( character varying, character varying, character varying, int );
+
+        CREATE OR REPLACE FUNCTION public.list_votes_by_comment_voter
+        (
+          in _VOTER hive_accounts.name%TYPE,
+          in _AUTHOR hive_accounts.name%TYPE,
+          in _PERMLINK hive_permlink_data.permlink%TYPE,
+          in _LIMIT INT
+        )
+        RETURNS SETOF database_api_vote
+        LANGUAGE 'plpgsql'
+        AS
+        $function$
+        DECLARE _VOTER_ID INT;
+        DECLARE _POST_ID INT;
+        BEGIN
+
+        IF _VOTER = '' THEN
+          _VOTER_ID = 0;
         ELSE
-          RETURN QUERY
-          (
-                  SELECT
-                      v.voter,
-                      v.author,
-                      v.permlink,
-                      v.weight,
-                      v.rshares,
-                      v.percent,
-                      v.last_update,
-                      v.num_changes,
-                      v.reputation
-                  FROM
-                      hive_votes_accounts_permlinks_view v
-                      WHERE
-                          ( v.permlink_id = _PERMLINK_ID and v.voter_id >= _VOTER_ID )
-                          OR
-                          ( v.permlink_id > _PERMLINK_ID )
-                      ORDER BY
-                        permlink_id,
-                        voter_id
-                  LIMIT _LIMIT
-          );
+          _VOTER_ID =
+                      (
+                        SELECT id FROM hive_accounts
+                        WHERE name=_VOTER
+                      );
         END IF;
 
+        _POST_ID = find_comment_id( _AUTHOR, _PERMLINK, True);
+
+        RETURN QUERY
+        (
+                SELECT
+                    v.voter,
+                    v.author,
+                    v.permlink,
+                    v.weight,
+                    v.rshares,
+                    v.percent,
+                    v.last_update,
+                    v.num_changes,
+                    v.reputation
+                FROM
+                    hive_votes_view v
+                    WHERE
+                        ( v.post_id = _POST_ID and v.voter_id >= _VOTER_ID )
+                        OR
+                        ( v.post_id > _POST_ID )
+                    ORDER BY
+                      post_id,
+                      voter_id
+                LIMIT _LIMIT
+        );
+
         END
-        $BODY$;
+        $function$;
     """
     db.query_no_return(sql)
 
diff --git a/hive/server/bridge_api/methods.py b/hive/server/bridge_api/methods.py
index aa47254af..970629487 100644
--- a/hive/server/bridge_api/methods.py
+++ b/hive/server/bridge_api/methods.py
@@ -2,7 +2,7 @@
 
 import hive.server.bridge_api.cursor as cursor
 from hive.server.bridge_api.objects import load_posts, load_posts_reblogs, load_profiles, _bridge_post_object
-from hive.server.database_api.methods import find_votes, VotesPresentation
+from hive.server.database_api.methods import find_votes_impl, VotesPresentation
 from hive.server.common.helpers import (
     return_error_info,
     valid_account,
@@ -109,7 +109,7 @@ async def get_post(context, author, permlink, observer=None):
     result = await db.query_all(sql, author=author, permlink=permlink)
     assert len(result) == 1, 'invalid author/permlink or post not found in cache'
     post = _bridge_post_object(result[0])
-    post['active_votes'] = await find_votes({'db':db}, author, permlink, VotesPresentation.BridgeApi)
+    post['active_votes'] = await find_votes_impl({'db':db}, author, permlink, VotesPresentation.BridgeApi)
     post = await append_statistics_to_post(post, result[0], False, blacklists_for_user)
     return post
 
@@ -213,7 +213,7 @@ async def get_ranked_posts(context, sort, start_author='', start_permlink='',
         pinned_result = await db.query_all(pinned_sql, author=start_author, limit=limit, tag=tag, permlink=start_permlink, community_name=tag, observer=observer)
         for row in pinned_result:
             post = _bridge_post_object(row)
-            post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
+            post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
             post = await append_statistics_to_post(post, row, True, blacklists_for_user)
             limit = limit - 1
             posts.append(post)
@@ -222,7 +222,7 @@ async def get_ranked_posts(context, sort, start_author='', start_permlink='',
     sql_result = await db.query_all(sql, author=start_author, limit=limit, tag=tag, permlink=start_permlink, community_name=tag, observer=observer)
     for row in sql_result:
         post = _bridge_post_object(row)
-        post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
         post = await append_statistics_to_post(post, row, False, blacklists_for_user)
         if post['post_id'] in pinned_post_ids:
             continue
@@ -314,7 +314,7 @@ async def get_account_posts(context, sort, account, start_author='', start_perml
     sql_result = await db.query_all(sql, account=account, author=start_author, permlink=start_permlink, limit=limit)
     for row in sql_result:
         post = _bridge_post_object(row)
-        post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
         post = await append_statistics_to_post(post, row, False, blacklists_for_user)
         posts.append(post)
     return posts
diff --git a/hive/server/bridge_api/objects.py b/hive/server/bridge_api/objects.py
index e7d7e20d6..1f7ddbbcd 100644
--- a/hive/server/bridge_api/objects.py
+++ b/hive/server/bridge_api/objects.py
@@ -5,7 +5,7 @@ import ujson as json
 
 from hive.server.common.mutes import Mutes
 from hive.server.common.helpers import json_date
-from hive.server.database_api.methods import find_votes, VotesPresentation
+from hive.server.database_api.methods import find_votes_impl, VotesPresentation
 from hive.utils.normalize import sbd_amount
 from hive.indexer.votes import Votes
 
@@ -100,7 +100,7 @@ async def load_posts_keyed(db, ids, truncate_body=0):
 
         row['author_rep'] = author['reputation']
         post = _bridge_post_object(row, truncate_body=truncate_body)
-        post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.BridgeApi)
 
         post['blacklists'] = Mutes.lists(post['author'], author['reputation'])
 
diff --git a/hive/server/bridge_api/thread.py b/hive/server/bridge_api/thread.py
index af8f07e19..145534810 100644
--- a/hive/server/bridge_api/thread.py
+++ b/hive/server/bridge_api/thread.py
@@ -4,7 +4,7 @@ import logging
 
 from hive.server.bridge_api.objects import load_posts_keyed, _bridge_post_object
 from hive.server.bridge_api.methods import append_statistics_to_post
-from hive.server.database_api.methods import find_votes, VotesPresentation
+from hive.server.database_api.methods import find_votes_impl, VotesPresentation
 from hive.server.common.helpers import (
     return_error_info,
     valid_account,
@@ -95,7 +95,7 @@ async def get_discussion(context, author, permlink, observer=None):
     root_id = rows[0]['id']
     all_posts = {}
     root_post = _bridge_post_object(rows[0])
-    root_post['active_votes'] = await find_votes({'db':db}, rows[0]['author'], rows[0]['permlink'], VotesPresentation.BridgeApi)
+    root_post['active_votes'] = await find_votes_impl({'db':db}, rows[0]['author'], rows[0]['permlink'], VotesPresentation.BridgeApi)
     root_post = await append_statistics_to_post(root_post, rows[0], False, blacklists_for_user)
     root_post['replies'] = []
     all_posts[root_id] = root_post
@@ -108,7 +108,7 @@ async def get_discussion(context, author, permlink, observer=None):
             parent_to_children_id_map[parent_id] = []
         parent_to_children_id_map[parent_id].append(rows[index]['id'])
         post = _bridge_post_object(rows[index])
-        post['active_votes'] = await find_votes({'db':db}, rows[index]['author'], rows[index]['permlink'], VotesPresentation.BridgeApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, rows[index]['author'], rows[index]['permlink'], VotesPresentation.BridgeApi)
         post = await append_statistics_to_post(post, rows[index], False, blacklists_for_user)
         post['replies'] = []
         all_posts[post['post_id']] = post
diff --git a/hive/server/condenser_api/methods.py b/hive/server/condenser_api/methods.py
index 81c90efaa..bf3375c65 100644
--- a/hive/server/condenser_api/methods.py
+++ b/hive/server/condenser_api/methods.py
@@ -15,7 +15,7 @@ from hive.server.common.helpers import (
     valid_limit,
     valid_follow_type)
 from hive.server.common.mutes import Mutes
-from hive.server.database_api.methods import find_votes, VotesPresentation
+from hive.server.database_api.methods import find_votes_impl, VotesPresentation
 from hive.utils.normalize import rep_to_raw, time_string_with_t
 
 # pylint: disable=too-many-arguments,line-too-long,too-many-lines
@@ -154,7 +154,7 @@ async def get_content(context, author: str, permlink: str, observer=None):
     if result:
         result = dict(result[0])
         post = _condenser_post_object(result, 0)
-        post['active_votes'] = await find_votes(context, author, permlink, VotesPresentation.CondenserApi)
+        post['active_votes'] = await find_votes_impl(context, author, permlink, VotesPresentation.CondenserApi)
         if not observer:
             post['active_votes'] = _mute_votes(post['active_votes'], Mutes.all())
         else:
@@ -272,7 +272,7 @@ async def get_discussions_by(discussion_type, context, start_author: str = '',
     posts = []
     for row in result:
         post = _condenser_post_object(row, truncate_body)
-        post['active_votes'] = await find_votes(context, post['author'], post['permlink'])
+        post['active_votes'] = await find_votes_impl(context, post['author'], post['permlink'], VotesPresentation.DatabaseApi )
         post['active_votes'] = _mute_votes(post['active_votes'], Mutes.all())
         posts.append(post)
     #posts = await resultset_to_posts(db=db, resultset=result, truncate_body=truncate_body)
@@ -381,7 +381,7 @@ async def get_discussions_by_blog(context, tag: str = None, start_author: str =
     for row in result:
         row = dict(row)
         post = _condenser_post_object(row, truncate_body=truncate_body)
-        post['active_votes'] = await find_votes(context, post['author'], post['permlink'], votes_presentation = VotesPresentation.CondenserApi)
+        post['active_votes'] = await find_votes_impl(context, post['author'], post['permlink'], VotesPresentation.CondenserApi)
         post['active_votes'] = _mute_votes(post['active_votes'], Mutes.all())
         #posts_by_id[row['post_id']] = post
         posts_by_id.append(post)
@@ -439,7 +439,7 @@ async def get_discussions_by_comments(context, start_author: str = None, start_p
     for row in result:
         row = dict(row)
         post = _condenser_post_object(row, truncate_body=truncate_body)
-        post['active_votes'] = await find_votes(context, post['author'], post['permlink'], VotesPresentation.CondenserApi)
+        post['active_votes'] = await find_votes_impl(context, post['author'], post['permlink'], VotesPresentation.CondenserApi)
         post['active_votes'] = _mute_votes(post['active_votes'], Mutes.all())
         posts.append(post)
 
@@ -592,4 +592,4 @@ async def get_active_votes(context, author: str, permlink: str):
     valid_permlink(permlink)
     db = context['db']
 
-    return await find_votes( {'db':db}, author, permlink, VotesPresentation.ActiveVotes  )
+    return await find_votes_impl( {'db':db}, author, permlink, VotesPresentation.ActiveVotes  )
diff --git a/hive/server/condenser_api/objects.py b/hive/server/condenser_api/objects.py
index 37a040aa6..9fef75b65 100644
--- a/hive/server/condenser_api/objects.py
+++ b/hive/server/condenser_api/objects.py
@@ -6,7 +6,7 @@ import ujson as json
 from hive.utils.normalize import sbd_amount, rep_to_raw
 from hive.server.common.mutes import Mutes
 from hive.server.common.helpers import json_date
-from hive.server.database_api.methods import find_votes, VotesPresentation
+from hive.server.database_api.methods import find_votes_impl, VotesPresentation
 
 log = logging.getLogger(__name__)
 
@@ -89,7 +89,7 @@ async def load_posts_keyed(db, ids, truncate_body=0):
         row['author_rep'] = author_reps[row['author']]
         post = _condenser_post_object(row, truncate_body=truncate_body)
 
-        post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.CondenserApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.CondenserApi)
         posts_by_id[row['id']] = post
 
     return posts_by_id
@@ -139,7 +139,7 @@ async def resultset_to_posts(db, resultset, truncate_body=0):
         row = dict(row)
         row['author_rep'] = author_reps[row['author']]
         post = _condenser_post_object(row, truncate_body=truncate_body)
-        post['active_votes'] = await find_votes({'db':db}, row['author'], row['permlink'], VotesPresentation.CondenserApi)
+        post['active_votes'] = await find_votes_impl({'db':db}, row['author'], row['permlink'], VotesPresentation.CondenserApi)
         posts.append(post)
 
     return posts
diff --git a/hive/server/database_api/methods.py b/hive/server/database_api/methods.py
index 914f1b9d6..bf906e013 100644
--- a/hive/server/database_api/methods.py
+++ b/hive/server/database_api/methods.py
@@ -188,7 +188,7 @@ def api_vote_info(rows, votes_presentation):
   return ret
 
 @return_error_info
-async def find_votes(context, author: str, permlink: str, votes_presentation = VotesPresentation.DatabaseApi):
+async def find_votes_impl(context, author: str, permlink: str, votes_presentation):
     """ Returns all votes for the given post """
     valid_account(author)
     valid_permlink(permlink)
@@ -205,7 +205,7 @@ async def find_votes(context, author: str, permlink: str, votes_presentation = V
             num_changes,
             reputation
         FROM
-            hive_votes_accounts_permlinks_view
+            hive_votes_view
         WHERE
             author = :author AND permlink = :permlink
         ORDER BY 
@@ -216,7 +216,7 @@ async def find_votes(context, author: str, permlink: str, votes_presentation = V
     return api_vote_info(rows, votes_presentation)
 
 @return_error_info
-async def list_votes(context, start: list, limit: int, order: str, votes_presentation = VotesPresentation.DatabaseApi):
+async def list_votes_impl(context, start: list, limit: int, order: str, votes_presentation):
     """ Returns all votes, starting with the specified voter and/or author and permlink. """
     supported_order_list = ["by_comment_voter", "by_voter_comment"]
     assert order in supported_order_list, "Order {} is not supported".format(order)
@@ -227,10 +227,18 @@ async def list_votes(context, start: list, limit: int, order: str, votes_present
     sql=""
 
     if order == "by_voter_comment":
-        sql = "select * from list_votes( '{}', '{}', '{}', {}, true )".format( start[0], start[1], start[2], limit )
+        sql = "select * from list_votes_by_voter_comment( '{}', '{}', '{}', {} )".format( start[0], start[1], start[2], limit )
     else:
-        sql = "select * from list_votes( '{}', '{}', '{}', {}, false )".format( start[2], start[0], start[1], limit )
+        sql = "select * from list_votes_by_comment_voter( '{}', '{}', '{}', {} )".format( start[2], start[0], start[1], limit )
 
     rows = await db.query_all(sql)
 
     return api_vote_info(rows, votes_presentation)
+
+@return_error_info
+async def find_votes(context, author: str, permlink: str):
+  return await find_votes_impl( context, author, permlink, VotesPresentation.DatabaseApi)
+
+@return_error_info
+async def list_votes(context, start: list, limit: int, order: str):
+  return await list_votes_impl( context, start, limit, order, VotesPresentation.DatabaseApi)
diff --git a/hive/server/tags_api/methods.py b/hive/server/tags_api/methods.py
index 5d8570d52..ff1d14766 100644
--- a/hive/server/tags_api/methods.py
+++ b/hive/server/tags_api/methods.py
@@ -21,7 +21,7 @@ async def get_active_votes(context, author: str, permlink: str):
             hv.last_update,
             hv.num_changes
         FROM
-            hive_votes_accounts_permlinks_view hv
+            hive_votes_view hv
         WHERE hv.author = :author AND hv.permlink = :permlink
     """
     ret = []
-- 
GitLab