From 7e1e9e82843b026ae7afb06a4452f10e89514487 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Wed, 15 Jan 2020 09:19:29 -0600 Subject: [PATCH 01/22] Example for v3, update cert-schema, issuing v3 w/ old MerkleProof --- .../verifiable-credential.json | 17 +++++++++++++++++ requirements.txt | 2 +- 2 files changed, 18 insertions(+), 1 deletion(-) create mode 100644 examples/data-testnet/unsigned_certificates/verifiable-credential.json diff --git a/examples/data-testnet/unsigned_certificates/verifiable-credential.json b/examples/data-testnet/unsigned_certificates/verifiable-credential.json new file mode 100644 index 00000000..d3178aea --- /dev/null +++ b/examples/data-testnet/unsigned_certificates/verifiable-credential.json @@ -0,0 +1,17 @@ +{ + "@context": [ + "https://www.w3.org/2018/credentials/v1", + "https://www.blockcerts.org/schema/3.0-alpha/context.json", + "https://www.w3.org/2018/credentials/examples/v1" + ], + "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", + "type": ["VerifiableCredential"], + "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", + "issuanceDate": "2010-01-01T19:73:24Z", + "credentialSubject": { + "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", + "alumniOf": { + "id": "did:example:c276e12ec21ebfeb1f712ebc6f1" + } + } +} \ No newline at end of file diff --git a/requirements.txt b/requirements.txt index e37c7d3d..b8ab2269 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ cert-core>=2.1.9 -cert-schema>=2.1.5 +cert-schema>=3.0.0a1 chainpoint>=0.0.2 configargparse==0.12.0 glob2==0.6 From e7051ca3c4b251bfb1c0eb891ad41ad53052e713 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Tue, 21 Jan 2020 12:33:59 -0600 Subject: [PATCH 02/22] Integrated MerkleProof2019 into v3 issuing --- .../blockchain_handlers/bitcoin/__init__.py | 12 ++- .../blockchain_handlers/ethereum/__init__.py | 7 +- cert_issuer/certificate_handlers.py | 8 +- cert_issuer/helpers.py | 18 ++++ cert_issuer/merkle_tree_generator.py | 33 +++++-- cert_issuer/models.py | 3 +- requirements.txt | 3 +- tests/test_certificate_handler.py | 92 ++++++------------- tests/test_merkle_tree_generator.py | 79 +++++++++++++--- 9 files changed, 157 insertions(+), 98 deletions(-) diff --git a/cert_issuer/blockchain_handlers/bitcoin/__init__.py b/cert_issuer/blockchain_handlers/bitcoin/__init__.py index 4280bde3..8b34a87b 100644 --- a/cert_issuer/blockchain_handlers/bitcoin/__init__.py +++ b/cert_issuer/blockchain_handlers/bitcoin/__init__.py @@ -7,7 +7,7 @@ from cert_issuer.blockchain_handlers.bitcoin.connectors import BitcoinServiceProviderConnector, MockServiceProviderConnector from cert_issuer.blockchain_handlers.bitcoin.signer import BitcoinSigner from cert_issuer.blockchain_handlers.bitcoin.transaction_handlers import BitcoinTransactionHandler -from cert_issuer.certificate_handlers import CertificateBatchHandler, CertificateV2Handler, CertificateBatchWebHandler, CertificateWebV2Handler +from cert_issuer.certificate_handlers import CertificateBatchHandler, CertificateV3Handler, CertificateBatchWebHandler, CertificateWebV3Handler from cert_issuer.merkle_tree_generator import MerkleTreeGenerator from cert_issuer.models import MockTransactionHandler from cert_issuer.signer import FileSecretManager @@ -50,12 +50,14 @@ def instantiate_blockchain_handlers(app_config, file_mode=True): if file_mode: certificate_batch_handler = CertificateBatchHandler(secret_manager=secret_manager, - certificate_handler=CertificateV2Handler(), - merkle_tree=MerkleTreeGenerator()) + certificate_handler=CertificateV3Handler(), + merkle_tree=MerkleTreeGenerator(), + config=app_config) else: certificate_batch_handler = CertificateBatchWebHandler(secret_manager=secret_manager, - certificate_handler=CertificateWebV2Handler(), - merkle_tree=MerkleTreeGenerator()) + certificate_handler=CertificateWebV3Handler(), + merkle_tree=MerkleTreeGenerator(), + config=app_config) if chain == Chain.mockchain: transaction_handler = MockTransactionHandler() connector = MockServiceProviderConnector() diff --git a/cert_issuer/blockchain_handlers/ethereum/__init__.py b/cert_issuer/blockchain_handlers/ethereum/__init__.py index c2b37ca3..6fbac259 100644 --- a/cert_issuer/blockchain_handlers/ethereum/__init__.py +++ b/cert_issuer/blockchain_handlers/ethereum/__init__.py @@ -4,7 +4,7 @@ from cert_core import BlockchainType from cert_core import Chain, UnknownChainError -from cert_issuer.certificate_handlers import CertificateBatchHandler, CertificateV2Handler +from cert_issuer.certificate_handlers import CertificateBatchHandler, CertificateV3Handler from cert_issuer.blockchain_handlers.ethereum.connectors import EthereumServiceProviderConnector from cert_issuer.blockchain_handlers.ethereum.signer import EthereumSigner from cert_issuer.blockchain_handlers.ethereum.transaction_handlers import EthereumTransactionHandler @@ -54,8 +54,9 @@ def instantiate_blockchain_handlers(app_config): chain = app_config.chain secret_manager = initialize_signer(app_config) certificate_batch_handler = CertificateBatchHandler(secret_manager=secret_manager, - certificate_handler=CertificateV2Handler(), - merkle_tree=MerkleTreeGenerator()) + certificate_handler=CertificateV3Handler(), + merkle_tree=MerkleTreeGenerator(), + config=app_config) if chain == Chain.mockchain: transaction_handler = MockTransactionHandler() # ethereum chains diff --git a/cert_issuer/certificate_handlers.py b/cert_issuer/certificate_handlers.py index c1f050d0..33f5f8fb 100644 --- a/cert_issuer/certificate_handlers.py +++ b/cert_issuer/certificate_handlers.py @@ -9,7 +9,7 @@ from cert_issuer.signer import FinalizableSigner -class CertificateV2Handler(CertificateHandler): +class CertificateV3Handler(CertificateHandler): def get_byte_array_to_issue(self, certificate_metadata): certificate_json = self._get_certificate_to_issue(certificate_metadata) normalized = normalize_jsonld(certificate_json, detect_unmapped_fields=False) @@ -32,7 +32,7 @@ def _get_certificate_to_issue(self, certificate_metadata): certificate_json = json.load(unsigned_cert_file) return certificate_json -class CertificateWebV2Handler(CertificateHandler): +class CertificateWebV3Handler(CertificateHandler): def get_byte_array_to_issue(self, certificate_json): normalized = normalize_jsonld(certificate_json, detect_unmapped_fields=False) return normalized.encode('utf-8') @@ -49,7 +49,7 @@ def add_proof(self, certificate_json, merkle_proof): class CertificateBatchWebHandler(BatchHandler): def finish_batch(self, tx_id, chain): self.proof = [] - proof_generator = self.merkle_tree.get_proof_generator(tx_id, chain) + proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.issuing_address, chain) for metadata in self.certificates_to_issue: proof = next(proof_generator) self.proof.append(self.certificate_handler.add_proof(metadata, proof)) @@ -120,7 +120,7 @@ def get_certificate_generator(self): yield data_to_issue def finish_batch(self, tx_id, chain): - proof_generator = self.merkle_tree.get_proof_generator(tx_id, chain) + proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.issuing_address, chain) for _, metadata in self.certificates_to_issue.items(): proof = next(proof_generator) self.certificate_handler.add_proof(metadata, proof) diff --git a/cert_issuer/helpers.py b/cert_issuer/helpers.py index c953e7ef..91718b4b 100644 --- a/cert_issuer/helpers.py +++ b/cert_issuer/helpers.py @@ -100,3 +100,21 @@ def to_pycoin_chain(chain): return 'BTC' else: raise UnknownChainError(chain.name) + +def tx_to_blink(chain, tx_id): + blink = 'blink:' + if chain == Chain.bitcoin_regtest: + blink += 'btc:regtest:' + elif chain == Chain.bitcoin_testnet: + blink += 'btc:testnet:' + elif chain == Chain.bitcoin_mainnet: + blink += 'btc:mainnet:' + elif chain == Chain.ethereum_ropsten: + blink += 'eth:ropsten:' + elif chain == Chain.ethereum_mainnet: + blink += 'eth:mainnet:' + elif chain == Chain.mockchain: + blink += 'mocknet:' + else: + raise UnknownChainError(chain.name) + return blink + tx_id \ No newline at end of file diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index 6d10815e..9d10e9de 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -1,8 +1,12 @@ import hashlib +import logging +from datetime import datetime from cert_core import Chain from chainpoint.chainpoint import MerkleTools from pycoin.serialize import h2b +from lds_merkle_proof_2019.merkle_proof_2019 import MerkleProof2019 +from cert_issuer import helpers def hash_byte_array(data): @@ -40,7 +44,7 @@ def get_blockchain_data(self): merkle_root = self.tree.get_merkle_root() return h2b(ensure_string(merkle_root)) - def get_proof_generator(self, tx_id, chain=Chain.bitcoin_mainnet): + def get_proof_generator(self, tx_id, issuing_address, chain=Chain.bitcoin_mainnet): """ Returns a generator (1-time iterator) of proofs in insertion order. @@ -59,16 +63,25 @@ def get_proof_generator(self, tx_id, chain=Chain.bitcoin_mainnet): dict2[key] = ensure_string(value) proof2.append(dict2) target_hash = ensure_string(self.tree.get_leaf(index)) + mp2019 = MerkleProof2019() + merkle_json = { + "path": proof2, + "merkleRoot": root, + "targetHash": target_hash, + "anchors": [ + helpers.tx_to_blink(chain, tx_id) + ] + } + logging.info('merkle_json: %s', str(merkle_json)) + + proof_value = mp2019.encode(merkle_json) merkle_proof = { - "type": ['MerkleProof2017', 'Extension'], - "merkleRoot": root, - "targetHash": target_hash, - "proof": proof2, - "anchors": [{ - "sourceId": to_source_id(tx_id, chain), - "type": chain.blockchain_type.external_display_value, - "chain": chain.external_display_value - }]} + "type": "MerkleProof2019", + "created": datetime.now().isoformat(), + "proofValue": proof_value.decode('utf8'), + "proofPurpose": "assertionMethod", + "verificationMethod": "ecdsa-koblitz-pubkey:" + issuing_address + } yield merkle_proof diff --git a/cert_issuer/models.py b/cert_issuer/models.py index ce6279a6..90505d9e 100644 --- a/cert_issuer/models.py +++ b/cert_issuer/models.py @@ -3,10 +3,11 @@ from cert_issuer.config import ESTIMATE_NUM_INPUTS class BatchHandler(object): - def __init__(self, secret_manager, certificate_handler, merkle_tree): + def __init__(self, secret_manager, certificate_handler, merkle_tree, config): self.certificate_handler = certificate_handler self.secret_manager = secret_manager self.merkle_tree = merkle_tree + self.config = config @abstractmethod def pre_batch_actions(self, config): diff --git a/requirements.txt b/requirements.txt index b8ab2269..6edf6a0c 100644 --- a/requirements.txt +++ b/requirements.txt @@ -10,4 +10,5 @@ pyld>=1.0.3 pysha3>=1.0.2 python-bitcoinlib>=0.10.1 tox>=3.0.0 -jsonschema<3.0.0 \ No newline at end of file +jsonschema<3.0.0 +lds-merkle-proof-2019>=0.0.1 \ No newline at end of file diff --git a/tests/test_certificate_handler.py b/tests/test_certificate_handler.py index 4975413c..49b19cae 100644 --- a/tests/test_certificate_handler.py +++ b/tests/test_certificate_handler.py @@ -6,63 +6,22 @@ from pycoin.serialize import b2h from mock import patch, mock_open -from cert_issuer.certificate_handlers import CertificateWebV2Handler, CertificateV2Handler, CertificateBatchHandler, CertificateHandler, CertificateBatchWebHandler +from cert_issuer.certificate_handlers import CertificateWebV3Handler, CertificateV3Handler, CertificateBatchHandler, CertificateHandler, CertificateBatchWebHandler from cert_issuer.merkle_tree_generator import MerkleTreeGenerator from cert_issuer import helpers +from cert_core import Chain from mock import ANY class TestCertificateHandler(unittest.TestCase): def _proof_helper(self, chain): proof = { - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'type': ['MerkleProof2017', 'Extension'], - 'targetHash': ANY, - 'anchors': [ - { - 'sourceId': ANY, - 'type': chain.blockchain_type.external_display_value, - 'chain': chain.external_display_value - } - ], - 'proof': [ - {'right': 'd4735e3a265e16eee03f59718b9b5d03019c07d8b6c51f90da3a666eec13ab35'}, - {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'} - ], + 'type': 'MerkleProof2019', + 'created': ANY, + 'proofValue': ANY, + 'proofPurpose': 'assertionMethod', + 'verificationMethod': ANY } - - proof_1 = { - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'type': ['MerkleProof2017', 'Extension'], - 'targetHash': ANY, - 'anchors': [ - { - 'sourceId': ANY, - 'type': chain.blockchain_type.external_display_value, - 'chain': chain.external_display_value - } - ], - 'proof': [ - {'left': '6b86b273ff34fce19d6b804eff5a3f5747ada4eaa22f1d49c01e52ddb7875b4b'}, - {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'} - ] - } - - proof_2 = { - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'type': ['MerkleProof2017', 'Extension'], - 'targetHash': ANY, - 'anchors': [ - { - 'sourceId': ANY, - 'type': chain.blockchain_type.external_display_value, - 'chain': chain.external_display_value - } - ], - 'proof': [ - {'left': '4295f72eeb1e3507b8461e240e3b8d18c1e7bd2f1122b11fc9ec40a65894031a'} - ] - } - return proof, proof_1, proof_2 + return proof def _helper_mock_call(self, *args): helper_mock = mock.MagicMock() @@ -83,10 +42,14 @@ def _get_certificate_batch_web_handler(self): certificates_to_issue['2'] = mock.Mock() certificates_to_issue['3'] = mock.Mock() + config = mock.Mock() + config.issuing_address = "123" + handler = CertificateBatchWebHandler( secret_manager=secret_manager, certificate_handler=DummyCertificateHandler(), - merkle_tree=MerkleTreeGenerator()) + merkle_tree=MerkleTreeGenerator(), + config=config) return handler, certificates_to_issue @@ -97,10 +60,14 @@ def _get_certificate_batch_handler(self): certificates_to_issue['2'] = mock.Mock() certificates_to_issue['3'] = mock.Mock() + config = mock.Mock() + config.issuing_address = "123" + handler = CertificateBatchHandler( secret_manager=secret_manager, certificate_handler=DummyCertificateHandler(), - merkle_tree=MerkleTreeGenerator()) + merkle_tree=MerkleTreeGenerator(), + config=config) return handler, certificates_to_issue @@ -126,8 +93,8 @@ def test_batch_web_handler_finish_batch(self): certificate_batch_handler.set_certificates_in_batch(certificates_to_issue) result = certificate_batch_handler.prepare_batch() - chain = mock.Mock() - proof, proof_1, proof_2 = self._proof_helper(chain) + chain = Chain.bitcoin_mainnet + proof = self._proof_helper(chain) with patch.object(DummyCertificateHandler, 'add_proof', return_value= {"cert": "cert"} ) as mock_method: result = certificate_batch_handler.finish_batch( @@ -135,8 +102,6 @@ def test_batch_web_handler_finish_batch(self): ) self.assertEqual(certificate_batch_handler.proof, [{'cert': 'cert'}, {'cert': 'cert'}, {'cert': 'cert'}]) mock_method.assert_any_call(ANY, proof) - mock_method.assert_any_call(ANY, proof_1) - mock_method.assert_any_call(ANY, proof_2) def test_batch_handler_finish_batch(self): certificate_batch_handler, certificates_to_issue = self._get_certificate_batch_handler() @@ -144,8 +109,11 @@ def test_batch_handler_finish_batch(self): certificate_batch_handler.set_certificates_in_batch(certificates_to_issue) result = certificate_batch_handler.prepare_batch() - chain = mock.Mock() - proof, proof_1, proof_2 = self._proof_helper(chain) + chain = Chain.bitcoin_mainnet + proof = self._proof_helper(chain) + + config = mock.Mock() + config.issuing_address = "123" with patch.object(DummyCertificateHandler, 'add_proof') as mock_method: result = certificate_batch_handler.finish_batch( @@ -153,8 +121,6 @@ def test_batch_handler_finish_batch(self): ) mock_method.assert_any_call(ANY, proof) - mock_method.assert_any_call(ANY, proof_1) - mock_method.assert_any_call(ANY, proof_2) def test_pre_batch_actions(self): self.directory_count = 1 @@ -196,7 +162,7 @@ def test_pre_batch_actions_empty_directories(self): @mock.patch("builtins.open", create=True) def test_add_proof(self,mock_open): - handler = CertificateV2Handler() + handler = CertificateV3Handler() cert_to_issue = {'kek':'kek'} proof = {'a': 'merkel'} @@ -207,7 +173,7 @@ def test_add_proof(self,mock_open): metadata.blockchain_cert_file_name = 'file_path.nfo' with patch.object( - CertificateV2Handler, '_get_certificate_to_issue', return_value=cert_to_issue) as mock_method: + CertificateV3Handler, '_get_certificate_to_issue', return_value=cert_to_issue) as mock_method: handler.add_proof(metadata, proof) mock_open.assert_any_call('file_path.nfo','w') @@ -216,7 +182,7 @@ def test_add_proof(self,mock_open): assert file_call in call_strings def test_web_add_proof(self): - handler = CertificateWebV2Handler() + handler = CertificateWebV3Handler() proof = {'a': 'merkel'} chain = mock.Mock() certificate_json = {'kek': 'kek'} @@ -226,6 +192,8 @@ def test_web_add_proof(self): class DummyCertificateHandler(CertificateHandler): def __init__(self): + self.config = mock.Mock() + self.config.issuing_address = "123" self.counter = 0 def validate_certificate(self, certificate_metadata): diff --git a/tests/test_merkle_tree_generator.py b/tests/test_merkle_tree_generator.py index 20bd788d..1438f29c 100644 --- a/tests/test_merkle_tree_generator.py +++ b/tests/test_merkle_tree_generator.py @@ -4,6 +4,8 @@ from pycoin.serialize import b2h from cert_issuer.merkle_tree_generator import MerkleTreeGenerator +from cert_issuer import helpers +from lds_merkle_proof_2019.merkle_proof_2019 import MerkleProof2019 def get_test_data_generator(): @@ -29,36 +31,89 @@ def test_proofs_bitcoin_testnet(self): self.do_test_signature(Chain.bitcoin_testnet, 'bitcoinTestnet', 'BTCOpReturn') def test_proofs_bitcoin_regtest(self): - self.do_test_signature(Chain.bitcoin_regtest, 'bitcoinRegtest', 'BTCOpReturn') + ''' TODO merkle proof 2019 library with regtest ''' + # self.do_test_signature(Chain.bitcoin_regtest, 'bitcoinRegtest', 'BTCOpReturn') def test_proofs_mock(self): - self.do_test_signature(Chain.mockchain, 'mockchain', 'Mock') + ''' TODO merkle proof 2019 library with mocknet ''' + # self.do_test_signature(Chain.mockchain, 'mockchain', 'Mock') def do_test_signature(self, chain, display_chain, type): merkle_tree_generator = MerkleTreeGenerator() merkle_tree_generator.populate(get_test_data_generator()) _ = merkle_tree_generator.get_blockchain_data() gen = merkle_tree_generator.get_proof_generator( - '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', chain) + '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', '123', chain) p1 = next(gen) _ = next(gen) p3 = next(gen) - p1_expected = {'type': ['MerkleProof2017', 'Extension'], - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'targetHash': '6b86b273ff34fce19d6b804eff5a3f5747ada4eaa22f1d49c01e52ddb7875b4b', - 'proof': [{'right': 'd4735e3a265e16eee03f59718b9b5d03019c07d8b6c51f90da3a666eec13ab35'}, - {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'}], - 'anchors': [ - {'sourceId': '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', + p1_expected_old = { + 'type': ['MerkleProof2017', 'Extension'], + 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', + 'targetHash': '6b86b273ff34fce19d6b804eff5a3f5747ada4eaa22f1d49c01e52ddb7875b4b', + 'proof': [ + {'right': 'd4735e3a265e16eee03f59718b9b5d03019c07d8b6c51f90da3a666eec13ab35'}, + {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'} + ], + 'anchors': [ + { + 'sourceId': '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', 'type': type, - 'chain': display_chain}]} - p3_expected = {'type': ['MerkleProof2017', 'Extension'], + 'chain': display_chain + } + ] + } + + p1_json_proof = { + 'path': [ + {'right': 'd4735e3a265e16eee03f59718b9b5d03019c07d8b6c51f90da3a666eec13ab35'}, + {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'} + ], + 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', + 'targetHash': '6b86b273ff34fce19d6b804eff5a3f5747ada4eaa22f1d49c01e52ddb7875b4b', + 'anchors': [ + helpers.tx_to_blink(chain, '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582') + ] + } + mp2019 = MerkleProof2019() + proof_value = mp2019.encode(p1_json_proof) + + p1_expected = { + "type": "MerkleProof2019", + "created": p1['created'], + "proofValue": proof_value.decode('utf8'), + "proofPurpose": "assertionMethod", + "verificationMethod": "ecdsa-koblitz-pubkey:123" + } + + p3_expected_old = {'type': ['MerkleProof2017', 'Extension'], 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', 'targetHash': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce', 'proof': [{'left': '4295f72eeb1e3507b8461e240e3b8d18c1e7bd2f1122b11fc9ec40a65894031a'}], 'anchors': [{'sourceId': '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', 'type': type, 'chain': display_chain}]} + p3_json_proof = { + 'path': [ + {'left': '4295f72eeb1e3507b8461e240e3b8d18c1e7bd2f1122b11fc9ec40a65894031a'} + ], + 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', + 'targetHash': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce', + 'anchors': [ + helpers.tx_to_blink(chain, '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582') + ] + } + mp2019 = MerkleProof2019() + proof_value = mp2019.encode(p3_json_proof) + + p3_expected = { + "type": "MerkleProof2019", + "created": p3['created'], + "proofValue": proof_value.decode('utf8'), + "proofPurpose": "assertionMethod", + "verificationMethod": "ecdsa-koblitz-pubkey:123" + } + self.assertEqual(p1, p1_expected) self.assertEqual(p3, p3_expected) From 2a3adb4d157924b1ea3124e24f0c893960663301 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Thu, 23 Jan 2020 11:46:08 -0600 Subject: [PATCH 03/22] #164 - Updated MerkleProof2019 to support mocknet & bitcoin regtest --- cert_issuer/__init__.py | 2 +- cert_issuer/merkle_tree_generator.py | 2 +- requirements.txt | 2 +- tests/test_merkle_tree_generator.py | 29 ++-------------------------- tox.ini | 1 + 5 files changed, 6 insertions(+), 30 deletions(-) diff --git a/cert_issuer/__init__.py b/cert_issuer/__init__.py index b03a62d0..6e648d3c 100644 --- a/cert_issuer/__init__.py +++ b/cert_issuer/__init__.py @@ -1 +1 @@ -__version__ = '2.0.20' +__version__ = '3.0.0a1' diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index 9d10e9de..08f02078 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -72,7 +72,7 @@ def get_proof_generator(self, tx_id, issuing_address, chain=Chain.bitcoin_mainne helpers.tx_to_blink(chain, tx_id) ] } - logging.info('merkle_json: %s', str(merkle_json)) + logging.debug('merkle_json: %s', str(merkle_json)) proof_value = mp2019.encode(merkle_json) merkle_proof = { diff --git a/requirements.txt b/requirements.txt index 6edf6a0c..a6cf0206 100644 --- a/requirements.txt +++ b/requirements.txt @@ -11,4 +11,4 @@ pysha3>=1.0.2 python-bitcoinlib>=0.10.1 tox>=3.0.0 jsonschema<3.0.0 -lds-merkle-proof-2019>=0.0.1 \ No newline at end of file +lds-merkle-proof-2019>=0.0.2 \ No newline at end of file diff --git a/tests/test_merkle_tree_generator.py b/tests/test_merkle_tree_generator.py index 1438f29c..88648abd 100644 --- a/tests/test_merkle_tree_generator.py +++ b/tests/test_merkle_tree_generator.py @@ -31,12 +31,10 @@ def test_proofs_bitcoin_testnet(self): self.do_test_signature(Chain.bitcoin_testnet, 'bitcoinTestnet', 'BTCOpReturn') def test_proofs_bitcoin_regtest(self): - ''' TODO merkle proof 2019 library with regtest ''' - # self.do_test_signature(Chain.bitcoin_regtest, 'bitcoinRegtest', 'BTCOpReturn') + self.do_test_signature(Chain.bitcoin_regtest, 'bitcoinRegtest', 'BTCOpReturn') def test_proofs_mock(self): - ''' TODO merkle proof 2019 library with mocknet ''' - # self.do_test_signature(Chain.mockchain, 'mockchain', 'Mock') + self.do_test_signature(Chain.mockchain, 'mockchain', 'Mock') def do_test_signature(self, chain, display_chain, type): merkle_tree_generator = MerkleTreeGenerator() @@ -47,22 +45,6 @@ def do_test_signature(self, chain, display_chain, type): p1 = next(gen) _ = next(gen) p3 = next(gen) - p1_expected_old = { - 'type': ['MerkleProof2017', 'Extension'], - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'targetHash': '6b86b273ff34fce19d6b804eff5a3f5747ada4eaa22f1d49c01e52ddb7875b4b', - 'proof': [ - {'right': 'd4735e3a265e16eee03f59718b9b5d03019c07d8b6c51f90da3a666eec13ab35'}, - {'right': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce'} - ], - 'anchors': [ - { - 'sourceId': '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', - 'type': type, - 'chain': display_chain - } - ] - } p1_json_proof = { 'path': [ @@ -86,13 +68,6 @@ def do_test_signature(self, chain, display_chain, type): "verificationMethod": "ecdsa-koblitz-pubkey:123" } - p3_expected_old = {'type': ['MerkleProof2017', 'Extension'], - 'merkleRoot': '0932f1d2e98219f7d7452801e2b64ebd9e5c005539db12d9b1ddabe7834d9044', - 'targetHash': '4e07408562bedb8b60ce05c1decfe3ad16b72230967de01f640b7e4729b49fce', - 'proof': [{'left': '4295f72eeb1e3507b8461e240e3b8d18c1e7bd2f1122b11fc9ec40a65894031a'}], - 'anchors': [{'sourceId': '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', - 'type': type, - 'chain': display_chain}]} p3_json_proof = { 'path': [ {'left': '4295f72eeb1e3507b8461e240e3b8d18c1e7bd2f1122b11fc9ec40a65894031a'} diff --git a/tox.ini b/tox.ini index f731e862..fa5e7bc5 100644 --- a/tox.ini +++ b/tox.ini @@ -11,5 +11,6 @@ envlist = py36 changedir=tests deps= pytest + -rrequirements.txt commands=py.test --basetemp={envtmpdir} {posargs} # substitute with tox' positional arguments From a40a7720ba8c36b8391b1224bff6ec3217e36b45 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Mon, 27 Jan 2020 10:02:45 -0600 Subject: [PATCH 04/22] Allowing issuer url/did's for verification method of ld merkle proof --- README.md | 3 +++ cert_issuer/__init__.py | 2 +- cert_issuer/certificate_handlers.py | 4 ++-- cert_issuer/config.py | 1 + cert_issuer/merkle_tree_generator.py | 6 +++--- conf_template.ini | 3 +++ tests/test_certificate_handler.py | 8 ++++---- tests/test_merkle_tree_generator.py | 6 +++--- 8 files changed, 20 insertions(+), 13 deletions(-) diff --git a/README.md b/README.md index 34883f41..22b168c2 100644 --- a/README.md +++ b/README.md @@ -260,6 +260,9 @@ Edit your conf.ini file (the config file for this application). ``` issuing_address = +# issuer URL / DID +verification_method = + chain= usb_name = diff --git a/cert_issuer/__init__.py b/cert_issuer/__init__.py index 6e648d3c..e3dc11f6 100644 --- a/cert_issuer/__init__.py +++ b/cert_issuer/__init__.py @@ -1 +1 @@ -__version__ = '3.0.0a1' +__version__ = '3.0.0a2' diff --git a/cert_issuer/certificate_handlers.py b/cert_issuer/certificate_handlers.py index 33f5f8fb..ad29b3b2 100644 --- a/cert_issuer/certificate_handlers.py +++ b/cert_issuer/certificate_handlers.py @@ -49,7 +49,7 @@ def add_proof(self, certificate_json, merkle_proof): class CertificateBatchWebHandler(BatchHandler): def finish_batch(self, tx_id, chain): self.proof = [] - proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.issuing_address, chain) + proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.verification_method, chain) for metadata in self.certificates_to_issue: proof = next(proof_generator) self.proof.append(self.certificate_handler.add_proof(metadata, proof)) @@ -120,7 +120,7 @@ def get_certificate_generator(self): yield data_to_issue def finish_batch(self, tx_id, chain): - proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.issuing_address, chain) + proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.verification_method, chain) for _, metadata in self.certificates_to_issue.items(): proof = next(proof_generator) self.certificate_handler.add_proof(metadata, proof) diff --git a/cert_issuer/config.py b/cert_issuer/config.py index bbdfbc2e..169be1f1 100644 --- a/cert_issuer/config.py +++ b/cert_issuer/config.py @@ -34,6 +34,7 @@ def add_arguments(p): p.add('-c', '--my-config', required=False, env_var='CONFIG_FILE', is_config_file=True, help='config file path') p.add_argument('--issuing_address', required=True, help='issuing address', env_var='ISSUING_ADDRESS') + p.add_argument('--verification_method', required=True, help='Verification method for the Linked Data Proof', env_var='VERIFICATION_METHOD') p.add_argument('--usb_name', required=True, help='usb path to key_file', env_var='USB_NAME') p.add_argument('--key_file', required=True, help='name of file on USB containing private key', env_var='KEY_FILE') diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index 08f02078..372fd94f 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -44,7 +44,7 @@ def get_blockchain_data(self): merkle_root = self.tree.get_merkle_root() return h2b(ensure_string(merkle_root)) - def get_proof_generator(self, tx_id, issuing_address, chain=Chain.bitcoin_mainnet): + def get_proof_generator(self, tx_id, verification_method, chain=Chain.bitcoin_mainnet): """ Returns a generator (1-time iterator) of proofs in insertion order. @@ -72,7 +72,7 @@ def get_proof_generator(self, tx_id, issuing_address, chain=Chain.bitcoin_mainne helpers.tx_to_blink(chain, tx_id) ] } - logging.debug('merkle_json: %s', str(merkle_json)) + logging.info('merkle_json: %s', str(merkle_json)) proof_value = mp2019.encode(merkle_json) merkle_proof = { @@ -80,7 +80,7 @@ def get_proof_generator(self, tx_id, issuing_address, chain=Chain.bitcoin_mainne "created": datetime.now().isoformat(), "proofValue": proof_value.decode('utf8'), "proofPurpose": "assertionMethod", - "verificationMethod": "ecdsa-koblitz-pubkey:" + issuing_address + "verificationMethod": verification_method } yield merkle_proof diff --git a/conf_template.ini b/conf_template.ini index 499ab702..07c6d1a7 100644 --- a/conf_template.ini +++ b/conf_template.ini @@ -1,5 +1,8 @@ issuing_address = +# Issuer URL / DID as the verification method +verification_method = + # put your unsigned certificates here for signing. Default is /data/unsigned_certificates unsigned_certificates_dir= # signed certificates are the output from the cert signing step; input to the cert issuing step. Default is /data/signed_certificates diff --git a/tests/test_certificate_handler.py b/tests/test_certificate_handler.py index 49b19cae..7c7e4923 100644 --- a/tests/test_certificate_handler.py +++ b/tests/test_certificate_handler.py @@ -43,7 +43,7 @@ def _get_certificate_batch_web_handler(self): certificates_to_issue['3'] = mock.Mock() config = mock.Mock() - config.issuing_address = "123" + config.issuing_address = "http://example.com" handler = CertificateBatchWebHandler( secret_manager=secret_manager, @@ -61,7 +61,7 @@ def _get_certificate_batch_handler(self): certificates_to_issue['3'] = mock.Mock() config = mock.Mock() - config.issuing_address = "123" + config.issuing_address = "http://example.com" handler = CertificateBatchHandler( secret_manager=secret_manager, @@ -113,7 +113,7 @@ def test_batch_handler_finish_batch(self): proof = self._proof_helper(chain) config = mock.Mock() - config.issuing_address = "123" + config.issuing_address = "http://example.com" with patch.object(DummyCertificateHandler, 'add_proof') as mock_method: result = certificate_batch_handler.finish_batch( @@ -193,7 +193,7 @@ def test_web_add_proof(self): class DummyCertificateHandler(CertificateHandler): def __init__(self): self.config = mock.Mock() - self.config.issuing_address = "123" + self.config.issuing_address = "http://example.com" self.counter = 0 def validate_certificate(self, certificate_metadata): diff --git a/tests/test_merkle_tree_generator.py b/tests/test_merkle_tree_generator.py index 88648abd..426a138c 100644 --- a/tests/test_merkle_tree_generator.py +++ b/tests/test_merkle_tree_generator.py @@ -41,7 +41,7 @@ def do_test_signature(self, chain, display_chain, type): merkle_tree_generator.populate(get_test_data_generator()) _ = merkle_tree_generator.get_blockchain_data() gen = merkle_tree_generator.get_proof_generator( - '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', '123', chain) + '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', 'http://example.com', chain) p1 = next(gen) _ = next(gen) p3 = next(gen) @@ -65,7 +65,7 @@ def do_test_signature(self, chain, display_chain, type): "created": p1['created'], "proofValue": proof_value.decode('utf8'), "proofPurpose": "assertionMethod", - "verificationMethod": "ecdsa-koblitz-pubkey:123" + "verificationMethod": "http://example.com" } p3_json_proof = { @@ -86,7 +86,7 @@ def do_test_signature(self, chain, display_chain, type): "created": p3['created'], "proofValue": proof_value.decode('utf8'), "proofPurpose": "assertionMethod", - "verificationMethod": "ecdsa-koblitz-pubkey:123" + "verificationMethod": "http://example.com" } self.assertEqual(p1, p1_expected) From b1c464742d7bd42002a3b558bd1b1166e886fd30 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Thu, 13 Feb 2020 10:47:59 -0600 Subject: [PATCH 05/22] bump cert-schema to 3.0.0a2 --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index a6cf0206..ab4661c0 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ cert-core>=2.1.9 -cert-schema>=3.0.0a1 +cert-schema>=3.0.0a2 chainpoint>=0.0.2 configargparse==0.12.0 glob2==0.6 From 830ee9d8ff2860976138dca8f2440c571e5d2170 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Wed, 19 Feb 2020 15:44:27 -0600 Subject: [PATCH 06/22] RI etherscan changes from v2 to v3 --- .travis.yml | 2 +- .../blockchain_handlers/ethereum/connectors.py | 13 +++++++++---- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/.travis.yml b/.travis.yml index 923b0188..beaa2d02 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,6 +1,6 @@ sudo: false language: python python: - - "3.4" + - "3.6" install: pip install tox-travis script: tox diff --git a/cert_issuer/blockchain_handlers/ethereum/connectors.py b/cert_issuer/blockchain_handlers/ethereum/connectors.py index 1f75e84b..632289b2 100644 --- a/cert_issuer/blockchain_handlers/ethereum/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum/connectors.py @@ -95,12 +95,14 @@ def broadcast_tx(self, tx, api_token): broadcast_url = self.base_url + '?module=proxy&action=eth_sendRawTransaction' if api_token: - '&apikey=%s' % api_token + broadcast_url += '&apikey=%s' % api_token response = requests.post(broadcast_url, data={'hex': tx_hex}) if 'error' in response.json(): logging.error("Etherscan returned an error: %s", response.json()['error']) raise BroadcastError(response.json()['error']) if int(response.status_code) == 200: + if response.json().get('message', None) == 'NOTOK': + raise BroadcastError(response.json().get('result', None)) tx_id = response.json().get('result', None) logging.info("Transaction ID obtained from broadcast through Etherscan: %s", tx_id) return tx_id @@ -116,9 +118,11 @@ def get_balance(self, address, api_token): broadcast_url += '&address=%s' % address broadcast_url += '&tag=latest' if api_token: - '&apikey=%s' % api_token + broadcast_url += '&apikey=%s' % api_token response = requests.get(broadcast_url) if int(response.status_code) == 200: + if response.json().get('message', None) == 'NOTOK': + raise BroadcastError(response.json().get('result', None)) balance = int(response.json().get('result', None)) logging.info('Balance check succeeded: %s', response.json()) return balance @@ -133,10 +137,11 @@ def get_address_nonce(self, address, api_token): broadcast_url += '&address=%s' % address broadcast_url += '&tag=latest' if api_token: - '&apikey=%s' % api_token + broadcast_url += '&apikey=%s' % api_token response = requests.get(broadcast_url, ) if int(response.status_code) == 200: - # the int(res, 0) transforms the hex nonce to int + if response.json().get('message', None) == 'NOTOK': + raise BroadcastError(response.json().get('result', None)) nonce = int(response.json().get('result', None), 0) logging.info('Nonce check went correct: %s', response.json()) return nonce From 625b1f169fa52e3accd35eb3e56be644c72db190 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Wed, 19 Feb 2020 16:20:13 -0600 Subject: [PATCH 07/22] v3-alpha schema updates --- cert_issuer/__init__.py | 2 +- cert_issuer/certificate_handlers.py | 2 +- .../unsigned_certificates/verifiable-credential.json | 9 ++++++--- 3 files changed, 8 insertions(+), 5 deletions(-) diff --git a/cert_issuer/__init__.py b/cert_issuer/__init__.py index e3dc11f6..7ccf291d 100644 --- a/cert_issuer/__init__.py +++ b/cert_issuer/__init__.py @@ -1 +1 @@ -__version__ = '3.0.0a2' +__version__ = '3.0.0a3' diff --git a/cert_issuer/certificate_handlers.py b/cert_issuer/certificate_handlers.py index ad29b3b2..3684ebf5 100644 --- a/cert_issuer/certificate_handlers.py +++ b/cert_issuer/certificate_handlers.py @@ -22,7 +22,7 @@ def add_proof(self, certificate_metadata, merkle_proof): :return: """ certificate_json = self._get_certificate_to_issue(certificate_metadata) - certificate_json['signature'] = merkle_proof + certificate_json['proof'] = merkle_proof with open(certificate_metadata.blockchain_cert_file_name, 'w') as out_file: out_file.write(json.dumps(certificate_json)) diff --git a/examples/data-testnet/unsigned_certificates/verifiable-credential.json b/examples/data-testnet/unsigned_certificates/verifiable-credential.json index d3178aea..37ab8c1d 100644 --- a/examples/data-testnet/unsigned_certificates/verifiable-credential.json +++ b/examples/data-testnet/unsigned_certificates/verifiable-credential.json @@ -5,13 +5,16 @@ "https://www.w3.org/2018/credentials/examples/v1" ], "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", - "type": ["VerifiableCredential"], + "type": [ + "VerifiableCredential", + "BlockcertsCredential" + ], "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", - "issuanceDate": "2010-01-01T19:73:24Z", + "issuanceDate": "2010-01-01T19:33:24Z", "credentialSubject": { "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", "alumniOf": { "id": "did:example:c276e12ec21ebfeb1f712ebc6f1" } } -} \ No newline at end of file +} From df2a44f8b36c8885a0da2df3080e05a006b7496e Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Thu, 20 Feb 2020 10:55:34 -0600 Subject: [PATCH 08/22] Fix test using proof instead of signature for v3 --- tests/test_certificate_handler.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_certificate_handler.py b/tests/test_certificate_handler.py index 7c7e4923..9a25689b 100644 --- a/tests/test_certificate_handler.py +++ b/tests/test_certificate_handler.py @@ -166,7 +166,7 @@ def test_add_proof(self,mock_open): cert_to_issue = {'kek':'kek'} proof = {'a': 'merkel'} - file_call = 'call().__enter__().write(\'{"kek": "kek", "signature": {"a": "merkel"}}\')' + file_call = 'call().__enter__().write(\'{"kek": "kek", "proof": {"a": "merkel"}}\')' chain = mock.Mock() metadata = mock.Mock() From c64f477e1000307a9d7db43c632d3d7c749f4383 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Thu, 20 Feb 2020 10:55:56 -0600 Subject: [PATCH 09/22] Bump v3-alpha schema to 3.0.0a3 --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index ab4661c0..3275b404 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ cert-core>=2.1.9 -cert-schema>=3.0.0a2 +cert-schema>=3.0.0a3 chainpoint>=0.0.2 configargparse==0.12.0 glob2==0.6 From 24a4f4f8ff202e51b77cc585b128ce9d862d49e2 Mon Sep 17 00:00:00 2001 From: AnthonyRonning Date: Wed, 11 Mar 2020 10:18:14 -0500 Subject: [PATCH 10/22] Bump v3-alpha schema to 3.0.0a4 --- requirements.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/requirements.txt b/requirements.txt index 3275b404..65ce1f19 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,5 @@ cert-core>=2.1.9 -cert-schema>=3.0.0a3 +cert-schema>=3.0.0a4 chainpoint>=0.0.2 configargparse==0.12.0 glob2==0.6 From f85bd555541044c2e561473e7e45ded521599002 Mon Sep 17 00:00:00 2001 From: James Date: Tue, 30 Jun 2020 11:03:18 +0200 Subject: [PATCH 11/22] updated for bloxberg --- .../ethereum_sc/.connectors.py.swp | Bin 0 -> 16384 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 cert_issuer/blockchain_handlers/ethereum_sc/.connectors.py.swp diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/.connectors.py.swp b/cert_issuer/blockchain_handlers/ethereum_sc/.connectors.py.swp new file mode 100644 index 0000000000000000000000000000000000000000..a52c6d26e00a98a5dab66211c9ce1c4db5112ee0 GIT binary patch literal 16384 zcmeHNO^h5z6)wzA_{)zRky8$pv58Y8ibx+r* zs%K{wD8^h7Rn=GTRllyT(fMo7KF1$V&odkkF!tu}|8V1thaY0Eon~xb%%!FByBtpL+%;vY zr6``#>yyO4>I64OgSH?rE%w(KZjU;GoJ}nW;0Qa3u)}8tY*^4 zjL|Nn76V{ZYsfu8_B25te*1J{77 zz>~mdfQNwJ-N)E(fY*UnfL-7UZ~=H2I0gLuUdH|g{1x~P@FK7a+ylJ(amKy@+yJfv zj{^??_XBTzjInP6Uk0uKDR2(>3l<~32YeUU0lo-)3i#(o7<&i!CGbt)3E&ez^y4>0xv;QPSK zzzRr!+wVu+zzDbooCjXTBFj&KTfjDOAMg%%e;XkFe??>PpC25@SWb9m)wIkiBT6gs zO6fAOvX~6{!pMa%a-3D`jf-f#Ka8xFrsUOHjhiyM;9H#Z(vi_(oC{m^O`?w4RHH=I zx}uwRQsILej zaI5xKM4#DfR5r4MiQ&~zQiXo)9TvyVR zqTr5Y!|hBj3+PI|q{ga}V?WIPvUCH^SE`z|mL_dCAWHXoeXfY2yg;vcj+&?fj(s+~ z<3Vk+c4axXAgvzkt_-+!YVE!q*SnQamE{v{pk$;4saCe;0zu@rp+4`f{Q*DDVKXCh zsTNflG%-Za2s=w>a-Fn8Rjq?yaIb82A%#1o)V42Zv)6{omD*wgof$0kj8MT*3l?-- z5vz`cq-F*|mYUv;+7@-RJK(;vjnXlZiGj71S>_19b=iz+ue50vNflL7OI3`!ZAN$Xf;`+FiL95J`Z$iWQ(_xx7OHH>IuA95FS{ zAU5nvl)7{V>hSR+F_l%PJ==O&R*so83>A12%cA;xp3}XJ)@IvukZ`L%FEnk$t>!!{ z)Ld1)HWhk0Rpk`^U`qTagQsa4pw8h}AE+_HW^~9)S*^6W$@PSP((;*_Vlx_C9x(&s zUcSC(W#vP1qNc;fvq3PQL#^Ty%s?sMDWt%}bs&wJ98&z$``awSFwONm9SqxMcTVdb zw7!YCsOcSlvOTZmI+4>z)pvr|AB+1uOMsRP~A(@w`osmC%~nqs5V459)JBd-P-I7@)6I$X%sd1PTb znLw*?>GJM&*t(ju^=!DfBU7yjQkob&=d?%G?~6^_=WDYsmL6tdUu-H%%hi0LO~t!a zF)~I6k$mq~LqC+qZS{v8aSgX%wEvm?;&Qp;Ra|isP9J{l;dUQeE;xv%9@3%>f=M!O>bL zIOzu3cLvj5FFOkfA0<>HjZ6+>+_V6#v3wF^ktO3vmQ7S4vn&K?7|(^a zR#BFJAc`eMiYN;eUgU#1j%W)y9P9dy(aKE)YsVk({GWU#Xff z9ilJ^5|qI)7qu(QEg=+Xo*DA)L{rOjBYTdSrf7oqB(LzsCB5G?!pFS(Fh2?SgfIma-Prxs{&eusV9+82_Kx7~? z5E+OJL<@VT(F zutim!{Jrq(3wp`%krYezhgdn}m9D>s369Sey>)zA>^s}8k;ru}7g%v~e^}>lmxo=2 z^*7e$D-WZ>a5SY@>kxTfrBu-S_SyMD+SZ5`cDhB1(67n!A5~CBA4^hNjG#q3|3#&< IHnze31&%J9QUCw| literal 0 HcmV?d00001 From 7f0d1dbf3cff91a8fec9970b07386d913e862a19 Mon Sep 17 00:00:00 2001 From: James Date: Fri, 3 Jul 2020 12:11:31 +0200 Subject: [PATCH 12/22] updated pathing for merkle tree --- .gitignore | 2 +- cert_issuer/blockchain_handlers/__init__.py | 1 + cert_issuer/blockchain_handlers/ethereum/connectors.py | 2 +- cert_issuer/blockchain_handlers/ethereum_sc/connectors.py | 4 ++-- cert_issuer/blockchain_handlers/ethereum_sc/signer.py | 4 ++-- cert_issuer/config.py | 4 ++-- cert_issuer/merkle_tree_generator.py | 2 +- 7 files changed, 10 insertions(+), 9 deletions(-) diff --git a/.gitignore b/.gitignore index 14ef34ad..978ba581 100644 --- a/.gitignore +++ b/.gitignore @@ -4,7 +4,7 @@ venv*/ virtual\ usb/ secrets.py __pycache__ - +pk certs_dbdump.csv data/unsigned_certificates/ diff --git a/cert_issuer/blockchain_handlers/__init__.py b/cert_issuer/blockchain_handlers/__init__.py index e69de29b..b03a62d0 100644 --- a/cert_issuer/blockchain_handlers/__init__.py +++ b/cert_issuer/blockchain_handlers/__init__.py @@ -0,0 +1 @@ +__version__ = '2.0.20' diff --git a/cert_issuer/blockchain_handlers/ethereum/connectors.py b/cert_issuer/blockchain_handlers/ethereum/connectors.py index 1f75e84b..1b0bf581 100644 --- a/cert_issuer/blockchain_handlers/ethereum/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum/connectors.py @@ -221,7 +221,7 @@ def get_address_nonce(self, address, api_token): # Configure Ethereum Ropsten testnet connectors rop_provider_list = [] -rop_provider_list.append(EtherscanBroadcaster('https://ropsten.etherscan.io/api')) +rop_provider_list.append(EtherscanBroadcaster('https://blockexplorer.bloxberg.org/api')) rop_provider_list.append(MyEtherWalletBroadcaster('https://api.myetherwallet.com/rop')) connectors[Chain.ethereum_ropsten] = rop_provider_list diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py index c3ab981b..2b2781d5 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py @@ -1,7 +1,7 @@ import json import os import logging -from errors import UnableToSignTxError +from cert_issuer.errors import UnableToSignTxError from cert_issuer.models import ServiceProviderConnector from web3 import Web3, HTTPProvider @@ -14,7 +14,7 @@ def get_abi(contract): """ directory = os.path.dirname(os.path.abspath(__file__)) - path = os.path.join(directory, f"data/{contract}_abi.json") + path = os.path.join(directory, f"/home/james/bloxberg-certs/cert-issuer/cert_issuer/blockchain_handlers/ethereum_sc/data/{contract}_abi.json") with open(path, "r") as f: raw = f.read() diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/signer.py b/cert_issuer/blockchain_handlers/ethereum_sc/signer.py index dcddd8c5..61a107aa 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/signer.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/signer.py @@ -12,9 +12,9 @@ def __init__(self, ethereum_chain): self.ethereum_chain = ethereum_chain # Netcode ensures replay protection (see EIP155) if ethereum_chain.external_display_value == 'ethereumMainnet': - self.netcode = 1 + self.netcode = 8995 elif ethereum_chain.external_display_value == 'ethereumRopsten': - self.netcode = 3 + self.netcode = 8995 else: self.netcode = None diff --git a/cert_issuer/config.py b/cert_issuer/config.py index faffa53d..8abcedb0 100644 --- a/cert_issuer/config.py +++ b/cert_issuer/config.py @@ -89,9 +89,9 @@ def add_arguments(p): help='ens_name that points to the smart contract to which to issue', env_var='ENS_NAME') p.add_argument('--revocation_list_file', required=False, help='list of certificates or batches to be revokes', env_var='REVOCATION_LIST_FILE') - p.add_argument('--ens_registry_ropsten', required=False, default="0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", + p.add_argument('--ens_registry_ropsten', required=False, default="0xde68Fcf6814D81Ee910bf35703622571718E07a7", help='ENS registry address on ropsten', env_var='ENS_RESGISTRY_ROPSTEN') - p.add_argument('--ens_registry_mainnet', required=False, default="0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e", + p.add_argument('--ens_registry_mainnet', required=False, default="0xde68Fcf6814D81Ee910bf35703622571718E07a7", help='ENS registry address on ropsten', env_var='ENS_RESGISTRY_MAINNET') def get_config(): diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index e6f5e3cb..3f3b6fc4 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -61,7 +61,7 @@ def get_proof_generator(self, tx_id, app_config, chain=Chain.bitcoin_mainnet): proof2.append(dict2) target_hash = ensure_string(self.tree.get_leaf(index)) if app_config.issuing_method == "smart_contract": - from blockchain_handlers.ethereum_sc.ens import ENSConnector + from cert_issuer.blockchain_handlers.ethereum_sc.ens import ENSConnector ens = ENSConnector(app_config) abi = ens.get_abi() From e35071c1722f9f44116ac542f7da74cb43e7d25a Mon Sep 17 00:00:00 2001 From: James Date: Mon, 6 Jul 2020 17:05:30 +0200 Subject: [PATCH 13/22] updated with ERC721 bloxberg contracts --- .../ethereum/transaction_handlers.py | 2 +- .../ethereum_sc/transaction_handlers.py | 12 ++++++++---- cert_issuer/issuer.py | 10 ++++++++-- cert_issuer/models.py | 4 ++-- 4 files changed, 19 insertions(+), 9 deletions(-) diff --git a/cert_issuer/blockchain_handlers/ethereum/transaction_handlers.py b/cert_issuer/blockchain_handlers/ethereum/transaction_handlers.py index 8a483f5f..cdf1898b 100644 --- a/cert_issuer/blockchain_handlers/ethereum/transaction_handlers.py +++ b/cert_issuer/blockchain_handlers/ethereum/transaction_handlers.py @@ -54,7 +54,7 @@ def ensure_balance(self): logging.error(error_message) raise InsufficientFundsError(error_message) - def issue_transaction(self, blockchain_bytes, app_config): + def issue_transaction(self, recipient_address, token_uri, blockchain_bytes, app_config): eth_data_field = b2h(blockchain_bytes) prepared_tx = self.create_transaction(blockchain_bytes) signed_tx = self.sign_transaction(prepared_tx) diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py index 60721188..92442af2 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py @@ -29,11 +29,15 @@ def ensure_balance(self): def revoke_transaction(self, blockchain_bytes, app_config): return self.make_transaction(blockchain_bytes, app_config, "revoke_hash") - def issue_transaction(self, blockchain_bytes, app_config): - return self.make_transaction(blockchain_bytes, app_config, "issue_hash") + #Outdated Function: TODO - REMOVE + #def issue_transaction(self, blockchain_bytes, app_config): + # return self.make_transaction(blockchain_bytes, app_config, "issue_hash") - def make_transaction(self, blockchain_bytes, app_config, method): - prepared_tx = self.connector.create_transaction(method, blockchain_bytes) + def issue_transaction(self, recipient_address, token_uri, blockchain_bytes, app_config): + return self.make_transaction(recipient_address, token_uri, blockchain_bytes, app_config, "createCertificate") + + def make_transaction(self, recipient_address, token_uri, blockchain_bytes, app_config, method): + prepared_tx = self.connector.create_transaction(method, recipient_address, token_uri, blockchain_bytes) signed_tx = self.sign_transaction(prepared_tx) logging.info('Broadcasting transaction to the blockchain...') diff --git a/cert_issuer/issuer.py b/cert_issuer/issuer.py index dc43b228..304c441e 100644 --- a/cert_issuer/issuer.py +++ b/cert_issuer/issuer.py @@ -2,7 +2,8 @@ Base class for building blockchain transactions to issue Blockchain Certificates. """ import logging - +from eth_utils import is_checksum_address +from web3 import Web3 from cert_issuer.errors import BroadcastError MAX_TX_RETRIES = 5 @@ -21,10 +22,15 @@ def issue(self, chain, app_config): """ blockchain_bytes = self.certificate_batch_handler.prepare_batch() + #Change to parameters that get passed in app.py TODO + recipient_address = Web3.toChecksumAddress('0xaA84378fA41da83a9B6523bA46E45A664FbEBFC8') + print(is_checksum_address(recipient_address)) + token_uri = "https://bloxberg.org" + blockchain_bytes = str(blockchain_bytes, 'latin-1') for attempt_number in range(0, self.max_retry): try: - txid = self.transaction_handler.issue_transaction(blockchain_bytes, app_config) + txid = self.transaction_handler.issue_transaction(recipient_address, token_uri, blockchain_bytes, app_config) self.certificate_batch_handler.finish_batch(txid, chain, app_config) logging.info('Broadcast transaction with txid %s', txid) return txid diff --git a/cert_issuer/models.py b/cert_issuer/models.py index ce6279a6..22d61fb8 100644 --- a/cert_issuer/models.py +++ b/cert_issuer/models.py @@ -90,7 +90,7 @@ def ensure_balance(self): pass @abstractmethod - def issue_transaction(self, blockchain_bytes): + def issue_transaction(self, recipient_address, token_uri, blockchain_bytes): pass @@ -98,7 +98,7 @@ class MockTransactionHandler(TransactionHandler): def ensure_balance(self): pass - def issue_transaction(self, op_return_bytes): + def issue_transaction(self, recipient_address, token_uri, op_return_bytes): return 'This has not been issued on a blockchain and is for testing only' From c8dd8d2c6288007f49061de9c2e97c9246540a4e Mon Sep 17 00:00:00 2001 From: James Date: Fri, 17 Jul 2020 09:48:24 +0200 Subject: [PATCH 14/22] updated issuing for bloxberg --- .app.py.swp | Bin 0 -> 12288 bytes app.py | 6 ++++-- .../ethereum_sc/connectors.py | 9 +++++++-- .../3bc1a96a-3501-46ed-8f75-49612bbac257.json | 1 + work/unsigned_certificates/.placeholder | 0 .../3bc1a96a-3501-46ed-8f75-49612bbac257.json | 1 + 6 files changed, 13 insertions(+), 4 deletions(-) create mode 100644 .app.py.swp create mode 100644 work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json create mode 100644 work/unsigned_certificates/.placeholder create mode 100644 work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json diff --git a/.app.py.swp b/.app.py.swp new file mode 100644 index 0000000000000000000000000000000000000000..8aabc4ae320ea3e0ce904862309e964017a33073 GIT binary patch literal 12288 zcmeI2J#W-77{^^$kZ1vc1QX!WAvx)qT-yazI)sp@gv6T!bm1j(a?YLGoE>5ZdKAGY zU;zd`1`^DC0QOEyOboyW02UrQiMXp;b?8*|`1C(3$@BWNGn~$GXZ6|*dNJq_7$*q% za`?}qr>oQC^JzlX*pO>`e~;np=2t~wN|s$w8@rjmaE4)=sg0Ny{rQ9!Mu+x$UT9tN zB4l|Up)OPr7Z5==U)~eu2J$wm_diuRxoiXP`@<1<)DL z)(j!9LDxawA>T*P2hcl^&A$o4fdr5M5Y9&G#r| z$Ks(`B_&UcDrlEJAU1gw6fXO=a+kqxkseQ}Bj>l4$z`{Esx;j9+(GV|Ega+);gf}+ z6FO6`O^4hJR8M#Bd$(`hx$8Y_O)O&--DRD~Jic6&O)yc6HHMwiRNAFCmE_AVPCrvI z%j!q2>TyrlMU9j5ALuMN za!sKMr^^`jZNjx~JQJ2>)w9%=+1z$Dx80i{(oq}U47Z!wl*LeblgI8YGB+ERx(H(- L!+ZqmA(zNs!|84j literal 0 HcmV?d00001 diff --git a/app.py b/app.py index 16aae656..d3d03a8f 100755 --- a/app.py +++ b/app.py @@ -4,7 +4,7 @@ from subprocess import call import cert_issuer.config -from cert_issuer.blockchain_handlers import bitcoin +from cert_issuer.blockchain_handlers import ethereum_sc import cert_issuer.issue_certificates app = Flask(__name__) @@ -20,7 +20,9 @@ def get_config(): def issue(): config = get_config() certificate_batch_handler, transaction_handler, connector = \ - bitcoin.instantiate_blockchain_handlers(config, False) + ethereum_sc.instantiate_blockchain_handlers(config) + #Removed File mode from ethereum_sc + #bitcoin.instantiate_blockchain_handlers(config, False) certificate_batch_handler.set_certificates_in_batch(request.json) cert_issuer.issue_certificates.issue(config, certificate_batch_handler, transaction_handler) return json.dumps(certificate_batch_handler.proof) diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py index 2b2781d5..fbdbdac1 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py @@ -49,7 +49,7 @@ def get_balance(self, address): def create_transaction(self, method, *argv): gas_limit = self.cost_constants.get_gas_limit() - estimated_gas = self._contract_obj.functions[method](*argv).estimateGas() * 2 + estimated_gas = self._contract_obj.functions[method](*argv).estimateGas() * 5 if estimated_gas > gas_limit: logging.warning("Estimated gas of %s more than gas limit of %s, transaction might fail. Please verify on etherescan.com.", estimated_gas, gas_limit) estimated_gas = gas_limit @@ -57,13 +57,18 @@ def create_transaction(self, method, *argv): gas_price = self._w3.eth.gasPrice gas_price_limit = self.cost_constants.get_gas_price() + print("gas price: " + str(gas_price)) + print("gas_price_limit: " + str(gas_price_limit)) + print("estimated_gas: " + str(estimated_gas)) + if gas_price > gas_price_limit: logging.warning("Gas price provided by network of %s higher than gas price of %s set in config, transaction might fail. Please verify on etherescan.com.", gas_price, gas_price_limit) gas_price = gas_price_limit tx_options = { 'nonce': self._w3.eth.getTransactionCount(self._w3.eth.defaultAccount), - 'gas': estimated_gas, + 'gas': 700000, + #'gas': estimated_gas, 'gasPrice': gas_price } diff --git a/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json b/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json new file mode 100644 index 00000000..63394b26 --- /dev/null +++ b/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json @@ -0,0 +1 @@ +{"issuedOn": "2017-07-20T09:33:47.490752+00:00", "recipient": {"identity": "eularia@landroth.org", "type": "email", "hashed": false}, "type": "Assertion", "verification": {"publicKey": "ecdsa-koblitz-pubkey:msBCHdwaQ7N2ypBYupkp6uNxtr9Pg76imj", "type": ["MerkleProofVerification2017", "Extension"]}, "@context": ["https://w3id.org/openbadges/v2", "https://w3id.org/blockcerts/v2"], "badge": {"issuer": {"url": "https://www.issuer.org", "name": "University of Learning", "email": "contact@issuer.org", "type": "Profile", "id": "https://www.blockcerts.org/samples/2.0/issuer-testnet.json", "image": "", "revocationList": "https://www.blockcerts.org/samples/2.0/revocation-list-testnet.json"}, "name": "Certificate of Accomplishment", "type": "BadgeClass", "criteria": {"narrative": "Nibh iriure ei nam, modo ridens neglegentur mel eu. At his cibo mucius."}, "image": "", "id": "urn:uuid:82a4c9f2-3588-457b-80ea-da695571b8fc", "description": "Lorem ipsum dolor sit amet, mei docendi concludaturque ad, cu nec partem graece. Est aperiam consetetur cu, expetenda moderatius neglegentur ei nam, suas dolor laudem eam an.", "signatureLines": [{"jobTitle": "University Issuer", "name": "Your signature", "type": ["SignatureLine", "Extension"], "image": ""}]}, "id": "urn:uuid:3bc1a96a-3501-46ed-8f75-49612bbac257", "recipientProfile": {"publicKey": "ecdsa-koblitz-pubkey:mtr98kany9G1XYNU74pRnfBQmaCg2FZLmc", "name": "Eularia Landroth", "type": ["RecipientProfile", "Extension"]}, "signature": {"type": ["MerkleProof2017", "Extension"], "merkleRoot": "c65c6184e3d5a945ddb5437e93ea312411fd33aa1def22b0746d6ecd4aa30f20", "targetHash": "c65c6184e3d5a945ddb5437e93ea312411fd33aa1def22b0746d6ecd4aa30f20", "proof": [], "anchors": [{"sourceId": "0xd85776b42af81083517d5399db02ff63b87ca96b5e86ee36a96cbf968b0352f0", "type": "ETHSmartContract", "chain": "ethereumMainnet", "contract_address": "0x7aF019edb78B6932f796996a7Eaa1fC8d8e183CF", "ens_name": "mpdl.berg", "contract_abi": [{"inputs": [], "stateMutability": "nonpayable", "type": "constructor"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "approved", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Approval", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "operator", "type": "address"}, {"indexed": false, "internalType": "bool", "name": "approved", "type": "bool"}], "name": "ApprovalForAll", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "from", "type": "address"}, {"indexed": true, "internalType": "address", "name": "to", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Transfer", "type": "event"}, {"inputs": [{"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "approve", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}], "name": "balanceOf", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "baseURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "recipient", "type": "address"}, {"internalType": "string", "name": "tokenURI", "type": "string"}, {"internalType": "string", "name": "tokenHash", "type": "string"}], "name": "createCertificate", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "getApproved", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "address", "name": "operator", "type": "address"}], "name": "isApprovedForAll", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "name", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "ownerOf", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}, {"internalType": "bytes", "name": "_data", "type": "bytes"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "operator", "type": "address"}, {"internalType": "bool", "name": "approved", "type": "bool"}], "name": "setApprovalForAll", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "bytes4", "name": "interfaceId", "type": "bytes4"}], "name": "supportsInterface", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "symbol", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenHash", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenOfOwnerByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "totalSupply", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "transferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}]}]}} \ No newline at end of file diff --git a/work/unsigned_certificates/.placeholder b/work/unsigned_certificates/.placeholder new file mode 100644 index 00000000..e69de29b diff --git a/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json b/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json new file mode 100644 index 00000000..716deae0 --- /dev/null +++ b/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json @@ -0,0 +1 @@ +{"issuedOn": "2017-07-20T09:33:47.490752+00:00", "recipient": {"identity": "eularia@landroth.org", "type": "email", "hashed": false}, "type": "Assertion", "verification": {"publicKey": "ecdsa-koblitz-pubkey:msBCHdwaQ7N2ypBYupkp6uNxtr9Pg76imj", "type": ["MerkleProofVerification2017", "Extension"]}, "@context": ["https://w3id.org/openbadges/v2", "https://w3id.org/blockcerts/v2"], "badge": {"issuer": {"url": "https://www.issuer.org", "name": "University of Learning", "email": "contact@issuer.org", "type": "Profile", "id": "https://www.blockcerts.org/samples/2.0/issuer-testnet.json", "image": "", "revocationList": "https://www.blockcerts.org/samples/2.0/revocation-list-testnet.json"}, "name": "Certificate of Accomplishment", "type": "BadgeClass", "criteria": {"narrative": "Nibh iriure ei nam, modo ridens neglegentur mel eu. At his cibo mucius."}, "image": "", "id": "urn:uuid:82a4c9f2-3588-457b-80ea-da695571b8fc", "description": "Lorem ipsum dolor sit amet, mei docendi concludaturque ad, cu nec partem graece. Est aperiam consetetur cu, expetenda moderatius neglegentur ei nam, suas dolor laudem eam an.", "signatureLines": [{"jobTitle": "University Issuer", "name": "Your signature", "type": ["SignatureLine", "Extension"], "image": ""}]}, "id": "urn:uuid:3bc1a96a-3501-46ed-8f75-49612bbac257", "recipientProfile": {"publicKey": "ecdsa-koblitz-pubkey:mtr98kany9G1XYNU74pRnfBQmaCg2FZLmc", "name": "Eularia Landroth", "type": ["RecipientProfile", "Extension"]}} \ No newline at end of file From 80b2a3347745c58119d03e209ce0bf195772058c Mon Sep 17 00:00:00 2001 From: James Date: Thu, 23 Jul 2020 15:51:40 +0200 Subject: [PATCH 15/22] updated API for bloxberg --- .dockerignore | 1 + Dockerfilebloxberg | 12 +++++ app/main.py | 48 +++++++++++++++++++ .../ethereum_sc/connectors.py | 3 +- cert_issuer/config.py | 2 +- cert_issuer/issuer.py | 1 + ethereum_smart_contract_requirements.txt | 16 +++++++ .../3bc1a96a-3501-46ed-8f75-49612bbac257.json | 1 - .../3bc1a96a-3501-46ed-8f75-49612bbac257.json | 1 - work/unsigned_certificates/testcert.json | 27 +++++++++++ 10 files changed, 108 insertions(+), 4 deletions(-) create mode 100644 Dockerfilebloxberg create mode 100644 app/main.py delete mode 100644 work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json delete mode 100644 work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json create mode 100644 work/unsigned_certificates/testcert.json diff --git a/.dockerignore b/.dockerignore index 4f37e5ab..ba79feef 100644 --- a/.dockerignore +++ b/.dockerignore @@ -2,3 +2,4 @@ .tox *egg-info venv +pk diff --git a/Dockerfilebloxberg b/Dockerfilebloxberg new file mode 100644 index 00000000..858f7d4f --- /dev/null +++ b/Dockerfilebloxberg @@ -0,0 +1,12 @@ +FROM tiangolo/uvicorn-gunicorn-fastapi:python3.7 + +EXPOSE 80 + +COPY ./app /app/app +COPY ./cert_issuer /app/cert_issuer +COPY ./sample_data /app/sample_data +COPY ethereum_smart_contract_requirements.txt . +COPY conf.ini . + +RUN pip install -r ethereum_smart_contract_requirements.txt + diff --git a/app/main.py b/app/main.py new file mode 100644 index 00000000..0a179337 --- /dev/null +++ b/app/main.py @@ -0,0 +1,48 @@ +from typing import List, Optional +from functools import lru_cache +from fastapi import Depends, FastAPI +from cert_tools import instantiate_v3_alpha_certificate_batch, create_v3_alpha_certificate_template +from pydantic import BaseModel +import configargparse +import os + +app = FastAPI() + + +@app.get("/") +def read_root(): + return {"Hello": "World"} + +class Batch(BaseModel): + publicKey: str + recipient_name: Optional[str] + email: Optional[str] + SHA256: List[str] + + class Config: + schema_extra = { + "example": { + "publicKey": "0x69575606E8b8F0cAaA5A3BD1fc5D032024Bb85AF", + "recipient_name": "Albert Einstein", + "email": "einstein@mpg.de", + "SHA256": ["0x0e4ded5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6", "0xfda3124d5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6"], + } + } + + +@app.post("/createUnsignedCertificateBatch") +def createUnsignedCertificate(batch: Batch): + #create_v3_alpha_certificate_template(recipient_name, email) + conf = create_v3_alpha_certificate_template.get_config() + create_v3_alpha_certificate_template.write_certificate_template(conf, batch.recipient_name, batch.email) + conf_instantiate = instantiate_v3_alpha_certificate_batch.get_config() + instantiate_v3_alpha_certificate_batch.instantiate_batch(conf_instantiate, batch.publicKey, batch.recipient_name, batch.email, batch.SHA256) + + return {"Unsigned Certificate Created!"} + +@app.post("/createCertificateBatch") +def createCertificateBatch(batch: Batch): + conf = instantiate_v3_alpha_certificate_batch.get_config() + instantiate_v3_alpha_certificate_batch.instantiate_batch(conf, batch.publicKey, batch.recipient_name, batch.email, batch.SHA256) + + return {"Batch Created!"} diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py index fbdbdac1..5caa4e0b 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py @@ -14,7 +14,8 @@ def get_abi(contract): """ directory = os.path.dirname(os.path.abspath(__file__)) - path = os.path.join(directory, f"/home/james/bloxberg-certs/cert-issuer/cert_issuer/blockchain_handlers/ethereum_sc/data/{contract}_abi.json") + path = os.path.join(directory, f"data/{contract}_abi.json") + print(path) with open(path, "r") as f: raw = f.read() diff --git a/cert_issuer/config.py b/cert_issuer/config.py index 8abcedb0..4c47309c 100644 --- a/cert_issuer/config.py +++ b/cert_issuer/config.py @@ -1,7 +1,7 @@ import logging import os -import bitcoin +#import bitcoin import configargparse from cert_core import BlockchainType, Chain, chain_to_bitcoin_network, UnknownChainError diff --git a/cert_issuer/issuer.py b/cert_issuer/issuer.py index 304c441e..44fc19d5 100644 --- a/cert_issuer/issuer.py +++ b/cert_issuer/issuer.py @@ -25,6 +25,7 @@ def issue(self, chain, app_config): #Change to parameters that get passed in app.py TODO recipient_address = Web3.toChecksumAddress('0xaA84378fA41da83a9B6523bA46E45A664FbEBFC8') print(is_checksum_address(recipient_address)) + #change token_uri address token_uri = "https://bloxberg.org" blockchain_bytes = str(blockchain_bytes, 'latin-1') diff --git a/ethereum_smart_contract_requirements.txt b/ethereum_smart_contract_requirements.txt index b9e889d4..b8b03e9d 100644 --- a/ethereum_smart_contract_requirements.txt +++ b/ethereum_smart_contract_requirements.txt @@ -1,2 +1,18 @@ coincurve==7.1.0 web3>=5 + + + +configargparse==0.12.0 +cert-core +cert-schema +chainpoint3>=0.0.2 +glob2==0.6 +mock==2.0.0 +requests[security]>=2.18.4 +pycoin==0.80 +pyld +pysha3>=1.0.2 +python-bitcoinlib>=0.10.1 +tox>=3.0.0 +jsonschema<3 diff --git a/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json b/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json deleted file mode 100644 index 63394b26..00000000 --- a/work/blockchain_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json +++ /dev/null @@ -1 +0,0 @@ -{"issuedOn": "2017-07-20T09:33:47.490752+00:00", "recipient": {"identity": "eularia@landroth.org", "type": "email", "hashed": false}, "type": "Assertion", "verification": {"publicKey": "ecdsa-koblitz-pubkey:msBCHdwaQ7N2ypBYupkp6uNxtr9Pg76imj", "type": ["MerkleProofVerification2017", "Extension"]}, "@context": ["https://w3id.org/openbadges/v2", "https://w3id.org/blockcerts/v2"], "badge": {"issuer": {"url": "https://www.issuer.org", "name": "University of Learning", "email": "contact@issuer.org", "type": "Profile", "id": "https://www.blockcerts.org/samples/2.0/issuer-testnet.json", "image": "", "revocationList": "https://www.blockcerts.org/samples/2.0/revocation-list-testnet.json"}, "name": "Certificate of Accomplishment", "type": "BadgeClass", "criteria": {"narrative": "Nibh iriure ei nam, modo ridens neglegentur mel eu. At his cibo mucius."}, "image": "", "id": "urn:uuid:82a4c9f2-3588-457b-80ea-da695571b8fc", "description": "Lorem ipsum dolor sit amet, mei docendi concludaturque ad, cu nec partem graece. Est aperiam consetetur cu, expetenda moderatius neglegentur ei nam, suas dolor laudem eam an.", "signatureLines": [{"jobTitle": "University Issuer", "name": "Your signature", "type": ["SignatureLine", "Extension"], "image": ""}]}, "id": "urn:uuid:3bc1a96a-3501-46ed-8f75-49612bbac257", "recipientProfile": {"publicKey": "ecdsa-koblitz-pubkey:mtr98kany9G1XYNU74pRnfBQmaCg2FZLmc", "name": "Eularia Landroth", "type": ["RecipientProfile", "Extension"]}, "signature": {"type": ["MerkleProof2017", "Extension"], "merkleRoot": "c65c6184e3d5a945ddb5437e93ea312411fd33aa1def22b0746d6ecd4aa30f20", "targetHash": "c65c6184e3d5a945ddb5437e93ea312411fd33aa1def22b0746d6ecd4aa30f20", "proof": [], "anchors": [{"sourceId": "0xd85776b42af81083517d5399db02ff63b87ca96b5e86ee36a96cbf968b0352f0", "type": "ETHSmartContract", "chain": "ethereumMainnet", "contract_address": "0x7aF019edb78B6932f796996a7Eaa1fC8d8e183CF", "ens_name": "mpdl.berg", "contract_abi": [{"inputs": [], "stateMutability": "nonpayable", "type": "constructor"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "approved", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Approval", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "operator", "type": "address"}, {"indexed": false, "internalType": "bool", "name": "approved", "type": "bool"}], "name": "ApprovalForAll", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "from", "type": "address"}, {"indexed": true, "internalType": "address", "name": "to", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Transfer", "type": "event"}, {"inputs": [{"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "approve", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}], "name": "balanceOf", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "baseURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "recipient", "type": "address"}, {"internalType": "string", "name": "tokenURI", "type": "string"}, {"internalType": "string", "name": "tokenHash", "type": "string"}], "name": "createCertificate", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "getApproved", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "address", "name": "operator", "type": "address"}], "name": "isApprovedForAll", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "name", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "ownerOf", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}, {"internalType": "bytes", "name": "_data", "type": "bytes"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "operator", "type": "address"}, {"internalType": "bool", "name": "approved", "type": "bool"}], "name": "setApprovalForAll", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "bytes4", "name": "interfaceId", "type": "bytes4"}], "name": "supportsInterface", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "symbol", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenHash", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenOfOwnerByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "totalSupply", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "transferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}]}]}} \ No newline at end of file diff --git a/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json b/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json deleted file mode 100644 index 716deae0..00000000 --- a/work/unsigned_certificates/3bc1a96a-3501-46ed-8f75-49612bbac257.json +++ /dev/null @@ -1 +0,0 @@ -{"issuedOn": "2017-07-20T09:33:47.490752+00:00", "recipient": {"identity": "eularia@landroth.org", "type": "email", "hashed": false}, "type": "Assertion", "verification": {"publicKey": "ecdsa-koblitz-pubkey:msBCHdwaQ7N2ypBYupkp6uNxtr9Pg76imj", "type": ["MerkleProofVerification2017", "Extension"]}, "@context": ["https://w3id.org/openbadges/v2", "https://w3id.org/blockcerts/v2"], "badge": {"issuer": {"url": "https://www.issuer.org", "name": "University of Learning", "email": "contact@issuer.org", "type": "Profile", "id": "https://www.blockcerts.org/samples/2.0/issuer-testnet.json", "image": "", "revocationList": "https://www.blockcerts.org/samples/2.0/revocation-list-testnet.json"}, "name": "Certificate of Accomplishment", "type": "BadgeClass", "criteria": {"narrative": "Nibh iriure ei nam, modo ridens neglegentur mel eu. At his cibo mucius."}, "image": "", "id": "urn:uuid:82a4c9f2-3588-457b-80ea-da695571b8fc", "description": "Lorem ipsum dolor sit amet, mei docendi concludaturque ad, cu nec partem graece. Est aperiam consetetur cu, expetenda moderatius neglegentur ei nam, suas dolor laudem eam an.", "signatureLines": [{"jobTitle": "University Issuer", "name": "Your signature", "type": ["SignatureLine", "Extension"], "image": ""}]}, "id": "urn:uuid:3bc1a96a-3501-46ed-8f75-49612bbac257", "recipientProfile": {"publicKey": "ecdsa-koblitz-pubkey:mtr98kany9G1XYNU74pRnfBQmaCg2FZLmc", "name": "Eularia Landroth", "type": ["RecipientProfile", "Extension"]}} \ No newline at end of file diff --git a/work/unsigned_certificates/testcert.json b/work/unsigned_certificates/testcert.json new file mode 100644 index 00000000..d816a6ea --- /dev/null +++ b/work/unsigned_certificates/testcert.json @@ -0,0 +1,27 @@ +{ + "@context":[ + "https://www.w3.org/2018/credentials/v1", + "https://w3id.org/blockcerts/schema/3.0-alpha/context.json", + "https://www.w3.org/2018/credentials/examples/v1" + ], + "type":[ + "VerifiableCredential", + "BlockcertsCredential" + ], + "issuer":"https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", + "issuanceDate":"2020-03-27T19:33:48.260278+00:00", + "id":"urn:uuid:ea856760-c1e9-4243-8e95-1a36c70f9615", + "credentialSubject":{ + "id":"ecdsa-koblitz-pubkey:mkwntSiQmc14H65YxwckLenxY3DsEpvFbe", + "alumniOf":{ + "id":"https://www.issuer.org" + } + }, + "proof":{ + "type":"MerkleProof2019", + "created":"2020-03-27T14:34:08.976994", + "proofValue":"zG7NYSuZNW6m3AqhgaP3kTHowuF5vwFNeVaNGisY5KQckwVZCf1Ze6QUeEghjGN1va4WVq9HBamzEFG4gq6AUxWrn6728njszRaXudPjGFbjAzNMpuRd6Bb27XzKzygXUEk4uPRPtKSr78Aibka5gMNmTf8LhWSsfNB6DdQVoUVc9tM1ctBy2jXY2sL5YDt1it66RhgRdirCaj8tUDhAfXMoe14DTyJvDDZgQmpYNtFqhrg7d71raPFPmo4AgHpFipGLs3NjxrSoBd29ex6Fae3rJAwHWHp173chNrXuvGDT99M2YxJqnxnr4LitxrGz39V57QrC5x4WqqpB1osMnKzfUHgtxetvzkKStuTzZsCLhFrZT7eSjrjy9wuU4er1dzk3nN23A1cDpWWFQM7mw79Rs8tv4U7As6r9wvUjm7VSRhfH9ur2qJxVbwBgEs7WDhmrbMeF7demEmKcU5Fq91JZRowXouyz6cGxWHb4QhLTHhtcQMx34", + "proofPurpose":"assertionMethod", + "verificationMethod":"ecdsa-koblitz-pubkey:0x7e30a37763e6Ba1fFeDE1750bBeFB4c60b17a1B3" + } +} From 5cdb14b4f7d60305c76e8ceacd2dc66e07676486 Mon Sep 17 00:00:00 2001 From: James Date: Fri, 24 Jul 2020 10:50:09 +0200 Subject: [PATCH 16/22] updated smart contract merkle proof anchors --- .../ethereum_sc/__init__.py | 5 +- cert_issuer/certificate_handlers.py | 7 +- cert_issuer/merkle_tree_generator.py | 77 ++++++++++++++----- ethereum_smart_contract_requirements.txt | 2 +- tests/test_merkle_tree_generator.py | 4 - work/blockchain_certificates/testcert.json | 1 + work/unsigned_certificates/testcert.json | 41 ++++------ 7 files changed, 82 insertions(+), 55 deletions(-) create mode 100644 work/blockchain_certificates/testcert.json diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py b/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py index 3b9d026b..755c34d3 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py @@ -82,8 +82,9 @@ def instantiate_blockchain_handlers(app_config): chain = app_config.chain secret_manager = initialize_signer(app_config) certificate_batch_handler = CertificateBatchHandler(secret_manager=secret_manager, - certificate_handler=CertificateV2Handler(), - merkle_tree=MerkleTreeGenerator()) + certificate_handler=CertificateV3Handler(), + merkle_tree=MerkleTreeGenerator(), + config=app_config) if chain == Chain.mockchain: transaction_handler = MockTransactionHandler() # ethereum chains diff --git a/cert_issuer/certificate_handlers.py b/cert_issuer/certificate_handlers.py index 533c9cd8..b82dc050 100644 --- a/cert_issuer/certificate_handlers.py +++ b/cert_issuer/certificate_handlers.py @@ -47,9 +47,10 @@ def add_proof(self, certificate_json, merkle_proof): class CertificateBatchWebHandler(BatchHandler): + #Smart contract version has app_config as variable - removed in V3 def finish_batch(self, tx_id, chain, app_config): self.proof = [] - proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.verification_method, chain) + proof_generator = self.merkle_tree.get_proof_generator(tx_id, app_config, self.config.verification_method, chain) for metadata in self.certificates_to_issue: proof = next(proof_generator) self.proof.append(self.certificate_handler.add_proof(metadata, proof)) @@ -119,8 +120,8 @@ def get_certificate_generator(self): data_to_issue = self.certificate_handler.get_byte_array_to_issue(metadata) yield data_to_issue - def finish_batch(self, tx_id, chain): - proof_generator = self.merkle_tree.get_proof_generator(tx_id, self.config.verification_method, chain) + def finish_batch(self, tx_id, chain, app_config): + proof_generator = self.merkle_tree.get_proof_generator(tx_id, app_config, self.config.verification_method, chain) for _, metadata in self.certificates_to_issue.items(): proof = next(proof_generator) self.certificate_handler.add_proof(metadata, proof) diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index ffa89c83..1451e20a 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -45,7 +45,7 @@ def get_blockchain_data(self): merkle_root = self.tree.get_merkle_root() return h2b(ensure_string(merkle_root)) - def get_proof_generator(self, tx_id, verification_method, chain=Chain.bitcoin_mainnet): + def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chain.bitcoin_mainnet): """ Returns a generator (1-time iterator) of proofs in insertion order. @@ -57,33 +57,70 @@ def get_proof_generator(self, tx_id, verification_method, chain=Chain.bitcoin_ma for index in range(0, node_count): proof = self.tree.get_proof(index) proof2 = [] - + #Change back to smart contract proof & verification for p in proof: dict2 = dict() for key, value in p.items(): dict2[key] = ensure_string(value) proof2.append(dict2) target_hash = ensure_string(self.tree.get_leaf(index)) - mp2019 = MerkleProof2019() - merkle_json = { - "path": proof2, - "merkleRoot": root, - "targetHash": target_hash, - "anchors": [ - helpers.tx_to_blink(chain, tx_id) - ] + + """ + Add additional parameters for smart contract certification + """ + if app_config.issuing_method == "smart_contract": + from cert_issuer.blockchain_handlers.ethereum_sc.ens import ENSConnector + + ens = ENSConnector(app_config) + abi = ens.get_abi() + + mp2019 = MerkleProof2019() + merkle_json = { + "path": proof2, + "merkleRoot": root, + "targetHash": target_hash + } + logging.info('merkle_json: %s', str(merkle_json)) + + proof_value = mp2019.encode(merkle_json) + merkle_proof = { + "type": "MerkleProof2019", + "created": datetime.now().isoformat(), + "proofValue": proof_value.decode('utf8'), + "proofPurpose": "assertionMethod", + "verificationMethod": verification_method, + "anchors": [{ + #helpers.tx_to_blink(chain, tx_id), + "sourceId": to_source_id(tx_id, chain), + "type": "ETHSmartContract", + "contract_address": app_config.contract_address, + "ens_name": app_config.ens_name, + "contract_abi": abi + }] + } + else: + mp2019 = MerkleProof2019() + merkle_json = { + "path": proof2, + "merkleRoot": root, + "targetHash": target_hash, + "anchors": [ + helpers.tx_to_blink(chain, tx_id) + ] + } + logging.info('merkle_json: %s', str(merkle_json)) + + proof_value = mp2019.encode(merkle_json) + merkle_proof = { + "type": "MerkleProof2019", + "created": datetime.now().isoformat(), + "proofValue": proof_value.decode('utf8'), + "proofPurpose": "assertionMethod", + "verificationMethod": verification_method } - logging.info('merkle_json: %s', str(merkle_json)) - - proof_value = mp2019.encode(merkle_json) - merkle_proof = { - "type": "MerkleProof2019", - "created": datetime.now().isoformat(), - "proofValue": proof_value.decode('utf8'), - "proofPurpose": "assertionMethod", - "verificationMethod": verification_method - } + yield merkle_proof + def to_source_id(txid, chain): diff --git a/ethereum_smart_contract_requirements.txt b/ethereum_smart_contract_requirements.txt index c16ed4bb..297bae74 100644 --- a/ethereum_smart_contract_requirements.txt +++ b/ethereum_smart_contract_requirements.txt @@ -2,7 +2,7 @@ coincurve==7.1.0 web3>=5 - +lds-merkle-proof-2019 configargparse==0.12.0 cert-core>=2.1.9 cert-schema>=2.1.5 diff --git a/tests/test_merkle_tree_generator.py b/tests/test_merkle_tree_generator.py index caa6bd19..922b4938 100644 --- a/tests/test_merkle_tree_generator.py +++ b/tests/test_merkle_tree_generator.py @@ -47,11 +47,7 @@ def do_test_signature(self, chain, display_chain, type): merkle_tree_generator.populate(get_test_data_generator()) _ = merkle_tree_generator.get_blockchain_data() gen = merkle_tree_generator.get_proof_generator( -<<<<<<< HEAD - '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', app_config, chain) -======= '8087c03e7b7bc9ca7b355de9d9d8165cc5c76307f337f0deb8a204d002c8e582', 'http://example.com', chain) ->>>>>>> 24a4f4f8ff202e51b77cc585b128ce9d862d49e2 p1 = next(gen) _ = next(gen) p3 = next(gen) diff --git a/work/blockchain_certificates/testcert.json b/work/blockchain_certificates/testcert.json new file mode 100644 index 00000000..dd347d2f --- /dev/null +++ b/work/blockchain_certificates/testcert.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1"], "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", "issuanceDate": "2010-01-01T19:33:24Z", "credentialSubject": {"id": "did:example:ebfeb1f712ebc6f1c276e12ec21", "alumniOf": {"id": "did:example:c276e12ec21ebfeb1f712ebc6f1"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-24T08:47:20.600951", "proofValue": "z8LQoL6KgPhoogRUFwfZ1rLhC36BBqheSDxHGE2MVeJ72eAakygWPg7uPh5ai5NF7mXpnQP8s7b4pS5Bz1W35EBrXAHSF3vbasxikUa578zWkkrpuAk5d9AVgtpyyTBeuscD2tQzQ8U1hqqAjq382hYmZecknwpJuYyVhBQ4S3aL4uiu8fYNK7Yn3m6KMVG3f2nhpN", "proofPurpose": "assertionMethod", "verificationMethod": "url", "anchors": [{"sourceId": "0x862b3de090744a2e719b6598bd8e9362198e64e3c3523b68afc986708c2004ab", "type": "ETHSmartContract", "contract_address": "0x7aF019edb78B6932f796996a7Eaa1fC8d8e183CF", "ens_name": "mpdl.berg", "contract_abi": [{"inputs": [], "stateMutability": "nonpayable", "type": "constructor"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "approved", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Approval", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "operator", "type": "address"}, {"indexed": false, "internalType": "bool", "name": "approved", "type": "bool"}], "name": "ApprovalForAll", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "from", "type": "address"}, {"indexed": true, "internalType": "address", "name": "to", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Transfer", "type": "event"}, {"inputs": [{"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "approve", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}], "name": "balanceOf", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "baseURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "recipient", "type": "address"}, {"internalType": "string", "name": "tokenURI", "type": "string"}, {"internalType": "string", "name": "tokenHash", "type": "string"}], "name": "createCertificate", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "getApproved", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "address", "name": "operator", "type": "address"}], "name": "isApprovedForAll", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "name", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "ownerOf", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}, {"internalType": "bytes", "name": "_data", "type": "bytes"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "operator", "type": "address"}, {"internalType": "bool", "name": "approved", "type": "bool"}], "name": "setApprovalForAll", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "bytes4", "name": "interfaceId", "type": "bytes4"}], "name": "supportsInterface", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "symbol", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenHash", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenOfOwnerByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "totalSupply", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "transferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}]}]}} \ No newline at end of file diff --git a/work/unsigned_certificates/testcert.json b/work/unsigned_certificates/testcert.json index d816a6ea..96dd6abf 100644 --- a/work/unsigned_certificates/testcert.json +++ b/work/unsigned_certificates/testcert.json @@ -1,27 +1,18 @@ { - "@context":[ - "https://www.w3.org/2018/credentials/v1", - "https://w3id.org/blockcerts/schema/3.0-alpha/context.json", - "https://www.w3.org/2018/credentials/examples/v1" - ], - "type":[ - "VerifiableCredential", - "BlockcertsCredential" - ], - "issuer":"https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", - "issuanceDate":"2020-03-27T19:33:48.260278+00:00", - "id":"urn:uuid:ea856760-c1e9-4243-8e95-1a36c70f9615", - "credentialSubject":{ - "id":"ecdsa-koblitz-pubkey:mkwntSiQmc14H65YxwckLenxY3DsEpvFbe", - "alumniOf":{ - "id":"https://www.issuer.org" - } - }, - "proof":{ - "type":"MerkleProof2019", - "created":"2020-03-27T14:34:08.976994", - "proofValue":"zG7NYSuZNW6m3AqhgaP3kTHowuF5vwFNeVaNGisY5KQckwVZCf1Ze6QUeEghjGN1va4WVq9HBamzEFG4gq6AUxWrn6728njszRaXudPjGFbjAzNMpuRd6Bb27XzKzygXUEk4uPRPtKSr78Aibka5gMNmTf8LhWSsfNB6DdQVoUVc9tM1ctBy2jXY2sL5YDt1it66RhgRdirCaj8tUDhAfXMoe14DTyJvDDZgQmpYNtFqhrg7d71raPFPmo4AgHpFipGLs3NjxrSoBd29ex6Fae3rJAwHWHp173chNrXuvGDT99M2YxJqnxnr4LitxrGz39V57QrC5x4WqqpB1osMnKzfUHgtxetvzkKStuTzZsCLhFrZT7eSjrjy9wuU4er1dzk3nN23A1cDpWWFQM7mw79Rs8tv4U7As6r9wvUjm7VSRhfH9ur2qJxVbwBgEs7WDhmrbMeF7demEmKcU5Fq91JZRowXouyz6cGxWHb4QhLTHhtcQMx34", - "proofPurpose":"assertionMethod", - "verificationMethod":"ecdsa-koblitz-pubkey:0x7e30a37763e6Ba1fFeDE1750bBeFB4c60b17a1B3" - } + "@context": [ + "https://www.w3.org/2018/credentials/v1" + ], + "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", + "type": [ + "VerifiableCredential", + "BlockcertsCredential" + ], + "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", + "issuanceDate": "2010-01-01T19:33:24Z", + "credentialSubject": { + "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", + "alumniOf": { + "id": "did:example:c276e12ec21ebfeb1f712ebc6f1" + } + } } From c3074b26f8c7cc1e20590a660a279286bab46bfe Mon Sep 17 00:00:00 2001 From: James Date: Thu, 30 Jul 2020 14:37:26 +0200 Subject: [PATCH 17/22] changed anchor structure in merkle tree --- cert_issuer/issue_certificates.py | 4 +-- cert_issuer/issuer.py | 12 ++++---- cert_issuer/merkle_tree_generator.py | 28 ++++++++++++------- data/blockchain_certificates/.placeholder | 0 ethereum_smart_contract_requirements.txt | 2 +- .../35b6c347-51e4-4881-96bf-287ffbccdfce.json | 1 + .../7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json | 1 + work/blockchain_certificates/testcert.json | 1 - .../35b6c347-51e4-4881-96bf-287ffbccdfce.json | 1 + .../7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json | 1 + work/unsigned_certificates/testcert.json | 18 ------------ 11 files changed, 31 insertions(+), 38 deletions(-) mode change 100755 => 100644 data/blockchain_certificates/.placeholder create mode 100644 work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json create mode 100644 work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json delete mode 100644 work/blockchain_certificates/testcert.json create mode 100644 work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json create mode 100644 work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json delete mode 100644 work/unsigned_certificates/testcert.json diff --git a/cert_issuer/issue_certificates.py b/cert_issuer/issue_certificates.py index feeb4345..4b506e4b 100644 --- a/cert_issuer/issue_certificates.py +++ b/cert_issuer/issue_certificates.py @@ -11,7 +11,7 @@ sys.exit(1) -def issue(app_config, certificate_batch_handler, transaction_handler): +def issue(app_config, certificate_batch_handler, transaction_handler, recipient_address): certificate_batch_handler.pre_batch_actions(app_config) transaction_handler.ensure_balance() @@ -20,7 +20,7 @@ def issue(app_config, certificate_batch_handler, transaction_handler): certificate_batch_handler=certificate_batch_handler, transaction_handler=transaction_handler, max_retry=app_config.max_retry) - tx_id = issuer.issue(app_config.chain, app_config) + tx_id = issuer.issue(app_config.chain, app_config, recipient_address) certificate_batch_handler.post_batch_actions(app_config) return tx_id diff --git a/cert_issuer/issuer.py b/cert_issuer/issuer.py index 44fc19d5..77141983 100644 --- a/cert_issuer/issuer.py +++ b/cert_issuer/issuer.py @@ -15,20 +15,20 @@ def __init__(self, certificate_batch_handler, transaction_handler, max_retry=MAX self.transaction_handler = transaction_handler self.max_retry = max_retry - def issue(self, chain, app_config): + def issue(self, chain, app_config, recipient_address): """ Issue the certificates on the blockchain :return: """ blockchain_bytes = self.certificate_batch_handler.prepare_batch() - #Change to parameters that get passed in app.py TODO - recipient_address = Web3.toChecksumAddress('0xaA84378fA41da83a9B6523bA46E45A664FbEBFC8') + + recipient_address = Web3.toChecksumAddress(recipient_address) print(is_checksum_address(recipient_address)) - #change token_uri address + #change token_uri address token_uri = "https://bloxberg.org" - blockchain_bytes = str(blockchain_bytes, 'latin-1') - + #blockchain_bytes = str(blockchain_bytes, 'latin-1') + blockchain_bytes = blockchain_bytes.hex() for attempt_number in range(0, self.max_retry): try: txid = self.transaction_handler.issue_transaction(recipient_address, token_uri, blockchain_bytes, app_config) diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index 1451e20a..f96af55a 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -78,7 +78,11 @@ def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chai merkle_json = { "path": proof2, "merkleRoot": root, - "targetHash": target_hash + "targetHash": target_hash, + #Possibly adjust anchor to merkle_proof dict + "anchors": [ + helpers.tx_to_blink(chain, tx_id) + ] } logging.info('merkle_json: %s', str(merkle_json)) @@ -88,16 +92,20 @@ def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chai "created": datetime.now().isoformat(), "proofValue": proof_value.decode('utf8'), "proofPurpose": "assertionMethod", - "verificationMethod": verification_method, - "anchors": [{ - #helpers.tx_to_blink(chain, tx_id), - "sourceId": to_source_id(tx_id, chain), - "type": "ETHSmartContract", - "contract_address": app_config.contract_address, - "ens_name": app_config.ens_name, - "contract_abi": abi - }] + "verificationMethod": verification_method } + #Uncomment after checking verification + """ + "anchors": [{ + #helpers.tx_to_blink(chain, tx_id), + "sourceId": to_source_id(tx_id, chain), + "type": "ETHSmartContract", + "contract_address": app_config.contract_address, + "ens_name": app_config.ens_name, + "contract_abi": abi + }] + """ + else: mp2019 = MerkleProof2019() merkle_json = { diff --git a/data/blockchain_certificates/.placeholder b/data/blockchain_certificates/.placeholder old mode 100755 new mode 100644 diff --git a/ethereum_smart_contract_requirements.txt b/ethereum_smart_contract_requirements.txt index 297bae74..7a96b013 100644 --- a/ethereum_smart_contract_requirements.txt +++ b/ethereum_smart_contract_requirements.txt @@ -1,7 +1,7 @@ coincurve==7.1.0 web3>=5 - +ipfshttpclient lds-merkle-proof-2019 configargparse==0.12.0 cert-core>=2.1.9 diff --git a/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json b/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json new file mode 100644 index 00000000..8fa6a469 --- /dev/null +++ b/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:35b6c347-51e4-4881-96bf-287ffbccdfce", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-30T12:36:26.247132", "proofValue": "z2LuLBVSfogU8YhUjfqryx2c8kYrVZ6jAJz2XGxy12hmXpzP2u1h5aWhceYdMwZQM9YqD1Kk7qW3WLArFSp1xgSGquLRThhk15ou5FkXtEDHtGCV4HE4aLxYjcfQX8DnS5GM1gA9npS9SjQ5UDaHMdh8qvqQA4gPumEMWLhVavs7PqtsqJpaDozD7XAkyv1RVfQVBW9EcJsF22KBAWKZvrUnU2jhzGBojt19Av2AsEp7rJM6HeW6dPRx3cKKmSEZqNUCogWBuoUNhsMiD3hmo9dBdHTcnnDhgrcCQ2hjgfa98ftN6NeV25TTe7nHCqLyQ9yWaV2ShkKg6ah7ma6347h3eEmcbkPqSctxZQ6dMQoXqRDAcmM66JYnnWttWXwAGhMPYGqU33Ys5Uwr1qvGfb", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json b/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json new file mode 100644 index 00000000..b4ce0e12 --- /dev/null +++ b/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:7f2e45da-be39-4f4d-86ed-15ea4d0512bf", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-30T12:36:26.404988", "proofValue": "z2LuLBVSfnVzaQtw5CHiYTk8VQf8pSv2ppMr3tnknHhNQFFDTA1KMRa8kLyeXj1MogtTaDdCmfd74QBzKff9vcJsHvxYmbUQPBqdAcpfD5oM8zo8XiXGQd9eXfPh1wkNhwngkt2XGDKP2T2fewubVZRAyNDBFgKwuUThraP8KDvzEjMXyxkjA69Y9ukXTR25FL7znrkbsHy2yP13WV87RkpmNb1NQuG8sWkV4RAu9aXh79RoSmFz8Ty91HPXRkc7TwesgqHQMLvFMeZY3ZoLA3fvzfNY3xaUiPcuceUxdmjuC2i8NQZuf6j84YeZT3aY2gsMsMtgdgJxqWH3GqPwSEQu1SFZ6dmhvDF5kmHUkPD3esGtXiGc898Xn2f9RtL4b5kihT3oFqFfAffPzSDN9F", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/testcert.json b/work/blockchain_certificates/testcert.json deleted file mode 100644 index dd347d2f..00000000 --- a/work/blockchain_certificates/testcert.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1"], "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", "issuanceDate": "2010-01-01T19:33:24Z", "credentialSubject": {"id": "did:example:ebfeb1f712ebc6f1c276e12ec21", "alumniOf": {"id": "did:example:c276e12ec21ebfeb1f712ebc6f1"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-24T08:47:20.600951", "proofValue": "z8LQoL6KgPhoogRUFwfZ1rLhC36BBqheSDxHGE2MVeJ72eAakygWPg7uPh5ai5NF7mXpnQP8s7b4pS5Bz1W35EBrXAHSF3vbasxikUa578zWkkrpuAk5d9AVgtpyyTBeuscD2tQzQ8U1hqqAjq382hYmZecknwpJuYyVhBQ4S3aL4uiu8fYNK7Yn3m6KMVG3f2nhpN", "proofPurpose": "assertionMethod", "verificationMethod": "url", "anchors": [{"sourceId": "0x862b3de090744a2e719b6598bd8e9362198e64e3c3523b68afc986708c2004ab", "type": "ETHSmartContract", "contract_address": "0x7aF019edb78B6932f796996a7Eaa1fC8d8e183CF", "ens_name": "mpdl.berg", "contract_abi": [{"inputs": [], "stateMutability": "nonpayable", "type": "constructor"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "approved", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Approval", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "owner", "type": "address"}, {"indexed": true, "internalType": "address", "name": "operator", "type": "address"}, {"indexed": false, "internalType": "bool", "name": "approved", "type": "bool"}], "name": "ApprovalForAll", "type": "event"}, {"anonymous": false, "inputs": [{"indexed": true, "internalType": "address", "name": "from", "type": "address"}, {"indexed": true, "internalType": "address", "name": "to", "type": "address"}, {"indexed": true, "internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "Transfer", "type": "event"}, {"inputs": [{"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "approve", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}], "name": "balanceOf", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "baseURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "recipient", "type": "address"}, {"internalType": "string", "name": "tokenURI", "type": "string"}, {"internalType": "string", "name": "tokenHash", "type": "string"}], "name": "createCertificate", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "getApproved", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "address", "name": "operator", "type": "address"}], "name": "isApprovedForAll", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "name", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "ownerOf", "outputs": [{"internalType": "address", "name": "", "type": "address"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}, {"internalType": "bytes", "name": "_data", "type": "bytes"}], "name": "safeTransferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "address", "name": "operator", "type": "address"}, {"internalType": "bool", "name": "approved", "type": "bool"}], "name": "setApprovalForAll", "outputs": [], "stateMutability": "nonpayable", "type": "function"}, {"inputs": [{"internalType": "bytes4", "name": "interfaceId", "type": "bytes4"}], "name": "supportsInterface", "outputs": [{"internalType": "bool", "name": "", "type": "bool"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "symbol", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenHash", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "owner", "type": "address"}, {"internalType": "uint256", "name": "index", "type": "uint256"}], "name": "tokenOfOwnerByIndex", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "tokenURI", "outputs": [{"internalType": "string", "name": "", "type": "string"}], "stateMutability": "view", "type": "function"}, {"inputs": [], "name": "totalSupply", "outputs": [{"internalType": "uint256", "name": "", "type": "uint256"}], "stateMutability": "view", "type": "function"}, {"inputs": [{"internalType": "address", "name": "from", "type": "address"}, {"internalType": "address", "name": "to", "type": "address"}, {"internalType": "uint256", "name": "tokenId", "type": "uint256"}], "name": "transferFrom", "outputs": [], "stateMutability": "nonpayable", "type": "function"}]}]}} \ No newline at end of file diff --git a/work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json b/work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json new file mode 100644 index 00000000..425f6de7 --- /dev/null +++ b/work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:35b6c347-51e4-4881-96bf-287ffbccdfce", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json b/work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json new file mode 100644 index 00000000..76097d1e --- /dev/null +++ b/work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:7f2e45da-be39-4f4d-86ed-15ea4d0512bf", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/testcert.json b/work/unsigned_certificates/testcert.json deleted file mode 100644 index 96dd6abf..00000000 --- a/work/unsigned_certificates/testcert.json +++ /dev/null @@ -1,18 +0,0 @@ -{ - "@context": [ - "https://www.w3.org/2018/credentials/v1" - ], - "id": "urn:uuid:bbba8553-8ec1-445f-82c9-a57251dd731c", - "type": [ - "VerifiableCredential", - "BlockcertsCredential" - ], - "issuer": "did:example:23adb1f712ebc6f1c276eba4dfa", - "issuanceDate": "2010-01-01T19:33:24Z", - "credentialSubject": { - "id": "did:example:ebfeb1f712ebc6f1c276e12ec21", - "alumniOf": { - "id": "did:example:c276e12ec21ebfeb1f712ebc6f1" - } - } -} From 2459d4b0e6adaac7e822cddb018853638fed93d4 Mon Sep 17 00:00:00 2001 From: James Date: Wed, 12 Aug 2020 10:27:33 +0200 Subject: [PATCH 18/22] updated bloxberg as blink --- README.md | 8 ++++---- cert_issuer/blockchain_handlers/ethereum/__init__.py | 2 +- .../blockchain_handlers/ethereum/connectors.py | 4 ++-- cert_issuer/blockchain_handlers/ethereum/signer.py | 2 +- .../blockchain_handlers/ethereum_sc/__init__.py | 2 +- cert_issuer/blockchain_handlers/ethereum_sc/ens.py | 4 ++-- cert_issuer/blockchain_handlers/ethereum_sc/signer.py | 2 +- cert_issuer/config.py | 11 ++++++----- cert_issuer/helpers.py | 4 ++-- cert_issuer/issue_certificates.py | 2 +- cert_issuer/merkle_tree_generator.py | 2 +- docs/ethereum_smart_contract.md | 4 ++-- ethereum_smart_contract_requirements.txt | 7 +++---- requirements.txt | 2 +- .../15186643-ec29-4055-bebc-aa464d928849.json | 1 + .../1f0d1af0-4c14-4836-ade3-0df9facb3ced.json | 1 + .../35b6c347-51e4-4881-96bf-287ffbccdfce.json | 1 - .../7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json | 1 - .../a2a99a9e-457b-4de9-930b-3146442275a8.json | 1 + .../c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json | 1 + ...json => 15186643-ec29-4055-bebc-aa464d928849.json} | 2 +- ...json => 1f0d1af0-4c14-4836-ade3-0df9facb3ced.json} | 2 +- .../a2a99a9e-457b-4de9-930b-3146442275a8.json | 1 + .../c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json | 1 + 24 files changed, 36 insertions(+), 32 deletions(-) create mode 100644 work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json create mode 100644 work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json delete mode 100644 work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json delete mode 100644 work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json create mode 100644 work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json create mode 100644 work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json rename work/unsigned_certificates/{35b6c347-51e4-4881-96bf-287ffbccdfce.json => 15186643-ec29-4055-bebc-aa464d928849.json} (63%) rename work/unsigned_certificates/{7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json => 1f0d1af0-4c14-4836-ade3-0df9facb3ced.json} (63%) create mode 100644 work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json create mode 100644 work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json diff --git a/README.md b/README.md index 6b3ed604..d884bdac 100644 --- a/README.md +++ b/README.md @@ -223,7 +223,7 @@ If you are using a local bitcoin node, you can create addresses by command line. ### Create an Ethereum issuing address -Currently Blockcerts just supports issuing to the Ropsten Ethereum testnet, and the Ethereum mainnet. In Ethereum a public/private key pair is the same accross all test/main networks. +Currently Blockcerts just supports issuing to the Bloxberg Ethereum testnet, and the Ethereum mainnet. In Ethereum a public/private key pair is the same accross all test/main networks. __These steps involve storing secure information on a USB. Do not plug in this USB when your computer's wifi is on.__ @@ -268,7 +268,7 @@ issuing_address = # issuer URL / DID verification_method = -chain= +chain= usb_name = key_file = @@ -301,7 +301,7 @@ python cert-issuer -c conf.ini - The Blockchain Certificates will be located in data/blockchain_certificates. - If you ran in the mainnet or testnet mode, you can also see your transaction on a live blockchain explorer. - For Bitcoin, Blockr.io has explorers for both [testnet](https://tbtc.blockr.io/) and [mainnet](https://blockr.io/). - - For Ethereum, Etherscan has explorers for [ropsten](https://ropsten.etherscan.io/) and [mainnet](https://etherscan.io/) + - For Ethereum, Etherscan has explorers for [bloxberg](https://bloxberg.etherscan.io/) and [mainnet](https://etherscan.io/) - The transaction id is located in the Blockchain Certificate under `signature.anchors[0].sourceId` @@ -400,7 +400,7 @@ For an Ethereum transaction, you'll need to use a different explorer, which migh output. To view a transaction in a web browser, you might try something like this: - Ethereum Mainnet: https://etherscan.io/tx/0xf537d81667c8011e34e1f450e18fd1c5a8a10c770cd0acdc91a79746696f36a3 -- Ethereum Ropsten (testnet): https://ropsten.etherscan.io/tx/0xf537d81667c8011e34e1f450e18fd1c5a8a10c770cd0acdc91a79746696f36a3 +- Ethereum Bloxberg (testnet): https://bloxberg.etherscan.io/tx/0xf537d81667c8011e34e1f450e18fd1c5a8a10c770cd0acdc91a79746696f36a3 ## Mac scrypt problems diff --git a/cert_issuer/blockchain_handlers/ethereum/__init__.py b/cert_issuer/blockchain_handlers/ethereum/__init__.py index 6fbac259..5d2d3291 100644 --- a/cert_issuer/blockchain_handlers/ethereum/__init__.py +++ b/cert_issuer/blockchain_handlers/ethereum/__init__.py @@ -60,7 +60,7 @@ def instantiate_blockchain_handlers(app_config): if chain == Chain.mockchain: transaction_handler = MockTransactionHandler() # ethereum chains - elif chain == Chain.ethereum_mainnet or chain == Chain.ethereum_ropsten: + elif chain == Chain.ethereum_mainnet or chain == Chain.ethereum_bloxberg: cost_constants = EthereumTransactionCostConstants(app_config.gas_price, app_config.gas_limit) connector = EthereumServiceProviderConnector(chain, app_config.api_token) transaction_handler = EthereumTransactionHandler(connector, cost_constants, secret_manager, diff --git a/cert_issuer/blockchain_handlers/ethereum/connectors.py b/cert_issuer/blockchain_handlers/ethereum/connectors.py index cff82a4e..89b7ee6f 100644 --- a/cert_issuer/blockchain_handlers/ethereum/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum/connectors.py @@ -224,11 +224,11 @@ def get_address_nonce(self, address, api_token): eth_provider_list.append(MyEtherWalletBroadcaster('https://api.myetherwallet.com/eth')) connectors[Chain.ethereum_mainnet] = eth_provider_list -# Configure Ethereum Ropsten testnet connectors +# Configure Ethereum Bloxberg testnet connectors rop_provider_list = [] rop_provider_list.append(EtherscanBroadcaster('https://blockexplorer.bloxberg.org/api')) rop_provider_list.append(MyEtherWalletBroadcaster('https://api.myetherwallet.com/rop')) -connectors[Chain.ethereum_ropsten] = rop_provider_list +connectors[Chain.ethereum_bloxberg] = rop_provider_list def get_providers_for_chain(chain, local_node=False): diff --git a/cert_issuer/blockchain_handlers/ethereum/signer.py b/cert_issuer/blockchain_handlers/ethereum/signer.py index fe6358d2..e6619f6f 100644 --- a/cert_issuer/blockchain_handlers/ethereum/signer.py +++ b/cert_issuer/blockchain_handlers/ethereum/signer.py @@ -12,7 +12,7 @@ def __init__(self, ethereum_chain): # Netcode ensures replay protection (see EIP155) if ethereum_chain.external_display_value == 'ethereumMainnet': self.netcode = 1 - elif ethereum_chain.external_display_value == 'ethereumRopsten': + elif ethereum_chain.external_display_value == 'ethereumBloxberg': self.netcode = 3 else: self.netcode = None diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py b/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py index 755c34d3..cdd7eb79 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/__init__.py @@ -88,7 +88,7 @@ def instantiate_blockchain_handlers(app_config): if chain == Chain.mockchain: transaction_handler = MockTransactionHandler() # ethereum chains - elif chain == Chain.ethereum_mainnet or chain == Chain.ethereum_ropsten: + elif chain == Chain.ethereum_mainnet or chain == Chain.ethereum_bloxberg: cost_constants = EthereumTransactionCostConstants(app_config.gas_price, app_config.gas_limit) connector = instantiate_connector(app_config, cost_constants) transaction_handler = EthereumSCTransactionHandler(connector, cost_constants, secret_manager, diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/ens.py b/cert_issuer/blockchain_handlers/ethereum_sc/ens.py index 9edf1e76..7b62834e 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/ens.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/ens.py @@ -11,8 +11,8 @@ def __init__(self, app_config): self._w3 = Web3(HTTPProvider()) def get_registry_address(self): - if self.app_config.chain == Chain.ethereum_ropsten: - addr = self.app_config.ens_registry_ropsten + if self.app_config.chain == Chain.ethereum_bloxberg: + addr = self.app_config.ens_registry_bloxberg else: addr = self.app_config.ens_registry_mainnet diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/signer.py b/cert_issuer/blockchain_handlers/ethereum_sc/signer.py index 61a107aa..69c54584 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/signer.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/signer.py @@ -13,7 +13,7 @@ def __init__(self, ethereum_chain): # Netcode ensures replay protection (see EIP155) if ethereum_chain.external_display_value == 'ethereumMainnet': self.netcode = 8995 - elif ethereum_chain.external_display_value == 'ethereumRopsten': + elif ethereum_chain.external_display_value == 'ethereumBloxberg': self.netcode = 8995 else: self.netcode = None diff --git a/cert_issuer/config.py b/cert_issuer/config.py index 4a80200e..aa9790c7 100644 --- a/cert_issuer/config.py +++ b/cert_issuer/config.py @@ -53,7 +53,7 @@ def add_arguments(p): p.add_argument('--max_retry', default=10, type=int, help='Maximum attempts to retry transaction on failure', env_var='MAX_RETRY') p.add_argument('--chain', default='bitcoin_regtest', help=('Which chain to use. Default is bitcoin_regtest (which is how the docker container is configured). Other options are ' - 'bitcoin_testnet bitcoin_mainnet, mockchain, ethereum_mainnet, ethereum_ropsten'), env_var='CHAIN') + 'bitcoin_testnet bitcoin_mainnet, mockchain, ethereum_mainnet, ethereum_bloxberg'), env_var='CHAIN') p.add_argument('--safe_mode', dest='safe_mode', default=True, action='store_true', help='Used to make sure your private key is not plugged in with the wifi.', env_var='SAFE_MODE') @@ -90,10 +90,10 @@ def add_arguments(p): help='ens_name that points to the smart contract to which to issue', env_var='ENS_NAME') p.add_argument('--revocation_list_file', required=False, help='list of certificates or batches to be revokes', env_var='REVOCATION_LIST_FILE') - p.add_argument('--ens_registry_ropsten', required=False, default="0xde68Fcf6814D81Ee910bf35703622571718E07a7", - help='ENS registry address on ropsten', env_var='ENS_RESGISTRY_ROPSTEN') + p.add_argument('--ens_registry_bloxberg', required=False, default="0xde68Fcf6814D81Ee910bf35703622571718E07a7", + help='ENS registry address on bloxberg', env_var='ENS_RESGISTRY_BLOXBERG') p.add_argument('--ens_registry_mainnet', required=False, default="0xde68Fcf6814D81Ee910bf35703622571718E07a7", - help='ENS registry address on ropsten', env_var='ENS_RESGISTRY_MAINNET') + help='ENS registry address on bloxberg', env_var='ENS_RESGISTRY_MAINNET') def get_config(): configure_logger() @@ -110,11 +110,12 @@ def get_config(): parsed_config.chain = Chain.parse_from_chain(parsed_config.chain) # ensure it's a supported chain + """ if parsed_config.chain.blockchain_type != BlockchainType.bitcoin and \ parsed_config.chain.blockchain_type != BlockchainType.ethereum and \ parsed_config.chain.blockchain_type != BlockchainType.mock: raise UnknownChainError(parsed_config.chain.name) - + """ logging.info('This run will try to issue on the %s chain', parsed_config.chain.name) if parsed_config.chain.blockchain_type == BlockchainType.bitcoin: diff --git a/cert_issuer/helpers.py b/cert_issuer/helpers.py index 91718b4b..83953ec1 100644 --- a/cert_issuer/helpers.py +++ b/cert_issuer/helpers.py @@ -109,8 +109,8 @@ def tx_to_blink(chain, tx_id): blink += 'btc:testnet:' elif chain == Chain.bitcoin_mainnet: blink += 'btc:mainnet:' - elif chain == Chain.ethereum_ropsten: - blink += 'eth:ropsten:' + elif chain == Chain.ethereum_bloxberg: + blink += 'eth:bloxberg:' elif chain == Chain.ethereum_mainnet: blink += 'eth:mainnet:' elif chain == Chain.mockchain: diff --git a/cert_issuer/issue_certificates.py b/cert_issuer/issue_certificates.py index 4b506e4b..deda492a 100644 --- a/cert_issuer/issue_certificates.py +++ b/cert_issuer/issue_certificates.py @@ -38,7 +38,7 @@ def revoke_certificates(app_config, transaction_handler): def main(app_config): chain = app_config.chain - if chain == Chain.ethereum_mainnet or chain == Chain.ethereum_ropsten: + if chain == Chain.ethereum_mainnet or chain == Chain.ethereum_bloxberg: if app_config.issuing_method == "smart_contract": from cert_issuer.blockchain_handlers import ethereum_sc certificate_batch_handler, transaction_handler, connector = ethereum_sc.instantiate_blockchain_handlers(app_config) diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index f96af55a..081f82fe 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -135,7 +135,7 @@ def to_source_id(txid, chain): # workaround return txid # previously the == operator to actually compare with 'chain' was missing - this caused the below text to be returned, breaking the tests - if chain == Chain.bitcoin_mainnet or chain == Chain.bitcoin_testnet or chain == Chain.ethereum_mainnet or chain == Chain.ethereum_ropsten: + if chain == Chain.bitcoin_mainnet or chain == Chain.bitcoin_testnet or chain == Chain.ethereum_mainnet or chain == Chain.ethereum_bloxberg: return txid else: return 'This has not been issued on a blockchain and is for testing only' diff --git a/docs/ethereum_smart_contract.md b/docs/ethereum_smart_contract.md index 8fe2335e..8270604f 100644 --- a/docs/ethereum_smart_contract.md +++ b/docs/ethereum_smart_contract.md @@ -62,8 +62,8 @@ Configuration is done via the `conf_eth.ini` file. Cert-deployer's configuration `deploying_address = ` The ethereum account's address. -`chain = ` -Choice of deployment on Ethereum Mainnet or the Ropsten test network. +`chain = ` +Choice of deployment on Ethereum Mainnet or the Bloxberg test network. `node_url = ` The web3py library requires an ethereum node that is compatible with the json-rpc interface. The easiest option is to use a public node e.g. infura’s, or connect to a locally-run node such as geth or parity. diff --git a/ethereum_smart_contract_requirements.txt b/ethereum_smart_contract_requirements.txt index 7a96b013..cbdc14f8 100644 --- a/ethereum_smart_contract_requirements.txt +++ b/ethereum_smart_contract_requirements.txt @@ -1,11 +1,10 @@ coincurve==7.1.0 web3>=5 - ipfshttpclient -lds-merkle-proof-2019 configargparse==0.12.0 -cert-core>=2.1.9 -cert-schema>=2.1.5 +lds-merkle-proof-2019-bloxberg +cert-core-bloxberg==2.1.2 +cert-schema chainpoint3>=0.0.2 configargparse==0.12.0 glob2==0.6 diff --git a/requirements.txt b/requirements.txt index fd717916..0fd28a51 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,4 +1,4 @@ -cert-core>=2.1.9 +cert-core-bloxberg==2.1.2 cert-schema>=2.1.5 chainpoint3>=0.0.2 configargparse==0.12.0 diff --git a/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json b/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json new file mode 100644 index 00000000..f175368d --- /dev/null +++ b/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:15186643-ec29-4055-bebc-aa464d928849", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:35.956897", "proofValue": "z65N6GBZC3gTCaERU11XMMh9LYfKrT5n29yM87W2recMTB7wmPxH2BB9mroHqganzTNnt6qxV922pCfacfb8z2cLu4V67vkJXBME9EksBapcD6vQKed1vcovX7h9w1XAMgJqJNRjYCw4kdFUdjAQR4MWbZkYphneJn1D5zfA4X2MkcSkj6pV8BXbEQ49ZZ5ugoCZew2t7aXEQES7mKGrmbPqTpXMrjGdtVwrMqeH6FU7QDjEzs2wJCmZV3jL42PpgbYSvFVY6KXCK2Z3P4iTWZYwL3DLb46Txy6ukPXwJznfTt2n1fybDFGHUhZJKEw5qDZPjUTY3V8VbnH3ySxofxxST39PTQStrbKTHNV27qY4wmK73buZzwmGjPj1t52AfEtx1TEXeFbtFgQCuxTFsTceCuWC8rgp9NrogVvTuZmukYBGUhr7sHTNQDD5wDmL71RcBtVmSmHqy1bE3VD9EB6wjgZKEYZGoF2BTRyLerbwVv1UxmTfXh4T", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json b/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json new file mode 100644 index 00000000..217b2388 --- /dev/null +++ b/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:1f0d1af0-4c14-4836-ade3-0df9facb3ced", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.179793", "proofValue": "z65N6GBZByDiB7gLEoeUugVq4yyHARNdri8J9zNzUMpaSAW1WQg7wVLQjEwVWn6jgLjJjEgqNaapzty73k922PVp82VpC7w9fkVhwHzrcohsj1akZ5V3RLfJgsdoz7ZvdpnPQYd2yb6BuzbrZHo9YBgR7CdBKJp6ZQP6sQdhpEmGPnzweNvxXSP672xBXZo3fZzbhcNBKJQZeGuJBqbdnbn5xQgevnxhE37LyTKomxQnpMZS3fDLYpMqazGhdLkTqzaUujMBoCwy6ZFnpxT5HMLMSUegZKC39KBSHHF5GWVuNhTSGVdRN7cvVan46vmNW9rChv5YaWsbK8RD3a2G8HSLHc7BZ1DANyzdJFRDHj4wVRWLgy9hscCZKmRvENY7rcqZyaTHfC6zAZQV7R5oikr4QpFMc9uVav72zBDfGkfTR7fnDc2mTdhXSinJJsng1Swci57TuVEaf5emnGUDdZf9vKsQV4ji5SfDQsZ2z9PxG3xNnuzcmSsq", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json b/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json deleted file mode 100644 index 8fa6a469..00000000 --- a/work/blockchain_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:35b6c347-51e4-4881-96bf-287ffbccdfce", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-30T12:36:26.247132", "proofValue": "z2LuLBVSfogU8YhUjfqryx2c8kYrVZ6jAJz2XGxy12hmXpzP2u1h5aWhceYdMwZQM9YqD1Kk7qW3WLArFSp1xgSGquLRThhk15ou5FkXtEDHtGCV4HE4aLxYjcfQX8DnS5GM1gA9npS9SjQ5UDaHMdh8qvqQA4gPumEMWLhVavs7PqtsqJpaDozD7XAkyv1RVfQVBW9EcJsF22KBAWKZvrUnU2jhzGBojt19Av2AsEp7rJM6HeW6dPRx3cKKmSEZqNUCogWBuoUNhsMiD3hmo9dBdHTcnnDhgrcCQ2hjgfa98ftN6NeV25TTe7nHCqLyQ9yWaV2ShkKg6ah7ma6347h3eEmcbkPqSctxZQ6dMQoXqRDAcmM66JYnnWttWXwAGhMPYGqU33Ys5Uwr1qvGfb", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json b/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json deleted file mode 100644 index b4ce0e12..00000000 --- a/work/blockchain_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:7f2e45da-be39-4f4d-86ed-15ea4d0512bf", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-07-30T12:36:26.404988", "proofValue": "z2LuLBVSfnVzaQtw5CHiYTk8VQf8pSv2ppMr3tnknHhNQFFDTA1KMRa8kLyeXj1MogtTaDdCmfd74QBzKff9vcJsHvxYmbUQPBqdAcpfD5oM8zo8XiXGQd9eXfPh1wkNhwngkt2XGDKP2T2fewubVZRAyNDBFgKwuUThraP8KDvzEjMXyxkjA69Y9ukXTR25FL7znrkbsHy2yP13WV87RkpmNb1NQuG8sWkV4RAu9aXh79RoSmFz8Ty91HPXRkc7TwesgqHQMLvFMeZY3ZoLA3fvzfNY3xaUiPcuceUxdmjuC2i8NQZuf6j84YeZT3aY2gsMsMtgdgJxqWH3GqPwSEQu1SFZ6dmhvDF5kmHUkPD3esGtXiGc898Xn2f9RtL4b5kihT3oFqFfAffPzSDN9F", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json b/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json new file mode 100644 index 00000000..78d061a2 --- /dev/null +++ b/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:a2a99a9e-457b-4de9-930b-3146442275a8", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.387999", "proofValue": "z65N6GBZC3gTCaEYPUvvvYq6o1PFB7EKAW5cTVe58KTaNHLkkHkZ2he7UrP7RgxqfXyMRtVA1DdLLnj4zWtweJ5rkVGtuvpsjfz5jndcehadaxvZe9TqxVZDjgrxrtLcRwyXvhnS8Li8jYcb2nefsAvPKVgkhLMNu3iJDLxakdPB5xbMQhStMTFeoGFYgJtEFuXK17zMzeK9yrTxx96rVhefiZHLem6oMSgDDJDxjJQ13Wqt9T8aShTkmsKh1iT37vHjubsfQD73SFMQcpbkbcyCPM2LsbFVqHphECMyLN5dmhTkPciJdhtqao7XFP52W47sQeSiEnT1oar2Wo7NvEse1yxUDkwv8hUFUHpUBU1BgxqjKqVMAkGcjT45VBRKGdcWbwZj73LKsYAFy2nSQWDHT73eeZfDNSga7Mo8S2hNGkKjVprstLuRaXx4KQ8WLtB9so6bqWddS3yWmurFF2U7Bcim99DE6BShrjKjoLve6ij5TK6A8Dhq", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json b/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json new file mode 100644 index 00000000..17dc5dfc --- /dev/null +++ b/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:c7494b9c-cc9c-4740-b7b8-9037f3cd64ec", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.533486", "proofValue": "z65N6GBZByDiB7gKxdycoYqEhCB8kmgsvjzyQiNfe8x9sppWwf3aaBAu9T3jrkhKUigceUycsRtrcaouqCNBDMUCMpFpN376pUSsZPkNPjHgkSUumzKihYEimm2gS2S2w5wXDmjaLrLg9KQ9jSTxnZbFf4h9bTJxQqj33MNcar4hkojJrgPR8kSHRCrRx7jWfRWyzCekXWFB5U8d9bB9aip2apBNwDFGA282Ce15hTYRjdd8FVjFnstFNLwthYeABnk9VZFBhKZFswDhBGHZYbrf3XDALNMrJdxrdm8CZZTe1Hpqp1iJJHo7WzmTuRR2HwxWWoxg46UhTBhut7ZR2YCW3Kx7XgtDQ5ZqseWhqucjy4dYFsEXgUKVFSCsS7PYXLqiscoUc6Z2DSHmRpTQy8ZAhjLcLnWpyW6uxWLZeqx2RCFSJNJs34n2ZYdVnn9jnFnA1m3epnyt68KqgLqaS66WLB3hrciXssjYjxqF22gXBeAX9bGriR3D", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json b/work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json similarity index 63% rename from work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json rename to work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json index 425f6de7..bd7943dc 100644 --- a/work/unsigned_certificates/35b6c347-51e4-4881-96bf-287ffbccdfce.json +++ b/work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json @@ -1 +1 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:35b6c347-51e4-4881-96bf-287ffbccdfce", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:15186643-ec29-4055-bebc-aa464d928849", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json b/work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json similarity index 63% rename from work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json rename to work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json index 76097d1e..ad0c45cb 100644 --- a/work/unsigned_certificates/7f2e45da-be39-4f4d-86ed-15ea4d0512bf.json +++ b/work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json @@ -1 +1 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:7f2e45da-be39-4f4d-86ed-15ea4d0512bf", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-07-30T12:04:01.829304+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:1f0d1af0-4c14-4836-ade3-0df9facb3ced", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json b/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json new file mode 100644 index 00000000..213f0f4a --- /dev/null +++ b/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:a2a99a9e-457b-4de9-930b-3146442275a8", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json b/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json new file mode 100644 index 00000000..ad06da4d --- /dev/null +++ b/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:c7494b9c-cc9c-4740-b7b8-9037f3cd64ec", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file From a8b3a7286f61d1fe33d208fdbdd7e9b20cd8557f Mon Sep 17 00:00:00 2001 From: James Date: Thu, 20 Aug 2020 14:45:19 +0200 Subject: [PATCH 19/22] updated proof mapping and initial IPFS functions --- .../blockchain_handlers/ethereum_sc/connectors.py | 8 +++++++- .../ethereum_sc/transaction_handlers.py | 14 ++++++++++++++ cert_issuer/issuer.py | 8 ++++++++ cert_issuer/merkle_tree_generator.py | 7 +++++-- .../15186643-ec29-4055-bebc-aa464d928849.json | 1 - .../1f0d1af0-4c14-4836-ade3-0df9facb3ced.json | 1 - .../96e4392d-461e-473c-b751-60f0dc408136.json | 1 + .../a2a99a9e-457b-4de9-930b-3146442275a8.json | 1 - .../bd2a927d-9374-4b5a-a923-e52aa215a4f7.json | 1 + .../c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json | 1 - .../15186643-ec29-4055-bebc-aa464d928849.json | 1 - .../1f0d1af0-4c14-4836-ade3-0df9facb3ced.json | 1 - .../96e4392d-461e-473c-b751-60f0dc408136.json | 1 + .../a2a99a9e-457b-4de9-930b-3146442275a8.json | 1 - .../bd2a927d-9374-4b5a-a923-e52aa215a4f7.json | 1 + .../c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json | 1 - 16 files changed, 38 insertions(+), 11 deletions(-) delete mode 100644 work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json delete mode 100644 work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json create mode 100644 work/blockchain_certificates/96e4392d-461e-473c-b751-60f0dc408136.json delete mode 100644 work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json create mode 100644 work/blockchain_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json delete mode 100644 work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json delete mode 100644 work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json delete mode 100644 work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json create mode 100644 work/unsigned_certificates/96e4392d-461e-473c-b751-60f0dc408136.json delete mode 100644 work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json create mode 100644 work/unsigned_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json delete mode 100644 work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py index 5caa4e0b..d68a346c 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py @@ -52,7 +52,7 @@ def create_transaction(self, method, *argv): gas_limit = self.cost_constants.get_gas_limit() estimated_gas = self._contract_obj.functions[method](*argv).estimateGas() * 5 if estimated_gas > gas_limit: - logging.warning("Estimated gas of %s more than gas limit of %s, transaction might fail. Please verify on etherescan.com.", estimated_gas, gas_limit) + logging.warning("Estimated gas of %s more than gas limit of %s, transaction might fail. Please verify on blockexplorer.bloxberg.org.", estimated_gas, gas_limit) estimated_gas = gas_limit gas_price = self._w3.eth.gasPrice @@ -81,6 +81,12 @@ def broadcast_tx(self, signed_tx): tx_receipt = self._w3.eth.waitForTransactionReceipt(tx_hash) return tx_receipt.transactionHash.hex() + def get_tokenID(self, txid): + tx_receipt = receipt = self._w3.eth.getTransactionReceipt(txid) + logs = self._contract_obj.events.Transfer().processReceipt(tx_receipt) + tokenID = logs[0]['args']['tokenId'] + return tokenID + def transact(self, method, *argv): """ Sends a signed transaction on the blockchain and waits for a response. diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py index 92442af2..4a0bb236 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py @@ -45,6 +45,18 @@ def make_transaction(self, recipient_address, token_uri, blockchain_bytes, app_c txid = self.broadcast_transaction(signed_tx) return txid + def ipfs_transaction(self, token_id, token_uri, blockchain_bytes, app_config): + return self.update_ipfs(token_id, token_uri, blockchain_bytes, app_config, "updateTokenURI") + + def update_ipfs(self, token_id, token_uri, blockchain_bytes, app_config): + prepared_tx = self.connector.create_transaction(method, token_id, token_uri, blockchain_bytes) + signed_tx = self.sign_transaction(prepared_tx) + + logging.info('Updating tokenURI field with ipfs link..') + + txid = self.broadcast_transaction(signed_tx) + return txid + def sign_transaction(self, prepared_tx): # stubbed from BitcoinTransactionHandler @@ -57,3 +69,5 @@ def sign_transaction(self, prepared_tx): def broadcast_transaction(self, signed_tx): txid = self.connector.broadcast_tx(signed_tx) return txid + + diff --git a/cert_issuer/issuer.py b/cert_issuer/issuer.py index 77141983..afb1589d 100644 --- a/cert_issuer/issuer.py +++ b/cert_issuer/issuer.py @@ -34,6 +34,7 @@ def issue(self, chain, app_config, recipient_address): txid = self.transaction_handler.issue_transaction(recipient_address, token_uri, blockchain_bytes, app_config) self.certificate_batch_handler.finish_batch(txid, chain, app_config) logging.info('Broadcast transaction with txid %s', txid) + return txid except BroadcastError: logging.warning( @@ -41,3 +42,10 @@ def issue(self, chain, app_config, recipient_address): attempt_number) logging.error('All attempts to broadcast failed. Try rerunning issuer.') raise BroadcastError('All attempts to broadcast failed. Try rerunning issuer.') + + def updateIPFS(self, chain, app_config): + """ + Update IPFS link on smart contract + """ + txid = self.transaction_handler.ipfs_transaction(token_id, token_uri, blockchain_bytes, app_config) + diff --git a/cert_issuer/merkle_tree_generator.py b/cert_issuer/merkle_tree_generator.py index 081f82fe..b7c49386 100644 --- a/cert_issuer/merkle_tree_generator.py +++ b/cert_issuer/merkle_tree_generator.py @@ -75,6 +75,7 @@ def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chai abi = ens.get_abi() mp2019 = MerkleProof2019() + print(helpers.tx_to_blink(chain, tx_id)) merkle_json = { "path": proof2, "merkleRoot": root, @@ -83,7 +84,7 @@ def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chai "anchors": [ helpers.tx_to_blink(chain, tx_id) ] - } + } logging.info('merkle_json: %s', str(merkle_json)) proof_value = mp2019.encode(merkle_json) @@ -92,7 +93,9 @@ def get_proof_generator(self, tx_id, app_config, verification_method, chain=Chai "created": datetime.now().isoformat(), "proofValue": proof_value.decode('utf8'), "proofPurpose": "assertionMethod", - "verificationMethod": verification_method + "verificationMethod": verification_method, + #Add ENS name for issuer validation + "ens_name": app_config.ens_name } #Uncomment after checking verification """ diff --git a/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json b/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json deleted file mode 100644 index f175368d..00000000 --- a/work/blockchain_certificates/15186643-ec29-4055-bebc-aa464d928849.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:15186643-ec29-4055-bebc-aa464d928849", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:35.956897", "proofValue": "z65N6GBZC3gTCaERU11XMMh9LYfKrT5n29yM87W2recMTB7wmPxH2BB9mroHqganzTNnt6qxV922pCfacfb8z2cLu4V67vkJXBME9EksBapcD6vQKed1vcovX7h9w1XAMgJqJNRjYCw4kdFUdjAQR4MWbZkYphneJn1D5zfA4X2MkcSkj6pV8BXbEQ49ZZ5ugoCZew2t7aXEQES7mKGrmbPqTpXMrjGdtVwrMqeH6FU7QDjEzs2wJCmZV3jL42PpgbYSvFVY6KXCK2Z3P4iTWZYwL3DLb46Txy6ukPXwJznfTt2n1fybDFGHUhZJKEw5qDZPjUTY3V8VbnH3ySxofxxST39PTQStrbKTHNV27qY4wmK73buZzwmGjPj1t52AfEtx1TEXeFbtFgQCuxTFsTceCuWC8rgp9NrogVvTuZmukYBGUhr7sHTNQDD5wDmL71RcBtVmSmHqy1bE3VD9EB6wjgZKEYZGoF2BTRyLerbwVv1UxmTfXh4T", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json b/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json deleted file mode 100644 index 217b2388..00000000 --- a/work/blockchain_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:1f0d1af0-4c14-4836-ade3-0df9facb3ced", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.179793", "proofValue": "z65N6GBZByDiB7gLEoeUugVq4yyHARNdri8J9zNzUMpaSAW1WQg7wVLQjEwVWn6jgLjJjEgqNaapzty73k922PVp82VpC7w9fkVhwHzrcohsj1akZ5V3RLfJgsdoz7ZvdpnPQYd2yb6BuzbrZHo9YBgR7CdBKJp6ZQP6sQdhpEmGPnzweNvxXSP672xBXZo3fZzbhcNBKJQZeGuJBqbdnbn5xQgevnxhE37LyTKomxQnpMZS3fDLYpMqazGhdLkTqzaUujMBoCwy6ZFnpxT5HMLMSUegZKC39KBSHHF5GWVuNhTSGVdRN7cvVan46vmNW9rChv5YaWsbK8RD3a2G8HSLHc7BZ1DANyzdJFRDHj4wVRWLgy9hscCZKmRvENY7rcqZyaTHfC6zAZQV7R5oikr4QpFMc9uVav72zBDfGkfTR7fnDc2mTdhXSinJJsng1Swci57TuVEaf5emnGUDdZf9vKsQV4ji5SfDQsZ2z9PxG3xNnuzcmSsq", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/96e4392d-461e-473c-b751-60f0dc408136.json b/work/blockchain_certificates/96e4392d-461e-473c-b751-60f0dc408136.json new file mode 100644 index 00000000..c6e0aa06 --- /dev/null +++ b/work/blockchain_certificates/96e4392d-461e-473c-b751-60f0dc408136.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:96e4392d-461e-473c-b751-60f0dc408136", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-20T12:14:10.363378+00:00", "credentialSubject": {"id": "einstein@mpg.de", "alumniOf": {"id": "https://bloxberg.org"}}, "SHA256Hash": "0x0e4ded5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6", "proof": {"type": "MerkleProof2019", "created": "2020-08-20T12:15:50.473528", "proofValue": "zTAfB64R9FnirA3SRy33KqoBNysBwHuFKYP2P5kyMMWHX6hyyMZgJJRzzPjQRMwtCPxZNyHR8eeCwXCx8gRPP1cAccZLEotjmkQYVUNfUGC92fnJBuMq5BatkuGV9GZsszwzevfEGtXdCVhXmMxFevDhS8ELPQBHoNS7B4RFmKBNGjDJmpbrARY7iDC9hQLdWqFQmX6ucyGUNKFmibBeKWNNkGuboVZpzDsokDmsRAwFjDk3vbb4EZsTtwFq4h8WsqmWZGsVBKBasL5fgY9eegoVdHFeaVG9mxy6CRtBVSSQcaXPQJ3D7isPA8PCT5k6eb4ZpHCUvrug3zh181nPFci6FomhtyPD7mgCQiDtMjQNpFMUiLYswSHFA2EWx1iCBg6gCkUodLCvKEpQcWid7LhQ", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e", "ens_name": "mpdl.berg"}} \ No newline at end of file diff --git a/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json b/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json deleted file mode 100644 index 78d061a2..00000000 --- a/work/blockchain_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:a2a99a9e-457b-4de9-930b-3146442275a8", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.387999", "proofValue": "z65N6GBZC3gTCaEYPUvvvYq6o1PFB7EKAW5cTVe58KTaNHLkkHkZ2he7UrP7RgxqfXyMRtVA1DdLLnj4zWtweJ5rkVGtuvpsjfz5jndcehadaxvZe9TqxVZDjgrxrtLcRwyXvhnS8Li8jYcb2nefsAvPKVgkhLMNu3iJDLxakdPB5xbMQhStMTFeoGFYgJtEFuXK17zMzeK9yrTxx96rVhefiZHLem6oMSgDDJDxjJQ13Wqt9T8aShTkmsKh1iT37vHjubsfQD73SFMQcpbkbcyCPM2LsbFVqHphECMyLN5dmhTkPciJdhtqao7XFP52W47sQeSiEnT1oar2Wo7NvEse1yxUDkwv8hUFUHpUBU1BgxqjKqVMAkGcjT45VBRKGdcWbwZj73LKsYAFy2nSQWDHT73eeZfDNSga7Mo8S2hNGkKjVprstLuRaXx4KQ8WLtB9so6bqWddS3yWmurFF2U7Bcim99DE6BShrjKjoLve6ij5TK6A8Dhq", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/blockchain_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json b/work/blockchain_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json new file mode 100644 index 00000000..db11e3fe --- /dev/null +++ b/work/blockchain_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:bd2a927d-9374-4b5a-a923-e52aa215a4f7", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-20T12:14:10.363378+00:00", "credentialSubject": {"id": "einstein@mpg.de", "alumniOf": {"id": "https://bloxberg.org"}}, "SHA256Hash": "0xfda3124d5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6", "proof": {"type": "MerkleProof2019", "created": "2020-08-20T12:15:50.692579", "proofValue": "zTAfB64R8rnijzFj9xZceVEQsH7yrpVvrc2bDidHKGYJKPpe1K3dtYgspAdJDDsNyd3yNnbrAAM1ZXwqVzCYvcAtHhqVxFxt6v4PXrCdbF6hCaFed3tQoUwhf8yXLAhWHNtjaSYawFBvyTMSJZSeYRQqXY8iedcrDb1Qx646TsBkGMaSVz7jhTNJt6Eeo6Jq1tN2Z7kR7WCCmqBNyjBZw7K39Lvw6n9qBTuNsm88LEbMXnb8ESEbRccQeUqJJXHXmhjDshuvbhpHGfHmJZ31AMEV33qmXzipsdAD4vKhdVRjnovm1TYkgnzNWg8Wv37Smi6oXySvMDjhptE5XTYLZNxuXyAh9NMdKrnEJeAXBNHvqS3ayARHemaTBGqyGJFxJNh14xX8FkKtnQTNwbu98Dcn", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e", "ens_name": "mpdl.berg"}} \ No newline at end of file diff --git a/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json b/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json deleted file mode 100644 index 17dc5dfc..00000000 --- a/work/blockchain_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:c7494b9c-cc9c-4740-b7b8-9037f3cd64ec", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}, "proof": {"type": "MerkleProof2019", "created": "2020-08-11T14:39:36.533486", "proofValue": "z65N6GBZByDiB7gKxdycoYqEhCB8kmgsvjzyQiNfe8x9sppWwf3aaBAu9T3jrkhKUigceUycsRtrcaouqCNBDMUCMpFpN376pUSsZPkNPjHgkSUumzKihYEimm2gS2S2w5wXDmjaLrLg9KQ9jSTxnZbFf4h9bTJxQqj33MNcar4hkojJrgPR8kSHRCrRx7jWfRWyzCekXWFB5U8d9bB9aip2apBNwDFGA282Ce15hTYRjdd8FVjFnstFNLwthYeABnk9VZFBhKZFswDhBGHZYbrf3XDALNMrJdxrdm8CZZTe1Hpqp1iJJHo7WzmTuRR2HwxWWoxg46UhTBhut7ZR2YCW3Kx7XgtDQ5ZqseWhqucjy4dYFsEXgUKVFSCsS7PYXLqiscoUc6Z2DSHmRpTQy8ZAhjLcLnWpyW6uxWLZeqx2RCFSJNJs34n2ZYdVnn9jnFnA1m3epnyt68KqgLqaS66WLB3hrciXssjYjxqF22gXBeAX9bGriR3D", "proofPurpose": "assertionMethod", "verificationMethod": "ecdsa-koblitz-pubkey:0xD748BF41264b906093460923169643f45BDbC32e"}} \ No newline at end of file diff --git a/work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json b/work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json deleted file mode 100644 index bd7943dc..00000000 --- a/work/unsigned_certificates/15186643-ec29-4055-bebc-aa464d928849.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:15186643-ec29-4055-bebc-aa464d928849", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json b/work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json deleted file mode 100644 index ad0c45cb..00000000 --- a/work/unsigned_certificates/1f0d1af0-4c14-4836-ade3-0df9facb3ced.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:1f0d1af0-4c14-4836-ade3-0df9facb3ced", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/96e4392d-461e-473c-b751-60f0dc408136.json b/work/unsigned_certificates/96e4392d-461e-473c-b751-60f0dc408136.json new file mode 100644 index 00000000..8c7deda5 --- /dev/null +++ b/work/unsigned_certificates/96e4392d-461e-473c-b751-60f0dc408136.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:96e4392d-461e-473c-b751-60f0dc408136", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-20T12:14:10.363378+00:00", "credentialSubject": {"id": "einstein@mpg.de", "alumniOf": {"id": "https://bloxberg.org"}}, "SHA256Hash": "0x0e4ded5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6"} \ No newline at end of file diff --git a/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json b/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json deleted file mode 100644 index 213f0f4a..00000000 --- a/work/unsigned_certificates/a2a99a9e-457b-4de9-930b-3146442275a8.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:a2a99a9e-457b-4de9-930b-3146442275a8", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T12:48:28.879555+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file diff --git a/work/unsigned_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json b/work/unsigned_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json new file mode 100644 index 00000000..ff25841a --- /dev/null +++ b/work/unsigned_certificates/bd2a927d-9374-4b5a-a923-e52aa215a4f7.json @@ -0,0 +1 @@ +{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:bd2a927d-9374-4b5a-a923-e52aa215a4f7", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-20T12:14:10.363378+00:00", "credentialSubject": {"id": "einstein@mpg.de", "alumniOf": {"id": "https://bloxberg.org"}}, "SHA256Hash": "0xfda3124d5319861c8daac00d425c53a16bd180a7d01a340a0e00f7dede40d2c9f6"} \ No newline at end of file diff --git a/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json b/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json deleted file mode 100644 index ad06da4d..00000000 --- a/work/unsigned_certificates/c7494b9c-cc9c-4740-b7b8-9037f3cd64ec.json +++ /dev/null @@ -1 +0,0 @@ -{"@context": ["https://www.w3.org/2018/credentials/v1", "https://w3id.org/blockcerts/schema/3.0-alpha/context.json"], "id": "urn:uuid:c7494b9c-cc9c-4740-b7b8-9037f3cd64ec", "type": ["VerifiableCredential", "BlockcertsCredential"], "issuer": "https://raw.githubusercontent.com/AnthonyRonning/https-github.com-labnol-files/master/issuer-eth.json", "issuanceDate": "2020-08-10T07:40:30.304362+00:00", "credentialSubject": {"id": "did:key:z6Mkq3L1jEDDZ5R7eT523FMLxC4k6MCpzqD7ff1CrkWpoJwM", "alumniOf": {"id": "https://www.issuer.org"}}} \ No newline at end of file From 05ffa02488dc2b28ad5c656a36e031e592a80edc Mon Sep 17 00:00:00 2001 From: Andrei Ionita Date: Tue, 25 Aug 2020 13:18:23 +0200 Subject: [PATCH 20/22] added certificate validation for v3 --- cert_issuer/certificate_handlers.py | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/cert_issuer/certificate_handlers.py b/cert_issuer/certificate_handlers.py index b82dc050..7b1f7bb3 100644 --- a/cert_issuer/certificate_handlers.py +++ b/cert_issuer/certificate_handlers.py @@ -1,7 +1,7 @@ import json import logging -from cert_schema import normalize_jsonld +from cert_schema import normalize_jsonld, validate_v3_alpha from cert_issuer import helpers from pycoin.serialize import b2h from cert_issuer.models import CertificateHandler, BatchHandler @@ -31,6 +31,15 @@ def _get_certificate_to_issue(self, certificate_metadata): certificate_json = json.load(unsigned_cert_file) return certificate_json + def validate_certificate(self, certificate_metadata): + certificate_json = self._get_certificate_to_issue(certificate_metadata) + return validate_v3_alpha(certificate_json) + + def sign_certificate(self, signer, certificate_metadata): + # TODO + return self.sign_certificate(signer, certificate_metadata) + + class CertificateWebV3Handler(CertificateHandler): def get_byte_array_to_issue(self, certificate_json): normalized = normalize_jsonld(certificate_json, detect_unmapped_fields=False) @@ -45,6 +54,14 @@ def add_proof(self, certificate_json, merkle_proof): certificate_json['signature'] = merkle_proof return certificate_json + def validate_certificate(self, certificate_metadata): + certificate_json = self._get_certificate_to_issue(certificate_metadata) + return validate_v3_alpha(certificate_json) + + def sign_certificate(self, signer, certificate_metadata): + # TODO + return self.sign_certificate(signer, certificate_metadata) + class CertificateBatchWebHandler(BatchHandler): #Smart contract version has app_config as variable - removed in V3 From f2d4efe959f89c72d0f93f31693decfca2197f72 Mon Sep 17 00:00:00 2001 From: Andrei Ionita Date: Tue, 25 Aug 2020 13:33:45 +0200 Subject: [PATCH 21/22] added ipfs_batch_file in conf --- conf_regtest.ini | 2 ++ conf_template.ini | 5 ++++- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/conf_regtest.ini b/conf_regtest.ini index 08a7103c..f30ecde0 100644 --- a/conf_regtest.ini +++ b/conf_regtest.ini @@ -10,4 +10,6 @@ unsigned_certificates_dir=/etc/cert-issuer/data/unsigned_certificates blockchain_certificates_dir=/etc/cert-issuer/data/blockchain_certificates work_dir=/etc/cert-issuer/work +ipfs_batch_file=ipfsBatch.json + no_safe_mode diff --git a/conf_template.ini b/conf_template.ini index 07c6d1a7..773ef370 100644 --- a/conf_template.ini +++ b/conf_template.ini @@ -15,8 +15,11 @@ work_dir= usb_name = key_file = +# a list of the IPFS hashes of the uploded certificates +ipfs_batch_file= + # which blockchain; bitcoin_regtest is default chain= # this disables the wifi check, and should only be used recommended during testing -no_safe_mode \ No newline at end of file +no_safe_mode From bba566720a178c825f0e6827d6341e381f51055c Mon Sep 17 00:00:00 2001 From: Andrei Ionita Date: Tue, 25 Aug 2020 13:35:17 +0200 Subject: [PATCH 22/22] added updating of token uri with ipfs batch file --- .../ethereum_sc/connectors.py | 12 ++++----- .../ethereum_sc/transaction_handlers.py | 24 ++++++------------ cert_issuer/issue_certificates.py | 10 ++++++-- cert_issuer/issuer.py | 25 +++++++++++++++---- 4 files changed, 42 insertions(+), 29 deletions(-) diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py index d68a346c..564a62aa 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/connectors.py @@ -81,12 +81,6 @@ def broadcast_tx(self, signed_tx): tx_receipt = self._w3.eth.waitForTransactionReceipt(tx_hash) return tx_receipt.transactionHash.hex() - def get_tokenID(self, txid): - tx_receipt = receipt = self._w3.eth.getTransactionReceipt(txid) - logs = self._contract_obj.events.Transfer().processReceipt(tx_receipt) - tokenID = logs[0]['args']['tokenId'] - return tokenID - def transact(self, method, *argv): """ Sends a signed transaction on the blockchain and waits for a response. @@ -113,3 +107,9 @@ def _sign_transaction(self, prepared_tx): def call(self, method, *argv): return self._contract_obj.functions[method](*argv).call() + + def get_event_args(self, tx_hash, event): + tx_receipt = self._w3.eth.getTransactionReceipt(tx_hash) + logs = self._contract_obj.events[event]().processReceipt(tx_receipt) + return logs[0]['args'] + diff --git a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py index 4a0bb236..8c388519 100644 --- a/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py +++ b/cert_issuer/blockchain_handlers/ethereum_sc/transaction_handlers.py @@ -34,25 +34,16 @@ def revoke_transaction(self, blockchain_bytes, app_config): # return self.make_transaction(blockchain_bytes, app_config, "issue_hash") def issue_transaction(self, recipient_address, token_uri, blockchain_bytes, app_config): - return self.make_transaction(recipient_address, token_uri, blockchain_bytes, app_config, "createCertificate") + return self.make_transaction(app_config, "createCertificate", recipient_address, token_uri, blockchain_bytes) - def make_transaction(self, recipient_address, token_uri, blockchain_bytes, app_config, method): - prepared_tx = self.connector.create_transaction(method, recipient_address, token_uri, blockchain_bytes) - signed_tx = self.sign_transaction(prepared_tx) - - logging.info('Broadcasting transaction to the blockchain...') + def update_token_uri(self, token_id, token_uri, app_config): + return self.make_transaction(app_config, "updateTokenURI", token_id, token_uri) - txid = self.broadcast_transaction(signed_tx) - return txid - - def ipfs_transaction(self, token_id, token_uri, blockchain_bytes, app_config): - return self.update_ipfs(token_id, token_uri, blockchain_bytes, app_config, "updateTokenURI") - - def update_ipfs(self, token_id, token_uri, blockchain_bytes, app_config): - prepared_tx = self.connector.create_transaction(method, token_id, token_uri, blockchain_bytes) + def make_transaction(self, app_config, method, *argv): + prepared_tx = self.connector.create_transaction(method, *argv) signed_tx = self.sign_transaction(prepared_tx) - logging.info('Updating tokenURI field with ipfs link..') + logging.info('Broadcasting transaction to the blockchain...') txid = self.broadcast_transaction(signed_tx) return txid @@ -70,4 +61,5 @@ def broadcast_transaction(self, signed_tx): txid = self.connector.broadcast_tx(signed_tx) return txid - + def get_event_args(self, tx_hash, event): + return self.connector(tx_hash, event) diff --git a/cert_issuer/issue_certificates.py b/cert_issuer/issue_certificates.py index deda492a..7d29fd45 100644 --- a/cert_issuer/issue_certificates.py +++ b/cert_issuer/issue_certificates.py @@ -20,10 +20,10 @@ def issue(app_config, certificate_batch_handler, transaction_handler, recipient_ certificate_batch_handler=certificate_batch_handler, transaction_handler=transaction_handler, max_retry=app_config.max_retry) - tx_id = issuer.issue(app_config.chain, app_config, recipient_address) + (tx_id, token_id) = issuer.issue(app_config.chain, app_config, recipient_address) certificate_batch_handler.post_batch_actions(app_config) - return tx_id + return (tx_id, token_id) def revoke_certificates(app_config, transaction_handler): # revocations are executed one hash at a time - balance is ensure before each tx @@ -36,6 +36,12 @@ def revoke_certificates(app_config, transaction_handler): return tx_id +def update_token_uri(app_config, token_id, token_uri, transaction_handler): + issuer = Issuer(transaction_handler=transaction_handler, + max_retry=app_config.max_retry) + + issuer.update_token_uri(token_id, token_uri, app_config) + def main(app_config): chain = app_config.chain if chain == Chain.ethereum_mainnet or chain == Chain.ethereum_bloxberg: diff --git a/cert_issuer/issuer.py b/cert_issuer/issuer.py index afb1589d..66c6a4b1 100644 --- a/cert_issuer/issuer.py +++ b/cert_issuer/issuer.py @@ -32,10 +32,12 @@ def issue(self, chain, app_config, recipient_address): for attempt_number in range(0, self.max_retry): try: txid = self.transaction_handler.issue_transaction(recipient_address, token_uri, blockchain_bytes, app_config) + event_args = self.transaction_handler.get_event_args(txid, 'Transfer') + token_id = event_args['tokenId'] + self.certificate_batch_handler.finish_batch(txid, chain, app_config) logging.info('Broadcast transaction with txid %s', txid) - - return txid + return (txid, token_id) except BroadcastError: logging.warning( 'Failed broadcast reattempts. Trying to recreate transaction. This is attempt number %d', @@ -43,9 +45,22 @@ def issue(self, chain, app_config, recipient_address): logging.error('All attempts to broadcast failed. Try rerunning issuer.') raise BroadcastError('All attempts to broadcast failed. Try rerunning issuer.') - def updateIPFS(self, chain, app_config): + def update_token_uri(self, token_id, token_uri, app_config): """ - Update IPFS link on smart contract + Update the tokenURI for the issued certificate batch + :return: transaction id """ - txid = self.transaction_handler.ipfs_transaction(token_id, token_uri, blockchain_bytes, app_config) + + for attempt_number in range(0, self.max_retry): + try: + txid = self.transaction_handler.update_token_uri(token_id, token_uri, app_config) + logging.info('Updating tokenURI field with ipfs link. Txid: %s', txid) + return txid + except BroadcastError: + logging.warning( + 'Failed broadcast reattempts. Trying to recreate transaction. This is attempt number %d', + attempt_number) + logging.error('All attempts to broadcast failed. Try rerunning issuer.') + raise BroadcastError('All attempts to broadcast failed. Try rerunning issuer.') +