Skip to content

Commit

Permalink
This PR fixes issues with credential registries (WebOfTrust#197)
Browse files Browse the repository at this point in the history
* Fix credential registry list endpoint to account for a not yet fully committed registry.  Closes WebOfTrust#147

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Fix long running endpoint to allow for an operaiton for a KEL event not yet processed.

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Remove extra test condition pulled in from development

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Add test for invalid sequence number in long running witness operation.

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Add test for satisfied witness receipts for long running operation

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Fix reference to revocation anchor event.

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Add test coverage for multisig revocation.

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

* Rev version number

Signed-off-by: pfeairheller <pfeairheller@gmail.com>

---------

Signed-off-by: pfeairheller <pfeairheller@gmail.com>
  • Loading branch information
pfeairheller authored Feb 26, 2024
1 parent c1b638c commit 31e81dd
Show file tree
Hide file tree
Showing 9 changed files with 128 additions and 26 deletions.
2 changes: 1 addition & 1 deletion Makefile
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
.PHONY: build-keria
build-keria:
@docker buildx build --platform=linux/amd64 --no-cache -f images/keria.dockerfile --tag weboftrust/keria:0.1.0 --tag weboftrust/keria:latest .
@docker buildx build --platform=linux/amd64 --no-cache -f images/keria.dockerfile --tag weboftrust/keria:0.1.1 --tag weboftrust/keria:latest .

publish-keria:
@docker push weboftrust/keria --all-tags
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@

setup(
name='keria',
version='0.1.0', # also change in src/keria/__init__.py
version='0.1.1', # also change in src/keria/__init__.py
license='Apache Software License 2.0',
description='KERIA: KERI Agent in the cloud',
long_description="KERIA: KERI Agent in the cloud.",
Expand Down
2 changes: 1 addition & 1 deletion src/keria/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,5 +3,5 @@
main package
"""

__version__ = '0.1.0' # also change in setup.py
__version__ = '0.1.1' # also change in setup.py

8 changes: 6 additions & 2 deletions src/keria/app/credentialing.py
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,9 @@ def on_get(req, rep, name):

res = []
for name, registry in agent.rgy.regs.items():
if registry.regk not in registry.tevers: # defensive programming for a registry not being fully committed
continue

if registry.hab.pre == hab.pre:
rd = dict(
name=registry.name,
Expand Down Expand Up @@ -808,8 +811,9 @@ def revoke(self, regk, rserder, anc):
self.rgy.reger.tpwe.add(keys=(vcid, rseq.qb64), val=(hab.kever.prefixer, seqner, saider))
return vcid, rseq.sn
else:
sn = anc.sn
said = anc.said
serder = serdering.SerderKERI(sad=anc)
sn = serder.sn
said = serder.said

prefixer = coring.Prefixer(qb64=hab.pre)
seqner = coring.Seqner(sn=sn)
Expand Down
36 changes: 20 additions & 16 deletions src/keria/core/longrunning.py
Original file line number Diff line number Diff line change
Expand Up @@ -197,26 +197,30 @@ def status(self, op):
sn = op.metadata["sn"]
kever = self.hby.kevers[op.oid]
sdig = self.hby.db.getKeLast(key=dbing.snKey(pre=kever.prefixer.qb64b, sn=sn))
if sdig is not None:

dgkey = dbing.dgKey(kever.prefixer.qb64b, bytes(sdig))
wigs = self.hby.db.getWigs(dgkey)
dgkey = dbing.dgKey(kever.prefixer.qb64b, bytes(sdig))
wigs = self.hby.db.getWigs(dgkey)

if len(wigs) >= kever.toader.num:
evt = self.hby.db.getEvt(dbing.dgKey(pre=kever.prefixer.qb64, dig=bytes(sdig)))
serder = serdering.SerderKERI(raw=bytes(evt))
operation.done = True
operation.response = serder.ked

else:
start = helping.fromIso8601(op.start)
dtnow = helping.nowUTC()
if (dtnow - start) > datetime.timedelta(seconds=eventing.Kevery.TimeoutPWE):
if len(wigs) >= kever.toader.num:
evt = self.hby.db.getEvt(dbing.dgKey(pre=kever.prefixer.qb64, dig=bytes(sdig)))
serder = serdering.SerderKERI(raw=bytes(evt))
operation.done = True
operation.error = Status(code=408, # Using HTTP error codes here for lack of a better alternative
message=f"long running {op.type} for {op.oid} operation timed out before "
f"receiving sufficient witness receipts")
operation.response = serder.ked

else:
operation.done = False
start = helping.fromIso8601(op.start)
dtnow = helping.nowUTC()
if (dtnow - start) > datetime.timedelta(seconds=eventing.Kevery.TimeoutPWE):
operation.done = True
operation.error = Status(code=408, # Using HTTP error codes here for lack of a better alternative
message=f"long running {op.type} for {op.oid} operation timed out before "
f"receiving sufficient witness receipts")
else:
operation.done = False

else:
operation.done = False

elif op.type in (OpTypes.oobi,):
if "oobi" not in op.metadata:
Expand Down
36 changes: 35 additions & 1 deletion tests/app/test_aiding.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,13 +18,14 @@
from keri.core.coring import MtrDex
from keri.db.basing import LocationRecord
from keri.peer import exchanging
from keri.db import basing
from keri.db import basing, dbing
from keri import kering
from keri.vdr import credentialing
from hio.base import doing

from keria.app import aiding, agenting
from keria.app.aiding import IdentifierOOBICollectionEnd, RpyEscrowCollectionEnd
from keria.core import longrunning


def test_load_ends(helpers):
Expand Down Expand Up @@ -252,6 +253,11 @@ def test_identifier_collection_end(helpers):
groupEnd = aiding.GroupMemberCollectionEnd()
app.add_route("/identifiers/{name}/members", groupEnd)

opColEnd = longrunning.OperationCollectionEnd()
app.add_route("/operations", opColEnd)
opResEnd = longrunning.OperationResourceEnd()
app.add_route("/operations/{name}", opResEnd)

client = testing.TestClient(app)

res = client.simulate_post(path="/identifiers", body=b'{}')
Expand Down Expand Up @@ -393,6 +399,23 @@ def test_identifier_collection_end(helpers):
res = client.simulate_post(path="/identifiers", body=json.dumps(body))
assert res.status_code == 202

op = res.json
name = op['name']

res = client.simulate_get(path=f"/operations/{name}")
assert res.status_code == 200
assert res.json['done'] is False

# Modify sequence number to test invalid sn
op = agent.monitor.opr.ops.get(keys=(name,))
op.metadata['sn'] = 4
agent.monitor.opr.ops.pin(keys=(name,), val=op)

res = client.simulate_get(path=f"/operations/{name}")
assert res.status_code == 200
assert res.json['done'] is False


assert len(agent.witners) == 1
res = client.simulate_get(path="/identifiers")
assert res.status_code == 200
Expand All @@ -403,6 +426,17 @@ def test_identifier_collection_end(helpers):
ss = aid[Algos.salty]
assert ss["pidx"] == 3

# Reset sn
op.metadata['sn'] = 0
agent.monitor.opr.ops.pin(keys=(name,), val=op)

# Add fake witness receipts to test satified witnessing
dgkey = dbing.dgKey(serder.preb, serder.preb)
agent.hby.db.putWigs(dgkey, vals=[b'A', b'B', b'C'])
res = client.simulate_get(path=f"/operations/{name}")
assert res.status_code == 200
assert res.json['done'] is True

res = client.simulate_get(path=f"/identifiers/aid1")
mhab = res.json
agent0 = mhab["state"]
Expand Down
19 changes: 15 additions & 4 deletions tests/app/test_credentialing.py
Original file line number Diff line number Diff line change
Expand Up @@ -131,6 +131,10 @@ def test_registry_end(helpers, seeder):
assert metadata["anchor"] == anchor
assert result.status == falcon.HTTP_202

result = client.simulate_get(path="/identifiers/test/registries")
assert result.status == falcon.HTTP_200
assert result.json == []

tock = 0.03125
limit = 1.0
doist = doing.Doist(limit=limit, tock=tock, real=True)
Expand All @@ -143,11 +147,15 @@ def test_registry_end(helpers, seeder):

assert regser.pre in agent.tvy.tevers

result = client.simulate_get(path="/identifiers/test/registries")
assert result.status == falcon.HTTP_200
assert len(result.json) == 1

body = dict(name="test", alias="test", vcp=regser.ked, ixn=serder.ked, sigs=sigers)
result = client.simulate_post(path="/identifiers/bad_test/registries", body=json.dumps(body).encode("utf-8"))
assert result.status == falcon.HTTP_404
assert result.json == {'description': 'alias is not a valid reference to an identifier', 'title': '404 Not Found'}
assert result.json == {'description': 'alias is not a valid reference to an identifier',
'title': '404 Not Found'}


result = client.simulate_get(path="/identifiers/not_test/registries")
Expand Down Expand Up @@ -513,7 +521,8 @@ def test_revoke_credential(helpers, seeder):
rev=sad,
ixn=serder.ked,
sigs=sigers)
res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}", body=json.dumps(badbody).encode("utf-8"))
res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}",
body=json.dumps(badbody).encode("utf-8"))
assert res.status_code == 404
assert res.json == {'description': 'revocation against invalid registry SAID '
'EIVtei3pGKGUw8H2Ri0h1uOevtSA6QGAq5wifbtHIaNI',
Expand All @@ -527,12 +536,14 @@ def test_revoke_credential(helpers, seeder):
rev=sad,
ixn=serder.ked,
sigs=sigers)
res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}", body=json.dumps(badbody).encode("utf-8"))
res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}",
body=json.dumps(badbody).encode("utf-8"))
assert res.status_code == 400
assert res.json == {'description': "invalid revocation event.",
'title': '400 Bad Request'}

res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}", body=json.dumps(body).encode("utf-8"))
res = client.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}",
body=json.dumps(body).encode("utf-8"))
assert res.status_code == 200

while not agent.registrar.complete(creder.said, sn=1):
Expand Down
47 changes: 47 additions & 0 deletions tests/app/test_ipexing.py
Original file line number Diff line number Diff line change
Expand Up @@ -368,6 +368,9 @@ def test_multisig_grant_admit(seeder, helpers):
endRolesEnd = aiding.EndRoleCollectionEnd()
app.add_route("/identifiers/{name}/endroles", endRolesEnd)

credentialResourceDelEnd = credentialing.CredentialResourceDeleteEnd(aidEnd)
app.add_route("/identifiers/{name}/credentials/{said}", credentialResourceDelEnd)

# Create Issuer Participant 0
ipsalt0 = b'0123456789abcM00'
op = helpers.createAid(client0, "issuerParticipant0", ipsalt0)
Expand Down Expand Up @@ -852,6 +855,50 @@ def test_multisig_grant_admit(seeder, helpers):
# Ensure that the credential has been persisted by both agents
assert hagent1.rgy.reger.saved.get(keys=(creder.said,)) is not None

# Get latest state
ip0 = client0.simulate_get("/identifiers/issuerParticipant0").json
ip1 = client1.simulate_get("/identifiers/issuerParticipant1").json

# Create the revocation event from the credential SAID and the registry SAID
issueSaid = regser.said
regser = veventing.revoke(vcdig=creder.said, regk=regk, dt=dt, dig=issueSaid)

anchor = dict(i=regser.ked['i'], s=regser.ked["s"], d=regser.said)
interact = eventing.interact(pre=issuerPre, dig=interact.said, sn=3, data=[anchor])
sigs = [issuerSigner0.sign(ser=interact.raw, index=0).qb64, issuerSigner1.sign(ser=interact.raw, index=1).qb64]

# Submit the Revocation to Agent 0
body = dict(
rev=regser.ked,
ixn=interact.ked,
sigs=sigs,
group={
"mhab": ip0,
"keys": ikeys
}
)

result = client0.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}",
body=json.dumps(body).encode("utf-8"))

assert result.status == falcon.HTTP_200

# Submit the Revocation to Agent 1
body = dict(
rev=regser.ked,
ixn=interact.ked,
sigs=sigs,
group={
"mhab": ip1,
"keys": ikeys
}
)

result = client1.simulate_delete(path=f"/identifiers/issuer/credentials/{creder.said}",
body=json.dumps(body).encode("utf-8"))

assert result.status == falcon.HTTP_200


def test_granter(helpers):
with helpers.openKeria() as (agency, agent, app, client):
Expand Down
2 changes: 2 additions & 0 deletions tests/core/test_longrunning.py
Original file line number Diff line number Diff line change
Expand Up @@ -185,6 +185,8 @@ def test_error(helpers):
except ValidationError as e:
err = e

assert err is not None

res = client.simulate_get(path="/operations")
assert isinstance(res.json, list)
assert len(res.json) == 1
Expand Down

0 comments on commit 31e81dd

Please sign in to comment.