From 7f903efb078280ed31e17a837e330a244cd5b459 Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Thu, 28 Oct 2021 11:36:37 +0300 Subject: [PATCH 1/5] Project: open correct web address when starting UI in development mode (npm start) When UI is started in development mode, browser window is opened to 0.0.0.0:8000, instead it should be local-ip:8000. This commit fixes this problem --- monkey/monkey_island/cc/ui/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/monkey/monkey_island/cc/ui/package.json b/monkey/monkey_island/cc/ui/package.json index da6200c25..f7c86151b 100644 --- a/monkey/monkey_island/cc/ui/package.json +++ b/monkey/monkey_island/cc/ui/package.json @@ -15,7 +15,7 @@ "release:patch": "npm version patch && npm publish && git push --follow-tags", "serve": "node server.js --env=dev", "serve:dist": "node server.js --env=dist", - "start": "webpack-dev-server --mode development --open --history-api-fallback --port 8000", + "start": "webpack-dev-server --mode development --open --history-api-fallback --port 8000 --host local-ip", "snyk-protect": "snyk protect", "prepare": "npm run snyk-protect" }, From 7425bf1bbd0c389f90958a651cd0318d570a3ae9 Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Thu, 28 Oct 2021 14:37:11 +0300 Subject: [PATCH 2/5] UI: refactor TelemetryLog component out of MapPage and migrate it to hook with typescript --- .../cc/ui/src/components/map/TelemetryLog.tsx | 96 +++++++++++++++++++ .../cc/ui/src/components/pages/MapPage.js | 89 +---------------- 2 files changed, 100 insertions(+), 85 deletions(-) create mode 100644 monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx diff --git a/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx new file mode 100644 index 000000000..f247bbf91 --- /dev/null +++ b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx @@ -0,0 +1,96 @@ +import React, {useEffect, useRef, useState} from 'react'; +import AuthComponent from '../AuthComponent'; + + +const authComponent = new AuthComponent({}); + +const TelemetryLog = (props: { onStatusChange: Function }) => { + + let [telemetryUpdateInProgress, setTelemetryUpdateInProgress] = useState(false); + let [telemetry, setTelemetry] = useState([]); + let [lastTelemetryTimestamp, setLastTelemetryTimestamp] = useState(null); + let [isScrolledUp, setIsScrolledUp] = useState(false); + let [telemetryLines, setTelemetryLines] = useState(0); + let [telemetryCurrentLine, setTelemetryCurrentLine] = useState(0); + + let scrollTop = 0; + const telemetryConsole = useRef(null); + + useEffect(() => { + const interval = setInterval(updateTelemetryFromServer, 5000); + return function cleanup() { + clearInterval(interval); + }; + }, []); + + function updateTelemetryFromServer() { + if (telemetryUpdateInProgress) { + return + } + setTelemetryUpdateInProgress(true); + authComponent.authFetch('/api/telemetry-feed?timestamp=' + lastTelemetryTimestamp) + .then(res => res.json()) + .then(res => { + if ('telemetries' in res) { + let newTelem = telemetry.concat(res['telemetries']); + setTelemetry(newTelem); + setLastTelemetryTimestamp(res['timestamp']); + setTelemetryUpdateInProgress(false); + props.onStatusChange(); + + let telemConsoleRef = telemetryConsole.current; + if (!isScrolledUp) { + telemConsoleRef.scrollTop = telemConsoleRef.scrollHeight - telemConsoleRef.clientHeight; + scrollTop = telemConsoleRef.scrollTop; + } + } + }); + } + + function handleScroll(e) { + let element = e.target; + + let telemetryStyle = window.getComputedStyle(element); + let telemetryLineHeight = parseInt((telemetryStyle.lineHeight).replace('px', '')); + + setIsScrolledUp((element.scrollTop < scrollTop)); + setTelemetryCurrentLine(Math.trunc(element.scrollTop / telemetryLineHeight) + 1); + setTelemetryLines(Math.trunc(element.scrollHeight / telemetryLineHeight)); + } + + function renderTelemetryConsole() { + return ( +
+ { + telemetry.map(renderTelemetryEntry) + } +
+ ); + } + + function renderTelemetryEntry(telemetry) { + return ( +
+ {telemetry.timestamp} + {telemetry.hostname}: + {telemetry.brief} +
+ ); + } + + function renderTelemetryLineCount() { + return ( +
+ [{telemetryCurrentLine}/{telemetryLines}] +
+ ); + } + + return ( + <> + {renderTelemetryLineCount()} + {renderTelemetryConsole()} + ); +} + +export default TelemetryLog; diff --git a/monkey/monkey_island/cc/ui/src/components/pages/MapPage.js b/monkey/monkey_island/cc/ui/src/components/pages/MapPage.js index da11c7ed6..6026cebb6 100644 --- a/monkey/monkey_island/cc/ui/src/components/pages/MapPage.js +++ b/monkey/monkey_island/cc/ui/src/components/pages/MapPage.js @@ -10,6 +10,7 @@ import {getOptions, edgeGroupToColor} from 'components/map/MapOptions'; import AuthComponent from '../AuthComponent'; import '../../styles/components/Map.scss'; import {faInfoCircle} from '@fortawesome/free-solid-svg-icons/faInfoCircle'; +import TelemetryLog from '../map/TelemetryLog'; class MapPageComponent extends AuthComponent { constructor(props) { @@ -20,17 +21,8 @@ class MapPageComponent extends AuthComponent { selected: null, selectedType: null, killPressed: false, - showKillDialog: false, - telemetry: [], - telemetryLastTimestamp: null, - isScrolledUp: false, - telemetryLines: 0, - telemetryCurrentLine: 0, - telemetryUpdateInProgress: false + showKillDialog: false }; - this.telemConsole = React.createRef(); - this.handleScroll = this.handleScroll.bind(this); - this.scrollTop = 0; } events = { @@ -40,7 +32,7 @@ class MapPageComponent extends AuthComponent { componentDidMount() { this.getNodeStateListFromServer(); this.updateMapFromServer(); - this.interval = setInterval(this.timedEvents, 5000); + this.interval = setInterval(this.updateMapFromServer, 5000); } componentWillUnmount() { @@ -55,11 +47,6 @@ class MapPageComponent extends AuthComponent { }); }; - timedEvents = () => { - this.updateMapFromServer(); - this.updateTelemetryFromServer(); - }; - updateMapFromServer = () => { this.authFetch('/api/netmap') .then(res => res.json()) @@ -74,33 +61,6 @@ class MapPageComponent extends AuthComponent { }); }; - updateTelemetryFromServer = () => { - if (this.state.telemetryUpdateInProgress) { - return - } - this.setState({telemetryUpdateInProgress: true}); - this.authFetch('/api/telemetry-feed?timestamp=' + this.state.telemetryLastTimestamp) - .then(res => res.json()) - .then(res => { - if ('telemetries' in res) { - let newTelem = this.state.telemetry.concat(res['telemetries']); - this.setState( - { - telemetry: newTelem, - telemetryLastTimestamp: res['timestamp'], - telemetryUpdateInProgress: false - }); - this.props.onStatusChange(); - - let telemConsoleRef = this.telemConsole.current; - if (!this.state.isScrolledUp) { - telemConsoleRef.scrollTop = telemConsoleRef.scrollHeight - telemConsoleRef.clientHeight; - this.scrollTop = telemConsoleRef.scrollTop; - } - } - }); - }; - selectionChanged(event) { if (event.nodes.length === 1) { this.authFetch('/api/netmap/node?id=' + event.nodes[0]) @@ -157,46 +117,6 @@ class MapPageComponent extends AuthComponent { ) }; - renderTelemetryEntry(telemetry) { - return ( -
- {telemetry.timestamp} - {telemetry.hostname}: - {telemetry.brief} -
- ); - } - - handleScroll(e) { - let element = e.target; - - let telemetryStyle = window.getComputedStyle(element); - let telemetryLineHeight = parseInt((telemetryStyle.lineHeight).replace('px', '')); - - this.setState({ - isScrolledUp: (element.scrollTop < this.scrollTop), - telemetryCurrentLine: Math.trunc(element.scrollTop / telemetryLineHeight) + 1, - telemetryLines: Math.trunc(element.scrollHeight / telemetryLineHeight) - }); - } - - renderTelemetryConsole() { - return ( -
- { - this.state.telemetry.map(this.renderTelemetryEntry) - } -
- ); - } - - renderTelemetryLineCount() { - return ( -
- [{this.state.telemetryCurrentLine}/{this.state.telemetryLines}] -
- ); - } render() { return ( @@ -220,10 +140,9 @@ class MapPageComponent extends AuthComponent { Island Communication
- {this.renderTelemetryLineCount()} - {this.renderTelemetryConsole()} +
From 97327f08a24ce8530db623c2f6f24e6bbf7a7ce6 Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Thu, 28 Oct 2021 16:21:26 +0300 Subject: [PATCH 3/5] UI: add loading icon to TelemetryLog.tsx --- .../cc/ui/src/components/map/TelemetryLog.tsx | 8 ++++++-- monkey/monkey_island/cc/ui/src/styles/components/Map.scss | 7 +++++++ 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx index f247bbf91..dfbcad8ff 100644 --- a/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx +++ b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx @@ -1,11 +1,13 @@ import React, {useEffect, useRef, useState} from 'react'; import AuthComponent from '../AuthComponent'; +import LoadingIcon from '../ui-components/LoadingIcon'; const authComponent = new AuthComponent({}); const TelemetryLog = (props: { onStatusChange: Function }) => { + let [telemetriesLoading, setTelemetriesLoading] = useState(true); let [telemetryUpdateInProgress, setTelemetryUpdateInProgress] = useState(false); let [telemetry, setTelemetry] = useState([]); let [lastTelemetryTimestamp, setLastTelemetryTimestamp] = useState(null); @@ -36,6 +38,7 @@ const TelemetryLog = (props: { onStatusChange: Function }) => { setTelemetry(newTelem); setLastTelemetryTimestamp(res['timestamp']); setTelemetryUpdateInProgress(false); + setTelemetriesLoading(false); props.onStatusChange(); let telemConsoleRef = telemetryConsole.current; @@ -87,10 +90,11 @@ const TelemetryLog = (props: { onStatusChange: Function }) => { } return ( - <> +
+ {telemetriesLoading && } {renderTelemetryLineCount()} {renderTelemetryConsole()} - ); +
); } export default TelemetryLog; diff --git a/monkey/monkey_island/cc/ui/src/styles/components/Map.scss b/monkey/monkey_island/cc/ui/src/styles/components/Map.scss index ebeb7c687..adbfe8b05 100644 --- a/monkey/monkey_island/cc/ui/src/styles/components/Map.scss +++ b/monkey/monkey_island/cc/ui/src/styles/components/Map.scss @@ -46,3 +46,10 @@ width: 100%; padding-bottom: 10px; } + +.telemetry-log-section .loading-icon{ + position: relative; + color: white; + z-index: 5; + margin-top: 30px; +} From 063286e899acce2d55c308ad749788c68861d932 Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Thu, 28 Oct 2021 16:22:24 +0300 Subject: [PATCH 4/5] UI: fix long telemetry log loading bug --- .../monkey_island/cc/ui/src/components/map/TelemetryLog.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx index dfbcad8ff..45e303a9c 100644 --- a/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx +++ b/monkey/monkey_island/cc/ui/src/components/map/TelemetryLog.tsx @@ -4,6 +4,7 @@ import LoadingIcon from '../ui-components/LoadingIcon'; const authComponent = new AuthComponent({}); +const telemetryUpdatePeriod = 5000; // UI will fetch telemetries every N milliseconds. const TelemetryLog = (props: { onStatusChange: Function }) => { @@ -19,7 +20,8 @@ const TelemetryLog = (props: { onStatusChange: Function }) => { const telemetryConsole = useRef(null); useEffect(() => { - const interval = setInterval(updateTelemetryFromServer, 5000); + updateTelemetryFromServer(); + const interval = setInterval(updateTelemetryFromServer, telemetryUpdatePeriod); return function cleanup() { clearInterval(interval); }; From 5cfe6de9270d6cbe71da5a6ab4f58363b5520c0c Mon Sep 17 00:00:00 2001 From: VakarisZ Date: Thu, 28 Oct 2021 16:31:18 +0300 Subject: [PATCH 5/5] Changelog: add entry for fixed #1545 --- CHANGELOG.md | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 295fc9442..18a655320 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,19 @@ file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/). +## [Unreleased] +### Added + +### Changed + +### Removed + +### Fixed +- A bug in network map page that caused delay of telemetry log loading. #1545 + +### Security + + ## [1.12.0] - 2021-10-27 ### Added - A new exploiter that allows propagation via PowerShell Remoting. #1246