Skip to content

Commit d5ab39d

Browse files
Merge pull request #259 from nyx-space/quickfix/residuals-plot
Quickfix/residuals plot
2 parents a081425 + 8e01cf4 commit d5ab39d

File tree

4 files changed

+106
-32
lines changed

4 files changed

+106
-32
lines changed

python/nyx_space/plots/__init__.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717
"""
1818

1919
from .gauss_markov import plot_gauss_markov
20-
from .od import plot_covar, plot_estimates, plot_measurements
20+
from .od import plot_covar, plot_estimates, plot_measurements, overlay_measurements
2121
from .traj import plot_traj, plot_ground_track, plot_traj_errors
2222

2323
__all__ = [
@@ -28,4 +28,5 @@
2828
"plot_traj_errors",
2929
"plot_ground_track",
3030
"plot_measurements",
31+
"overlay_measurements",
3132
]

python/nyx_space/plots/od.py

+97-25
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717
"""
1818

1919
import plotly.graph_objects as go
20+
import plotly.express as px
21+
from datetime import datetime
2022

2123
from .utils import plot_with_error, plot_line, finalize_plot, colors
2224

@@ -28,6 +30,7 @@
2830
import numpy as np
2931
from scipy.stats import norm
3032

33+
from nyx_space.time import Epoch
3134

