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

ucar.nc2.iosp.gempak.GempakSoundingIOSP 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 (c) 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.iosp.gempak;


import ucar.ma2.*;

import ucar.nc2.*;
import ucar.nc2.constants.AxisType;
import ucar.nc2.constants.CF;
import ucar.nc2.constants._Coordinate;

import ucar.nc2.iosp.IOServiceProvider;
import ucar.nc2.util.CancelTask;

import ucar.unidata.io.RandomAccessFile;

import visad.util.Trace;

import java.io.IOException;

import java.nio.ByteBuffer;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;


/**
 * An IOSP for Gempak Sounding (SN) data.
 *
 * @author Unidata Java Development Team
 */
public class GempakSoundingIOSP extends GempakStationFileIOSP {

    /** static for shared dimension of length 4 */
    protected final static Dimension DIM_MAXMERGELEVELS =
        new Dimension("maxMergeLevels", 50, true);

    /**
     * Make the station reader for this type
     *
     * @return a GempakSoundingFileReader
     */
    protected AbstractGempakStationFileReader makeStationReader() {
        return new GempakSoundingFileReader();
    }

    /**
     * Is this a valid file?
     *
     * @param raf  RandomAccessFile to check
     *
     * @return true if a valid Gempak grid file
     *
     * @throws IOException  problem reading file
     */
    public boolean isValidFile(RandomAccessFile raf) throws IOException {
        if ( !super.isValidFile(raf)) {
            return false;
        }
        // TODO:  handle other types of surface files 
        return gemreader.getFileSubType()
            .equals(GempakSoundingFileReader.MERGED) || gemreader
            .getFileSubType().equals(GempakSoundingFileReader.UNMERGED);
    }

    /**
     * Get the file type id
     *
     * @return the file type id
     */
    public String getFileTypeId() {
        return "GempakSounding";
    }

    /**
     * Get the file type description
     *
     * @return the file type description
     */
    public String getFileTypeDescription() {
        return "GEMPAK Sounding Obs Data";
    }

    /**
     * Get the CF feature type
     * @return the feature type
     */
    public String getCFFeatureType() {
        return CF.FeatureType.timeSeriesProfile.toString();
    }

    /**
     * Read the data for the variable
     * @param v2  Variable to read
     * @param section   section infomation
     * @return Array of data
     *
     * @throws IOException problem reading from file
     * @throws InvalidRangeException  invalid Range
     */
    public Array readData(Variable v2, Section section)
            throws IOException, InvalidRangeException {
        if (gemreader == null) {
            return null;
        }
        //long start = System.currentTimeMillis();
        //System.out.println("looking for " + v2);
        //System.out.println("Section = " + section);
        //Trace.call1("GEMPAKSIOSP: readData");
        Array array = readSoundingData(
                          v2, section,
                          gemreader.getFileSubType().equals(
                              GempakSoundingFileReader.MERGED));
        //long took = System.currentTimeMillis() - start;
        //System.out.println("  read data took=" + took + " msec ");
        //Trace.call2("GEMPAKSIOSP: readData");
        return array;
    }

    /**
     * Read in the data for the variable.  In this case, it should be
     * a Structure.  The section should be rank 2 (station, time).
     *
     * @param v2  variable to read
     * @param section  section of the variable
     * @param isMerged flag for merged data or not
     *
     * @return the array of data
     *
     * @throws IOException  problem reading the file
     */
    private Array readSoundingData(Variable v2, Section section,
                                   boolean isMerged)
            throws IOException {

        Array array = null;
        if (v2 instanceof Structure) {

            Range                         stationRange = section.getRange(0);
            Range                         timeRange    = section.getRange(1);
            int size = stationRange.length() * timeRange.length();

            Structure                     pdata        = (Structure) v2;
            StructureMembers members = pdata.makeStructureMembers();
            ArrayStructureBB.setOffsets(members);
            ArrayStructureBB abb = new ArrayStructureBB(members,
                                       new int[] { size });
            ByteBuffer buf = abb.getByteBuffer();

            //Trace.call1("GEMPAKSIOSP: readMergedData" , section.toString());
            for (int y = stationRange.first(); y <= stationRange.last();
                    y += stationRange.stride()) {
                for (int x = timeRange.first(); x <= timeRange.last();
                        x += timeRange.stride()) {
                    List parts = (isMerged)
                                         ? ((GempakSoundingFileReader) gemreader)
                                             .getMergedParts()
                                         : ((GempakSoundingFileReader) gemreader)
                                             .getUnmergedParts();
                    boolean allMissing = true;
                    for (String part : parts) {

                        List params =
                            gemreader.getParameters(part);
                        GempakFileReader.RData vals = gemreader.DM_RDTR(x
                                                          + 1, y + 1, part);
                        ArraySequence aseq = null;
                        Sequence seq = (Sequence) pdata.findVariable(part);
                        if (vals == null) {
                            aseq = makeEmptySequence(seq);
                        } else {
                            allMissing = false;
                            aseq = makeArraySequence(seq, params, vals.data);
                        }
                        int index = abb.addObjectToHeap(aseq);
                        buf.putInt(index);
                    }
                    buf.put((byte) (allMissing
                                    ? 1
                                    : 0));
                }
            }
            array = abb;
            Trace.call2("GEMPAKSIOSP: readMergedData");
        }
        return array;
    }

