import Spectre @testable import Stencil import Foundation func testForNode() { describe("ForNode") { let context = Context(dictionary: [ "items": [1, 2, 3], "emptyItems": [Int](), "dict": [ "one": "I", "two": "II", ], "tuples": [(1, 2, 3), (4, 5, 6)] ]) $0.it("renders the given nodes for each item") { let nodes: [NodeType] = [VariableNode(variable: "item")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "123" } $0.it("renders the given empty nodes when no items found item") { let nodes: [NodeType] = [VariableNode(variable: "item")] let emptyNodes: [NodeType] = [TextNode(text: "empty")] let node = ForNode(resolvable: Variable("emptyItems"), loopVariables: ["item"], nodes: nodes, emptyNodes: emptyNodes) try expect(try node.render(context)) == "empty" } $0.it("renders a context variable of type Array") { let any_context = Context(dictionary: [ "items": ([1, 2, 3] as [Any]) ]) let nodes: [NodeType] = [VariableNode(variable: "item")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(any_context)) == "123" } $0.it("renders a context variable of type CountableClosedRange") { let context = Context(dictionary: ["range": 1...3]) let nodes: [NodeType] = [VariableNode(variable: "item")] let node = ForNode(resolvable: Variable("range"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "123" } $0.it("renders a context variable of type CountableRange") { let context = Context(dictionary: ["range": 1..<4]) let nodes: [NodeType] = [VariableNode(variable: "item")] let node = ForNode(resolvable: Variable("range"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "123" } #if os(OSX) $0.it("renders a context variable of type NSArray") { let nsarray_context = Context(dictionary: [ "items": NSArray(array: [1, 2, 3]) ]) let nodes: [NodeType] = [VariableNode(variable: "item")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(nsarray_context)) == "123" } #endif $0.it("renders the given nodes while providing if the item is first in the context") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.first")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "1true2false3false" } $0.it("renders the given nodes while providing if the item is last in the context") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.last")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "1false2false3true" } $0.it("renders the given nodes while providing item counter") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.counter")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "112233" } $0.it("renders the given nodes while providing item counter") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.counter0")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "102132" } $0.it("renders the given nodes while providing loop length") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.length")] let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: []) try expect(try node.render(context)) == "132333" } $0.it("renders the given nodes while filtering items using where expression") { let nodes: [NodeType] = [VariableNode(variable: "item"), VariableNode(variable: "forloop.counter")] let `where` = try parseExpression(components: ["item", ">", "1"], tokenParser: TokenParser(tokens: [], environment: Environment()), token: .text(value: "", at: .unknown)) let node = ForNode(resolvable: Variable("items"), loopVariables: ["item"], nodes: nodes, emptyNodes: [], where: `where`) try expect(try node.render(context)) == "2132" } $0.it("renders the given empty nodes when all items filtered out with where expression") { let nodes: [NodeType] = [VariableNode(variable: "item")] let emptyNodes: [NodeType] = [TextNode(text: "empty")] let `where` = try parseExpression(components: ["item", "==", "0"], tokenParser: TokenParser(tokens: [], environment: Environment()), token: .text(value: "", at: .unknown)) let node = ForNode(resolvable: Variable("emptyItems"), loopVariables: ["item"], nodes: nodes, emptyNodes: emptyNodes, where: `where`) try expect(try node.render(context)) == "empty" } $0.it("can render a filter with spaces") { let templateString = """ {% for article in ars | default: a, b , articles %}\ - {{ article.title }} by {{ article.author }}. {% endfor %} """ let context = Context(dictionary: [ "articles": [ Article(title: "Migrating from OCUnit to XCTest", author: "Kyle Fuller"), Article(title: "Memory Management with ARC", author: "Kyle Fuller"), ] ]) let template = Template(templateString: templateString) let result = try template.render(context) try expect(result) == """ - Migrating from OCUnit to XCTest by Kyle Fuller. - Memory Management with ARC by Kyle Fuller. """ } $0.context("given array of tuples") { $0.it("can iterate over all tuple values") { let templateString = """ {% for first,second,third in tuples %}\ {{ first }}, {{ second }}, {{ third }} {% endfor %} """ let template = Template(templateString: templateString) let result = try template.render(context) try expect(result) == """ 1, 2, 3 4, 5, 6 """ } $0.it("can iterate with less number of variables") { let templateString = """ {% for first,second in tuples %}\ {{ first }}, {{ second }} {% endfor %} """ let template = Template(templateString: templateString) let result = try template.render(context) try expect(result) == """ 1, 2 4, 5 """ } $0.it("can use _ to skip variables") { let templateString = """ {% for first,_,third in tuples %}\ {{ first }}, {{ third }} {% endfor %} """ let template = Template(templateString: templateString) let result = try template.render(context) try expect(result) == """ 1, 3 4, 6 """ } $0.it("throws when number of variables is more than number of tuple values") { let templateString = """ {% for key,value,smth in dict %} {% endfor %} """ let template = Template(templateString: templateString) try expect(template.render(context)).toThrow() } } $0.it("can iterate over dictionary") { let templateString = """ {% for key, value in dict %}\ {{ key }}: {{ value }},\ {% endfor %} """ let template = Template(templateString: templateString) let result = try template.render(context) try expect(result) == """ one: I,two: II, """ } $0.it("renders supports iterating over dictionary") { let nodes: [NodeType] = [ VariableNode(variable: "key"), TextNode(text: ","), ] let emptyNodes: [NodeType] = [TextNode(text: "empty")] let node = ForNode(resolvable: Variable("dict"), loopVariables: ["key"], nodes: nodes, emptyNodes: emptyNodes, where: nil) let result = try node.render(context) try expect(result) == """ one,two, """ } $0.it("renders supports iterating over dictionary") { let nodes: [NodeType] = [ VariableNode(variable: "key"), TextNode(text: "="), VariableNode(variable: "value"), TextNode(text: ","), ] let emptyNodes: [NodeType] = [TextNode(text: "empty")] let node = ForNode(resolvable: Variable("dict"), loopVariables: ["key", "value"], nodes: nodes, emptyNodes: emptyNodes, where: nil) let result = try node.render(context) try expect(result) == """ one=I,two=II, """ } $0.it("handles invalid input") { let token = Token.block(value: "for i", at: .unknown) let parser = TokenParser(tokens: [token], environment: Environment()) let error = TemplateSyntaxError(reason: "'for' statements should use the syntax: `for in [where ]`.", token: token) try expect(try parser.parse()).toThrow(error) } $0.it("can iterate over struct properties") { struct MyStruct { let string: String let number: Int } let context = Context(dictionary: [ "struct": MyStruct(string: "abc", number: 123) ]) let nodes: [NodeType] = [ VariableNode(variable: "property"), TextNode(text: "="), VariableNode(variable: "value"), TextNode(text: "\n"), ] let node = ForNode(resolvable: Variable("struct"), loopVariables: ["property", "value"], nodes: nodes, emptyNodes: []) let result = try node.render(context) try expect(result) == """ string=abc number=123 """ } $0.it("can iterate tuple items") { let context = Context(dictionary: [ "tuple": (one: 1, two: "dva"), ]) let nodes: [NodeType] = [ VariableNode(variable: "label"), TextNode(text: "="), VariableNode(variable: "value"), TextNode(text: "\n"), ] let node = ForNode(resolvable: Variable("tuple"), loopVariables: ["label", "value"], nodes: nodes, emptyNodes: []) let result = try node.render(context) try expect(result) == """ one=1 two=dva """ } $0.it("can iterate over class properties") { class MyClass { var baseString: String var baseInt: Int init(_ string: String, _ int: Int) { baseString = string baseInt = int } } class MySubclass: MyClass { var childString: String init(_ childString: String, _ string: String, _ int: Int) { self.childString = childString super.init(string, int) } } let context = Context(dictionary: [ "class": MySubclass("child", "base", 1) ]) let nodes: [NodeType] = [ VariableNode(variable: "label"), TextNode(text: "="), VariableNode(variable: "value"), TextNode(text: "\n"), ] let node = ForNode(resolvable: Variable("class"), loopVariables: ["label", "value"], nodes: nodes, emptyNodes: []) let result = try node.render(context) try expect(result) == """ childString=child baseString=base baseInt=1 """ } $0.it("can iterate in range of variables") { let template: Template = "{% for i in 1...j %}{{ i }}{% endfor %}" try expect(try template.render(Context(dictionary: ["j": 3]))) == "123" } } } fileprivate struct Article { let title: String let author: String }