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 all 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
22 changes: 19 additions & 3 deletions .eslintrc
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,13 @@
],
"rules": {
/* general */
"comma-dangle": 0,
"arrow-parens": [1, "as-needed"],
"no-restricted-syntax": 0,
"comma-dangle": 0,
"no-continue": 0,
"no-plusplus": 0,
"no-restricted-syntax": 0,
"no-self-compare": 0,
"no-underscore-dangle": 0,

/* jsx */
"jsx-a11y/no-static-element-interactions": 1,
Expand All @@ -29,7 +32,20 @@
"react/forbid-prop-types": 1,
"react/require-default-props": 1,
"react/no-array-index-key": 1,

"react/sort-comp": [2, {
"order": [
"type-annotations",
"statics",
"state",
"propTypes",
"static-methods",
"constructor",
"lifecycle",
"everything-else",
"/^on.+$/",
"render"
]
}],

/* import */
"import/prefer-default-export": 1,
Expand Down
3 changes: 3 additions & 0 deletions .flowconfig
Original file line number Diff line number Diff line change
Expand Up @@ -10,3 +10,6 @@
[libs]

[options]

[version]
0.53.1
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
"homepage": null,
"devDependencies": {
"babel-eslint": "^7.2.3",
"bluebird": "^3.5.0",
"enzyme": "^2.9.1",
"eslint": "^4.5.0",
"eslint-config-airbnb": "^15.1.0",
Expand Down
64 changes: 28 additions & 36 deletions src/components/TracePage/TraceSpanGraph.js
Original file line number Diff line number Diff line change
Expand Up @@ -25,15 +25,13 @@ import PropTypes from 'prop-types';
import SpanGraph from './SpanGraph';
import SpanGraphTickHeader from './SpanGraph/SpanGraphTickHeader';
import TimelineScrubber from './TimelineScrubber';
import { getTraceId, getTraceTimestamp, getTraceEndTimestamp, getTraceDuration } from '../../selectors/trace';
import { getPercentageOfInterval } from '../../utils/date';

const TIMELINE_TICK_INTERVAL = 4;

export default class TraceSpanGraph extends Component {
static get propTypes() {
return {
xformedTrace: PropTypes.object,
trace: PropTypes.object,
height: PropTypes.number.isRequired,
};
Expand Down Expand Up @@ -71,13 +69,31 @@ export default class TraceSpanGraph extends Component {
const newRightBound = newTimeRangeFilter[1];

return (
getTraceId(trace) !== getTraceId(newTrace) ||
trace.traceID !== newTrace.traceID ||
leftBound !== newLeftBound ||
rightBound !== newRightBound ||
currentlyDragging !== newCurrentlyDragging
);
}

startDragging(boundName, { clientX }) {
this.setState({ currentlyDragging: boundName, prevX: clientX });

const mouseMoveHandler = (...args) => this.onMouseMove(...args);
const mouseUpHandler = () => {
this.stopDragging();
window.removeEventListener('mouseup', mouseUpHandler);
window.removeEventListener('mousemove', mouseMoveHandler);
};

window.addEventListener('mouseup', mouseUpHandler);
window.addEventListener('mousemove', mouseMoveHandler);
}

stopDragging() {
this.setState({ currentlyDragging: null, prevX: null });
}

onMouseMove({ clientX }) {
const { trace } = this.props;
const { prevX, currentlyDragging } = this.state;
Expand All @@ -91,19 +107,16 @@ export default class TraceSpanGraph extends Component {
let rightBound = timeRangeFilter[1];

const deltaX = (clientX - prevX) / this.svg.clientWidth;
const timestamp = getTraceTimestamp(trace);
const endTimestamp = getTraceEndTimestamp(trace);
const duration = getTraceDuration(this.props.trace);
const prevValue = { leftBound, rightBound }[currentlyDragging];
const newValue = prevValue + duration * deltaX;
const newValue = prevValue + trace.duration * deltaX;

// enforce the edges of the graph
switch (currentlyDragging) {
case 'leftBound':
leftBound = Math.max(timestamp, newValue);
leftBound = Math.max(trace.startTime, newValue);
break;
case 'rightBound':
rightBound = Math.min(endTimestamp, newValue);
rightBound = Math.min(trace.endTime, newValue);
break;
/* istanbul ignore next */ default:
break;
Expand All @@ -115,26 +128,8 @@ export default class TraceSpanGraph extends Component {
}
}

startDragging(boundName, { clientX }) {
this.setState({ currentlyDragging: boundName, prevX: clientX });

const mouseMoveHandler = (...args) => this.onMouseMove(...args);
const mouseUpHandler = () => {
this.stopDragging();
window.removeEventListener('mouseup', mouseUpHandler);
window.removeEventListener('mousemove', mouseMoveHandler);
};

window.addEventListener('mouseup', mouseUpHandler);
window.addEventListener('mousemove', mouseMoveHandler);
}

stopDragging() {
this.setState({ currentlyDragging: null, prevX: null });
}

render() {
const { trace, xformedTrace, height } = this.props;
const { trace, height } = this.props;
const { currentlyDragging } = this.state;
const { timeRangeFilter } = this.context;
const leftBound = timeRangeFilter[0];
Expand All @@ -144,23 +139,20 @@ export default class TraceSpanGraph extends Component {
return <div />;
}

const initialTimestamp = getTraceTimestamp(trace);
const traceDuration = getTraceDuration(trace);

let leftInactive;
if (leftBound) {
leftInactive = getPercentageOfInterval(leftBound, initialTimestamp, traceDuration);
leftInactive = getPercentageOfInterval(leftBound, trace.startTime, trace.duration);
}

let rightInactive;
if (rightBound) {
rightInactive = 100 - getPercentageOfInterval(rightBound, initialTimestamp, traceDuration);
rightInactive = 100 - getPercentageOfInterval(rightBound, trace.startTime, trace.duration);
}

return (
<div>
<div className="trace-page-timeline--tick-container">
<SpanGraphTickHeader numTicks={TIMELINE_TICK_INTERVAL} duration={traceDuration} />
<SpanGraphTickHeader numTicks={TIMELINE_TICK_INTERVAL} duration={trace.duration} />
</div>
<div>
<svg
Expand All @@ -187,9 +179,9 @@ export default class TraceSpanGraph extends Component {
className="trace-page-timeline__graph--inactive"
/>}
<SpanGraph
valueWidth={xformedTrace.duration}
valueWidth={trace.duration}
numTicks={TIMELINE_TICK_INTERVAL}
items={xformedTrace.spans.map(span => ({
items={trace.spans.map(span => ({
valueOffset: span.relativeStartTime,
valueWidth: span.duration,
serviceName: span.process.serviceName,
Expand Down
30 changes: 11 additions & 19 deletions src/components/TracePage/TraceSpanGraph.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,21 +27,14 @@ import TraceSpanGraph from './TraceSpanGraph';
import SpanGraphTickHeader from './SpanGraph/SpanGraphTickHeader';
import TimelineScrubber from './TimelineScrubber';
import traceGenerator from '../../../src/demo/trace-generators';
import { transformTrace } from './TraceTimelineViewer/transforms';
import { hydrateSpansWithProcesses } from '../../selectors/trace';
import transformTraceData from '../../model/transform-trace-data';

describe('<TraceSpanGraph>', () => {
const trace = hydrateSpansWithProcesses(traceGenerator.trace({}));
const xformedTrace = transformTrace(trace);

const props = {
trace,
xformedTrace,
};

const trace = transformTraceData(traceGenerator.trace({}));
const props = { trace };
const options = {
context: {
timeRangeFilter: [trace.timestamp, trace.timestamp + trace.duration],
timeRangeFilter: [trace.startTime, trace.startTime + trace.duration],
updateTimeRangeFilter: () => {},
},
};
Expand All @@ -68,7 +61,7 @@ describe('<TraceSpanGraph>', () => {
it('renders a filtering box if leftBound exists', () => {
const context = {
...options.context,
timeRangeFilter: [trace.timestamp + 0.2 * trace.duration, trace.timestamp + trace.duration],
timeRangeFilter: [trace.startTime + 0.2 * trace.duration, trace.startTime + trace.duration],
};
wrapper = shallow(<TraceSpanGraph {...props} />, { ...options, context });
const leftBox = wrapper.find('.trace-page-timeline__graph--inactive');
Expand All @@ -82,7 +75,7 @@ describe('<TraceSpanGraph>', () => {
it('renders a filtering box if rightBound exists', () => {
const context = {
...options.context,
timeRangeFilter: [trace.timestamp, trace.timestamp + 0.8 * trace.duration],
timeRangeFilter: [trace.startTime, trace.startTime + 0.8 * trace.duration],
};
wrapper = shallow(<TraceSpanGraph {...props} />, { ...options, context });
const rightBox = wrapper.find('.trace-page-timeline__graph--inactive');
Expand Down Expand Up @@ -132,7 +125,7 @@ describe('<TraceSpanGraph>', () => {

it('passes items to SpanGraph', () => {
const spanGraph = wrapper.find(SpanGraph).first();
const items = xformedTrace.spans.map(span => ({
const items = trace.spans.map(span => ({
valueOffset: span.relativeStartTime,
valueWidth: span.duration,
serviceName: span.process.serviceName,
Expand All @@ -151,9 +144,8 @@ describe('<TraceSpanGraph>', () => {
it('returns true for new trace', () => {
const state = wrapper.state();
const instance = wrapper.instance();
const trace2 = hydrateSpansWithProcesses(traceGenerator.trace({}));
const xformedTrace2 = transformTrace(trace2);
const altProps = { trace: trace2, xformedTrace: xformedTrace2 };
const trace2 = transformTraceData(traceGenerator.trace({}));
const altProps = { trace: trace2 };
expect(instance.shouldComponentUpdate(altProps, state, options.context)).toBe(true);
});

Expand Down Expand Up @@ -190,7 +182,7 @@ describe('<TraceSpanGraph>', () => {
});

it('updates the timeRangeFilter for the left handle', () => {
const timestamp = trace.timestamp;
const timestamp = trace.startTime;
const duration = trace.duration;
const updateTimeRangeFilter = sinon.spy();
const context = { ...options.context, updateTimeRangeFilter };
Expand All @@ -204,7 +196,7 @@ describe('<TraceSpanGraph>', () => {
});

it('updates the timeRangeFilter for the right handle', () => {
const timestamp = trace.timestamp;
const timestamp = trace.startTime;
const duration = trace.duration;
const updateTimeRangeFilter = sinon.spy();
const context = { ...options.context, updateTimeRangeFilter };
Expand Down
Loading