Skip to content

Commit

Permalink
test: add connectionless issuance end-to-end test
Browse files Browse the repository at this point in the history
Signed-off-by: chereseeriepa <[email protected]>
  • Loading branch information
chereseeriepa committed Nov 18, 2024
1 parent b65fa95 commit cf145f4
Show file tree
Hide file tree
Showing 11 changed files with 151 additions and 17 deletions.
6 changes: 3 additions & 3 deletions integration-tests/e2e-tests/.env.example
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
MEDIATOR_OOB_URL=
AGENT_URL=
APIKEY=
MEDIATOR_OOB_URL=https://mediator.mydomain.com/invitationOOB
AGENT_URL=https://issuer.mydomain.com/cloud-agent/
APIKEY=myDomainAPIKey
PUBLISHED_DID=
JWT_SCHEMA_GUID=
ANONCRED_DEFINITION_GUID=
16 changes: 8 additions & 8 deletions integration-tests/e2e-tests/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,14 +9,14 @@ This guide shows you how to run the end-to-end tests
- Copy `.env.example` to `.env`
- Fill the required properties

| Property | Description |
| ------------------------ | ----------------------------------------------------------- |
| MEDIATOR_OOB_URL | URL that returns the OOB url |
| AGENT_URL | URL for Cloud Agent - should end with a forward slash ("/") |
| APIKEY | (Optional) Apikey authentication |
| PUBLISHED_DID | (Optional) Existing DID |
| JWT_SCHEMA_GUID | (Optional) Existing jwt schema guid |
| ANONCRED_DEFINITION_GUID | (Optional) Existing anoncred definition guid |
| Property | Description |
| ------------------------ | ------------------------------------------------------------------------------------------------------------|
| MEDIATOR_OOB_URL | URL that returns the OOB url (e.g. https://mediator.mydomain.com/invitationOOB) |
| AGENT_URL | URL for Cloud Agent - should end with a forward slash ("/") (e.g. https://issuer.mydomain.com/cloud-agent/) |
| APIKEY | (Optional) Apikey authentication |
| PUBLISHED_DID | (Optional) Existing DID |
| JWT_SCHEMA_GUID | (Optional) Existing jwt schema guid |
| ANONCRED_DEFINITION_GUID | (Optional) Existing anoncred definition guid |

### Compile the SDK

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
@connectionless @credential-offer
Feature: Edge SDK Connectionless Credential Offer

Scenario: Receive a credential without a connection
Given Cloud Agent is not connected to Edge Agent
When Cloud Agent has a connectionless credential offer invitation
And Cloud Agent shares invitation to Edge Agent
Then Edge Agent accepts the connectionless credential offer invitation
Then Edge Agent should receive the connectionless credential offer
Then Edge Agent accepts the connectionless credential offer
Then Edge Agent should receive the connectionless credential
And Edge Agent processes the issued connectionless credential from Cloud Agent
2 changes: 1 addition & 1 deletion integration-tests/e2e-tests/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
"@amagyar-iohk/identus-cloud-agent-client-ts": "^1.36.1",
"@cucumber/cucumber": "^10.3.1",
"@cucumber/pretty-formatter": "^1.0.0",
"@hyperledger/identus-edge-agent-sdk": "../../hyperledger-identus-edge-agent-sdk-3.1.0.tgz",
"@hyperledger/identus-edge-agent-sdk": "../../",
"@serenity-js/assertions": "^3.29.2",
"@serenity-js/console-reporter": "^3.29.2",
"@serenity-js/core": "^3.29.2",
Expand Down
10 changes: 8 additions & 2 deletions integration-tests/e2e-tests/src/abilities/WalletSdk.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,8 @@ export class WalletSdk extends Ability implements Initialisable, Discardable {
issuedCredentialStack: SDK.Domain.Message[];
proofRequestStack: SDK.Domain.Message[];
revocationStack: SDK.Domain.Message[],
presentationMessagesStack: SDK.Domain.Message[]
presentationMessagesStack: SDK.Domain.Message[],
enqueue(message: SDK.Domain.Message): Promise<void>

}) => Promise<void>): Interaction {
return Interaction.where("#actor uses wallet sdk", async actor => {
Expand All @@ -72,7 +73,12 @@ export class WalletSdk extends Ability implements Initialisable, Discardable {
issuedCredentialStack: WalletSdk.as(actor).messages.issuedCredentialStack,
proofRequestStack: WalletSdk.as(actor).messages.proofRequestStack,
revocationStack: WalletSdk.as(actor).messages.revocationStack,
presentationMessagesStack: WalletSdk.as(actor).messages.presentationMessagesStack
presentationMessagesStack: WalletSdk.as(actor).messages.presentationMessagesStack,

enqueue: async (message: SDK.Domain.Message) => {
// Ensure to call the async method properly
await WalletSdk.as(actor).messages.enqueue(message);
}
})
})
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,7 @@ export class CloudAgentConfiguration {
try {
assert(this.anoncredDefinitionGuid != null)
assert(this.anoncredDefinitionGuid != "")

await axiosInstance.get(
`credential-definition-registry/definitions/${this.anoncredDefinitionGuid}`
)
Expand Down
14 changes: 13 additions & 1 deletion integration-tests/e2e-tests/src/steps/CloudAgentSteps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@ Given("{actor} has a connection invitation with '{}', '{}' and '{}' parameters",
const goalCode = rawGoalCode == "null" ? undefined : rawGoalCode
const goal = rawGoal == "null" ? undefined : rawGoal
await CloudAgentWorkflow.createConnection(cloudAgent, label, goalCode, goal)
})
}
)

