SinexParseInfo.java

  1. /* Copyright 2002-2025 CS GROUP
  2.  * Licensed to CS GROUP (CS) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * CS licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *   http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.orekit.files.sinex;

  18. import org.hipparchus.geometry.euclidean.threed.Vector3D;
  19. import org.orekit.errors.OrekitException;
  20. import org.orekit.errors.OrekitMessages;
  21. import org.orekit.gnss.GnssSignal;
  22. import org.orekit.gnss.SatInSystem;
  23. import org.orekit.models.earth.displacement.PsdCorrection;
  24. import org.orekit.time.AbsoluteDate;
  25. import org.orekit.time.TimeScales;
  26. import org.orekit.utils.TimeSpanMap;

  27. import java.util.HashMap;
  28. import java.util.Map;

  29. /** Parse information for Solution INdependent EXchange (SINEX) files.
  30.  * @author Bryan Cazabonne
  31.  * @author Luc Maisonobe
  32.  * @since 13.0
  33.  */
  34. public class SinexParseInfo extends ParseInfo<Sinex> {

  35.     /** Satellites antennas. */
  36.     private final Map<SatInSystem, Map<GnssSignal, Vector3D>> satellitesPhaseCenters;

  37.     /** Stations phase centers. */
  38.     private final Map<AntennaKey, Map<GnssSignal, Vector3D>> stationsPhaseCenters;

  39.     /** Station data. */
  40.     private final Map<String, Station> stations;

  41.     /** Earth Orientation Parameters data. */
  42.     private final Map<AbsoluteDate, SinexEopEntry> eop;

  43.     /** Station position X coordinate. */
  44.     private double px;

  45.     /** Station position Y coordinate. */
  46.     private double py;

  47.     /** Station position Z coordinate. */
  48.     private double pz;

  49.     /** Station velocity X coordinate. */
  50.     private double vx;

  51.     /** Station velocity Y coordinate. */
  52.     private double vy;

  53.     /** Station velocity Z coordinate. */
  54.     private double vz;

  55.     /** Correction axis. */
  56.     private PsdCorrection.Axis axis;

  57.     /** Correction time evolution. */
  58.     private PsdCorrection.TimeEvolution evolution;

  59.     /** Correction amplitude. */
  60.     private double amplitude;

  61.     /** Correction relaxation time. */
  62.     private double relaxationTime;

  63.     /** Phase centers. */
  64.     private final Map<GnssSignal, Vector3D> phaseCenters;

  65.     /** Simple constructor.
  66.      * @param timeScales time scales
  67.      */
  68.     SinexParseInfo(final TimeScales timeScales) {
  69.         super(timeScales);
  70.         this.satellitesPhaseCenters = new HashMap<>();
  71.         this.stationsPhaseCenters   = new HashMap<>();
  72.         this.stations               = new HashMap<>();
  73.         this.eop                    = new HashMap<>();
  74.         this.phaseCenters           = new HashMap<>();
  75.     }

  76.     /** {@inheritDoc} */
  77.     @Override
  78.     void newSource(final String name) {
  79.         super.newSource(name);
  80.         resetPosition();
  81.         resetVelocity();
  82.         resetPsdCorrection();
  83.     }

  84.     /** Add satellite phase center.
  85.      * @param satInSystem satellite id
  86.      * @param signal signal
  87.      * @param phaseCenter phase center
  88.      */
  89.     void addSatellitePhaseCenter(final SatInSystem satInSystem, final GnssSignal signal, final Vector3D phaseCenter) {
  90.         satellitesPhaseCenters.
  91.             computeIfAbsent(satInSystem, s -> new HashMap<>()).
  92.             put(signal, phaseCenter);
  93.     }

  94.     /** Add station phase center.
  95.      * @param key antenna key
  96.      * @param phaseCenter phase center
  97.      * @param signals signals to use in order
  98.      */
  99.     void addStationPhaseCenter(final AntennaKey key, final Vector3D phaseCenter, final GnssSignal[] signals) {
  100.         phaseCenters.put(signals[phaseCenters.size()], phaseCenter);
  101.         if (phaseCenters.size() == signals.length) {
  102.             // we have parsed all expected signals
  103.             stationsPhaseCenters.computeIfAbsent(key, k -> new HashMap<>()).putAll(phaseCenters);
  104.             phaseCenters.clear();
  105.         }
  106.     }

  107.     /** Add station.
  108.      * @param station station to add
  109.      */
  110.     void addStation(final Station station) {
  111.         stations.putIfAbsent(station.getSiteCode(), station);
  112.     }

  113.     /** Get station from current line.
  114.      * @param index index of station in current line
  115.      * @return station
  116.      */
  117.     Station getCurrentLineStation(final int index) {
  118.         return stations.get(parseString(index, 4));
  119.     }

  120.     /** Get start date from current line.
  121.      * @return start date
  122.      */
  123.     AbsoluteDate getCurrentLineStartDate() {
  124.         return stringEpochToAbsoluteDate(parseString(16, 12), true);
  125.     }

  126.     /** Get end date from current line.
  127.      * @return end date
  128.      */
  129.     AbsoluteDate getCurrentLineEndDate() {
  130.         return stringEpochToAbsoluteDate(parseString(29, 12), false);
  131.     }

  132.     /** Set station position X coordinate.
  133.      * @param x station position X coordinate
  134.      * @param station station
  135.      * @param epoch   coordinates epoch
  136.      */
  137.     void setPx(final double x, final Station station, final AbsoluteDate epoch) {
  138.         this.px = x;
  139.         finalizePositionIfComplete(station, epoch);
  140.     }

  141.     /** Set station position Y coordinate.
  142.      * @param y station position Y coordinate
  143.      * @param station station
  144.      * @param epoch   coordinates epoch
  145.      */
  146.     void setPy(final double y, final Station station, final AbsoluteDate epoch) {
  147.         this.py = y;
  148.         finalizePositionIfComplete(station, epoch);
  149.     }

  150.     /** Set station position Z coordinate.
  151.      * @param z station position Z coordinate
  152.      * @param station station
  153.      * @param epoch   coordinates epoch
  154.      */
  155.     void setPz(final double z, final Station station, final AbsoluteDate epoch) {
  156.         this.pz = z;
  157.         finalizePositionIfComplete(station, epoch);
  158.     }

  159.     /** Finalize station position if complete.
  160.      * @param station station
  161.      * @param epoch   coordinates epoch
  162.      */
  163.     private void finalizePositionIfComplete(final Station station, final AbsoluteDate epoch) {
  164.         if (!Double.isNaN(px + py + pz)) {
  165.             // all coordinates are available, position is complete
  166.             station.setEpoch(epoch);
  167.             station.setPosition(new Vector3D(px, py, pz));
  168.             resetPosition();
  169.         }
  170.     }

  171.     /** Reset position.
  172.      */
  173.     void resetPosition() {
  174.         px = Double.NaN;
  175.         py = Double.NaN;
  176.         pz = Double.NaN;
  177.     }

  178.     /** Set station velocity X coordinate.
  179.      * @param x station velocity X coordinate
  180.      * @param station station
  181.      */
  182.     void setVx(final double x, final Station station) {
  183.         this.vx = x;
  184.         finalizeVelocityIfComplete(station);
  185.     }

  186.     /** Set station velocity Y coordinate.
  187.      * @param y station velocity Y coordinate
  188.      * @param station station
  189.      */
  190.     void setVy(final double y, final Station station) {
  191.         this.vy = y;
  192.         finalizeVelocityIfComplete(station);
  193.     }

  194.     /** Set station velocity Z coordinate.
  195.      * @param z station velocity Z coordinate
  196.      * @param station station
  197.      */
  198.     void setVz(final double z, final Station station) {
  199.         this.vz = z;
  200.         finalizeVelocityIfComplete(station);
  201.     }

  202.     /** Finalize station velocity if complete.
  203.      * @param station station
  204.      */
  205.     private void finalizeVelocityIfComplete(final Station station) {
  206.         if (!Double.isNaN(vx + vy + vz)) {
  207.             // all coordinates are available, velocity is complete
  208.             station.setVelocity(new Vector3D(vx, vy, vz));
  209.             resetVelocity();
  210.         }
  211.     }

  212.     /** Reset velocity.
  213.      */
  214.     void resetVelocity() {
  215.         vx = Double.NaN;
  216.         vy = Double.NaN;
  217.         vz = Double.NaN;
  218.     }

  219.     /** Set correction axis.
  220.      * @param axis correction axis
  221.      */
  222.     void setAxis(final PsdCorrection.Axis axis) {
  223.         this.axis = axis;
  224.     }

  225.     /** Set correction time evolution.
  226.      * @param evolution correction time evolution
  227.      */
  228.     void setEvolution(final PsdCorrection.TimeEvolution evolution) {
  229.         this.evolution = evolution;
  230.     }

  231.     /** Set correction amplitude.
  232.      * @param correctionAmplitude correction amplitude
  233.      * @param station station
  234.      * @param epoch   coordinates epoch
  235.      */
  236.     void setAmplitude(final double correctionAmplitude, final Station station, final AbsoluteDate epoch) {
  237.         this.amplitude = correctionAmplitude;
  238.         finalizePsdCorrectionIfComplete(station, epoch);
  239.     }

  240.     /** Set correction relaxation time.
  241.      * @param correctionRelaxationTime correction relaxation time
  242.      * @param station station
  243.      * @param epoch   coordinates epoch
  244.      */
  245.     void setRelaxationTime(final double correctionRelaxationTime,
  246.                            final Station station, final AbsoluteDate epoch) {
  247.         this.relaxationTime = correctionRelaxationTime;
  248.         finalizePsdCorrectionIfComplete(station, epoch);
  249.     }

  250.     /** Finalize a Post-Seismic Deformation correction model if complete.
  251.      * @param station station
  252.      * @param epoch   coordinates epoch
  253.      */
  254.     private void finalizePsdCorrectionIfComplete(final Station station, final AbsoluteDate epoch) {
  255.         if (!Double.isNaN(amplitude + relaxationTime)) {
  256.             // both amplitude and relaxation time are available, correction is complete
  257.             final PsdCorrection correction = new PsdCorrection(axis, evolution, epoch, amplitude, relaxationTime);
  258.             station.addPsdCorrectionValidAfter(correction, epoch);
  259.             resetPsdCorrection();
  260.         }
  261.     }

  262.     /** Reset Post-Seismic Deformation correction model.
  263.      */
  264.     private void resetPsdCorrection() {
  265.         axis           = null;
  266.         evolution      = null;
  267.         amplitude      = Double.NaN;
  268.         relaxationTime = Double.NaN;
  269.     }

  270.     /** Create EOP entry.
  271.      * @param date EOP date
  272.      * @return EOP entry at date, creating it if needed
  273.      */
  274.     SinexEopEntry createEOPEntry(final AbsoluteDate date) {
  275.         return eop.computeIfAbsent(date, SinexEopEntry::new);
  276.     }

  277.     /** {@inheritDoc} */
  278.     @Override
  279.     protected Sinex build() {

  280.         // set up phase centers for stations
  281.         for (final Station station : stations.values()) {

  282.             // time span map we need to populate
  283.             final TimeSpanMap<Map<GnssSignal, Vector3D>> phaseCentersMap = station.getPhaseCentersMap();

  284.             if (station.getAntennaKeyTimeSpanMap().getSpansNumber() > 1) {
  285.                 for (TimeSpanMap.Span<AntennaKey> keySpan = station.getAntennaKeyTimeSpanMap().getFirstNonNullSpan();
  286.                      keySpan != null; keySpan = keySpan.next()) {

  287.                     // get the existing map for this span
  288.                     Map<GnssSignal, Vector3D> centers =
  289.                         phaseCentersMap.get(AbsoluteDate.createMedian(keySpan.getStart(), keySpan.getEnd()));
  290.                     if (centers == null) {
  291.                         // this is the first time we process this time span
  292.                         centers = new HashMap<>();
  293.                         phaseCentersMap.addValidBetween(centers, keySpan.getStart(), keySpan.getEnd());
  294.                     }

  295.                     if (keySpan.getData() != null) {

  296.                         // try to identify the closest match for antenna
  297.                         AntennaKey closestKey = null;
  298.                         for (final AntennaKey candidate : keySpan.getData().matchingCandidates()) {
  299.                             if (stationsPhaseCenters.containsKey(candidate)) {
  300.                                 closestKey = candidate;
  301.                                 break;
  302.                             }
  303.                         }
  304.                         if (closestKey == null) {
  305.                             throw new OrekitException(OrekitMessages.UNKNOWN_GNSS_ANTENNA,
  306.                                                       keySpan.getData().getName(),
  307.                                                       keySpan.getData().getRadomeCode(),
  308.                                                       keySpan.getData().getSerialNumber());
  309.                         }

  310.                         // add the phase centers for the closest key
  311.                         centers.putAll(stationsPhaseCenters.get(closestKey));

  312.                     }

  313.                 }
  314.             }

  315.         }

  316.         return new Sinex(getTimeScales(), getCreationDate(), getStartDate(), getEndDate(),
  317.                          satellitesPhaseCenters, stations, eop);
  318.     }

  319. }