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

com.hazelcast.shaded.org.apache.calcite.avatica.AvaticaUtils Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * 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.shaded.org.apache.calcite.avatica;

import com.hazelcast.shaded.org.apache.calcite.avatica.util.UnsynchronizedBuffer;

import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.lang.reflect.Field;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.AbstractList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

/** Avatica utilities. */
public class AvaticaUtils {

  private static final Map BOX;

  private static final MethodHandle SET_LARGE_MAX_ROWS =
      method(void.class, Statement.class, "setLargeMaxRows", long.class);
  private static final MethodHandle GET_LARGE_MAX_ROWS =
      method(long.class, Statement.class, "getLargeMaxRows");
  private static final MethodHandle GET_LARGE_UPDATE_COUNT =
      method(void.class, Statement.class, "getLargeUpdateCount");
  private static final MethodHandle EXECUTE_LARGE_BATCH =
      method(long[].class, Statement.class, "executeLargeBatch");

  private static final Set UNIQUE_STRINGS = new HashSet<>();

  private static final ThreadLocal PER_THREAD_BUFFER  = new ThreadLocal() {
    @Override protected byte[] initialValue() {
      return new byte[4096];
    }
  };

  private static final int SKIP_BUFFER_SIZE = 4096;

  private AvaticaUtils() {}

  static {
    BOX = new HashMap<>();
    BOX.put(boolean.class, Boolean.class);
    BOX.put(byte.class, Byte.class);
    BOX.put(char.class, Character.class);
    BOX.put(short.class, Short.class);
    BOX.put(int.class, Integer.class);
    BOX.put(long.class, Long.class);
    BOX.put(float.class, Float.class);
    BOX.put(double.class, Double.class);
  }

  private static MethodHandle method(Class returnType, Class targetType,
      String name, Class... argTypes) {
    final MethodHandles.Lookup lookup = MethodHandles.lookup();
    try {
      return lookup.findVirtual(targetType, name,
          MethodType.methodType(returnType, targetType, argTypes));
    } catch (NoSuchMethodException e) {
      return null;
    } catch (IllegalAccessException e) {
      throw new RuntimeException(e);
    }
  }

  /**
   * Does nothing with its argument. Call this method when you have a value
   * you are not interested in, but you don't want the compiler to warn that
   * you are not using it.
   */
  public static void discard(Object o) {
    if (false) {
      discard(o);
    }
  }

