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

thredds.catalog2.xml.parser.stax.DatasetNodeElementParserHelper Maven / Gradle / Ivy

Go to download

The NetCDF-Java Library is a Java interface to NetCDF files, as well as to many other types of scientific data formats.

There is a newer version: 4.3.22
Show newest version
/*
 * 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 thredds.catalog2.xml.parser.stax;

import thredds.catalog2.builder.*;
import thredds.catalog2.builder.util.ThreddsMetadataBuilderUtils;
import thredds.catalog2.builder.util.MetadataBuilderUtils;
import thredds.catalog2.xml.parser.ThreddsXmlParserException;
import thredds.catalog2.xml.names.DatasetNodeElementNames;
import thredds.catalog2.xml.names.DatasetElementNames;

import javax.xml.stream.events.StartElement;
import javax.xml.stream.events.Attribute;
import javax.xml.stream.XMLEventReader;
import java.util.List;
import java.util.ArrayList;
import java.util.Collections;

/**
 * Helper class for DatasetNodeElementParser that deals with metadata and other
 * information inherited from ancestor DatasetNodes.
 *
 * @author edavis
 * @since 4.0
 */
class DatasetNodeElementParserHelper
{
  private org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger( getClass() );

  private final ThreddsBuilderFactory builderFactory;

  private final PropertyElementParser.Factory propertyElemParserFactory;
  private final DatasetElementParser.Factory datasetElemParserFactory;
  private final CatalogRefElementParser.Factory catRefElemParserFactory;
  private final MetadataElementParser.Factory metadataElemParserFactory;
  private final ThreddsMetadataElementParser.Factory threddsMetadataElemParserFactory;

  private final DatasetNodeBuilder datasetNodeBuilder;

  private String defaultServiceNameInheritedFromAncestors;
  private String defaultServiceNameSpecifiedInSelf;
  private String defaultServiceNameToBeInheritedByDescendants;

  private String idAuthorityInheritedFromAncestors;
  private String idAuthoritySpecifiedInSelf;
  private String idAuthorityToBeInheritedByDescendants;

  // All metadata applicable to this dataset.
  private List metadataForThisDataset;
  // All metadata inherited by descendant datasets.
  private List metadataInheritedByDescendants;

  private ThreddsMetadataElementParser threddsMetadataElementParser;

  private SplitMetadata finalSplitMetadata;

  DatasetNodeElementParserHelper( DatasetNodeElementParserHelper parentDatasetNodeElementParserHelper,
                                  DatasetNodeBuilder datasetNodeBuilder,
                                  ThreddsBuilderFactory builderFactory )
  {
    this.datasetNodeBuilder = datasetNodeBuilder;
    this.builderFactory = builderFactory;

    this.propertyElemParserFactory = new PropertyElementParser.Factory();
    this.datasetElemParserFactory = new DatasetElementParser.Factory();
    this.catRefElemParserFactory = new CatalogRefElementParser.Factory();
    this.metadataElemParserFactory = new MetadataElementParser.Factory();
    this.threddsMetadataElemParserFactory = new ThreddsMetadataElementParser.Factory();

    if ( parentDatasetNodeElementParserHelper != null)
    {
      List metadataInheritedFromAncestors
              = parentDatasetNodeElementParserHelper.getMetadataInheritedByDescendants();

      if ( metadataInheritedFromAncestors != null
           && ! metadataInheritedFromAncestors.isEmpty() )
      {
        // Add to list of metadata inherited by descendant datasets.
        this.metadataInheritedByDescendants = new ArrayList();
        this.metadataInheritedByDescendants.addAll( metadataInheritedFromAncestors);

        // Add to list of metadata applicable to this dataset.
        this.metadataForThisDataset = new ArrayList();
        this.metadataForThisDataset.addAll( metadataInheritedFromAncestors);
      }

      this.defaultServiceNameInheritedFromAncestors
              = parentDatasetNodeElementParserHelper.getDefaultServiceNameToBeInheritedByDescendants();
      this.idAuthorityInheritedFromAncestors
              = parentDatasetNodeElementParserHelper.getIdAuthorityToBeInheritedByDescendants();
    }
  }

  String getIdAuthorityInheritedFromAncestors() {
    return this.idAuthorityInheritedFromAncestors;
  }

  void setIdAuthorityInheritedFromAncestors( String idAuthorityInheritedFromAncestors ) {
    this.idAuthorityInheritedFromAncestors = idAuthorityInheritedFromAncestors;
  }

  String getIdAuthoritySpecifiedInSelf() {
    return this.idAuthoritySpecifiedInSelf;
  }

  void setIdAuthoritySpecifiedInSelf( String idAuthoritySpecifiedInSelf ) {
    this.idAuthoritySpecifiedInSelf = idAuthoritySpecifiedInSelf;
  }

  String getIdAuthorityToBeInheritedByDescendants() {
    return this.idAuthorityToBeInheritedByDescendants;
  }

