wazuh-kibana-app/public/controllers/dev-tools/index.js
2018-09-11 11:54:48 +02:00

380 lines
11 KiB
JavaScript

/*
* Wazuh app - Dev tools controller
* Copyright (C) 2018 Wazuh, Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Find more information about this on the LICENSE file.
*/
import { uiModules } from 'ui/modules';
import CodeMirror from '../../utils/codemirror/lib/codemirror';
import jsonLint from '../../utils/codemirror/json-lint';
import queryString from 'querystring-browser';
import $ from 'jquery';
const app = uiModules.get('app/wazuh', []);
class DevToolsController {
constructor($scope, apiReq, $window, appState, errorHandler, $document) {
this.$scope = $scope;
this.apiReq = apiReq;
this.$window = $window;
this.appState = appState;
this.errorHandler = errorHandler;
this.$document = $document;
this.groups = [];
this.linesWithClass = [];
this.widgets = [];
}
$onInit() {
this.apiInputBox = CodeMirror.fromTextArea(
this.$document[0].getElementById('api_input'),
{
lineNumbers: true,
matchBrackets: true,
mode: { name: 'javascript', json: true },
theme: 'ttcn',
foldGutter: true,
styleSelectedText: true,
gutters: ['CodeMirror-foldgutter']
}
);
this.apiInputBox.on('change', () => {
this.groups = this.analyzeGroups();
const currentState = this.apiInputBox.getValue().toString();
this.appState.setCurrentDevTools(currentState);
const currentGroup = this.calculateWhichGroup();
if (currentGroup) {
const hasWidget = this.widgets.filter(
item => item.start === currentGroup.start
);
if (hasWidget.length)
this.apiInputBox.removeLineWidget(hasWidget[0].widget);
setTimeout(() => this.checkJsonParseError(), 450);
}
});
this.apiInputBox.on('cursorActivity', () => {
const currentGroup = this.calculateWhichGroup();
this.highlightGroup(currentGroup);
});
this.apiOutputBox = CodeMirror.fromTextArea(
this.$document[0].getElementById('api_output'),
{
lineNumbers: true,
matchBrackets: true,
mode: { name: 'javascript', json: true },
readOnly: true,
lineWrapping: true,
styleActiveLine: true,
theme: 'ttcn',
foldGutter: true,
gutters: ['CodeMirror-foldgutter']
}
);
this.$scope.send = firstTime => this.send(firstTime);
this.$scope.help = () => {
this.$window.open(
'https://documentation.wazuh.com/current/user-manual/api/reference.html'
);
};
this.init();
this.$scope.send(true);
}
analyzeGroups() {
try {
const currentState = this.apiInputBox.getValue().toString();
this.appState.setCurrentDevTools(currentState);
const tmpgroups = [];
const splitted = currentState
.split(/[\r\n]+(?=(?:GET|PUT|POST|DELETE)\b)/gm)
.filter(item => item.replace(/\s/g, '').length);
let start = 0;
let end = 0;
const slen = splitted.length;
for (let i = 0; i < slen; i++) {
let tmp = splitted[i].split('\n');
if (Array.isArray(tmp)) tmp = tmp.filter(item => !item.includes('#'));
const cursor = this.apiInputBox.getSearchCursor(splitted[i], null, {
multiline: true
});
if (cursor.findNext()) start = cursor.from().line;
else return [];
end = start + tmp.length;
const tmpRequestText = tmp[0];
let tmpRequestTextJson = '';
const tmplen = tmp.length;
for (let j = 1; j < tmplen; ++j) {
if (!!tmp[j] && !tmp[j].includes('#')) {
tmpRequestTextJson += tmp[j];
}
}
if (tmpRequestTextJson && typeof tmpRequestTextJson === 'string') {
let rtjlen = tmp.length;
while (rtjlen--) {
if (tmp[rtjlen].trim() === '}') break;
else end -= 1;
}
}
if (!tmpRequestTextJson && tmp.length > 1) {
tmp = [tmp[0]];
end = start + 1;
}
if (i === slen - 1 && !tmpRequestTextJson) {
if (tmp.length > 1) end -= tmp.length - 1;
}
end--;
tmpgroups.push({
requestText: tmpRequestText,
requestTextJson: tmpRequestTextJson,
start,
end
});
}
return tmpgroups;
} catch (error) {
return [];
}
}
highlightGroup(group) {
for (const line of this.linesWithClass) {
this.apiInputBox.removeLineClass(
line,
'background',
'CodeMirror-styled-background'
);
}
this.linesWithClass = [];
if (group) {
if (!group.requestTextJson) {
this.linesWithClass.push(
this.apiInputBox.addLineClass(
group.start,
'background',
'CodeMirror-styled-background'
)
);
return;
}
for (let i = group.start; i <= group.end; i++) {
this.linesWithClass.push(
this.apiInputBox.addLineClass(
i,
'background',
'CodeMirror-styled-background'
)
);
}
}
}
checkJsonParseError() {
const affectedGroups = [];
for (const widget of this.widgets) {
this.apiInputBox.removeLineWidget(widget.widget);
}
this.widgets = [];
for (const item of this.groups) {
if (item.requestTextJson) {
try {
jsonLint.parse(item.requestTextJson);
} catch (error) {
affectedGroups.push(item.requestText);
const msg = this.$document[0].createElement('div');
msg.id = new Date().getTime() / 1000;
const icon = msg.appendChild(this.$document[0].createElement('div'));
icon.className = 'lint-error-icon';
icon.id = new Date().getTime() / 1000;
icon.onmouseover = () => {
const advice = msg.appendChild(
this.$document[0].createElement('span')
);
advice.id = new Date().getTime() / 1000;
advice.innerText = error.message || 'Error parsing query';
advice.className = 'lint-block-wz';
};
icon.onmouseleave = () => {
msg.removeChild(msg.lastChild);
};
this.widgets.push({
start: item.start,
widget: this.apiInputBox.addLineWidget(item.start, msg, {
coverGutter: false,
noHScroll: true
})
});
}
}
}
return affectedGroups;
}
init() {
this.apiInputBox.setSize('auto', '100%');
this.apiOutputBox.setSize('auto', '100%');
const currentState = this.appState.getCurrentDevTools();
if (!currentState) {
const demoStr =
'GET /\n\n# Comment here\nGET /agents\n' +
JSON.stringify({ limit: 1 }, null, 2);
this.appState.setCurrentDevTools(demoStr);
this.apiInputBox.getDoc().setValue(demoStr);
} else {
this.apiInputBox.getDoc().setValue(currentState);
}
this.groups = this.analyzeGroups();
const currentGroup = this.calculateWhichGroup();
this.highlightGroup(currentGroup);
}
calculateWhichGroup(firstTime) {
try {
const selection = this.apiInputBox.getCursor();
const desiredGroup = firstTime
? this.groups.filter(item => item.requestText)
: this.groups.filter(
item =>
item.requestText &&
(item.end >= selection.line && item.start <= selection.line)
);
// Place play button at first line from the selected group
const cords = this.apiInputBox.cursorCoords({
line: desiredGroup[0].start,
ch: 0
});
if (!$('#play_button').is(':visible')) $('#play_button').show();
const currentPlayButton = $('#play_button').offset();
$('#play_button').offset({
top: cords.top,
left: currentPlayButton.left
});
if (firstTime) this.highlightGroup(desiredGroup[0]);
return desiredGroup[0];
} catch (error) {
$('#play_button').hide();
return null;
}
}
async send(firstTime) {
try {
this.groups = this.analyzeGroups();
const desiredGroup = this.calculateWhichGroup(firstTime);
if (desiredGroup) {
if (firstTime) {
const cords = this.apiInputBox.cursorCoords({
line: desiredGroup.start,
ch: 0
});
const currentPlayButton = $('#play_button').offset();
$('#play_button').offset({
top: cords.top + 10,
left: currentPlayButton.left
});
}
const affectedGroups = this.checkJsonParseError();
const filteredAffectedGroups = affectedGroups.filter(
item => item === desiredGroup.requestText
);
if (filteredAffectedGroups.length) {
this.apiOutputBox.setValue('Error parsing JSON query');
return;
}
const method = desiredGroup.requestText.startsWith('GET')
? 'GET'
: desiredGroup.requestText.startsWith('POST')
? 'POST'
: desiredGroup.requestText.startsWith('PUT')
? 'PUT'
: desiredGroup.requestText.startsWith('DELETE')
? 'DELETE'
: 'GET';
const requestCopy = desiredGroup.requestText.includes(method)
? desiredGroup.requestText.split(method)[1].trim()
: desiredGroup.requestText;
// Checks for inline parameters
const inlineSplit = requestCopy.split('?');
const extra =
inlineSplit && inlineSplit[1]
? queryString.parse(inlineSplit[1])
: {};
const req = requestCopy
? requestCopy.startsWith('/')
? requestCopy
: `/${requestCopy}`
: '/';
let JSONraw = {};
try {
JSONraw = JSON.parse(desiredGroup.requestTextJson);
} catch (error) {
JSONraw = {};
}
if (typeof extra.pretty !== 'undefined') delete extra.pretty;
if (typeof JSONraw.pretty !== 'undefined') delete JSONraw.pretty;
// Assign inline parameters
for (const key in extra) JSONraw[key] = extra[key];
const path = req.includes('?') ? req.split('?')[0] : req;
if (typeof JSONraw === 'object') JSONraw.devTools = true;
const output = await this.apiReq.request(method, path, JSONraw);
this.apiOutputBox.setValue(JSON.stringify(output.data.data, null, 2));
} else {
this.apiOutputBox.setValue('Welcome!');
}
} catch (error) {
const parsedError = this.errorHandler.handle(error, null, null, true);
if (typeof parsedError === 'string') {
return this.apiOutputBox.setValue(parsedError);
} else if (error && error.data && typeof error.data === 'object') {
return this.apiOutputBox.setValue(JSON.stringify(error.data));
} else {
return this.apiOutputBox.setValue('Empty');
}
}
}
}
// Logs controller
app.controller('devToolsController', DevToolsController);