ucar.nc2.Group Maven / Gradle / Ivy
/*
* Copyright 1998-2009 University Corporation for Atmospheric Research/Unidata
*
* Portions of this software were developed by the Unidata Program at the
* University Corporation for Atmospheric Research.
*
* Access and use of this software shall impose the following obligations
* and understandings on the user. The user is granted the right, without
* any fee or cost, to use, copy, modify, alter, enhance and distribute
* this software, and any derivative works thereof, and its supporting
* documentation for any purpose whatsoever, provided that this entire
* notice appears in all copies of the software, derivative works and
* supporting documentation. Further, UCAR requests that the user credit
* UCAR/Unidata in any publications that result from the use of this
* software or in any product that includes this software. The names UCAR
* and/or Unidata, however, may not be used in any advertising or publicity
* to endorse or promote any products or commercial entity unless specific
* written permission is obtained from UCAR/Unidata. The user also
* understands that UCAR/Unidata is not obligated to provide the user with
* any support, consulting, training or assistance of any kind with regard
* to the use, operation and performance of this software nor to provide
* the user with any updates, revisions, new versions or "bug fixes."
*
* THIS SOFTWARE IS PROVIDED BY UCAR/UNIDATA "AS IS" AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL UCAR/UNIDATA BE LIABLE FOR ANY SPECIAL,
* INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
* FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
* NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
* WITH THE ACCESS, USE OR PERFORMANCE OF THIS SOFTWARE.
*/
package ucar.nc2;
import ucar.ma2.DataType;
import java.util.ArrayList;
import java.util.List;
import java.util.Collections;
import java.io.PrintWriter;
/**
* A Group is a logical collection of Variables.
* The Groups in a Dataset form a hierarchical tree, like directories on a disk.
* A Group has a name and optionally a set of Attributes.
* There is always at least one Group in a dataset, the root Group, whose name is the empty string.
* Immutable if setImmutable() was called.
*
* @author caron
*/
public class Group {
protected NetcdfFile ncfile;
protected Group parent;
protected String shortName;
protected List variables = new ArrayList();
protected List dimensions = new ArrayList();
protected List groups = new ArrayList();
protected List attributes = new ArrayList();
protected List enumTypedefs = new ArrayList();
private boolean immutable = false;
/**
* Get the full name, starting from the root Group.
*
* @return group full name
*/
public String getName() {
return (parent == null) || (parent == ncfile.getRootGroup()) ? shortName : parent.getName() + "/" + shortName;
}
/**
* Is this the root group?
*
* @return true if root group
*/
public boolean isRoot() {
return parent == null;
}
/** Get the "short" name, unique within its parent Group.
* @return group short name
*/
public String getShortName() { return shortName; }
/** Get its parent Group, or null if its the root group.
* @return parent Group
*/
public Group getParentGroup() { return parent; }
/** Get the Variables contained directly in this group.
* @return List of type Variable; may be empty, not null.
*/
public java.util.List getVariables() { return variables; }
/**
* Find the Variable with the specified (short) name in this group.
* @param varShortName short name of Variable within this group.
* @return the Variable, or null if not found
*/
public Variable findVariable(String varShortName) {
if (varShortName == null) return null;
for (Variable v : variables) {
if (varShortName.equals(v.getShortName()))
return v;
}
return null;
}
/*
* Find the Variable with the specified escaped (short) name in this group.
* @param varShortNameEscaped escaped short name of Variable within this group.
* @return the Variable, or null if not found
* @see NetcdfFile#escapeName
* @see NetcdfFile#unescapeName
*
public Variable findVariableEscaped(String varShortNameEscaped) {
if (varShortNameEscaped == null) return null;
return findVariable( NetcdfFile.unescapeName(varShortNameEscaped));
} */
/**
* Find the Variable with the specified (short) name in this group or a parent group.
* @param varShortName short name of Variable.
* @return the Variable, or null if not found
*/
public Variable findVariableOrInParent(String varShortName) {
if (varShortName == null) return null;
Variable v = findVariable( varShortName);
if ((v == null) && (parent != null))
v = parent.findVariableOrInParent( varShortName);
return v;
}
/** Get the Groups contained directly in this Group.
* @return List of type Group; may be empty, not null.
*/
public java.util.List getGroups() { return groups; }
/** Get the owning NetcdfFile
* @return owning NetcdfFile.
*/
public NetcdfFile getNetcdfFile() { return ncfile; }
/**
* Retrieve the Group with the specified (short) name.
* @param groupShortName short name of the nested group you are looking for. May be escaped.
* @see NetcdfFile#unescapeName
* @return the Group, or null if not found
*/
public Group findGroup(String groupShortName) {
if ( groupShortName == null) return null;
groupShortName = NetcdfFile.unescapeName(groupShortName);
for (Group group : groups) {
if (groupShortName.equals(group.getShortName()))
return group;
}
return null;
}
/**
* Get the Dimensions contained directly in this group.
* @return List of type Dimension; may be empty, not null.
*/
public java.util.List getDimensions() { return dimensions; }
/**
* Get the enumerations contained directly in this group.
* @return List of type EnumTypedef; may be empty, not null.
*/
public java.util.List getEnumTypedefs() { return enumTypedefs; }
/**
* Retrieve a Dimension using its (short) name. If it doesnt exist in this group,
* recursively look in parent groups.
* @param name Dimension name.
* @return the Dimension, or null if not found
*/
public Dimension findDimension(String name) {
if (name == null) return null;
name = NetcdfFile.unescapeName(name);
Dimension d = findDimensionLocal( name);
if (d != null) return d;
if (parent != null)
return parent.findDimension( name);
return null;
}
/**
* Retrieve a Dimension using its (short) name, in this group only
* @param name Dimension name.
* @return the Dimension, or null if not found
*/
public Dimension findDimensionLocal(String name) {
if (name == null) return null;
name = NetcdfFile.unescapeName(name);
for (Dimension d : dimensions) {
if (name.equals(d.getName()))
return d;
}
return null;
}
/**
* Get the set of attributes contained directly in this Group.
* @return List of type Attribute; may be empty, not null.
*/
public java.util.List getAttributes() { return attributes; }
/**
* Find an Attribute in this Group by its name.
*
* @param name the name of the attribute. may be escaped
* @return the attribute, or null if not found
*/
public Attribute findAttribute(String name) {
if (name == null) return null;
name = NetcdfFile.unescapeName(name);
for (Attribute a : attributes) {
if (name.equals(a.getName()))
return a;
}
return null;
}
/**
* Find an Attribute in this Group by its name, ignore case.
*
* @param name the name of the attribute
* @return the attribute, or null if not found
*/
public Attribute findAttributeIgnoreCase(String name) {
if (name == null) return null;
name = NetcdfFile.unescapeName(name);
for (Attribute a : attributes) {
if (name.equalsIgnoreCase(a.getName()))
return a;
}
return null;
}
/**
* Find an Enumeration Typedef using its (short) name. If it doesnt exist in this group,
* recursively look in parent groups.
*
* @param name Enumeration name.
* @return the Enumeration, or null if not found
*/
public EnumTypedef findEnumeration(String name) {
if (name == null) return null;
name = NetcdfFile.unescapeName(name); // LOOK: why ??
for (EnumTypedef d : enumTypedefs) {
if (name.equals(d.getName()))
return d;
}
if (parent != null)
return parent.findEnumeration(name);
return null;
}
/**
* Get the common parent of this and the other group.
* Cant fail, since the root group is always a parent of any 2 groups.
* @param other the other group
* @return common parent of this and the other group
*/
public Group commonParent( Group other) {
if (isParent(other)) return this;
if (other.isParent(this)) return other;
while (!other.isParent(this))
other = other.getParentGroup();
return other;
}
/**
* Is this a parent of the other Group?
* @param other another Group
* @return true is it is equal or a parent
*/
public boolean isParent( Group other) {
while ((other != this) && (other.getParentGroup() != null))
other = other.getParentGroup();
return (other == this);
}
//////////////////////////////////////////////////////////////////////////////////////
/** Get String with name and attributes. Used in short descriptions like tooltips.
* @return name and attributes String.
*/
public String getNameAndAttributes() {
StringBuilder sbuff = new StringBuilder();
sbuff.append("Group ");
sbuff.append(getShortName());
sbuff.append("\n");
for (Attribute att : attributes) {
sbuff.append(" ").append(getShortName()).append(":");
sbuff.append(att.toString());
sbuff.append(";");
sbuff.append("\n");
}
return sbuff.toString();
}
protected void writeCDL(PrintWriter out, String indent, boolean strict) {
boolean hasE = (enumTypedefs.size() > 0);
boolean hasD = (dimensions.size() > 0);
boolean hasV = (variables.size() > 0);
boolean hasG = (groups.size() > 0);
boolean hasA = (attributes.size() > 0);
if (hasE) {
out.print(indent+" types:\n");
for (EnumTypedef e : enumTypedefs) {
out.print(indent + e.writeCDL(strict));
out.print(indent + "\n");
}
out.print(indent+"\n");
}
if (hasD)
out.print(indent+" dimensions:\n");
for (Dimension myd : dimensions) {
out.print(indent + myd.writeCDL(strict));
out.print(indent + "\n");
}
if (hasV)
out.print(indent+" variables:\n");
for (Variable v : variables) {
out.print(v.writeCDL(indent + " ", false, strict));
}
for (Group g : groups) {
String gname = strict ? NetcdfFile.escapeName(g.getShortName()) : g.getShortName();
out.print("\n " + indent + "Group " + gname + " {\n");
g.writeCDL(out, indent + " ", strict);
out.print(indent + " }\n");
}
if (hasA && (hasE || hasD || hasV || hasG))
out.print("\n");
if (hasA && strict)
out.print(indent + " // global attributes:\n");
for (Attribute att : attributes) {
String name = strict ? NetcdfFile.escapeName(getShortName()) : getShortName();
out.print(indent + " " + name + ":");
out.print( att.toString(strict));
out.print(";");
if (!strict && (att.getDataType() != DataType.STRING)) out.print(" // " + att.getDataType());
out.print("\n");
}
}
//////////////////////////////////////////////////////////////////////////////////////
/** Constructor
* @param ncfile NetcdfFile owns this Group
* @param parent parent of Group. If null, this is the root Group.
* @param shortName short name of Group.
*/
public Group(NetcdfFile ncfile, Group parent, String shortName) {
this.ncfile = ncfile;
this.parent = parent == null ? ncfile.getRootGroup() : parent ;
this.shortName = shortName;
}
/** Set the Group's parent Group
* @param parent parent group.
*/
public void setParentGroup( Group parent) {
if (immutable) throw new IllegalStateException("Cant modify");
this.parent = parent == null ? ncfile.getRootGroup() : parent ;
}
/** Set the Group short name
* @param shortName short name.
*/
public void setName( String shortName) {
if (immutable) throw new IllegalStateException("Cant modify");
this.shortName = shortName;
}
/**
* Add new Attribute; replace old if has same name.
* @param att add this Attribute.
*/
public void addAttribute(Attribute att) {
if (immutable) throw new IllegalStateException("Cant modify");
for (int i=0; i