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

ucar.nc2.dt.grid.GeoGrid Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 1998-2018 John Caron and University Corporation for Atmospheric Research/Unidata
 * See LICENSE for license information.
 */
package ucar.nc2.dt.grid;

import ucar.ma2.*;
import ucar.nc2.*;
import ucar.nc2.constants.AxisType;
import ucar.nc2.dt.GridCoordSystem;
import ucar.nc2.dt.GridDatatype;
import ucar.nc2.util.NamedObject;
import ucar.nc2.dataset.*;
import ucar.unidata.geoloc.*;
import ucar.unidata.util.Format;
import java.io.IOException;
import java.util.*;

/**
 * A georeferencing "gridded" VariableEnhanced, that has a GridCoordSys.
 * In VisAD data model, it is a sampled Field.
 * The dimension are put into canonical order: (rt, e, t, z, y, x).
 * 

*

* Implementation note: * If the Horizontal axes are 2D, the x and y dimensions are arbitrarily chosen to be * gcs.getXHorizAxis().getDimension(1), gcs.getXHorizAxis().getDimension(0) respectively. *

* * @author caron */ public class GeoGrid implements NamedObject, ucar.nc2.dt.GridDatatype { private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(GeoGrid.class); private static final boolean debugArrayShape = false; private final GridDataset dataset; private final GridCoordSys gcs; private final VariableDS vs; private int xDimOrgIndex, yDimOrgIndex, zDimOrgIndex = -1, tDimOrgIndex = -1, eDimOrgIndex = -1, rtDimOrgIndex = -1; private int xDimNewIndex = -1, yDimNewIndex = -1, zDimNewIndex = -1, tDimNewIndex = -1, eDimNewIndex = -1, rtDimNewIndex = -1; private final List mydims; /** * Constructor. * * @param dataset belongs to this dataset * @param dsvar wraps this Variable * @param gcs has this grid coordinate system */ public GeoGrid(GridDataset dataset, VariableDS dsvar, GridCoordSys gcs) { this.dataset = dataset; this.vs = dsvar; this.gcs = gcs; CoordinateAxis xaxis = gcs.getXHorizAxis(); if (xaxis instanceof CoordinateAxis1D) { xDimOrgIndex = findDimension(gcs.getXHorizAxis().getDimension(0)); yDimOrgIndex = findDimension(gcs.getYHorizAxis().getDimension(0)); } else { // 2D case // coverity[COPY_PASTE_ERROR] yDimOrgIndex = findDimension(gcs.getXHorizAxis().getDimension(0)); xDimOrgIndex = findDimension(gcs.getXHorizAxis().getDimension(1)); } if (gcs.getVerticalAxis() != null) zDimOrgIndex = findDimension(gcs.getVerticalAxis().getDimension(0)); if (gcs.getTimeAxis() != null) { if (gcs.getTimeAxis1D() != null) tDimOrgIndex = findDimension(gcs.getTimeAxis1D().getDimension(0)); else tDimOrgIndex = findDimension(gcs.getTimeAxis().getDimension(1)); // deal with swath where time not independent dimension LOOK rewrite ?? if ((tDimOrgIndex == yDimOrgIndex) || (tDimOrgIndex == xDimOrgIndex)) { tDimOrgIndex = -1; } } if (gcs.getEnsembleAxis() != null) eDimOrgIndex = findDimension(gcs.getEnsembleAxis().getDimension(0)); if (gcs.getRunTimeAxis() != null) rtDimOrgIndex = findDimension(gcs.getRunTimeAxis().getDimension(0)); // construct canonical dimension list int count = 0; this.mydims = new ArrayList<>(); if ((rtDimOrgIndex >= 0) && (rtDimOrgIndex != tDimOrgIndex)) { mydims.add(dsvar.getDimension(rtDimOrgIndex)); rtDimNewIndex = count++; } if (eDimOrgIndex >= 0) { mydims.add(dsvar.getDimension(eDimOrgIndex)); eDimNewIndex = count++; } if (tDimOrgIndex >= 0) { Dimension tdim = dsvar.getDimension(tDimOrgIndex); mydims.add(tdim); tDimNewIndex = count++; } if (zDimOrgIndex >= 0) { mydims.add(dsvar.getDimension(zDimOrgIndex)); zDimNewIndex = count++; } if (yDimOrgIndex >= 0) { mydims.add(dsvar.getDimension(yDimOrgIndex)); yDimNewIndex = count++; } if (xDimOrgIndex >= 0) { mydims.add(dsvar.getDimension(xDimOrgIndex)); xDimNewIndex = count; } } private int findDimension(Dimension want) { java.util.List dims = vs.getDimensions(); for (int i = 0; i < dims.size(); i++) { Dimension d = (Dimension) dims.get(i); if (d.equals(want)) return i; } return -1; } /** * Returns an ArrayList containing the dimensions used by this geoGrid. * The dimension are put into canonical order: (rt, e, t, z, y, x). Note that the z and t dimensions are optional. * If the Horizontal axes are 2D, the x and y dimensions are arbitrarily chosen to be * gcs.getXHorizAxis().getDimension(1), gcs.getXHorizAxis().getDimension(0), respectively. * * @return List with objects of type Dimension, in canonical order. */ public java.util.List getDimensions() { return new ArrayList<>(mydims); } /** * get the ith dimension * * @param i : which dimension * @return ith Dimension */ public Dimension getDimension(int i) { if ((i < 0) || (i >= mydims.size())) return null; return mydims.get(i); } /** * get the time Dimension, if it exists */ public Dimension getTimeDimension() { return tDimNewIndex < 0 ? null : getDimension(tDimNewIndex); } /** * get the z Dimension, if it exists */ public Dimension getZDimension() { return zDimNewIndex < 0 ? null : getDimension(zDimNewIndex); } /** * get the y Dimension, if it exists */ public Dimension getYDimension() { return yDimNewIndex < 0 ? null : getDimension(yDimNewIndex); } /** * get the x Dimension, if it exists */ public Dimension getXDimension() { return xDimNewIndex < 0 ? null : getDimension(xDimNewIndex); } /** * get the ensemble Dimension, if it exists */ public Dimension getEnsembleDimension() { return eDimNewIndex < 0 ? null : getDimension(eDimNewIndex); } /** * get the run time Dimension, if it exists */ public Dimension getRunTimeDimension() { return rtDimNewIndex < 0 ? null : getDimension(rtDimNewIndex); } /** * get the time Dimension index in the geogrid (canonical order), or -1 if none */ public int getTimeDimensionIndex() { return tDimNewIndex; } /** * get the z Dimension index in the geogrid (canonical order), or -1 if none */ public int getZDimensionIndex() { return zDimNewIndex; } /** * get the y Dimension index in the geogrid (canonical order) */ public int getYDimensionIndex() { return yDimNewIndex; } /** * get the x Dimension index in the geogrid (canonical order) */ public int getXDimensionIndex() { return xDimNewIndex; } /** * get the ensemble Dimension index in the geogrid (canonical order) */ public int getEnsembleDimensionIndex() { return eDimNewIndex; } /** * get the runtime Dimension index in the geogrid (canonical order) */ public int getRunTimeDimensionIndex() { return rtDimNewIndex; } /** * Convenience function; lookup Attribute by name. * * @param name the name of the attribute * @return the attribute, or null if not found */ public Attribute findAttributeIgnoreCase(String name) { return vs.attributes().findAttributeIgnoreCase(name); } /** * Convenience function; lookup Attribute value by name. Must be String valued * * @param attName name of the attribute * @param defaultValue if not found, use this as the default * @return Attribute string value, or default if not found. */ public String findAttValueIgnoreCase(String attName, String defaultValue) { return vs.attributes().findAttributeString(attName, defaultValue); } // implementation of GridDatatype interface /** * get the rank */ public int getRank() { return mydims.size(); } /** * get the shape */ public int[] getShape() { int[] shape = new int[mydims.size()]; for (int i = 0; i < mydims.size(); i++) { Dimension d = mydims.get(i); shape[i] = d.getLength(); } return shape; } /** * get the data type */ public DataType getDataType() { return vs.getDataType(); } public AttributeContainer attributes() { return vs.attributes(); } /** @deprecated use attributes() */ @Deprecated public List getAttributes() { return vs.getAttributes(); } public VariableDS getVariable() { return vs; } public String getFullName() { return vs.getFullName(); } public String getName() { return vs.getFullName(); } public String getShortName() { return vs.getShortName(); } /** * get the GridCoordSys for this GeoGrid. */ public GridCoordSystem getCoordinateSystem() { return gcs; } /** * get the Projection. * * @deprecated this will return Projection, not ProjectionImpl in 6. */ public ProjectionImpl getProjection() { return gcs.getProjection(); } /** * @return ArrayList of thredds.util.NamedObject, from the GridCoordSys. * @deprecated will move in ver 6 */ @Deprecated public List getLevels() { return gcs.getLevels(); } /** * @return ArrayList of thredds.util.NamedObject, from the GridCoordSys. * @deprecated will move in ver 6 */ @Deprecated public List getTimes() { return gcs.getTimes(); } /** * get the standardized description, or null if none. */ @Override public String getDescription() { return vs.getDescription(); } @Override public Object getValue() { return this; } /** * get the unit as a string */ public String getUnitsString() { String units = vs.getUnitsString(); return (units == null) ? "" : units; } /** * @return getUnitsString() * @deprecated use getUnitsString() */ public java.lang.String getUnitString() { return getUnitsString(); } // public ucar.unidata.geoloc.ProjectionImpl getProjection() { return gcs.getProjection(); } /** * true if there may be missing data, see VariableDS.hasMissing() */ public boolean hasMissingData() { return vs.hasMissing(); } /** * if val is missing data, see VariableDS.isMissingData() */ public boolean isMissingData(double val) { return vs.isMissing(val); } public boolean hasMissing() { return vs.hasMissing(); } public boolean isMissing(double val) { return vs.isMissing(val); } /** * Convert (in place) all values in the given array that are considered * as "missing" to Float.NaN, according to isMissingData(val). * * @param values input array * @return input array, with missing values converted to NaNs. */ public float[] setMissingToNaN(float[] values) { if (!vs.hasMissing()) return values; int length = values.length; for (int i = 0; i < length; i++) { double value = values[i]; if (vs.isMissing(value)) values[i] = Float.NaN; } return values; } /** * Get the minimum and the maximum data value of the previously read Array, * skipping missing values as defined by isMissingData(double val). * * @param a Array to get min/max values * @return both min and max value. */ public MAMath.MinMax getMinMaxSkipMissingData(Array a) { return MAMath.getMinMaxSkipMissingData(a, this); } /** * Reads in the data "volume" at the given time index. * If its a product set, put into canonical order (z-y-x). * If not a product set, reorder to (z,i,j), where i, j are from the * original * * @param t time index; ignored if no time axis. * @return data[z,y,x] or data[y,x] if no z axis. */ public Array readVolumeData(int t) throws java.io.IOException { return readDataSlice(t, -1, -1, -1); } /** * Reads a Y-X "horizontal slice" at the given time and vertical index. * If its a product set, put into canonical order (y-x). * * @param t time index; ignored if no time axis. * @param z vertical index; ignored if no z axis. * @return data[y,x] * @throws java.io.IOException on read error */ public Array readYXData(int t, int z) throws java.io.IOException { return readDataSlice(t, z, -1, -1); } /** * Reads a Z-Y "vertical slice" at the given time and x index. * If its a product set, put into canonical order (z-y). * * @param t time index; ignored if no time axis. * @param x x index; ignored if no x axis. * @return data[z,y] * @throws java.io.IOException on read error */ public Array readZYData(int t, int x) throws java.io.IOException { return readDataSlice(t, -1, -1, x); } /** * @throws java.io.IOException on read error * @deprecated use readDataSlice */ public Array getDataSlice(int t, int z, int y, int x) throws java.io.IOException { return readDataSlice(t, z, y, x); } /** * This reads an arbitrary data slice, returning the data in * canonical order (t-z-y-x). If any dimension does not exist, ignore it. * * @param t if < 0, get all of time dim; if valid index, fix slice to that value. * @param z if < 0, get all of z dim; if valid index, fix slice to that value. * @param y if < 0, get all of y dim; if valid index, fix slice to that value. * @param x if < 0, get all of x dim; if valid index, fix slice to that value. * @return data[t,z,y,x], eliminating missing or fixed dimension. */ public Array readDataSlice(int t, int z, int y, int x) throws java.io.IOException { return readDataSlice(0, 0, t, z, y, x); } /** * This reads an arbitrary data slice, returning the data in * canonical order (rt-e-t-z-y-x). If any dimension does not exist, ignore it. * * @param rt if < 0, get all of runtime dim; if valid index, fix slice to that value. * @param e if < 0, get all of ensemble dim; if valid index, fix slice to that value. * @param t if < 0, get all of time dim; if valid index, fix slice to that value. * @param z if < 0, get all of z dim; if valid index, fix slice to that value. * @param y if < 0, get all of y dim; if valid index, fix slice to that value. * @param x if < 0, get all of x dim; if valid index, fix slice to that value. * @return data[rt,e,t,z,y,x], eliminating missing or fixed dimension. */ public Array readDataSlice(int rt, int e, int t, int z, int y, int x) throws java.io.IOException { int rank = vs.getRank(); int[] start = new int[rank]; int[] shape = new int[rank]; for (int i = 0; i < rank; i++) { start[i] = 0; shape[i] = 1; } Dimension xdim = getXDimension(); Dimension ydim = getYDimension(); Dimension zdim = getZDimension(); Dimension tdim = getTimeDimension(); Dimension edim = getEnsembleDimension(); Dimension rtdim = getRunTimeDimension(); // construct the shape of the data volume to be read if (rtdim != null) { if ((rt >= 0) && (rt < rtdim.getLength())) start[rtDimOrgIndex] = rt; // fix rt else { shape[rtDimOrgIndex] = rtdim.getLength(); // all of rt } } if (edim != null) { if ((e >= 0) && (e < edim.getLength())) start[eDimOrgIndex] = e; // fix e else { shape[eDimOrgIndex] = edim.getLength(); // all of e } } if (tdim != null) { if ((t >= 0) && (t < tdim.getLength())) start[tDimOrgIndex] = t; // fix t else { shape[tDimOrgIndex] = tdim.getLength(); // all of t } } if (zdim != null) { if ((z >= 0) && (z < zdim.getLength())) start[zDimOrgIndex] = z; // fix z else { shape[zDimOrgIndex] = zdim.getLength(); // all of z } } if (ydim != null) { if ((y >= 0) && (y < ydim.getLength())) start[yDimOrgIndex] = y; // fix y else { shape[yDimOrgIndex] = ydim.getLength(); // all of y } } if (xdim != null) { if ((x >= 0) && (x < xdim.getLength())) // all of x start[xDimOrgIndex] = x; // fix x else { shape[xDimOrgIndex] = xdim.getLength(); // all of x } } if (debugArrayShape) { System.out.println("read shape from org variable = "); for (int i = 0; i < rank; i++) System.out.println( " start = " + start[i] + " shape = " + shape[i] + " name = " + vs.getDimension(i).getShortName()); } // read it Array dataVolume; try { dataVolume = vs.read(start, shape); } catch (Exception ex) { log.error("GeoGrid.getdataSlice() on variable " + vs.getFullName() + " dataset " + dataset.getLocation(), ex); throw new java.io.IOException(ex); } // LOOK: the real problem is the lack of named dimensions in the Array object // figure out correct permutation for canonical ordering for permute List oldDims = new ArrayList<>(vs.getDimensions()); int[] permuteIndex = new int[dataVolume.getRank()]; int count = 0; if (oldDims.contains(rtdim)) permuteIndex[count++] = oldDims.indexOf(rtdim); if (oldDims.contains(edim)) permuteIndex[count++] = oldDims.indexOf(edim); if (oldDims.contains(tdim)) permuteIndex[count++] = oldDims.indexOf(tdim); if (oldDims.contains(zdim)) permuteIndex[count++] = oldDims.indexOf(zdim); if (oldDims.contains(ydim)) permuteIndex[count++] = oldDims.indexOf(ydim); if (oldDims.contains(xdim)) permuteIndex[count] = oldDims.indexOf(xdim); if (debugArrayShape) { System.out.println("oldDims = "); for (Dimension oldDim : oldDims) System.out.println(" oldDim = " + oldDim.getShortName()); System.out.println("permute dims = "); for (int aPermuteIndex : permuteIndex) System.out.println(" oldDim index = " + aPermuteIndex); } // check to see if we need to permute boolean needPermute = false; for (int i = 0; i < permuteIndex.length; i++) { if (i != permuteIndex[i]) needPermute = true; } // permute to the order rt,e,t,z,y,x if (needPermute) dataVolume = dataVolume.permute(permuteIndex); // eliminate fixed dimensions, but not all dimensions of length 1. count = 0; if (rtdim != null) { if (rt >= 0) dataVolume = dataVolume.reduce(count); else count++; } if (edim != null) { if (e >= 0) dataVolume = dataVolume.reduce(count); else count++; } if (tdim != null) { if (t >= 0) dataVolume = dataVolume.reduce(count); else count++; } if (zdim != null) { if (z >= 0) dataVolume = dataVolume.reduce(count); else count++; } if (ydim != null) { if (y >= 0) dataVolume = dataVolume.reduce(count); else count++; } if (xdim != null) { if (x >= 0) dataVolume = dataVolume.reduce(count); } return dataVolume; } @Override public Array readSubset(List subset) throws InvalidRangeException, IOException { // get the ranges list in the order of the variable; a null range means "all" to vs.read() int rank = getRank(); Range[] varRange = new Range[rank]; for (Range r : subset) { AxisType type = AxisType.valueOf(r.getName()); switch (type) { case Lon: case GeoX: varRange[xDimOrgIndex] = r; break; case Lat: case GeoY: varRange[yDimOrgIndex] = r; break; case Height: case Pressure: case GeoZ: varRange[zDimOrgIndex] = r; break; case Time: varRange[tDimOrgIndex] = r; break; case RunTime: varRange[rtDimOrgIndex] = r; break; case Ensemble: varRange[eDimOrgIndex] = r; break; } } // read it Array dataVolume = vs.read(Arrays.asList(varRange)); // LOOK permute to canonical order if needed return dataVolume; } ////////////////////////////////// /** * Create a new GeoGrid that is a logical subset of this GeoGrid. * * @param t_range subset the time dimension, or null if you want all of it * @param z_range subset the vertical dimension, or null if you want all of it * @param bbox a lat/lon bounding box, or null if you want all x,y * @param z_stride use only if z_range is null, then take all z with this stride (1 means all) * @param y_stride use this stride on the y coordinate (1 means all) * @param x_stride use this stride on the x coordinate (1 means all) * @return subsetted GeoGrid * @throws InvalidRangeException if bbox does not intersect GeoGrid */ public GeoGrid subset(Range t_range, Range z_range, LatLonRect bbox, int z_stride, int y_stride, int x_stride) throws InvalidRangeException { if ((z_range == null) && (z_stride > 1)) { Dimension zdim = getZDimension(); if (zdim != null) z_range = new Range(0, zdim.getLength() - 1, z_stride); } Range y_range = null, x_range = null; if (bbox != null) { List yx_ranges = gcs.getRangesFromLatLonRect(bbox); y_range = (Range) yx_ranges.get(0); x_range = (Range) yx_ranges.get(1); } if (y_stride > 1) { if (y_range == null) { Dimension ydim = getYDimension(); if (ydim != null) y_range = new Range(0, ydim.getLength() - 1, y_stride); } else { y_range = new Range(y_range.first(), y_range.last(), y_stride); } } if (x_stride > 1) { if (x_range == null) { Dimension xdim = getXDimension(); if (xdim != null) x_range = new Range(0, xdim.getLength() - 1, x_stride); } else { x_range = new Range(x_range.first(), x_range.last(), x_stride); } } return subset(t_range, z_range, y_range, x_range); } public GridDatatype makeSubset(Range t_range, Range z_range, LatLonRect bbox, int z_stride, int y_stride, int x_stride) throws InvalidRangeException { return subset(t_range, z_range, bbox, z_stride, y_stride, x_stride); } /** * Create a new GeoGrid that is a logical subset of this GeoGrid. * * @param t_range subset the time dimension, or null if you want all of it * @param z_range subset the vertical dimension, or null if you want all of it * @param y_range subset the y dimension, or null if you want all of it * @param x_range subset the x dimension, or null if you want all of it * @return subsetted GeoGrid * @throws InvalidRangeException if any of the ranges are invalid */ public GeoGrid subset(Range t_range, Range z_range, Range y_range, Range x_range) throws InvalidRangeException { return (GeoGrid) makeSubset(null, null, t_range, z_range, y_range, x_range); } public GridDatatype makeSubset(Range rt_range, Range e_range, Range t_range, Range z_range, Range y_range, Range x_range) throws InvalidRangeException { // get the ranges list int rank = getRank(); Range[] ranges = new Range[rank]; if (null != getXDimension()) ranges[xDimOrgIndex] = x_range; if (null != getYDimension()) ranges[yDimOrgIndex] = y_range; if (null != getZDimension()) ranges[zDimOrgIndex] = z_range; if (null != getTimeDimension()) ranges[tDimOrgIndex] = t_range; if (null != getRunTimeDimension()) ranges[rtDimOrgIndex] = rt_range; if (null != getEnsembleDimension()) ranges[eDimOrgIndex] = e_range; List rangesList = Arrays.asList(ranges); // subset the variable VariableDS v_section = (VariableDS) vs.section(new Section(rangesList)); List dims = v_section.getDimensions(); for (Dimension dim : dims) { dim.setShared(true); // make them shared (section will make them unshared) } // subset the axes in the GridCoordSys GridCoordSys gcs_section = new GridCoordSys(gcs, rt_range, e_range, t_range, z_range, y_range, x_range); // now we can make the geogrid return new GeoGrid(dataset, v_section, gcs_section); } ///////////////////////////////////////////////////////////////////////////////// /** * Instances which have same name and coordinate system are equal. */ public boolean equals(Object oo) { if (this == oo) return true; if (!(oo instanceof GeoGrid)) return false; GeoGrid d = (GeoGrid) oo; if (!getFullName().equals(d.getFullName())) return false; return getCoordinateSystem().equals(d.getCoordinateSystem()); } /** * Override Object.hashCode() to be consistent with equals. */ public int hashCode() { if (hashCode == 0) { int result = 17; // result = 37*result + dataset.getName().hashCode(); result = 37 * result + getFullName().hashCode(); result = 37 * result + getCoordinateSystem().hashCode(); hashCode = result; } return hashCode; } private int hashCode; // Bloch, item 8 /** * string representation */ public String toString() { return getFullName(); } /** * nicely formatted information */ public String getInfo() { StringBuilder buf = new StringBuilder(200); buf.setLength(0); buf.append(getFullName()); Format.tab(buf, 30, true); buf.append(getUnitsString()); Format.tab(buf, 60, true); buf.append(hasMissingData()); Format.tab(buf, 66, true); buf.append(getDescription()); return buf.toString(); } public int compareTo(GridDatatype g) { return getFullName().compareTo(g.getFullName()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy