container list:_elem [
value:_elem
next:&:list:_elem
]
def push x:_elem, l:&:list:_elem -> l:&:list:_elem [
local-scope
load-ingredients
result:&:list:_elem <- new {(list _elem): type}
*result <- merge x, l
reply result
]
def first in:&:list:_elem -> result:_elem [
local-scope
load-ingredients
result <- get *in, value:offset
]
def rest in:&:list:_elem -> result:&:list:_elem/contained-in:in [
local-scope
load-ingredients
result <- get *in, next:offset
]
scenario list-handling [
run [
local-scope
x:&:list:num <- push 3, 0
x <- push 4, x
x <- push 5, x
10:num/raw <- first x
x <- rest x
11:num/raw <- first x
x <- rest x
12:num/raw <- first x
20:&:list:num/raw <- rest x
]
memory-should-contain [
10 <- 5
11 <- 4
12 <- 3
20 <- 0
]
]
def length l:&:list:_elem -> result:num [
local-scope
load-ingredients
return-unless l, 0
rest:&:list:_elem <- rest l
length-of-rest:num <- length rest
result <- add length-of-rest, 1
]
def insert x:_elem, in:&:list:_elem -> in:&:list:_elem [
local-scope
load-ingredients
new-node:&:list:_elem <- new {(list _elem): type}
*new-node <- put *new-node, value:offset, x
next-node:&:list:_elem <- get *in, next:offset
*in <- put *in, next:offset, new-node
*new-node <- put *new-node, next:offset, next-node
]
scenario inserting-into-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:list:char <- rest list
list2 <- insert 6, list2
list2 <- copy list
10:char/raw <- first list2
list2 <- rest list2
11:char/raw <- first list2
list2 <- rest list2
12:char/raw <- first list2
list2 <- rest list2
13:char/raw <- first list2
]
memory-should-contain [
10 <- 5
11 <- 4
12 <- 6
13 <- 3
]
]
scenario inserting-at-end-of-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:list:char <- rest list
list2 <- rest list2
list2 <- insert 6, list2
list2 <- copy list
10:char/raw <- first list2
list2 <- rest list2
11:char/raw <- first list2
list2 <- rest list2
12:char/raw <- first list2
list2 <- rest list2
13:char/raw <- first list2
]
memory-should-contain [
10 <- 5
11 <- 4
12 <- 3
13 <- 6
]
]
scenario inserting-after-start-of-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list <- insert 6, list
list2:&:list:char <- copy list
10:char/raw <- first list2
list2 <- rest list2
11:char/raw <- first list2
list2 <- rest list2
12:char/raw <- first list2
list2 <- rest list2
13:char/raw <- first list2
]
memory-should-contain [
10 <- 5
11 <- 6
12 <- 4
13 <- 3
]
]
def remove x:&:list:_elem/contained-in:in, in:&:list:_elem -> in:&:list:_elem [
local-scope
load-ingredients
return-unless x
next-node:&:list:_elem <- rest x
*x <- put *x, next:offset, 0
at-head?:bool <- equal x, in
return-if at-head?, next-node
prev-node:&:list:_elem <- copy in
curr:&:list:_elem <- rest prev-node
{
return-unless curr
found?:bool <- equal curr, x
break-if found?
prev-node <- copy curr
curr <- rest curr
}
*prev-node <- put *prev-node, next:offset, next-node
]
scenario removing-from-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:list:char <- rest list
list <- remove list2, list
10:bool/raw <- equal list2, 0
list2 <- copy list
11:char/raw <- first list2
list2 <- rest list2
12:char/raw <- first list2
20:&:list:char/raw <- rest list2
]
memory-should-contain [
10 <- 0
11 <- 5
12 <- 3
20 <- 0
]
]
scenario removing-from-start-of-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list <- remove list, list
list2:&:list:char <- copy list
10:char/raw <- first list2
list2 <- rest list2
11:char/raw <- first list2
20:&:list:char/raw <- rest list2
]
memory-should-contain [
10 <- 4
11 <- 3
20 <- 0
]
]
scenario removing-from-end-of-list [
local-scope
list:&:list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:list:char <- rest list
list2 <- rest list2
list <- remove list2, list
10:bool/raw <- equal list2, 0
list2 <- copy list
11:char/raw <- first list2
list2 <- rest list2
12:char/raw <- first list2
20:&:list:char/raw <- rest list2
]
memory-should-contain [
10 <- 0
11 <- 5
12 <- 4
20 <- 0
]
]
scenario removing-from-singleton-list [
local-scope
list:&:list:char <- push 3, 0
run [
list <- remove list, list
1:num/raw <- copy list
]
memory-should-contain [
1 <- 0
]
]
def reverse list:&:list:_elem temp:&:list:_elem/contained-in:result -> result:&:list:_elem [
local-scope
load-ingredients
reply-unless list, temp
object:_elem <- first, list
list <- rest list
temp <- push object, temp
result <- reverse list, temp
]
scenario reverse-list [
local-scope
list:&:list:number <- push 1, 0
list <- push 2, list
list <- push 3, list
run [
stash [list:], list
list <- reverse list
stash [reversed:], list
]
trace-should-contain [
app: list: 3 -> 2 -> 1
app: reversed: 1 -> 2 -> 3
]
]
def to-text in:&:list:_elem -> result:text [
local-scope
load-ingredients
buf:&:buffer <- new-buffer 80
buf <- to-buffer in, buf
result <- buffer-to-array buf
]
def to-text-line in:&:list:_elem -> result:text [
local-scope
load-ingredients
buf:&:buffer <- new-buffer 80
buf <- to-buffer in, buf, 6
result <- buffer-to-array buf
]
def to-buffer in:&:list:_elem, buf:&:buffer -> buf:&:buffer [
local-scope
load-ingredients
{
break-if in
buf <- append buf, [[]]
return
}
val:_elem <- get *in, value:offset
buf <- append buf, val
next:&:list:_elem <- rest in
nextn:num <- copy next
return-unless next
buf <- append buf, [ -> ]
remaining:num, optional-ingredient-found?:bool <- next-ingredient
{
break-if optional-ingredient-found?
buf <- to-buffer next, buf
return
}
{
break-unless remaining
remaining <- subtract remaining, 1
buf <- to-buffer next, buf, remaining
return
}
append buf, [...]
]
scenario stash-empty-list [
local-scope
x:&:list:num <- copy 0
run [
stash x
]
trace-should-contain [
app: []
]
]