    /**
     * Create an empty ArraySequence for missing data
     *
     * @param seq  the Sequence variable
     *
     * @return the empty sequence
     */
    private ArraySequence makeEmptySequence(Sequence seq) {
        StructureMembers members = seq.makeStructureMembers();
        return new ArraySequence(members, new EmptyStructureDataIterator(),
                                 -1);
    }

    /**
     * Create an ArraySequence to hold the data
     *
     * @param seq    the Sequence variable
     * @param params the list of all GempakParameters possible in that sequence
     * @param values the values that were read
     *
     * @return the ArraySequence
     */
    private ArraySequence makeArraySequence(Sequence seq,
                                            List params,
                                            float[] values) {

        if (values == null) {
            return makeEmptySequence(seq);
        }
        int numLevels = values.length / params.size();
        StructureMembers              members = seq.makeStructureMembers();
        int offset = ArrayStructureBB.setOffsets(members);

        int        size  = offset * numLevels;
        byte[]     bytes = new byte[size];
        ByteBuffer buf   = ByteBuffer.wrap(bytes);
        ArrayStructureBB abb = new ArrayStructureBB(members,
                                   new int[] { numLevels }, buf, 0);
        int var = 0;
        for (int i = 0; i < numLevels; i++) {
            for (GempakParameter param : params) {
                if (members.findMember(param.getName()) != null) {
                    buf.putFloat(values[var]);
                }
                var++;
            }
        }
        return new ArraySequence(members,
                                 new SequenceIterator(numLevels, abb),
                                 numLevels);
    }

    /**
     * Test this.
     *
     * @param args file name
     *
     * @throws IOException  problem reading the file
     */
    public static void main(String[] args) throws IOException {
        IOServiceProvider mciosp = new GempakSoundingIOSP();
        RandomAccessFile  rf     = new RandomAccessFile(args[0], "r", 2048);
        NetcdfFile ncfile = new MakeNetcdfFile(mciosp, rf, args[0], null);
        if (args.length > 1) {
            ucar.nc2.FileWriter.writeToFile(ncfile, args[1]);
        } else {
            System.out.println(ncfile);
        }
    }

    /**
     *   TODO:  generalize this
     *   static class for testing
     */
    protected static class MakeNetcdfFile extends NetcdfFile {

        /**
         * Ctor
         *
         * @param spi IOServiceProvider
         * @param raf RandomAccessFile
         * @param location   location of file?
         * @param cancelTask CancelTask
         *
         * @throws IOException problem opening the file
         */
        MakeNetcdfFile(IOServiceProvider spi, RandomAccessFile raf,
                       String location, CancelTask cancelTask)
                throws IOException {
            super(spi, raf, location, cancelTask);
        }
    }

    /**
     * Build the netCDF file
     *
     * @throws IOException   problem reading the file
     */
    protected void fillNCFile() throws IOException {
        String fileType = gemreader.getFileSubType();
        buildFile(fileType.equals(GempakSoundingFileReader.MERGED));
    }

    /**
     * Build a standard station structure
     *
     * @param isMerged  true if this is a merged file
     */
    private void buildFile(boolean isMerged) {

        // Build station list
        List stations = gemreader.getStations();
        Dimension station = new Dimension("station", stations.size(), true);
        ncfile.addDimension(null, station);
        ncfile.addDimension(null, DIM_LEN8);
        ncfile.addDimension(null, DIM_LEN4);
        ncfile.addDimension(null, DIM_LEN2);
        List stationVars = makeStationVars(stations, station);
        // loop through and add to ncfile
        for (Variable stnVar : stationVars) {
            ncfile.addVariable(null, stnVar);
        }


        // Build variable list (var(station,time))
        // time
        List timeList = gemreader.getDates();
        int        numTimes = timeList.size();
        Dimension  times    = new Dimension(TIME_VAR, numTimes, true);
        ncfile.addDimension(null, times);
        Array varArray = null;
        Variable timeVar = new Variable(ncfile, null, null, TIME_VAR,
                                        DataType.DOUBLE, TIME_VAR);
        timeVar.addAttribute(
            new Attribute("units", "seconds since 1970-01-01 00:00:00"));
        timeVar.addAttribute(new Attribute("long_name", TIME_VAR));
        varArray = new ArrayDouble.D1(numTimes);
        int i = 0;
        for (Date date : timeList) {
            ((ArrayDouble.D1) varArray).set(i, date.getTime() / 1000.d);
            i++;
        }
        timeVar.setCachedData(varArray, false);
        ncfile.addVariable(null, timeVar);


        // build the data structure
        List stationTime = new ArrayList();
        stationTime.add(station);
        stationTime.add(times);
        String structName = (isMerged)
                            ? GempakSoundingFileReader.MERGED
                            : GempakSoundingFileReader.UNMERGED;
        structName = structName + "Sounding";
        Structure sVar = new Structure(ncfile, null, null, structName);
        sVar.setDimensions(stationTime);
        sVar.addAttribute(new Attribute("coordinates",
                                        "time SLAT SLON SELV"));
        List sequenceNames;
        if (isMerged) {
            sequenceNames = new ArrayList();
            sequenceNames.add(GempakSoundingFileReader.SNDT);
        } else {
            sequenceNames =
                ((GempakSoundingFileReader) gemreader).getUnmergedParts();
        }
        for (String seqName : sequenceNames) {
            Sequence paramData = makeSequence(sVar, seqName, false);
            if (paramData == null) {
                continue;
            }
            sVar.addMemberVariable(paramData);
        }
        sVar.addMemberVariable(makeMissingVariable());
        ncfile.addAttribute(
            null,
            new Attribute(
                "CF:featureType", CF.FeatureType.timeSeriesProfile.toString()));
        ncfile.addVariable(null, sVar);
    }


