mirror of
https://github.com/ntop/ntopng.git
synced 2026-05-03 09:20:10 +00:00
318 lines
9.1 KiB
Vue
318 lines
9.1 KiB
Vue
<!-- (C) 2022 - ntop.org -->
|
|
<template>
|
|
<modal @showed="showed()" ref="modal_id">
|
|
<template v-slot:title>{{title}}</template>
|
|
<template v-slot:body>
|
|
|
|
<!-- Repeater Type -->
|
|
<div class="form-group ms-2 me-2 mt-3 row">
|
|
<label class="col-form-label col-sm-3">
|
|
<b>{{_i18n("nedge.page_repeater_config.modal_repeater_config.repeater_type")}}</b>
|
|
</label>
|
|
<div class="col-7">
|
|
<SelectSearch v-model:selected_option="selected_repeater_type"
|
|
@select_option="change_repeater_type()"
|
|
:options="repeater_type_array">
|
|
</SelectSearch>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- IP -->
|
|
<div v-if="selected_repeater_type.value == 'custom'" >
|
|
<div class="form-group ms-2 me-2 mt-3 row">
|
|
|
|
<label class="col-form-label col-sm-3" >
|
|
<b>{{_i18n("nedge.page_repeater_config.ip")}}</b>
|
|
</label>
|
|
<div class="col-7">
|
|
<input v-model="ip" @focusout="check_empty_host" class="form-control col-7" type="text" :placeholder="host_placeholder" required>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Interface -->
|
|
<div class="form-group ms-2 me-2 mt-3 row">
|
|
<label class="col-form-label col-sm-3" >
|
|
<b>{{_i18n("nedge.page_repeater_config.interfaces")}}</b>
|
|
</label>
|
|
<div class="col-7">
|
|
|
|
<SelectSearch ref="interfaces_search"
|
|
v-model:selected_options="selected_interfaces"
|
|
:options="interface_array"
|
|
:multiple="true"
|
|
@select_option="update_interfaces_selected"
|
|
@unselect_option="remove_interfaces_selected"
|
|
@change_selected_options="all_criteria">
|
|
</SelectSearch>
|
|
|
|
|
|
</div>
|
|
</div>
|
|
|
|
<div class="form-group ms-2 me-2 mt-3 row">
|
|
<label class="col-form-label col-sm-3" >
|
|
<b>{{_i18n("nedge.page_repeater_config.restricted_interfaces")}}</b>
|
|
</label>
|
|
<div class="col-7">
|
|
|
|
<SelectSearch ref="restricted_interfaces_search"
|
|
v-model:selected_options="selected_restricted_interfaces"
|
|
:options="interface_array"
|
|
:multiple="true"
|
|
@select_option="update_restricted_interfaces_selected"
|
|
@unselect_option="remove_restricted_interfaces_selected"
|
|
@change_selected_options="all_criteria">
|
|
</SelectSearch>
|
|
|
|
|
|
</div>
|
|
<small class="form-text text-muted">
|
|
{{_i18n("nedge.page_repeater_config.restricted_interfaces_note")}}
|
|
</small>
|
|
</div>
|
|
|
|
</template>
|
|
<template v-slot:footer>
|
|
<button type="button" :disabled="disable_add" @click="apply" class="btn btn-primary">{{button_text}}</button>
|
|
</template>
|
|
</modal>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, onMounted, computed, watch } from "vue";
|
|
import { ntopng_utility } from "../services/context/ntopng_globals_services.js";
|
|
import { default as modal } from "./modal.vue";
|
|
import { default as SelectSearch } from "./select-search.vue";
|
|
import regexValidation from "../utilities/regex-validation.js";
|
|
|
|
const _i18n = (t) => i18n(t);
|
|
const host_placeholder = i18n('if_stats_config.multicast_ip_placeholder')
|
|
const modal_id = ref(null);
|
|
const selected_interfaces = ref([]);
|
|
const selected_restricted_interfaces = ref([]);
|
|
const ip = ref(null);
|
|
const repeater_type = ref({value: "mdns", label: "MDNS" });
|
|
const emit = defineEmits(['edit', 'add'])
|
|
|
|
const interfaces_search = ref(null);
|
|
const restricted_interfaces_search = ref(null);
|
|
|
|
const showed = () => {};
|
|
|
|
const props = defineProps({});
|
|
|
|
const disable_add = ref(true);
|
|
const invalid_iface_number = ref(true);
|
|
const not_valid_ip = ref(true);
|
|
|
|
const check_empty_host = () => {
|
|
let regex = new RegExp(regexValidation.get_data_pattern('ip'));
|
|
not_valid_ip.value = !(regex.test(ip.value) || ip.value === '*');
|
|
disable_add.value = update_disable_add();
|
|
}
|
|
|
|
|
|
const update_disable_add = () => {
|
|
if (repeater_type.value.value == 'custom') {
|
|
return (invalid_iface_number.value || not_valid_ip.value);
|
|
} else {
|
|
return (invalid_iface_number.value);
|
|
}
|
|
}
|
|
|
|
const title = ref("");
|
|
|
|
const repeater_type_array = [
|
|
{ label: _i18n("nedge.page_repeater_config.modal_repeater_config.mdns"), value: "mdns", default: true },
|
|
{ label: _i18n("nedge.page_repeater_config.modal_repeater_config.custom"), value: "custom" },
|
|
];
|
|
|
|
const repeater_id = ref(0);
|
|
|
|
const selected_repeater_type = ref({});
|
|
|
|
const interface_list_url = `${http_prefix}/lua/pro/rest/v2/get/nedge/interfaces.lua`;
|
|
let interface_list;
|
|
const interface_array = ref([]);
|
|
|
|
const button_text = ref("");
|
|
|
|
const all_criteria = (item) => {
|
|
invalid_iface_number.value = ((selected_interfaces.value.length + selected_restricted_interfaces.value.length) < 2);
|
|
disable_add.value = update_disable_add();
|
|
}
|
|
|
|
const update_interfaces_selected = (items) => {
|
|
selected_interfaces.value = items;
|
|
}
|
|
|
|
const remove_interfaces_selected = (item_to_delete) => {
|
|
selected_interfaces.value = selected_interfaces.value.filter((item) => item.label != item_to_delete.label);
|
|
}
|
|
|
|
const update_restricted_interfaces_selected = (items) => {
|
|
selected_restricted_interfaces.value = items;
|
|
}
|
|
|
|
const remove_restricted_interfaces_selected = (item_to_delete) => {
|
|
selected_restricted_interfaces.value = selected_restricted_interfaces.value.filter((item) => item.label != item_to_delete.label);
|
|
}
|
|
|
|
const reset_modal = () => {
|
|
selected_repeater_type.value = {};
|
|
ip.value = "";
|
|
selected_interfaces.value = [];
|
|
selected_restricted_interfaces.value = [];
|
|
not_valid_ip.value = true;
|
|
invalid_iface_number.value = true;
|
|
}
|
|
|
|
const show = (row ) => {
|
|
reset_modal();
|
|
init(row);
|
|
modal_id.value.show();
|
|
};
|
|
|
|
const is_open_in_add = ref(true);
|
|
|
|
function init(row) {
|
|
is_open_in_add.value = row == null;
|
|
|
|
// check if we need open in edit
|
|
if (is_open_in_add.value == false) {
|
|
not_valid_ip.value = false;
|
|
invalid_iface_number.value = false;
|
|
disable_add.value = false;
|
|
title.value = _i18n("nedge.page_rules_config.modal_rule_config.title_edit");
|
|
button_text.value = _i18n("edit");
|
|
repeater_id.value = row.repeater_id;
|
|
selected_repeater_type.value = repeater_type_array.find((s) => (s.value == row.type));
|
|
if (selected_repeater_type.value.value == 'custom') {
|
|
ip.value = row.ip;
|
|
}
|
|
change_repeater_type(row)
|
|
|
|
} else {
|
|
title.value = _i18n("nedge.page_rules_config.modal_rule_config.title_add");
|
|
button_text.value = _i18n("add");
|
|
let default_type = repeater_type_array.find((s) => s.default == true);
|
|
}
|
|
|
|
if (is_open_in_add.value == false) {
|
|
const row_interfaces = row.interfaces.split(",");
|
|
let tmp_selected_interfaces = [];
|
|
row_interfaces.forEach((row_iface) => {
|
|
if (row_iface != '' && row_iface != null) {
|
|
tmp_selected_interfaces.push(interface_array.value.find((iface) => iface.value == row_iface));
|
|
}
|
|
})
|
|
selected_interfaces.value = tmp_selected_interfaces;
|
|
|
|
const row_restricted_interfaces = row.restricted_interfaces.split(",");
|
|
let tmp_selected_restricted_interfaces = [];
|
|
row_restricted_interfaces.forEach((row_iface) => {
|
|
if (row_iface != '' && row_iface != null) {
|
|
tmp_selected_restricted_interfaces.push(interface_array.value.find((iface) => iface.value == row_iface));
|
|
}
|
|
})
|
|
selected_restricted_interfaces.value = tmp_selected_restricted_interfaces;
|
|
}
|
|
}
|
|
|
|
async function change_repeater_type(type) {
|
|
repeater_type.value = selected_repeater_type.value;
|
|
if (repeater_type.value.value == "custom") {
|
|
await set_interface_array();
|
|
}
|
|
}
|
|
|
|
let is_set_interface_array = false;
|
|
async function set_interface_array() {
|
|
if (is_set_interface_array == true) { return; }
|
|
if (interface_list == null) {
|
|
interface_list = ntopng_utility.http_request(interface_list_url);
|
|
}
|
|
let res_interface_list = await interface_list;
|
|
interface_array.value = res_interface_list.filter(i => i.role == "lan").map((i) => {
|
|
return {
|
|
label: i.label,
|
|
value: i.ifname,
|
|
};
|
|
});
|
|
is_set_interface_array = true;
|
|
}
|
|
|
|
const apply = () => {
|
|
let repeater_t = repeater_type.value.value;
|
|
let obj = {
|
|
repeater_type: repeater_t,
|
|
};
|
|
if (repeater_type.value.value == "custom") {
|
|
let ip_t = ip.value;
|
|
obj = {
|
|
repeater_type: repeater_t,
|
|
ip: ip_t,
|
|
};
|
|
}
|
|
|
|
let event = "add";
|
|
if (is_open_in_add.value == false) {
|
|
obj.repeater_id = repeater_id.value;
|
|
event = "edit";
|
|
}
|
|
|
|
let interfaces = [];
|
|
let details = [];
|
|
selected_interfaces.value.forEach((i) => {
|
|
interfaces.push(i.value);
|
|
|
|
if(i.value != i.label && !i.label.includes(i.value)) {
|
|
details.push(i.label+" ("+i.value+")");
|
|
} else {
|
|
details.push(i.label);
|
|
}
|
|
});
|
|
const tmp_interfaces = interfaces.join(",");
|
|
const tmp_details = details.join(",");
|
|
obj.interfaces = tmp_interfaces;
|
|
obj.interface_details = tmp_details;
|
|
|
|
let restricted_interfaces = [];
|
|
let restricted_details = [];
|
|
selected_restricted_interfaces.value.forEach((i) => {
|
|
restricted_interfaces.push(i.value);
|
|
|
|
if(i.value != i.label && !i.label.includes(i.value)) {
|
|
restricted_details.push(i.label+" ("+i.value+")");
|
|
} else {
|
|
restricted_details.push(i.label);
|
|
}
|
|
});
|
|
const tmp_restricted_interfaces = restricted_interfaces.join(",");
|
|
const tmp_restricted_details = restricted_details.join(",");
|
|
obj.restricted_interfaces = tmp_restricted_interfaces;
|
|
obj.restricted_interface_details = tmp_restricted_details;
|
|
|
|
emit(event, obj);
|
|
close();
|
|
};
|
|
|
|
const close = () => {
|
|
modal_id.value.close();
|
|
};
|
|
|
|
defineExpose({ show, close });
|
|
|
|
onMounted(async () => {
|
|
await set_interface_array();
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
<style scoped>
|
|
input:invalid {
|
|
border-color: #ff0000;
|
|
}
|
|
</style>
|