From 0bb78cae74320ba77446df23292a9e7f93efcb4d Mon Sep 17 00:00:00 2001
From: Moul <moul@moul.re>
Date: Sun, 8 Dec 2019 22:19:05 +0200
Subject: [PATCH] =?UTF-8?q?[feat]=20#262:=20Add=20'verify'=20command=20to?=
 =?UTF-8?q?=20check=20blocks=E2=80=99=20signatures?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

- Based on DuniterPy 0.56.0 which implements block signature
  verification
- #183: Click progress bar
- Handle BMA anti-spam protection whith an exception
- Possibility to pass starting and ending block numbers to verify them
- Default to 0,0(head) for the full blockchain
- Get rid of the client singleton, in order to have full control over it
---
 silkaj/blocks.py    | 115 ++++++++++++++++++++++++++++++++++++++++++++
 silkaj/cli.py       |   2 +
 silkaj/constants.py |   1 +
 3 files changed, 118 insertions(+)
 create mode 100644 silkaj/blocks.py

diff --git a/silkaj/blocks.py b/silkaj/blocks.py
new file mode 100644
index 00000000..33e14abe
--- /dev/null
+++ b/silkaj/blocks.py
@@ -0,0 +1,115 @@
+"""
+Copyright  2016-2020 Maël Azimi <m.a@moul.re>
+
+Silkaj is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+Silkaj is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with Silkaj. If not, see <https://www.gnu.org/licenses/>.
+"""
+
+import logging
+from asyncio import sleep
+from click import command, argument, INT, progressbar
+from aiohttp.client_exceptions import ServerDisconnectedError
+
+from duniterpy.api import bma
+from duniterpy.api.client import Client
+from duniterpy.api.errors import DuniterError
+from duniterpy.documents import Block
+from duniterpy.key.verifying_key import VerifyingKey
+
+from silkaj.tools import message_exit, coroutine
+from silkaj.network_tools import EndPoint
+from silkaj.constants import BMA_MAX_BLOCKS_CHUNK_SIZE
+
+
+@command(
+    "verify",
+    help="Verify blocks’ signatures. \
+If only FROM_BLOCK is specified, it verifies from this block to the last block. \
+If nothing specified, the whole blockchain gets verified.",
+)
+@argument("from_block", default=0, type=INT)
+@argument("to_block", default=0, type=INT)
+@coroutine
+async def verify_blocks_signatures(from_block, to_block):
+    client = Client(EndPoint().BMA_ENDPOINT)
+    to_block = await check_passed_blocks_range(client, from_block, to_block)
+    invalid_blocks_signatures = list()
+    chunks_from = range(from_block, to_block + 1, BMA_MAX_BLOCKS_CHUNK_SIZE)
+    with progressbar(chunks_from, label="Processing blocks verification") as bar:
+        for chunk_from in bar:
+            chunk_size = get_chunk_size(from_block, to_block, chunks_from, chunk_from)
+            logging.info(
+                "Processing chunk from block {} to {}".format(
+                    chunk_from, chunk_from + chunk_size
+                )
+            )
+            chunk = await get_chunk(client, chunk_size, chunk_from)
+
+            for block in chunk:
+                block = Block.from_signed_raw(block["raw"] + block["signature"] + "\n")
+                verify_block_signature(invalid_blocks_signatures, block)
+
+    await client.close()
+    display_result(from_block, to_block, invalid_blocks_signatures)
+
+
+async def check_passed_blocks_range(client, from_block, to_block):
+    head_number = (await client(bma.blockchain.current))["number"]
+    if to_block == 0:
+        to_block = head_number
+    if to_block > head_number:
+        await client.close()
+        message_exit(
+            "Passed TO_BLOCK argument is bigger than the head block: "
+            + str(head_number)
+        )
+    if from_block > to_block:
+        await client.close()
+        message_exit("TO_BLOCK should be bigger or equal to FROM_BLOCK")
+    return to_block
+
+
+def get_chunk_size(from_block, to_block, chunks_from, chunk_from):
+    """If not last chunk, take the maximum size
+    Otherwise, calculate the size for the last chunk"""
+    if chunk_from != chunks_from[-1]:
+        return BMA_MAX_BLOCKS_CHUNK_SIZE
+    else:
+        return (to_block + 1 - from_block) % BMA_MAX_BLOCKS_CHUNK_SIZE
+
+
+async def get_chunk(client, chunk_size, chunk_from):
+    try:
+        return await client(bma.blockchain.blocks, chunk_size, chunk_from)
+    except ServerDisconnectedError:
+        logging.info("Reach BMA anti-spam protection. Waiting two seconds")
+        await sleep(2)
+        return await client(bma.blockchain.blocks, chunk_size, chunk_from)
+    except DuniterError as error:
+        logging.error(error)
+
+
+def verify_block_signature(invalid_blocks_signatures, block):
+    key = VerifyingKey(block.issuer)
+    if not key.verify_document(block):
+        invalid_blocks_signatures.append(block.number)
+
+
+def display_result(from_block, to_block, invalid_blocks_signatures):
+    result = "Within {0}-{1} range, ".format(from_block, to_block)
+    if invalid_blocks_signatures:
+        result += "blocks with a wrong signature: "
+        result += " ".join(str(n) for n in invalid_blocks_signatures)
+    else:
+        result += "no blocks with a wrong signature."
+    print(result)
diff --git a/silkaj/cli.py b/silkaj/cli.py
index 09c81a50..1e78ff03 100644
--- a/silkaj/cli.py
+++ b/silkaj/cli.py
@@ -33,6 +33,7 @@ from silkaj.commands import (
 from silkaj.wot import received_sent_certifications, id_pubkey_correspondence
 from silkaj.auth import generate_auth_file
 from silkaj.license import license_command
+from silkaj.blocks import verify_blocks_signatures
 from silkaj.constants import SILKAJ_VERSION
 
 
@@ -95,6 +96,7 @@ cli.add_command(currency_info)
 cli.add_command(license_command)
 cli.add_command(network_info)
 cli.add_command(send_transaction)
+cli.add_command(verify_blocks_signatures)
 cli.add_command(received_sent_certifications)
 
 
diff --git a/silkaj/constants.py b/silkaj/constants.py
index ae35b2ca..17185490 100644
--- a/silkaj/constants.py
+++ b/silkaj/constants.py
@@ -25,3 +25,4 @@ ASYNC_SLEEP = 0.1
 SOURCES_PER_TX = 40
 SUCCESS_EXIT_STATUS = 0
 FAILURE_EXIT_STATUS = 1
+BMA_MAX_BLOCKS_CHUNK_SIZE = 5000
-- 
GitLab