| /* |
| * 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.lang |
| |
| import groovy.test.GroovyTestCase |
| |
| /** |
| * Tests for method missing handling in Groovy |
| * |
| * @since 1.5 |
| */ |
| class MethodMissingTest extends GroovyTestCase { |
| |
| void tearDown() { |
| MMTest.metaClass = null |
| MMTest2.metaClass = null |
| MMTest3.metaClass = null |
| MMTest5.metaClass = null |
| MMTest6.metaClass = null |
| } |
| |
| void testOverrideStaticMethodMissingTwice() { |
| MMTest2.metaClass.'static'.methodMissing = { String name, args -> "foo" } |
| assertEquals "foo",MMTest2.doStuff() |
| MMTest2.metaClass.'static'.methodMissing = { String name, args -> "bar" } |
| assertEquals "bar",MMTest2.doStuff() |
| } |
| |
| void testSimpleMethodMissing() { |
| def t = new MMTest() |
| assertEquals "world", t.hello() |
| assertEquals "foo", t.stuff() |
| } |
| |
| void testMethodMissingViaMetaClass() { |
| def t1 = new MMTest2() |
| shouldFail(MissingMethodException) { |
| t1.stuff() |
| } |
| assertEquals "world", t1.hello() |
| MMTest2.metaClass.methodMissing = { String name, args -> |
| "foo" |
| } |
| def t2 = new MMTest2() |
| assertEquals "world", t2.hello() |
| assertEquals "foo", t2.stuff() |
| } |
| |
| void testMethodMissingWithRegistration() { |
| MMTest2.metaClass.methodMissing = { String name, args -> |
| MMTest2.metaClass."$name" = {-> "bar" } |
| "foo" |
| } |
| def t2 = new MMTest2() |
| assertEquals "world", t2.hello() |
| assertEquals "foo", t2.stuff() |
| assertEquals "bar", t2.stuff() |
| } |
| |
| |
| void testStaticMethodMissingViaMetaClass() { |
| assertEquals "world", MMTest3.hello() |
| shouldFail(MissingMethodException) { |
| MMTest3.stuff() |
| } |
| MMTest3.metaClass.'static'.methodMissing = { String name, args-> |
| "foo" |
| } |
| assertEquals "world", MMTest3.hello() |
| assertEquals "foo", MMTest3.stuff() |
| } |
| |
| void testMethodMissingWithInheritance() { |
| assertEquals "world",MMTest6.hello() |
| assertEquals "cruel world",MMTest6.goodbye() |
| |
| MMTest6.metaClass.'static'.methodMissing = { String name, args -> |
| "foo" |
| } |
| assertEquals "foo",MMTest6.bar() |
| |
| shouldFail(MissingMethodException) { |
| MMTest5.bar() |
| } |
| } |
| |
| // GROOVY-4701 |
| void testMethodMissingExceptionWithBrokenToString() { |
| def exception = new MissingMethodException("methodName", getClass(), new Broken()) |
| def sw = new StringWriter() |
| sw.withPrintWriter { pw -> |
| exception.printStackTrace(pw) |
| assert sw.toString().contains("groovy.lang.MissingMethodException: No signature of method") |
| assert sw.toString().contains("is applicable for argument types: (Broken) values: [<Broken@????>]") |
| assert !sw.toString().contains("toString broken") |
| } |
| sw = new StringWriter() |
| exception = new MissingMethodException("methodName", getClass(), new Broken(hash: 9)) |
| sw.withPrintWriter { pw -> |
| exception.printStackTrace(pw) |
| assert sw.toString().contains("groovy.lang.MissingMethodException: No signature of method") |
| assert sw.toString().contains("is applicable for argument types: (Broken) values: [<Broken@9>]") |
| assert !sw.toString().contains("toString broken") |
| } |
| } |
| } |
| |
| class MMTest { |
| def hello() { "world" } |
| def methodMissing(String name, args) { |
| "foo" |
| } |
| } |
| |
| class MMTest2 { |
| def hello() { "world" } |
| } |
| |
| class MMTest3 { |
| static hello() { "world" } |
| } |
| |
| class MMTest5 { |
| static hello() { "world" } |
| } |
| |
| class MMTest6 extends MMTest5 { |
| static goodbye() { "cruel world" } |
| } |
| |
| class Broken { |
| int hash = -1 |
| @Override |
| String toString() { |
| throw new RuntimeException("toString broken") |
| } |
| @Override |
| int hashCode() { |
| if (hash == -1) throw new RuntimeException("hashCode broken") |
| else hash |
| } |
| } |