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

com.hazelcast.org.apache.calcite.rel.metadata.ReflectiveRelMetadataProvider Maven / Gradle / Ivy

/*
 * 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 com.hazelcast.org.apache.calcite.rel.metadata;

import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.runtime.FlatLists;
import com.hazelcast.org.apache.calcite.util.BuiltInMethod;
import com.hazelcast.org.apache.calcite.util.ImmutableNullableList;
import com.hazelcast.org.apache.calcite.util.Pair;
import com.hazelcast.org.apache.calcite.util.ReflectiveVisitor;
import com.hazelcast.org.apache.calcite.util.Util;

import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.com.google.common.collect.ImmutableMultimap;
import com.hazelcast.com.google.common.collect.Multimap;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

/**
 * Implementation of the {@link RelMetadataProvider} interface that dispatches
 * metadata methods to methods on a given object via reflection.
 *
 * 

The methods on the target object must be public and non-static, and have * the same signature as the implemented metadata method except for an * additional first parameter of type {@link RelNode} or a sub-class. That * parameter gives this provider an indication of that relational expressions it * can handle.

* *

For an example, see {@link RelMdColumnOrigins#SOURCE}. */ public class ReflectiveRelMetadataProvider implements RelMetadataProvider, ReflectiveVisitor { //~ Instance fields -------------------------------------------------------- private final ConcurrentMap, UnboundMetadata> map; private final Class metadataClass0; private final ImmutableMultimap handlerMap; //~ Constructors ----------------------------------------------------------- /** * Creates a ReflectiveRelMetadataProvider. * * @param map Map * @param metadataClass0 Metadata class * @param handlerMap Methods handled and the objects to call them on */ protected ReflectiveRelMetadataProvider( ConcurrentMap, UnboundMetadata> map, Class metadataClass0, Multimap handlerMap) { assert !map.isEmpty() : "are your methods named wrong?"; this.map = map; this.metadataClass0 = metadataClass0; this.handlerMap = ImmutableMultimap.copyOf(handlerMap); } /** Returns an implementation of {@link RelMetadataProvider} that scans for * methods with a preceding argument. * *

For example, {@link BuiltInMetadata.Selectivity} has a method * {@link BuiltInMetadata.Selectivity#getSelectivity(RexNode)}. * A class

* *

   * class RelMdSelectivity {
   *   public Double getSelectivity(Union rel, RexNode predicate) { }
   *   public Double getSelectivity(Filter rel, RexNode predicate) { }
   * 
* *

provides implementations of selectivity for relational expressions * that extend {@link com.hazelcast.org.apache.calcite.rel.core.Union} * or {@link com.hazelcast.org.apache.calcite.rel.core.Filter}.

*/ public static RelMetadataProvider reflectiveSource(Method method, MetadataHandler target) { return reflectiveSource(target, ImmutableList.of(method)); } /** Returns a reflective metadata provider that implements several * methods. */ public static RelMetadataProvider reflectiveSource(MetadataHandler target, Method... methods) { return reflectiveSource(target, ImmutableList.copyOf(methods)); } private static RelMetadataProvider reflectiveSource( final MetadataHandler target, final ImmutableList methods) { final Space2 space = Space2.create(target, methods); // This needs to be a concurrent map since RelMetadataProvider are cached in static // fields, thus the map is subject to concurrent modifications later. // See map.put in com.hazelcast.org.apache.calcite.rel.metadata.ReflectiveRelMetadataProvider.apply( // java.lang.Class) final ConcurrentMap, UnboundMetadata> methodsMap = new ConcurrentHashMap<>(); for (Class key : space.classes) { ImmutableNullableList.Builder builder = ImmutableNullableList.builder(); for (final Method method : methods) { builder.add(space.find(key, method)); } final List handlerMethods = builder.build(); final UnboundMetadata function = (rel, mq) -> (Metadata) Proxy.newProxyInstance( space.metadataClass0.getClassLoader(), new Class[]{space.metadataClass0}, (proxy, method, args) -> { // Suppose we are an implementation of Selectivity // that wraps "filter", a LogicalFilter. Then we // implement // Selectivity.selectivity(rex) // by calling method // new SelectivityImpl().selectivity(filter, rex) if (method.equals(BuiltInMethod.METADATA_REL.method)) { return rel; } if (method.equals(BuiltInMethod.OBJECT_TO_STRING.method)) { return space.metadataClass0.getSimpleName() + "(" + rel + ")"; } int i = methods.indexOf(method); if (i < 0) { throw new AssertionError("not handled: " + method + " for " + rel); } final Method handlerMethod = handlerMethods.get(i); if (handlerMethod == null) { throw new AssertionError("not handled: " + method + " for " + rel); } final Object[] args1; final List key1; if (args == null) { args1 = new Object[]{rel, mq}; key1 = FlatLists.of(rel, method); } else { args1 = new Object[args.length + 2]; args1[0] = rel; args1[1] = mq; System.arraycopy(args, 0, args1, 2, args.length); final Object[] args2 = args1.clone(); args2[1] = method; // replace RelMetadataQuery with method for (int j = 0; j < args2.length; j++) { if (args2[j] == null) { args2[j] = NullSentinel.INSTANCE; } else if (args2[j] instanceof RexNode) { // Can't use RexNode.equals - it is not deep args2[j] = args2[j].toString(); } } key1 = FlatLists.copyOf(args2); } if (mq.map.put(rel, key1, NullSentinel.INSTANCE) != null) { throw new CyclicMetadataException(); } try { return handlerMethod.invoke(target, args1); } catch (InvocationTargetException | UndeclaredThrowableException e) { Util.throwIfUnchecked(e.getCause()); throw new RuntimeException(e.getCause()); } finally { mq.map.remove(rel, key1); } }); methodsMap.put(key, function); } return new ReflectiveRelMetadataProvider(methodsMap, space.metadataClass0, space.providerMap); } public Multimap> handlers( MetadataDef def) { final ImmutableMultimap.Builder> builder = ImmutableMultimap.builder(); for (Map.Entry entry : handlerMap.entries()) { if (def.methods.contains(entry.getKey())) { //noinspection unchecked builder.put(entry.getKey(), entry.getValue()); } } return builder.build(); } private static boolean couldImplement(Method handlerMethod, Method method) { if (!handlerMethod.getName().equals(method.getName()) || (handlerMethod.getModifiers() & Modifier.STATIC) != 0 || (handlerMethod.getModifiers() & Modifier.PUBLIC) == 0) { return false; } final Class[] parameterTypes1 = handlerMethod.getParameterTypes(); final Class[] parameterTypes = method.getParameterTypes(); return parameterTypes1.length == parameterTypes.length + 2 && RelNode.class.isAssignableFrom(parameterTypes1[0]) && RelMetadataQuery.class == parameterTypes1[1] && Arrays.asList(parameterTypes) .equals(Util.skip(Arrays.asList(parameterTypes1), 2)); } //~ Methods ---------------------------------------------------------------- public UnboundMetadata apply( Class relClass, Class metadataClass) { if (metadataClass == metadataClass0) { return apply(relClass); } else { return null; } } @SuppressWarnings({ "unchecked", "SuspiciousMethodCalls" }) public UnboundMetadata apply( Class relClass) { List> newSources = new ArrayList<>(); for (;;) { UnboundMetadata function = map.get(relClass); if (function != null) { for (@SuppressWarnings("rawtypes") Class clazz : newSources) { map.put(clazz, function); } return function; } else { newSources.add(relClass); } for (Class interfaceClass : relClass.getInterfaces()) { if (RelNode.class.isAssignableFrom(interfaceClass)) { final UnboundMetadata function2 = map.get(interfaceClass); if (function2 != null) { for (@SuppressWarnings("rawtypes") Class clazz : newSources) { map.put(clazz, function2); } return function2; } } } if (RelNode.class.isAssignableFrom(relClass.getSuperclass())) { relClass = (Class) relClass.getSuperclass(); } else { return null; } } } /** Workspace for computing which methods can act as handlers for * given metadata methods. */ static class Space { final Set> classes = new HashSet<>(); final Map, Method>, Method> handlerMap = new HashMap<>(); final ImmutableMultimap providerMap; Space(Multimap providerMap) { this.providerMap = ImmutableMultimap.copyOf(providerMap); // Find the distinct set of RelNode classes handled by this provider, // ordered base-class first. for (Map.Entry entry : providerMap.entries()) { final Method method = entry.getKey(); final MetadataHandler provider = entry.getValue(); for (final Method handlerMethod : provider.getClass().getMethods()) { if (couldImplement(handlerMethod, method)) { @SuppressWarnings("unchecked") final Class relNodeClass = (Class) handlerMethod.getParameterTypes()[0]; classes.add(relNodeClass); handlerMap.put(Pair.of(relNodeClass, method), handlerMethod); } } } } /** Finds an implementation of a method for {@code relNodeClass} or its * nearest base class. Assumes that base classes have already been added to * {@code map}. */ @SuppressWarnings({ "unchecked", "SuspiciousMethodCalls" }) Method find(final Class relNodeClass, Method method) { Objects.requireNonNull(relNodeClass); for (Class r = relNodeClass;;) { Method implementingMethod = handlerMap.get(Pair.of(r, method)); if (implementingMethod != null) { return implementingMethod; } for (Class clazz : r.getInterfaces()) { if (RelNode.class.isAssignableFrom(clazz)) { implementingMethod = handlerMap.get(Pair.of(clazz, method)); if (implementingMethod != null) { return implementingMethod; } } } r = r.getSuperclass(); if (r == null || !RelNode.class.isAssignableFrom(r)) { throw new IllegalArgumentException("No handler for method [" + method + "] applied to argument of type [" + relNodeClass + "]; we recommend you create a catch-all (RelNode) handler"); } } } } /** Extended work space. */ static class Space2 extends Space { private Class metadataClass0; Space2(Class metadataClass0, ImmutableMultimap providerMap) { super(providerMap); this.metadataClass0 = metadataClass0; } public static Space2 create(MetadataHandler target, ImmutableList methods) { assert methods.size() > 0; final Method method0 = methods.get(0); //noinspection unchecked Class metadataClass0 = (Class) method0.getDeclaringClass(); assert Metadata.class.isAssignableFrom(metadataClass0); for (Method method : methods) { assert method.getDeclaringClass() == metadataClass0; } final ImmutableMultimap.Builder providerBuilder = ImmutableMultimap.builder(); for (final Method method : methods) { providerBuilder.put(method, target); } return new Space2(metadataClass0, providerBuilder.build()); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy