Skip to content

Commit

Permalink
WIP: Check response payload.
Browse files Browse the repository at this point in the history
Signed-off-by: dblock <[email protected]>
  • Loading branch information
dblock committed Jun 19, 2024
1 parent 2134bcf commit 0dece09
Show file tree
Hide file tree
Showing 6 changed files with 55 additions and 9 deletions.
10 changes: 10 additions & 0 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@
"eslint-plugin-yml": "^1.14.0",
"globals": "^15.0.0",
"jest": "^29.7.0",
"json-diff-ts": "^4.0.1",
"json-schema-to-typescript": "^14.0.4",
"ts-jest": "^29.1.2"
}
Expand Down
4 changes: 4 additions & 0 deletions tests/_core/info.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,7 @@ chapters:
pretty: false
response:
status: 200
payload:
version:
distribution: opensearch
tagline: 'The OpenSearch Project: https://opensearch.org/'
34 changes: 29 additions & 5 deletions tools/src/tester/ChapterEvaluator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* compatible open source license.
*/

import { type Chapter, type ActualResponse } from './types/story.types'
import { type Chapter, type ActualResponse, type Payload } from './types/story.types'
import { type ChapterEvaluation, type Evaluation, Result } from './types/eval.types'
import { type ParsedOperation } from './types/spec.types'
import { overall_result } from './helpers'
Expand All @@ -16,6 +16,8 @@ import type OperationLocator from './OperationLocator'
import type SchemaValidator from './SchemaValidator'
import { type StoryOutputs } from './StoryOutputs'
import { ChapterOutput } from './ChapterOutput'
import { Operation, atomizeChangeset, diff } from 'json-diff-ts'
import _ from 'lodash'

export default class ChapterEvaluator {
private readonly _operation_locator: OperationLocator
Expand All @@ -36,13 +38,20 @@ export default class ChapterEvaluator {
const params = this.#evaluate_parameters(chapter, operation)
const request_body = this.#evaluate_request_body(chapter, operation)
const status = this.#evaluate_status(chapter, response)
const payload = status.result === Result.PASSED ? this.#evaluate_payload(response, operation) : { result: Result.SKIPPED }
const payload_body_evaluation = status.result === Result.PASSED ? this.#evaluate_payload_body(response, chapter.response?.payload) : { result: Result.SKIPPED }
const payload_schema_evaluation = status.result === Result.PASSED ? this.#evaluate_payload_schema(response, operation) : { result: Result.SKIPPED }
const output_values = ChapterOutput.extract_output_values(response, chapter.output)
return {
title: chapter.synopsis,
overall: { result: overall_result(Object.values(params).concat([request_body, status, payload]).concat(output_values ? [output_values] : [])) },
overall: { result: overall_result(Object.values(params).concat([
request_body, status, payload_body_evaluation, payload_schema_evaluation
]).concat(output_values ? [output_values] : [])) },
request: { parameters: params, request_body },
response: { status, payload },
response: {
status,
payload_body: payload_body_evaluation,
payload_schema: payload_schema_evaluation
},
...(output_values ? { output_values } : {})
}
}
Expand Down Expand Up @@ -74,7 +83,22 @@ export default class ChapterEvaluator {
}
}

#evaluate_payload(response: ActualResponse, operation: ParsedOperation): Evaluation {
#evaluate_payload_body(response: ActualResponse, expected_payload?: Payload): Evaluation {
if (expected_payload == null) return { result: Result.PASSED }
const delta = atomizeChangeset(diff(expected_payload, response.payload))
var messages: string[] = []
delta.forEach((value, _index, _array) => {
switch(value.type) {
case Operation.UPDATE:
return messages.push(`expected ${value.path.replace('$.', '')}='${value.oldValue}', got '${value.value}'`)
case Operation.REMOVE:
return messages.push(`missing ${value.path.replace('$.', '')}=${value.value}`)
}
})
return messages.length > 0 ? { result: Result.FAILED, message: _.join(messages, ', ')} : { result: Result.PASSED }
}

#evaluate_payload_schema(response: ActualResponse, operation: ParsedOperation): Evaluation {
const content_type = response.content_type ?? 'application/json'
const content = operation.responses[response.status]?.content[content_type]
const schema = content?.schema
Expand Down
12 changes: 9 additions & 3 deletions tools/src/tester/ResultLogger.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,8 @@ export class ConsoleResultLogger implements ResultLogger {
this.#log_parameters(chapter.request?.parameters ?? {})
this.#log_request_body(chapter.request?.request_body)
this.#log_status(chapter.response?.status)
this.#log_payload(chapter.response?.payload)
this.#log_payload_body(chapter.response?.payload_body)
this.#log_payload_schema(chapter.response?.payload_schema)
}

#log_parameters (parameters: Record<string, Evaluation>): void {
Expand All @@ -75,9 +76,14 @@ export class ConsoleResultLogger implements ResultLogger {
this.#log_evaluation(evaluation, 'RESPONSE STATUS', this._tab_width * 3)
}

#log_payload (evaluation: Evaluation | undefined): void {
#log_payload_body (evaluation: Evaluation | undefined): void {
if (evaluation == null) return
this.#log_evaluation(evaluation, 'RESPONSE PAYLOAD', this._tab_width * 3)
this.#log_evaluation(evaluation, 'RESPONSE PAYLOAD BODY', this._tab_width * 3)
}

#log_payload_schema (evaluation: Evaluation | undefined): void {
if (evaluation == null) return
this.#log_evaluation(evaluation, 'RESPONSE PAYLOAD SCHEMA', this._tab_width * 3)
}

#log_evaluation (evaluation: Evaluation, title: string, prefix: number = 0): void {
Expand Down
3 changes: 2 additions & 1 deletion tools/src/tester/types/eval.types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,8 @@ export interface ChapterEvaluation {
}
response?: {
status: Evaluation
payload: Evaluation
payload_body: Evaluation,
payload_schema: Evaluation
}
output_values?: EvaluationWithOutput
}
Expand Down

0 comments on commit 0dece09

Please sign in to comment.