Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Virtualized scrolling for trace detail view #68

Merged
merged 44 commits into from
Sep 8, 2017
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
fe35bbd
Update prettier and wrap at 110 instead of 80
tiffon Aug 3, 2017
acbfdee
WIP commit for refactoring trace detail view
tiffon Aug 11, 2017
4fa5c8a
WIP commit for refactoring trace detail view
tiffon Aug 14, 2017
93a6fb7
WIP refactor trace detail, split out css, start fixing tests
tiffon Aug 17, 2017
b8945ab
WIP refactor trace detail, tests working again
tiffon Aug 20, 2017
401b083
WIP refactor trace, yarn upgrade
tiffon Aug 20, 2017
c34bbd3
WIP refactor trace, test maintenance
tiffon Aug 20, 2017
c041353
remove unused import
tiffon Aug 20, 2017
9f54454
add license to css files
tiffon Aug 20, 2017
c657cf1
Revert sub-page scrolling for trace detail
tiffon Aug 20, 2017
2226744
Support URL prefix via homepage in package.json
tiffon Aug 20, 2017
f555e8b
Prevent collision of logs in log entries table
tiffon Aug 20, 2017
bef7f5b
Add comments to new utils
tiffon Aug 21, 2017
8101efd
WIP Use react-virtualized in trace detail view
tiffon Aug 21, 2017
fa6f016
Fix #59 - "Span Name" to "Service & Operation"
tiffon Aug 21, 2017
3797f5b
Fix unreleased regression - ellipsis on span name
tiffon Aug 21, 2017
0fd43f5
Address PR comment on search results scatter plot
tiffon Aug 21, 2017
542c1ba
Misc cleanup from PR comment
tiffon Aug 21, 2017
42736e0
PR comment - Remove ms and use nano seconds
tiffon Aug 21, 2017
d16097b
PR comment - Adjust export, relates to recompose
tiffon Aug 21, 2017
83b28d5
Reorganize span detail components
tiffon Aug 22, 2017
bdd6d7e
PR comment - Comment getTraceSpanIdsAsTree()
tiffon Aug 22, 2017
7204684
PR comment - Add "SpanID:" label
tiffon Aug 22, 2017
9bdf8b6
Merge branch 'refactor-trace-detail-selectors' into span-detail-reorg
tiffon Aug 22, 2017
da7bbb2
Merge branch 'refactor-trace-detail-selectors' into milestone1/issue-…
tiffon Aug 22, 2017
762b1b1
Merge branch 'span-detail-reorg' into milestone1/issue-60-react-virtu…
tiffon Aug 22, 2017
0ed6fde
WIP react-virtualized layout progress
tiffon Aug 22, 2017
284c816
WIP react-virtualized layout progress
tiffon Aug 23, 2017
0113985
Merge branch 'master' into span-detail-reorg
tiffon Aug 24, 2017
5c7f896
PR comment - Flow typing for props
tiffon Aug 24, 2017
650c60b
Merge branch 'span-detail-reorg' into milestone1/issue-60-react-virtu…
tiffon Aug 24, 2017
d1fd592
WIP Switching react-virtualized to custom solution
tiffon Aug 28, 2017
831aa9c
fix file-header licenses
tiffon Aug 28, 2017
3eeec40
WIP Switching react-virtualized to custom solution
tiffon Aug 28, 2017
c749135
Merge branch 'master' into milestone1/issue-60-react-virtualized
tiffon Aug 28, 2017
11fa23e
Add flow to ListView, comment ListView, Positions
tiffon Aug 29, 2017
c88e723
Positions tests, fix edge-case bug in Positions
tiffon Aug 29, 2017
f8dd6b8
Fix comment issue from prettier
tiffon Aug 29, 2017
cd45e62
Unit tests for ListView, remove unused propType/*
tiffon Aug 30, 2017
f45d360
Minor changes to a comment and a test case description
tiffon Aug 30, 2017
ff2ae49
Unit tests for TraceTimelineViewer/duck
tiffon Aug 31, 2017
3c8057e
remove react-virtualized from package.json
tiffon Sep 1, 2017
6059fed
PR feedback - JSDoc, move trace transform
tiffon Sep 5, 2017
ff779db
Use top-level redux store for traceTimeline state
tiffon Sep 8, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
115 changes: 58 additions & 57 deletions src/components/SearchTracePage/TraceResultsScatterPlot.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,77 +18,78 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.

import React, { Component } from 'react';
import React from 'react';
import moment from 'moment';
import PropTypes from 'prop-types';
import dimensions from 'react-dimensions';
import { XYPlot, XAxis, YAxis, MarkSeries, Hint } from 'react-vis';
import { compose, withState, withProps } from 'recompose';

import { formatDuration } from '../../utils/date';

import './react-vis.css';
import './TraceResultsScatterPlot.css';

class TraceResultsScatterPlot extends Component {
constructor(props) {
super(props);
this.state = {
overValue: null,
};
this.onValueOver = this.onValueOver.bind(this);
this.onValueOut = this.onValueOut.bind(this);
}

onValueOver(data) {
this.setState({ overValue: data });
}

onValueOut() {
this.setState({ overValue: null });
}

render() {
const { data, containerWidth, onValueClick } = this.props;
const { overValue } = this.state;
return (
<div className="TraceResultsScatterPlot">
<XYPlot
margin={{
left: 50,
}}
width={containerWidth}
height={200}
>
<XAxis title="Time" tickTotal={4} tickFormat={t => moment(t).format('hh:mm:ss a')} />
<YAxis title="Duration" tickTotal={3} tickFormat={t => formatDuration(t, 'milliseconds')} />
<MarkSeries
sizeRange={[3, 10]}
opacity={0.5}
onValueClick={onValueClick}
onValueMouseOver={this.onValueOver}
onValueMouseOut={this.onValueOut}
data={data}
/>
{overValue &&
<Hint value={overValue}>
<h4 className="scatter-plot-hint">
{overValue.name || '¯\\_(ツ)_/¯'}
</h4>
</Hint>}
</XYPlot>
</div>
);
}
function TraceResultsScatterPlotBase(props) {
const { data, containerWidth, onValueClick, overValue, onValueOver, onValueOut } = props;
return (
<div className="TraceResultsScatterPlot">
<XYPlot
margin={{
left: 50,
}}
width={containerWidth}
height={200}
>
<XAxis title="Time" tickTotal={4} tickFormat={t => moment(t).format('hh:mm:ss a')} />
<YAxis title="Duration" tickTotal={3} tickFormat={t => formatDuration(t, 'milliseconds')} />
<MarkSeries
sizeRange={[3, 10]}
opacity={0.5}
onValueClick={onValueClick}
onValueMouseOver={onValueOver}
onValueMouseOut={onValueOut}
data={data}
/>
{overValue &&
<Hint value={overValue}>
<h4 className="scatter-plot-hint">
{overValue.name || '¯\\_(ツ)_/¯'}
</h4>
</Hint>}
</XYPlot>
</div>
);
}

export default dimensions()(TraceResultsScatterPlot);
const valueShape = PropTypes.shape({
x: PropTypes.number,
y: PropTypes.number,
traceID: PropTypes.string,
size: PropTypes.number,
name: PropTypes.string,
});

TraceResultsScatterPlot.propTypes = {
data: PropTypes.array.isRequired,
TraceResultsScatterPlotBase.propTypes = {
containerWidth: PropTypes.number,
onValueClick: PropTypes.func,
data: PropTypes.arrayOf(valueShape).isRequired,
overValue: valueShape,
onValueClick: PropTypes.func.isRequired,
onValueOut: PropTypes.func.isRequired,
onValueOver: PropTypes.func.isRequired,
};

TraceResultsScatterPlot.defaultProps = {
data: [],
TraceResultsScatterPlotBase.defaultProps = {
containerWidth: null,
overValue: null,
};

const TraceResultsScatterPlot = compose(
withState('overValue', 'setOverValue', null),
withProps(({ setOverValue }) => ({
onValueOver: value => setOverValue(value),
onValueOut: () => setOverValue(null),
}))
)(TraceResultsScatterPlotBase);

export default dimensions()(TraceResultsScatterPlot);
8 changes: 4 additions & 4 deletions src/components/TracePage/TracePageHeader.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,12 +28,12 @@ export const HEADER_ITEMS = [
{
key: 'timestamp',
title: 'Trace Start',
renderer: props => formatDatetime(props.timestampMs * 1000),
renderer: props => formatDatetime(props.timestamp),
},
{
key: 'duration',
title: 'Duration',
renderer: props => formatDuration(props.durationMs * 1000),
renderer: props => formatDuration(props.duration),
},
{
key: 'service-count',
Expand Down Expand Up @@ -120,8 +120,8 @@ TracePageHeader.propTypes = {
maxDepth: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
numServices: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
numSpans: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
durationMs: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
timestampMs: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
duration: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
timestamp: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
slimView: PropTypes.bool,
onSlimViewClicked: PropTypes.func,
};
Expand Down
8 changes: 0 additions & 8 deletions src/components/TracePage/TracePageHeader.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,14 +28,6 @@ describe('<TracePageHeader>', () => {
const defaultProps = {
traceID: 'some-trace-id',
name: 'some-trace-name',

// maxDepth: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
// numServices: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
// numSpans: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
// durationMs: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
// timestampMs: PropTypes.number, // eslint-disable-line react/no-unused-prop-types
// slimView: PropTypes.bool,
// onSlimViewClicked: PropTypes.func,
};

const defaultOptions = {
Expand Down
12 changes: 5 additions & 7 deletions src/components/TracePage/TraceTimelineViewer/SpanBar.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ function toPercent(value) {
return `${value * 100}%`;
}

function SpanBarImpl(props) {
function SpanBar(props) {
const { viewEnd, viewStart, color, label, hintSide, onClick, onMouseOver, onMouseOut, rpc } = props;

return (
Expand All @@ -55,7 +55,7 @@ function SpanBarImpl(props) {
);
}

SpanBarImpl.propTypes = {
SpanBar.propTypes = {
rpc: PropTypes.shape({
viewStart: PropTypes.number,
viewEnd: PropTypes.number,
Expand All @@ -71,20 +71,18 @@ SpanBarImpl.propTypes = {
onMouseOut: PropTypes.func,
};

SpanBarImpl.defaultProps = {
SpanBar.defaultProps = {
rpc: null,
onClick: null,
onMouseOver: null,
onMouseOut: null,
};

const SpanBar = compose(
export default compose(
withState('label', 'setLabel', props => props.shortLabel),
withProps(({ setLabel, shortLabel, longLabel }) => ({
onMouseOver: () => setLabel(longLabel),
onMouseOut: () => setLabel(shortLabel),
})),
onlyUpdateForKeys(['viewStart', 'viewEnd', 'label', 'rpc'])
)(SpanBarImpl);

export default SpanBar;
)(SpanBar);
5 changes: 5 additions & 0 deletions src/components/TracePage/TraceTimelineViewer/SpanBarRow.css
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,11 @@ THE SOFTWARE.
z-index: 1;
}

.span-name-wrapper {
overflow: hidden;
text-overflow: ellipsis;
}

.span-name {
outline: none;
color: #000;
Expand Down
2 changes: 1 addition & 1 deletion src/components/TracePage/TraceTimelineViewer/SpanBarRow.js
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,7 @@ export default function SpanBarRow(props) {
`}
>
<TimelineRow.Left className="span-name-column">
<div className="overflow-hidden" title={title}>
<div className="span-name-wrapper" title={title}>
<SpanTreeOffset
level={depth + 1}
hasChildren={isParent}
Expand Down
2 changes: 1 addition & 1 deletion src/components/TracePage/TraceTimelineViewer/TraceView.js
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ export default function TraceView(props) {
<div className="mx0 px1 overflow-hidden">
<TimelineRow style={{ backgroundColor: '#e8e8e8', borderBottom: '1px solid #ccc' }}>
<TimelineRow.Left>
<h3 className="m0 p1">Span Name</h3>
<h3 className="m0 p1">Service & Operation</h3>
</TimelineRow.Left>
<TimelineRow.Right>
<Ticks
Expand Down
4 changes: 2 additions & 2 deletions src/components/TracePage/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -164,13 +164,13 @@ export default class TracePage extends Component {
<div className="trace-page" id={`jaeger-trace-${id}`}>
<section className="trace-page-header-section" ref={this.setHeaderHeight}>
<TracePageHeader
durationMs={duration / 1000}
duration={duration}
maxDepth={maxSpanDepth}
name={getTraceName(spans, processes)}
numServices={numberOfServices}
numSpans={spans.length}
slimView={slimView}
timestampMs={startTime / 1000}
timestamp={startTime}
traceID={traceID}
onSlimViewClicked={this.toggleSlimView}
/>
Expand Down
14 changes: 14 additions & 0 deletions src/selectors/trace.js
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,20 @@ export const getTraceSpansAsMap = createSelector(getTraceSpans, spans =>

export const TREE_ROOT_ID = '__root__';

/**
* Build a tree of { value: spanID, children } items derived from the
* `span.references` information. The tree represents the grouping of parent /
* child relationships. The root-most node is nominal in that
* `.value === TREE_ROOT_ID`. This is done because a root span (the main trace
* span) is not always included with the trace data. Thus, there can be
* multiple top-level spans, and the root node acts as their common parent.
*
* The children are sorted by `span.startTime` after the tree is built.
*
* @param {Trace} trace The trace to build the tree of spanIDs.
* @return {TreeNode} A tree of spanIDs derived from the relationships
* between spans in the trace.
*/
export function getTraceSpanIdsAsTree(trace) {
const nodesById = new Map(trace.spans.map(span => [span.spanID, new TreeNode(span.spanID)]));
const spansById = new Map(trace.spans.map(span => [span.spanID, span]));
Expand Down