Merge pull request #7771 from nim-lang/nim-in-action-fixes

Higher Nim in Action coverage plus regression fix.
This commit is contained in:
Andreas Rumpf
2018-06-19 17:52:05 +02:00
committed by GitHub
12 changed files with 646 additions and 1 deletions

View File

@@ -17,6 +17,10 @@ import parseutils
from math import pow, round, floor, log10
from algorithm import reverse
when defined(nimVmExportFixed):
from unicode import toLower, toUpper
export toLower, toUpper
{.deadCodeElim: on.} # dce option deprecated
{.push debugger:off .} # the user does not want to trace a part

View File

@@ -0,0 +1,45 @@
discard """
exitCode: 0
outputsub: "Woof!"
"""
import strutils
echo("hello".to_upper())
echo("world".toUpper())
type
Dog = object #<1>
age: int #<2>
let dog = Dog(age: 3) #<3>
proc showNumber(num: int | float) =
echo(num)
showNumber(3.14)
showNumber(42)
for i in 0 .. <10:
echo(i)
block: # Block added due to clash.
type
Dog = object
proc bark(self: Dog) = #<1>
echo("Woof!")
let dog = Dog()
dog.bark() #<2>
import sequtils, future, strutils
let list = @["Dominik Picheta", "Andreas Rumpf", "Desmond Hume"]
list.map(
(x: string) -> (string, string) => (x.split[0], x.split[1])
).echo
import strutils
let list1 = @["Dominik Picheta", "Andreas Rumpf", "Desmond Hume"]
for name in list1:
echo((name.split[0], name.split[1]))

View File

@@ -0,0 +1,7 @@
discard """
line: 7
errormsg: "has to be discarded"
"""
proc myProc(name: string): string = "Hello " & name
myProc("Dominik")

View File

@@ -0,0 +1,16 @@
discard """
line: 16
errormsg: "type mismatch"
"""
type
Dog = object
name: string
Cat = object
name: string
let dog: Dog = Dog(name: "Fluffy")
let cat: Cat = Cat(name: "Fluffy")
echo(dog == cat)

View File

@@ -0,0 +1,7 @@
discard """
line: 6
errormsg: "type mismatch"
"""
for i in 5:
echo i

View File

@@ -0,0 +1,6 @@
discard """
line: 6
errormsg: "cannot infer the type of the sequence"
"""
var list = @[]

View File

@@ -0,0 +1,28 @@
discard """
output: ""
"""
# Page 35.
proc implicit: string =
"I will be returned"
proc discarded: string =
discard "I will not be returned"
proc explicit: string =
return "I will be returned"
proc resultVar: string =
result = "I will be returned"
proc resultVar2: string =
result = ""
result.add("I will be ")
result.add("returned")
doAssert implicit() == "I will be returned"
doAssert discarded() == nil
doAssert explicit() == "I will be returned"
doAssert resultVar() == "I will be returned"
doAssert resultVar2() == "I will be returned"

View File

@@ -0,0 +1,33 @@
discard """
line: 27
errormsg: "has to be discarded"
"""
# Page 35.
proc implicit: string =
"I will be returned"
proc discarded: string =
discard "I will not be returned"
proc explicit: string =
return "I will be returned"
proc resultVar: string =
result = "I will be returned"
proc resultVar2: string =
result = ""
result.add("I will be ")
result.add("returned")
proc resultVar3: string =
result = "I am the result"
"I will cause an error"
doAssert implicit() == "I will be returned"
doAssert discarded() == nil
doAssert explicit() == "I will be returned"
doAssert resultVar() == "I will be returned"
doAssert resultVar2() == "I will be returned"

View File

