Improving the tutorial.
This commit is contained in:
parent
64ef8c1ac8
commit
13cab23e20
@ -56,9 +56,92 @@ $lessons = [
|
|||||||
# Tests:
|
# Tests:
|
||||||
assume (the sum of [1, 2, 3, 4, 5]) == 15
|
assume (the sum of [1, 2, 3, 4, 5]) == 15
|
||||||
assume (the sum of [100, 200]) == 300
|
assume (the sum of [100, 200]) == 300
|
||||||
|
|
||||||
|
lesson "Variable Scopes":
|
||||||
|
# A nomsu variable that has not yet been assigned to is (nil)
|
||||||
|
assume $never_assigned == (nil)
|
||||||
|
|
||||||
|
# Nomsu's variables are local by default, and actions have their own scopes:
|
||||||
|
$x = 1
|
||||||
|
$y = 2
|
||||||
|
|
||||||
|
# Define an action that sets a variable:
|
||||||
|
(do something) means:
|
||||||
|
# The variable $y is never set in this action, so it has the same value
|
||||||
|
it has outside this action.
|
||||||
|
assume $y == (???)
|
||||||
|
|
||||||
|
# $x is set inside this action, and actions have their own scopes.
|
||||||
|
$x = $y
|
||||||
|
|
||||||
|
# What number should $x be here?
|
||||||
|
assume $x == (???)
|
||||||
|
|
||||||
|
# After running the action, what value should $x have?
|
||||||
|
do something
|
||||||
|
assume $x == (???)
|
||||||
|
|
||||||
|
lesson "More Variable Scopes":
|
||||||
|
# Loops and conditionals do *not* have their own scopes:
|
||||||
|
$z = 1
|
||||||
|
if (1 == 1):
|
||||||
|
# Set $z inside a conditional:
|
||||||
|
$z = 2
|
||||||
|
|
||||||
|
# After assigning in a conditional, what should $z be?
|
||||||
|
assume $z == (???)
|
||||||
|
for $ in 1 to 1:
|
||||||
|
# Set $z inside a loop:
|
||||||
|
$z = 3
|
||||||
|
|
||||||
|
# After assigning in a loop, what should $z be?
|
||||||
|
assume $z == (???)
|
||||||
|
|
||||||
|
lesson "Externals":
|
||||||
|
# The 'external' block lets you modify variables outside an action:
|
||||||
|
$x = 1
|
||||||
|
(do something) means:
|
||||||
|
external: $x = 2
|
||||||
|
do something
|
||||||
|
|
||||||
|
# After running the action that sets $x in an 'external' block, what should $x be?
|
||||||
|
assume $x == (???)
|
||||||
|
|
||||||
|
lesson "Locals":
|
||||||
|
# The 'with' block lets you create a local scope for the variables you list:
|
||||||
|
$y = 1
|
||||||
|
$z = 1
|
||||||
|
with [$y]:
|
||||||
|
$y = 2
|
||||||
|
$z = 2
|
||||||
|
|
||||||
|
# After setting $y and $z in the 'with [$y]' block, what should $y and $z be?
|
||||||
|
assume $y == (???)
|
||||||
|
assume $z == (???)
|
||||||
|
|
||||||
|
lesson "Failure and Recovery":
|
||||||
|
$what_happened = "nothing"
|
||||||
|
# In Nomsu, sometimes things fail, but you can recover from failures with 'try':
|
||||||
|
try:
|
||||||
|
# The 'fail' action triggers failure
|
||||||
|
fail "Oh no!"
|
||||||
|
..if it fails:
|
||||||
|
$what_happened = "failure"
|
||||||
|
..if it succeeds:
|
||||||
|
$what_happened = "success"
|
||||||
|
|
||||||
|
# What do you think happened?
|
||||||
|
assume $what_happened == (???)
|
||||||
|
|
||||||
|
# Note: a 'try' block will silence failures, so this has no effect:
|
||||||
|
try: fail
|
||||||
]
|
]
|
||||||
|
|
||||||
command line program with $args:
|
command line program with $args:
|
||||||
|
if ($args.help or $args.h):
|
||||||
|
say "Nomsu tutorial usage: nomsu -t tutorial [-x] [<location to put tutorial files>]"
|
||||||
|
exit
|
||||||
|
|
||||||
say
|
say
|
||||||
bold ("
|
bold ("
|
||||||
|
|
||||||
@ -69,84 +152,103 @@ command line program with $args:
|
|||||||
")
|
")
|
||||||
|
|
||||||
if ($args.extras is empty):
|
if ($args.extras is empty):
|
||||||
say (bold (red "Please specify a directory where the tutorial files will go."))
|
$tutorial_dir = "./nomsu_tutorial"
|
||||||
say "For example: nomsu -t tutorial ~/nomsu_tutorial"
|
if (not ($Files.exists $tutorial_dir)):
|
||||||
exit 1
|
when (ask "The Nomsu tutorial files will be in \(bold $tutorial_dir) is that okay? [Y/n] ") is:
|
||||||
|
"n" "N" "no":
|
||||||
|
say "Okay. If you want to specify another file location, run `nomsu -t tutorial <your location>`"
|
||||||
|
exit
|
||||||
|
..else:
|
||||||
|
$tutorial_dir = $args.extras.1
|
||||||
|
|
||||||
if (not ($Files.exists $args.extras.1)):
|
if (not ($Files.exists $tutorial_dir)):
|
||||||
sh> "mkdir \($args.extras.1)"
|
sh> "mkdir \$tutorial_dir"
|
||||||
|
|
||||||
say "The tutorial files are located in \($args.extras.1)"
|
if ($args.x):
|
||||||
|
say ("
|
||||||
|
The tutorial files are located in \$tutorial_dir.
|
||||||
|
You're in charge of editing them yourself.
|
||||||
|
")
|
||||||
|
..else:
|
||||||
$EDITOR = (($os.getenv "EDITOR") or "nano")
|
$EDITOR = (($os.getenv "EDITOR") or "nano")
|
||||||
(filename of $i) means "\($args.extras.1)/lesson\$i.nom"
|
say ("
|
||||||
|
The tutorial will use \(yellow $EDITOR) for editing files.
|
||||||
|
If you'd rather edit the files in another window, use the \(yellow "-x") flag.
|
||||||
|
e.g. \(yellow "nomsu -t tutorial -x tutorials/")
|
||||||
|
")
|
||||||
|
(filename of $i) means ("\$tutorial_dir/lesson\$i.nom", with "//" -> "/")
|
||||||
for $lesson in $lessons at $i:
|
for $lesson in $lessons at $i:
|
||||||
$filename = (filename of $i)
|
$filename = (filename of $i)
|
||||||
unless ($Files.exists $filename):
|
unless ($Files.exists $filename):
|
||||||
write $lesson.lesson to file $filename
|
write $lesson.lesson to file $filename
|
||||||
$prev_pass = (nil)
|
|
||||||
repeat:
|
(get failures) means [
|
||||||
say ""
|
|
||||||
say (bold "Lessons:")
|
|
||||||
$failures = [
|
|
||||||
: for $lesson in $lessons at $i:
|
: for $lesson in $lessons at $i:
|
||||||
$filename = (filename of $i)
|
$filename = (filename of $i)
|
||||||
$file = (read file $filename)
|
$file = (read file $filename)
|
||||||
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
|
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
|
||||||
try:
|
try:
|
||||||
run $file
|
run $file
|
||||||
..if it fails with $msg: add $msg
|
..if it fails with $msg:
|
||||||
..if it succeeds:
|
$msg = ($msg, with "\n *stack traceback:.*" -> "")
|
||||||
add (no)
|
add {.lesson_number = $i, .failure = $msg}
|
||||||
]
|
]
|
||||||
$first_failure = (nil)
|
|
||||||
$pass = 0
|
|
||||||
for $lesson in $lessons at $i:
|
|
||||||
$filename = (filename of $i)
|
|
||||||
if $failures.$i:
|
|
||||||
say (bold (red " \$i. \($lesson.name) [incomplete]"))
|
|
||||||
$first_failure or= $i
|
|
||||||
..else:
|
|
||||||
say (bold (green " \$i. \($lesson.name) [passed]"))
|
|
||||||
$pass += 1
|
|
||||||
say ""
|
say ""
|
||||||
when:
|
say (bold "Lessons:")
|
||||||
($pass == $prev_pass):
|
|
||||||
say "\(bold "Your progress:") \(20 wide ($pass / (#$lessons)) progress bar)"
|
|
||||||
say "\n\(bold (red "Sorry, that didn't work :("))"
|
|
||||||
|
|
||||||
($prev_pass and ($prev_pass != $pass)):
|
(show first failure from $failures) means:
|
||||||
$N = 100
|
|
||||||
for $ in 0 to $N:
|
|
||||||
$k = (($ / $N) smoothed by 2)
|
|
||||||
$progress = (($prev_pass to $pass mixed by $k) / (#$lessons))
|
|
||||||
say "\r\(bold "Your progress:") \(20 wide $progress progress bar)" inline
|
|
||||||
$io.flush()
|
|
||||||
sh> "sleep \(1 / $N)"
|
|
||||||
say "\n\n\(bold (green "Nice job!"))"
|
|
||||||
|
|
||||||
else:
|
|
||||||
say "\(bold "Your progress:") \(20 wide ($pass / (#$lessons)) progress bar)"
|
|
||||||
$prev_pass = $pass
|
|
||||||
if $first_failure:
|
|
||||||
say ("
|
say ("
|
||||||
|
|
||||||
\(bold "Next thing to fix:") \(
|
\(bold "Next thing to fix:") \(
|
||||||
bold (red "Lesson \$first_failure: \($lessons.$first_failure.name)")
|
bold (yellow "Lesson \($failures.1.lesson_number): \($lessons.($failures.1.lesson_number).name)")
|
||||||
)
|
)
|
||||||
|
|
||||||
\($failures.$first_failure, indented)
|
\($failures.1.failure, indented)
|
||||||
|
|
||||||
")
|
")
|
||||||
|
|
||||||
$confirm =
|
$failures = (get failures)
|
||||||
ask
|
for $lesson in $lessons at $i:
|
||||||
bold "Do you want to edit \(filename of $first_failure) to get it to pass? [Y/n] "
|
for $f in $failures:
|
||||||
|
if ($f.lesson_number == $i):
|
||||||
|
say (bold (red " \$i. \($lesson.name) [incomplete]"))
|
||||||
|
do next $lesson
|
||||||
|
say (bold (green " \$i. \($lesson.name) [passed]"))
|
||||||
|
say "\n\(bold "Your progress:") \(20 wide (((#$lessons) - (#$failures)) / (#$lessons)) progress bar)"
|
||||||
|
|
||||||
unless {.n, .no, .q}.$confirm:
|
repeat while (not ($failures is empty)):
|
||||||
$filename = (filename of $first_failure)
|
show first failure from $failures
|
||||||
|
if ($args.x):
|
||||||
|
$filename = (filename of $failures.1.lesson_number)
|
||||||
|
say "\(yellow "Waiting for you to fix ")\(bold $filename) \(yellow "(press ctrl+c to exit)...")"
|
||||||
|
try:
|
||||||
|
$files = [: for $ in 1 to (#$lessons): add (read file (filename of $))]
|
||||||
|
repeat:
|
||||||
|
sh> "sleep 0.5"
|
||||||
|
$new_files = [: for $ in 1 to (#$lessons): add (read file (filename of $))]
|
||||||
|
if ($new_files != $files):
|
||||||
|
$files = $new_files
|
||||||
|
stop
|
||||||
|
..if it fails:
|
||||||
|
say "\nGoodbye."
|
||||||
|
exit
|
||||||
|
..else:
|
||||||
|
$filename = (filename of $failures.1.lesson_number)
|
||||||
|
--- (retry file) ---
|
||||||
|
when
|
||||||
|
ask (bold (cyan "Edit \$filename to get it to pass? [Y/n/exit] "))
|
||||||
|
..is:
|
||||||
|
"q" "quit" "exit" "n" "N" "no":
|
||||||
|
exit
|
||||||
|
|
||||||
|
"y" "Y" "yes" "":
|
||||||
$f = (read file $filename)
|
$f = (read file $filename)
|
||||||
$pos = (($f, find "<your code here>" 1 (yes)) or ($f, find "???" 1 (yes)))
|
$cursor_positions = []
|
||||||
if $pos:
|
$cursor_positions, add ($f, position of "<your code here>" 1 (yes))
|
||||||
|
$cursor_positions, add ($f, position of "???" 1 (yes))
|
||||||
|
unless ($cursor_positions is empty):
|
||||||
|
$pos = (min of $cursor_positions)
|
||||||
[$line, $col] = [($f, line number at $pos), ($f, line position at $pos)]
|
[$line, $col] = [($f, line number at $pos), ($f, line position at $pos)]
|
||||||
when:
|
when:
|
||||||
($EDITOR, matches "vim$"):
|
($EDITOR, matches "vim$"):
|
||||||
@ -162,15 +264,45 @@ command line program with $args:
|
|||||||
sh> "\$EDITOR \$filename"
|
sh> "\$EDITOR \$filename"
|
||||||
..else:
|
..else:
|
||||||
sh> "\$EDITOR \$filename"
|
sh> "\$EDITOR \$filename"
|
||||||
..else: stop
|
else:
|
||||||
|
say "Sorry, I don't understand that."
|
||||||
|
go to (retry file)
|
||||||
|
|
||||||
|
$file = (read file $filename)
|
||||||
|
$file = ($NomsuCode, from (Source $filename 1 (#$file)) $file)
|
||||||
|
try:
|
||||||
|
run $file
|
||||||
|
..if it fails with $msg:
|
||||||
|
say (bold (red "\n There's a bit more to fix:"))
|
||||||
|
$msg = ($msg, with "\n *stack traceback:.*" -> "")
|
||||||
|
say ($msg, indented)
|
||||||
|
say ""
|
||||||
|
go to (retry file)
|
||||||
|
|
||||||
|
$prev_progress = (((#$lessons) - (#$failures)) / (#$lessons))
|
||||||
|
$failures = (get failures)
|
||||||
|
$progress = (((#$lessons) - (#$failures)) / (#$lessons))
|
||||||
|
if ($progress != $prev_progress):
|
||||||
|
if ($progress > $prev_progress):
|
||||||
|
say (bold (green "\nSuccess!\n"))
|
||||||
..else:
|
..else:
|
||||||
|
say (bold (red "\nUh oh, that broke something.\n"))
|
||||||
|
$N = 100
|
||||||
|
for $ in 0 to $N:
|
||||||
|
$k = (($ / $N) smoothed by 2)
|
||||||
|
$p = ($prev_progress to $progress mixed by $k)
|
||||||
|
say "\r\(bold "Your progress:") \(20 wide $p progress bar)" inline
|
||||||
|
$io.flush()
|
||||||
|
sh> "sleep \(1 / $N)"
|
||||||
|
say ""
|
||||||
|
|
||||||
say ("
|
say ("
|
||||||
|
|
||||||
\(bold "\(slow blink "Congratulations!")")
|
\(bold "\(slow blink "Congratulations!")")
|
||||||
Everything works!
|
|
||||||
|
You've passed the tutorial!
|
||||||
|
|
||||||
\\(^ᴗ^)/
|
\\(^ᴗ^)/
|
||||||
|
|
||||||
|
|
||||||
")
|
")
|
||||||
stop
|
|
||||||
|
Loading…
Reference in New Issue
Block a user