File ‹output_style_custom.ML›

(*  Title:      SpecCheck/Output_Style/output_style_custom.ML
    Author:     Lukas Bulwahn, Nicolai Schaffroth, Kevin Kappelmann TU Muenchen
    Author:     Christopher League

Custom-made, QuickCheck-inspired output style for SpecCheck.
*)

signature SPECCHECK_OUTPUT_STYLE_CUSTOM =
sig
  include SPECCHECK_OUTPUT_STYLE
  val style_custom : (string -> unit) -> (string -> unit) ->
    'a SpecCheck_Output_Style_Types.output_style
end

structure SpecCheck_Output_Style_Custom : SPECCHECK_OUTPUT_STYLE_CUSTOM =
struct

open SpecCheck_Base

fun print_success writeln_info stats =
  let
    val num_success_tests = string_of_int (#num_success_tests stats)
    val num_discarded_tests = #num_discarded_tests stats
    val discarded_str =
      if num_discarded_tests = 0
      then "."
      else SpecCheck_Util.spaces [";", string_of_int num_discarded_tests,  "discarded."]
  in
    implode ["OK, passed ", num_success_tests, " tests", discarded_str]
    |> writeln_info
  end

fun print_gave_up writeln_warning stats =
  let
    val num_success_tests = string_of_int (#num_success_tests stats)
    val num_discarded_tests = string_of_int (#num_discarded_tests stats)
  in
    SpecCheck_Util.spaces ["Gave up! Passed only", num_success_tests, "test(s);", num_discarded_tests,
      "discarded test(s)."]
    |> writeln_warning
  end

fun print_failure_data writeln_warning ctxt show_opt failure_data =
  case #the_exception failure_data of
    SOME exn =>
      cat_lines ["Exception during test run:", exnMessage exn]
      |> writeln_warning
  | NONE => case show_opt of
      NONE => ()
    | SOME show =>
      let
        val sort_counterexamples = Config.get ctxt SpecCheck_Configuration.sort_counterexamples
        val maybe_sort = if sort_counterexamples then sort (int_ord o apply2 size) else I
        val counterexamples =
          #counterexamples failure_data
          |> map (Pretty.string_of o show)
          |> maybe_sort
      in fold (fn x => fn _ => writeln_warning x) counterexamples () end

fun print_failure writeln_warning ctxt show_opt (stats, failure_data) =
  ((SpecCheck_Util.spaces ["Failed! Falsified (after", string_of_int (num_tests stats), "test(s) and ",
    string_of_int (num_shrinks stats), "shrink(s)):"] |> writeln_warning);
  print_failure_data writeln_warning ctxt show_opt failure_data)

fun print_stats writeln_info ctxt stats total_time =
  let
    val show_stats = Config.get ctxt SpecCheck_Configuration.show_stats
    (*the time spent in the test function in relation to the total time spent;
      the latter includes generating test cases and overhead from the framework*)
    fun show_time {elapsed, ...} =
      implode ["Time: ", Time.toString elapsed, "s (run) / ", Time.toString (#elapsed total_time),
        "s (total)"]
  in
    if not show_stats
    then ()
    else writeln_info (show_time (#timing stats))
  end

fun style_custom writeln_info writeln_warning show_opt ctxt name total_time
  result =
  let val stats = stats_of_result result
  in
    writeln_info (SpecCheck_Util.spaces ["Testing:", name]);
    (case result of
      Success _ => print_success writeln_info stats
    | Gave_Up _ => print_gave_up writeln_warning stats
    | Failure data => print_failure writeln_warning ctxt show_opt data);
    print_stats writeln_info ctxt stats total_time
  end

(*pass `error` instead of `warning` to make builds fail on test failure.
TODO: passing `error` will block subsequent message printing; in particular,
the failed test case will not be printed to the build log.*)
fun style show_opt = style_custom writeln warning show_opt

end