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

org.soulwing.jdbc.source.ResourceSQLSource Maven / Gradle / Ivy

/*
 * File created on Aug 5, 2015
 *
 * Copyright (c) 2015 Carl Harris, Jr
 * and others as noted
 *
 * 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.soulwing.jdbc.source;

import java.io.BufferedReader;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URL;

/**
 * An {@link SQLSource} that reads SQL statements from a resource that can be
 * located using a URL.
 *
 * @author Carl Harris
 */
public class ResourceSQLSource extends ReaderSQLSource {

  public static final String DEFAULT_ENCODING = "UTF-8";

  /**
   * Constructs a new SQL source for a resource at the specified location,
   * which uses the specified encoding.
   * @param location location of the resource; this URL may use the
   *    {@code classpath:} scheme to specify a resource relative to the
   *    root of the classpath which will be accessing using the thread context
   *    class loader
   * @param encoding character set name
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public ResourceSQLSource(URL location, String encoding) {
    super(openReader(location, encoding));
  }

  /**
   * Creates a new SQL source for a resource in the same package as the given
   * class, which uses the {@link #DEFAULT_ENCODING default encoding}.
   * @param name name of the resource
   * @param relativeToClass class that provides the package location of the
   *    resource
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, Class relativeToClass) {
    return with(name, relativeToClass, DEFAULT_ENCODING);
  }

  /**
   * Creates a new SQL source for a resource in the same package as the given
   * class which uses the specified encoding.
   * @param name name of the resource
   * @param relativeToClass class that provides the package location of the
   *    resource
   * @param encoding character set name
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, Class relativeToClass,
      String encoding) {
    return with(name, new ClassResourceAccessor(relativeToClass), encoding);
  }

  /**
   * Creates a new SQL source for a resource relative to the root of the
   * given class loader, which uses the {@link #DEFAULT_ENCODING default encoding}.
   * @param name name of the resource
   * @param classLoader class loader that will be used to access the resource
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, ClassLoader classLoader) {
    return with(name, classLoader, DEFAULT_ENCODING);
  }

  /**
   * Creates a new SQL source for a resource relative to the root of the
   * given class loader, which uses the specified encoding.
   * @param name name of the resource
   * @param classLoader class loader that will be used to access the resource
   * @param encoding character set name
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, ClassLoader classLoader,
      String encoding) {
    return with(name, new ClassLoaderResourceAccessor(classLoader), encoding);
  }

  /**
   * Creates a new SQL source for a resource that will be obtained from the
   * given resource accessor and which uses the {@link #DEFAULT_ENCODING default encoding}.
   * @param name name of the resource
   * @param accessor accessor that will be used to obtain a URL for the resource
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, ResourceAccessor accessor) {
    return with(name, accessor, DEFAULT_ENCODING);
  }

  /**
   * Creates a new SQL source for a resource that will be obtained from the
   * given resource accessor and which uses the {@link #DEFAULT_ENCODING default encoding}.
   * @param name name of the resource
   * @param accessor accessor that will be used to obtain a URL for the resource
   * @param encoding character set name
   * @return SQL source
   * @throws SQLResourceNotFoundException if the specified resource is not found
   * @throws SQLInputException if an I/O errors when accessing the resource
   */
  public static ResourceSQLSource with(String name, ResourceAccessor accessor,
      String encoding) {
    return with(accessor.getResource(name), encoding);
  }