  void setIdAuthorityToBeInheritedByDescendants( String idAuthorityToBeInheritedByDescendants ) {
    this.idAuthorityToBeInheritedByDescendants = idAuthorityToBeInheritedByDescendants;
  }

  String getDefaultServiceNameInheritedFromAncestors() {
    return this.defaultServiceNameInheritedFromAncestors;
  }

  void setDefaultServiceNameInheritedFromAncestors( String defaultServiceNameInheritedFromAncestors) {
    this.defaultServiceNameInheritedFromAncestors = defaultServiceNameInheritedFromAncestors;
  }

  String getDefaultServiceNameSpecifiedInSelf() {
    return this.defaultServiceNameSpecifiedInSelf;
  }

  void setDefaultServiceNameSpecifiedInSelf( String defaultServiceNameSpecifiedInSelf ) {
    this.defaultServiceNameSpecifiedInSelf = defaultServiceNameSpecifiedInSelf;
  }

  String getDefaultServiceNameToBeInheritedByDescendants() {
    return this.defaultServiceNameToBeInheritedByDescendants;
  }

  void setDefaultServiceNameToBeInheritedByDescendants( String defaultServiceNameToBeInheritedByDescendants ) {
    this.defaultServiceNameToBeInheritedByDescendants = defaultServiceNameToBeInheritedByDescendants;
  }

  public List getMetadataForThisDataset() {
    if ( this.metadataForThisDataset == null )
      return Collections.emptyList();
    return Collections.unmodifiableList( this.metadataForThisDataset);
  }

  List getMetadataInheritedByDescendants() {
    if ( this.metadataInheritedByDescendants == null )
      return Collections.emptyList();
    return Collections.unmodifiableList( this.metadataInheritedByDescendants);
  }

  void parseStartElementNameAttribute( StartElement startElement )
  {
    Attribute att = startElement.getAttributeByName( DatasetElementNames.DatasetElement_Name );
    if ( att != null )
      this.datasetNodeBuilder.setName( att.getValue() );
  }

  void parseStartElementIdAttribute( StartElement startElement )
  {
    Attribute att = startElement.getAttributeByName( DatasetNodeElementNames.DatasetNodeElement_Id );
    if ( att != null )
      this.datasetNodeBuilder.setId( att.getValue() );
  }

  void parseStartElementIdAuthorityAttribute( StartElement startElement )
  {
    Attribute att = startElement.getAttributeByName( DatasetNodeElementNames.DatasetNodeElement_Authority );
    if ( att != null )
      this.setIdAuthoritySpecifiedInSelf( att.getValue() );
  }

  /**
   * Handle child start elements that are used in "dataset" elements and "catalogRef" elements.
   *
   * @param startElement the StartElement to handle.
   * @param reader the XMLEventReader with the given StartElement as the next event.
   * @param dsNodeBuilder the DatasetNodeBuilder this is helping to build
   * @return true if the given StartElement was recognized and handled.
   * @throws ThreddsXmlParserException if there was a problem parsing the XML.
   */
  boolean handleBasicChildStartElement( StartElement startElement,
                                        XMLEventReader reader,
                                        DatasetNodeBuilder dsNodeBuilder )
          throws ThreddsXmlParserException
  {
    if ( this.propertyElemParserFactory.isEventMyStartElement( startElement ))
    {
      PropertyElementParser parser = this.propertyElemParserFactory.getNewParser( reader,
                                                                                  this.builderFactory,
                                                                                  dsNodeBuilder );
      parser.parse();
      return true;
    }
    else if ( this.metadataElemParserFactory.isEventMyStartElement( startElement ))
    {
      MetadataElementParser parser = this.metadataElemParserFactory.getNewParser( reader, this.builderFactory,
                                                                                  dsNodeBuilder, this );
      parser.parse();

      if ( this.metadataForThisDataset == null )
        this.metadataForThisDataset = new ArrayList();
      this.metadataForThisDataset.add( parser);

      if ( parser.doesMetadataElementGetInherited())
      {
        if ( this.metadataInheritedByDescendants == null )
          this.metadataInheritedByDescendants = new ArrayList();
        this.metadataInheritedByDescendants.add( parser );
      }

      return true;
    }
    else if ( this.threddsMetadataElemParserFactory.isEventMyStartElement( startElement ))
    {
      if ( this.threddsMetadataElementParser == null )
        this.threddsMetadataElementParser = this.threddsMetadataElemParserFactory.getNewParser( reader,
                                                                                                this.builderFactory,
                                                                                                dsNodeBuilder,
                                                                                                this, false );
      this.threddsMetadataElementParser.parse();
      return true;
    }
    else
      return false;
  }

