1209 lines
60 KiB
HTML
1209 lines
60 KiB
HTML
<!DOCTYPE html>
|
||
<html lang="en">
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||
<title>RSCM-101</title>
|
||
<link rel="stylesheet" type="text/css" href="/style.css">
|
||
<link rel="stylesheet" type="text/css" href="/fields.css">
|
||
<style>
|
||
header {
|
||
position: fixed;
|
||
top: 0;
|
||
left: 0;
|
||
width: 100%;
|
||
z-index: 10;
|
||
background: var(--bg-selected);
|
||
}
|
||
|
||
body { /* значение по-умолчанию */ --header-height: 60px; }
|
||
|
||
#content {
|
||
padding-top: var(--header-height);
|
||
}
|
||
</style>
|
||
</head>
|
||
<body>
|
||
|
||
<div id="app" hidden>
|
||
<header>
|
||
<span class="nav-bar-element">Прием: <span :class="{ indicator_bad: stat_rx.state === false, indicator_good: stat_rx.state === true, indicator: true }"></span></span>
|
||
<span class="nav-bar-element">Передача: <span :class="{ indicator_good: stat_tx.state === true, indicator: true }"></span></span>
|
||
<span class="nav-bar-element">Тест: <span :class="{ indicator_good: (param.general.isTestInputData === true || param.general.modulatorMode === 'test'), indicator: true }"></span></span>
|
||
<!-- Последнее обновление: {{ lastUpdateTime }}-->
|
||
<div :class="{ value_bad: initState !== 'Успешная инициализация системы' }">{{ initState }}</div>
|
||
|
||
<div class="tabs-header">
|
||
<span style="font-weight:bold">RSCM-101</span>
|
||
<a href="#monitoring" class="tabs-btn" @click="activeTab = 'monitoring'" :class="{ active: activeTab === 'monitoring' }">Мониторинг</a>
|
||
<a href="#setup" class="tabs-btn" @click="activeTab = 'setup'" :class="{ active: activeTab === 'setup' }">Настройки</a>
|
||
<a href="#qos" class="tabs-btn" @click="activeTab = 'qos'" :class="{ active: activeTab === 'qos' }">QoS</a>
|
||
<a href="#admin" class="tabs-btn" @click="activeTab = 'admin'" :class="{ active: activeTab === 'admin' }">Администрирование</a>
|
||
<a href="/logout" class="tabs-btn">Выход</a>
|
||
</div>
|
||
</header>
|
||
<div id="content">
|
||
<div class="tabs-body-item tabs-item-flex-container" v-if="activeTab === 'monitoring'">
|
||
<div class="settings-set-container">
|
||
<h2>Статистика приема</h2>
|
||
<table>
|
||
<tbody>
|
||
<tr><th>Прием</th><td><span :class="{ indicator_bad: stat_rx.state === false, indicator_good: stat_rx.state === true, indicator: true }"></span></td></tr>
|
||
<tr><th>Захват символьной</th><td><span :class="{ indicator_bad: stat_rx.sym_sync_lock === false, indicator_good: stat_rx.sym_sync_lock === true, indicator: true }"></span></td></tr>
|
||
<tr><th>Захват ФАПЧ</th><td><span :class="{ indicator_bad: stat_rx.afc_lock === false, indicator_good: stat_rx.afc_lock === true, indicator: true }"></span></td></tr>
|
||
<tr><th>Захват поиска по частоте</th><td><span :class="{ indicator_bad: stat_rx.freq_search_lock === false, indicator_good: stat_rx.freq_search_lock === true, indicator: true }"></span></td></tr>
|
||
<tr><th>Захват пакетной синхр.</th><td><span :class="{ indicator_bad: stat_rx.pkt_sync === false, indicator_good: stat_rx.pkt_sync === true, indicator: true }"></span></td></tr>
|
||
<tr><th>ОСШ/RSSI</th><td>{{ stat_rx.snr }} / {{ stat_rx.rssi }}</td></tr>
|
||
<tr><th>Modcod</th><td>{{ stat_rx.modcod }}</td></tr>
|
||
<tr><th>Размер кадра</th><td>{{ stat_rx.frameSize }}</td></tr>
|
||
<tr><th>Пилот-символы</th><td>{{ stat_rx.pilots }}</td></tr>
|
||
<tr><th>Символьная ошибка</th><td>{{ stat_rx.symError }}</td></tr>
|
||
<tr><th>Грубая/точная част. ошибка, Гц</th><td>{{ stat_rx.freqErr }} / {{ stat_rx.freqErrAcc }}</td></tr>
|
||
<tr><th>Ур. входного сигнала</th><td>{{ stat_rx.inputSignalLevel }}</td></tr>
|
||
<tr><th>Ошибка ФАПЧ</th><td>{{ stat_rx.pllError }}</td></tr>
|
||
<tr><th>Инф. скорость на приеме</th><td>{{ stat_rx.speedOnRxKbit }} kbit/s</td></tr>
|
||
<tr><th>Инф. скорость на интерфейсе</th><td>{{ stat_rx.speedOnIifKbit }} kbit/s</td></tr>
|
||
</tbody>
|
||
</table>
|
||
<p> Статистика пакетов </p>
|
||
<table>
|
||
<tbody>
|
||
<tr><th>Качественных пакетов</th><td>{{ stat_rx.packetsOk }}</td></tr>
|
||
<tr><th>Поврежденных пакетов</th><td>{{ stat_rx.packetsBad }}</td></tr>
|
||
<tr><th>DUMMY</th><td>{{ stat_rx.packetsDummy }}</td></tr>
|
||
</tbody>
|
||
</table>
|
||
<button class="action-button" @click="resetPacketsStatistics()"> Сброс статистики </button>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h2>Статистика передачи</h2>
|
||
<table>
|
||
<tbody>
|
||
<tr><th>Передача</th><td><span :class="{ indicator_bad: stat_tx.state === false, indicator_good: stat_tx.state === true, indicator: true }"></span></td></tr>
|
||
<tr><th>ОСШ дальнего приема</th><td>{{ stat_tx.snr }}</td></tr>
|
||
<tr><th>Modcod</th><td>{{ stat_tx.modcod }}</td></tr>
|
||
<tr><th>Размер кадра</th><td>{{ stat_tx.frameSize }}</td></tr>
|
||
<tr><th>Пилот-символы</th><td>{{ stat_tx.pilots }}</td></tr>
|
||
<tr><th>Инф. скорость на передаче</th><td>{{ stat_tx.speedOnTxKbit }} kbit/s</td></tr>
|
||
<tr><th>Инф. скорость на интерфейсе</th><td>{{ stat_tx.speedOnIifKbit }} kbit/s</td></tr>
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
<div class="settings-set-container" v-if="isCinC === true">
|
||
<h2>Статистика режима CinC</h2>
|
||
<table>
|
||
<tbody>
|
||
<tr><th>ОСС</th><td>{{ stat_cinc.occ }}</td></tr>
|
||
<tr><th>Захват коррелятора</th><td><span :class="{ indicator_bad: stat_cinc.correlator === false, indicator_good: stat_cinc.correlator === true, indicator: true }"></span></td></tr>
|
||
<tr><th>Кол-во срывов коррелятора</th><td>{{ stat_cinc.correlatorFails }}</td></tr>
|
||
<tr><th>Грубая/точная част. ошибка, Гц</th><td>{{ stat_cinc.freqErr }} / {{ stat_cinc.freqErrAcc }}</td></tr>
|
||
<tr><th>Задержка в канале, мс</th><td>{{ stat_cinc.channelDelay }}</td></tr>
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h2>Состояние устройства</h2>
|
||
<table>
|
||
<tbody>
|
||
<tr><th>Температура ADRV</th><td>{{ stat_device.adrv }} °C</td></tr>
|
||
<tr><th>Температура ZYNQ ULTRASUCK</th><td>{{ stat_device.zynq }} °C</td></tr>
|
||
<tr><th>Температура FPGA</th><td>{{ stat_device.fpga }} °C</td></tr>
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
</div>
|
||
<div class="tabs-body-item" v-if="activeTab === 'setup' && settingFetchComplete">
|
||
<h2>Настройки приема/передачи</h2>
|
||
<div class="settings-set-container">
|
||
<label>
|
||
<span>Режим работы</span>
|
||
<select v-model="param.general.isCinC">
|
||
<option value="false">SCPC</option>
|
||
<option value="true">CinC</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
<div class="tabs-item-flex-container">
|
||
<div class="settings-set-container">
|
||
<h3>Настройки передатчика</h3>
|
||
<label>
|
||
<span>Включить передатчик</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.general.txEn" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Автоматический запуск передатчика</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.general.autoStartTx" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Режим работы модулятора</span>
|
||
<select v-model="param.general.modulatorMode">
|
||
<option value="normal">Нормальный</option>
|
||
<option value="test">Тест (CW)</option>
|
||
</select>
|
||
</label>
|
||
<label>
|
||
<span>Входные данные</span>
|
||
<select v-model="param.general.isTestInputData">
|
||
<option value="false">Ethernet</option>
|
||
<option value="true">Тест (CW)</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Параметры передачи</h3>
|
||
<label>
|
||
<span>Центральная частота, КГц</span>
|
||
<input v-model="param.tx.centerFreq" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Символьная скорость, Бод</span>
|
||
<input v-model="param.tx.cymRate" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Roll-off</span>
|
||
<select v-model="param.tx.rolloff">
|
||
<option value="5">0.05</option>
|
||
<option value="10">0.10</option>
|
||
<option value="15">0.15</option>
|
||
<option value="20">0.20</option>
|
||
<option value="25">0.25</option>
|
||
</select>
|
||
</label>
|
||
<label>
|
||
<span>Ослабление, дБ</span>
|
||
<input v-model="param.tx.attenuation" type="number"/>
|
||
</label>
|
||
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Режим работы DVB-S2</h3>
|
||
<label>
|
||
<span>Период служебных пакетов, сек</span>
|
||
<input v-model="param.dvbs2.servicePacketPeriod" type="number">
|
||
</label>
|
||
<label>
|
||
<span>Режим</span>
|
||
<select v-model="param.dvbs2.mode">
|
||
<option value="ccm">CCM</option>
|
||
<option value="acm">ACM</option>
|
||
</select>
|
||
</label>
|
||
<label>
|
||
<span>Размер кадра</span>
|
||
<select v-model="param.dvbs2.frameSize">
|
||
<option value="normal">normal</option>
|
||
<option value="short">short</option>
|
||
</select>
|
||
</label>
|
||
<!-- <label>-->
|
||
<!-- <span>Пилот-символы</span>-->
|
||
<!-- <select v-model="param.dvbs2.pilots">-->
|
||
<!-- <option value="true">pilots</option>-->
|
||
<!-- <option value="false">no pilots</option>-->
|
||
<!-- </select>-->
|
||
<!-- </label>-->
|
||
|
||
<label v-show="param.dvbs2.mode === 'ccm'">
|
||
<span>Модуляция</span>
|
||
<select v-model="param.dvbs2.ccm_modulation">
|
||
<option value="qpsk">QPSK</option>
|
||
<option value="8psk">8PSK</option>
|
||
<option value="16apsk">16APSK</option>
|
||
<option value="32apsk">32APSK</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.dvbs2.mode === 'ccm'">
|
||
<span>Скорость кода</span>
|
||
<select v-model="param.dvbs2.ccm_speed">
|
||
<option v-for="speed in getAvailableModcods(param.dvbs2.ccm_modulation)" v-bind:value="speed">{{ speed }}</option>
|
||
</select>
|
||
</label>
|
||
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Текущий модкод</span>
|
||
<input v-model="stat_rx.modcod" readonly>
|
||
</label>
|
||
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Модуляция (макс. режим)</span>
|
||
<select v-model="param.dvbs2.acm_maxModulation">
|
||
<option value="qpsk">QPSK</option>
|
||
<option value="8psk">8PSK</option>
|
||
<option value="16apsk">16APSK</option>
|
||
<option value="32apsk">32APSK</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Скорость кода (макс. режим)</span>
|
||
<select v-model="param.dvbs2.acm_maxSpeed">
|
||
<option v-for="speed in getAvailableModcods(param.dvbs2.acm_maxModulation)" v-bind:value="speed">{{ speed }}</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Модуляция (мин. режим)</span>
|
||
<select v-model="param.dvbs2.acm_minModulation">
|
||
<option value="qpsk">QPSK</option>
|
||
<option value="8psk">8PSK</option>
|
||
<option value="16apsk">16APSK</option>
|
||
<option value="32apsk">32APSK</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Скорость кода (мин. режим)</span>
|
||
<select v-model="param.dvbs2.acm_minSpeed">
|
||
<option v-for="speed in getAvailableModcods(param.dvbs2.acm_minModulation)" v-bind:value="speed">{{ speed }}</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.dvbs2.mode === 'acm'">
|
||
<span>Запас ОСШ</span>
|
||
<input v-model="param.dvbs2.snrReserve" type="number">
|
||
</label>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Авто-регулировка мощности</h3>
|
||
<label>
|
||
<span>Авто-регулировка мощности</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.acm.en" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Максимальное ослабление</span>
|
||
<input v-model="param.acm.maxAttenuation" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Минимальное ослабление</span>
|
||
<input v-model="param.acm.minAttenuation" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Требуемое ОСШ</span>
|
||
<input v-model="param.acm.requiredSnr" type="number"/>
|
||
</label>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Настройка приемника</h3>
|
||
<label>
|
||
<span>Режим управления усилением</span>
|
||
<select v-model="param.rx.gainMode">
|
||
<option value="auto">АРУ</option>
|
||
<option value="manual">РРУ</option>
|
||
</select>
|
||
</label>
|
||
<label v-show="param.rx.gainMode === 'manual'">
|
||
<span>Усиление, dB</span>
|
||
<input v-model="param.rx.manualGain" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Инверсия спектра</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.rx.spectrumInversion" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Центральная частота, кГц</span>
|
||
<input v-model="param.rx.centerFreq" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Символьная скорость, Бод</span>
|
||
<input v-model="param.rx.cymRate" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Roll-off</span>
|
||
<select v-model="param.rx.rolloff">
|
||
<option value="5">0.05</option>
|
||
<option value="10">0.10</option>
|
||
<option value="15">0.15</option>
|
||
<option value="20">0.20</option>
|
||
<option value="25">0.25</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
<button class="action-button" @click="settingsSubmitRxTx()">Сохранить <span class="submit-spinner" v-show="submitStatus.rxTx"></span></button>
|
||
|
||
<h2 v-show="param.general.isCinC">Настройки режима CinC</h2>
|
||
<div v-show="param.general.isCinC" class="settings-set-container">
|
||
<label>
|
||
<span>Метод расчета задержки</span>
|
||
<select v-model="param.cinc.mode">
|
||
<option value="positional">Позиционированием</option>
|
||
<option value="delay">Окном задержки</option>
|
||
</select>
|
||
</label>
|
||
|
||
<h3 v-show="param.cinc.mode === 'positional'">Настройки позиционирования</h3>
|
||
<label v-show="param.cinc.mode === 'positional'">
|
||
<span>Широта станции</span>
|
||
<input v-model="param.cinc.position.station.latitude" type="number"/>
|
||
</label>
|
||
<label v-show="param.cinc.mode === 'positional'">
|
||
<span>Долгота станции</span>
|
||
<input v-model="param.cinc.position.station.longitude" type="number"/>
|
||
</label>
|
||
<label v-show="param.cinc.mode === 'positional'">
|
||
<span>Подспутниковая точка</span>
|
||
<input v-model="param.cinc.position.satelliteLongitude" type="number"/>
|
||
</label>
|
||
|
||
<h3 v-show="param.cinc.mode === 'delay'">Задержка до спутника</h3>
|
||
<label v-show="param.cinc.mode === 'delay'">
|
||
<span>от, мс</span>
|
||
<input v-model="param.cinc.delayMin" type="number"/>
|
||
</label>
|
||
<label v-show="param.cinc.mode === 'delay'">
|
||
<span>до, мс</span>
|
||
<input v-model="param.cinc.delayMax" type="number"/>
|
||
</label>
|
||
</div>
|
||
<button class="action-button" v-show="param.general.isCinC" @click="settingsSubmitCinC()" type="submit">Сохранить <span class="submit-spinner" v-show="submitStatus.cinc"></span></button>
|
||
|
||
<h2>Настройки питания и опорного генератора</h2>
|
||
<div class="tabs-item-flex-container">
|
||
<div class="settings-set-container">
|
||
<h3>Настройки BUC</h3>
|
||
<label>
|
||
<span>Подача опоры 10МГц</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.buc.refClk10M" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Питание BUC</span>
|
||
<select v-model="param.buc.powering">
|
||
<option value="0">выкл</option>
|
||
<option value="24">24В</option>
|
||
<option value="48">48В</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Настройки LNB</h3>
|
||
<label>
|
||
<span>Подача опоры 10МГц</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.lnb.refClk10M" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Питание LNB</span>
|
||
<select v-model="param.lnb.powering">
|
||
<option value="0">выкл</option>
|
||
<option value="13">13В</option>
|
||
<option value="18">18В</option>
|
||
<option value="24">24В</option>
|
||
</select>
|
||
</label>
|
||
</div>
|
||
<div class="settings-set-container">
|
||
<h3>Сервисные настройки</h3>
|
||
<label>
|
||
<span>Подача опоры 10МГц на 'Выход 10МГц'</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.serviceSettings.refClk10M" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<label>
|
||
<span>Автозапуск BUC и LNB при включении</span>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="param.serviceSettings.autoStart" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
<button class="action-button" @click="settingsSubmitBucLnb()">Сохранить <span class="submit-spinner" v-show="submitStatus.bucLnb"></span></button>
|
||
</div>
|
||
<div class="tabs-body-item" v-if="activeTab === 'qos'">
|
||
<h2>Настройки QoS</h2>
|
||
<div class="settings-set-container">
|
||
<label>
|
||
<span>Активировать QoS</span>
|
||
<span class="toggle-input"><input type="checkbox" v-model="qos.en" /><span class="slider"></span></span>
|
||
</label>
|
||
</div>
|
||
<template>
|
||
<div v-for="classesGroup in ['rt1', 'rt2', 'rt3', 'cd']">
|
||
<h3>Классы {{ classesGroup.toUpperCase() }}</h3>
|
||
<button class="action-button" @click="qosAddClass(classesGroup)">Добавить класс {{ classesGroup.toUpperCase() }}</button>
|
||
<details v-for="(qosClass, index) in qos[classesGroup]" :key="index" class="settings-set-container">
|
||
<summary>
|
||
<span v-if="classesGroup === 'cd'">#{{ index }} CIR={{ qosClass.cir }}кбит, PIR={{ qosClass.pir }}кбит {{ qosClass.description }}</span>
|
||
<span v-if="classesGroup !== 'cd'">#{{ index }} CBR={{ qosClass.cir }}кбит {{ qosClass.description }}</span>
|
||
<span class="summary-actions">
|
||
<label>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="qosClass.isEnabled" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
</span>
|
||
</summary>
|
||
<label>
|
||
<span v-if="classesGroup === 'cd'">CIR</span> <span v-if="classesGroup !== 'cd'">CBR</span>
|
||
<input v-model="qosClass.cir" type="number"/>
|
||
</label>
|
||
<label v-if="classesGroup === 'cd'">
|
||
<span>PIR</span>
|
||
<input v-model="qosClass.pir" type="number"/>
|
||
</label>
|
||
<label>
|
||
<span>Описание</span>
|
||
<input v-model="qosClass.description"/>
|
||
</label>
|
||
|
||
<h3>Фильтры ({{ qosClass.filters.length }})</h3>
|
||
<div>
|
||
<button class="action-button" @click="qosClassAddRule(classesGroup, index)">Добавить правило</button>
|
||
</div>
|
||
<details v-for="(filter, filterIndex) in qosClass.filters" :key="filterIndex" class="settings-set-container">
|
||
<summary>
|
||
<span>#{{ filterIndex }} {{ qosGenerateRuleDescription(filter) }}</span>
|
||
<span class="summary-actions">
|
||
<label>
|
||
<span class="toggle-input">
|
||
<input type="checkbox" v-model="filter.isEnabled" />
|
||
<span class="slider"></span>
|
||
</span>
|
||
</label>
|
||
<button class="dangerous-button" @click="qosDelFilter(classesGroup, index, filterIndex)">Del</button>
|
||
</span>
|
||
</summary>
|
||
<label>
|
||
<span>VLAN ID</span>
|
||
<input v-model="filter.vlan" type="text">
|
||
</label>
|
||
<label>
|
||
<span>Протокол L3</span>
|
||
<input v-model="filter.proto" type="text">
|
||
</label>
|
||
<label>
|
||
<span>Порт источника</span>
|
||
<input v-model="filter.sport" type="text">
|
||
</label>
|
||
<label>
|
||
<span>Порт назначения</span>
|
||
<input v-model="filter.dport" type="text">
|
||
</label>
|
||
<label>
|
||
<span>IP источника</span>
|
||
<input v-model="filter.ip_src" type="text">
|
||
</label>
|
||
<label>
|
||
<span>IP назначения</span>
|
||
<input v-model="filter.ip_dest" type="text">
|
||
</label>
|
||
<label>
|
||
<span>Метка IP.DSCP</span>
|
||
<input v-model="filter.dscp" type="text">
|
||
</label>
|
||
</details>
|
||
|
||
<div>
|
||
<button class="dangerous-button" @click="qosDelClass(classesGroup, index)">Удалить класс QoS</button>
|
||
</div>
|
||
</details>
|
||
</div>
|
||
</template>
|
||
|
||
<button class="action-button" @click="settingsSubmitQoS()">Применить <span class="submit-spinner" v-show="submitStatus.qos"></span></button>
|
||
</div>
|
||
<div class="tabs-body-item" v-if="activeTab === 'admin'">
|
||
<p>
|
||
Эти настройки пока недоступны, но скоро разработчик это поправит. А пока смотри на крокодила, или купи разработчику банку <strike>пива</strike> колы для ускорения процесса)
|
||
</p>
|
||
<img loading="lazy" src="/images/krokodil_vzryvaetsya_hd.gif" alt="krokodil">
|
||
<video preload="auto" controls style="max-width: 100%">
|
||
<source src="/vid/video_2024-11-06_15-49-35.mp4" type="video/mp4" />
|
||
</video>
|
||
</div>
|
||
<p>Последнее обновление статистики: {{ lastUpdateTime }}</p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Версия для разработки включает в себя возможность вывода в консоль полезных уведомлений -->
|
||
<script src="/js/vue.js"></script>
|
||
<script>
|
||
window.addEventListener('load', updateHeaderHeight);
|
||
window.addEventListener('resize', updateHeaderHeight);
|
||
|
||
function updateHeaderHeight() {
|
||
const header = document.querySelector('header');
|
||
document.body.style.setProperty('--header-height', `${header.offsetHeight}px`);
|
||
}
|
||
|
||
// const router = useRouter();
|
||
const availableTabs = ['monitoring', 'setup', 'qos', 'admin']
|
||
const defaultTab = availableTabs[0]
|
||
|
||
function getCurrentTab() {
|
||
const sl = window.location.hash.slice(1)
|
||
if (availableTabs.indexOf(sl) >= 0) {
|
||
return sl
|
||
}
|
||
return defaultTab
|
||
}
|
||
|
||
function modcodToStr(modcod) {
|
||
// модкоды из раздела 5.5.2.2 https://www.etsi.org/deliver/etsi_en/302300_302399/302307/01.01.02_60/en_302307v010102p.pdf
|
||
|
||
// NOTE модкоды со скоростью хода 3/5 не работают
|
||
const modcods = [
|
||
"DUMMY",
|
||
"QPSK 1/4",
|
||
"QPSK 1/3",
|
||
"QPSK 2/5",
|
||
"QPSK 1/2",
|
||
"QPSK 3/5", // отключено
|
||
"QPSK 2/3",
|
||
"QPSK 3/4",
|
||
"QPSK 4/5",
|
||
"QPSK 5/6",
|
||
"QPSK 8/9",
|
||
"QPSK 9/10",
|
||
|
||
"8PSK 3/5", // отключено
|
||
"8PSK 2/3",
|
||
"8PSK 3/4",
|
||
"8PSK 5/6",
|
||
"8PSK 8/9",
|
||
"8PSK 9/10",
|
||
|
||
"16APSK 2/3",
|
||
"16APSK 3/4",
|
||
"16APSK 4/5",
|
||
"16APSK 5/6",
|
||
"16APSK 8/9",
|
||
"16APSK 9/10",
|
||
|
||
"32APSK 3/4",
|
||
"32APSK 4/5",
|
||
"32APSK 5/6",
|
||
"32APSK 8/9",
|
||
"32APSK 9/10",
|
||
]
|
||
if (typeof modcod != "number" || modcod < 0 || modcod >= modcod.length) {
|
||
return "?";
|
||
}
|
||
return modcods[modcod]
|
||
}
|
||
|
||
function extractModulationAndSpeedFromModcod(modcod) {
|
||
switch (modcod) {
|
||
case 1: return { modulation: 'qpsk', speed: '1/4' }
|
||
case 2: return { modulation: 'qpsk', speed: '1/3' }
|
||
case 3: return { modulation: 'qpsk', speed: '2/5' }
|
||
case 4: return { modulation: 'qpsk', speed: '1/2' }
|
||
case 5: return { modulation: 'qpsk', speed: '3/5' }
|
||
case 6: return { modulation: 'qpsk', speed: '2/3' }
|
||
case 7: return { modulation: 'qpsk', speed: '3/4' }
|
||
case 8: return { modulation: 'qpsk', speed: '4/5' }
|
||
case 9: return { modulation: 'qpsk', speed: '5/6' }
|
||
case 10: return { modulation: 'qpsk', speed: '8/9' }
|
||
case 11: return { modulation: 'qpsk', speed: '9/10' }
|
||
case 12: return { modulation: '8psk', speed: '3/5' }
|
||
case 13: return { modulation: '8psk', speed: '2/3' }
|
||
case 14: return { modulation: '8psk', speed: '3/4' }
|
||
case 15: return { modulation: '8psk', speed: '5/6' }
|
||
case 16: return { modulation: '8psk', speed: '8/9' }
|
||
case 17: return { modulation: '8psk', speed: '9/10' }
|
||
case 18: return { modulation: '16apsk', speed: '2/3' }
|
||
case 19: return { modulation: '16apsk', speed: '3/4' }
|
||
case 20: return { modulation: '16apsk', speed: '4/5' }
|
||
case 21: return { modulation: '16apsk', speed: '5/6' }
|
||
case 22: return { modulation: '16apsk', speed: '8/9' }
|
||
case 23: return { modulation: '16apsk', speed: '9/10' }
|
||
case 24: return { modulation: '32apsk', speed: '3/4' }
|
||
case 25: return { modulation: '32apsk', speed: '4/5' }
|
||
case 26: return { modulation: '32apsk', speed: '5/6' }
|
||
case 27: return { modulation: '32apsk', speed: '8/9' }
|
||
case 28: return { modulation: '32apsk', speed: '9/10' }
|
||
}
|
||
return { modulation: 'qpsk', speed: '1/4' }
|
||
}
|
||
|
||
const app = new Vue({
|
||
el: '#app',
|
||
data: {
|
||
isCinC: null,
|
||
|
||
// false - означает что статистика не отправляется, true - отправляется
|
||
submitStatus: {
|
||
rxTx: false,
|
||
cinc: false,
|
||
bucLnb: false,
|
||
qos: false,
|
||
},
|
||
|
||
stat_rx: {
|
||
// индикаторы
|
||
state: '?', // общее состояние
|
||
sym_sync_lock: '?', // захват символьной
|
||
freq_search_lock: '?', // Захват поиска по частоте
|
||
afc_lock: '?', // захват ФАПЧ
|
||
pkt_sync: '?', // захват пакетной синхронизации
|
||
|
||
// куча других параметров, идет в том же порядке, что и в таблице
|
||
snr: '?', rssi: '?',
|
||
modcod: '?', frameSize: '?',
|
||
pilots: '?',
|
||
symError: '?',
|
||
freqErr: '?', freqErrAcc: '?',
|
||
inputSignalLevel: '?',
|
||
pllError: '?',
|
||
speedOnRxKbit: '?',
|
||
speedOnIifKbit: '?',
|
||
|
||
// статистика пакетов
|
||
packetsOk: '?', packetsBad: '?', packetsDummy: '?',
|
||
},
|
||
stat_tx: {
|
||
// состояние
|
||
state: '?',
|
||
|
||
// прочие поля
|
||
snr: '?', modcod: '?', frameSize: '?', pilots: '?', speedOnTxKbit: '?', speedOnIifKbit: '?',
|
||
},
|
||
stat_cinc: {
|
||
occ: '?',
|
||
correlator: null,
|
||
correlatorFails: '?',
|
||
freqErr: '?', freqErrAcc: '?',
|
||
channelDelay: '?'
|
||
},
|
||
stat_device: { // температурные датчики
|
||
adrv: 0, zynq: 0, fpga: 0
|
||
},
|
||
|
||
param: {
|
||
general: {
|
||
isCinC: false,
|
||
txEn: false, // включен/выключен
|
||
modulatorMode: 'normal', // режим работы модулятора
|
||
autoStartTx: false, // было "режим работы передатчика"
|
||
isTestInputData: false, // входные данные: eth или test
|
||
},
|
||
tx: {
|
||
attenuation: null, // ослабление
|
||
rolloff: null,
|
||
cymRate: null,
|
||
centerFreq: null,
|
||
},
|
||
dvbs2: {
|
||
mode: null, // ccm/acm
|
||
frameSize: null, // 'normal' / 'short'
|
||
// pilots: false,
|
||
|
||
// CCM
|
||
ccm_modulation: null,
|
||
ccm_speed: null,
|
||
|
||
// ACM
|
||
acm_maxModulation: null,
|
||
acm_maxSpeed: null,
|
||
acm_minModulation: null,
|
||
acm_minSpeed: null,
|
||
|
||
snrReserve: null,
|
||
servicePacketPeriod: null,
|
||
},
|
||
// авто-регулировка мощности
|
||
acm: {
|
||
en: false,
|
||
maxAttenuation: null,
|
||
minAttenuation: null,
|
||
requiredSnr: null,
|
||
},
|
||
rx: {
|
||
gainMode: null, // 'auto'/'manual' режим управления усилением
|
||
manualGain: 0, // усиление, только для ручного режима
|
||
spectrumInversion: false,
|
||
rolloff: 0,
|
||
cymRate: 100000,
|
||
centerFreq: 1200000.0,
|
||
},
|
||
|
||
cinc: {
|
||
mode: null, // 'positional' | 'delay'
|
||
searchBandwidth: 0, // полоса поиска в кГц
|
||
position: {
|
||
station: {
|
||
latitude: 0,
|
||
longitude: 0
|
||
},
|
||
satelliteLongitude: 0,
|
||
},
|
||
delayMin: 0,
|
||
delayMax: 0
|
||
},
|
||
buc: {
|
||
refClk10M: false, // подача опоры 10MHz
|
||
powering: 0 // 0, 24, 48
|
||
},
|
||
lnb: {
|
||
refClk10M: false, // подача опоры 10MHz
|
||
powering: 0 // 0, 13, 18, 24
|
||
},
|
||
serviceSettings: {
|
||
refClk10M: false, // подача опоры 10MHz
|
||
autoStart: false
|
||
},
|
||
},
|
||
|
||
qos: {
|
||
en: false,
|
||
rt1: [],
|
||
rt2: [],
|
||
rt3: [],
|
||
cd: [],
|
||
},
|
||
|
||
testState: false,
|
||
initState: '',
|
||
lastUpdateTime: new Date(),
|
||
activeTab: getCurrentTab(),
|
||
settingFetchComplete: false
|
||
},
|
||
methods: {
|
||
getAvailableModcods(modulation) {
|
||
// NOTE модкоды со скоростью хода 3/5 не работают
|
||
switch (modulation) {
|
||
case 'qpsk':
|
||
return ['1/4', '1/3', '2/5', '1/2', /* '3/5',*/ '2/3', '3/4', '4/5', '5/6', '8/9', '9/10']
|
||
case '8psk':
|
||
return [/* '3/5',*/ '2/3', '3/4', '5/6', '8/9', '9/10']
|
||
case '16apsk':
|
||
return ['2/3', '3/4', '4/5', '5/6', '8/9', '9/10']
|
||
case '32apsk':
|
||
return ['3/4', '4/5', '5/6', '8/9', '9/10']
|
||
default:
|
||
return []
|
||
}
|
||
},
|
||
|
||
updateStatistics(vals) {
|
||
this.lastUpdateTime = new Date();
|
||
this.isCinC = vals["mainState"]["isCinC"]
|
||
|
||
this.stat_rx.state = vals["mainState"]["rx.state"]
|
||
this.stat_rx.sym_sync_lock = vals["mainState"]["rx.sym_sync_lock"]
|
||
this.stat_rx.freq_search_lock = vals["mainState"]["rx.freq_search_lock"]
|
||
this.stat_rx.afc_lock = vals["mainState"]["rx.afc_lock"]
|
||
this.stat_rx.pkt_sync = vals["mainState"]["rx.pkt_sync"]
|
||
this.stat_rx.snr = vals["mainState"]["rx.snr"]
|
||
this.stat_rx.rssi = vals["mainState"]["rx.rssi"]
|
||
this.stat_rx.modcod = modcodToStr(vals["mainState"]["rx.modcod"])
|
||
this.stat_rx.frameSize = vals["mainState"]["rx.frameSize"]
|
||
this.stat_rx.pilots = vals["mainState"]["rx.pilots"]
|
||
this.stat_rx.symError = vals["mainState"]["rx.symError"]
|
||
this.stat_rx.freqErr = vals["mainState"]["rx.freqErr"]
|
||
this.stat_rx.freqErrAcc = vals["mainState"]["rx.freqErrAcc"]
|
||
this.stat_rx.inputSignalLevel = vals["mainState"]["rx.inputSignalLevel"]
|
||
this.stat_rx.pllError = vals["mainState"]["rx.pllError"]
|
||
this.stat_rx.speedOnRxKbit = vals["mainState"]["rx.speedOnRxKbit"]
|
||
this.stat_rx.speedOnIifKbit = vals["mainState"]["rx.speedOnIifKbit"]
|
||
this.stat_rx.packetsOk = vals["mainState"]["rx.packetsOk"]
|
||
this.stat_rx.packetsBad = vals["mainState"]["rx.packetsBad"]
|
||
this.stat_rx.packetsDummy = vals["mainState"]["rx.packetsDummy"]
|
||
|
||
this.stat_tx.state = vals["mainState"]["tx.state"]
|
||
this.stat_tx.snr = vals["mainState"]["tx.snr"]
|
||
this.stat_tx.modcod = modcodToStr(vals["mainState"]["tx.modcod"])
|
||
this.stat_tx.frameSize = vals["mainState"]["tx.frameSize"]
|
||
this.stat_tx.pilots = vals["mainState"]["tx.pilots"]
|
||
this.stat_tx.speedOnTxKbit = vals["mainState"]["tx.speedOnTxKbit"]
|
||
this.stat_tx.speedOnIifKbit = vals["mainState"]["tx.speedOnIifKbit"]
|
||
|
||
this.stat_cinc.occ = vals["mainState"]["cinc.occ"]
|
||
this.stat_cinc.correlator = vals["mainState"]["cinc.correlator"]
|
||
this.stat_cinc.correlatorFails = vals["mainState"]["cinc.correlatorFails"]
|
||
this.stat_cinc.freqErr = vals["mainState"]["cinc.freqErr"]
|
||
this.stat_cinc.freqErrAcc = vals["mainState"]["cinc.freqErrAcc"]
|
||
this.stat_cinc.channelDelay = vals["mainState"]["cinc.channelDelay"]
|
||
|
||
this.stat_device.adrv = vals["mainState"]["device.adrv"]
|
||
this.stat_device.zynq = vals["mainState"]["device.zynq"]
|
||
this.stat_device.fpga = vals["mainState"]["device.fpga"]
|
||
|
||
this.testState = vals["mainState"]["testState"]
|
||
},
|
||
|
||
resetPacketsStatistics() {
|
||
fetch('/api/resetPacketStatistics', {
|
||
method: 'POST'
|
||
}).then(() => {
|
||
this.stat_rx.packetsOk = 0
|
||
this.stat_rx.packetsBad = 0
|
||
this.stat_rx.packetsDummy = 0
|
||
})
|
||
},
|
||
|
||
settingsSubmitRxTx() {
|
||
if (this.submitStatus.rxTx) { return }
|
||
this.submitStatus.rxTx = true
|
||
},
|
||
|
||
settingsSubmitCinC() {
|
||
if (this.submitStatus.cinc) { return }
|
||
this.submitStatus.cinc = true
|
||
},
|
||
|
||
settingsSubmitBucLnb() {
|
||
this.submitStatus.bucLnb = true
|
||
// TODO сделать всплывающее окно с подтверждением того, что настройки действительно нужно применить
|
||
let query = {
|
||
"buc.refClk10M": this.param.buc.refClk10M,
|
||
"buc.powering": parseInt(this.param.buc.powering),
|
||
"lnb.refClk10M": this.param.lnb.refClk10M,
|
||
"lnb.powering": parseInt(this.param.lnb.powering),
|
||
"serviceSettings.refClk10M": this.param.serviceSettings.refClk10M,
|
||
"serviceSettings.autoStart": this.param.serviceSettings.autoStart
|
||
}
|
||
fetch('/api/set/bucLnb', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json'
|
||
},
|
||
body: JSON.stringify(query)
|
||
}).then(async (resp) => {
|
||
this.submitStatus.bucLnb = false
|
||
this.updateBucLnbSettings(await resp.json())
|
||
})
|
||
},
|
||
|
||
settingsSubmitQoS() {
|
||
if (this.submitStatus.qos) { return }
|
||
this.submitStatus.qos = true
|
||
function _translateQosClass(trafficClass, qc) {
|
||
let res = {
|
||
cir: qc['cir'],
|
||
description: qc['description'],
|
||
filters: []
|
||
}
|
||
if (trafficClass === 'cd') {
|
||
res.pir = qc.pir
|
||
}
|
||
if (!qc.isEnabled) {
|
||
res.disabled = true
|
||
}
|
||
for (const fi in qc.filters) {
|
||
let filter = {}
|
||
if (qc['filters'][fi].vlan !== "") { filter['vlan'] = qc['filters'][fi].vlan }
|
||
if (qc['filters'][fi].proto !== "") { filter['proto'] = qc['filters'][fi].proto }
|
||
if (qc['filters'][fi].sport !== "") { filter['sport'] = qc['filters'][fi].sport }
|
||
if (qc['filters'][fi].dport !== "") { filter['dport'] = qc['filters'][fi].dport }
|
||
if (qc['filters'][fi].ip_src !== "") { filter['ip.src'] = qc['filters'][fi].ip_src }
|
||
if (qc['filters'][fi].ip_dest !== "") { filter['ip.dest'] = qc['filters'][fi].ip_dest }
|
||
if (qc['filters'][fi].dscp !== "") { filter['dscp'] = qc['filters'][fi].dscp }
|
||
|
||
if (Object.keys(filter).length === 0) { continue }
|
||
if (!qc.filters[fi].isEnabled) { filter['disabled'] = true }
|
||
|
||
res.filters.push(filter)
|
||
}
|
||
|
||
return res
|
||
}
|
||
let query = {
|
||
"en": this.qos.en,
|
||
"rt1": [],
|
||
"rt2": [],
|
||
"rt3": [],
|
||
"cd": []
|
||
}
|
||
for (let i = 0; i < this.qos.rt1.length; i++) { query.rt1.push(_translateQosClass('rt', this.qos.rt1[i])) }
|
||
for (let i = 0; i < this.qos.rt2.length; i++) { query.rt2.push(_translateQosClass('rt', this.qos.rt2[i])) }
|
||
for (let i = 0; i < this.qos.rt3.length; i++) { query.rt3.push(_translateQosClass('rt', this.qos.rt3[i])) }
|
||
for (let i = 0; i < this.qos.cd.length; i++) { query.cd.push(_translateQosClass('rt', this.qos.cd[i])) }
|
||
|
||
console.log(query)
|
||
fetch('/api/set/qos', {
|
||
method: 'POST',
|
||
headers: {
|
||
'Content-Type': 'application/json'
|
||
},
|
||
body: JSON.stringify(query)
|
||
}).then(async (resp) => {
|
||
this.submitStatus.qos = false
|
||
this.updateQosSettings(await resp.json())
|
||
})
|
||
},
|
||
|
||
performUpdateSettings(reloadParts) {
|
||
const doFetchSettings = async () => {
|
||
let d = await fetch("/api/get/settings")
|
||
let vals = await d.json()
|
||
if (reloadParts !== undefined) {
|
||
if (reloadParts.indexOf('rxtx')) { this.updateRxTxSettings(vals) }
|
||
if (reloadParts.indexOf('cinc')) { this.updateCincSettings(vals) }
|
||
if (reloadParts.indexOf('buclnb')) { this.updateBucLnbSettings(vals) }
|
||
if (reloadParts.indexOf('qos')) { this.updateQosSettings(vals) }
|
||
} else {
|
||
this.updateSettings(vals)
|
||
}
|
||
}
|
||
|
||
doFetchSettings().then(() => {})
|
||
},
|
||
|
||
updateRxTxSettings(vals) {
|
||
this.submitStatus.rxTx = false
|
||
this.param.general.isCinC = vals["settings"]["general.isCinC"]
|
||
this.param.general.txEn = vals["settings"]["general.txEn"]
|
||
this.param.general.modulatorMode = vals["settings"]["general.modulatorMode"]
|
||
this.param.general.autoStartTx = vals["settings"]["general.autoStartTx"]
|
||
this.param.general.isTestInputData = vals["settings"]["general.isTestInputData"]
|
||
|
||
this.param.tx.attenuation = vals["settings"]["tx.attenuation"]
|
||
this.param.tx.rolloff = vals["settings"]["tx.rolloff"]
|
||
this.param.tx.cymRate = vals["settings"]["tx.cymRate"]
|
||
this.param.tx.centerFreq = vals["settings"]["tx.centerFreq"]
|
||
|
||
this.param.dvbs2.mode = (vals["settings"]["dvbs2.isAcm"] ? 'acm' : 'ccm')
|
||
this.param.dvbs2.frameSize = vals["settings"]["dvbs2.frameSize"]
|
||
// this.param.dvbs2.pilots = vals["settings"]["dvbs2.pilots"]
|
||
|
||
let m = extractModulationAndSpeedFromModcod(vals["settings"]["dvbs2.ccm_modcod"])
|
||
this.param.dvbs2.ccm_modulation = m.modulation
|
||
this.param.dvbs2.ccm_speed = m.speed
|
||
m = extractModulationAndSpeedFromModcod(vals["settings"]["dvbs2.acm_maxModcod"])
|
||
this.param.dvbs2.acm_maxModulation = m.modulation
|
||
this.param.dvbs2.acm_maxSpeed = m.speed
|
||
m = extractModulationAndSpeedFromModcod(vals["settings"]["dvbs2.acm_minModcod"])
|
||
this.param.dvbs2.acm_minModulation = m.modulation
|
||
this.param.dvbs2.acm_minSpeed = m.speed
|
||
|
||
this.param.dvbs2.snrReserve = vals["settings"]["dvbs2.snrReserve"]
|
||
this.param.dvbs2.servicePacketPeriod = vals["settings"]["dvbs2.servicePacketPeriod"]
|
||
|
||
this.param.acm.en = vals["settings"]["acm.en"]
|
||
this.param.acm.maxAttenuation = vals["settings"]["acm.maxAttenuation"]
|
||
this.param.acm.minAttenuation = vals["settings"]["acm.minAttenuation"]
|
||
this.param.acm.requiredSnr = vals["settings"]["acm.requiredSnr"]
|
||
|
||
this.param.rx.gainMode = vals["settings"]["rx.gainMode"]
|
||
this.param.rx.manualGain = vals["settings"]["rx.manualGain"]
|
||
this.param.rx.spectrumInversion = vals["settings"]["rx.spectrumInversion"]
|
||
this.param.rx.rolloff = vals["settings"]["rx.rolloff"]
|
||
this.param.rx.cymRate = vals["settings"]["rx.cymRate"]
|
||
this.param.rx.centerFreq = vals["settings"]["rx.centerFreq"]
|
||
},
|
||
|
||
updateCincSettings(vals) {
|
||
this.submitStatus.cinc = false
|
||
this.param.cinc.mode = vals["settings"]["cinc.mode"]
|
||
this.param.cinc.searchBandwidth = vals["settings"]["cinc.searchBandwidth"]
|
||
this.param.cinc.position.station.latitude = vals["settings"]["cinc.position.station.latitude"]
|
||
this.param.cinc.position.station.longitude = vals["settings"]["cinc.position.station.longitude"]
|
||
this.param.cinc.position.satelliteLongitude = vals["settings"]["cinc.position.satelliteLongitude"]
|
||
this.param.cinc.delayMin = vals["settings"]["cinc.delayMin"]
|
||
this.param.cinc.delayMax = vals["settings"]["cinc.delayMax"]
|
||
},
|
||
|
||
updateBucLnbSettings(vals) {
|
||
this.submitStatus.bucLnb = false
|
||
this.param.buc.refClk10M = vals["settings"]["buc.refClk10M"]
|
||
this.param.buc.powering = vals["settings"]["buc.powering"]
|
||
this.param.lnb.refClk10M = vals["settings"]["lnb.refClk10M"]
|
||
this.param.lnb.powering = vals["settings"]["lnb.powering"]
|
||
this.param.serviceSettings.refClk10M = vals["settings"]["serviceSettings.refClk10M"]
|
||
this.param.serviceSettings.autoStart = vals["settings"]["serviceSettings.autoStart"]
|
||
},
|
||
|
||
updateQosSettings(vals) {
|
||
this.submitStatus.qos = false
|
||
this.qos.en = vals["settings"]["qos.enabled"]
|
||
|
||
const qosProfile = vals["settings"]["qos.profile"]
|
||
if (qosProfile !== null && qosProfile !== undefined) {
|
||
this.qos.rt1 = [] // .splice(0, this.qos.rt1.length)
|
||
this.qos.rt2 = [] // .splice(0, this.qos.rt2.length)
|
||
this.qos.rt3 = [] // .splice(0, this.qos.rt3.length)
|
||
this.qos.cd = [] // .splice(0, this.qos.cd.length)
|
||
|
||
for (let trafficClass in qosProfile) {
|
||
if (['rt1', 'rt2', 'rt3', 'cd'].indexOf(trafficClass) < 0) {
|
||
continue
|
||
}
|
||
|
||
if (Array.isArray(qosProfile[trafficClass])) {
|
||
for (let i = 0; i < qosProfile[trafficClass].length; i++) {
|
||
const qc = qosProfile[trafficClass][i]
|
||
let result = {
|
||
isEnabled: !qc.hasOwnProperty('disabled'),
|
||
cir: qc['cir'],
|
||
pir: 0,
|
||
description: qc['description'],
|
||
filters: []
|
||
}
|
||
if (trafficClass === 'cd') {
|
||
if (qc['pir']) {
|
||
result.pir = qc['pir']
|
||
}
|
||
}
|
||
for (let fi = 0; fi < qc['filters'].length; fi++) {
|
||
result.filters.push({
|
||
isEnabled: !qc['filters'][fi].hasOwnProperty('disabled'),
|
||
vlan: qc['filters'][fi].hasOwnProperty('vlan') ? qc['filters'][fi]['vlan'] : '',
|
||
proto: qc['filters'][fi].hasOwnProperty('proto') ? qc['filters'][fi]['proto'] : '',
|
||
sport: qc['filters'][fi].hasOwnProperty('sport') ? qc['filters'][fi]['sport'] : '',
|
||
dport: qc['filters'][fi].hasOwnProperty('dport') ? qc['filters'][fi]['dport'] : '',
|
||
ip_src: qc['filters'][fi].hasOwnProperty('ip.src') ? qc['filters'][fi]['ip.src'] : '',
|
||
ip_dest: qc['filters'][fi].hasOwnProperty('ip.dest') ? qc['filters'][fi]['ip.dest'] : '',
|
||
dscp: qc['filters'][fi].hasOwnProperty('dscp') ? qc['filters'][fi]['dscp'] : ''
|
||
})
|
||
}
|
||
switch (trafficClass) {
|
||
case 'rt1': this.qos.rt1.push(result); break
|
||
case 'rt2': this.qos.rt2.push(result); break
|
||
case 'rt3': this.qos.rt3.push(result); break
|
||
case 'cd': this.qos.cd.push(result); break
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
},
|
||
|
||
updateSettings(vals) {
|
||
this.settingFetchComplete = true
|
||
this.updateRxTxSettings(vals)
|
||
this.updateCincSettings(vals)
|
||
this.updateBucLnbSettings(vals)
|
||
this.updateQosSettings(vals)
|
||
},
|
||
|
||
// addQosClass()
|
||
qosAddClass(name) {
|
||
let res = {
|
||
isEnabled: true,
|
||
cir: 0,
|
||
pir: 0,
|
||
description: "",
|
||
filters: []
|
||
}
|
||
switch (name) {
|
||
case 'rt1': this.qos.rt1.push(res); break
|
||
case 'rt2': this.qos.rt2.push(res); break
|
||
case 'rt3': this.qos.rt3.push(res); break
|
||
case 'cd': this.qos.cd.push(res); break
|
||
}
|
||
},
|
||
|
||
qosClassAddRule(name, index) {
|
||
let rule = {
|
||
isEnabled: true,
|
||
vlan: "",
|
||
proto: "",
|
||
sport: "",
|
||
dport: "",
|
||
ip_src: "",
|
||
ip_dest: "",
|
||
dscp: ""
|
||
}
|
||
switch (name) {
|
||
case 'rt1': this.qos.rt1[index].filters.push(rule); break
|
||
case 'rt2': this.qos.rt2[index].filters.push(rule); break
|
||
case 'rt3': this.qos.rt3[index].filters.push(rule); break
|
||
case 'cd': this.qos.cd[index].filters.push(rule); break
|
||
}
|
||
},
|
||
|
||
qosDelClass(name, index) {
|
||
switch (name) {
|
||
case 'rt1': this.qos.rt1.splice(index, 1); break
|
||
case 'rt2': this.qos.rt2.splice(index, 1); break
|
||
case 'rt3': this.qos.rt3.splice(index, 1); break
|
||
case 'cd': this.qos.cd.splice(index, 1); break
|
||
}
|
||
},
|
||
|
||
qosDelFilter(name, index, filterIndex) {
|
||
switch (name) {
|
||
case 'rt1': this.qos.rt1[index].filters.splice(filterIndex, 1); break
|
||
case 'rt2': this.qos.rt2[index].filters.splice(filterIndex, 1); break
|
||
case 'rt3': this.qos.rt3[index].filters.splice(filterIndex, 1); break
|
||
case 'cd': this.qos.cd[index].filters.splice(filterIndex, 1); break
|
||
}
|
||
},
|
||
|
||
qosGenerateRuleDescription(filter) {
|
||
// попытка 1: просто отобразить все фильтры
|
||
let result = ""
|
||
let isFirst = true;
|
||
for (const key in filter) {
|
||
if (key === "isEnabled") {
|
||
continue
|
||
}
|
||
if (!filter[key]) {
|
||
continue
|
||
}
|
||
if (isFirst) {
|
||
isFirst = false;
|
||
} else {
|
||
result += '; '
|
||
}
|
||
result += `${key}: ${filter[key]}`
|
||
}
|
||
if (result === "") {
|
||
return "пустой"
|
||
}
|
||
|
||
const maxResultLen = 60
|
||
|
||
if (result.length > maxResultLen) {
|
||
// попытка 2, отобразить что вообще в этом фильтре использовалось
|
||
result = ""
|
||
isFirst = true;
|
||
for (const key in filter) {
|
||
if (key === "isEnabled") {
|
||
continue
|
||
}
|
||
if (!filter[key]) {
|
||
continue
|
||
}
|
||
if (isFirst) {
|
||
isFirst = false;
|
||
} else {
|
||
result += ', '
|
||
}
|
||
result += `${key}`
|
||
}
|
||
}
|
||
|
||
return result
|
||
}
|
||
},
|
||
mounted() {
|
||
const doFetchStatistics = async () => {
|
||
let d = await fetch("/api/get/statistics")
|
||
this.updateStatistics(await d.json())
|
||
|
||
setTimeout(() => {
|
||
doFetchStatistics()
|
||
}, 1000)
|
||
}
|
||
|
||
doFetchStatistics().then(() => {})
|
||
|
||
this.performUpdateSettings()
|
||
|
||
document.getElementById("app").removeAttribute("hidden")
|
||
}
|
||
})
|
||
|
||
// import MyComponent from './modules/header'
|
||
// const sh = new Vue(MyComponent)
|
||
</script>
|
||
</body>
|
||
</html> |