@@ -0,0 +1,369 @@
discard """
exitCode: 0
outputsub: '''42 is greater than 0'''
"""
if 42 >= 0:
echo "42 is greater than 0"
echo("Output: ",
5)
echo(5 +
5)
# --- Removed code that is supposed to fail here. Not going to test those. ---
# Single-line comment
#[
Multiline comment
]#
when false:
echo("Commented-out code")
let decimal = 42
let hex = 0x42
let octal = 0o42
let binary = 0b101010
let a: int16 = 42
let b = 42'i8
let c = 1'f32 # --- Changed names here to avoid clashes ---
let d = 1.0e19
let e = false
let f = true
let g = 'A'
let h = '\109'
let i = '\x79'
let text = "The book title is \"Nim in Action\""
let filepath = "C:\\Program Files\\Nim"
# --- Changed name here to avoid clashes ---
let filepath1 = r"C:\Program Files\Nim"
let multiLine = """foo
bar
baz
"""
echo multiLine
import strutils
# --- Changed name here to avoid clashes ---
let multiLine1 = """foo
bar
baz
"""
echo multiLine1.unindent
doAssert multiLine1.unindent == "foo\nbar\nbaz\n"
proc fillString(): string =
result = ""
echo("Generating string")
for i in 0 .. 4:
result.add($i) #<1>
const count = fillString()
var
text1 = "hello"
number: int = 10
isTrue = false
var = "Fire"
let ogień = true
var `var` = "Hello"
echo(`var`)
proc myProc(name: string): string = "Hello " & name
discard myProc("Dominik")
proc bar(): int #<1>
proc foo(): float = bar().float
proc bar(): int = foo().int
proc noReturn() = echo("Hello")
proc noReturn2(): void = echo("Hello")
proc noReturn3 = echo("Hello")
proc message(recipient: string): auto =
"Hello " & recipient
doAssert message("Dom") == "Hello Dom"
proc max(a: int, b: int): int =
if a > b: a else: b
doAssert max(5, 10) == 10
proc max2(a, b: int): int =
if a > b: a else: b
proc genHello(name: string, surname = "Doe"): string =
"Hello " & name & " " & surname
# -- Leaving these as asserts as that is in the original code, just in case
# -- somehow in the future `assert` is removed :)
assert genHello("Peter") == "Hello Peter Doe"
assert genHello("Peter", "Smith") == "Hello Peter Smith"
proc genHello2(names: varargs[string]): string =
result = ""
for name in names:
result.add("Hello " & name & "\n")
doAssert genHello2("John", "Bob") == "Hello John\nHello Bob\n"
proc getUserCity(firstName, lastName: string): string =
case firstName
of "Damien": return "Tokyo"
of "Alex": return "New York"
else: return "Unknown"
proc getUserCity(userID: int): string =
case userID
of 1: return "Tokyo"
of 2: return "New York"
else: return "Unknown"
doAssert getUserCity("Damien", "Lundi") == "Tokyo"
doAssert getUserCity(2) == "New York" # -- Errata here: missing closing "
import sequtils
let numbers = @[1, 2, 3, 4, 5, 6]
let odd = filter(numbers, proc (x: int): bool = x mod 2 != 0)
doAssert odd == @[1, 3, 5]
import sequtils, future
let numbers1 = @[1, 2, 3, 4, 5, 6]
let odd1 = filter(numbers1, (x: int) -> bool => x mod 2 != 0)
assert odd1 == @[1, 3, 5]
proc isValid(x: int, validator: proc (x: int): bool) =
if validator(x): echo(x, " is valid")
else: echo(x, " is NOT valid")
import future
proc isValid2(x: int, validator: (x: int) -> bool) =
if validator(x): echo(x, " is valid")
else: echo(x, " is NOT valid")
var list: array[3, int]
list[0] = 1
list[1] = 42
assert list[0] == 1
assert list[1] == 42
assert list[2] == 0 #<1>
echo list.repr #<2>
# echo list[500]
var list2: array[-10 .. -9, int]
list2[-10] = 1
list2[-9] = 2
var list3 = ["Hi", "There"]
var list4 = ["My", "name", "is", "Dominik"]
for item in list4:
echo(item)
for i in list4.low .. list4.high:
echo(list4[i])
var list5: seq[int] = @[]
doAssertRaises(IndexError):
list5[0] = 1
list5.add(1)
assert list5[0] == 1
doAssertRaises(IndexError):
echo list5[42]
# -- Errata: var list: seq[int]; echo(list[0]). This now creates an exception,
# -- not a SIGSEGV.
block:
var list = newSeq[string](3)
assert list[0] == nil
list[0] = "Foo"
list[1] = "Bar"
list[2] = "Baz"
list.add("Lorem")
block:
let list = @[4, 8, 15, 16, 23, 42]
for i in 0 .. <list.len:
stdout.write($list[i] & " ")
var collection: set[int16]
doAssert collection == {}
block:
let collection = {'a', 'x', 'r'}
doAssert 'a' in collection
block:
let collection = {'a', 'T', 'z'}
let isAllLowerCase = {'A' .. 'Z'} * collection == {}
doAssert(not isAllLowerCase)
let age = 10
if age > 0 and age <= 10:
echo("You're still a child")
elif age > 10 and age < 18:
echo("You're a teenager")
else:
echo("You're an adult")
let variable = "Arthur"
case variable
of "Arthur", "Zaphod", "Ford":
echo("Male")
of "Marvin":
echo("Robot")
of "Trillian":
echo("Female")
else:
echo("Unknown")
let ageDesc = if age < 18: "Non-Adult" else: "Adult"
block:
var i = 0
while i < 3:
echo(i)
i.inc
block label:
var i = 0
while true:
while i < 5:
if i > 3: break label
i.inc
iterator values(): int =
var i = 0
while i < 5:
yield i
i.inc
for value in values():
echo(value)
import os
for filename in walkFiles("*.nim"):
echo(filename)
for item in @[1, 2, 3]:
echo(item)
for i, value in @[1, 2, 3]: echo("Value at ", i, ": ", value)
doAssertRaises(IOError):
proc second() =
raise newException(IOError, "Somebody set us up the bomb")
proc first() =
second()
first()
block:
proc second() =
raise newException(IOError, "Somebody set us up the bomb")
proc first() =
try:
second()
except:
echo("Cannot perform second action because: " &
getCurrentExceptionMsg())
first()
block:
type
Person = object
name: string
age: int
var person: Person
var person1 = Person(name: "Neo", age: 28)
block:
type
PersonObj = object
name: string
age: int
PersonRef = ref PersonObj
# proc setName(person: PersonObj) =
# person.name = "George"
proc setName(person: PersonRef) =
person.name = "George"
block:
type
Dog = object
name: string
Cat = object
name: string
let dog: Dog = Dog(name: "Fluffy")
let cat: Cat = Cat(name: "Fluffy")
block:
type
Dog = tuple
name: string
Cat = tuple
name: string
let dog: Dog = (name: "Fluffy")
let cat: Cat = (name: "Fluffy")
echo(dog == cat)
block:
type
Point = tuple[x, y: int]
Point2 = (int, int)
let pos: Point = (x: 100, y: 50)
doAssert pos == (100, 50)
let pos1: Point = (x: 100, y: 50)
let (x, y) = pos1 #<1>
let (left, _) = pos1
doAssert x == pos1[0]
doAssert y == pos1[1]
doAssert left == x
block:
type
Color = enum
colRed,
colGreen,
colBlue
let color: Color = colRed
block:
type
Color {.pure.} = enum
red, green, blue
let color = Color.red

View File

@@ -0,0 +1,93 @@
import threadpool
proc foo: string = "Dog"
var x: FlowVar[string] = spawn foo()
assert(^x == "Dog")
block:
type
Box = object
case empty: bool
of false:
contents: string
else:
discard
var obj = Box(empty: false, contents: "Hello")
assert obj.contents == "Hello"
var obj2 = Box(empty: true)
doAssertRaises(FieldError):
echo(obj2.contents)
import json
assert parseJson("null").kind == JNull
assert parseJson("true").kind == JBool
assert parseJson("42").kind == JInt
assert parseJson("3.14").kind == JFloat
assert parseJson("\"Hi\"").kind == JString
assert parseJson("""{ "key": "value" }""").kind == JObject
assert parseJson("[1, 2, 3, 4]").kind == JArray
import json
let data = """
{"username": "Dominik"}
"""
let obj = parseJson(data)
assert obj.kind == JObject
assert obj["username"].kind == JString
assert obj["username"].str == "Dominik"
block:
proc count10(): int =
for i in 0 .. <10:
result.inc
assert count10() == 10
type
Point = tuple[x, y: int]
var point = (5, 10)
var point2 = (x: 5, y: 10)
type
Human = object
name: string
age: int
var jeff = Human(name: "Jeff", age: 23)
var amy = Human(name: "Amy", age: 20)
import asyncdispatch
var future = newFuture[int]()
doAssert(not future.finished)
future.callback =
proc (future: Future[int]) =
echo("Future is no longer empty, ", future.read)
future.complete(42)
import asyncdispatch, asyncfile
when false:
var file = openAsync("")
let dataFut = file.readAll()
dataFut.callback =
proc (future: Future[string]) =
echo(future.read())
asyncdispatch.runForever()
import asyncdispatch, asyncfile, os
proc readFiles() {.async.} =
# --- Changed to getTempDir here.
var file = openAsync(getTempDir() / "test.txt", fmReadWrite)
let data = await file.readAll()
echo(data)
await file.write("Hello!\n")
waitFor readFiles()

