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

thredds.client.catalog.Catalog Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 1998-2018 University Corporation for Atmospheric Research/Unidata
 * See LICENSE for license information.
 */
package thredds.client.catalog;

import org.jdom2.Namespace;
import thredds.client.catalog.builder.AccessBuilder;
import thredds.client.catalog.builder.CatalogBuilder;
import thredds.client.catalog.builder.CatalogRefBuilder;
import thredds.client.catalog.builder.DatasetBuilder;
import ucar.nc2.time.CalendarDate;
import ucar.nc2.util.URLnaming;
import javax.annotation.concurrent.Immutable;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * A Client Catalog
 *
 * @author caron
 * @since 1/7/2015
 */
@Immutable
public class Catalog extends DatasetNode {
  public static final String CATALOG_NAMESPACE_10 = "http://www.unidata.ucar.edu/namespaces/thredds/InvCatalog/v1.0";
  public static final Namespace defNS = Namespace.getNamespace(CATALOG_NAMESPACE_10);
  public static final String NJ22_NAMESPACE = "http://www.unidata.ucar.edu/namespaces/netcdf/ncml-2.2";
  // The following line (and related code) handles incorrect HTTPS variant of NcML Namespace URI.
  // NOTE: The HTTPS variant is incorrect (according to recent, Oct 2021, decision).
  // The variant probably appeared when Unidata servers started requiring HTTPS.
  // NcML is only place HTTPS namespace variants are handled (and tested in testReadHttps.xml).
  // Should probably just be dropped.
  public static final String NJ22_NAMESPACE_HTTPS = "https://www.unidata.ucar.edu/namespaces/netcdf/ncml-2.2";
  public static final Namespace ncmlNS = Namespace.getNamespace("ncml", NJ22_NAMESPACE);
  public static final Namespace ncmlNSHttps = Namespace.getNamespace("ncml", NJ22_NAMESPACE_HTTPS);
  public static final String XLINK_NAMESPACE = "http://www.w3.org/1999/xlink";
  public static final Namespace xlinkNS = Namespace.getNamespace("xlink", XLINK_NAMESPACE);
  public static final Namespace xsiNS = Namespace.getNamespace("xsi", "http://www.w3.org/2001/XMLSchema-instance");

  // all of these are catalog only
  // public static final String CatalogScan = "CatalogScan"; // List
  public static final String DatasetHash = "DatasetHash"; // Map
  public static final String DatasetRoots = "DatasetRoots"; // List
  public static final String Expires = "Expires"; // CalendarDate
  public static final String Services = "Services"; // List (LOOK what about using Map ?)
  public static final String Version = "Version"; // String

  //////////////////////////////////////////////////////////////////////////////////////////
  private final URI baseURI; // LOOK its possible we never want to use this. perhaps "location" instead ??

  public Catalog(URI baseURI, String name, Map flds, List datasets) {
    super(null, name, flds, datasets);
    this.baseURI = baseURI;

    Map datasetMap = new HashMap<>();
    addDatasetsToHash(getDatasetsLocal(), datasetMap);
    if (!datasetMap.isEmpty())
      flds.put(Catalog.DatasetHash, datasetMap);
  }

  private void addDatasetsToHash(List datasets, Map datasetMap) {
    if (datasets == null)
      return;
    for (Dataset ds : datasets) {
      String id = ds.getIdOrPath();
      if (id != null)
        datasetMap.put(id, ds);
      if (ds instanceof CatalogRef)
        continue; // dont recurse into catrefs
      addDatasetsToHash(ds.getDatasetsLocal(), datasetMap);
    }
  }

  public URI getBaseURI() {
    return baseURI;
  }

  public CalendarDate getExpires() {
    return (CalendarDate) flds.get(Catalog.Expires);
  }

  public String getVersion() {
    return (String) flds.get(Catalog.Version);
  }

  public List getServices() {
    List services = (List) flds.get(Catalog.Services);
    return services == null ? new ArrayList<>(0) : services;
  }

  public boolean hasService(String name) {
    for (Service s : getServices())
      if (s.getName().equalsIgnoreCase(name))
        return true;
    return false;
  }

  public Service findService(String serviceName) {
    if (serviceName == null)
      return null;
    List services = (List) flds.get(Catalog.Services);
    return findService(services, serviceName);
  }

  public Service findService(ServiceType type) {
    if (type == null)
      return null;
    List services = (List) flds.get(Catalog.Services);
    return findService(services, type);
  }

  private Service findService(List services, String want) {
    if (services == null)
      return null;
    for (Service s : services) {
      if (s.getName().equals(want))
        return s;
    }
    for (Service s : services) {
      Service result = findService(s.getNestedServices(), want);
      if (result != null)
        return result;
    }
    return null;
  }

  private Service findService(List services, ServiceType type) {
    if (services == null)
      return null;
    for (Service s : services) {
      if (s.getType() == type)
        return s;
    }
    for (Service s : services) {
      Service result = findService(s.getNestedServices(), type);
      if (result != null)
        return result;
    }
    return null;
  }