3235
def plot_estimates(
3336
dfs,
@@ -93,8 +96,8 @@ def plot_estimates(
9396
epoch = epoch.replace("UTC", "").strip()
9497
if "." not in epoch:
9598
epoch += ".0"
96-
pd_ok_epochs += [epoch]
97-
time_col = pd.to_datetime(pd_ok_epochs)
99+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
100+
time_col = pd.Series(pd_ok_epochs)
98101
x_title = "Epoch {}".format(time_col_name[-3:])
99102

100103
# Check that the requested covariance frame exists
@@ -247,12 +250,12 @@ def plot_estimates(
247250

248251
if msr_df is not None:
249252
# Plot the measurements on both plots
250-
pos_fig = plot_measurements(
251-
msr_df, title, time_col_name, fig=pos_fig, show=False
253+
pos_fig = overlay_measurements(
254+
pos_fig, msr_df, title, time_col_name, show=False
252255
)
253256

254-
vel_fig = plot_measurements(
255-
msr_df, title, time_col_name, fig=vel_fig, show=False
257+
vel_fig = overlay_measurements(
258+
vel_fig, msr_df, title, time_col_name, show=False
256259
)
257260

258261
if html_out:
@@ -333,8 +336,8 @@ def plot_covar(
333336
epoch = epoch.replace("UTC", "").strip()
334337
if "." not in epoch:
335338
epoch += ".0"
336-
pd_ok_epochs += [epoch]
337-
time_col = pd.to_datetime(pd_ok_epochs)
339+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
340+
time_col = pd.Series(pd_ok_epochs)
338341
x_title = "Epoch {}".format(time_col_name[-3:])
339342

340343
# Check that the requested covariance frame exists
@@ -454,12 +457,12 @@ def plot_covar(
454457

455458
if msr_df is not None:
456459
# Plot the measurements on both plots
457-
pos_fig = plot_measurements(
458-
msr_df, title, time_col_name, fig=pos_fig, show=False
460+
pos_fig = overlay_measurements(
461+
pos_fig, msr_df, title, time_col_name, show=False
459462
)
460463

461-
vel_fig = plot_measurements(
462-
msr_df, title, time_col_name, fig=vel_fig, show=False
464+
vel_fig = overlay_measurements(
465+
vel_fig, msr_df, title, time_col_name, show=False
463466
)
464467

465468
if html_out:
@@ -481,21 +484,22 @@ def plot_covar(
481484
return pos_fig, vel_fig
482485

483486

484-
def plot_measurements(
487+
def overlay_measurements(
488+
fig,
485489
dfs,
486490
title,
487491
time_col_name="Epoch:Gregorian UTC",
488492
html_out=None,
489493
copyright=None,
490-
fig=None,
491494
show=True,
492495
):
496+
"""
497+
Given a plotly figure, overlay the measurements as shaded regions on top of the existing plot.
498+
For a plot of measurements only, use `plot_measurements`.
499+
"""
493500
if not isinstance(dfs, list):
494501
dfs = [dfs]
495502

496-
if fig is None:
497-
fig = go.Figure()
498-
499503
color_values = list(colors.values())
500504

501505
station_colors = {}
@@ -518,8 +522,8 @@ def plot_measurements(
518522
epoch = epoch.replace("UTC", "").strip()
519523
if "." not in epoch:
520524
epoch += ".0"
521-
pd_ok_epochs += [epoch]
522-
time_col = pd.to_datetime(pd_ok_epochs)
525+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
526+
time_col = pd.Series(pd_ok_epochs)
523527
x_title = "Epoch {}".format(time_col_name[-3:])
524528

525529
# Diff the epochs of the measurements to find when there is a start and end.
@@ -571,7 +575,7 @@ def plot_measurements(
571575
line_width=0,
572576
)
573577

574-
finalize_plot(fig, title, x_title, copyright, show)
578+
finalize_plot(fig, title, x_title, None, copyright)
575579

576580
if html_out:
577581
with open(html_out, "w") as f:
@@ -595,7 +599,7 @@ def plot_residuals(
595599
show=True,
596600
):
597601
"""
598-
Plot of residuals, with 3-σ lines
602+
Plot of residuals, with 3-σ lines. Returns a tuple of the plots if show=False.
599603
"""
600604

601605
try:
@@ -615,12 +619,14 @@ def plot_residuals(
615619
epoch = epoch.replace("UTC", "").strip()
616620
if "." not in epoch:
617621
epoch += ".0"
618-
pd_ok_epochs += [epoch]
619-
time_col = pd.to_datetime(pd_ok_epochs)
622+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
623+
time_col = pd.Series(pd_ok_epochs)
620624
x_title = "Epoch {}".format(time_col_name[-3:])
621625

622626
plt_any = False
623627

628+
rtn_plots = []
629+
624630
for col in df.columns:
625631
if col.startswith(kind):
626632
fig = go.Figure()
@@ -671,8 +677,8 @@ def plot_residuals(
671677

672678
if msr_df is not None:
673679
# Plot the measurements on both plots
674-
fig = plot_measurements(
675-
msr_df, title, time_col_name, fig=fig, show=False
680+
fig = overlay_measurements(
681+
fig, msr_df, title, time_col_name, show=False
676682
)
677683

678684
finalize_plot(
@@ -689,10 +695,15 @@ def plot_residuals(
689695

690696
if show:
691697
fig.show()
698+
else:
699+
rtn_plots += [fig]
692700

693701
if not plt_any:
694702
raise ValueError(f"No columns ending with {kind} found -- nothing plotted")
695703

704+
if not show:
705+
return rtn_plots
706+
696707

697708
def plot_residual_histogram(
698709
df, title, kind="Prefit", copyright=None, html_out=None, show=True
@@ -737,3 +748,64 @@ def plot_residual_histogram(
737748

738749
if show:
739750
fig.show()
751+
752+
def plot_measurements(
753+
df,
754+
msr_type=None,
755+
title=None,
756+
time_col_name="Epoch:Gregorian UTC",
757+
html_out=None,
758+
copyright=None,
759+
show=True,
760+
):
761+
"""
762+
Plot the provided measurement type, fuzzy matching of the column name, or plot all as a strip
763+
"""
764+
765+
if title is None:
766+
# Build a title
767+
station_names = ", ".join([name for name in df["Tracking device"].unique()])
768+
start = Epoch(df["Epoch:Gregorian UTC"].iloc[0])
769+
end = Epoch(df["Epoch:Gregorian UTC"].iloc[-1])
770+
arc_duration = end.timedelta(start)
771+
title = f"Measurements from {station_names} spanning {start} to {end} ({arc_duration})"
772+
773+
try:
774+
orig_tim_col = df[time_col_name]
775+
except KeyError:
776+
# Find the time column
777+
try:
778+
col_name = [x for x in df.columns if x.startswith("Epoch")][0]
779+
except IndexError:
780+
raise KeyError("Could not find any Epoch column")
781+
print(f"Could not find time column {time_col_name}, using `{col_name}`")
782+
orig_tim_col = df[col_name]
783+
784+
# Build a Python datetime column
785+
pd_ok_epochs = []
786+
for epoch in orig_tim_col:
787+
epoch = epoch.replace("UTC", "").strip()
788+
if "." not in epoch:
789+
epoch += ".0"
790+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
791+
df["time_col"] = pd.Series(pd_ok_epochs)
792+
x_title = "Epoch {}".format(time_col_name[-3:])
793+
794+
if msr_type is None:
795+
fig = px.strip(df, x="time_col", y="Tracking device", color="Tracking device")
796+
finalize_plot(fig, title, x_title, "All tracking data", copyright)
797+
else:
798+
msr_col_name = [col for col in df.columns if msr_type in col.lower()]
799+
800+
fig = px.scatter(df, x="time_col", y=msr_col_name, color="Tracking device")
801+
finalize_plot(fig, title, x_title, msr_col_name[0], copyright)
802+
803+
if html_out:
804+
with open(html_out, "w") as f:
805+
f.write(fig.to_html())
806+
print(f"Saved HTML to {html_out}")
807+
808+
if show:
809+
fig.show()
810+
else:
811+
return fig

python/nyx_space/plots/traj.py

+5-4
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
import plotly.graph_objects as go
2020
from plotly.subplots import make_subplots
2121
import pandas as pd
22+
from datetime import datetime
2223

2324
from .utils import (
2425
radii,
@@ -219,8 +220,8 @@ def plot_orbit_elements(
219220
epoch = epoch.replace("UTC", "").strip()
220221
if "." not in epoch:
221222
epoch += ".0"
222-
pd_ok_epochs += [epoch]
223-
df["Epoch"] = pd.to_datetime(pd_ok_epochs)
223+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
224+
df["Epoch"] = pd.Series(pd_ok_epochs)
224225

225226
if not isinstance(names, list):
226227
names = [names]
@@ -317,8 +318,8 @@ def plot_traj_errors(
317318
epoch = epoch.replace("UTC", "").strip()
318319
if "." not in epoch:
319320
epoch += ".0"
320-
pd_ok_epochs += [epoch]
321-
df["Epoch"] = pd.to_datetime(pd_ok_epochs)
321+
pd_ok_epochs += [datetime.fromisoformat(str(epoch).replace("UTC", "").strip())]
322+
df["Epoch"] = pd.Series(pd_ok_epochs)
322323

323324
if not isinstance(names, list):
324325
names = [names]

python/nyx_space/plots/utils.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ def _add_watermark(who):
104104
nyx_tpl.layout.annotations = [
105105
dict(
106106
name="watermark",
107-
text=f"Nyx Space 🄯 AGPLv3 {year}",
107+
text=f"Powered by Nyx Space © {year}",
108108
opacity=0.75,
109109
font=dict(color="#3d84e8", size=12),
110110
xref="paper",
@@ -201,7 +201,7 @@ def finalize_plot(fig, title, xtitle=None, ytitle=None, copyright=None):
201201
"""
202202

203203
annotations = [dict(templateitemname="watermark")]
204-
if copyright:
204+
if copyright is not None:
205205
annotations += [
206206
dict(
207207
templateitemname="watermark",

0 commit comments

Comments
 (0)