Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add validation to JSON transformer #830

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
3d91c49
feat: wip add validation to JSON transformer
Jun 2, 2022
0c350cf
feat: wip address code review feedback.
Jun 3, 2022
5de2d99
chore: merge branch 'main' into feat/json-tranform-validator
Jun 3, 2022
19a8f48
chore: wip fix validation issues
Jun 3, 2022
89e691a
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 3, 2022
29d26f5
chore: fixing more issues
Jun 3, 2022
29f4e6d
chore: fix som emore errors
Jun 3, 2022
587d438
chore: code review feedback
Jun 6, 2022
8d235b5
chore: come more code review feedback
Jun 6, 2022
9ae21e3
chore: fix type check
Jun 6, 2022
e6f8637
chore: fix tests
Jun 6, 2022
f474eba
chore: fix TransportDecorator test
Jun 7, 2022
1eb8def
chore: enable validate for ConnectionService
Jun 7, 2022
71aca28
chore: code review feedback
Jun 9, 2022
f189eca
chore: add test for message validator
Jun 9, 2022
8721bb7
chore: merge branch 'main' into feat/json-tranform-validator
Jun 9, 2022
d13c3c9
chore: fix accDecorator tests
Jun 9, 2022
686e0fe
chore: add tests for validating error array
Jun 9, 2022
ea761dc
chore: wip this fails
Jun 9, 2022
f3e9023
chore: code review feedback
Jun 9, 2022
793117c
chore: refactor
Jun 9, 2022
502244e
chore: wip code review feedback fixing tests
Jun 10, 2022
c2c70b7
chore: fix tests
Jun 10, 2022
7824356
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 10, 2022
3cabb43
chore: fix typo
Jun 10, 2022
50365d7
chore: fix newline in in template literal
Jun 13, 2022
9e04a1b
chore: code review feedback
Jun 13, 2022
c0e45c7
chore: code review feedback
Jun 14, 2022
7695eaa
chore: merge branch 'main' into feat/json-tranform-validator
Jun 15, 2022
992862a
chore: code review feedback
Jun 15, 2022
bdab6bc
chore: code review feedback
Jun 15, 2022
80fee1c
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 15, 2022
cbb0f62
chore: code review feedback
Jun 15, 2022
5a4a9bf
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 15, 2022
d5eecb9
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 16, 2022
8504119
chore: code review feedback
Jun 16, 2022
e71fe1c
chore: merge remote-tracking branch 'upstream/main' into feat/json-tr…
Jun 16, 2022
cee5211
chore: remove unused es-lint ignore line
Jun 16, 2022
d8174fc
fix: variable declaration
Jun 17, 2022
ff330a2
Merge remote-tracking branch 'upstream/main' into feat/json-tranform-…
Jun 17, 2022
baf038b
merge remote-tracking branch 'upstream/main' into feat/json-tranform-…
Jun 20, 2022
3291848
Merge branch 'main' into feat/json-tranform-validator
TimoGlastra Jun 20, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion packages/core/src/agent/Agent.ts
Original file line number Diff line number Diff line change
Expand Up @@ -277,7 +277,7 @@ export class Agent {
}

private async getMediationConnection(mediatorInvitationUrl: string) {
const outOfBandInvitation = await this.oob.parseInvitation(mediatorInvitationUrl)
const outOfBandInvitation = this.oob.parseInvitation(mediatorInvitationUrl)
const outOfBandRecord = await this.oob.findByInvitationId(outOfBandInvitation.id)
const [connection] = outOfBandRecord ? await this.connections.findAllByOutOfBandId(outOfBandRecord.id) : []

Expand Down
16 changes: 4 additions & 12 deletions packages/core/src/agent/MessageReceiver.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,6 @@ import { ConnectionsModule } from '../modules/connections'
import { ProblemReportError, ProblemReportMessage, ProblemReportReason } from '../modules/problem-reports'
import { isValidJweStructure } from '../utils/JWE'
import { JsonTransformer } from '../utils/JsonTransformer'
import { MessageValidator } from '../utils/MessageValidator'
import { canHandleMessageType, parseMessageType, replaceLegacyDidSovPrefixOnMessage } from '../utils/messageType'

import { AgentConfig } from './AgentConfig'
Expand Down Expand Up @@ -168,7 +167,6 @@ export class MessageReceiver {
let message: AgentMessage
try {
message = await this.transformMessage(plaintextMessage)
await this.validateMessage(message)
} catch (error) {
if (connection) await this.sendProblemReportMessage(error.message, connection, plaintextMessage)
throw error
Expand Down Expand Up @@ -209,25 +207,19 @@ export class MessageReceiver {
}

// Cast the plain JSON object to specific instance of Message extended from AgentMessage
return JsonTransformer.fromJSON(message, MessageClass)
}

/**
* Validate an AgentMessage instance.
* @param message agent message to validate
*/
private async validateMessage(message: AgentMessage) {
let messageTransformed: AgentMessage
try {
await MessageValidator.validate(message)
messageTransformed = JsonTransformer.fromJSON(message, MessageClass)
} catch (error) {
this.logger.error(`Error validating message ${message.type}`, {
errors: error,
message: message.toJSON(),
message: JSON.stringify(message),
})
throw new ProblemReportError(`Error validating message ${message.type}`, {
problemCode: ProblemReportReason.MessageParseFailure,
})
}
return messageTransformed
}

/**
Expand Down
2 changes: 1 addition & 1 deletion packages/core/src/agent/MessageSender.ts
Original file line number Diff line number Diff line change
Expand Up @@ -313,7 +313,7 @@ export class MessageSender {
}

try {
await MessageValidator.validate(message)
MessageValidator.validateSync(message)
} catch (error) {
this.logger.error(
`Aborting sending outbound message ${message.type} to ${service.serviceEndpoint}. Message validation failed`,
Expand Down
51 changes: 35 additions & 16 deletions packages/core/src/agent/__tests__/AgentMessage.test.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { TestMessage } from '../../../tests/TestMessage'
import { ClassValidationError } from '../../error/ClassValidationError'
import { JsonTransformer } from '../../utils'
import { MessageValidator } from '../../utils/MessageValidator'
import { IsValidMessageType, parseMessageType } from '../../utils/messageType'
import { AgentMessage } from '../AgentMessage'

Expand Down Expand Up @@ -32,7 +32,7 @@ describe('AgentMessage', () => {

const message = JsonTransformer.fromJSON(json, CustomProtocolMessage)

await expect(MessageValidator.validate(message)).resolves.toBeUndefined()
expect(message).toBeInstanceOf(CustomProtocolMessage)
})

it('successfully validates if the message type minor version is lower than the supported message type', async () => {
Expand All @@ -43,37 +43,56 @@ describe('AgentMessage', () => {

const message = JsonTransformer.fromJSON(json, CustomProtocolMessage)

await expect(MessageValidator.validate(message)).resolves.toBeUndefined()
expect(message).toBeInstanceOf(CustomProtocolMessage)
})

it('successfully validates if the message type minor version is higher than the supported message type', async () => {
it('successfully validates if the message type minor version is higher than the supported message type', () => {
const json = {
'@id': 'd61c7e3d-d4af-469b-8d42-33fd14262e17',
'@type': 'https://didcomm.org/fake-protocol/1.8/message',
}

const message = JsonTransformer.fromJSON(json, CustomProtocolMessage)

await expect(MessageValidator.validate(message)).resolves.toBeUndefined()
expect(message).toBeInstanceOf(CustomProtocolMessage)
})

it('throws a validation error if the message type major version differs from the supported message type', async () => {
expect.assertions(1)

it('throws a validation error if the message type major version differs from the supported message type', () => {
const json = {
'@id': 'd61c7e3d-d4af-469b-8d42-33fd14262e17',
'@type': 'https://didcomm.org/fake-protocol/2.0/message',
}

const message = JsonTransformer.fromJSON(json, CustomProtocolMessage)

await expect(MessageValidator.validate(message)).rejects.toMatchObject([
{
constraints: {
isValidMessageType: 'type does not match the expected message type (only minor version may be lower)',
expect(() => JsonTransformer.fromJSON(json, CustomProtocolMessage)).toThrowError(ClassValidationError)
try {
JsonTransformer.fromJSON(json, CustomProtocolMessage)
} catch (error) {
const thrownError = error as ClassValidationError
expect(thrownError.message).toEqual(
'CustomProtocolMessage: Failed to validate class.\nAn instance of CustomProtocolMessage has failed the validation:\n - property type has failed the following constraints: isValidMessageType \n'
)
expect(thrownError.validationErrors).toMatchObject([
{
target: {
appendedAttachments: undefined,
id: 'd61c7e3d-d4af-469b-8d42-33fd14262e17',
l10n: undefined,
pleaseAck: undefined,
service: undefined,
thread: undefined,
timing: undefined,
transport: undefined,
type: 'https://didcomm.org/fake-protocol/2.0/message',
},
value: 'https://didcomm.org/fake-protocol/2.0/message',
property: 'type',
children: [],
constraints: {
isValidMessageType: 'type does not match the expected message type (only minor version may be lower)',
},
},
},
])
])
}
})
})
})
97 changes: 48 additions & 49 deletions packages/core/src/decorators/ack/AckDecorator.test.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { BaseMessage } from '../../agent/BaseMessage'
import { ClassValidationError } from '../../error/ClassValidationError'
import { JsonTransformer } from '../../utils/JsonTransformer'
import { MessageValidator } from '../../utils/MessageValidator'
import { Compose } from '../../utils/mixins'
Expand All @@ -25,22 +26,9 @@ describe('Decorators | AckDecoratorExtension', () => {
})

test('transforms Json to AckDecorator class', () => {
const transformed = JsonTransformer.fromJSON({ '~please_ack': {} }, TestMessage)

expect(transformed).toEqual({
pleaseAck: {
on: ['RECEIPT'],
},
})
expect(transformed).toBeInstanceOf(TestMessage)
})

test('successfully transforms ack decorator with on field present', () => {
const transformed = JsonTransformer.fromJSON(
{
'~please_ack': {
on: ['RECEIPT'],
},
'~please_ack': {},
'@id': '7517433f-1150-46f2-8495-723da61b872a',
'@type': 'https://didcomm.org/test-protocol/1.0/test-message',
},
Expand Down Expand Up @@ -88,45 +76,56 @@ describe('Decorators | AckDecoratorExtension', () => {
TestMessage
)

await expect(MessageValidator.validate(transformedWithDefault)).resolves.toBeUndefined()
expect(MessageValidator.validateSync(transformedWithDefault)).toBeUndefined()
})

const transformedWithoutDefault = JsonTransformer.fromJSON(
{
'~please_ack': {
on: ['OUTCOME'],
test('transforms Json to AckDecorator class', () => {
const transformed = () =>
JsonTransformer.fromJSON(
{
'~please_ack': {},
'@id': undefined,
'@type': undefined,
},
'@id': '7517433f-1150-46f2-8495-723da61b872a',
'@type': 'https://didcomm.org/test-protocol/1.0/test-message',
},
TestMessage
)

await expect(MessageValidator.validate(transformedWithoutDefault)).resolves.toBeUndefined()
TestMessage
)

const transformedWithIncorrectValue = JsonTransformer.fromJSON(
{
'~please_ack': {
on: ['NOT_A_VALID_VALUE'],
expect(() => transformed()).toThrow(ClassValidationError)
try {
transformed()
} catch (e) {
const caughtError = e as ClassValidationError
expect(caughtError.message).toEqual(
'TestMessage: Failed to validate class.\nAn instance of TestMessage has failed the validation:\n - property id has failed the following constraints: matches \n\nAn instance of TestMessage has failed the validation:\n - property type has failed the following constraints: matches \n'
)
expect(caughtError.validationErrors).toMatchObject([
{
children: [],
constraints: {
matches: 'id must match /[-_./a-zA-Z0-9]{8,64}/ regular expression',
},
property: 'id',
target: {
pleaseAck: {
on: ['RECEIPT'],
},
},
value: undefined,
},
'@id': '7517433f-1150-46f2-8495-723da61b872a',
'@type': 'https://didcomm.org/test-protocol/1.0/test-message',
},
TestMessage
)

await expect(MessageValidator.validate(transformedWithIncorrectValue)).rejects.toMatchObject([
{
children: [
{
children: [],
constraints: { isEnum: 'each value in on must be a valid enum value' },
property: 'on',
target: { on: ['NOT_A_VALID_VALUE'] },
value: ['NOT_A_VALID_VALUE'],
{
children: [],
constraints: {
matches: 'type must match /(.*?)([a-zA-Z0-9._-]+)\\/(\\d[^/]*)\\/([a-zA-Z0-9._-]+)$/ regular expression',
},
],
property: 'pleaseAck',
},
])
property: 'type',
target: {
pleaseAck: {
on: ['RECEIPT'],
},
},
value: undefined,
},
])
}
})
})
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,10 @@ describe('Decorators | ServiceDecoratorExtension', () => {
})

test('transforms Json to ServiceDecorator class', () => {
const transformed = JsonTransformer.fromJSON({ '~service': service }, TestMessage)
const transformed = JsonTransformer.fromJSON(
{ '@id': 'randomID', '@type': 'https://didcomm.org/fake-protocol/1.5/message', '~service': service },
TestMessage
)

expect(transformed.service).toEqual(service)
expect(transformed).toBeInstanceOf(TestMessage)
Expand Down
24 changes: 12 additions & 12 deletions packages/core/src/decorators/transport/TransportDecorator.test.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,14 @@
import { ClassValidationError } from '../../error/ClassValidationError'
import { JsonTransformer } from '../../utils/JsonTransformer'
import { MessageValidator } from '../../utils/MessageValidator'

import { TransportDecorator, ReturnRouteTypes } from './TransportDecorator'

const validTransport = (transportJson: Record<string, unknown>) =>
MessageValidator.validate(JsonTransformer.fromJSON(transportJson, TransportDecorator))
const expectValid = (transportJson: Record<string, unknown>) =>
expect(validTransport(transportJson)).resolves.toBeUndefined()
MessageValidator.validateSync(JsonTransformer.fromJSON(transportJson, TransportDecorator))
const expectValid = (transportJson: Record<string, unknown>) => expect(validTransport(transportJson)).toBeUndefined()
const expectInvalid = (transportJson: Record<string, unknown>) =>
expect(validTransport(transportJson)).rejects.not.toBeNull()
expect(() => validTransport(transportJson)).toThrowError(ClassValidationError)

const valid = {
all: {
Expand Down Expand Up @@ -60,18 +60,18 @@ describe('Decorators | TransportDecorator', () => {
expect(json).toEqual(transformed)
})

it('should only allow correct return_route values', async () => {
it('should only allow correct return_route values', () => {
expect.assertions(4)
await expectValid(valid.all)
await expectValid(valid.none)
await expectValid(valid.thread)
await expectInvalid(invalid.random)
expectValid(valid.all)
expectValid(valid.none)
expectValid(valid.thread)
expectInvalid(invalid.random)
})

it('should require return_route_thread when return_route is thread', async () => {
expect.assertions(3)
await expectValid(valid.thread)
await expectInvalid(invalid.invalidThreadId)
await expectInvalid(invalid.missingThreadId)
expectValid(valid.thread)
expectInvalid(invalid.invalidThreadId)
expectInvalid(invalid.missingThreadId)
})
})
24 changes: 24 additions & 0 deletions packages/core/src/error/ClassValidationError.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import type { ValidationError } from 'class-validator'

import { AriesFrameworkError } from './AriesFrameworkError'

export class ClassValidationError extends AriesFrameworkError {
public validationErrors: ValidationError[]

public validationErrorsToString() {
return this.validationErrors?.map((error) => error.toString(true)).join('\n') ?? ''
}

public constructor(
message: string,
{ classType, cause, validationErrors }: { classType: string; cause?: Error; validationErrors?: ValidationError[] }
) {
const validationErrorsStringified = validationErrors?.map((error) => error.toString()).join('\n')
super(
`${classType}: ${message}
${validationErrorsStringified}`,
{ cause }
)
this.validationErrors = validationErrors ?? []
}
}
9 changes: 9 additions & 0 deletions packages/core/src/error/ValidationErrorUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { ValidationError } from 'class-validator'

export function isValidationErrorArray(e: ValidationError[] | unknown): boolean {
if (Array.isArray(e)) {
const isErrorArray = e.length > 0 && e.every((err) => err instanceof ValidationError)
return isErrorArray
}
return false
}
24 changes: 24 additions & 0 deletions packages/core/src/error/__tests__/ValidationErrorUtils.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { ValidationError } from 'class-validator'

import { isValidationErrorArray } from '../ValidationErrorUtils'

describe('ValidationErrorUtils', () => {
test('returns true for an array of ValidationErrors', () => {
const error = new ValidationError()
const errorArray = [error, error]
const isErrorArray = isValidationErrorArray(errorArray)
expect(isErrorArray).toBeTruthy
})

test('returns false for an array of strings', () => {
const errorArray = ['hello', 'world']
const isErrorArray = isValidationErrorArray(errorArray)
expect(isErrorArray).toBeFalsy
})

test('returns false for a non array', () => {
const error = new ValidationError()
const isErrorArray = isValidationErrorArray(error)
expect(isErrorArray).toBeFalsy
})
})
1 change: 1 addition & 0 deletions packages/core/src/error/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,4 @@ export * from './AriesFrameworkError'
export * from './RecordNotFoundError'
export * from './RecordDuplicateError'
export * from './IndySdkError'
export * from './ClassValidationError'
Loading