  public List getProperties() {
    List properties = (List) flds.get(Dataset.Properties);
    return properties == null ? new ArrayList<>(0) : properties;
  }

  public Dataset findDatasetByID(String id) {
    Map datasetMap = (Map) flds.get(Catalog.DatasetHash);
    return datasetMap == null ? null : datasetMap.get(id);
  }

  // get all datasets contained directly in this catalog
  public Iterable getAllDatasets() {
    List all = new ArrayList<>();
    addAll(this, all);
    return all;
  }

  private void addAll(DatasetNode node, List all) {
    all.addAll(node.getDatasetsLocal());
    for (DatasetNode nested : node.getDatasetsLocal())
      addAll(nested, all);
  }

  /**
   * Resolve relative URIs, using the catalog's base URI. If the uriString is not relative, then
   * no resolution is done. This also allows baseURI to be a file: scheme.
   *
   * @param uriString any url, relative or absolute
   * @return resolved url string, or null on error
   * @throws java.net.URISyntaxException if uriString violates RFC 2396
   */
  public URI resolveUri(String uriString) throws URISyntaxException {
    if (baseURI == null)
      return new URI(uriString);
    String resolved = URLnaming.resolve(baseURI.toString(), uriString);
    return new URI(resolved);
  }

  // look is this different than URLnaming ??
  public static URI resolveUri(URI baseURI, String uriString) throws URISyntaxException {
    URI want = new URI(uriString);
    if ((baseURI == null) || want.isAbsolute())
      return want;

    // gotta deal with file ourself
    String scheme = baseURI.getScheme();
    if ("file".equals(scheme)) {
      String baseString = baseURI.toString();
      if ((!uriString.isEmpty()) && (uriString.charAt(0) == '#'))
        return new URI(baseString + uriString);
      int pos = baseString.lastIndexOf('/');
      if (pos > 0) {
        String r = baseString.substring(0, pos + 1) + uriString;
        return new URI(r);
      }
    }

    // otherwise let the URI class resolve it
    return baseURI.resolve(want);
  }

  public String getUriString() {
    URI baseURI = getBaseURI();
    return baseURI == null ? null : baseURI.toString();
  }

  //////////////////////////////////////////////////////////////////////////////////
  // from DeepCopyUtils, for subsetting

  public Catalog subsetCatalogOnDataset(Dataset dataset) {
    if (dataset == null)
      throw new IllegalArgumentException("Dataset may not be null.");
    if (dataset.getParentCatalog() != this)
      throw new IllegalArgumentException("Catalog must contain the dataset.");

    CatalogBuilder builder = new CatalogBuilder();

    URI docBaseUri = formDocBaseUriForSubsetCatalog(dataset);
    builder.setBaseURI(docBaseUri);
    builder.setName(dataset.getName());

    List neededServices = new ArrayList<>();
    DatasetBuilder topDs = copyDataset(null, dataset, neededServices, true); // LOOK, cant set catalog as datasetNode
                                                                             // parent

    for (Service s : neededServices)
      builder.addService(s);

    builder.addDataset(topDs);

    return builder.makeCatalog();
  }

  private DatasetBuilder copyDataset(DatasetBuilder parent, Dataset dataset, List neededServices,
      boolean copyInherited) {

    neededServices.add(dataset.getServiceDefault());

    DatasetBuilder result;

    if (dataset instanceof CatalogRef) {
      CatalogRef catRef = (CatalogRef) dataset;
      CatalogRefBuilder catBuilder = new CatalogRefBuilder(parent);
      catBuilder.setHref(catRef.getXlinkHref());
      catBuilder.setTitle(catRef.getName());
      result = catBuilder;

    } else {
      result = new DatasetBuilder(parent);

      List access = dataset.getLocalFieldAsList(Dataset.Access); // dont expand
      for (Access curAccess : access) {
        result.addAccess(copyAccess(result, curAccess, neededServices));
      }

      List datasets = dataset.getLocalFieldAsList(Dataset.Datasets); // dont expand
      for (Dataset currDs : datasets) {
        result.addDataset(copyDataset(result, currDs, neededServices, copyInherited));
      }
    }

    result.setName(dataset.getName());
    result.transferMetadata(dataset, copyInherited); // make a copy of all local metadata
    return result;
  }

  private AccessBuilder copyAccess(DatasetBuilder parent, Access access, List neededServices) {
    neededServices.add(access.getService()); // LOOK may get dups
    return new AccessBuilder(parent, access.getUrlPath(), access.getService(), access.getDataFormatName(),
        access.getDataSize());
  }

  private URI formDocBaseUriForSubsetCatalog(Dataset dataset) {
    String catDocBaseUri = getUriString();
    String subsetDocBaseUriString =
        catDocBaseUri + "/" + (dataset.getID() != null ? dataset.getID() : dataset.getName());
    try {
      return new URI(subsetDocBaseUriString);
    } catch (URISyntaxException e) { // This shouldn't happen. But just in case ...
      throw new IllegalStateException("Bad document Base URI for new catalog [" + catDocBaseUri + "/"
          + (dataset.getID() != null ? dataset.getID() : dataset.getName()) + "].", e);
    }
  }



}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy