All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.model.RuleSourceAppliedByRuleMethodIntegrationTest.groovy Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2015 the original author or authors.
 *
 * Licensed 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 org.gradle.model

import org.gradle.integtests.fixtures.AbstractIntegrationSpec
import org.gradle.language.base.LanguageSourceSet

class RuleSourceAppliedByRuleMethodIntegrationTest extends AbstractIntegrationSpec {
    def "@Rule method can apply rules to a particular target"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                    assert t.name == 'default'
                    t.name = 'p1'
                }

                @Model
                void p2(Thing t) {
                    assert t.name == null
                    t.name = 'p2'
                }

                @Rules
                void rules(CalculateName rules, @Path('p1') Thing t) {
                    println "applying rules to $t"
                    assert rules != null
                    assert t != null
                }
            }

            class CalculateName extends RuleSource {
                @Defaults
                void defaultName(Thing t) {
                    t.name = 'default'
                }

                @Finalize
                void finalizeName(Thing t) {
                    assert t.name == 'p1'
                    t.name = 'p1 finalized'
                }
            }

            apply plugin: MyPlugin

            model {
                tasks {
                    show(Task) {
                        doLast {
                            println "p1 = " + $.p1.name
                            println "p2 = " + $.p2.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("applying rules to Thing 'p1'")
        output.contains("p1 = p1 finalized")
        output.contains("p2 = p2")
    }

    def "@Rule method can apply rules to target of current rule source"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t) {
                }
            }

            class CalculateName extends RuleSource {
                @Rules
                void rules(SpecializeName rules, Thing t) {
                }
            }

            class SpecializeName extends RuleSource {
                @Defaults
                void defaultName(Thing t) {
                    t.name = 'default'
                }

                @Finalize
                void finalizeName(Thing t) {
                    assert t.name == 'default'
                    t.name = 'finalized'
                }
            }

            apply plugin: MyPlugin

            model {
                tasks {
                    show(Task) {
                        doLast {
                            println "p1 = " + $.p1.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("p1 = finalized")
    }

    def "@Rule method can apply abstract RuleSource"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t) {
                    println "applying " + rules + " to " + t
                    assert rules == rules
                }
            }

            abstract class CalculateName extends RuleSource {
                @Defaults
                void defaultName(Thing t) {
                    println "applying defaults from " + toString()
                    assert this == this
                    t.name = 'default'
                }
            }

            apply plugin: MyPlugin

            model {
                tasks {
                    show(Task) {
                        doLast {
                            println "p1 = " + $.p1.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("applying rule source CalculateName to Thing 'p1'")
        output.contains("applying defaults from rule source CalculateName")
        output.contains("p1 = default")
    }

    def "@Rule method can apply abstract RuleSource with scalar input properties"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t) {
                    assert rules.defaultName == null
                    rules.defaultName = 'default'
                    assert rules.defaultName == 'default'
                }
            }

            abstract class CalculateName extends RuleSource {
                @RuleInput
                abstract String getDefaultName()
                abstract void setDefaultName(String n)

                @Defaults
                void defaultName(Thing t) {
                    assert defaultName == 'default'
                    t.name = defaultName
                }
            }

            apply plugin: MyPlugin

            model {
                tasks {
                    show(Task) {
                        doLast {
                            println "p1 = " + $.p1.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("p1 = default")
    }

    def "elements referenced by @RuleInput property are treated as implicit input of rules on RuleSource"() {
        buildFile << '''
            @Managed
            interface SomeThings {
                Thing getThingA()
                Thing getThingB()
                Thing getThingC()
            }

            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void things(SomeThings t) {
                }

                @Rules
                void rules(CalculateName rules, SomeThings t) {
                    assert rules.thing == null
                    rules.thing = t.thingA
                    assert rules.thing == t.thingA
                }
            }

            abstract class CalculateName extends RuleSource {
                @RuleInput
                abstract Thing getThing()
                abstract void setThing(Thing t)

                @Defaults
                void defaultB(@Path('thingB') Thing thingB) {
                    assert thing.name == 'thing a from dsl'
                    thingB.name = thing.name
                }

                @Finalize
                void defaultC(@Path('thingC') Thing thingC) {
                    assert thing.name == 'thing a from dsl'
                    thingC.name = 'thing c from rule'
                }
            }

            apply plugin: MyPlugin

            model {
                things.thingA {
                    name = 'thing a from dsl'
                }
                things.thingB {
                    assert name == 'thing a from dsl'
                    name = 'thing b from dsl'
                }
                tasks {
                    show(Task) {
                        doLast {
                            def things = $.things
                            println "thingA = " + things.thingA.name
                            println "thingB = " + things.thingB.name
                            println "thingC = " + things.thingC.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("thingA = thing a from dsl")
        output.contains("thingB = thing b from dsl")
        output.contains("thingC = thing c from rule")
    }

    def "element referenced by @RuleTarget property is treated as target of RuleSource"() {
        buildFile << '''
            @Managed
            interface SomeThings {
                Thing getThingA()
                Thing getThingB()
            }

            @Managed
            interface Thing {
                String getName()
                void setName(String name)
                Address getAddress()
            }

            @Managed
            interface Address {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void things(SomeThings t) {
                }

                @Rules
                void rules(CalculateName rules, SomeThings t) {
                    assert rules.thing == null
                    rules.thing = t.thingA
                    assert rules.thing == t.thingA
                }
            }

            abstract class CalculateName extends RuleSource {
                @RuleTarget
                abstract Thing getThing()
                abstract void setThing(Thing t)

                @Mutate
                void thing(Thing subject) {
                    assert subject == thing
                    subject.name = 'thing a from rule'
                    subject.address.name = 'address a from rule'
                }

                @Mutate
                void address(Address address) {
                    assert address == thing.address
                    assert address.name == 'address a from rule'
                    address.name = 'modified address'
                }
            }

            apply plugin: MyPlugin

            model {
                tasks {
                    show(Task) {
                        doLast {
                            def things = $.things
                            println "thingA = " + things.thingA.name
                            println "thingA.address = " + things.thingA.address.name
                        }
                    }
                }
            }
        '''

        when:
        run 'show'

        then:
        output.contains("thingA = thing a from rule")
        output.contains("thingA.address = modified address")
    }

    def "reports exception thrown by @Rules method"() {
        buildFile << '''
            class MyPlugin extends RuleSource {
                @Model
                void strings(List s) {}

                @Rules
                void rules(OtherRules rules, List s) {
                    throw new RuntimeException("broken")
                }
            }

            class OtherRules extends RuleSource {
            }

            apply plugin: MyPlugin
        '''

        expect:
        fails("model")
        failure.assertHasCause("Exception thrown while executing model rule: MyPlugin#rules")
        failure.assertHasCause("broken")
    }

    def "reports exception thrown by rule method on applied RuleSource"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t) {
                }
            }

            class CalculateName extends RuleSource {
                @Defaults
                void defaultName(Thing t) {
                    throw new RuntimeException("broken")
                }
            }

            apply plugin: MyPlugin
        '''

        expect:
        fails 'model'
        failure.assertHasCause("Exception thrown while executing model rule: CalculateName#defaultName")
        failure.assertHasCause("broken")
    }

    def "reports exception thrown by rule source constructor"() {
        buildFile << '''
            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void p1(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t) {
                }
            }

            class CalculateName extends RuleSource {
                CalculateName() {
                    throw new RuntimeException("broken")
                }
            }

            apply plugin: MyPlugin
        '''

        expect:
        fails 'model'
        failure.assertHasCause("Exception thrown while executing model rule: MyPlugin#rules")
        failure.assertHasCause("broken")
    }

    def "@Rules method is not executed when target is not required"() {
        buildFile << '''
            class MyPlugin extends RuleSource {
                @Model
                void strings(List s) {}

                @Rules
                void rules(RuleSource rules, List s) {
                    throw new RuntimeException("broken")
                }
            }

            apply plugin: MyPlugin
        '''

        expect:
        succeeds("tasks")
        fails("model")
    }

    def "first parameter of @Rules method must be assignable to RuleSource"() {
        buildFile << '''
            class MyPlugin extends RuleSource {
                @Rules
                void rules(LanguageSourceSet lss, String string) {
                }
            }
            apply plugin: MyPlugin
        '''

        expect:
        fails 'model'
        failure.assertHasCause("""Type MyPlugin is not a valid rule source:
- Method rules(${LanguageSourceSet.name}, ${String.name}) is not a valid rule method: The first parameter of a method annotated with @Rules must be a subtype of ${RuleSource.name}""")
    }

    def "reports declaration problem with applied RuleSource"() {
        buildFile << '''
            class MyPlugin extends RuleSource {
                @Model
                void strings(List s) {}

                @Rules
                void rules(BrokenRuleSource rules, List s) {
                }
            }

            class BrokenRuleSource extends RuleSource {
                @Validate
                private void broken() { }
            }

            apply plugin: MyPlugin
        '''

        expect:
        fails("model")
        failure.assertHasCause("Exception thrown while executing model rule: MyPlugin#rules")
        failure.assertHasCause('''Type BrokenRuleSource is not a valid rule source:
- Method broken() is not a valid rule method: A rule method cannot be private
- Method broken() is not a valid rule method: A method annotated with @Validate must have at least one parameter''')
    }

    def "reports unbound parameters for rules on applied RuleSource"() {
        buildScript '''
            class UnboundRuleSource extends RuleSource {
                @Mutate
                void unboundRule(String string, Integer integer, @Path("some.inner.path") String withPath) {
                }
            }

            class MyPlugin extends RuleSource {
                @Model
                void strings(List s) {}

                @Rules
                void rules(UnboundRuleSource rules, List s) {
                }
            }

            apply type: MyPlugin
        '''

        expect:
        fails "model"
        failure.assertHasCause('''The following model rules could not be applied due to unbound inputs and/or subjects:

  UnboundRuleSource#unboundRule(String, Integer, String)
    subject:
      -  String (parameter 1) [*]
          scope: strings
    inputs:
      -  Integer (parameter 2) [*]
      - strings.some.inner.path String (parameter 3) [*]
''')
    }

    def "cannot mutate implicit input of RuleSource method"() {
        buildFile << '''
            @Managed
            interface SomeThings {
                Thing getThingA()
            }

            @Managed
            interface Thing {
                String getName()
                void setName(String name)
            }

            class MyPlugin extends RuleSource {
                @Model
                void things(SomeThings t) {
                }

                @Model
                void calculated(Thing t) {
                }

                @Rules
                void rules(CalculateName rules, Thing t, SomeThings others) {
                    rules.thing = others.thingA
                }
            }

            abstract class CalculateName extends RuleSource {
                @RuleInput
                abstract Thing getThing()
                abstract void setThing(Thing t)

                @Mutate
                void broken(Thing t) {
                    println "check"
                    assert thing.name == 'thing a'
                    thing.name = 'broken'
                }
            }

            apply plugin: MyPlugin

            model {
                things.thingA {
                    name = 'thing a'
                }
                tasks {
                    show(Task) {
                        doLast {
                            println $.calculated
                        }
                    }
                }
            }
        '''

        when:
        fails 'show'

        then:
        failure.assertHasCause("Exception thrown while executing model rule: CalculateName#broken(Thing)")
        failure.assertHasCause("Attempt to modify a read only view of model element 'things.thingA' of type 'Thing' given to rule CalculateName#broken(Thing)")
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy