/* * GeoTools - The Open Source Java GIS Toolkit * http://geotools.org * * (C) 2007-2015, Open Source Geospatial Foundation (OSGeo) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. */ package org.geotools.coverage.io.netcdf; import it.geosolutions.imageio.stream.input.FileImageInputStreamExtImpl; import it.geosolutions.imageio.stream.input.URIImageInputStream; import it.geosolutions.imageio.stream.input.URIImageInputStreamImpl; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.Serializable; import java.net.URL; import java.net.URLDecoder; import java.util.Arrays; import java.util.EnumSet; import java.util.List; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; import javax.imageio.ImageIO; import javax.imageio.spi.ImageReaderSpi; import javax.imageio.stream.ImageInputStream; import org.geotools.coverage.io.CoverageAccess; import org.geotools.coverage.io.Driver; import org.geotools.coverage.io.FileDriver; import org.geotools.coverage.io.impl.DefaultFileDriver; import org.geotools.factory.Hints; import org.geotools.imageio.netcdf.NetCDFImageReaderSpi; import org.geotools.imageio.netcdf.utilities.NetCDFUtilities; import org.opengis.util.ProgressListener; /** * NetCDF Driver * */ public class NetCDFDriver extends DefaultFileDriver implements FileDriver, Driver { private static final List<String> EXTENSIONS = NetCDFUtilities.isGribAvailable() ? Arrays .asList("nc", "grb", "grb2", "grib") : Arrays.asList("nc"); /** Logger. */ private final static Logger LOGGER = org.geotools.util.logging.Logging .getLogger("org.geotools.coverage.io.netcdf"); /** A static {@link ImageReaderSpi} to be used to call canDecodeInput */ final static ImageReaderSpi SPI; static { ImageReaderSpi temp = null; if (checkNetCDF()) { try { temp = new NetCDFImageReaderSpi(); } catch (Throwable e) { if (LOGGER.isLoggable(Level.SEVERE)) LOGGER.log(Level.SEVERE, e.getLocalizedMessage(), e); temp = null; } } // assign the INTERNAL_STORE_SPI SPI = temp; } public NetCDFDriver() { super( "NetCDF", "NetCDF Coverage Format", "NetCDF Coverage Format", null, EXTENSIONS, EnumSet.of(DriverCapabilities.CONNECT)); } public CoverageAccess connect(java.net.URL source, Map<String, Serializable> params, Hints hints, ProgressListener listener) throws IOException { return new NetCDFAccess(this, source, params, hints, listener); } public CoverageAccess connect(Map<String, Serializable> params, Hints hints, ProgressListener listener) throws IOException { return new NetCDFAccess(this, null, params, hints, listener); } public boolean isAvailable() { return checkNetCDF(); } public static boolean checkNetCDF() { boolean available = true; // if these classes are here, then the runtime environment has // access to JAI and the JAI ImageI/O toolbox. try { Class.forName("javax.media.jai.JAI"); Class.forName("com.sun.media.jai.operator.ImageReadDescriptor"); if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("NetCDFDriver is available."); } catch (ClassNotFoundException cnf) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("NetCDFDriver is not available."); available = false; } return available; } @Override protected boolean canConnect(URL url, Map<String, Serializable> params) { if (url == null) { return false; } ImageInputStream inputStream = null; Object source = null; URIImageInputStream uriInStream = null; try { // // URL management // In case the URL points to a file we need to get to the file // directly and avoid caching. In case it points to http or ftp // or it is an open stream we have very small to do and we need // to enable caching. // if (url.getProtocol().equalsIgnoreCase("file")) { File file = new File(URLDecoder.decode(url.getFile(), "UTF-8")); if (file.exists()) { if (!file.canRead() || !file.isFile()) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("File cannot be read or it is a directory"); return false; } if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("Provided URL is a file"); // setting source source = file; } else { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("File doesn't exist"); return false; } } else { if (url.getProtocol().toLowerCase().startsWith("http") || url.getProtocol().equalsIgnoreCase("dods")) { uriInStream = new URIImageInputStreamImpl(url.toURI()); source = uriInStream; if (!SPI.canDecodeInput(source)) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("Unable to decode the inputStream"); return false; } return true; } else if (url.getProtocol().equalsIgnoreCase("ftp")) { source = url.openStream(); } else { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("Unsupported protocol"); return false; } } // get a stream inputStream = (ImageInputStream) ((source instanceof ImageInputStream) ? source : (source instanceof File) ? new FileImageInputStreamExtImpl((File) source) : ImageIO.createImageInputStream(source)); if (inputStream == null) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("Unable to get an ImageInputStream"); return false; } inputStream.mark(); if (!SPI.canDecodeInput(inputStream)) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.fine("Unable to decode the inputStream"); return false; } return true; } catch (Throwable e) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.log(Level.FINE, e.getLocalizedMessage(), e); return false; } finally { if (inputStream != null) { try { inputStream.close(); } catch (Exception e) { } } if (source != null && (source instanceof InputStream)) { try { ((InputStream) source).close(); } catch (Exception e) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.log(Level.FINE, e.getLocalizedMessage(), e); } } if (uriInStream != null) { try { uriInStream.close(); } catch (Exception e) { if (LOGGER.isLoggable(Level.FINE)) LOGGER.log(Level.FINE, e.getLocalizedMessage(), e); } } } } }