View File

@@ -0,0 +1 @@
threads:on

View File

@@ -266,8 +266,17 @@ proc testNimInAction(r: var TResults, cat: Category, options: string) =
testSpec r, makeTest(filename, options, cat, actionCompile), targetCPP
let tests = [
"niminaction/Chapter1/various1",
"niminaction/Chapter2/various2",
"niminaction/Chapter2/resultaccept",
"niminaction/Chapter2/resultreject",
"niminaction/Chapter2/explicit_discard",
"niminaction/Chapter2/no_def_eq",
"niminaction/Chapter2/no_iterator",
"niminaction/Chapter2/no_seq_type",
"niminaction/Chapter3/ChatApp/src/server",
"niminaction/Chapter3/ChatApp/src/client",
"niminaction/Chapter3/various3",
"niminaction/Chapter6/WikipediaStats/concurrency_regex",
"niminaction/Chapter6/WikipediaStats/concurrency",
"niminaction/Chapter6/WikipediaStats/naive",
@@ -278,8 +287,34 @@ proc testNimInAction(r: var TResults, cat: Category, options: string) =
"niminaction/Chapter7/Tweeter/src/tweeter",
"niminaction/Chapter7/Tweeter/src/createDatabase",
"niminaction/Chapter7/Tweeter/tests/database_test",
"niminaction/Chapter8/sdl/sdl_test",
"niminaction/Chapter8/sdl/sdl_test"
]
# Verify that the files have not been modified. Death shall fall upon
# whoever edits these hashes without dom96's permission, j/k. But please only
# edit when making a conscious breaking change, also please try to make your
# commit message clear and notify me so I can easily compile an errata later.
var testHashes: seq[string] = @[]
for test in tests:
testHashes.add(getMD5(readFile("tests" / test.addFileExt("nim")).string))
const refHashes = @[
"51afdfa84b3ca3d810809d6c4e5037ba", "30f07e4cd5eaec981f67868d4e91cfcf",
"d14e7c032de36d219c9548066a97e846", "2e40bfd5daadb268268727da91bb4e81",
"c5d3853ed0aba04bf6d35ba28a98dca0", "058603145ff92d46c009006b06e5b228",
"7b94a029b94ddb7efafddd546c965ff6", "586d74514394e49f2370dfc01dd9e830",
"e1901837b757c9357dc8d259fd0ef0f6", "097670c7ae12e825debaf8ec3995227b",
"a8cb7b78cc78d28535ab467361db5d6e", "bfaec2816a1848991b530c1ad17a0184",
"47cb71bb4c1198d6d29cdbee05aa10b9", "87e4436809f9d73324cfc4f57f116770",
"7b7db5cddc8cf8fa9b6776eef1d0a31d", "e6e40219f0f2b877869b738737b7685e",
"6532ee87d819f2605a443d5e94f9422a", "9a8fe78c588d08018843b64b57409a02",
"03a801275b8b76b4170c870cd0da079d", "20bb7d3e2d38d43b0cb5fcff4909a4a8",
"af6844598f534fab6942abfa4dfe9ab2", "2a7a17f84f6503d9bc89a5ab8feea127"
]
doAssert testHashes == refHashes, "Nim in Action tests were changed."
# Run the tests.
for testfile in tests:
test "tests/" & testfile & ".nim", actionCompile
@@ -291,6 +326,7 @@ proc testNimInAction(r: var TResults, cat: Category, options: string) =
# ------------------------- manyloc -------------------------------------------
#proc runSpecialTests(r: var TResults, options: string) =
# for t in ["lib/packages/docutils/highlite"]: