More lessons
This commit is contained in:
parent
b41fde8f82
commit
04a491c551
3
koans.tm
3
koans.tm
@ -22,6 +22,9 @@ LESSONS := [
|
||||
Lesson((./lessons/lesson-11-enums.tm), "Enums"),
|
||||
Lesson((./lessons/lesson-12-allocating.tm), "Allocating Memory"),
|
||||
Lesson((./lessons/lesson-13-paths.tm), "File Paths"),
|
||||
Lesson((./lessons/lesson-14-langs.tm), "Embedded Languages"),
|
||||
Lesson((./lessons/lesson-15-min-max.tm), "Min and Max"),
|
||||
Lesson((./lessons/lesson-16-reducers.tm), "Reducers"),
|
||||
]
|
||||
|
||||
enum TestResult(Success(output:Text), Error(err:Text), WrongOutput(actual:Text, expected:Text)):
|
||||
|
31
lesson-templates/lesson-14-langs.tm
Normal file
31
lesson-templates/lesson-14-langs.tm
Normal file
@ -0,0 +1,31 @@
|
||||
# Langs (Safe Embedded Languages)
|
||||
|
||||
# `lang` defines custom text types with automatic escaping.
|
||||
lang HTML:
|
||||
|
||||
# Custom escaping rules can be created with `convert`
|
||||
convert(t:Text -> HTML):
|
||||
t = t:replace_all({$/&/="&", $/</="<", $/>/=">"})
|
||||
return HTML.from_text(t)
|
||||
|
||||
func paragraph(content:HTML -> HTML):
|
||||
return $HTML"<p>$content</p>"
|
||||
|
||||
|
||||
# Type safety prevents injection:
|
||||
func greet(name:HTML -> HTML):
|
||||
return $HTML"Hello, $name!"
|
||||
|
||||
func main():
|
||||
|
||||
malicious_input := "<b>hello</b>"
|
||||
|
||||
safe := $HTML"User said: $malicious_input"
|
||||
|
||||
>> safe
|
||||
= ???
|
||||
|
||||
>> safe:paragraph()
|
||||
= ???
|
||||
|
||||
greeting := greet(malicious_input) # This should fail
|
22
lesson-templates/lesson-15-min-max.tm
Normal file
22
lesson-templates/lesson-15-min-max.tm
Normal file
@ -0,0 +1,22 @@
|
||||
# Min and Max
|
||||
|
||||
struct Point(x:Int, y:Int)
|
||||
|
||||
func main():
|
||||
|
||||
# `_min_` and `_max_` return the smaller or larger of two values:
|
||||
>> 7 _min_ 3
|
||||
= ???
|
||||
>> "apple" _max_ "banana"
|
||||
= "???"
|
||||
|
||||
# You can use an indexed key for choosing a maximum:
|
||||
>> Point(1, 2) _max_.y Point(999, 0)
|
||||
= Point(x=???, y=???)
|
||||
|
||||
>> [1, 2, 3] _min_.length [99, 100]
|
||||
= [???]
|
||||
|
||||
# You can also use a method for choosing a maximum:
|
||||
>> 5 _max_:abs() -999
|
||||
= ???
|
31
lesson-templates/lesson-16-reducers.tm
Normal file
31
lesson-templates/lesson-16-reducers.tm
Normal file
@ -0,0 +1,31 @@
|
||||
# Reductions
|
||||
|
||||
func main():
|
||||
|
||||
# Reductions fold collections into a single value:
|
||||
>> (+: [1, 2, 3])!
|
||||
= ???
|
||||
>> (*: [2, 3, 4])!
|
||||
= ???
|
||||
|
||||
# If an empty argument is given, a `none` value is returned
|
||||
empty := [:Int]
|
||||
>> (+: empty)
|
||||
= none
|
||||
|
||||
# Use `or` to provide a fallback:
|
||||
>> (+: empty) or 100
|
||||
= ???
|
||||
|
||||
# `_min_` and `_max_` work as reducers:
|
||||
>> (_max_: [10, 30, 20])!
|
||||
= ???
|
||||
>> (_min_.length: ["x", "abcd", "yz"])!
|
||||
= "???"
|
||||
|
||||
# Comprehensions inside reductions:
|
||||
>> (+: i * 2 for i in [1, 2, 3])!
|
||||
= ???
|
||||
|
||||
>> (+: i for i in 10 if i:is_prime())!
|
||||
= 0
|
Loading…
Reference in New Issue
Block a user