ucar.nc2.internal.dataset.conv.Cosmic1Convention Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 1998-2020 John Caron and University Corporation for Atmospheric Research/Unidata
* See LICENSE for license information.
*/
package ucar.nc2.internal.dataset.conv;
import java.io.IOException;
import ucar.ma2.Array;
import ucar.ma2.ArrayDouble;
import ucar.ma2.ArrayFloat;
import ucar.ma2.DataType;
import ucar.ma2.IndexIterator;
import ucar.nc2.Attribute;
import ucar.nc2.AttributeContainer;
import ucar.nc2.Dimension;
import ucar.nc2.NetcdfFile;
import ucar.nc2.Variable;
import ucar.nc2.constants.AxisType;
import ucar.nc2.constants.CDM;
import ucar.nc2.constants._Coordinate;
import ucar.nc2.dataset.NetcdfDataset;
import ucar.nc2.dataset.VariableDS;
import ucar.nc2.dataset.VariableEnhanced;
import ucar.nc2.dataset.spi.CoordSystemBuilderFactory;
import ucar.nc2.internal.dataset.CoordSystemBuilder;
import ucar.nc2.util.CancelTask;
/** Cosmic data - version 1 and 3 */
public class Cosmic1Convention extends CoordSystemBuilder {
private static final String CONVENTION_NAME = "Cosmic1";
public static class Factory implements CoordSystemBuilderFactory {
@Override
public String getConventionName() {
return CONVENTION_NAME;
}
@Override
public boolean isMine(NetcdfFile ncfile) {
if ((null == ncfile.findDimension("MSL_alt")) && (null == ncfile.findDimension("time"))) {
return false;
}
String center = ncfile.getRootGroup().attributes().findAttributeString("center", null);
return "UCAR/CDAAC".equals(center);
}
@Override
public CoordSystemBuilder open(NetcdfDataset.Builder datasetBuilder) {
return new Cosmic1Convention(datasetBuilder);
}
}
private Cosmic1Convention(NetcdfDataset.Builder datasetBuilder) {
super(datasetBuilder);
this.conventionName = CONVENTION_NAME;
}
@Override
public void augmentDataset(CancelTask cancelTask) throws IOException {
AttributeContainer gatts = rootGroup.getAttributeContainer();
Attribute leoAtt = gatts.findAttribute("leoId");
if (leoAtt == null) {
if (!rootGroup.findVariableLocal("time").isPresent()) {
// create a time Variable.Builder - assume its linear along the vertical dimension
double start = gatts.findAttributeDouble("start_time", Double.NaN);
double stop = gatts.findAttributeDouble("stop_time", Double.NaN);
if (Double.isNaN(start) && Double.isNaN(stop)) {
double top = gatts.findAttributeDouble("toptime", Double.NaN);
double bot = gatts.findAttributeDouble("bottime", Double.NaN);
this.conventionName = "Cosmic2";
if (top > bot) {
stop = top;
start = bot;
} else {
stop = bot;
start = top;
}
}
Dimension dim = rootGroup.findDimension("MSL_alt").get();
VariableDS.Builder dimV = (VariableDS.Builder) rootGroup.findVariableLocal("MSL_alt").get();
Array dimU = dimV.orgVar.read();
int inscr = (dimU.getFloat(1) - dimU.getFloat(0)) > 0 ? 1 : 0;
int n = dim.getLength();
double incr = (stop - start) / n;
String timeUnits = "seconds since 1980-01-06 00:00:00";
VariableDS.Builder timeVar = VariableDS.builder().setName("time").setDataType(DataType.DOUBLE)
.setParentGroupBuilder(rootGroup).setDimensionsByName(dim.getShortName()).setUnits(timeUnits);
rootGroup.addVariable(timeVar);
timeVar.setUnits(timeUnits);
timeVar.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Time.toString()));
int dir = gatts.findAttributeInteger("irs", 1);
ArrayDouble.D1 data = (ArrayDouble.D1) Array.factory(DataType.DOUBLE, new int[] {n});
if (inscr == 0) {
if (dir == 1) {
for (int i = 0; i < n; i++) {
data.set(i, start + i * incr);
}
} else {
for (int i = 0; i < n; i++) {
data.set(i, stop - i * incr);
}
}
} else {
for (int i = 0; i < n; i++) {
data.set(i, stop - i * incr);
}
}
timeVar.setCachedData(data, false);
}
Variable.Builder lat =
rootGroup.findVariableLocal("Lat").orElse(rootGroup.findVariableLocal("GEO_lat").orElse(null));
lat.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Lat.toString()));
Variable.Builder lon =
rootGroup.findVariableLocal("Lon").orElse(rootGroup.findVariableLocal("GEO_lon").orElse(null));
lon.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Lon.toString()));
Variable.Builder alt = rootGroup.findVariableLocal("MSL_alt").orElse(null);
alt.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Height.toString()));
} else {
Dimension dim = rootGroup.findDimension("time").get();
int n = dim.getLength();
Variable.Builder latVar = VariableDS.builder().setName("Lat").setDataType(DataType.FLOAT)
.setParentGroupBuilder(rootGroup).setDimensionsByName(dim.getShortName()).setUnits("degree");
latVar.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Lat.toString()));
rootGroup.addVariable(latVar);
Variable.Builder lonVar = VariableDS.builder().setName("Lon").setDataType(DataType.FLOAT)
.setParentGroupBuilder(rootGroup).setDimensionsByName(dim.getShortName()).setUnits("degree");
lonVar.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Lon.toString()));
rootGroup.addVariable(lonVar);
Variable.Builder altVar = VariableDS.builder().setName("MSL_alt").setDataType(DataType.FLOAT)
.setParentGroupBuilder(rootGroup).setDimensionsByName(dim.getShortName()).setUnits("meter");
altVar.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Height.toString()));
rootGroup.addVariable(altVar);
// cal data array
ArrayFloat.D1 latData = (ArrayFloat.D1) Array.factory(DataType.FLOAT, new int[] {n});
ArrayFloat.D1 lonData = (ArrayFloat.D1) Array.factory(DataType.FLOAT, new int[] {n});
ArrayFloat.D1 altData = (ArrayFloat.D1) Array.factory(DataType.FLOAT, new int[] {n});
ArrayDouble.D1 timeData = (ArrayDouble.D1) Array.factory(DataType.DOUBLE, new int[] {n});
this.conventionName = "Cosmic3";
int iyr = gatts.findAttributeInteger("year", 2009);
int mon = gatts.findAttributeInteger("month", 0);
int iday = gatts.findAttributeInteger("day", 0);
int ihr = gatts.findAttributeInteger("hour", 0);
int min = gatts.findAttributeInteger("minute", 0);
int sec = gatts.findAttributeInteger("second", 0);
double start = gatts.findAttributeDouble("startTime", Double.NaN);
double stop = gatts.findAttributeDouble("stopTime", Double.NaN);
double incr = (stop - start) / n;
int t = 0;
// double julian = juday(mon, iday, iyr);
// cal the dtheta based pm attributes
double dtheta = gast(iyr, mon, iday, ihr, min, sec, t);
VariableDS.Builder tVar = (VariableDS.Builder) rootGroup.findVariableLocal("time").get();
String timeUnits = "seconds since 1980-01-06 00:00:00"; // dtime.getUnit().toString();
tVar.getAttributeContainer().removeAttributeIgnoreCase(CDM.VALID_RANGE);
tVar.getAttributeContainer().removeAttributeIgnoreCase(CDM.UNITS);
tVar.setUnits(timeUnits);
tVar.addAttribute(new Attribute(_Coordinate.AxisType, AxisType.Time.toString()));
VariableDS.Builder xLeoVar = (VariableDS.Builder) rootGroup.findVariableLocal("xLeo").get();
Array xLeoData = xLeoVar.orgVar.read();
VariableDS.Builder yLeoVar = (VariableDS.Builder) rootGroup.findVariableLocal("yLeo").get();
Array yLeoData = yLeoVar.orgVar.read();
VariableDS.Builder zLeoVar = (VariableDS.Builder) rootGroup.findVariableLocal("zLeo").get();
Array zLeoData = zLeoVar.orgVar.read();
double a = 6378.1370;
double b = 6356.7523142;
IndexIterator xLeoIter = xLeoData.getIndexIterator();
IndexIterator yLeoIter = yLeoData.getIndexIterator();
IndexIterator zLeoIter = zLeoData.getIndexIterator();
int i = 0;
while (xLeoIter.hasNext()) {
double[] v_inertial = new double[3];
v_inertial[0] = xLeoIter.getDoubleNext();
v_inertial[1] = yLeoIter.getDoubleNext();
v_inertial[2] = zLeoIter.getDoubleNext();
double[] uvz = new double[3];
uvz[0] = 0.0;
uvz[1] = 0.0;
uvz[2] = 1.0;
// v_ecef should be in the (approximate) ECEF frame
// double[] v_ecf = execute(v_inertial, julian);
double[] v_ecf = spin(v_inertial, uvz, -1 * dtheta);
// cal lat/lon here
// double [] llh = ECFtoLLA(v_ecf[0]*1000, v_ecf[1]*1000, v_ecf[2]*1000, a, b);
double[] llh = xyzell(a, b, v_ecf);
latData.set(i, (float) llh[0]);
lonData.set(i, (float) llh[1]);
altData.set(i, (float) llh[2]);
timeData.set(i, start + i * incr);
i++;
}
latVar.setCachedData(latData, false);
lonVar.setCachedData(lonData, false);
altVar.setCachedData(altData, false);
tVar.setCachedData(timeData, false);
}
}
protected AxisType getAxisType(NetcdfDataset ncDataset, VariableEnhanced v) {
String name = v.getShortName();
if (name.equals("time")) {
return AxisType.Time;
}
if (name.equals("Lat") || name.equals("GEO_lat")) {
return AxisType.Lat;
}
if (name.equals("Lon") || name.equals("GEO_lon")) {
return AxisType.Lon;
}
// if (name.equals("xLeo") ) return AxisType.GeoX;
// if (name.equals("yLeo") ) return AxisType.GeoY;
if (name.equals("MSL_alt")) {
return AxisType.Height;
}
return null;
}
/**
* NAME : XYZELL
*
* CALL XYZELL(A,B,XSTAT,XSTELL)
*
* PURPOSE : COMPUTATION OF ELLIPSOIDAL COORDINATES "XSTELL"
* GIVEN THE CARTESIAN COORDINATES "XSTAT"
*
* PARAMETERS :
* IN : A : SEMI-MAJOR AXIS OF THE REFERENCE R*8
* ELLIPSOID IN METERS
* B : SEMI-MINOR AXIS OF THE REFERENCE R*8
* ELLIPSOID IN METERS
* DXELL(3): TRANSLATION COMPONENTS FROM THE R*8
* ORIGIN OF THE CART. COORD. SYSTEM
* (X,Y,Z) TO THE CENTER OF THE REF.
* ELLIPSOID (IN METRES)
* SCELL : SCALE FACTOR BETWEEN REF. ELLIPSOID R*8
* AND WGS-84
* XSTAT(3): CARTESIAN COORDINATES (M) R*8
* OUT : XSTELL(3): ELLIPSOIDAL COORDINATES R*8
* XSTELL(1): ELL. LATITUDE (RADIAN)
* XSTELL(2): ELL. LONGITUDE (RADIAN)
* XSTELL(3): ELL. HEIGHT (M)
*
* SR CALLED : DMLMTV
*
* REMARKS : ---
*
* AUTHOR : M. ROTHACHER
*
* VERSION : 3.4 (JAN 93)
*
* CREATED : 87/11/03 12:32 LAST MODIFIED : 88/11/21 17:36
*
* COPYRIGHT : ASTRONOMICAL INSTITUTE
* 1987 UNIVERSITY OF BERNE
* SWITZERLAND
*/
private double[] xyzell(double a, double b, double[] xstat) {
double[] xstell = new double[3];
double e2, s, rlam, zps, h, phi, n, hp, phip;
int i, niter;
e2 = (a * a - b * b) / (a * a);
s = Math.sqrt(xstat[0] * xstat[0] + xstat[1] * xstat[1]);
rlam = Math.atan2(xstat[1], xstat[0]);
zps = xstat[2] / s;
h = Math.sqrt(xstat[0] * xstat[0] + xstat[1] * xstat[1] + xstat[2] * xstat[2]) - a;
phi = Math.atan(zps / (1.0 - e2 * a / (a + h)));
niter = 0;
for (i = 1; i <= 10000000; i++) {
n = a / Math.sqrt(1.0 - e2 * Math.sin(phi) * Math.sin(phi));
hp = h;
phip = phi;
h = s / Math.cos(phi) - n;
phi = Math.atan(zps / (1.0 - e2 * n / (n + h)));
niter = niter + 1;
if ((Math.abs(phip - phi) <= 1.e-11) && (Math.abs(hp - h) <= 1.e-5)) {
break;
}
if (niter >= 10) {
phi = -999.0;
rlam = -999.0;
h = -999.0;
break;
}
}
xstell[0] = phi * 180 / 3.1415926;
xstell[1] = rlam * 180 / 3.1415926;
xstell[2] = h;
return xstell;
}
/*
* gast.f
* Compute hour angle dtheta
*
* ! iyr, mon, iday, ihr, min and sec form a base (epoch) time,
* ! t is an offset from the base time in seconds
* ! dtheta is the output hour angle in radians
*
* Calculation of local time
*
* ! glon -- East longitude in degrees, -180 to 180
*
* call vprod.f spin.f rnorm.f
* Calculation of the unit vector normal to the occultation plane
* (clockwise rotated from GPS to LEO)
*/
/*
* double dtheta = gast(iyr,mon,iday,ihr,min,sec,t)
* utc=ihr*1.d0+min/60.d0
* timloc=utc+24.d0*glon/360.d0
* if (timloc.gt.24.d0) timloc=timloc-24.d0
* if (timloc.lt.0.d0) timloc=timloc+24.d0
*/
// In the inertial reference frame
/*
* v_inertial(1)= ! Inertial GPS position vectors, XYZ
* v_inertial(2)=
* v_inertial(3)=
*/
// In the Earth-fixed reference frame
// Z axis to rotate around (unit vector Z)
/*
* uvz(1)=0.0;
* uvz(2)=0.0;
* uvz(3)=1.0;
*
* double [] v_ecf = spin(v_inertial,uvz,-180.d0*dtheta/pi)
*/
// after this call, v_ecef should be in the (approximate) ECEF frame
/**
* ----------------------------------------------------------------------
* gast.f
*
* This subroutine computes the Greenwich Apparent Siderial
* Time angle given a UTC date and time.
*
* parameter Input parameters:
* Inputs:
*
* @param iyr integer year 1995
* @param imon integer month 5
* @param iday integer day 5
* @param ihr integer hour 5
* @param imin integer minute 5
* @param sec double second 31.0
* @param dsec double second 0.0
* Outputs:
* @return theta GAST angle in radians
*
* author Bill Schreiner
* @since May 1995
* version $URL: svn://ursa.cosmic.ucar.edu/trunk/src/roam/gast.f $ $Id: gast.f 10129 2008-07-30 17:10:52Z
* dhunt $
*/
private double gast(int iyr, int imon, int iday, int ihr, int imin, double sec, double dsec) {
//
// implicit double precision (a-h,o-z)
// character(len=*), parameter :: header = '$URL: svn://ursa.cosmic.ucar.edu/trunk/src/roam/gast.f $ $Id: gast.f
// 10129 2008-07-30 17:10:52Z dhunt $'
//
// Coordinate transform from the celestial inertial reference frame to the geo-
// centered Greenwich reference frame.
// Call a subroutine to calculate the Julian day "djd":
double djd = juday(imon, iday, iyr); // djd=julean day.
double tu = (djd - 2451545.0) / 36525.0;
double gmst = 24110.548410 + 8640184.8128660 * tu + 0.093104 * tu * tu - 6.2E-6 * Math.pow(tu, 3); // !gmst=Greenwich
// mean...
double utco = (ihr * 3600) + (imin * 60) + sec;
return togreenw(dsec, utco, gmst);
}
/**
* JDAY calculates the Julian Day number (JD) from the Gregorian month day, and year (M,D,Y). (NOT VALID BEFORE
* 10/15/1582)
*/
private double juday(int M, int D, int Y) {
double JD;
double IY = Y - (12 - M) / 10.0;
double IM = M + 1 + 12 * ((12 - M) / 10.0);
double I = IY / 100;
double J = 2 - I + I / 4 + Math.round(365.25 * IY) + Math.round(30.6001 * IM);
JD = (J + D + 1720994.50);
return JD;
}
/**
* This subroutine is to transform the locations and velocities of the GPS and
* LEO satellites from the celestial inertial reference frame to the Earth
* centered Greenwich reference frame.
* The dummy arguments iyear, month and iday are the calender year, month and
* day of the occultation event. The dummy arguments ihour, minute and sec
* are the UTC time.
* Reference: Astronomical Alamanus, 1993
*
* Modified subroutine from Dasheng's code.
*/
private double togreenw(double rectt, double utco, double gmst) {
double pi = Math.acos(-1.00);
//
// For each occultation ID, its TU and GMST are the same. However, every
// occultation event takes place at gmst+uts, uts is progressively increasing
// with every occultation event.
double utc = (utco + rectt) * 1.0027379093;
gmst = gmst + utc; // in seconds, without eoe correction.
// gmst may be a positive number or may be a negative number.
while (gmst < 0.0) {
gmst = gmst + 86400.00;
}
while (gmst > 86400.00) {
gmst = gmst - 86400.00;
}
// gmst = the Greenwich mean sidereal time.
// This gmst is without the corrections from the equation of equinoxes. For
// GPS/MET applications, the corrections from equation of equinoxes is not
// necessary because of the accurary needed.
return gmst * 2.0 * pi / 86400.0; // !*** This is the THETA in radian.
}
/**
* ----------------------------------------------------------------------
* file spin.f
*
* This subroutine rotates vector V1 around vector VS
* at angle A. V2 is the vector after the rotation.
*
*
* parameter Input parameters:
* v1 - Vector to be rotated
* vs - Vector around which to rotate v1
* a - angle of rotation
* Output parameters:
* v2 - output vector
*
* S.V.Sokolovskiy
* URL: svn://ursa.cosmic.ucar.edu/trunk/src/roam/spin.f $ $Id: spin.f 10129 2008-07-30 17:10:52Z dhunt $
* -----------------------------------------------------------------------
*
* @param v1 - Vector to be rotated
* @param vs - Vector around which to rotate v1
* @param a - angle of rotation
* @return _more_
*/
public double[] spin(double[] v1, double[] vs, double a) {
// implicit real*8(a-h,o-z)
// dimension v1(3),vs(3),vsn(3),v2(3),v3(3),s(3,3)
// Calculation of the unit vector around which
// the rotation should be done.
double[] v2 = new double[3];
double[] vsn = new double[3];
double[] v3 = new double[3];
double vsabs = Math.sqrt(vs[0] * vs[0] + vs[1] * vs[1] + vs[2] * vs[2]);
for (int i = 0; i < 3; i++) {
vsn[i] = vs[i] / vsabs;
}
// Calculation of the rotation matrix.
double a1 = Math.cos(a);
double a2 = 1.0 - a1;
double a3 = Math.sin(a);
double[][] s = new double[3][3];
s[0][0] = a2 * vsn[0] * vsn[0] + a1;
s[0][1] = a2 * vsn[0] * vsn[1] - a3 * vsn[2];
s[0][2] = a2 * vsn[0] * vsn[2] + a3 * vsn[1];
s[1][0] = a2 * vsn[1] * vsn[0] + a3 * vsn[2];
s[1][1] = a2 * vsn[1] * vsn[1] + a1;
s[1][2] = a2 * vsn[1] * vsn[2] - a3 * vsn[0];
s[2][0] = a2 * vsn[2] * vsn[0] - a3 * vsn[1];
s[2][1] = a2 * vsn[2] * vsn[1] + a3 * vsn[0];
s[2][2] = a2 * vsn[2] * vsn[2] + a1;
// Calculation of the rotated vector.
for (int i = 0; i < 3; i++) {
v3[i] = s[i][0] * v1[0] + s[i][1] * v1[1] + s[i][2] * v1[2];
}
System.arraycopy(v3, 0, v2, 0, 3);
return v2;
}
private static final double RTD = 180. / Math.PI;
private static final double DTR = Math.PI / 180.;
public double[] execute(double[] eci, double julian) {
double Xi = eci[0];
double Yi = eci[1];
double Zi = eci[2];
double c, s;
double GHA;
double[] ecef = new double[3];
// Compute GHAD
/* System generated locals */
double d__1, d__2, d__3;
/* Local variables */
double tsec, tday, gmst, t, omega, tfrac, tu, dat;
/* INPUT IS TIME "secondsSince1970" IN SECONDS AND "TDAY" */
/* WHICH IS WHOLE DAYS FROM 1970 JAN 1 0H */
/* THE OUTPUT IS GREENWICH HOUR ANGLE IN DEGREES */
/* XOMEGA IS ROTATION RATE IN DEGREES/SEC */
/* FOR COMPATABILITY */
tday = (double) ((int) (julian / 86400.));
tsec = julian - tday * 86400;
/* THE NUMBER OF DAYS FROM THE J2000 EPOCH */
/* TO 1970 JAN 1 0H UT1 IS -10957.5 */
t = tday - (float) 10957.5;
tfrac = tsec / 86400.;
dat = t;
tu = dat / 36525.;
/* Computing 2nd power */
d__1 = tu;
/* Computing 3rd power */
d__2 = tu;
d__3 = d__2;
gmst = tu * 8640184.812866 + 24110.54841 + d__1 * d__1 * .093104 - d__3 * (d__2 * d__2) * 6.2e-6;
/* COMPUTE THE EARTH'S ROTATION RATE */
/* Computing 2nd power */
d__1 = tu;
omega = tu * 5.098097e-6 + 86636.55536790872 - d__1 * d__1 * 5.09e-10;
/* COMPUTE THE GMST AND GHA */
// da is earth nutation - currently unused
double da = 0.0;
gmst = gmst + omega * tfrac + da * RTD * 86400. / 360.;
gmst = gmst % 86400;
if (gmst < 0.) {
gmst += 86400.;
}
gmst = gmst / 86400. * 360.;
// ghan = gmst;
// returns gha in radians
gmst = gmst * DTR;
GHA = gmst;
// RotateZ
c = Math.cos(GHA);
s = Math.sin(GHA);
double X = c * Xi + s * Yi;
double Y = -s * Xi + c * Yi;
// Set outputs
ecef[0] = X;
ecef[1] = Y;
ecef[2] = Zi;
return ecef;
}
/** comparing api to others */
public static double[] ECFtoLLA(double x, double y, double z, double a, double b) {
double longitude = Math.atan2(y, x);
double ePrimeSquared = (a * a - b * b) / (b * b);
double p = Math.sqrt(x * x + y * y);
double theta = Math.atan((z * a) / (p * b));
double sineTheta = Math.sin(theta);
double cosTheta = Math.cos(theta);
double f = 1 / 298.257223563;
double e2 = 2 * f - f * f;
double top = z + ePrimeSquared * b * sineTheta * sineTheta * sineTheta;
double bottom = p - e2 * a * cosTheta * cosTheta * cosTheta;
double geodeticLat = Math.atan(top / bottom);
double sineLat = Math.sin(geodeticLat);
double N = a / Math.sqrt(1 - e2 * sineLat * sineLat);
double altitude = (p / Math.cos(geodeticLat)) - N;
// maintain longitude btw -PI and PI
if (longitude > Math.PI) {
longitude -= 2 * Math.PI;
} else if (longitude < -Math.PI) {
longitude += 2 * Math.PI;
}
return new double[] {geodeticLat, longitude, altitude};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy