diff --git a/pkg/metrics/metrics.jsx b/pkg/metrics/metrics.jsx
index c2095fe4abbf..870f78e14acb 100644
--- a/pkg/metrics/metrics.jsx
+++ b/pkg/metrics/metrics.jsx
@@ -1101,7 +1101,20 @@ class MetricsMinute extends React.Component {
});
let desc;
- if (this.props.isExpanded && this.props.events) {
+ if (this.props.isExpanded && this.props.booted) {
+ const timestamp = this.props.startTime + (this.props.minute * 60000);
+ desc = (
+
+
+
+
+
+ {_("Boot")}
+
+
+
+ );
+ } else if (this.props.isExpanded && this.props.events) {
const timestamp = this.props.startTime + (this.props.minute * 60000);
const logsPanel = (
@@ -1170,6 +1183,7 @@ class MetricsHour extends React.Component {
if (this.state.dataItems !== nextProps.data.length ||
this.state.isHourExpanded !== nextState.isHourExpanded ||
this.props.startTime !== nextProps.startTime ||
+ this.props.boots !== nextProps.boots ||
Object.keys(this.props.selectedVisibility).some(itm => this.props.selectedVisibility[itm] != nextProps.selectedVisibility[itm])) {
this.updateGraphs(nextProps.data, nextProps.startTime, nextProps.selectedVisibility, nextState.isHourExpanded);
return false;
@@ -1237,6 +1251,8 @@ class MetricsHour extends React.Component {
const dataOffset = minute * SAMPLES_PER_MIN;
const dataSlice = normData.slice(dataOffset, dataOffset + SAMPLES_PER_MIN);
const rawSlice = this.props.data.slice(dataOffset, dataOffset + SAMPLES_PER_MIN);
+ const is_boot = this.props.boots.includes(minute);
+
minuteGraphs.push(
+ selectedVisibility={selectedVisibility}
+ booted={is_boot} />
);
}
@@ -1559,7 +1576,8 @@ class MetricsHistory extends React.Component {
packagekitExists: false,
isBeibootBridge: false,
isPythonPCPInstalled: null,
- selectedVisibility: this.columns.reduce((a, v) => ({ ...a, [v[0]]: true }), {})
+ selectedVisibility: this.columns.reduce((a, v) => ({ ...a, [v[0]]: true }), {}),
+ boots: [], // journalctl --list-boots as [{started: Date, ended: Date}]
};
this.handleMoreData = this.handleMoreData.bind(this);
@@ -1621,8 +1639,24 @@ class MetricsHistory extends React.Component {
} catch (_ex) {}
const isBeibootBridge = cmdline?.includes("ic# cockpit-bridge");
-
this.setState({ packagekitExists, isBeibootBridge });
+
+ try {
+ // Only 14 days of metrics are shown
+ // Requires superuser on Debian/Ubuntu, on Fedora/Arch users in the wheel group can list without superuser.
+ const output = await cockpit.spawn(["journalctl", "--list-boots", "--since", "-15d", "--output", "json"], { superuser: "try" });
+ const list_boots = JSON.parse(output);
+ const boots = list_boots.map(boot => {
+ return {
+ started: new Date(boot?.first_entry / 1000),
+ ended: new Date(boot?.last_entry / 1000),
+ current_boot: boot?.index === 0,
+ };
+ });
+ this.setState({ boots });
+ } catch (exc) {
+ console.warn("journalctl --list-boots failed", exc);
+ }
}
handleMoreData() {
@@ -1927,6 +1961,11 @@ class MetricsHistory extends React.Component {
{ this.state.hours.map((time, i) => {
+ const date_time = new Date(time);
+ const boot_minutes = this.state.boots.filter(reboot => reboot.started.getDay() === date_time.getDay() &&
+ reboot.started.getYear() === date_time.getYear() &&
+ reboot.started.getHours() === date_time.getHours())
+ .map(reboot => reboot.started.getMinutes());
const showHeader = i == 0 || timeformat.date(time) != timeformat.date(this.state.hours[i - 1]);
return (
@@ -1934,7 +1973,8 @@ class MetricsHistory extends React.Component {
{showHeader && }
+ data={this.data[time]} clipLeading={i == 0}
+ boots={boot_minutes} />
);
})}
diff --git a/test/verify/check-metrics b/test/verify/check-metrics
index c67d1a617613..01ccd8ce801e 100755
--- a/test/verify/check-metrics
+++ b/test/verify/check-metrics
@@ -440,14 +440,23 @@ class TestHistoryMetrics(testlib.MachineCase):
return
m.upload(["verify/files/metrics-archives/journal.journal.gz"], "/tmp")
+ # we need to move all other existing journals out of the way, otherwise boot order is going back in time
m.execute("""gunzip /tmp/journal.journal.gz
+ systemctl stop systemd-journald
+ rm /var/log/journal/*/*.journal
cp /tmp/journal.journal /var/log/journal/*/""")
b.reload()
b.enter_page("/metrics")
- # details for above load event
+ # Expand metrics when loaded
b.wait_in_text(".metrics-heading", "CPU")
b.click("#metrics-hour-1615197600000 button.metrics-events-expander")
+
+ # Show boot as event
+ with b.wait_timeout(60):
+ b.wait_in_text(".metrics-minute[data-minute='35']", "Boot")
+
+ # details for above load event
b.click(f"{load_minute_sel} .metrics-events button.spikes_info")
b.wait_in_text(".cockpit-log-panel", "load-hog")
action = "Stopping" if load_minute == 39 else "Started"