Skip to content

Commit

Permalink
Add proverb exercise
Browse files Browse the repository at this point in the history
  • Loading branch information
keiravillekode committed Feb 16, 2025
1 parent 3dbdc5c commit cc29ea2
Show file tree
Hide file tree
Showing 8 changed files with 344 additions and 0 deletions.
8 changes: 8 additions & 0 deletions config.json
Original file line number Diff line number Diff line change
Expand Up @@ -383,6 +383,14 @@
"prerequisites": [],
"difficulty": 3
},
{
"slug": "proverb",
"name": "Proverb",
"uuid": "e4d5a3e5-051d-4c0a-ae24-5e055752b90d",
"practices": [],
"prerequisites": [],
"difficulty": 3
},
{
"slug": "rotational-cipher",
"name": "Rotational Cipher",
Expand Down
19 changes: 19 additions & 0 deletions exercises/practice/proverb/.docs/instructions.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# Instructions

For want of a horseshoe nail, a kingdom was lost, or so the saying goes.

Given a list of inputs, generate the relevant proverb.
For example, given the list `["nail", "shoe", "horse", "rider", "message", "battle", "kingdom"]`, you will output the full text of this proverbial rhyme:

```text
For want of a nail the shoe was lost.
For want of a shoe the horse was lost.
For want of a horse the rider was lost.
For want of a rider the message was lost.
For want of a message the battle was lost.
For want of a battle the kingdom was lost.
And all for the want of a nail.
```

Note that the list of inputs may vary; your solution should be able to handle lists of arbitrary length and content.
No line of the output text should be a static, unchanging string; all should vary according to the input given.
19 changes: 19 additions & 0 deletions exercises/practice/proverb/.meta/config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
{
"authors": [
"keiravillekode"
],
"files": {
"solution": [
"proverb.sml"
],
"test": [
"test.sml"
],
"example": [
".meta/example.sml"
]
},
"blurb": "For want of a horseshoe nail, a kingdom was lost, or so the saying goes. Output the full text of this proverbial rhyme.",
"source": "Wikipedia",
"source_url": "https://en.wikipedia.org/wiki/For_Want_of_a_Nail"
}
9 changes: 9 additions & 0 deletions exercises/practice/proverb/.meta/example.sml
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
local
fun recurse (previous: string, inputs: string list): string =
case inputs of
nil => ""
| current :: rest => "For want of a " ^ previous ^ " the " ^ current ^ " was lost.\n" ^ (recurse (current, rest))
in
fun recite nil = ""
| recite (first :: rest) = (recurse (first, rest)) ^ "And all for the want of a " ^ first ^ "."
end
28 changes: 28 additions & 0 deletions exercises/practice/proverb/.meta/tests.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
# This is an auto-generated file.
#
# Regenerating this file via `configlet sync` will:
# - Recreate every `description` key/value pair
# - Recreate every `reimplements` key/value pair, where they exist in problem-specifications
# - Remove any `include = true` key/value pair (an omitted `include` key implies inclusion)
# - Preserve any other key/value pair
#
# As user-added comments (using the # character) will be removed when this file
# is regenerated, comments can be added via a `comment` key.

[e974b73e-7851-484f-8d6d-92e07fe742fc]
description = "zero pieces"

[2fcd5f5e-8b82-4e74-b51d-df28a5e0faa4]
description = "one piece"

[d9d0a8a1-d933-46e2-aa94-eecf679f4b0e]
description = "two pieces"

[c95ef757-5e94-4f0d-a6cb-d2083f5e5a83]
description = "three pieces"

[433fb91c-35a2-4d41-aeab-4de1e82b2126]
description = "full proverb"

[c1eefa5a-e8d9-41c7-91d4-99fab6d6b9f7]
description = "four pieces modernized"
2 changes: 2 additions & 0 deletions exercises/practice/proverb/proverb.sml
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
fun recite (inputs: string list): string =
raise Fail "'recite' is not implemented"
99 changes: 99 additions & 0 deletions exercises/practice/proverb/test.sml
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
(* version 1.0.0 *)

use "testlib.sml";
use "proverb.sml";

infixr |>
fun x |> f = f x

val testsuite =
describe "proverb" [
test "zero pieces"
(fn _ => let
val inputs = []
val expected = ""
in
recite inputs |> Expect.equalTo expected
end),

test "one piece"
(fn _ => let
val inputs = [
"nail"
]
val expected =
"And all for the want of a nail."
in
recite inputs |> Expect.equalTo expected
end),

test "two pieces"
(fn _ => let
val inputs = [
"nail",
"shoe"
]
val expected =
"For want of a nail the shoe was lost.\n\
\And all for the want of a nail."
in
recite inputs |> Expect.equalTo expected
end),

test "three pieces"
(fn _ => let
val inputs = [
"nail",
"shoe",
"horse"
]
val expected =
"For want of a nail the shoe was lost.\n\
\For want of a shoe the horse was lost.\n\
\And all for the want of a nail."
in
recite inputs |> Expect.equalTo expected
end),

test "full proverb"
(fn _ => let
val inputs = [
"nail",
"shoe",
"horse",
"rider",
"message",
"battle",
"kingdom"
]
val expected =
"For want of a nail the shoe was lost.\n\
\For want of a shoe the horse was lost.\n\
\For want of a horse the rider was lost.\n\
\For want of a rider the message was lost.\n\
\For want of a message the battle was lost.\n\
\For want of a battle the kingdom was lost.\n\
\And all for the want of a nail."
in
recite inputs |> Expect.equalTo expected
end),

test "four pieces modernized"
(fn _ => let
val inputs = [
"pin",
"gun",
"soldier",
"battle"
]
val expected =
"For want of a pin the gun was lost.\n\
\For want of a gun the soldier was lost.\n\
\For want of a soldier the battle was lost.\n\
\And all for the want of a pin."
in
recite inputs |> Expect.equalTo expected
end)
]

val _ = Test.run testsuite
160 changes: 160 additions & 0 deletions exercises/practice/proverb/testlib.sml
Original file line number Diff line number Diff line change
@@ -0,0 +1,160 @@
structure Expect =
struct
datatype expectation = Pass | Fail of string * string

local
fun failEq b a =
Fail ("Expected: " ^ b, "Got: " ^ a)

fun failExn b a =
Fail ("Expected: " ^ b, "Raised: " ^ a)

fun exnName (e: exn): string = General.exnName e
in
fun truthy a =
if a
then Pass
else failEq "true" "false"

fun falsy a =
if a
then failEq "false" "true"
else Pass

fun equalTo b a =
if a = b
then Pass
else failEq (PolyML.makestring b) (PolyML.makestring a)

fun nearTo delta b a =
if Real.abs (a - b) <= delta * Real.abs a orelse
Real.abs (a - b) <= delta * Real.abs b
then Pass
else failEq (Real.toString b ^ " +/- " ^ Real.toString delta) (Real.toString a)

fun anyError f =
(
f ();
failExn "an exception" "Nothing"
) handle _ => Pass

fun error e f =
(
f ();
failExn (exnName e) "Nothing"
) handle e' => if exnMessage e' = exnMessage e
then Pass
else failExn (exnMessage e) (exnMessage e')
end
end

structure TermColor =
struct
datatype color = Red | Green | Yellow | Normal

fun f Red = "\027[31m"
| f Green = "\027[32m"
| f Yellow = "\027[33m"
| f Normal = "\027[0m"

fun colorize color s = (f color) ^ s ^ (f Normal)

val redit = colorize Red

val greenit = colorize Green

val yellowit = colorize Yellow
end

structure Test =
struct
datatype testnode = TestGroup of string * testnode list
| Test of string * (unit -> Expect.expectation)

local
datatype evaluation = Success of string
| Failure of string * string * string
| Error of string * string

fun indent n s = (implode (List.tabulate (n, fn _ => #" "))) ^ s

fun fmt indentlvl ev =
let
val check = TermColor.greenit "\226\156\148 " (**)
val cross = TermColor.redit "\226\156\150 " (**)
val indentlvl = indentlvl * 2
in
case ev of
Success descr => indent indentlvl (check ^ descr)
| Failure (descr, exp, got) =>
String.concatWith "\n" [indent indentlvl (cross ^ descr),
indent (indentlvl + 2) exp,
indent (indentlvl + 2) got]
| Error (descr, reason) =>
String.concatWith "\n" [indent indentlvl (cross ^ descr),
indent (indentlvl + 2) (TermColor.redit reason)]
end

fun eval (TestGroup _) = raise Fail "Only a 'Test' can be evaluated"
| eval (Test (descr, thunk)) =
(
case thunk () of
Expect.Pass => ((1, 0, 0), Success descr)
| Expect.Fail (s, s') => ((0, 1, 0), Failure (descr, s, s'))
)
handle e => ((0, 0, 1), Error (descr, "Unexpected error: " ^ exnMessage e))

fun flatten depth testnode =
let
fun sum (x, y, z) (a, b, c) = (x + a, y + b, z + c)

fun aux (t, (counter, acc)) =
let
val (counter', texts) = flatten (depth + 1) t
in
(sum counter' counter, texts :: acc)
end
in
case testnode of
TestGroup (descr, ts) =>
let
val (counter, texts) = foldr aux ((0, 0, 0), []) ts
in
(counter, (indent (depth * 2) descr) :: List.concat texts)
end
| Test _ =>
let
val (counter, evaluation) = eval testnode
in
(counter, [fmt depth evaluation])
end
end

fun println s = print (s ^ "\n")
in
fun run suite =
let
val ((succeeded, failed, errored), texts) = flatten 0 suite

val summary = String.concatWith ", " [
TermColor.greenit ((Int.toString succeeded) ^ " passed"),
TermColor.redit ((Int.toString failed) ^ " failed"),
TermColor.redit ((Int.toString errored) ^ " errored"),
(Int.toString (succeeded + failed + errored)) ^ " total"
]

val status = if failed = 0 andalso errored = 0
then OS.Process.success
else OS.Process.failure

in
List.app println texts;
println "";
println ("Tests: " ^ summary);
OS.Process.exit status
end
end
end

fun describe description tests = Test.TestGroup (description, tests)
fun test description thunk = Test.Test (description, thunk)

0 comments on commit cc29ea2

Please sign in to comment.