blob: 681410546304c4e82908d8ec98afbb3e3d24c3d0 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package groovy
import groovy.test.GroovyTestCase
import groovy.transform.CompileStatic
import java.util.function.Function
/**
* Tests for Closure composition
*/
class ClosureComposeTest extends GroovyTestCase {
void testComposeMultiply() {
def twice = { a -> a * 2 }
def thrice = { a -> a * 3 }
def times6 = twice >> thrice
assert times6(3) == 18
}
void testReverseComposeAndCallShortcut() {
def twice = { a -> a * 2 }
def thrice = { a -> a * 3 }
assert twice << thrice << 3 == 18
}
void testComposeAndLonghand() {
def twice = { a -> a * 2 }
def inc = { b -> b + 1 }
def f = inc >> twice
def g = { x -> twice(inc(x)) }
assert f(10) == 22
assert g(10) == 22
}
// GROOVY-4512
void testClosureCompositionInstance() {
def inst = new ComposeTestHelper()
assert inst.composedA.call() == 42
assert inst.composedA() == 42
assert inst.composedB.call(3) == 122
assert inst.composedB(3) == 122
}
void testComposeWithMethodClosure() {
def s2c = { it.chars[0] }
def p = Integer.&toHexString >> s2c >> Character.&toUpperCase
assert p(15) == 'F'
}
void testComposeWithMultipleArgs() {
def multiply = { a, b -> a * b }
def identity = { a -> [a, a] }
def sq = identity >> multiply
assert (1..5).collect { sq(it) } == [1, 4, 9, 16, 25]
}
void testReverseCompositionWithMultipleArgs() {
def multiply = { a, b -> a * b }
def identity = { a -> [a, a] }
def sq = multiply << identity
assert (1..5).collect { sq(it) } == [1, 4, 9, 16, 25]
}
void testComposeWithCurriedClosures() {
def add3 = { a, b, c -> a + b + c }
def add2plus10 = add3.curry(10)
def multBoth = { a, b, c -> [a * c, b * c] }
def twiceBoth = multBoth.rcurry(2)
def twiceBothPlus10 = twiceBoth >> add2plus10
assert twiceBothPlus10(5, 10) == 40
}
void testDelegate() {
// Groovy-4994 failed with MissingPropertyException
assertScript """
def a = { foo }
def b = { bar }
class O {
def foo = 'foo'
def bar = 'bar'
}
def ab = a >> b
ab.delegate = new O()
ab()
"""
}
@CompileStatic
void testAndThenCS() {
Function<String, String> lower = String::toLowerCase
Function<String, String> upper = String::toUpperCase
Function<String, String> lu = lower.andThen(upper)
Function<? super String, String> ul = upper.andThen(lower)
assert lower('Hi') == ul('Hi')
assert upper('Hi') == lu('Hi')
}
void testAndThen() {
def lower = String::toLowerCase
def upper = String::toUpperCase
def lu1 = lower.rightShift(upper)
def ul1 = upper.rightShift(lower)
assert lower('Hi') == ul1('Hi')
assert upper('Hi') == lu1('Hi')
def lu2 = lower.andThen(upper)
def ul2 = upper.andThen(lower)
assert lower('Hi') == ul2('Hi')
assert upper('Hi') == lu2('Hi')
}
void testAndThenSelf() {
def inc = String::next
def inc2 = inc.andThenSelf()
def inc4 = inc.andThenSelf(3)
assert inc('abc') == 'abd'
assert inc2('abc') == 'abe'
assert inc4('abc') == 'abg'
}
@CompileStatic
void testComposeCS() {
Function<String, String> lower = String::toLowerCase
Function<String, String> upper = String::toUpperCase
Function<String, String> ul = lower.compose(upper)
Function<String, ? extends String> lu = upper.compose(lower)
assert lower('Hi') == ul('Hi')
assert upper('Hi') == lu('Hi')
}
void testCompose() {
def lower = String::toLowerCase
def upper = String::toUpperCase
def ul1 = lower.leftShift(upper)
def lu1 = upper.leftShift(lower)
assert lower('Hi') == ul1('Hi')
assert upper('Hi') == lu1('Hi')
def ul2 = lower.compose(upper)
def lu2 = upper.compose(lower)
assert lower('Hi') == ul2('Hi')
assert upper('Hi') == lu2('Hi')
}
void testComposeSelf() {
def inc = String::next
def inc2 = inc.composeSelf()
def inc4 = inc.composeSelf(3)
assert inc('abc') == 'abd'
assert inc2('abc') == 'abe'
assert inc4('abc') == 'abg'
}
class ComposeTestHelper {
def closure1 = { 40 }
def closure2 = { it * 40 }
def closure3 = { it + 2 }
def composedA = closure1 >> closure3
def composedB = closure2 >> closure3
}
}