  /**
   * Use this method to flag temporary code.
   *
   * 

Example #1: *

   * if (AvaticaUtils.remark("baz fixed") == null) {
   *   baz();
   * }
* *

Example #2: *

   * /** @see AvaticaUtils#remark Remove before checking in */
   * void uselessMethod() {}
   * 
*/ public static T remark(T remark) { return remark; } /** * Use this method to flag code that should be re-visited after upgrading * a component. * *

If the intended change is that a class or member be removed, flag * instead using a {@link Deprecated} annotation followed by a comment such as * "to be removed before 2.0". */ public static boolean upgrade(String remark) { discard(remark); return false; } /** * Adapts a primitive array into a {@link List}. For example, * {@code asList(new double[2])} returns a {@code List<Double>}. */ public static List primitiveList(final Object array) { // REVIEW: A per-type list might be more efficient. (Or might not.) return new AbstractList() { public Object get(int index) { return java.lang.reflect.Array.get(array, index); } public int size() { return java.lang.reflect.Array.getLength(array); } }; } /** * Converts a camelCase name into an upper-case underscore-separated name. * For example, {@code camelToUpper("myJdbcDriver")} returns * "MY_JDBC_DRIVER". */ public static String camelToUpper(String name) { StringBuilder buf = new StringBuilder(); for (int i = 0; i < name.length(); i++) { char c = name.charAt(i); if (Character.isUpperCase(c)) { buf.append('_'); } else { c = Character.toUpperCase(c); } buf.append(c); } return buf.toString(); } /** * Converts an underscore-separated name into a camelCase name. * For example, {@code uncamel("MY_JDBC_DRIVER")} returns "myJdbcDriver". */ public static String toCamelCase(String name) { StringBuilder buf = new StringBuilder(); int nextUpper = -1; for (int i = 0; i < name.length(); i++) { char c = name.charAt(i); if (c == '_') { nextUpper = i + 1; continue; } if (nextUpper == i) { c = Character.toUpperCase(c); } else { c = Character.toLowerCase(c); } buf.append(c); } return buf.toString(); } /** Returns the boxed class. For example, {@code box(int.class)} * returns {@code java.lang.Integer}. */ public static Class box(Class clazz) { if (clazz.isPrimitive()) { return BOX.get(clazz); } return clazz; } /** Creates an instance of a plugin class. * *

First looks for a static member called "{@code INSTANCE}", * then calls a public default constructor. * *

If {@code className} contains a "#", instead looks for a static field. * *

In the "#" case, if the static field is a {@link ThreadLocal}, this * method dereferences the {@code ThreadLocal} by calling * {@link ThreadLocal#get()}. This behavior allows, for example, client code * to pass an object to a JDBC driver. The JDBC driver needs to be running in * the same JVM and the same thread as the client. * * @param pluginClass Class (or interface) to instantiate * @param className Name of implementing class * @param Class * @return Plugin instance */ public static T instantiatePlugin(Class pluginClass, String className) { String right = null; String left = null; Object value = null; try { // Given a static field, say "com.example.MyClass#FOO_INSTANCE", return // the value of that static field. if (className.contains("#")) { int i = className.indexOf('#'); left = className.substring(0, i); right = className.substring(i + 1); //noinspection unchecked final Class clazz = (Class) Class.forName(left); final Field field; field = clazz.getField(right); final Object fieldValue = field.get(null); if (fieldValue instanceof ThreadLocal) { value = ((ThreadLocal) fieldValue).get(); } else { value = fieldValue; } return pluginClass.cast(value); } //noinspection unchecked final Class clazz = (Class) Class.forName(className); try { // We assume that if there is an INSTANCE field it is static and // has the right type. final Field field = clazz.getField("INSTANCE"); value = field.get(null); return pluginClass.cast(value); } catch (NoSuchFieldException e) { // ignore } if (!pluginClass.isAssignableFrom(clazz)) { throw new RuntimeException("Property '" + className + "' not valid for plugin type " + pluginClass.getName()); } return clazz.getConstructor().newInstance(); } catch (ClassNotFoundException e) { throw new RuntimeException("Property '" + className + "' not valid as '" + className + "' not found in the classpath", e); } catch (NoSuchFieldException e) { // We can't ignore it because the right field is user configured. throw new RuntimeException("Property '" + className + "' not valid as there is no '" + right + "' field in the class of '" + left + "'", e); } catch (ClassCastException e) { throw new RuntimeException("Property '" + className + "' not valid as cannot convert " + (value == null ? "null" : value.getClass().getName()) + " to " + pluginClass.getCanonicalName(), e); } catch (NoSuchMethodException e) { throw new RuntimeException("Property '" + className + "' not valid as " + "the default constructor is necessary, " + "but not found in the class of '" + className + "'", e); } catch (ReflectiveOperationException e) { throw new RuntimeException("Property '" + className + "' not valid. The exception info here : " + e.getMessage(), e); } } /** Reads the contents of an input stream and returns as a string. */ public static String readFully(InputStream inputStream) throws IOException { return readFully(inputStream, new UnsynchronizedBuffer(1024)); } /** Reads the contents of an input stream and returns as a string. */ public static String readFully(InputStream inputStream, UnsynchronizedBuffer buffer) throws IOException { // Variant that lets us use a pooled Buffer final byte[] bytes = _readFully(inputStream, buffer); return AvaticaUtils.newStringUtf8(bytes); } /** Reads the contents of an input stream and returns as a string. */ public static byte[] readFullyToBytes(InputStream inputStream) throws IOException { return readFullyToBytes(inputStream, new UnsynchronizedBuffer(1024)); } /** Reads the contents of an input stream and returns as a string. */ public static byte[] readFullyToBytes(InputStream inputStream, UnsynchronizedBuffer buffer) throws IOException { // Variant that lets us use a pooled Buffer return _readFully(inputStream, buffer); } /** * Reads the contents of an input stream and returns a byte array. * * @param inputStream the input to read from. * @return A byte array whose length is equal to the number of bytes contained. */ static byte[] _readFully(InputStream inputStream, UnsynchronizedBuffer buffer) throws IOException { final byte[] bytes = PER_THREAD_BUFFER.get(); for (;;) { int count = inputStream.read(bytes, 0, bytes.length); if (count < 0) { break; } buffer.write(bytes, 0, count); } return buffer.toArray(); } /** * Reads and discards all data available on the InputStream. */ public static void skipFully(InputStream inputStream) throws IOException { byte[] temp = null; while (true) { long bytesSkipped = inputStream.skip(Long.MAX_VALUE); if (bytesSkipped == 0) { if (temp == null) { temp = new byte[SKIP_BUFFER_SIZE]; } int bytesRead = inputStream.read(temp, 0, SKIP_BUFFER_SIZE); if (bytesRead < 0) { // EOF return; } } } } /** Invokes {@code Statement#setLargeMaxRows}, falling back on * {@link Statement#setMaxRows(int)} if the method does not exist (before * JDK 1.8) or throws {@link UnsupportedOperationException}. */ public static void setLargeMaxRows(Statement statement, long n) throws SQLException { if (SET_LARGE_MAX_ROWS != null) { try { // Call Statement.setLargeMaxRows SET_LARGE_MAX_ROWS.invokeExact(n); return; } catch (UnsupportedOperationException e) { // ignore, and fall through to call Statement.setMaxRows } catch (Error | RuntimeException | SQLException e) { throw e; } catch (Throwable e) { throw new RuntimeException(e); } } int i = (int) Math.max(Math.min(n, Integer.MAX_VALUE), Integer.MIN_VALUE); statement.setMaxRows(i); } /** Invokes {@code Statement#getLargeMaxRows}, falling back on * {@link Statement#getMaxRows()} if the method does not exist (before * JDK 1.8) or throws {@link UnsupportedOperationException}. */ public static long getLargeMaxRows(Statement statement) throws SQLException { if (GET_LARGE_MAX_ROWS != null) { try { // Call Statement.getLargeMaxRows return (long) GET_LARGE_MAX_ROWS.invokeExact(); } catch (UnsupportedOperationException e) { // ignore, and fall through to call Statement.getMaxRows } catch (Error | RuntimeException | SQLException e) { throw e; } catch (Throwable e) { throw new RuntimeException(e); } } return statement.getMaxRows(); } /** Invokes {@code Statement#getLargeUpdateCount}, falling back on * {@link Statement#getUpdateCount()} if the method does not exist (before * JDK 1.8) or throws {@link UnsupportedOperationException}. */ public static long getLargeUpdateCount(Statement statement) throws SQLException { if (GET_LARGE_UPDATE_COUNT != null) { try { // Call Statement.getLargeUpdateCount return (long) GET_LARGE_UPDATE_COUNT.invokeExact(); } catch (UnsupportedOperationException e) { // ignore, and fall through to call Statement.getUpdateCount } catch (Error | RuntimeException | SQLException e) { throw e; } catch (Throwable e) { throw new RuntimeException(e); } } return statement.getUpdateCount(); } /** Invokes {@code Statement#executeLargeBatch}, falling back on * {@link PreparedStatement#executeBatch} if the method does not exist * (before JDK 1.8) or throws {@link UnsupportedOperationException}. */ public static long[] executeLargeBatch(Statement statement) throws SQLException { if (EXECUTE_LARGE_BATCH != null) { try { // Call Statement.executeLargeBatch return (long[]) EXECUTE_LARGE_BATCH.invokeExact(); } catch (UnsupportedOperationException e) { // ignore, and fall through to call Statement.executeBatch } catch (Error | RuntimeException | SQLException e) { throw e; } catch (Throwable e) { throw new RuntimeException(e); } } return toLongs(statement.executeBatch()); } /** Generates a string that is unique in the execution of the JVM. * It is used by tests to ensure that they create distinct temporary tables. * The strings are never thrown away, so don't put too much in there! * Thread safe. */ public static String unique(String base) { synchronized (UNIQUE_STRINGS) { String s = base; while (!UNIQUE_STRINGS.add(s)) { s = base + "_" + UNIQUE_STRINGS.size(); } return s; } } /** Converts a {@code long} to {@code int}, rounding as little as possible * if the value is outside the legal range for an {@code int}. */ public static int toSaturatedInt(long value) { if (value > Integer.MAX_VALUE) { return Integer.MAX_VALUE; } if (value < Integer.MIN_VALUE) { return Integer.MIN_VALUE; } return (int) value; } /** * Converts an array of {@code long} values to an array of {@code int} * values, truncating values outside the legal range for an {@code int} * to {@link Integer#MIN_VALUE} or {@link Integer#MAX_VALUE}. * * @param longs An array of {@code long}s * @return An array of {@code int}s */ public static int[] toSaturatedInts(long[] longs) { final int[] ints = new int[longs.length]; for (int i = 0; i < longs.length; i++) { ints[i] = toSaturatedInt(longs[i]); } return ints; } /** Converts an array of {@code int} values to an array of {@code long} * values. */ public static long[] toLongs(int[] ints) { final long[] longs = new long[ints.length]; for (int i = 0; i < ints.length; i++) { longs[i] = ints[i]; } return longs; } public static String newStringUtf8(final byte[] bytes) { return newString(bytes, StandardCharsets.UTF_8); } //CHECKSTYLE: OFF public static String newString(final byte[] bytes, final Charset charset) { return new String(bytes, charset); } public static String newString(final byte[] bytes, final String charsetName) throws UnsupportedEncodingException { return new String(bytes, charsetName); } //CHECKSTYLE:ON } // End AvaticaUtils.java