Given("{actor} is connected to {actor}", async function (cloudAgent: Actor, edgeAgent: Actor) {
await CloudAgentWorkflow.createConnection(cloudAgent)
Expand Down Expand Up @@ -45,6 +46,7 @@ When("{actor} offers '{int}' anonymous credential", async function (cloudAgent:
)
})


When("{actor} asks for present-proof", async function (cloudAgent: Actor) {
await CloudAgentWorkflow.askForPresentProof(cloudAgent)
})
Expand Down Expand Up @@ -77,9 +79,19 @@ Then("{actor} should see the present-proof is not verified", async (cloudAgent:
await CloudAgentWorkflow.verifyPresentProof(cloudAgent, "PresentationFailed")
})


Then("{actor} should see all credentials were accepted", async (cloudAgent: Actor) => {
const recordIdList = await cloudAgent.answer<string[]>(Notepad.notes().get("recordIdList"))
for (const recordId of recordIdList) {
await CloudAgentWorkflow.verifyCredentialState(cloudAgent, recordId, "CredentialSent")
}
})

Given("{actor} is not connected to Edge Agent", async function (cloudAgent: Actor) {
await CloudAgentWorkflow.verifyNoConnection(cloudAgent)
})

Given("{actor} has a connectionless credential offer invitation", async function (cloudAgent: Actor) {
await CloudAgentWorkflow.createConnectionlessCredentialOfferInvitation(cloudAgent)
})

46 changes: 46 additions & 0 deletions integration-tests/e2e-tests/src/steps/EdgeAgentSteps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -233,3 +233,49 @@ Then("{actor} should see the verification proof is verified false", async (edgeA
await EdgeAgentWorkflow.waitForPresentationMessage(edgeAgent)
await EdgeAgentWorkflow.verifyPresentation(edgeAgent, false)
})


When("{actor} accepts the connectionless credential offer invitation",
async function (edgeAgent: Actor) {
await EdgeAgentWorkflow.acceptCredentialOfferInvitation(edgeAgent)
}
)

Then("{actor} should receive the connectionless credential offer",
async function (edgeAgent: Actor) {
try {
await EdgeAgentWorkflow.waitForCredentialOffer(edgeAgent, 1)
} catch (error) {
// NOTE: sometimes the listener fails, so we fall back to getting the messages from
// pluto
await EdgeAgentWorkflow.loadMessagesFromPluto(edgeAgent)
await EdgeAgentWorkflow.waitForCredentialOffer(edgeAgent, 1)
}
}
)

When("{actor} accepts the connectionless credential offer",
async function (edgeAgent: Actor) {
await EdgeAgentWorkflow.acceptCredential(edgeAgent)
}
)

Then("{actor} should receive the connectionless credential",
async function (edgeAgent: Actor) {
try {
await EdgeAgentWorkflow.waitToReceiveCredentialIssuance(edgeAgent, 1)
} catch (error) {
// NOTE: sometimes the listener fails, so we fall back to getting the messages from
// pluto
await EdgeAgentWorkflow.loadMessagesFromPluto(edgeAgent)
await EdgeAgentWorkflow.waitToReceiveCredentialIssuance(edgeAgent, 1)
}
}
)

Then("{actor} processes the issued connectionless credential from {actor}",
async function (edgeAgent: Actor, cloudAgent: Actor) {
const recordId = await cloudAgent.answer<string>(Notepad.notes().get("recordId"))
await EdgeAgentWorkflow.processIssuedCredential(edgeAgent, recordId)
}
)
40 changes: 40 additions & 0 deletions integration-tests/e2e-tests/src/workflow/CloudAgentWorkflow.ts
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,16 @@ export class CloudAgentWorkflow {
)
}

static async verifyNoConnection(cloudAgent: Actor) {
const connectionId = await cloudAgent.answer(
Notepad.notes().get("connectionId")
).catch(() => null)

await cloudAgent.attemptsTo(
Ensure.that(connectionId, equals(null))
)
}

static async verifyCredentialState(cloudAgent: Actor, recordId: string, state: string) {
await cloudAgent.attemptsTo(
Wait.upTo(Duration.ofSeconds(60)).until(
Expand All @@ -73,6 +83,36 @@ export class CloudAgentWorkflow {
)
}

static async createConnectionlessCredentialOfferInvitation(cloudAgent: Actor) {
const credentialOffer = {
claims: {
emailAddress: "sampleEmail",
familyName: "",
dateOfIssuance: "2023-01-01T02:02:02Z",
drivingLicenseID: "",
drivingClass: 1,
},
goalCode: "issue-vc",
goal: "Request issuance",
credentialFormat: "JWT",
issuingDID: CloudAgentConfiguration.publishedDid,
automaticIssuance: true
}

await cloudAgent.attemptsTo(
Send.a(PostRequest.to("issue-credentials/credential-offers/invitation").with(credentialOffer)),
Ensure.that(LastResponse.status(), equals(HttpStatusCode.Created)),
Notepad.notes().set(
"invitation",
LastResponse.body().invitation.invitationUrl
),
Notepad.notes().set(
"recordId",
LastResponse.body().recordId
)
)
}

static async offerCredential(cloudAgent: Actor) {
const credential = new CreateIssueCredentialRecordRequest()
credential.claims = {
Expand Down
18 changes: 18 additions & 0 deletions integration-tests/e2e-tests/src/workflow/EdgeAgentWorkflow.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,10 @@ export class EdgeAgentWorkflow {
)
}

static async acceptCredentialOfferInvitation(edgeAgent: Actor): Promise<void> {
await this.connect(edgeAgent)
}

static async waitForCredentialOffer(edgeAgent: Actor, numberOfCredentialOffer: number) {
await edgeAgent.attemptsTo(
Wait.upTo(Duration.ofSeconds(60)).until(
Expand All @@ -32,6 +36,20 @@ export class EdgeAgentWorkflow {
)
}

// NOTE: sometimes the listener fails, so we have to fallback to
// the messages in pluto
static async loadMessagesFromPluto (edgeAgent: Actor) {
await edgeAgent.attemptsTo(
WalletSdk.execute(async (sdk, messages) => {
const msgs = await sdk.pluto.getAllMessages()

await Promise.all(
msgs.map(msg => messages.enqueue(msg))
)
})
)
}

static async waitToReceiveCredentialIssuance(edgeAgent: Actor, expectedNumberOfCredentials: number) {
await edgeAgent.attemptsTo(
Wait.upTo(Duration.ofSeconds(60)).until(
Expand Down
3 changes: 1 addition & 2 deletions integration-tests/e2e-tests/yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -678,9 +678,8 @@
hash.js "1.1.7"
stream-browserify "3.0.0"

"@hyperledger/identus-edge-agent-sdk@../../hyperledger-identus-edge-agent-sdk-3.1.0.tgz":
"@hyperledger/identus-edge-agent-sdk@../../":
version "3.1.0"
resolved "../../hyperledger-identus-edge-agent-sdk-3.1.0.tgz#9eaf45ae9ac72fd590192eedaead852a7b9671fd"
dependencies:
"@hyperledger/identus-apollo" "^1.4.3"
"@noble/ciphers" "^0.6.0"
Expand Down

0 comments on commit cf145f4

Please sign in to comment.