    /**
     * Make a Sequence for the part
     *
     * @param parent  parent structure
     * @param partName   partname
     * @param includeMissing  true to include the missing variable
     *
     * @return  a Structure
     */
    protected Sequence makeSequence(Structure parent, String partName,
                                    boolean includeMissing) {
        List params = gemreader.getParameters(partName);
        if (params == null) {
            return null;
        }
        Sequence sVar = new Sequence(ncfile, null, parent, partName);
        sVar.setDimensions("");
        for (GempakParameter param : params) {
            Variable v = makeParamVariable(param, null);
            addVerticalCoordAttribute(v);
            sVar.addMemberVariable(v);
        }
        if (includeMissing) {
            sVar.addMemberVariable(makeMissingVariable());
        }
        return sVar;
    }

    /**
     * Add the vertical coordinate variables if necessary
     *
     * @param v  the variable
     */
    private void addVerticalCoordAttribute(Variable v) {

        GempakSoundingFileReader gsfr = (GempakSoundingFileReader) gemreader;
        int                      vertType = gsfr.getVerticalCoordinate();
        String                   pName    = v.getName();
        if (gemreader.getFileSubType().equals(
                GempakSoundingFileReader.MERGED)) {
            if ((vertType == gsfr.PRES_COORD) && pName.equals("PRES")) {
                v.addAttribute(new Attribute(_Coordinate.AxisType,
                                             AxisType.Pressure.name()));
            } else if ((vertType == gsfr.HGHT_COORD)
                       && (pName.equals("HGHT") || pName.equals("MHGT")
                           || pName.equals("DHGT"))) {
                v.addAttribute(new Attribute(_Coordinate.AxisType,
                                             AxisType.Height.name()));
            }
        } else if (pName.equals("PRES")) {
            v.addAttribute(new Attribute(_Coordinate.AxisType,
                                         AxisType.Pressure.name()));
        }
    }

    /**
     * An empty sequence iterator
     *
     * @author Unidata Development Team
     */
    class EmptyStructureDataIterator implements StructureDataIterator {

        /**
         * Do we have more?
         *
         * @return false
         *
         * @throws IOException problem with read
         */
        @Override public boolean hasNext() throws IOException {
            return false;
        }

        /**
         * Get the next data
         *
         * @return null
         *
         * @throws IOException problem with read
         */
        @Override public StructureData next() throws IOException {
            return null;
        }

        /**
         * Set the buffer size
         *
         * @param bytes the buffer size
         */
        @Override public void setBufferSize(int bytes) {}

        /**
         * Reset the iterator
         *
         * @return this
         */
        @Override public StructureDataIterator reset() {
            return this;
        }

        /**
         * Get the current record number
         *
         * @return -1
         */
        @Override public int getCurrentRecno() {
            return -1;
        }
    }

    /**
     * A data iterator for a sequence
     *
     * @author Unidata Development Team
     */
    private class SequenceIterator implements StructureDataIterator {

        /** the number of records */
        private int count;

        /** the backing structure */
        private ArrayStructure abb;

        /** the iterator */
        private StructureDataIterator siter;

        /**
         * Create a new iterator for the ArrayStructure
         *
         * @param count the number of records
         * @param abb   the backing store
         */
        SequenceIterator(int count, ArrayStructure abb) {
            this.count = count;
            this.abb   = abb;
        }

        /**
         * Do we have more?
         *
         * @return true if we do
         *
         * @throws IOException problem with read
         */
        @Override public boolean hasNext() throws IOException {
            if (siter == null) {
                siter = abb.getStructureDataIterator();
            }
            return siter.hasNext();
        }

        /**
         * Get the next StructureData
         *
         * @return the next StructureData
         *
         * @throws IOException problem with read
         */
        @Override public StructureData next() throws IOException {
            return siter.next();
        }

        /**
         * Set the buffer size
         *
         * @param bytes the buffer size
         */
        @Override public void setBufferSize(int bytes) {
            siter.setBufferSize(bytes);
        }

        /**
         * Reset the iterator
         *
         * @return  this
         */
        @Override public StructureDataIterator reset() {
            siter = null;
            return this;
        }

        /**
         * Get the current record number
         *
         * @return the current record number
         */
        @Override public int getCurrentRecno() {
            return siter.getCurrentRecno();
        }

    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy