Skip to content

Commit

Permalink
Fix: display output collection errors. (#450)
Browse files Browse the repository at this point in the history
* Fix: display output collection errors.

Signed-off-by: dblock <[email protected]>
  • Loading branch information
dblock authored Jul 25, 2024
1 parent b9bdac2 commit b9388d4
Show file tree
Hide file tree
Showing 15 changed files with 165 additions and 39 deletions.
32 changes: 23 additions & 9 deletions tools/src/tester/ChapterEvaluator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
*/

import { type Chapter, type ActualResponse, type Payload } from './types/story.types'
import { type ChapterEvaluation, type Evaluation, Result } from './types/eval.types'
import { type ChapterEvaluation, type Evaluation, Result, EvaluationWithOutput } from './types/eval.types'
import { type ParsedOperation } from './types/spec.types'
import { overall_result } from './helpers'
import type ChapterReader from './ChapterReader'
Expand Down Expand Up @@ -45,21 +45,35 @@ export default class ChapterEvaluator {
const status = this.#evaluate_status(chapter, response)
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(chapter, response, operation) : { result: Result.SKIPPED }
const output_values = ChapterOutput.extract_output_values(response, chapter.output)
return {
const output_values_evaluation: EvaluationWithOutput = status.result === Result.PASSED ? ChapterOutput.extract_output_values(response, chapter.output) : { evaluation: { result: Result.SKIPPED } }

const evaluations = _.compact(_.concat(
Object.values(params),
request_body,
status,
payload_body_evaluation,
payload_schema_evaluation,
output_values_evaluation.evaluation
))

var result: ChapterEvaluation = {
title: chapter.synopsis,
path: `${chapter.method} ${chapter.path}`,
overall: { result: overall_result(Object.values(params).concat([
request_body, status, payload_body_evaluation, payload_schema_evaluation
]).concat(output_values ? [output_values] : [])) },
overall: { result: overall_result(evaluations) },
request: { parameters: params, request_body },
response: {
status,
payload_body: payload_body_evaluation,
payload_schema: payload_schema_evaluation
},
...(output_values ? { output_values } : {})
payload_schema: payload_schema_evaluation,
output_values: output_values_evaluation.evaluation
}
}

if (output_values_evaluation?.output !== undefined) {
result.output = output_values_evaluation?.output
}

return result
}

#evaluate_parameters(chapter: Chapter, operation: ParsedOperation): Record<string, Evaluation> {
Expand Down
12 changes: 6 additions & 6 deletions tools/src/tester/ChapterOutput.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,26 +26,26 @@ export class ChapterOutput {
this.outputs[name] = value
}

static extract_output_values(response: ActualResponse, output?: Output): EvaluationWithOutput | undefined {
if (!output) return undefined
static extract_output_values(response: ActualResponse, output?: Output): EvaluationWithOutput {
if (!output) return { evaluation: { result: Result.SKIPPED } }
const chapter_output = new ChapterOutput({})
for (const [name, path] of Object.entries(output)) {
const [source, ...rest] = path.split('.')
const keys = rest.join('.')
let value: any
if (source === 'payload') {
if (response.payload === undefined) {
return { result: Result.ERROR, message: 'No payload found in response, but expected output: ' + path }
return { evaluation: { result: Result.ERROR, message: 'No payload found in response, but expected output: ' + path } }
}
value = keys.length === 0 ? response.payload : _.get(response.payload, keys)
if (value === undefined) {
return { result: Result.ERROR, message: `Expected to find non undefined value at \`${path}\`.` }
return { evaluation: { result: Result.ERROR, message: `Expected to find non undefined value at \`${path}\`.` } }
}
} else {
return { result: Result.ERROR, message: 'Unknown output source: ' + source }
return { evaluation: { result: Result.ERROR, message: 'Unknown output source: ' + source } }
}
chapter_output.set(name, value)
}
return { result: Result.PASSED, output: chapter_output }
return { evaluation: { result: Result.PASSED }, output: chapter_output }
}
}
6 changes: 6 additions & 0 deletions tools/src/tester/ResultLogger.ts
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ export class ConsoleResultLogger implements ResultLogger {
this.#log_status(chapter.response?.status)
this.#log_payload_body(chapter.response?.payload_body)
this.#log_payload_schema(chapter.response?.payload_schema)
this.#log_output_values(chapter.response?.output_values)
}

#log_parameters (parameters: Record<string, Evaluation>): void {
Expand Down Expand Up @@ -96,6 +97,11 @@ export class ConsoleResultLogger implements ResultLogger {
this.#log_evaluation(evaluation, 'RESPONSE PAYLOAD SCHEMA', this._tab_width * 3)
}

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

#log_evaluation (evaluation: Evaluation, title: string, prefix: number = 0): void {
const result = ansi.padding(this.#result(evaluation.result), 0, prefix)

Expand Down
10 changes: 5 additions & 5 deletions tools/src/tester/StoryEvaluator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,8 +67,8 @@ export default class StoryEvaluator {
} else {
const evaluation = await this._chapter_evaluator.evaluate(chapter, has_errors, story_outputs)
has_errors = has_errors || evaluation.overall.result === Result.ERROR
if (evaluation.output_values?.output !== undefined && chapter.id !== undefined) {
story_outputs.set_chapter_output(chapter.id, evaluation.output_values?.output)
if (evaluation.output !== undefined && chapter.id !== undefined) {
story_outputs.set_chapter_output(chapter.id, evaluation.output)
}
evaluations.push(evaluation)
}
Expand All @@ -86,8 +86,8 @@ export default class StoryEvaluator {
} else {
const { evaluation, evaluation_error } = await this._supplemental_chapter_evaluator.evaluate(chapter, story_outputs)
has_errors = has_errors || evaluation_error
if (evaluation.output_values?.output !== undefined && chapter.id !== undefined) {
story_outputs.set_chapter_output(chapter.id, evaluation.output_values?.output)
if (evaluation.output !== undefined && chapter.id !== undefined) {
story_outputs.set_chapter_output(chapter.id, evaluation.output)
}
evaluations.push(evaluation)
}
Expand Down Expand Up @@ -117,7 +117,7 @@ export default class StoryEvaluator {
prologues,
chapters,
epilogues,
message: 'The story was defined with incorrect variables'
message: 'The story was defined with incorrect variables.'
}
}
}
Expand Down
42 changes: 32 additions & 10 deletions tools/src/tester/SupplementalChapterEvaluator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
* compatible open source license.
*/

import _ from "lodash";
import { ChapterOutput } from "./ChapterOutput";
import ChapterReader from "./ChapterReader";
import { StoryOutputs } from "./StoryOutputs";
Expand All @@ -25,34 +26,55 @@ export default class SupplementalChapterEvaluator {
const title = `${chapter.method} ${chapter.path}`
const response = await this._chapter_reader.read(chapter, story_outputs)
const status = chapter.status ?? [200, 201]
const output_values = ChapterOutput.extract_output_values(response, chapter.output)
const output_values_evaluation = ChapterOutput.extract_output_values(response, chapter.output)
let response_evaluation: ChapterEvaluation
const passed_evaluation = { title, overall: { result: Result.PASSED } }
if (status.includes(response.status)) {
response_evaluation = passed_evaluation
} else {
response_evaluation = { title, overall: { result: Result.ERROR, message: response.message, error: response.error as Error }, output_values }
response_evaluation = {
title,
overall: {
result: Result.ERROR,
message: response.message,
error: response.error as Error
}
}
}
if (output_values) {
response_evaluation.output_values = output_values

if (output_values_evaluation.output) {
response_evaluation.output = output_values_evaluation.output
}
const result = overall_result([response_evaluation.overall].concat(output_values ? [output_values] : []))

const result = overall_result(_.compact([
response_evaluation.overall,
output_values_evaluation.evaluation
]))

if (result === Result.PASSED) {
return { evaluation: passed_evaluation, evaluation_error: false }
} else {
const message_segments = []

if (response_evaluation.overall.result === Result.ERROR) {
message_segments.push(`${response_evaluation.overall.message}`)
}
if (output_values !== undefined && output_values.result === Result.ERROR) {
message_segments.push(`${output_values.message}`)

if (output_values_evaluation.evaluation.message !== undefined && output_values_evaluation.evaluation.result === Result.ERROR) {
message_segments.push(`${output_values_evaluation.evaluation.message}`)
}

const message = message_segments.join('\n')
const evaluation = {

var evaluation: ChapterEvaluation = {
title,
overall: { result: Result.ERROR, message, error: response.error as Error },
...(output_values ? { output_values } : {})
overall: { result: Result.ERROR, message, error: response.error as Error }
}

if (output_values_evaluation.output) {
evaluation.output = output_values_evaluation.output
}

return { evaluation, evaluation_error: true }
}
}
Expand Down
8 changes: 5 additions & 3 deletions tools/src/tester/types/eval.types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,8 +44,9 @@ export interface ChapterEvaluation {
status: Evaluation
payload_body: Evaluation,
payload_schema: Evaluation
}
output_values?: EvaluationWithOutput
output_values: Evaluation
},
output?: ChapterOutput
}

export class ChaptersEvaluations {
Expand All @@ -63,7 +64,8 @@ export interface Evaluation {
error?: Error | string
}

export type EvaluationWithOutput = Evaluation & {
export type EvaluationWithOutput = {
evaluation: Evaluation,
output?: ChapterOutput
}

Expand Down
2 changes: 2 additions & 0 deletions tools/tests/tester/fixtures/evals/error/chapter_error.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ chapters:
result: SKIPPED
payload_schema:
result: SKIPPED
output_values:
result: SKIPPED
- title: This chapter should be skipped.
overall:
result: SKIPPED
Expand Down
29 changes: 29 additions & 0 deletions tools/tests/tester/fixtures/evals/error/output_error.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
display_path: error/output_error.yaml
full_path: tools/tests/tester/fixtures/stories/error/output_error.yaml

result: ERROR
description: This story has an error in the output.
prologues: []
epilogues: []
chapters:
- title: This chapter expects a `cursor` in the output.
overall:
result: ERROR
path: GET /_cat/health
request:
parameters:
format:
result: PASSED
request_body:
result: PASSED
response:
payload_body:
result: PASSED
payload_schema:
result: PASSED
output_values:
result: ERROR
message: Expected to find non undefined value at `payload.does_not_exist`.
status:
result: PASSED

10 changes: 10 additions & 0 deletions tools/tests/tester/fixtures/evals/failed/invalid_data.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ chapters:
result: PASSED
payload_schema:
result: PASSED
output_values:
result: SKIPPED
- title: This chapter should fail because the request body is invalid.
overall:
result: FAILED
Expand All @@ -43,6 +45,8 @@ chapters:
result: PASSED
payload_schema:
result: PASSED
output_values:
result: SKIPPED
- title: This chapter should fail because the response content type does not match.
overall:
result: FAILED
Expand All @@ -63,6 +67,8 @@ chapters:
payload_schema:
result: FAILED
message: 'Expected content type application/json, but received application/yaml.'
output_values:
result: SKIPPED
- title: This chapter should fail because the response data and schema are invalid.
overall:
result: FAILED
Expand All @@ -82,6 +88,8 @@ chapters:
payload_schema:
result: FAILED
message: 'data contains unsupported properties: acknowledged'
output_values:
result: SKIPPED
- title: This chapter should fail because the response status does not match.
overall:
result: ERROR
Expand All @@ -100,6 +108,8 @@ chapters:
result: SKIPPED
payload_schema:
result: SKIPPED
output_values:
result: SKIPPED

epilogues:
- title: DELETE /books
Expand Down
6 changes: 6 additions & 0 deletions tools/tests/tester/fixtures/evals/failed/not_found.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,8 @@ chapters:
result: PASSED
payload_schema:
result: PASSED
output_values:
result: SKIPPED
- title: This chapter should fail because the request body is not defined in the spec.
overall:
result: FAILED
Expand All @@ -49,6 +51,8 @@ chapters:
result: PASSED
payload_schema:
result: PASSED
output_values:
result: SKIPPED
- title: This chapter should fail because the response is not defined in the spec.
overall:
result: FAILED
Expand All @@ -67,6 +71,8 @@ chapters:
payload_schema:
result: FAILED
message: 'Schema for "404: application/json" response not found in the spec.'
output_values:
result: SKIPPED

epilogues:
- title: DELETE /books
Expand Down
Loading

0 comments on commit b9388d4

Please sign in to comment.