xref: /aosp_15_r20/art/test/044-proxy/src/NarrowingTest.java (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2011 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.*;
17*795d594fSAndroid Build Coastguard Worker import java.util.Arrays;
18*795d594fSAndroid Build Coastguard Worker 
19*795d594fSAndroid Build Coastguard Worker class NarrowingTest {
20*795d594fSAndroid Build Coastguard Worker 
21*795d594fSAndroid Build Coastguard Worker    interface I1 {
foo()22*795d594fSAndroid Build Coastguard Worker        public Object foo();
23*795d594fSAndroid Build Coastguard Worker    }
24*795d594fSAndroid Build Coastguard Worker 
25*795d594fSAndroid Build Coastguard Worker    interface I2 extends I1 {
26*795d594fSAndroid Build Coastguard Worker        // Note that this method declaration narrows the return type.
27*795d594fSAndroid Build Coastguard Worker        @Override
foo()28*795d594fSAndroid Build Coastguard Worker        public String foo();
29*795d594fSAndroid Build Coastguard Worker    }
30*795d594fSAndroid Build Coastguard Worker 
main(String[] args)31*795d594fSAndroid Build Coastguard Worker    public static void main(String[] args) {
32*795d594fSAndroid Build Coastguard Worker        I2 proxy = (I2) Proxy.newProxyInstance(NarrowingTest.class.getClassLoader(),
33*795d594fSAndroid Build Coastguard Worker                                               new Class<?>[] { I2.class },
34*795d594fSAndroid Build Coastguard Worker                new InvocationHandler() {
35*795d594fSAndroid Build Coastguard Worker                    int count = 0;
36*795d594fSAndroid Build Coastguard Worker                    @Override
37*795d594fSAndroid Build Coastguard Worker                    public Object invoke(Object proxy, Method method,
38*795d594fSAndroid Build Coastguard Worker                                         Object[] args) throws Throwable {
39*795d594fSAndroid Build Coastguard Worker                        System.out.println("Invocation of " + method);
40*795d594fSAndroid Build Coastguard Worker                        if (count == 0) {
41*795d594fSAndroid Build Coastguard Worker                            count++;
42*795d594fSAndroid Build Coastguard Worker                            return "hello";
43*795d594fSAndroid Build Coastguard Worker                        } else {
44*795d594fSAndroid Build Coastguard Worker                            return Integer.valueOf(1);
45*795d594fSAndroid Build Coastguard Worker                        }
46*795d594fSAndroid Build Coastguard Worker                    }
47*795d594fSAndroid Build Coastguard Worker                });
48*795d594fSAndroid Build Coastguard Worker        Main.registerProxyClassName(proxy.getClass().getCanonicalName());
49*795d594fSAndroid Build Coastguard Worker 
50*795d594fSAndroid Build Coastguard Worker        Method[] methods = proxy.getClass().getDeclaredMethods();
51*795d594fSAndroid Build Coastguard Worker        Arrays.sort(methods, new MethodComparator());
52*795d594fSAndroid Build Coastguard Worker        System.out.println("Proxy methods: " +
53*795d594fSAndroid Build Coastguard Worker                           Main.replaceProxyClassNamesForOutput(Arrays.deepToString(methods)));
54*795d594fSAndroid Build Coastguard Worker 
55*795d594fSAndroid Build Coastguard Worker        System.out.println("Invoking foo using I2 type: " + proxy.foo());
56*795d594fSAndroid Build Coastguard Worker 
57*795d594fSAndroid Build Coastguard Worker        I1 proxyAsParent = proxy;
58*795d594fSAndroid Build Coastguard Worker        System.out.println("Invoking foo using I1 type: " + proxyAsParent.foo());
59*795d594fSAndroid Build Coastguard Worker 
60*795d594fSAndroid Build Coastguard Worker        try {
61*795d594fSAndroid Build Coastguard Worker            proxy.foo();
62*795d594fSAndroid Build Coastguard Worker            System.out.println("Didn't get expected exception");
63*795d594fSAndroid Build Coastguard Worker        } catch (ClassCastException e) {
64*795d594fSAndroid Build Coastguard Worker            // With an I2 invocation returning an integer is an exception.
65*795d594fSAndroid Build Coastguard Worker            System.out.println("Got expected exception");
66*795d594fSAndroid Build Coastguard Worker        }
67*795d594fSAndroid Build Coastguard Worker 
68*795d594fSAndroid Build Coastguard Worker        System.out.println("Proxy narrowed invocation return type passed");
69*795d594fSAndroid Build Coastguard Worker    }
70*795d594fSAndroid Build Coastguard Worker }
71