Skip to content

Commit

Permalink
Implement ExecStats for page panel
Browse files Browse the repository at this point in the history
This shows the execution status, including the age of the execution and
execution duration if available.

It's also where we put the button to request a recomputation of an
existing page.

The component's data is populated by the TimesSquareHtmlEventsContext.
We show how to mock the context in Storybook to design different states.
  • Loading branch information
jonathansick committed Apr 10, 2024
1 parent 66fe54a commit f4277a0
Show file tree
Hide file tree
Showing 5 changed files with 136 additions and 12 deletions.
1 change: 1 addition & 0 deletions apps/squareone/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
"@reach/alert": "^0.17.0",
"@reach/menu-button": "^0.17.0",
"ajv": "^8.11.0",
"date-fns": "^3.6.0",
"formik": "^2.2.9",
"js-yaml": "^4.1.0",
"next": "^12.2.4",
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
/*
* ExecStats provides a summary of the execution status and timing of the
* notebook execution. It also provides a button to request the recomputation
* of the already-executed notebook.
*/

import React from 'react';
import styled from 'styled-components';
import { parseISO, formatDistanceToNow } from 'date-fns';

import { TimesSquareHtmlEventsContext } from '../TimesSquareHtmlEventsProvider';
import { GhostButton } from '../Button';

export default function ExecStats({}) {
const htmlEvent = React.useContext(TimesSquareHtmlEventsContext);

const handleRecompute = async (event) => {
event.preventDefault();

await fetch(htmlEvent.htmlUrl, {
method: 'DELETE',
headers: {
'Content-Type': 'application/json',
},
});
};

if (htmlEvent.executionStatus == 'complete') {
const dateFinished = parseISO(htmlEvent.dateFinished);
const formattedDuration = Number(htmlEvent.executionDuration).toFixed(1);
return (
<StyledContainer>
<StyledContent>
Computed{' '}
<time
dateTime={htmlEvent.dateFinished}
title={htmlEvent.dateFinished}
>
{formatDistanceToNow(dateFinished, { addSuffix: true })}
</time>{' '}
in {formattedDuration} seconds.
</StyledContent>
<GhostButton onClick={handleRecompute}>Recompute</GhostButton>
</StyledContainer>
);
}

if (htmlEvent.executionStatus == 'in_progress') {
return (
<StyledContainer>
<p>Computing…</p>
</StyledContainer>
);
}

return null;
}

const StyledContainer = styled.div`
display: flex;
flex-direction: column;
gap: 0.5rem;
margin-top: 2rem;
`;

const StyledContent = styled.p`
font-size: 0.8rem;
margin-bottom: 0;
`;
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
import React from 'react';

import { TimesSquareHtmlEventsContext } from '../TimesSquareHtmlEventsProvider';
import ExecStats from './ExecStats';

export default {
component: ExecStats,
title: 'Components/TimesSquare/ExecStats',
parameters: {
viewport: {
viewports: {
sidebar: {
name: 'Sidebar',
styles: {
width: '280px',
height: '900px',
},
},
},
},
defaultViewport: 'sidebar',
},
};

const Template = (args) => (
<TimesSquareHtmlEventsContext.Provider value={args}>
<ExecStats />
</TimesSquareHtmlEventsContext.Provider>
);

export const Default = Template.bind({});
Default.args = {
dateSubmitted: '2021-09-01T12:00:00Z',
dateStarted: '2021-09-01T12:00:01Z',
dateFinished: '2021-09-01T12:00:10Z',
executionStatus: 'complete',
executionDuration: 10.12,
};

export const InProgressNew = Template.bind({});
InProgressNew.args = {
dateSubmitted: '2021-09-01T12:00:10Z',
dateStarted: null,
dateFinished: null,
executionStatus: 'in_progress',
executionDuration: null,
};

export const InProgressExisting = Template.bind({});
InProgressExisting.args = {
dateSubmitted: '2021-09-01T12:00:00Z',
dateStarted: '2021-09-01T12:00:01Z',
dateFinished: '2021-09-01T12:00:10Z',
executionStatus: 'in_progress',
executionDuration: 10.12,
};
Original file line number Diff line number Diff line change
Expand Up @@ -12,14 +12,12 @@ import Error from 'next/error';

import useTimesSquarePage from '../../hooks/useTimesSquarePage';
import TimesSquareParameters from '../TimesSquareParameters';
import { TimesSquareHtmlEventsContext } from '../TimesSquareHtmlEventsProvider';
import ExecStats from './ExecStats';

export default function TimesSquareGitHubPagePanel({}) {
const { publicRuntimeConfig } = getConfig();
const pageData = useTimesSquarePage();

const htmlEvent = React.useContext(TimesSquareHtmlEventsContext);

if (pageData.loading) {
return <p>Loading...</p>;
}
Expand All @@ -40,15 +38,8 @@ export default function TimesSquareGitHubPagePanel({}) {
<div dangerouslySetInnerHTML={{ __html: description.html }}></div>
)}
<TimesSquareParameters />
{htmlEvent.executionStatus != 'complete' && (
<p>{htmlEvent.executionStatus}</p>
)}
{htmlEvent.executionStatus == 'complete' && (
<>
<p>Computed {htmlEvent.dateFinished}</p>
<p>Executed in {htmlEvent.executionDuration} seconds</p>
</>
)}

<ExecStats />
</div>
</PagePanelContainer>
);
Expand Down
7 changes: 7 additions & 0 deletions pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit f4277a0

Please sign in to comment.