From 350c7d93fa67de6d94b88db2e2f5f32efae08972 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Fri, 7 Jun 2019 11:51:47 +0300
Subject: [PATCH 01/12] T1003 credential dumping implemented
---
.../cc/services/attack/attack_report.py | 5 ++--
.../attack/technique_reports/T1003.py | 25 +++++++++++++++++++
.../src/components/attack/techniques/T1003.js | 24 ++++++++++++++++++
.../cc/ui/src/components/pages/ReportPage.js | 2 +-
.../report-components/AttackReport.js | 6 +++--
5 files changed, 57 insertions(+), 5 deletions(-)
create mode 100644 monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
create mode 100644 monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
diff --git a/monkey/monkey_island/cc/services/attack/attack_report.py b/monkey/monkey_island/cc/services/attack/attack_report.py
index 49d51ffe5..b1079d8c7 100644
--- a/monkey/monkey_island/cc/services/attack/attack_report.py
+++ b/monkey/monkey_island/cc/services/attack/attack_report.py
@@ -1,5 +1,5 @@
import logging
-from monkey_island.cc.services.attack.technique_reports import T1210, T1197, T1110, T1075
+from monkey_island.cc.services.attack.technique_reports import T1210, T1197, T1110, T1075, T1003
from monkey_island.cc.services.attack.attack_telem import AttackTelemService
from monkey_island.cc.services.attack.attack_config import AttackConfig
from monkey_island.cc.database import mongo
@@ -12,7 +12,8 @@ LOG = logging.getLogger(__name__)
TECHNIQUES = {'T1210': T1210.T1210,
'T1197': T1197.T1197,
'T1110': T1110.T1110,
- 'T1075': T1075.T1075}
+ 'T1075': T1075.T1075,
+ 'T1003': T1003.T1003}
REPORT_NAME = 'new_report'
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
new file mode 100644
index 000000000..002e2eeb7
--- /dev/null
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
@@ -0,0 +1,25 @@
+from monkey_island.cc.services.attack.technique_reports import AttackTechnique
+from common.utils.attack_utils import ScanStatus
+from monkey_island.cc.database import mongo
+
+__author__ = "VakarisZ"
+
+
+class T1003(AttackTechnique):
+
+ tech_id = "T1003"
+ unscanned_msg = "Monkey tried to obtain credentials from systems in the network but didn't find any or failed."
+ scanned_msg = "Monkey tried to obtain credentials from systems in the network but didn't find any or failed."
+ used_msg = "Monkey successfully obtained some credentials from systems on the network."
+
+ query = {'telem_type': 'system_info_collection', '$and': [{'data.credentials': {'$exists': True}},
+ {'data.credentials': {'$gt': {}}}]}
+
+ @staticmethod
+ def get_report_data():
+ data = {'title': T1003.technique_title(T1003.tech_id)}
+ if mongo.db.telemetry.count_documents(T1003.query):
+ data.update({'message': T1003.used_msg, 'status': ScanStatus.USED.name})
+ else:
+ data.update({'message': T1003.unscanned_msg, 'status': ScanStatus.UNSCANNED.name})
+ return data
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
new file mode 100644
index 000000000..1ec0b000e
--- /dev/null
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
@@ -0,0 +1,24 @@
+import React from 'react';
+import '../../../styles/Collapse.scss'
+import '../../report-components/StolenPasswords'
+import StolenPasswordsComponent from "../../report-components/StolenPasswords";
+
+
+class T1003 extends React.Component {
+
+ constructor(props) {
+ super(props);
+ }
+
+ render() {
+ return (
+
+
{this.props.data.message}
+
+
+
+ );
+ }
+}
+
+export default T1003;
diff --git a/monkey/monkey_island/cc/ui/src/components/pages/ReportPage.js b/monkey/monkey_island/cc/ui/src/components/pages/ReportPage.js
index 72aeca574..3a21b721e 100644
--- a/monkey/monkey_island/cc/ui/src/components/pages/ReportPage.js
+++ b/monkey/monkey_island/cc/ui/src/components/pages/ReportPage.js
@@ -520,7 +520,7 @@ class ReportPageComponent extends AuthComponent {
This report shows information about ATT&CK techniques used by Infection Monkey.
)
diff --git a/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js b/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
index 58f8ea4d7..b10b1ff7d 100644
--- a/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
+++ b/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
@@ -9,12 +9,14 @@ import T1210 from '../attack/techniques/T1210';
import T1197 from '../attack/techniques/T1197';
import T1110 from '../attack/techniques/T1110';
import T1075 from "../attack/techniques/T1075";
+import T1003 from "../attack/techniques/T1003";
const tech_components = {
'T1210': T1210,
'T1197': T1197,
'T1110': T1110,
- 'T1075': T1075
+ 'T1075': T1075,
+ 'T1003': T1003
};
const classNames = require('classnames');
@@ -101,7 +103,7 @@ class AttackReportPageComponent extends AuthComponent {
const TechniqueComponent = tech_components[technique];
return (
-
+
);
}
From c99ceff21df431e588461efe55c4dbf27b055de2 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Fri, 7 Jun 2019 17:00:08 +0300
Subject: [PATCH 02/12] Table not shown if no passwords were stolen
---
.../cc/ui/src/components/attack/techniques/T1003.js | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
index 1ec0b000e..d7783714a 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1003.js
@@ -15,7 +15,9 @@ class T1003 extends React.Component {
{this.props.data.message}
-
+ {this.props.data.status === 'USED' ?
+
+ : ""}
);
}
From 9b08e606f8e23ec3bb3a77d5997353a5f9f9b72e Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Mon, 10 Jun 2019 08:46:35 +0300
Subject: [PATCH 03/12] CLI implementation started
---
monkey/infection_monkey/exploit/__init__.py | 3 +-
monkey/infection_monkey/exploit/hadoop.py | 1 +
monkey/infection_monkey/exploit/mssqlexec.py | 2 +-
monkey/infection_monkey/exploit/rdpgrinder.py | 2 +-
monkey/infection_monkey/exploit/shellshock.py | 1 +
monkey/infection_monkey/exploit/sshexec.py | 1 +
monkey/infection_monkey/exploit/vsftpd.py | 1 +
monkey/infection_monkey/exploit/web_rce.py | 2 +
.../infection_monkey/exploit/win_ms08_067.py | 5 ++-
.../cc/services/attack/attack_report.py | 5 ++-
.../cc/services/attack/attack_schema.py | 14 +++++++
.../attack/technique_reports/T1059.py | 30 +++++++++++++++
.../src/components/attack/techniques/T1059.js | 38 +++++++++++++++++++
.../src/components/attack/techniques/T1075.js | 4 +-
14 files changed, 100 insertions(+), 9 deletions(-)
create mode 100644 monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
create mode 100644 monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
diff --git a/monkey/infection_monkey/exploit/__init__.py b/monkey/infection_monkey/exploit/__init__.py
index 7353d77bc..84dd81393 100644
--- a/monkey/infection_monkey/exploit/__init__.py
+++ b/monkey/infection_monkey/exploit/__init__.py
@@ -24,7 +24,8 @@ class HostExploiter(object):
'started': '',
'finished': '',
'vulnerable_urls': [],
- 'vulnerable_ports': []}
+ 'vulnerable_ports': [],
+ 'executed_cmds': []}
self._exploit_attempts = []
self.host = host
diff --git a/monkey/infection_monkey/exploit/hadoop.py b/monkey/infection_monkey/exploit/hadoop.py
index f02c4f3d3..1e8ac9d8e 100644
--- a/monkey/infection_monkey/exploit/hadoop.py
+++ b/monkey/infection_monkey/exploit/hadoop.py
@@ -49,6 +49,7 @@ class HadoopExploiter(WebRCE):
return False
http_thread.join(self.DOWNLOAD_TIMEOUT)
http_thread.stop()
+ self._exploit_info['executed_cmds'].append(command)
return True
def exploit(self, url, command):
diff --git a/monkey/infection_monkey/exploit/mssqlexec.py b/monkey/infection_monkey/exploit/mssqlexec.py
index c7d29c8c2..1f21f9ecd 100644
--- a/monkey/infection_monkey/exploit/mssqlexec.py
+++ b/monkey/infection_monkey/exploit/mssqlexec.py
@@ -76,7 +76,7 @@ class MSSQLExploiter(HostExploiter):
commands.extend(monkey_args)
MSSQLExploiter.execute_command(cursor, commands)
MSSQLExploiter.run_file(cursor, tmp_file_path)
-
+ self._exploit_info['executed_cmds'].append(commands[-1])
return True
@staticmethod
diff --git a/monkey/infection_monkey/exploit/rdpgrinder.py b/monkey/infection_monkey/exploit/rdpgrinder.py
index 8e219b5c8..dcbc01b1d 100644
--- a/monkey/infection_monkey/exploit/rdpgrinder.py
+++ b/monkey/infection_monkey/exploit/rdpgrinder.py
@@ -343,5 +343,5 @@ class RdpExploiter(HostExploiter):
LOG.info("Executed monkey '%s' on remote victim %r",
os.path.basename(src_path), self.host)
-
+ self._exploit_info['executed_cmds'].append(command)
return True
diff --git a/monkey/infection_monkey/exploit/shellshock.py b/monkey/infection_monkey/exploit/shellshock.py
index 2f6e3516f..b308c57fe 100644
--- a/monkey/infection_monkey/exploit/shellshock.py
+++ b/monkey/infection_monkey/exploit/shellshock.py
@@ -144,6 +144,7 @@ class ShellShockExploiter(HostExploiter):
if not (self.check_remote_file_exists(url, header, exploit, self._config.monkey_log_path_linux)):
LOG.info("Log file does not exist, monkey might not have run")
continue
+ self._exploit_info['executed_cmds'].append(cmdline)
return True
return False
diff --git a/monkey/infection_monkey/exploit/sshexec.py b/monkey/infection_monkey/exploit/sshexec.py
index 09982876d..9aafae2a4 100644
--- a/monkey/infection_monkey/exploit/sshexec.py
+++ b/monkey/infection_monkey/exploit/sshexec.py
@@ -178,6 +178,7 @@ class SSHExploiter(HostExploiter):
self._config.dropper_target_path_linux, self.host, cmdline)
ssh.close()
+ self._exploit_info['executed_cmds'].append(cmdline)
return True
except Exception as exc:
diff --git a/monkey/infection_monkey/exploit/vsftpd.py b/monkey/infection_monkey/exploit/vsftpd.py
index 3f6a7c304..23a89a96e 100644
--- a/monkey/infection_monkey/exploit/vsftpd.py
+++ b/monkey/infection_monkey/exploit/vsftpd.py
@@ -138,6 +138,7 @@ class VSFTPDExploiter(HostExploiter):
if backdoor_socket.send(run_monkey):
LOG.info("Executed monkey '%s' on remote victim %r (cmdline=%r)", self._config.dropper_target_path_linux,
self.host, run_monkey)
+ self._exploit_info['executed_cmds'].append(run_monkey)
return True
else:
return False
diff --git a/monkey/infection_monkey/exploit/web_rce.py b/monkey/infection_monkey/exploit/web_rce.py
index 2b08575c3..57a4cfa60 100644
--- a/monkey/infection_monkey/exploit/web_rce.py
+++ b/monkey/infection_monkey/exploit/web_rce.py
@@ -419,6 +419,8 @@ class WebRCE(HostExploiter):
LOG.error("Something went wrong when trying to execute remote monkey: %s" % e)
return False
LOG.info("Execution attempt finished")
+
+ self._exploit_info['executed_cmds'].append(command)
return resp
def get_monkey_upload_path(self, url_to_monkey):
diff --git a/monkey/infection_monkey/exploit/win_ms08_067.py b/monkey/infection_monkey/exploit/win_ms08_067.py
index 41b3820d5..72fca6d1d 100644
--- a/monkey/infection_monkey/exploit/win_ms08_067.py
+++ b/monkey/infection_monkey/exploit/win_ms08_067.py
@@ -92,7 +92,7 @@ class SRVSVC_Exploit(object):
def get_telnet_port(self):
"""get_telnet_port()
-
+
The port on which the Telnet service will listen.
"""
@@ -100,7 +100,7 @@ class SRVSVC_Exploit(object):
def start(self):
"""start() -> socket
-
+
Exploit the target machine and return a socket connected to it's
listening Telnet service.
"""
@@ -153,6 +153,7 @@ class SRVSVC_Exploit(object):
class Ms08_067_Exploiter(HostExploiter):
_TARGET_OS_TYPE = ['windows']
+ _EXPLOITED_SERVICE = 'Microsoft Server Service'
_windows_versions = {'Windows Server 2003 3790 Service Pack 2': WindowsVersion.Windows2003_SP2,
'Windows Server 2003 R2 3790 Service Pack 2': WindowsVersion.Windows2003_SP2}
diff --git a/monkey/monkey_island/cc/services/attack/attack_report.py b/monkey/monkey_island/cc/services/attack/attack_report.py
index b1079d8c7..4ec62d7a2 100644
--- a/monkey/monkey_island/cc/services/attack/attack_report.py
+++ b/monkey/monkey_island/cc/services/attack/attack_report.py
@@ -1,5 +1,5 @@
import logging
-from monkey_island.cc.services.attack.technique_reports import T1210, T1197, T1110, T1075, T1003
+from monkey_island.cc.services.attack.technique_reports import T1210, T1197, T1110, T1075, T1003, T1059
from monkey_island.cc.services.attack.attack_telem import AttackTelemService
from monkey_island.cc.services.attack.attack_config import AttackConfig
from monkey_island.cc.database import mongo
@@ -13,7 +13,8 @@ TECHNIQUES = {'T1210': T1210.T1210,
'T1197': T1197.T1197,
'T1110': T1110.T1110,
'T1075': T1075.T1075,
- 'T1003': T1003.T1003}
+ 'T1003': T1003.T1003,
+ 'T1059': T1059.T1059}
REPORT_NAME = 'new_report'
diff --git a/monkey/monkey_island/cc/services/attack/attack_schema.py b/monkey/monkey_island/cc/services/attack/attack_schema.py
index 2a59103ad..a79b57a87 100644
--- a/monkey/monkey_island/cc/services/attack/attack_schema.py
+++ b/monkey/monkey_island/cc/services/attack/attack_schema.py
@@ -84,5 +84,19 @@ SCHEMA = {
}
}
},
+ "execution": {
+ "title": "Execution",
+ "type": "object",
+ "properties": {
+ "T1059": {
+ "title": "T1059 Command line interface",
+ "type": "bool",
+ "value": True,
+ "necessary": True,
+ "description": "Adversaries may use command-line interfaces to interact with systems "
+ "and execute other software during the course of an operation.",
+ }
+ }
+ },
}
}
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
new file mode 100644
index 000000000..bf29247fd
--- /dev/null
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -0,0 +1,30 @@
+from monkey_island.cc.services.attack.technique_reports import AttackTechnique
+from common.utils.attack_utils import ScanStatus
+from monkey_island.cc.database import mongo
+
+__author__ = "VakarisZ"
+
+
+class T1059(AttackTechnique):
+
+ tech_id = "T1059"
+ unscanned_msg = "Monkey didn't exploit any machines to run commands at."
+ scanned_msg = ""
+ used_msg = "Monkey successfully ran commands on exploited machines in the network."
+
+ query = [{'$match': {'telem_type': 'exploit',
+ 'data.info.executed_cmds': {'$not': {'$size': 0}}}},
+ {'$project': {'_id': 0,
+ 'machine': '$data.machine',
+ 'info': '$data.info'}},
+ {'$group': {'_id': '$machine', 'data': {'$push': '$$ROOT'}}}]
+
+ @staticmethod
+ def get_report_data():
+ cmd_data = list(mongo.db.telemetry.aggregate(T1059.query))
+ data = {'title': T1059.technique_title(T1059.tech_id), 'data': cmd_data}
+ if cmd_data:
+ data.update({'message': T1059.used_msg, 'status': ScanStatus.USED.name})
+ else:
+ data.update({'message': T1059.unscanned_msg, 'status': ScanStatus.UNSCANNED.name})
+ return data
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
new file mode 100644
index 000000000..1cf93065f
--- /dev/null
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -0,0 +1,38 @@
+import React from 'react';
+import '../../../styles/Collapse.scss'
+import ReactTable from "react-table";
+import { RenderMachine } from "./Helpers"
+
+
+class T1059 extends React.Component {
+
+ constructor(props) {
+ super(props);
+ }
+
+ static getHashColumns() {
+ return ([{
+ columns: [
+ {Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.machine), style: { 'whiteSpace': 'unset' }},
+ {Header: 'Command', id: 'command', accessor: x => x.attempts[0].hashType, style: { 'whiteSpace': 'unset' }},
+ ]
+ }])};
+
+ render() {
+ return (
+
+
{this.props.data.message}
+
+ {this.props.data.status === 'USED' ?
+
: ""}
+
+ );
+ }
+}
+
+export default T1059;
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1075.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1075.js
index 6e80c9196..9b7e921b7 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1075.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1075.js
@@ -22,8 +22,8 @@ class T1075 extends React.Component {
columns: [
{Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.machine), style: { 'whiteSpace': 'unset' }},
{Header: 'Service', id: 'service', accessor: x => x.info.display_name, style: { 'whiteSpace': 'unset' }},
- {Header: 'Username', id: 'attempts', accessor: x => x.attempts[0].user, style: { 'whiteSpace': 'unset' }},
- {Header: 'Hash type', id: 'credentials', accessor: x => x.attempts[0].hashType, style: { 'whiteSpace': 'unset' }},
+ {Header: 'Username', id: 'username', accessor: x => x.attempts[0].user, style: { 'whiteSpace': 'unset' }},
+ {Header: 'Hash type', id: 'hash', accessor: x => x.attempts[0].hashType, style: { 'whiteSpace': 'unset' }},
]
}])};
From 908c531696cd0e61274f6e7442fad0a25793d6e1 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Mon, 10 Jun 2019 11:31:04 +0300
Subject: [PATCH 04/12] command line implementation finished
---
.../cc/services/attack/technique_reports/T1059.py | 2 +-
.../cc/ui/src/components/attack/techniques/T1059.js | 9 +++++----
.../ui/src/components/report-components/AttackReport.js | 4 +++-
3 files changed, 9 insertions(+), 6 deletions(-)
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
index bf29247fd..330b13c71 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -22,7 +22,7 @@ class T1059(AttackTechnique):
@staticmethod
def get_report_data():
cmd_data = list(mongo.db.telemetry.aggregate(T1059.query))
- data = {'title': T1059.technique_title(T1059.tech_id), 'data': cmd_data}
+ data = {'title': T1059.technique_title(T1059.tech_id), 'cmds': cmd_data}
if cmd_data:
data.update({'message': T1059.used_msg, 'status': ScanStatus.USED.name})
else:
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index 1cf93065f..4a65ee34a 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -13,8 +13,9 @@ class T1059 extends React.Component {
static getHashColumns() {
return ([{
columns: [
- {Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.machine), style: { 'whiteSpace': 'unset' }},
- {Header: 'Command', id: 'command', accessor: x => x.attempts[0].hashType, style: { 'whiteSpace': 'unset' }},
+ {Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
+ {Header: 'Approx. Time', id: 'time', accessor: x => x.data[0].info.finished, style: { 'whiteSpace': 'unset' }},
+ {Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds[0], style: { 'whiteSpace': 'unset' }},
]
}])};
@@ -26,9 +27,9 @@ class T1059 extends React.Component {
{this.props.data.status === 'USED' ?
: ""}
);
diff --git a/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js b/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
index b10b1ff7d..20250873a 100644
--- a/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
+++ b/monkey/monkey_island/cc/ui/src/components/report-components/AttackReport.js
@@ -10,13 +10,15 @@ import T1197 from '../attack/techniques/T1197';
import T1110 from '../attack/techniques/T1110';
import T1075 from "../attack/techniques/T1075";
import T1003 from "../attack/techniques/T1003";
+import T1059 from "../attack/techniques/T1059";
const tech_components = {
'T1210': T1210,
'T1197': T1197,
'T1110': T1110,
'T1075': T1075,
- 'T1003': T1003
+ 'T1003': T1003,
+ 'T1059': T1059
};
const classNames = require('classnames');
From 6636cd23e8b3aec3ed7f898b94ea683466c27739 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Mon, 10 Jun 2019 12:32:01 +0300
Subject: [PATCH 05/12] Changed cmds from array to dict
---
monkey/infection_monkey/exploit/__init__.py | 5 ++++-
monkey/infection_monkey/exploit/hadoop.py | 2 +-
monkey/infection_monkey/exploit/mssqlexec.py | 2 +-
monkey/infection_monkey/exploit/rdpgrinder.py | 2 +-
monkey/infection_monkey/exploit/shellshock.py | 2 +-
monkey/infection_monkey/exploit/sshexec.py | 2 +-
monkey/infection_monkey/exploit/vsftpd.py | 2 +-
monkey/infection_monkey/exploit/web_rce.py | 2 +-
monkey/infection_monkey/exploit/wmiexec.py | 2 +-
.../cc/services/attack/technique_reports/T1059.py | 2 +-
.../cc/ui/src/components/attack/techniques/T1059.js | 2 +-
11 files changed, 14 insertions(+), 11 deletions(-)
diff --git a/monkey/infection_monkey/exploit/__init__.py b/monkey/infection_monkey/exploit/__init__.py
index 84dd81393..7cf1ac4cf 100644
--- a/monkey/infection_monkey/exploit/__init__.py
+++ b/monkey/infection_monkey/exploit/__init__.py
@@ -25,7 +25,7 @@ class HostExploiter(object):
'finished': '',
'vulnerable_urls': [],
'vulnerable_ports': [],
- 'executed_cmds': []}
+ 'executed_cmds': {}}
self._exploit_attempts = []
self.host = host
@@ -59,6 +59,9 @@ class HostExploiter(object):
def add_vuln_port(self, port):
self._exploit_info['vulnerable_ports'].append(port)
+ def add_example_cmd(self, cmd):
+ self._exploit_info['executed_cmds']['example'] = cmd
+
from infection_monkey.exploit.win_ms08_067 import Ms08_067_Exploiter
from infection_monkey.exploit.wmiexec import WmiExploiter
diff --git a/monkey/infection_monkey/exploit/hadoop.py b/monkey/infection_monkey/exploit/hadoop.py
index 1e8ac9d8e..570575423 100644
--- a/monkey/infection_monkey/exploit/hadoop.py
+++ b/monkey/infection_monkey/exploit/hadoop.py
@@ -49,7 +49,7 @@ class HadoopExploiter(WebRCE):
return False
http_thread.join(self.DOWNLOAD_TIMEOUT)
http_thread.stop()
- self._exploit_info['executed_cmds'].append(command)
+ self.add_example_cmd(command)
return True
def exploit(self, url, command):
diff --git a/monkey/infection_monkey/exploit/mssqlexec.py b/monkey/infection_monkey/exploit/mssqlexec.py
index 1f21f9ecd..d738cba60 100644
--- a/monkey/infection_monkey/exploit/mssqlexec.py
+++ b/monkey/infection_monkey/exploit/mssqlexec.py
@@ -76,7 +76,7 @@ class MSSQLExploiter(HostExploiter):
commands.extend(monkey_args)
MSSQLExploiter.execute_command(cursor, commands)
MSSQLExploiter.run_file(cursor, tmp_file_path)
- self._exploit_info['executed_cmds'].append(commands[-1])
+ self.add_example_cmd(commands[-1])
return True
@staticmethod
diff --git a/monkey/infection_monkey/exploit/rdpgrinder.py b/monkey/infection_monkey/exploit/rdpgrinder.py
index dcbc01b1d..1e66040fc 100644
--- a/monkey/infection_monkey/exploit/rdpgrinder.py
+++ b/monkey/infection_monkey/exploit/rdpgrinder.py
@@ -343,5 +343,5 @@ class RdpExploiter(HostExploiter):
LOG.info("Executed monkey '%s' on remote victim %r",
os.path.basename(src_path), self.host)
- self._exploit_info['executed_cmds'].append(command)
+ self.add_example_cmd(command)
return True
diff --git a/monkey/infection_monkey/exploit/shellshock.py b/monkey/infection_monkey/exploit/shellshock.py
index b308c57fe..9db770905 100644
--- a/monkey/infection_monkey/exploit/shellshock.py
+++ b/monkey/infection_monkey/exploit/shellshock.py
@@ -144,7 +144,7 @@ class ShellShockExploiter(HostExploiter):
if not (self.check_remote_file_exists(url, header, exploit, self._config.monkey_log_path_linux)):
LOG.info("Log file does not exist, monkey might not have run")
continue
- self._exploit_info['executed_cmds'].append(cmdline)
+ self.add_example_cmd(cmdline)
return True
return False
diff --git a/monkey/infection_monkey/exploit/sshexec.py b/monkey/infection_monkey/exploit/sshexec.py
index 9aafae2a4..226ad8943 100644
--- a/monkey/infection_monkey/exploit/sshexec.py
+++ b/monkey/infection_monkey/exploit/sshexec.py
@@ -178,7 +178,7 @@ class SSHExploiter(HostExploiter):
self._config.dropper_target_path_linux, self.host, cmdline)
ssh.close()
- self._exploit_info['executed_cmds'].append(cmdline)
+ self.add_example_cmd(cmdline)
return True
except Exception as exc:
diff --git a/monkey/infection_monkey/exploit/vsftpd.py b/monkey/infection_monkey/exploit/vsftpd.py
index 23a89a96e..f3f13c073 100644
--- a/monkey/infection_monkey/exploit/vsftpd.py
+++ b/monkey/infection_monkey/exploit/vsftpd.py
@@ -138,7 +138,7 @@ class VSFTPDExploiter(HostExploiter):
if backdoor_socket.send(run_monkey):
LOG.info("Executed monkey '%s' on remote victim %r (cmdline=%r)", self._config.dropper_target_path_linux,
self.host, run_monkey)
- self._exploit_info['executed_cmds'].append(run_monkey)
+ self.add_example_cmd(run_monkey)
return True
else:
return False
diff --git a/monkey/infection_monkey/exploit/web_rce.py b/monkey/infection_monkey/exploit/web_rce.py
index 57a4cfa60..053737075 100644
--- a/monkey/infection_monkey/exploit/web_rce.py
+++ b/monkey/infection_monkey/exploit/web_rce.py
@@ -420,7 +420,7 @@ class WebRCE(HostExploiter):
return False
LOG.info("Execution attempt finished")
- self._exploit_info['executed_cmds'].append(command)
+ self.add_example_cmd(command)
return resp
def get_monkey_upload_path(self, url_to_monkey):
diff --git a/monkey/infection_monkey/exploit/wmiexec.py b/monkey/infection_monkey/exploit/wmiexec.py
index 29bc08981..b9decf433 100644
--- a/monkey/infection_monkey/exploit/wmiexec.py
+++ b/monkey/infection_monkey/exploit/wmiexec.py
@@ -114,7 +114,7 @@ class WmiExploiter(HostExploiter):
result.RemRelease()
wmi_connection.close()
-
+ self.add_example_cmd(cmdline)
return success
return False
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
index 330b13c71..e85e27415 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -13,7 +13,7 @@ class T1059(AttackTechnique):
used_msg = "Monkey successfully ran commands on exploited machines in the network."
query = [{'$match': {'telem_type': 'exploit',
- 'data.info.executed_cmds': {'$not': {'$size': 0}}}},
+ 'data.info.executed_cmds.example': {'$exists': True}}},
{'$project': {'_id': 0,
'machine': '$data.machine',
'info': '$data.info'}},
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index 4a65ee34a..466474624 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -15,7 +15,7 @@ class T1059 extends React.Component {
columns: [
{Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
{Header: 'Approx. Time', id: 'time', accessor: x => x.data[0].info.finished, style: { 'whiteSpace': 'unset' }},
- {Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds[0], style: { 'whiteSpace': 'unset' }},
+ {Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds.example, style: { 'whiteSpace': 'unset' }},
]
}])};
From 6ca33fff68c724d77c2788fe63b778bbbb8a8c20 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Mon, 10 Jun 2019 12:41:29 +0300
Subject: [PATCH 06/12] Added header to used commands table.
---
.../cc/ui/src/components/attack/techniques/T1059.js | 1 +
1 file changed, 1 insertion(+)
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index 466474624..d1c809651 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -12,6 +12,7 @@ class T1059 extends React.Component {
static getHashColumns() {
return ([{
+ Header: 'Example commands used',
columns: [
{Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
{Header: 'Approx. Time', id: 'time', accessor: x => x.data[0].info.finished, style: { 'whiteSpace': 'unset' }},
From 9cc526ca09ff6b597ba82371c07b8163fcb08c09 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Mon, 10 Jun 2019 17:33:00 +0300
Subject: [PATCH 07/12] web_rce bugfix
---
monkey/infection_monkey/exploit/web_rce.py | 1 +
1 file changed, 1 insertion(+)
diff --git a/monkey/infection_monkey/exploit/web_rce.py b/monkey/infection_monkey/exploit/web_rce.py
index 053737075..8c478a8ac 100644
--- a/monkey/infection_monkey/exploit/web_rce.py
+++ b/monkey/infection_monkey/exploit/web_rce.py
@@ -407,6 +407,7 @@ class WebRCE(HostExploiter):
# If exploiter returns True / False
if type(resp) is bool:
LOG.info("Execution attempt successfully finished")
+ self.add_example_cmd(command)
return resp
# If exploiter returns command output, we can check for execution errors
if 'is not recognized' in resp or 'command not found' in resp:
From ab4bbd437f6592c4461a8fe6d71db85a3045036f Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Wed, 19 Jun 2019 11:54:58 +0300
Subject: [PATCH 08/12] Readability improvements
---
.../attack/technique_reports/T1075.py | 7 ++++---
.../attack/technique_reports/T1110.py | 8 ++++----
.../attack/technique_reports/T1210.py | 7 ++++---
.../attack/technique_reports/__init__.py | 20 +++++++++----------
4 files changed, 21 insertions(+), 21 deletions(-)
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
index 18082dfc1..09eee9d44 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
@@ -35,9 +35,10 @@ class T1075(AttackTechnique):
successful_logins = list(mongo.db.telemetry.aggregate(T1075.query))
data.update({'successful_logins': successful_logins})
if successful_logins:
- data.update(T1075.get_message_and_status(ScanStatus.USED))
+ status = ScanStatus.USED
elif mongo.db.telemetry.count_documents(T1075.login_attempt_query):
- data.update(T1075.get_message_and_status(ScanStatus.SCANNED))
+ status = ScanStatus.SCANNED
else:
- data.update(T1075.get_message_and_status(ScanStatus.UNSCANNED))
+ status = ScanStatus.UNSCANNED
+ data.update(T1075.get_message_and_status(status))
return data
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
index e8e4a62c3..fe236f487 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
@@ -35,12 +35,12 @@ class T1110(AttackTechnique):
result['successful_creds'].append(T1110.parse_creds(attempt))
if succeeded:
- data = T1110.get_message_and_status(ScanStatus.USED)
+ status = ScanStatus.USED
elif attempts:
- data = T1110.get_message_and_status(ScanStatus.SCANNED)
+ status = ScanStatus.SCANNED
else:
- data = T1110.get_message_and_status(ScanStatus.UNSCANNED)
-
+ status = ScanStatus.UNSCANNED
+ data = T1110.get_message_and_status(status)
# Remove data with no successful brute force attempts
attempts = [attempt for attempt in attempts if attempt['attempts']]
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
index 677495c10..ff5d1caa0 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
@@ -18,11 +18,12 @@ class T1210(AttackTechnique):
scanned_services = T1210.get_scanned_services()
exploited_services = T1210.get_exploited_services()
if exploited_services:
- data.update({'status': ScanStatus.USED.name, 'message': T1210.used_msg})
+ status = ScanStatus.USED
elif scanned_services:
- data.update({'status': ScanStatus.SCANNED.name, 'message': T1210.scanned_msg})
+ status = ScanStatus.SCANNED
else:
- data.update({'status': ScanStatus.UNSCANNED.name, 'message': T1210.unscanned_msg})
+ status = ScanStatus.UNSCANNED.name
+ data.update(T1210.get_message_and_status(status))
data.update({'scanned_services': scanned_services, 'exploited_services': exploited_services})
return data
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py b/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
index 7faaf5afd..fe2beb424 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
@@ -46,20 +46,19 @@ class AttackTechnique(object):
"""
pass
- @staticmethod
- def technique_status(tech_id):
+ @classmethod
+ def technique_status(cls):
"""
Gets the status of a certain attack technique.
- :param tech_id: ID of attack technique, for e.g. T1110
:return: ScanStatus Enum object
"""
if mongo.db.attack_results.find_one({'telem_catagory': 'attack',
'status': ScanStatus.USED.value,
- 'technique': tech_id}):
+ 'technique': cls.tech_id}):
return ScanStatus.USED
elif mongo.db.attack_results.find_one({'telem_catagory': 'attack',
'status': ScanStatus.SCANNED.value,
- 'technique': tech_id}):
+ 'technique': cls.tech_id}):
return ScanStatus.SCANNED
else:
return ScanStatus.UNSCANNED
@@ -87,13 +86,12 @@ class AttackTechnique(object):
else:
return cls.used_msg
- @staticmethod
- def technique_title(tech_id):
+ @classmethod
+ def technique_title(cls):
"""
- :param tech_id: Technique's id. E.g. T1110
:return: techniques title. E.g. "T1110 Brute force"
"""
- return AttackConfig.get_technique(tech_id)['title']
+ return AttackConfig.get_technique(cls.tech_id)['title']
@classmethod
def get_tech_base_data(cls):
@@ -102,8 +100,8 @@ class AttackTechnique(object):
:return: dict E.g. {'message': 'Brute force used', 'status': 'Used', 'title': 'T1110 Brute force'}
"""
data = {}
- status = AttackTechnique.technique_status(cls.tech_id)
- title = AttackTechnique.technique_title(cls.tech_id)
+ status = cls.technique_status()
+ title = cls.technique_title()
data.update({'status': status.name,
'title': title,
'message': cls.get_message_by_status(status)})
From b3c6baf214d3fb124d0420c9d217957fb905b102 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Wed, 19 Jun 2019 11:54:58 +0300
Subject: [PATCH 09/12] Readability improvements
---
.../attack/technique_reports/T1075.py | 9 +++++----
.../attack/technique_reports/T1110.py | 10 +++++-----
.../attack/technique_reports/T1210.py | 9 +++++----
.../attack/technique_reports/__init__.py | 20 +++++++++----------
4 files changed, 24 insertions(+), 24 deletions(-)
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
index 18082dfc1..fa65a66c2 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1075.py
@@ -31,13 +31,14 @@ class T1075(AttackTechnique):
@staticmethod
def get_report_data():
- data = {'title': T1075.technique_title(T1075.tech_id)}
+ data = {'title': T1075.technique_title()}
successful_logins = list(mongo.db.telemetry.aggregate(T1075.query))
data.update({'successful_logins': successful_logins})
if successful_logins:
- data.update(T1075.get_message_and_status(ScanStatus.USED))
+ status = ScanStatus.USED
elif mongo.db.telemetry.count_documents(T1075.login_attempt_query):
- data.update(T1075.get_message_and_status(ScanStatus.SCANNED))
+ status = ScanStatus.SCANNED
else:
- data.update(T1075.get_message_and_status(ScanStatus.UNSCANNED))
+ status = ScanStatus.UNSCANNED
+ data.update(T1075.get_message_and_status(status))
return data
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
index e8e4a62c3..0f09fb0fe 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
@@ -35,16 +35,16 @@ class T1110(AttackTechnique):
result['successful_creds'].append(T1110.parse_creds(attempt))
if succeeded:
- data = T1110.get_message_and_status(ScanStatus.USED)
+ status = ScanStatus.USED
elif attempts:
- data = T1110.get_message_and_status(ScanStatus.SCANNED)
+ status = ScanStatus.SCANNED
else:
- data = T1110.get_message_and_status(ScanStatus.UNSCANNED)
-
+ status = ScanStatus.UNSCANNED
+ data = T1110.get_message_and_status(status)
# Remove data with no successful brute force attempts
attempts = [attempt for attempt in attempts if attempt['attempts']]
- data.update({'services': attempts, 'title': T1110.technique_title(T1110.tech_id)})
+ data.update({'services': attempts, 'title': T1110.technique_title()})
return data
@staticmethod
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
index 677495c10..08019699b 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1210.py
@@ -14,15 +14,16 @@ class T1210(AttackTechnique):
@staticmethod
def get_report_data():
- data = {'title': T1210.technique_title(T1210.tech_id)}
+ data = {'title': T1210.technique_title()}
scanned_services = T1210.get_scanned_services()
exploited_services = T1210.get_exploited_services()
if exploited_services:
- data.update({'status': ScanStatus.USED.name, 'message': T1210.used_msg})
+ status = ScanStatus.USED
elif scanned_services:
- data.update({'status': ScanStatus.SCANNED.name, 'message': T1210.scanned_msg})
+ status = ScanStatus.SCANNED
else:
- data.update({'status': ScanStatus.UNSCANNED.name, 'message': T1210.unscanned_msg})
+ status = ScanStatus.UNSCANNED.name
+ data.update(T1210.get_message_and_status(status))
data.update({'scanned_services': scanned_services, 'exploited_services': exploited_services})
return data
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py b/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
index 7faaf5afd..fe2beb424 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/__init__.py
@@ -46,20 +46,19 @@ class AttackTechnique(object):
"""
pass
- @staticmethod
- def technique_status(tech_id):
+ @classmethod
+ def technique_status(cls):
"""
Gets the status of a certain attack technique.
- :param tech_id: ID of attack technique, for e.g. T1110
:return: ScanStatus Enum object
"""
if mongo.db.attack_results.find_one({'telem_catagory': 'attack',
'status': ScanStatus.USED.value,
- 'technique': tech_id}):
+ 'technique': cls.tech_id}):
return ScanStatus.USED
elif mongo.db.attack_results.find_one({'telem_catagory': 'attack',
'status': ScanStatus.SCANNED.value,
- 'technique': tech_id}):
+ 'technique': cls.tech_id}):
return ScanStatus.SCANNED
else:
return ScanStatus.UNSCANNED
@@ -87,13 +86,12 @@ class AttackTechnique(object):
else:
return cls.used_msg
- @staticmethod
- def technique_title(tech_id):
+ @classmethod
+ def technique_title(cls):
"""
- :param tech_id: Technique's id. E.g. T1110
:return: techniques title. E.g. "T1110 Brute force"
"""
- return AttackConfig.get_technique(tech_id)['title']
+ return AttackConfig.get_technique(cls.tech_id)['title']
@classmethod
def get_tech_base_data(cls):
@@ -102,8 +100,8 @@ class AttackTechnique(object):
:return: dict E.g. {'message': 'Brute force used', 'status': 'Used', 'title': 'T1110 Brute force'}
"""
data = {}
- status = AttackTechnique.technique_status(cls.tech_id)
- title = AttackTechnique.technique_title(cls.tech_id)
+ status = cls.technique_status()
+ title = cls.technique_title()
data.update({'status': status.name,
'title': title,
'message': cls.get_message_by_status(status)})
From e38410a232409ec065442361d306c9c97a104ee6 Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Wed, 19 Jun 2019 12:43:44 +0300
Subject: [PATCH 10/12] CR improvements
---
monkey/infection_monkey/exploit/__init__.py | 2 +-
monkey/infection_monkey/exploit/hadoop.py | 2 +-
monkey/infection_monkey/exploit/mssqlexec.py | 2 +-
monkey/infection_monkey/exploit/rdpgrinder.py | 2 +-
monkey/infection_monkey/exploit/shellshock.py | 2 +-
monkey/infection_monkey/exploit/sshexec.py | 2 +-
monkey/infection_monkey/exploit/vsftpd.py | 2 +-
monkey/infection_monkey/exploit/web_rce.py | 4 ++--
monkey/infection_monkey/exploit/wmiexec.py | 2 +-
.../cc/services/attack/technique_reports/T1003.py | 9 +++++----
.../cc/services/attack/technique_reports/T1059.py | 7 ++++---
.../cc/ui/src/components/attack/techniques/T1059.js | 4 ++--
12 files changed, 21 insertions(+), 19 deletions(-)
diff --git a/monkey/infection_monkey/exploit/__init__.py b/monkey/infection_monkey/exploit/__init__.py
index 7cf1ac4cf..962240665 100644
--- a/monkey/infection_monkey/exploit/__init__.py
+++ b/monkey/infection_monkey/exploit/__init__.py
@@ -59,7 +59,7 @@ class HostExploiter(object):
def add_vuln_port(self, port):
self._exploit_info['vulnerable_ports'].append(port)
- def add_example_cmd(self, cmd):
+ def set_example_cmd(self, cmd):
self._exploit_info['executed_cmds']['example'] = cmd
diff --git a/monkey/infection_monkey/exploit/hadoop.py b/monkey/infection_monkey/exploit/hadoop.py
index 570575423..39edf0262 100644
--- a/monkey/infection_monkey/exploit/hadoop.py
+++ b/monkey/infection_monkey/exploit/hadoop.py
@@ -49,7 +49,7 @@ class HadoopExploiter(WebRCE):
return False
http_thread.join(self.DOWNLOAD_TIMEOUT)
http_thread.stop()
- self.add_example_cmd(command)
+ self.set_example_cmd(command)
return True
def exploit(self, url, command):
diff --git a/monkey/infection_monkey/exploit/mssqlexec.py b/monkey/infection_monkey/exploit/mssqlexec.py
index 1bf7f90d5..c1409ec6c 100644
--- a/monkey/infection_monkey/exploit/mssqlexec.py
+++ b/monkey/infection_monkey/exploit/mssqlexec.py
@@ -77,7 +77,7 @@ class MSSQLExploiter(HostExploiter):
commands.extend(monkey_args)
MSSQLExploiter.execute_command(cursor, commands)
MSSQLExploiter.run_file(cursor, tmp_file_path)
- self.add_example_cmd(commands[-1])
+ self.set_example_cmd(commands[-1])
return True
@staticmethod
diff --git a/monkey/infection_monkey/exploit/rdpgrinder.py b/monkey/infection_monkey/exploit/rdpgrinder.py
index 819ff00fb..828b03c20 100644
--- a/monkey/infection_monkey/exploit/rdpgrinder.py
+++ b/monkey/infection_monkey/exploit/rdpgrinder.py
@@ -343,5 +343,5 @@ class RdpExploiter(HostExploiter):
LOG.info("Executed monkey '%s' on remote victim %r",
os.path.basename(src_path), self.host)
- self.add_example_cmd(command)
+ self.set_example_cmd(command)
return True
diff --git a/monkey/infection_monkey/exploit/shellshock.py b/monkey/infection_monkey/exploit/shellshock.py
index 9db770905..26e9a743b 100644
--- a/monkey/infection_monkey/exploit/shellshock.py
+++ b/monkey/infection_monkey/exploit/shellshock.py
@@ -144,7 +144,7 @@ class ShellShockExploiter(HostExploiter):
if not (self.check_remote_file_exists(url, header, exploit, self._config.monkey_log_path_linux)):
LOG.info("Log file does not exist, monkey might not have run")
continue
- self.add_example_cmd(cmdline)
+ self.set_example_cmd(cmdline)
return True
return False
diff --git a/monkey/infection_monkey/exploit/sshexec.py b/monkey/infection_monkey/exploit/sshexec.py
index 226ad8943..e65d3cb19 100644
--- a/monkey/infection_monkey/exploit/sshexec.py
+++ b/monkey/infection_monkey/exploit/sshexec.py
@@ -178,7 +178,7 @@ class SSHExploiter(HostExploiter):
self._config.dropper_target_path_linux, self.host, cmdline)
ssh.close()
- self.add_example_cmd(cmdline)
+ self.set_example_cmd(cmdline)
return True
except Exception as exc:
diff --git a/monkey/infection_monkey/exploit/vsftpd.py b/monkey/infection_monkey/exploit/vsftpd.py
index f3f13c073..eddac620c 100644
--- a/monkey/infection_monkey/exploit/vsftpd.py
+++ b/monkey/infection_monkey/exploit/vsftpd.py
@@ -138,7 +138,7 @@ class VSFTPDExploiter(HostExploiter):
if backdoor_socket.send(run_monkey):
LOG.info("Executed monkey '%s' on remote victim %r (cmdline=%r)", self._config.dropper_target_path_linux,
self.host, run_monkey)
- self.add_example_cmd(run_monkey)
+ self.set_example_cmd(run_monkey)
return True
else:
return False
diff --git a/monkey/infection_monkey/exploit/web_rce.py b/monkey/infection_monkey/exploit/web_rce.py
index 43c1b5749..351eb7c17 100644
--- a/monkey/infection_monkey/exploit/web_rce.py
+++ b/monkey/infection_monkey/exploit/web_rce.py
@@ -408,7 +408,7 @@ class WebRCE(HostExploiter):
# If exploiter returns True / False
if type(resp) is bool:
LOG.info("Execution attempt successfully finished")
- self.add_example_cmd(command)
+ self.set_example_cmd(command)
return resp
# If exploiter returns command output, we can check for execution errors
if 'is not recognized' in resp or 'command not found' in resp:
@@ -422,7 +422,7 @@ class WebRCE(HostExploiter):
return False
LOG.info("Execution attempt finished")
- self.add_example_cmd(command)
+ self.set_example_cmd(command)
return resp
def get_monkey_upload_path(self, url_to_monkey):
diff --git a/monkey/infection_monkey/exploit/wmiexec.py b/monkey/infection_monkey/exploit/wmiexec.py
index b9decf433..648fb233d 100644
--- a/monkey/infection_monkey/exploit/wmiexec.py
+++ b/monkey/infection_monkey/exploit/wmiexec.py
@@ -114,7 +114,7 @@ class WmiExploiter(HostExploiter):
result.RemRelease()
wmi_connection.close()
- self.add_example_cmd(cmdline)
+ self.set_example_cmd(cmdline)
return success
return False
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
index 002e2eeb7..d30197e9a 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
@@ -9,7 +9,7 @@ class T1003(AttackTechnique):
tech_id = "T1003"
unscanned_msg = "Monkey tried to obtain credentials from systems in the network but didn't find any or failed."
- scanned_msg = "Monkey tried to obtain credentials from systems in the network but didn't find any or failed."
+ scanned_msg = ""
used_msg = "Monkey successfully obtained some credentials from systems on the network."
query = {'telem_type': 'system_info_collection', '$and': [{'data.credentials': {'$exists': True}},
@@ -17,9 +17,10 @@ class T1003(AttackTechnique):
@staticmethod
def get_report_data():
- data = {'title': T1003.technique_title(T1003.tech_id)}
+ data = {'title': T1003.technique_title()}
if mongo.db.telemetry.count_documents(T1003.query):
- data.update({'message': T1003.used_msg, 'status': ScanStatus.USED.name})
+ status = ScanStatus.USED
else:
- data.update({'message': T1003.unscanned_msg, 'status': ScanStatus.UNSCANNED.name})
+ status = ScanStatus.UNSCANNED
+ data.update(T1003.get_message_and_status(status))
return data
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
index e85e27415..6f126b175 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -22,9 +22,10 @@ class T1059(AttackTechnique):
@staticmethod
def get_report_data():
cmd_data = list(mongo.db.telemetry.aggregate(T1059.query))
- data = {'title': T1059.technique_title(T1059.tech_id), 'cmds': cmd_data}
+ data = {'title': T1059.technique_title(), 'cmds': cmd_data}
if cmd_data:
- data.update({'message': T1059.used_msg, 'status': ScanStatus.USED.name})
+ status = ScanStatus.USED
else:
- data.update({'message': T1059.unscanned_msg, 'status': ScanStatus.UNSCANNED.name})
+ status = ScanStatus.UNSCANNED
+ data.update(T1059.get_message_and_status(status))
return data
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index d1c809651..5678b8c14 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -10,7 +10,7 @@ class T1059 extends React.Component {
super(props);
}
- static getHashColumns() {
+ static getCommandColumns() {
return ([{
Header: 'Example commands used',
columns: [
@@ -27,7 +27,7 @@ class T1059 extends React.Component {
{this.props.data.status === 'USED' ?
Date: Tue, 25 Jun 2019 10:42:03 +0300
Subject: [PATCH 11/12] Executed cmds info variable refactored
---
monkey/infection_monkey/exploit/__init__.py | 11 ++++++++---
monkey/infection_monkey/exploit/hadoop.py | 2 +-
monkey/infection_monkey/exploit/mssqlexec.py | 2 +-
monkey/infection_monkey/exploit/rdpgrinder.py | 2 +-
monkey/infection_monkey/exploit/shellshock.py | 2 +-
monkey/infection_monkey/exploit/sshexec.py | 2 +-
monkey/infection_monkey/exploit/vsftpd.py | 2 +-
monkey/infection_monkey/exploit/web_rce.py | 4 ++--
monkey/infection_monkey/exploit/wmiexec.py | 2 +-
.../monkey_island/cc/services/attack/attack_report.py | 1 -
.../cc/services/attack/technique_reports/T1003.py | 1 +
.../cc/services/attack/technique_reports/T1059.py | 2 +-
.../cc/ui/src/components/attack/techniques/T1059.js | 2 +-
13 files changed, 20 insertions(+), 15 deletions(-)
diff --git a/monkey/infection_monkey/exploit/__init__.py b/monkey/infection_monkey/exploit/__init__.py
index 962240665..95a199923 100644
--- a/monkey/infection_monkey/exploit/__init__.py
+++ b/monkey/infection_monkey/exploit/__init__.py
@@ -25,7 +25,7 @@ class HostExploiter(object):
'finished': '',
'vulnerable_urls': [],
'vulnerable_ports': [],
- 'executed_cmds': {}}
+ 'executed_cmds': []}
self._exploit_attempts = []
self.host = host
@@ -59,8 +59,13 @@ class HostExploiter(object):
def add_vuln_port(self, port):
self._exploit_info['vulnerable_ports'].append(port)
- def set_example_cmd(self, cmd):
- self._exploit_info['executed_cmds']['example'] = cmd
+ def add_executed_cmd(self, cmd):
+ """
+ Appends command to exploiter's info.
+ :param cmd: String of executed command. e.g. 'echo Example'
+ """
+ command = {'cmd': cmd}
+ self._exploit_info['executed_cmds'].append(command)
from infection_monkey.exploit.win_ms08_067 import Ms08_067_Exploiter
diff --git a/monkey/infection_monkey/exploit/hadoop.py b/monkey/infection_monkey/exploit/hadoop.py
index 39edf0262..ac1cf784a 100644
--- a/monkey/infection_monkey/exploit/hadoop.py
+++ b/monkey/infection_monkey/exploit/hadoop.py
@@ -49,7 +49,7 @@ class HadoopExploiter(WebRCE):
return False
http_thread.join(self.DOWNLOAD_TIMEOUT)
http_thread.stop()
- self.set_example_cmd(command)
+ self.add_executed_cmd(command)
return True
def exploit(self, url, command):
diff --git a/monkey/infection_monkey/exploit/mssqlexec.py b/monkey/infection_monkey/exploit/mssqlexec.py
index c1409ec6c..4b5b258b9 100644
--- a/monkey/infection_monkey/exploit/mssqlexec.py
+++ b/monkey/infection_monkey/exploit/mssqlexec.py
@@ -77,7 +77,7 @@ class MSSQLExploiter(HostExploiter):
commands.extend(monkey_args)
MSSQLExploiter.execute_command(cursor, commands)
MSSQLExploiter.run_file(cursor, tmp_file_path)
- self.set_example_cmd(commands[-1])
+ self.add_executed_cmd(commands[-1])
return True
@staticmethod
diff --git a/monkey/infection_monkey/exploit/rdpgrinder.py b/monkey/infection_monkey/exploit/rdpgrinder.py
index 828b03c20..ea2bbb3f6 100644
--- a/monkey/infection_monkey/exploit/rdpgrinder.py
+++ b/monkey/infection_monkey/exploit/rdpgrinder.py
@@ -343,5 +343,5 @@ class RdpExploiter(HostExploiter):
LOG.info("Executed monkey '%s' on remote victim %r",
os.path.basename(src_path), self.host)
- self.set_example_cmd(command)
+ self.add_executed_cmd(command)
return True
diff --git a/monkey/infection_monkey/exploit/shellshock.py b/monkey/infection_monkey/exploit/shellshock.py
index 26e9a743b..5686be5d7 100644
--- a/monkey/infection_monkey/exploit/shellshock.py
+++ b/monkey/infection_monkey/exploit/shellshock.py
@@ -144,7 +144,7 @@ class ShellShockExploiter(HostExploiter):
if not (self.check_remote_file_exists(url, header, exploit, self._config.monkey_log_path_linux)):
LOG.info("Log file does not exist, monkey might not have run")
continue
- self.set_example_cmd(cmdline)
+ self.add_executed_cmd(cmdline)
return True
return False
diff --git a/monkey/infection_monkey/exploit/sshexec.py b/monkey/infection_monkey/exploit/sshexec.py
index e65d3cb19..e4b7a313c 100644
--- a/monkey/infection_monkey/exploit/sshexec.py
+++ b/monkey/infection_monkey/exploit/sshexec.py
@@ -178,7 +178,7 @@ class SSHExploiter(HostExploiter):
self._config.dropper_target_path_linux, self.host, cmdline)
ssh.close()
- self.set_example_cmd(cmdline)
+ self.add_executed_cmd(cmdline)
return True
except Exception as exc:
diff --git a/monkey/infection_monkey/exploit/vsftpd.py b/monkey/infection_monkey/exploit/vsftpd.py
index eddac620c..ced79f208 100644
--- a/monkey/infection_monkey/exploit/vsftpd.py
+++ b/monkey/infection_monkey/exploit/vsftpd.py
@@ -138,7 +138,7 @@ class VSFTPDExploiter(HostExploiter):
if backdoor_socket.send(run_monkey):
LOG.info("Executed monkey '%s' on remote victim %r (cmdline=%r)", self._config.dropper_target_path_linux,
self.host, run_monkey)
- self.set_example_cmd(run_monkey)
+ self.add_executed_cmd(run_monkey)
return True
else:
return False
diff --git a/monkey/infection_monkey/exploit/web_rce.py b/monkey/infection_monkey/exploit/web_rce.py
index 351eb7c17..d138e4cca 100644
--- a/monkey/infection_monkey/exploit/web_rce.py
+++ b/monkey/infection_monkey/exploit/web_rce.py
@@ -408,7 +408,7 @@ class WebRCE(HostExploiter):
# If exploiter returns True / False
if type(resp) is bool:
LOG.info("Execution attempt successfully finished")
- self.set_example_cmd(command)
+ self.add_executed_cmd(command)
return resp
# If exploiter returns command output, we can check for execution errors
if 'is not recognized' in resp or 'command not found' in resp:
@@ -422,7 +422,7 @@ class WebRCE(HostExploiter):
return False
LOG.info("Execution attempt finished")
- self.set_example_cmd(command)
+ self.add_executed_cmd(command)
return resp
def get_monkey_upload_path(self, url_to_monkey):
diff --git a/monkey/infection_monkey/exploit/wmiexec.py b/monkey/infection_monkey/exploit/wmiexec.py
index 648fb233d..88246ba76 100644
--- a/monkey/infection_monkey/exploit/wmiexec.py
+++ b/monkey/infection_monkey/exploit/wmiexec.py
@@ -114,7 +114,7 @@ class WmiExploiter(HostExploiter):
result.RemRelease()
wmi_connection.close()
- self.set_example_cmd(cmdline)
+ self.add_executed_cmd(cmdline)
return success
return False
diff --git a/monkey/monkey_island/cc/services/attack/attack_report.py b/monkey/monkey_island/cc/services/attack/attack_report.py
index 27fe53392..711f5103a 100644
--- a/monkey/monkey_island/cc/services/attack/attack_report.py
+++ b/monkey/monkey_island/cc/services/attack/attack_report.py
@@ -1,6 +1,5 @@
import logging
from monkey_island.cc.services.attack.technique_reports import T1210, T1197, T1110, T1075, T1003, T1059
-from monkey_island.cc.services.attack.attack_telem import AttackTelemService
from monkey_island.cc.services.attack.attack_config import AttackConfig
from monkey_island.cc.database import mongo
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
index d30197e9a..bd9e31c92 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
@@ -13,6 +13,7 @@ class T1003(AttackTechnique):
used_msg = "Monkey successfully obtained some credentials from systems on the network."
query = {'telem_type': 'system_info_collection', '$and': [{'data.credentials': {'$exists': True}},
+ # $gt: {} checks if field is not an empty object
{'data.credentials': {'$gt': {}}}]}
@staticmethod
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
index 6f126b175..5f0fa4433 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -13,7 +13,7 @@ class T1059(AttackTechnique):
used_msg = "Monkey successfully ran commands on exploited machines in the network."
query = [{'$match': {'telem_type': 'exploit',
- 'data.info.executed_cmds.example': {'$exists': True}}},
+ 'data.info.executed_cmds': {'$exists': True, '$ne': []}}},
{'$project': {'_id': 0,
'machine': '$data.machine',
'info': '$data.info'}},
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index 5678b8c14..2352772c0 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -16,7 +16,7 @@ class T1059 extends React.Component {
columns: [
{Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
{Header: 'Approx. Time', id: 'time', accessor: x => x.data[0].info.finished, style: { 'whiteSpace': 'unset' }},
- {Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds.example, style: { 'whiteSpace': 'unset' }},
+ {Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds[0].cmd, style: { 'whiteSpace': 'unset' }},
]
}])};
From 36f917bc8dddbb8bffc51aff22d912fe60ac651e Mon Sep 17 00:00:00 2001
From: VakarisZ
Date: Tue, 25 Jun 2019 15:43:02 +0300
Subject: [PATCH 12/12] Updated branch according to changes in dev.
---
.../services/attack/technique_reports/T1003.py | 2 +-
.../services/attack/technique_reports/T1059.py | 2 +-
.../services/attack/technique_reports/T1110.py | 2 +-
.../services/attack/technique_reports/T1197.py | 16 +++++++++-------
.../ui/src/components/attack/techniques/T1059.js | 5 +++--
5 files changed, 15 insertions(+), 12 deletions(-)
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
index bd9e31c92..cd1a538cb 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1003.py
@@ -12,7 +12,7 @@ class T1003(AttackTechnique):
scanned_msg = ""
used_msg = "Monkey successfully obtained some credentials from systems on the network."
- query = {'telem_type': 'system_info_collection', '$and': [{'data.credentials': {'$exists': True}},
+ query = {'telem_category': 'system_info_collection', '$and': [{'data.credentials': {'$exists': True}},
# $gt: {} checks if field is not an empty object
{'data.credentials': {'$gt': {}}}]}
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
index 5f0fa4433..488a8f547 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1059.py
@@ -12,7 +12,7 @@ class T1059(AttackTechnique):
scanned_msg = ""
used_msg = "Monkey successfully ran commands on exploited machines in the network."
- query = [{'$match': {'telem_type': 'exploit',
+ query = [{'$match': {'telem_category': 'exploit',
'data.info.executed_cmds': {'$exists': True, '$ne': []}}},
{'$project': {'_id': 0,
'machine': '$data.machine',
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
index 0f09fb0fe..60ae14c0b 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1110.py
@@ -13,7 +13,7 @@ class T1110(AttackTechnique):
used_msg = "Monkey successfully used brute force in the network."
# Gets data about brute force attempts
- query = [{'$match': {'telem_type': 'exploit',
+ query = [{'$match': {'telem_category': 'exploit',
'data.attempts': {'$not': {'$size': 0}}}},
{'$project': {'_id': 0,
'machine': '$data.machine',
diff --git a/monkey/monkey_island/cc/services/attack/technique_reports/T1197.py b/monkey/monkey_island/cc/services/attack/technique_reports/T1197.py
index 9a968e998..b6bd316af 100644
--- a/monkey/monkey_island/cc/services/attack/technique_reports/T1197.py
+++ b/monkey/monkey_island/cc/services/attack/technique_reports/T1197.py
@@ -13,13 +13,15 @@ class T1197(AttackTechnique):
@staticmethod
def get_report_data():
data = T1197.get_tech_base_data()
- bits_results = mongo.db.telemetry.aggregate([{'$match': {'telem_category': 'attack', 'data.technique': T1197.tech_id}},
- {'$group': {'_id': {'ip_addr': '$data.machine.ip_addr', 'usage': '$data.usage'},
- 'ip_addr': {'$first': '$data.machine.ip_addr'},
- 'domain_name': {'$first': '$data.machine.domain_name'},
- 'usage': {'$first': '$data.usage'},
- 'time': {'$first': '$timestamp'}}
- }])
+ bits_results = mongo.db.telemetry.aggregate([{'$match': {'telem_category': 'attack',
+ 'data.technique': T1197.tech_id}},
+ {'$group': {'_id': {'ip_addr': '$data.machine.ip_addr',
+ 'usage': '$data.usage'},
+ 'ip_addr': {'$first': '$data.machine.ip_addr'},
+ 'domain_name': {'$first': '$data.machine.domain_name'},
+ 'usage': {'$first': '$data.usage'},
+ 'time': {'$first': '$timestamp'}}
+ }])
bits_results = list(bits_results)
data.update({'bits_jobs': bits_results})
return data
diff --git a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
index 2352772c0..abca8987a 100644
--- a/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
+++ b/monkey/monkey_island/cc/ui/src/components/attack/techniques/T1059.js
@@ -1,7 +1,7 @@
import React from 'react';
import '../../../styles/Collapse.scss'
import ReactTable from "react-table";
-import { RenderMachine } from "./Helpers"
+import { renderMachine } from "./Helpers"
class T1059 extends React.Component {
@@ -14,13 +14,14 @@ class T1059 extends React.Component {
return ([{
Header: 'Example commands used',
columns: [
- {Header: 'Machine', id: 'machine', accessor: x => RenderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
+ {Header: 'Machine', id: 'machine', accessor: x => renderMachine(x.data[0].machine), style: { 'whiteSpace': 'unset'}, width: 160 },
{Header: 'Approx. Time', id: 'time', accessor: x => x.data[0].info.finished, style: { 'whiteSpace': 'unset' }},
{Header: 'Command', id: 'command', accessor: x => x.data[0].info.executed_cmds[0].cmd, style: { 'whiteSpace': 'unset' }},
]
}])};
render() {
+ console.log(this.props.data);
return (
{this.props.data.message}