  /**
   * Creates a new SQL source for a resource relative to the root of the
   * classpath, which uses the {@link #DEFAULT_ENCODING default encoding}.
   * 

* The resource will be located using the thread context class loader. * * @param location URL for the resource; may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath (which will be accessed using the thread context * class loader) * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(String location) { return with(location, DEFAULT_ENCODING); } /** * Creates a new SQL source for a resource relative to the root of the * classpath, which uses the specified encoding. *

* The resource will be located using the thread context class loader. * * @param location URL for the resource; may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath (which will be accessed using the thread context * class loader) * @param encoding character set name * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(String location, String encoding) { return with(URI.create(location), encoding); } /** * Creates a new SQL source for a resource at the specified location, * which uses the specified encoding. * @param location location of the resource; this URL may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath (which will be accessed using the thread context * class loader) * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(URI location) { return with(location, DEFAULT_ENCODING); } /** * Creates a new SQL source for a resource at the specified location, * which uses the specified encoding. * @param location location of the resource; this URL may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath (which will be accessed using the thread context * class loader) * @param encoding character set name * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(URI location, String encoding) { return new ResourceSQLSource(translate(location), encoding); } /** * Creates a new SQL source for a resource at the specified location, * which uses the specified encoding. * @param location location of the resource; this URL may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath which will be accessing using the thread context * class loader * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(URL location) { return new ResourceSQLSource(location, DEFAULT_ENCODING); } /** * Creates a new SQL source for a resource at the specified location, * which uses the specified encoding. * @param location location of the resource; this URL may use the * {@code classpath:} scheme to specify a resource relative to the * root of the classpath which will be accessing using the thread context * class loader * @param encoding character set name * @return SQL source * @throws SQLResourceNotFoundException if the specified resource is not found * @throws SQLInputException if an I/O errors when accessing the resource */ public static ResourceSQLSource with(URL location, String encoding) { return new ResourceSQLSource(location, encoding); } /** * An accessor for a resource. *

* This is an interface that should be defined in the JDK and * implemented by the {@link Class} and {@link ClassLoader} objects. Alas, * it is not. */ public interface ResourceAccessor { /** * Gets a URL for a resource. * @param name name of the resource * @return location of the resource * @throws SQLResourceNotFoundException if the resource cannot be found */ URL getResource(String name) throws SQLResourceNotFoundException; } /** * A {@link ResourceAccessor} based on a {@link ClassLoader}. */ private static class ClassLoaderResourceAccessor implements ResourceAccessor { private final ClassLoader classLoader; public ClassLoaderResourceAccessor(ClassLoader classLoader) { this.classLoader = classLoader; } @Override public URL getResource(String name) { URL location = classLoader.getResource(name); if (location == null) { throw new SQLResourceNotFoundException(name); } return location; } } /** * A {@link ResourceAccessor} based on a {@link Class}. */ private static class ClassResourceAccessor implements ResourceAccessor { private final Class clazz; public ClassResourceAccessor(Class clazz) { this.clazz = clazz; } @Override public URL getResource(String name) { URL location = clazz.getResource(name); if (location == null) { throw new SQLResourceNotFoundException(name, clazz); } return location; } } /** * Translates a URI to a URL. *

* The JDK URL class doesn't support the {@code classpath:} scheme, so we * translate it here, if necessary. * * @param uri the URL to translate * @return translated URL * @throws SQLResourceNotFoundException if a {@code classpath:} URL refers * to a non-existent resource * @throws NullPointerException if {@code url} is {@code null} */ private static URL translate(URI uri) { if (uri == null) { throw new NullPointerException("resource location is required"); } if ("classpath".equals(uri.getScheme())) { String path = uri.getSchemeSpecificPart(); if (path.startsWith("/")) { path = path.substring(1); } URL url = Thread.currentThread().getContextClassLoader() .getResource(path); if (url == null) { throw new SQLResourceNotFoundException(path); } return url; } try { return uri.toURL(); } catch (MalformedURLException ex) { throw new SQLInputException(uri.toString(), ex); } } /** * Opens an input stream to a resource. * @param location location of the resource * @param encoding character set name * @return buffered reader for the stream */ private static Reader openReader(URL location, String encoding) { try { InputStream inputStream = location.openStream(); return new BufferedReader(new InputStreamReader(inputStream, encoding)); } catch (FileNotFoundException ex) { throw new SQLResourceNotFoundException(location.toString()); } catch (IOException ex) { throw new SQLInputException(ex.getMessage(), ex); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy