ntopng/http_src/vue/dashboard.vue
2025-11-13 16:36:49 +01:00

1321 lines
47 KiB
Vue

<!-- (C) 2023 - ntop.org -->
<template>
<div class='row'>
<!-- <Dropdown v-for="(t, t_index) in top_table_array" -->
<!-- :ref="el => { top_table_dropdown_array[t_index] = el }"> -->
<!-- <template v-slot:title> -->
<!-- <Spinner :show="t.show_spinner" size="1rem" class="me-1"></Spinner> -->
<!-- <a class="ntopng-truncate" :title="t.title">{{ t.label }}</a> -->
<!-- </template> -->
<!-- <template v-slot:menu> -->
<!-- <a v-for="opt in t.options" style="cursor:pointer; display: block;" -->
<!-- @click="add_top_table_filter(opt, $event)" class="ntopng-truncate tag-filter " -->
<!-- :title="opt.value">{{ opt.label }}</a> -->
<!-- </template> -->
<!-- </Dropdown> -->
<DateTimeRangePicker v-if="enable_date_time_range_picker" class="dontprint"
:disabled_date_picker="disable_date_time_picker" id="dashboard-date-time-picker" :round_time="true"
min_time_interval_id="min" @epoch_change="set_components_epoch_interval">
<!-- Report Selector -->
<template v-slot:begin>
<div class="me-2">
<SelectSearch v-model:selected_option="selected_report_template" :options="reports_templates"
@select_option="select_report_template">
</SelectSearch>
</div>
</template>
<!-- Report Toolbox (Store, Save, ...) -->
<template v-slot:extra_buttons>
<button class="btn btn-link btn-sm" type="button" @click="show_store_report_modal"
:title="_i18n('dashboard.store')">
<i class="fa-solid fa-floppy-disk"></i>
</button>
<button class="btn btn-link btn-sm" type="button" @click="show_open_report_modal"
:title="_i18n('dashboard.open')">
<i class="fa-solid fa-folder-open"></i>
</button>
<button class="btn btn-link btn-sm" type="button" @click="download_report" :title="_i18n('download')">
<i class="fa-solid fa-file-arrow-down"></i>
</button>
<button class="btn btn-link btn-sm" type="button" @click="show_upload_report_modal"
:title="_i18n('upload')">
<i class="fa-solid fa-file-arrow-up"></i>
</button>
<button class="btn btn-link btn-sm" type="button" @click="print_report"
:title="_i18n('dashboard.print')">
<i class="fas fa-print"></i>
</button>
<button v-if="allow_edit" class="btn btn-link btn-sm" type="button" @click="show_new_template_modal"
:title="_i18n('dashboard.new_template')">
<i class="fas fa-folder-plus"></i>
</button>
<button v-if="allow_edit && selected_report_template.allow_edit" class="btn btn-link btn-sm"
:class="edit_mode ? 'text-warning' : ''" type="button" @click="toggle_edit_mode"
:title="_i18n('dashboard.edit_mode')">
<i class="fas fa-pen-to-square"></i>
</button>
</template>
</DateTimeRangePicker>
<!-- Filters -->
<div class="form-group d-flex align-items-end"
:class="[loading_filters ? 'justify-content-center align-items-center' : '']">
<Spinner :show="loading_filters" size="1rem" class="me-1"></Spinner>
<template v-for="filter_id in filters_to_show">
<div class="dropdown me-3 d-inline-block">
<label class="ms-1 my-auto me-2 filters-label"><b>{{ _i18n('db_search.' + filter_id) }}</b></label>
<SelectSearch v-model:selected_option="selected_filters[filter_id]"
:options="filtered_filters[filter_id]" :disabled="second_load" :style="'width: 50%;'"
theme="bootstrap-5" dropdown_size="small"
@select_option="select_filter(selected_filters[filter_id], filter_id)">
</SelectSearch>
</div>
</template>
<template v-if="Object.keys(filters_to_show).length > 0">
<div class="d-flex justify-content-center align-items-center">
<div class="me-2">
<div>
<label class="my-auto me-2"></label>
</div>
<button type="button" class="btn btn-sm btn-primary" @click="reset_filters">{{ _i18n('reset')
}}</button>
</div>
<div class="mt-4">
<Spinner :show="second_load" size="1rem" class="me-1"></Spinner>
</div>
</div>
</template>
</div>
<!-- Template Editor Toolbox -->
<div v-if="edit_mode" class="me-auto mt-2 flex-wrap d-flex">
<button class="btn btn-lg btn-link text-warning" type="button" @click="show_add_template_component_modal"
:title="_i18n('dashboard.add_component')">
<i class="fas fa-square-plus"></i>
</button>
<button class="btn btn-lg btn-link text-warning" type="button" @click="show_delete_template_modal"
:title="_i18n('dashboard.del_template')">
<i class="fas fa-trash-can"></i>
</button>
<h2 class="text-warning" style="margin-top: 0.5rem; margin-left: 1rem">{{ _i18n('dashboard.edit_mode') }}
</h2>
<!--
<div style="margin: auto"></div>
<h4><span class="badge bg-warning">{{ _i18n('dashboard.edit_mode') }}</span></h4>
-->
</div>
<!-- Report Title and small buttons -->
<div v-if="enable_report_title" class="mt-3" style="margin-bottom:-0.5rem; display: inline">
<h3 style="text-align:center;">{{ report_title }}
<span v-if="enable_small_picker">
<template v-if="enable_small_picker_actions">
<button class="btn btn-link btn-sm" type="button" @click="download_report"
:title="_i18n('download')">
<i class="fa-solid fa-file-arrow-down"></i>
</button>
<button class="btn btn-link btn-sm" type="button" @click="show_upload_report_modal"
:title="_i18n('upload')">
<i class="fa-solid fa-file-arrow-up"></i>
</button>
</template>
<button class="btn btn-link btn-sm" type="button" @click="print_report"
:title="_i18n('dashboard.print')">
<i class="fas fa-print"></i>
</button>
</span>
</h3>
</div>
<!-- Report/Dashboard Content -->
<div ref="report_box" id="drag-zone" class="row" :key="components">
<!-- Warning Message -->
<div v-if="warning_message" class="col-sm mt-1">
<div class="alert alert-warning">
{{ warning_message }}
</div>
</div>
<!-- Empty template message -->
<div v-if="components_loaded && !components.length && !edit_mode" class="col-sm mt-1">
<div class="alert alert-secondary sm-1 text-center"
style="width:40%; margin: auto; margin-top: 5vh; margin-bottom: 5vh">
<h4 class="alert-heading">
{{ _i18n("dashboard.empty_template") }}
</h4>
<p class="mb-0">
{{ _i18n("dashboard.empty_template_note") }}
</p>
</div>
</div>
<!-- Rendered Components -->
<template v-for="c in components">
<Box :color="(c.active && c.color) || c.inactive_color" :width="c.width" :height="c.height" :id="c.id"
class="drag-item">
<template v-slot:box_title>
<div v-if="c.i18n_name" class="mb-2 modal-header">
<h4 class="modal-title">
{{ c.custom_name ? c.custom_name : _i18n(c.i18n_name) }}
<span style="color: gray">
{{ c.time_offset ? _i18n('dashboard.' + (is_live ? 'time_ago' : 'time_offset_list')
+ '.' + c.time_offset) : '' }}
</span>
</h4>
<div v-if="edit_mode" class="modal-close">
<div class='btn-group'>
<button type="button" class="btn-close btn-edit me-1" :data-component-id="c.id"
@click="show_edit_template_component"></button>
<button type="button" class="btn-close" :data-component-id="c.id"
@click="remove_template_component"></button>
</div>
</div>
</div>
</template>
<template v-slot:box_content>
<div>
<Loading :isLoading="c.isLoading"></Loading>
<component :is="components_dict[c.component]" :id="c.id" :style="component_custom_style(c)"
:epoch_begin="c.epoch_begin" :epoch_end="c.epoch_end" :i18n_title="c.i18n_name"
:ifid="c.ifid ? c.ifid.toString() : context.ifid.toString()" :max_width="c.width"
:max_height="c.height" :params="c.params"
:get_component_data="get_component_data_func(c)"
:set_component_attr="set_component_attr_func(c)" :csrf="context.csrf"
:filters="c.filters">
</component>
</div>
</template>
<template v-slot:box_footer>
<span v-if="c.component != 'empty' && c.i18n_name && !disable_date"
style="color: lightgray;font-size:12px;">
{{ component_interval(c) }}
</span>
</template>
</Box>
</template>
</div>
</div> <!-- div row -->
<ModalSave ref="modal_store_report" :get_suggested_file_name="get_suggested_report_name" :store_file="store_report"
:csrf="context.csrf" :title="_i18n('dashboard.store')">
</ModalSave>
<ModalOpen ref="modal_open_report" :list_files="list_reports" :open_file="open_report" :delete_file="delete_report"
:csrf="context.csrf" :title="_i18n('dashboard.open')" :file_title="_i18n('report.report_name')">
</ModalOpen>
<ModalUpload ref="modal_upload_report" :upload_file="upload_report" :title="_i18n('upload')"
:file_title="_i18n('report.file')">
</ModalUpload>
<ModalSave ref="modal_new_template" :get_suggested_file_name="get_suggested_template_name"
:store_file="new_template" :allow_spaces="true" :csrf="context.csrf" :title="_i18n('dashboard.new_template')">
</ModalSave>
<ModalSelectComponent ref="modal_add_template_component" :list_components="list_template_components"
:add_component="add_template_component" :csrf="context.csrf" :title="_i18n('dashboard.add_component')">
</ModalSelectComponent>
<ModalEditComponent ref="modal_edit_template_component" :csrf="context.csrf" @edit="edit_template_component">
</ModalEditComponent>
<ModalDeleteConfirm ref="modal_delete_template" :title="_i18n('dashboard.del_template')"
:body="_i18n('dashboard.del_template_confirm')" @delete="delete_template">
</ModalDeleteConfirm>
</template>
<script setup>
import { ref, onMounted, onBeforeMount, computed, nextTick } from "vue";
import { ntopng_status_manager, ntopng_url_manager, ntopng_utility, ntopng_events_manager, ntopng_sync } from "../services/context/ntopng_globals_services";
import { default as DateTimeRangePicker } from "./date-time-range-picker.vue";
import { default as ModalSave } from "./modal-file-save.vue";
import { default as ModalOpen } from "./modal-file-open.vue";
import { default as ModalUpload } from "./modal-file-upload.vue";
import { default as ModalSelectComponent } from "./modal-select-component.vue";
import { default as ModalEditComponent } from "./modal-edit-component.vue";
import { default as ModalDeleteConfirm } from "./modal-delete-confirm.vue";
import { default as Spinner } from "./spinner.vue";
import { default as Loading } from "./loading.vue";
import { default as Box } from "./dashboard-box.vue";
import { default as EmptyComponent } from "./dashboard-empty.vue";
import { default as TableComponent } from "./dashboard-table.vue";
import { default as BadgeComponent } from "./dashboard-badge.vue";
import { default as PieComponent } from "./dashboard-pie.vue";
import { default as BarComponent } from "./dashboard-bar.vue";
import { default as LateralPieComponent } from "./dashboard-lateral-pie.vue";
import { default as BoxOverview } from "./dashboard-box-overview.vue";
import { default as TimeseriesComponent } from "./dashboard-timeseries.vue";
import { default as TopTimeseriesComponent } from "./dashboard-top-timeseries.vue";
import { default as SankeyComponent } from "./dashboard-sankey.vue";
import { default as SelectSearch } from "./select-search.vue";
import { default as dataUtils } from "../utilities/data-utils";
const _i18n = (t) => i18n(t);
const timeframes_dict = ntopng_utility.get_timeframes_dict();
const props = defineProps({
context: Object,
});
const components_dict = {
"badge": BadgeComponent,
"empty": EmptyComponent,
"pie": PieComponent,
"lateral-pie": LateralPieComponent,
"box-overview": BoxOverview,
"timeseries": TimeseriesComponent,
"top-timeseries": TopTimeseriesComponent,
"bar": BarComponent,
"table": TableComponent,
"sankey": SankeyComponent,
}
const loading_filters = ref(false);
const second_load = ref(false);
const page_id = "page-dashboard";
const allow_edit = props.context.allow_edit || false;
const report_box = ref(null);
const modal_store_report = ref(null);
const modal_open_report = ref(null);
const modal_upload_report = ref(null);
const modal_add_template_component = ref(null);
const modal_edit_template_component = ref(null);
const modal_delete_template = ref(null)
const main_epoch_interval = ref(null);
const components = ref([]);
const components_loaded = ref(false);
const selected_filters = ref({});
const all_available_filters = ref({});
const filtered_filters = ref({});
const filters_to_show = ref([]);
const template_filters = ref([]);
const reports_templates = ref([]);
const selected_report_template = ref({});
const warning_message = ref("");
let components_info = {};
let data_from_backup = false;
let printable = false;
const edit_mode = ref(false);
let template_sortable = null;
const modal_new_template = ref(null);
const isFirstLoading = ref(false);
const is_live = computed(() => {
return props.context.page != "report" && props.context.page != "vs-report";
});
const enable_date_time_range_picker = computed(() => {
return props.context.page == "report"
&& !printable;
});
const enable_small_picker = computed(() => {
return props.context.page == "vs-report";
});
const enable_small_picker_actions = computed(() => {
return true; // Set to false for hiding open/save actions in the small picker
});
const disable_date = computed(() => {
return (selected_report_template.value?.toolbox?.time?.hide == true) || props.context.disable_date;
});
const disable_date_time_picker = computed(() => {
const disabled = selected_report_template.value.is_open_report == true
|| disable_date.value;
return disabled;
});
const enable_report_title = computed(() => {
const enable = selected_report_template.value.is_open_report == true
|| props.context.page == "vs-report";
return enable;
});
const report_title = computed(() => {
let title = "";
if (selected_report_template.value.is_open_report) {
title = `Report: ${selected_report_template.value.value}`;
} else if (props.context.title) {
title = props.context.title;
}
return title;
});
const component_custom_style = computed(() => {
return (c) => {
if (c.params.custom_style != null && (!printable || c.params.custom_print_style)) {
return c.params.custom_style;
} else if (c.params.custom_print_style && printable == true) {
return c.params.custom_print_style;
}
return "";
};
});
const component_interval = computed(() => {
return (c) => {
const time_interval_string = get_time_interval_string(c.epoch_begin, c.epoch_end);
return time_interval_string;
};
});
/* Param report_template is optional (uses url report_template or props.context.template otherwise) */
async function set_template(report_template) {
let epoch_interval = null;
printable = ntopng_url_manager.get_url_entry("printable") == "true";
if (props.context.page == "report" || props.context.page == "vs-report") {
if (props.context.page == "report") {
epoch_interval = ntopng_utility.check_and_set_default_time_interval(undefined, undefined, true, "min");
} else if (props.context.page == "vs-report") {
epoch_interval = ntopng_utility.check_and_set_default_time_interval(undefined, undefined, true);
}
main_epoch_interval.value = epoch_interval;
}
await set_templates_list(report_template);
let report_name = ntopng_url_manager.get_url_entry("report_name");
if (report_name != null && report_name != "") {
/* Report name provided - open a report backup */
await open_report(report_name);
} else {
/* Load a template and build a new report */
await load_components(epoch_interval, selected_report_template.value.value);
}
if (printable == true) {
set_report_title();
// await nextTick();
// ntopng_sync.ready("print_report");
}
components_loaded.value = true
}
onBeforeMount(async () => {
set_template();
});
onMounted(async () => {
if (props.context.page == "dashboard") {
start_dashboard_refresh_loop();
}
// if (printable == true) {
// await ntopng_sync.on_ready("print_report");
// }
});
async function set_templates_list(report_template) {
const url_request = props.context.template_list_endpoint;
let res = await ntopng_utility.http_request(url_request);
if (res?.list == null) { return; }
let templates_list = res.list.map((t) => {
return {
value: t.name,
label: t.label,
disabled: false,
toolbox: t.toolbox,
is_open_report: false,
allow_edit: t.allow_edit,
};
});
templates_list.sort((a, b) => a.label.toUpperCase().localeCompare(b.label.toUpperCase()));
reports_templates.value = templates_list;
const report_template_value = report_template ||
ntopng_url_manager.get_url_entry("report_template") ||
props.context.template;
props.context.template = report_template_value;
selected_report_template.value = reports_templates.value.find((t) => t.value == report_template_value);
if (selected_report_template.value == null) {
selected_report_template.value = reports_templates.value[0];
}
if (!selected_report_template.value.allow_edit && edit_mode.value) {
toggle_edit_mode();
}
}
let dasboard_loop_interval;
/* Dashboard update interval/frequency */
const REFRESH_INTERVAL_SEC = 10;
function start_dashboard_refresh_loop() {
const loop_interval = REFRESH_INTERVAL_SEC * 1000;
dasboard_loop_interval = setInterval(() => {
set_components_epoch_interval();
}, loop_interval);
}
function set_components_filter(filter_id, filter_value) {
if (filter_value) { filter_value = filter_value + ";eq"; }
ntopng_url_manager.set_key_to_url(filter_id, filter_value);
components.value.forEach((c, i) => {
update_component_filters(c, filter_id, filter_value);
});
}
function set_components_epoch_interval(epoch_interval) {
if (epoch_interval) {
main_epoch_interval.value = epoch_interval;
}
components.value.forEach((c, i) => {
update_component_epoch_interval(c, epoch_interval);
});
}
/* This is used to reset the filters putting all of them to the ALL value */
async function reset_filters() {
/* Iterate all the filters available */
for (const [filter, value] of Object.entries(all_available_filters.value)) {
/* Set each filter to the ALL value (first value) */
set_components_filter(filter, value[0].value);
selected_filters.value[filter] = value[0];
/* Hide all the needed filters */
}
filters_to_show.value = await load_filters(template_filters.value, null, true);
}
/* This function loads the filters */
async function load_filters(filters_available, res, show_second_load) {
(!show_second_load) ? loading_filters.value = true : second_load.value = true;
const added_filters_list = [];
if (!res && (props.context.report_filters_endpoint != null)) {
let retrieve_filters = ''
let extra_params = ''
let tmp = {}
filters_available.forEach((element) => {
if (element.show_only_if_selected) {
const selected = ntopng_url_manager.get_url_entry(element.show_only_if_selected) || ''
if (selected !== '') {
retrieve_filters = `${retrieve_filters}${element.name},`
tmp[element.show_only_if_selected] = selected
}
} else {
retrieve_filters = `${retrieve_filters}${element.name},`
}
})
for (const [key, value] of Object.entries(tmp)) {
extra_params = `${key}=${value}&${extra_params}`
}
retrieve_filters = retrieve_filters.slice(0, -1);
/* Being a mix of flows, historical/hourly, keep in mind
* that not all the filters are available in the hourly, so retrieve only the
* ones available in both
*/
res = await ntopng_utility.http_request(`${props.context.report_filters_endpoint}?hide_exporters_name=true&filters_to_display=${retrieve_filters}&aggregated=true&${extra_params}`);
}
filters_available.forEach((element) => {
const id = element?.name || "";
const filter_options = res.find((el) => el.id == id)?.options;
/* Check the filters available, if no filter or only 1 filter is provided, hide the dropdown */
if (filter_options && filter_options.length > 0) {
const selected = ntopng_url_manager.get_url_entry(id)
let all_label = i18n('db_search.all.' + id)
if (dataUtils.isEmptyOrNull(all_label)) {
all_label = i18n('all') + " " + i18n('db_search.' + id);
}
/* Add the 'All' filter */
/* To be safe, add a default name */
filter_options.unshift({
value: null,
label: all_label,
display_more_filters: filter_options[0].display_more_filters
});
if (selected && selected !== '') {
const tmp_value = filter_options.find((el) => selected.startsWith(el.value))
selected_filters.value[id] = tmp_value
} else {
selected_filters.value[id] = filter_options[0];
}
all_available_filters.value[id] = filter_options;
filtered_filters.value[id] = filter_options
added_filters_list.push(id);
}
});
(!show_second_load) ? loading_filters.value = false : second_load.value = false;
return added_filters_list;
}
async function load_components(epoch_interval, template_name) {
/* Enable REST calls */
data_from_backup = false;
let url_request = `${props.context.template_endpoint}?template=${template_name}`;
let res = await ntopng_utility.http_request(url_request);
components.value = res.list.filter((c) => components_dict[c.component] != null)
.map((c, index) => {
let c_ext = {
filters: {},
component_id: `auto_${c.id}_${index}`,
isLoading: true,
...c
};
update_component_epoch_interval(c_ext, epoch_interval);
return c_ext;
});
reset_filters();
if (res.filters) {
template_filters.value = res.filters
filters_to_show.value = await load_filters(res.filters);
}
await nextTick();
}
function update_component_epoch_interval(c, epoch_interval) {
const interval_seconds = timeframes_dict[c.time_window || "5_min"];
if (epoch_interval == null) {
const epoch_end = ntopng_utility.get_utc_seconds();
epoch_interval = { epoch_begin: epoch_end - interval_seconds, epoch_end: epoch_end };
}
const utc_offset = timeframes_dict[c.time_offset] || 0;
c.epoch_begin = epoch_interval.epoch_begin - utc_offset;
c.epoch_end = epoch_interval.epoch_end - utc_offset;
}
function update_component_filters(c, filter_id, filter_value) {
c.filters[filter_id] = filter_value;
}
/* ********************************************* */
/* This function is called whenever a filter is clicked */
async function select_filter(option, filter_id) {
/* Set the filter, ready for the rest */
set_components_filter(filter_id, option.value);
if (option.display_more_filters) {
filters_to_show.value = await load_filters(template_filters.value, null, true);
}
}
/* ********************************************* */
function select_report_template() {
if (!selected_report_template.value.allow_edit && edit_mode.value) {
toggle_edit_mode();
}
if (printable == true) {
set_report_title();
}
if (selected_report_template.value.is_open_report == true) {
return;
}
components_info = {};
update_templates_list();
const global_status = ntopng_status_manager.get_status(true);
let epoch_interval = { epoch_begin: global_status.epoch_begin, epoch_end: global_status.epoch_end };
if (data_from_backup == true) { // last report selected it was a saved report and then we must to restore default timestamp
epoch_interval = ntopng_utility.set_default_time_interval(undefined, "min");
ntopng_events_manager.emit_event(ntopng_events.EPOCH_CHANGE, epoch_interval, props.context.page);
}
load_components(epoch_interval, selected_report_template.value.value);
}
function show_store_report_modal() {
modal_store_report.value.show();
}
function show_open_report_modal() {
modal_open_report.value.show();
}
function show_upload_report_modal() {
modal_upload_report.value.show();
}
function get_suggested_report_name() {
let name = "report";
if (props.context.page == "vs-report") {
name = props.context.title;
} else if (main_epoch_interval.value &&
main_epoch_interval.value.epoch_end) {
name += "-" + ntopng_utility.from_utc_to_server_date_format(main_epoch_interval.value.epoch_end * 1000, 'DD-MM-YYYY');
}
return name;
}
const upload_report = async (content_string) => {
let content = JSON.parse(content_string);
set_report(content, content.name);
ntopng_url_manager.delete_key_from_url("report_name");
}
function set_report(content, name) {
update_templates_list(name);
const epoch_status = { epoch_begin: content.epoch_begin, epoch_end: content.epoch_end };
ntopng_events_manager.emit_event(ntopng_events.EPOCH_CHANGE, epoch_status, props.context.page);
load_report(content);
}
const list_reports = async () => {
let url = `${props.context.report_list_endpoint}?ifid=${props.context.ifid}`;
let files_obj = await ntopng_utility.http_request(url);
let files = ntopng_utility.object_to_array(files_obj);
/* Return array of [{ name: String, epoch: Number }, ...] */
return files;
}
const load_report = async (content) => {
let tmp_epoch_interval = {
epoch_begin: content.epoch_begin,
epoch_end: content.epoch_end
};
let tmp_template = content.template;
let tmp_components_data = content.data;
let tmp_components_info = {};
for (let key in tmp_components_data) {
let info = {
data: tmp_components_data[key],
};
tmp_components_info[key] = info;
}
/* Disable REST calls */
data_from_backup = true;
/* Set the cached data from the backup */
components_info = tmp_components_info;
/* Change the components (template) from the backup */
components.value = tmp_template;
/* Change the time interval on components */
set_components_epoch_interval(tmp_epoch_interval);
}
const open_report = async (file_name) => {
let url = `${props.context.report_open_endpoint}?ifid=${props.context.ifid}&report_name=${file_name}`;
let content = await ntopng_utility.http_request(url);
if (content) {
set_report(content, file_name);
warning_message.value = "";
} else {
warning_message.value = _i18n("report.unable_to_open");
}
}
function update_templates_list(report_name_to_open) {
reports_templates.value = reports_templates.value.filter((t) => t.is_open_report == false);
if (report_name_to_open == null) { // in this case is selected a report_template
ntopng_url_manager.set_key_to_url("report_template", selected_report_template.value.value);
props.context.template = selected_report_template.value.value;
ntopng_url_manager.delete_key_from_url("report_name");
return;
}
let t_entry = {
value: report_name_to_open,
label: _i18n("dashboard.custom"),
disabled: false,
toolbox: null,
is_open_report: true,
allow_edit: false
};
reports_templates.value.push(t_entry);
selected_report_template.value = t_entry;
props.context.template = report_name_to_open;
ntopng_url_manager.set_key_to_url("report_name", selected_report_template.value.value);
ntopng_url_manager.delete_key_from_url("report_template");
}
const delete_report = async (file_name) => {
let success = false;
let params = {
csrf: props.context.csrf,
ifid: props.context.ifid,
report_name: file_name
};
let url = `${props.context.report_delete_endpoint}`;
try {
let headers = {
'Content-Type': 'application/json'
};
await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(params) });
success = true;
} catch (err) {
console.error(err);
}
return success;
}
/* Dump report content - keep in sync with dashboard_utils.build_report (lua) */
const serialize_report = async (name) => {
let tmp_components_data = {};
for (var key in components_info) {
tmp_components_data[key] = await components_info[key].data;
}
let content = {
version: "1.0", // Report dump version
name: name,
template: components.value,
data: tmp_components_data
};
if (main_epoch_interval.value &&
main_epoch_interval.value.epoch_begin &&
main_epoch_interval.value.epoch_end) {
content.epoch_begin = main_epoch_interval.value.epoch_begin;
content.epoch_end = main_epoch_interval.value.epoch_end;
}
return JSON.stringify(content);
}
const store_report = async (file_name) => {
let success = false;
let data = {
csrf: props.context.csrf,
ifid: props.context.ifid,
report_name: file_name,
content: await serialize_report(file_name)
};
let url = `${props.context.report_store_endpoint}`;
try {
let headers = {
'Content-Type': 'application/json'
};
await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(data) });
success = true;
} catch (err) {
console.error(err);
}
return success;
}
async function download_report() {
var name = get_suggested_report_name();
var filename = name + '.json';
var content = await serialize_report(name);
var element = document.createElement('a');
element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(content));
element.setAttribute('download', filename);
element.style.display = 'none';
document.body.appendChild(element);
element.click();
document.body.removeChild(element);
}
function print_report() {
if (printable == true) {
window.print();
return false;
}
let url_params_obj = ntopng_url_manager.get_url_object();
url_params_obj.printable = true;
const params = ntopng_url_manager.obj_to_url_params(url_params_obj);
let url = `${window.location.origin}${window.location.pathname}?${params}`;
//const print_key = "printable";
//ntopng_url_manager.set_key_to_url(print_key, true);
ntopng_url_manager.open_new_window(url);
//ntopng_url_manager.delete_key_from_url(print_key);
// $(report_box.value).print({mediaPrint: true, timeout: 1000});
// $(report_box.value).print();
}
function get_time_interval_string(epoch_begin, epoch_end) {
if (disable_date.value == true) { return ""; }
const epoch_begin_msec = epoch_begin * 1000;
const epoch_end_msec = epoch_end * 1000;
const begin_date = ntopng_utility.from_utc_to_server_date_format(epoch_begin_msec, 'DD/MM/YYYY');
const begin_time = ntopng_utility.from_utc_to_server_date_format(epoch_begin_msec, 'HH:mm:ss');
const end_date = ntopng_utility.from_utc_to_server_date_format(epoch_end_msec, 'DD/MM/YYYY');
const end_time = ntopng_utility.from_utc_to_server_date_format(epoch_end_msec, 'HH:mm:ss');
const begin = `${begin_date} ${begin_time}`;
const end = (begin_date == end_date) ? `${end_time}` : `${end_date} ${end_time}`;
return `${begin} - ${end}`;
}
function set_report_title() {
const epoch_interval = ntopng_status_manager.get_status(true);
const time_interval_string = get_time_interval_string(epoch_interval.epoch_begin, epoch_interval.epoch_end);
let title = `ntopng - Report ${selected_report_template.value.label} ${time_interval_string}`;
document.title = title;
}
/* *************************************************** */
/* This function returns true if the previous parameters and the current one
* are different. This is used in order to do not make the same request multiple times
* e.g. a component is not yet loaded and the same request is resent
*/
function check_diff_params(previous_params, current_params) {
/* Empty params, return true, they are different */
if (!previous_params || !current_params)
return true;
/* Check the length, if it's different return true, a new/removed filter/param */
if (Object.keys(previous_params).length != Object.keys(current_params).length)
return true;
/* Same length, so check that all the key - value are present in both, if not, they are different */
for (const [key, value] of Object.entries(previous_params)) {
/* Check for different params */
if (!current_params[key] || (current_params[key] !== value))
return true;
}
/* The for is ended, meaning that all parameters were equal */
return false;
}
/* *************************************************** */
/* Callback to request REST data from components */
function get_component_data_func(component) {
const get_component_data = async (url, query_params, post_params, refresh_epoch, skip) => {
if (skip === true) {
component.isLoading = false;
return;
}
if (!isFirstLoading.value) {
component.isLoading = true;
}
let info = {};
if (data_from_backup) {
// backward compatibility (component_id was not defined)
if (component.component_id == null) component.component_id = component.id;
if (!components_info[component.component_id]) { /* Safety check */
console.error("No data for " + component.component_id);
info.data = {};
} else {
info = components_info[component.component_id];
}
component.isLoading = false;
} else {
/* datasource_id is an optimization for components getting live data
* from the same endpoint (e.g. multiple badges in the infrastructure dashboard) */
let datasource_id = component.datasource_id ? component.datasource_id : component.component_id
/* Check if there is already a promise for the same request */
let pending = false;
if (components_info[datasource_id]) {
info = components_info[datasource_id];
if (info.data && !info.data.done) {
pending = true;
await info.data; /* wait in case of previous pending requests */
component.isLoading = false;
}
}
if (pending /* pending request from previous iteration (slow) */
/* or other component calling the same endpoint (same time slot) */
|| !check_diff_params(info.filters, query_params)) {
/* Use data from other/pending requests */
} else {
/* Request fresh data */
if (!url) {
component.isLoading = false
return;
}
/* If infrastructure monitor, call the aggregator endpoint */
if (props.context.is_infrastructure && !url.includes("infrastructure")) {
const infrastructure_proxy_url = "/lua/pro/rest/v2/get/infrastructure/aggregate.lua";
query_params['endpoint'] = url;
query_params['component'] = component.component;
url = infrastructure_proxy_url;
}
let url_params = ntopng_url_manager.obj_to_url_params(query_params);
/* Push ifid to the parameters (e.g. "ts_query=ifid:$IFID$" */
url_params = url_params.replaceAll("%24IFID%24" /* $IFID$ */, props.context.ifid);
const data_url = `${url}?${url_params}`;
if (post_params) {
info.data = ntopng_utility.http_post_request(data_url, post_params)
} else {
info.data = ntopng_utility.http_request(data_url);
}
info.filters = query_params;
components_info[datasource_id] = info;
info.data.then(() => {
info.data.done = true;
component.isLoading = false
/* Disable the periodic loading component in case of Dashboard */
if (!isFirstLoading.value && props.context.page !== "report") {
isFirstLoading.value = true;
}
});
}
}
return info.data;
};
return get_component_data
}
/* Callback to request REST data from components */
function set_component_attr_func(component) {
const set_component_attr = async (attr, value) => {
component[attr] = value;
}
return set_component_attr;
}
/* ********************************************* */
/* ************** Template Editor ************** */
function show_new_template_modal() {
modal_new_template.value.show();
}
function get_suggested_template_name() {
let name = "New Template";
return name;
}
const new_template = async (template_name) => {
let success = false;
let data = {
csrf: props.context.csrf,
template_name: template_name,
};
let url = `${props.context.template_add_endpoint}`;
try {
let headers = {
'Content-Type': 'application/json'
};
let res = await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(data) });
if (res && res.template_id) {
await set_template(res.template_id);
}
success = true;
} catch (err) {
console.error(err);
}
return success;
}
async function commit_template_change(e) {
let components_ids = components.value.map((c) => { return c.id; });
let data = {
csrf: props.context.csrf,
template: selected_report_template.value.value,
components: components_ids.toString()
};
let success = false;
let url = `${props.context.template_edit_endpoint}`;
try {
let headers = {
'Content-Type': 'application/json'
};
let res = await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(data) });
success = true;
} catch (err) {
console.error(err);
}
return success;
}
async function component_dragged(e) {
/* console.log(template_sortable.toArray().toString()); */
let new_components = template_sortable.toArray().map((id) => {
return components.value.find((c) => c.id == id);
});
unset_sortable_template();
components.value = new_components;
await nextTick();
set_sortable_template();
commit_template_change();
}
function set_sortable_template() {
if (template_sortable) {
return;
}
var drag_zone = document.getElementById('drag-zone');
template_sortable = window.Sortable.create(drag_zone, {
draggable: ".drag-item",
dataIdAttr: "id",
onUpdate: component_dragged
});
//console.log("Sortable ON");
}
function unset_sortable_template() {
if (template_sortable) {
template_sortable.destroy();
template_sortable = null;
//console.log("Sortable OFF");
}
}
function toggle_edit_mode() {
edit_mode.value = !edit_mode.value;
if (edit_mode.value) {
set_sortable_template();
} else {
unset_sortable_template();
}
}
const list_template_components = async () => {
let url = `${props.context.template_list_widgets_endpoint}`;
let res = await ntopng_utility.http_request(url);
if (res?.list == null) { return {}; }
let widgets = res.list;
widgets.sort((a, b) => _i18n(a.i18n_name).toUpperCase().localeCompare(_i18n(b.i18n_name).toUpperCase()));
return widgets;
}
function show_add_template_component_modal() {
modal_add_template_component.value.show();
}
function show_delete_template_modal() {
modal_delete_template.value.show();
}
const add_template_component = async (c) => {
unset_sortable_template();
let url = `${props.context.template_component_add_endpoint}`;
let params = {
csrf: props.context.csrf,
template: props.context.template,
component: c.id
};
let headers = {
'Content-Type': 'application/json'
};
try {
let content = await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(params) });
warning_message.value = "";
await set_template(selected_report_template.value.value);
} catch (err) {
warning_message.value = _i18n("report.unable_to_open");
}
await nextTick();
set_sortable_template();
}
async function remove_template_component(e) {
unset_sortable_template();
const component_id = e.target.dataset.componentId;
components.value = components.value.filter(c => c.id !== component_id);
await nextTick();
set_sortable_template();
commit_template_change();
}
/**
* @brief The method used to open the edit component modal
* involves triggering a click event on the 'Edit' (pencil icon) button.
* @param e event object
*/
function show_edit_template_component(e) {
const component_id = e.target.dataset.componentId;
const component = components.value.find(c => c.id === component_id);
const is_report_page = props.context.page == 'report';
modal_edit_template_component.value.show(component, is_report_page);
}
/**
*
* @brief The method used to update the selected component with new settings
* involves making a REST call.
* @param new_component An object with new settings for the component to edit.
*/
async function edit_template_component(new_component) {
unset_sortable_template();
let url = `${props.context.template_component_edit_endpoint}`;
let params = {
csrf: props.context.csrf,
template: props.context.template,
component: new_component.id,
component_title: new_component.title,
component_height: new_component.height,
component_width: new_component.width,
component_time_offset: new_component.time_offset,
component_time_window: new_component.time_window,
component_params: new_component.rest_params
};
let headers = {
'Content-Type': 'application/json'
};
try {
let content = await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(params) });
warning_message.value = "";
await set_template(selected_report_template.value.value);
} catch (err) {
warning_message.value = _i18n("report.unable_to_open");
}
await nextTick();
set_sortable_template();
}
async function delete_template() {
let success = false;
let data = {
csrf: props.context.csrf,
template: selected_report_template.value.value,
};
let url = `${props.context.template_delete_endpoint}`;
try {
let headers = {
'Content-Type': 'application/json'
};
let res = await ntopng_utility.http_request(url, { method: 'post', headers, body: JSON.stringify(data) });
await set_template();
success = true;
} catch (err) {
console.error(err);
}
return success;
}
/* ********************************************* */
</script>
<style scoped>
@media print {
.dontprint {
display: none;
}
.pagebreak-begin {
page-break-before: always;
}
.pagebreak-end {
page-break-after: always !important;
}
.print-element-class {
page-break-inside: avoid !important;
page-break-after: auto
}
}
/* @media print and (orientation: portrait) and (max-width: 297mm){ */
/* .col-4 { */
/* width: 50% !important; */
/* flex: 0 0 auto; */
/* } */
/* } */
@page {
/* size: A3 landscape; */
/* position:absolute; width:100%; top:0;left:0;right:0;bottom:0; padding:0; margin:-1px; */
}
/* Print on A4 */
@media print and (max-width: 297mm) and (min-width: 210mm) {
/* .row { */
/* padding-left: 0; */
/* padding-right: 0; */
/* margin-left: -10rem; */
/* margin-right: 0; */
/* } */
.col-4 {
width: 50% !important;
flex: 0 0 auto;
}
}
/* Print on A5 (commented out as this is not working on Chrome/Safari) */
/* @media print and (max-width: 148mm){ */
/* .col-4 { */
/* width: 100% !important; */
/* flex: 0 0 auto; */
/* } */
/* .col-6 { */
/* width: 100% !important; */
/* flex: 0 0 auto; */
/* } */
/* } */
.align-center {}
</style>