  /**
   * Handle those elements that are only contained in a "dataset" element.
   *
   * @param startElement the StartElement to handle.
   * @param reader the XMLEventReader with the given StartElement as the next event.
   * @param dsNodeBuilder the DatasetNodeBuilder this is helping to build
   * @return true if the given StartElement was recognized and handled.
   * @throws ThreddsXmlParserException if there was a problem parsing the XML.
   */
  boolean handleCollectionChildStartElement( StartElement startElement,
                                             XMLEventReader reader,
                                             DatasetNodeBuilder dsNodeBuilder )
          throws ThreddsXmlParserException
  {
    if ( this.datasetElemParserFactory.isEventMyStartElement( startElement ))
    {
      DatasetElementParser parser = this.datasetElemParserFactory.getNewParser( reader, this.builderFactory,
                                                                                dsNodeBuilder, this );
      parser.parse();
      return true;
    }
    else if ( this.catRefElemParserFactory.isEventMyStartElement( startElement ))
    {
      CatalogRefElementParser parser = this.catRefElemParserFactory.getNewParser( reader, this.builderFactory,
                                                                                  dsNodeBuilder, this );
      parser.parse();
      return true;
    }
    else
      return false;
  }

  void postProcessingAfterEndElement()
          throws ThreddsXmlParserException
  {
    if ( this.getDefaultServiceNameToBeInheritedByDescendants() == null )
      this.setDefaultServiceNameToBeInheritedByDescendants( this.getDefaultServiceNameInheritedFromAncestors() );

    if ( this.getIdAuthorityToBeInheritedByDescendants() == null )
      this.setIdAuthorityToBeInheritedByDescendants( this.getIdAuthorityInheritedFromAncestors() );

    this.datasetNodeBuilder.setIdAuthority( this.getIdAuthoritySpecifiedInSelf() != null
                                            ? this.getIdAuthoritySpecifiedInSelf()
                                            : this.getIdAuthorityInheritedFromAncestors() );

    if ( this.threddsMetadataElementParser != null )
      this.threddsMetadataElementParser.postProcessingAfterEndElement();

    this.finalSplitMetadata = new SplitMetadata( this.metadataForThisDataset);
  }

  /**
   * Add to the target DatasetNodeBuilder a ThreddsMetadataBuilder that contains
   * all the ThreddsMetadata for the target. The ThreddsMetadataBuilder merges
   * together:
   * 1) the local ThreddsMetadata that was not wrapped in a metadata element,
   * 2) all local ThreddsMetadata that was wrapped in a metadata element, and
   * 3) all inherited ThreddsMetadata.
   *
   * @param dsNodeBuilder the target DatasetNodeBuilder.
   */
  void addFinalThreddsMetadataToDatasetNodeBuilder( DatasetNodeBuilder dsNodeBuilder)
  {
    ThreddsMetadataBuilder unwrappedThreddsMetadataBuilder = null;
    boolean isUnwrappedEmpty = true;
    if ( this.threddsMetadataElementParser != null )
    {
      unwrappedThreddsMetadataBuilder = this.threddsMetadataElementParser.getSelfBuilder();
      isUnwrappedEmpty = unwrappedThreddsMetadataBuilder.isEmpty();
    }

    if ( ! isUnwrappedEmpty || ! this.finalSplitMetadata.threddsMetadata.isEmpty())
    {
      ThreddsMetadataBuilder result = dsNodeBuilder.setNewThreddsMetadataBuilder();
      if ( ! isUnwrappedEmpty )
        ThreddsMetadataBuilderUtils.copyThreddsMetadataBuilder( unwrappedThreddsMetadataBuilder, result );
      for ( MetadataElementParser mdElemParser : this.finalSplitMetadata.threddsMetadata )
        ThreddsMetadataBuilderUtils.copyThreddsMetadataBuilder( mdElemParser.getThreddsMetadataBuilder(), result );
    }
  }

  /**
   * Add to the target DatasetNodeBuilder the list of MetadataBuilders for the target
   * including all inherited and local metadata.
   *
   * @param dsNodeBuilder the target DatasetNodeBuilder.
   */
  void addFinalMetadataToDatasetNodeBuilder( DatasetNodeBuilder dsNodeBuilder )
  {
    for ( MetadataElementParser currentMetadataElemParser : this.finalSplitMetadata.nonThreddsMetadata )
    {
      MetadataBuilder newMetadataBuilder = dsNodeBuilder.addMetadata();
      MetadataBuilderUtils.copyMetadataBuilder( currentMetadataElemParser.getSelfBuilder(), newMetadataBuilder);
    }
  }

  private class SplitMetadata
  {
    final List threddsMetadata;
    final List nonThreddsMetadata;

    SplitMetadata( List metadata )
    {
      if ( metadata == null || metadata.isEmpty())
      {
        this.threddsMetadata = Collections.emptyList();
        this.nonThreddsMetadata = Collections.emptyList();
        return;
      }

      this.threddsMetadata = new ArrayList();
      this.nonThreddsMetadata = new ArrayList();

      for ( MetadataElementParser current : metadata )
      {
        if ( current.isContainsThreddsMetadata())
          this.threddsMetadata.add( current);
        else
          this.nonThreddsMetadata.add( current);
      }
    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy