diff --git a/public/controllers/agent/agents-preview.js b/public/controllers/agent/agents-preview.js
index a50b511b7e..681d427d88 100644
--- a/public/controllers/agent/agents-preview.js
+++ b/public/controllers/agent/agents-preview.js
@@ -72,12 +72,8 @@ export class AgentsPreviewController {
this.load();
}
- search(term) {
- this.$scope.$broadcast('wazuhSearch', { term });
- }
-
- filter(filter) {
- this.$scope.$broadcast('wazuhFilter', { filter });
+ query(query, search) {
+ this.$scope.$broadcast('wazuhQuery', { query, search });
}
showAgent(agent) {
@@ -128,6 +124,19 @@ export class AgentsPreviewController {
const [agentsUnique, agentsTop] = data;
const unique = agentsUnique.data.result;
+ this.searchBarModel = {
+ 'status': ['Active', 'Disconnected', 'Never connected'],
+ 'group': unique.groups,
+ 'node_name': unique.nodes,
+ 'version': unique.versions,
+ 'os.platform': unique.osPlatforms.map(x => x.platform),
+ 'os.version': unique.osPlatforms.map(x => x.version),
+ 'os.name': unique.osPlatforms.map(x => x.name),
+ };
+ this.searchBarModel['os.name'] = Array.from(new Set(this.searchBarModel['os.name']));
+ this.searchBarModel['os.version'] = Array.from(new Set(this.searchBarModel['os.version']));
+ this.searchBarModel['os.platform'] = Array.from(new Set(this.searchBarModel['os.platform']));
+
this.groups = unique.groups;
this.nodes = unique.nodes.map(item => ({ id: item }));
this.versions = unique.versions.map(item => ({ id: item }));
diff --git a/public/directives/index.js b/public/directives/index.js
index 91d9f92c5c..9127222ba9 100644
--- a/public/directives/index.js
+++ b/public/directives/index.js
@@ -18,4 +18,6 @@ import './wz-welcome-card/wz-welcome-card';
import './wz-no-config/wz-no-config';
import './wz-config-item/wz-config-item';
import './wz-config-item/wz-config-item.less';
-import './wz-config-viewer/wz-config-viewer';
\ No newline at end of file
+import './wz-tag-filter/wz-tag-filter';
+import './wz-tag-filter/wz-tag-filter.less';
+import './wz-config-viewer/wz-config-viewer';
diff --git a/public/directives/wz-table/lib/data.js b/public/directives/wz-table/lib/data.js
index 30624ecf79..4a2b16e92c 100644
--- a/public/directives/wz-table/lib/data.js
+++ b/public/directives/wz-table/lib/data.js
@@ -64,10 +64,48 @@ export async function filterData(
$scope.wazuh_table_loading = false;
$scope.error = `Error filtering by ${
filter ? filter.value : 'undefined'
- } - ${error.message || error}.`;
+ } - ${error.message || error}.`;
errorHandler.handle(
`Error filtering by ${
- filter ? filter.value : 'undefined'
+ filter ? filter.value : 'undefined'
+ }. ${error.message || error}`,
+ 'Data factory'
+ );
+ }
+ if (!$scope.$$phase) $scope.$digest();
+ return;
+}
+
+export async function queryData(
+ query,
+ term,
+ instance,
+ wzTableFilter,
+ $scope,
+ fetch,
+ errorHandler
+) {
+ try {
+ $scope.error = false;
+ $scope.wazuh_table_loading = true;
+ instance.removeFilters();
+ if (term) {
+ instance.addFilter('search', term);
+ }
+ if (query) {
+ instance.addFilter('q', query);
+ }
+ wzTableFilter.set(instance.filters);
+ await fetch();
+ $scope.wazuh_table_loading = false;
+ } catch (error) {
+ $scope.wazuh_table_loading = false;
+ $scope.error = `Query error ${
+ query ? query.value : 'undefined'
+ } - ${error.message || error}.`;
+ errorHandler.handle(
+ `Query error ${
+ query ? query.value : 'undefined'
}. ${error.message || error}`,
'Data factory'
);
diff --git a/public/directives/wz-table/lib/listeners.js b/public/directives/wz-table/lib/listeners.js
index 658afa2017..1428895cce 100644
--- a/public/directives/wz-table/lib/listeners.js
+++ b/public/directives/wz-table/lib/listeners.js
@@ -20,6 +20,10 @@ export function wazuhFilter(parameters, filter) {
return filter(parameters.filter);
}
+export function wazuhQuery(parameters, query) {
+ return query(parameters.query, parameters.search);
+}
+
export function wazuhSearch(parameters, instance, search) {
try {
const matchesSpecificPath =
diff --git a/public/directives/wz-table/wz-table-directive.js b/public/directives/wz-table/wz-table-directive.js
index 0004abf6c4..f411d8f902 100644
--- a/public/directives/wz-table/wz-table-directive.js
+++ b/public/directives/wz-table/wz-table-directive.js
@@ -19,14 +19,14 @@ import { parseValue } from './lib/parse-value';
import * as pagination from './lib/pagination';
import { sort } from './lib/sort';
import * as listeners from './lib/listeners';
-import { searchData, filterData } from './lib/data';
+import { searchData, filterData, queryData } from './lib/data';
import { clickAction } from './lib/click-action';
import { initTable } from './lib/init';
import { checkGap } from './lib/check-gap';
const app = uiModules.get('app/wazuh', []);
-app.directive('wzTable', function() {
+app.directive('wzTable', function () {
return {
restrict: 'E',
scope: {
@@ -131,6 +131,17 @@ app.directive('wzTable', function() {
errorHandler
);
+ const query = async (query, search) =>
+ queryData(
+ query,
+ search,
+ instance,
+ wzTableFilter,
+ $scope,
+ fetch,
+ errorHandler
+ );
+
const realTimeFunction = async () => {
try {
$scope.error = false;
@@ -180,7 +191,7 @@ app.directive('wzTable', function() {
$scope.prevPage = () => pagination.prevPage($scope);
$scope.nextPage = async currentPage =>
pagination.nextPage(currentPage, $scope, errorHandler, fetch);
- $scope.setPage = function() {
+ $scope.setPage = function () {
$scope.currentPage = this.n;
$scope.nextPage(this.n);
};
@@ -200,6 +211,10 @@ app.directive('wzTable', function() {
listeners.wazuhSearch(parameters, instance, search)
);
+ $scope.$on('wazuhQuery', (event, parameters) =>
+ listeners.wazuhQuery(parameters, query)
+ );
+
$scope.$on('wazuhRemoveFilter', (event, parameters) =>
listeners.wazuhRemoveFilter(parameters, instance, wzTableFilter, init)
);
diff --git a/public/directives/wz-tag-filter/wz-tag-filter.html b/public/directives/wz-tag-filter/wz-tag-filter.html
new file mode 100644
index 0000000000..75e440ea64
--- /dev/null
+++ b/public/directives/wz-tag-filter/wz-tag-filter.html
@@ -0,0 +1,33 @@
+
\ No newline at end of file
diff --git a/public/directives/wz-tag-filter/wz-tag-filter.js b/public/directives/wz-tag-filter/wz-tag-filter.js
new file mode 100644
index 0000000000..4963826bd5
--- /dev/null
+++ b/public/directives/wz-tag-filter/wz-tag-filter.js
@@ -0,0 +1,257 @@
+/*
+ * Wazuh app - Wazuh search and filter by tags bar
+ * 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 template from './wz-tag-filter.html';
+import { DataFactory } from '../../services/data-factory';
+import { uiModules } from 'ui/modules';
+
+const app = uiModules.get('app/wazuh', []);
+
+app.directive('wzTagFilter', function () {
+ return {
+ restrict: 'E',
+ scope: {
+ path: '=path',
+ queryFn: '&',
+ fieldsModel: '='
+ },
+ controller(
+ $scope,
+ $timeout,
+ apiReq,
+ $document,
+ errorHandler
+ ) {
+ const instance = new DataFactory(
+ apiReq,
+ $scope.path,
+ {}
+ );
+
+ $scope.tagList = [];
+ $scope.groupedTagList = [];
+ $scope.newTag = '';
+ $scope.isAutocomplete = false;
+ $scope.dataModel = [];
+
+ $scope.addTag = (flag = false) => {
+ try {
+ const input = $document[0].getElementById('wz-search-filter-bar-input');
+ input.blur();
+ const term = $scope.newTag.split(':');
+ const obj = { name: term[0], value: term[1] };
+ const isFilter = obj.value;
+ if ((isFilter && Object.keys($scope.fieldsModel).indexOf(obj.name) === -1) ||
+ (!isFilter && (!obj.name || /^\s*$/.test(obj.name)))) {
+ $scope.showAutocomplete(flag);
+ $scope.newTag = '';
+ } else {
+ const tag = {
+ 'id': generateUID(),
+ 'key': obj.name,
+ 'value': obj,
+ 'type': isFilter ? 'filter' : 'search'
+ };
+ const idxSearch = $scope.tagList.find(function (x) { return x.type === 'search' });
+ if (!isFilter && idxSearch) { $scope.removeTag(idxSearch.id, false) };
+ if (!$scope.tagList.find(function (x) { return x.type === 'filter' && x.key === tag.key && x.value.value === tag.value.value })) {
+ $scope.tagList.push(tag);
+ $scope.groupedTagList = groupBy($scope.tagList, 'key');
+ buildQuery($scope.groupedTagList);
+ }
+ $scope.showAutocomplete(flag);
+ $scope.newTag = '';
+ }
+ } catch (error) {
+ errorHandler.handle(error, 'Error adding filter');
+ }
+ };
+
+ const buildQuery = groups => {
+ try {
+ let queryObj = {
+ 'query': '',
+ 'search': ''
+ };
+ let first = true;
+ groups.forEach(function (group, idx1) {
+ const search = group.find(function (x) { return x.type === 'search' });
+ if (search) {
+ queryObj.search = search.value.name;
+ }
+ else {
+ if (!first) {
+ queryObj.query += ';';
+ }
+ const twoOrMoreElements = group.length > 1;
+ if (twoOrMoreElements) {
+ queryObj.query += '('
+ }
+ group.filter(function (x) { return x.type === 'filter' }).forEach(function (tag, idx2) {
+ queryObj.query += tag.key + '=' + tag.value.value;
+ if (idx2 != group.length - 1) {
+ queryObj.query += ',';
+ }
+ });
+ if (twoOrMoreElements) {
+ queryObj.query += ')'
+ }
+ first = false;
+ }
+ });
+ $scope.queryFn({ 'q': queryObj.query, 'search': queryObj.search });
+ } catch (error) {
+ errorHandler.handle(error, 'Error in query request');
+ }
+ }
+
+ const groupBy = (collection, property) => {
+ let i = 0, val, index,
+ values = [], result = [];
+ for (; i < collection.length; i++) {
+ val = collection[i][property];
+ index = values.indexOf(val);
+ if (index > -1 && collection[i].type === 'filter')
+ result[index].push(collection[i]);
+ else {
+ values.push(val);
+ result.push([collection[i]]);
+ }
+ }
+ return result;
+ }
+
+ $scope.addTagKey = (key) => {
+ $scope.newTag = key + ':';
+ $scope.showAutocomplete(true);
+ };
+
+ $scope.addTagValue = (value) => {
+ $scope.newTag = $scope.newTag.substring(0, $scope.newTag.indexOf(':') + 1);
+ $scope.newTag += value;
+ $scope.addTag();
+ };
+
+ $scope.removeTag = (id, deleteGroup) => {
+ if (deleteGroup) {
+ $scope.tagList = $scope.tagList.filter(function (x) { return x.key !== id });
+ } else {
+ $scope.tagList.splice($scope.tagList.findIndex(function (x) { return x.id === id }), 1);
+ }
+ $scope.groupedTagList = groupBy($scope.tagList, 'key');
+ buildQuery($scope.groupedTagList);
+ $scope.showAutocomplete(false);
+ };
+
+ $scope.showAutocomplete = (flag) => {
+ if (flag) {
+ $scope.getAutocompleteContent();
+ }
+ $scope.isAutocomplete = flag;
+ indexAutocomplete(flag);
+ };
+
+ $scope.getAutocompleteContent = () => {
+ const term = $scope.newTag.split(':');
+ const isKey = !term[1] && $scope.newTag.indexOf(':') === -1;
+ $scope.autocompleteContent = { 'title': '', 'isKey': isKey, 'list': [] };
+ $scope.autocompleteContent.title = isKey ? 'Filter keys' : 'Values';
+ if (isKey) {
+ for (let key in $scope.fieldsModel) {
+ if (key.toUpperCase().includes(term[0].toUpperCase())) {
+ $scope.autocompleteContent.list.push(key);
+ }
+ }
+ } else {
+ const model = $scope.dataModel.find(function (x) { return x.key === $scope.newTag.split(':')[0] })
+ if (model) {
+ $scope.autocompleteContent.list = [...new Set(model.list.filter(function (x) { return x.toUpperCase().includes(term[1].toUpperCase()); }))];
+ }
+ }
+ };
+
+ $scope.addSearchKey = (e) => {
+ if ($scope.autocompleteEnter) {
+ $scope.autocompleteEnter = false;
+ }
+ $scope.getAutocompleteContent();
+ };
+
+ const indexAutocomplete = (flag = true) => {
+ $timeout(function () {
+ const bar = $document[0].getElementById('wz-search-filter-bar');
+ const autocomplete = $document[0].getElementById('wz-search-filter-bar-autocomplete');
+ const input = $document[0].getElementById('wz-search-filter-bar-input');
+ autocomplete.style.left = input.offsetLeft - bar.scrollLeft + 'px';
+ if (flag) {
+ input.focus();
+ }
+ $('#wz-search-filter-bar-autocomplete-list li').removeClass('selected');
+ }, 100);
+ }
+
+ const load = async () => {
+ try {
+ const result = await instance.fetch();
+ Object.keys($scope.fieldsModel).forEach(function (key) {
+ $scope.dataModel.push({ 'key': key, 'list': $scope.fieldsModel[key] });
+ });
+ return;
+ } catch (error) {
+ return Promise.reject(error);
+ }
+ };
+
+ const generateUID = () => {
+ // I generate the UID from two parts here
+ // to ensure the random number provide enough bits.
+ let firstPart = (Math.random() * 46656) | 0;
+ let secondPart = (Math.random() * 46656) | 0;
+ firstPart = ('000' + firstPart.toString(36)).slice(-3);
+ secondPart = ('000' + secondPart.toString(36)).slice(-3);
+ return firstPart + secondPart;
+ }
+
+ $('#wz-search-filter-bar-input').bind('keydown', function (e) {
+ let $current = $('#wz-search-filter-bar-autocomplete-list li.selected');
+ if ($current.length === 0 && (e.keyCode === 38 || e.keyCode === 40)) {
+ $('#wz-search-filter-bar-autocomplete-list li').first().addClass('selected');
+ $current = $('#wz-search-filter-bar-autocomplete-list li.selected');
+ } else {
+ let $next;
+ switch (e.keyCode) {
+ case 13: // enter
+ if ($current.text().trim() && !/^\s*$/.test($current.text().trim())) {
+ $scope.autocompleteEnter = true;
+ $scope.autocompleteContent.isKey ? $scope.addTagKey($current.text().trim()) : $scope.addTagValue($current.text().trim());
+ }
+ break;
+ case 38: // if the UP key is pressed
+ $next = $current.prev();
+ break;
+ case 40: // if the DOWN key is pressed
+ $next = $current.next();
+ break;
+ }
+ if ($next && $next.length > 0 && (e.keyCode === 38 || e.keyCode === 40)) {
+ const input = $document[0].getElementById('wz-search-filter-bar-input');
+ input.focus();
+ $('#wz-search-filter-bar-autocomplete-list li').removeClass('selected');
+ $next.addClass('selected');
+ }
+ }
+ });
+ load();
+ },
+ template
+ };
+});
diff --git a/public/directives/wz-tag-filter/wz-tag-filter.less b/public/directives/wz-tag-filter/wz-tag-filter.less
new file mode 100644
index 0000000000..634f9ceea6
--- /dev/null
+++ b/public/directives/wz-tag-filter/wz-tag-filter.less
@@ -0,0 +1,119 @@
+/*
+ * Wazuh app - Search and filter by tags bar stylesheet
+ * 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.
+ */
+
+/* -------------------------------------------------------------------------- */
+/* ----------- Wazuh search and filter by tags bar stylesheet --------------- */
+/* -------------------------------------------------------------------------- */
+
+#wz-search-filter-bar {
+ padding: 8px;
+ margin-top: 15px;
+ overflow: auto;
+}
+
+#wz-search-filter-bar > .fa-search{
+ padding: 8px 3px;
+ float: left;
+ opacity: .5;
+ width: calc(~'0% + 25px');
+}
+
+.wz-tags {
+ height: 30px;
+ width: calc(~'100% - 25px');
+ display: flex;
+}
+
+.wz-tags > ul{
+ flex: 0 1 auto;
+ display: flex;
+}
+
+.wz-tags > ul li{
+ white-space: nowrap;
+ display: flex;
+}
+
+.wz-tags > ul li .grouped{
+ background: #d9d9d9;
+ border-radius: 3px;
+ padding: 3px;
+ height: 36px;
+ margin-top: -3px;
+ display: flex;
+}
+
+.wz-tag-item {
+ color: white;
+ border-radius: 3px;
+ display: inline-block;
+ padding: 3px 5px;
+ background: #0079a5;
+ margin: 1px 3px;
+}
+
+.wz-tag-item-connector {
+ font-size: 10px;
+ line-height: 30px;
+ color: #005571;
+ font-weight: bold;
+ padding: 0px 5px;
+}
+
+.wz-tag-remove-button{
+ color: #005571;
+}
+
+.wz-tag-remove-button-group{
+ padding: 4px 2px;
+ color: gray;
+}
+
+.wz-search-filter-bar-input{
+ border: none;
+ padding: 5px;
+ width: auto;
+ margin-left: 5px;
+ min-width: 225px;
+ flex: 1;
+}
+
+.wz-search-filter-bar-autocomplete{
+ min-width: 200px;
+ background: #000000d9;
+ position: absolute;
+ border: 1px solid black;
+ border-radius: 5px;
+ padding: 10px 0;
+ margin-top: 30px;
+}
+
+.wz-search-filter-bar-autocomplete p b{
+ color: white;
+ font-weight: 700;
+ padding: 10px;
+}
+
+.wz-search-filter-bar-autocomplete ul li{
+ color: white;
+ padding: 0 10px;
+}
+
+.wz-search-filter-bar-autocomplete ul li span{
+ padding-left: 5px;
+}
+
+.wz-search-filter-bar-autocomplete ul li:hover, .wz-search-filter-bar-autocomplete ul li.selected{
+ background: black;
+ color:#0079a5;
+ cursor: pointer;
+}
diff --git a/public/templates/agents-prev/agents-prev.html b/public/templates/agents-prev/agents-prev.html
index 8a1d03fc79..07ddc836d3 100644
--- a/public/templates/agents-prev/agents-prev.html
+++ b/public/templates/agents-prev/agents-prev.html
@@ -34,85 +34,26 @@
Last registered agent
-
{{ctrl.lastAgent.name}}
-
{{ctrl.lastAgent.name}} (manager)
+
{{ctrl.lastAgent.name}}
+
{{ctrl.lastAgent.name}}
+ (manager)
Higher activity
-
{{ctrl.mostActiveAgent.name}}
+
{{ctrl.mostActiveAgent.name}}
{{ctrl.mostActiveAgent.name}}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
+
-
+
@@ -122,4 +63,4 @@
-
+
\ No newline at end of file