AG-21212 - Custom logs and stats retention

Updates#3404

Squashed commit of the following:

commit b68a1d08b0676ebb7abbb13c9274c8d509cd6eed
Merge: 81265147 6d402dc8
Author: Artem Krisanov <a.krisanov@adguard.com>
Date:   Mon Apr 17 15:48:33 2023 +0300

    Merge master

commit 81265147b5613be11a6621a416f9588c0e1c0ef5
Author: Artem Krisanov <a.krisanov@adguard.com>
Date:   Thu Apr 13 10:54:39 2023 +0300

    Changed query log 'retention' --> 'rotation'.

commit 02c5dc0b54bca9ec293ee8629d769489bc5dc533
Author: Artem Krisanov <a.krisanov@adguard.com>
Date:   Wed Apr 12 13:22:22 2023 +0300

    Custom inputs for query log and stats configs.

commit 21dbfbd8aac868baeea0f8b25d14786aecf09a0d
Author: Artem Krisanov <a.krisanov@adguard.com>
Date:   Tue Apr 11 18:12:40 2023 +0300

    Temporary changes.
This commit is contained in:
Artem Krisanov 2023-04-17 15:57:57 +03:00
parent 6d402dc86c
commit e43ba17884
10 changed files with 233 additions and 24 deletions

View File

@ -257,12 +257,12 @@
"query_log_cleared": "The query log has been successfully cleared", "query_log_cleared": "The query log has been successfully cleared",
"query_log_updated": "The query log has been successfully updated", "query_log_updated": "The query log has been successfully updated",
"query_log_clear": "Clear query logs", "query_log_clear": "Clear query logs",
"query_log_retention": "Query logs retention", "query_log_retention": "Query logs rotation",
"query_log_enable": "Enable log", "query_log_enable": "Enable log",
"query_log_configuration": "Logs configuration", "query_log_configuration": "Logs configuration",
"query_log_disabled": "The query log is disabled and can be configured in the <0>settings</0>", "query_log_disabled": "The query log is disabled and can be configured in the <0>settings</0>",
"query_log_strict_search": "Use double quotes for strict search", "query_log_strict_search": "Use double quotes for strict search",
"query_log_retention_confirm": "Are you sure you want to change query log retention? If you decrease the interval value, some data will be lost", "query_log_retention_confirm": "Are you sure you want to change query log rotation? If you decrease the interval value, some data will be lost",
"anonymize_client_ip": "Anonymize client IP", "anonymize_client_ip": "Anonymize client IP",
"anonymize_client_ip_desc": "Don't save the client's full IP address to logs or statistics", "anonymize_client_ip_desc": "Don't save the client's full IP address to logs or statistics",
"dns_config": "DNS server configuration", "dns_config": "DNS server configuration",
@ -669,6 +669,8 @@
"disable_notify_for_hours_plural": "Disable protection for {{count}} hours", "disable_notify_for_hours_plural": "Disable protection for {{count}} hours",
"disable_notify_until_tomorrow": "Disable protection until tomorrow", "disable_notify_until_tomorrow": "Disable protection until tomorrow",
"enable_protection_timer": "Protection will be enabled in {{time}}", "enable_protection_timer": "Protection will be enabled in {{time}}",
"custom_retention_input": "Enter retention in hours",
"custom_rotation_input": "Enter rotation in hours",
"protection_section_label": "Protection", "protection_section_label": "Protection",
"log_and_stats_section_label": "Query log and statistics", "log_and_stats_section_label": "Query log and statistics",
"ignore_query_log": "Ignore this client in query log", "ignore_query_log": "Ignore this client in query log",

View File

@ -1,25 +1,37 @@
import React from 'react'; import React, { useEffect } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { Field, reduxForm } from 'redux-form'; import {
change,
Field,
formValueSelector,
reduxForm,
} from 'redux-form';
import { connect } from 'react-redux';
import { Trans, withTranslation } from 'react-i18next'; import { Trans, withTranslation } from 'react-i18next';
import flow from 'lodash/flow'; import flow from 'lodash/flow';
import { import {
CheckboxField, CheckboxField,
renderRadioField,
toFloatNumber, toFloatNumber,
renderTextareaField, renderTextareaField, renderInputField, renderRadioField,
} from '../../../helpers/form'; } from '../../../helpers/form';
import { import {
FORM_NAME, FORM_NAME,
QUERY_LOG_INTERVALS_DAYS, QUERY_LOG_INTERVALS_DAYS,
HOUR, HOUR,
DAY, DAY,
RETENTION_CUSTOM,
RETENTION_CUSTOM_INPUT,
RETENTION_RANGE,
CUSTOM_INTERVAL,
} from '../../../helpers/constants'; } from '../../../helpers/constants';
import '../FormButton.css'; import '../FormButton.css';
const getIntervalTitle = (interval, t) => { const getIntervalTitle = (interval, t) => {
switch (interval) { switch (interval) {
case RETENTION_CUSTOM:
return t('settings_custom');
case 6 * HOUR: case 6 * HOUR:
return t('interval_6_hour'); return t('interval_6_hour');
case DAY: case DAY:
@ -42,11 +54,26 @@ const getIntervalFields = (processing, t, toNumber) => QUERY_LOG_INTERVALS_DAYS.
/> />
)); ));
const Form = (props) => { let Form = (props) => {
const { const {
handleSubmit, submitting, invalid, processing, processingClear, handleClear, t, handleSubmit,
submitting,
invalid,
processing,
processingClear,
handleClear,
t,
interval,
customInterval,
dispatch,
} = props; } = props;
useEffect(() => {
if (QUERY_LOG_INTERVALS_DAYS.includes(interval)) {
dispatch(change(FORM_NAME.LOG_CONFIG, CUSTOM_INTERVAL, null));
}
}, [interval]);
return ( return (
<form onSubmit={handleSubmit}> <form onSubmit={handleSubmit}>
<div className="form__group form__group--settings"> <div className="form__group form__group--settings">
@ -73,6 +100,37 @@ const Form = (props) => {
</label> </label>
<div className="form__group form__group--settings"> <div className="form__group form__group--settings">
<div className="custom-controls-stacked"> <div className="custom-controls-stacked">
<Field
key={RETENTION_CUSTOM}
name="interval"
type="radio"
component={renderRadioField}
value={QUERY_LOG_INTERVALS_DAYS.includes(interval)
? RETENTION_CUSTOM
: interval
}
placeholder={getIntervalTitle(RETENTION_CUSTOM, t)}
normalize={toFloatNumber}
disabled={processing}
/>
{!QUERY_LOG_INTERVALS_DAYS.includes(interval) && (
<div className="form__group--input">
<div className="form__desc form__desc--top">
{t('custom_rotation_input')}
</div>
<Field
key={RETENTION_CUSTOM_INPUT}
name={CUSTOM_INTERVAL}
type="number"
className="form-control"
component={renderInputField}
disabled={processing}
normalize={toFloatNumber}
min={RETENTION_RANGE.MIN}
max={RETENTION_RANGE.MAX}
/>
</div>
)}
{getIntervalFields(processing, t, toFloatNumber)} {getIntervalFields(processing, t, toFloatNumber)}
</div> </div>
</div> </div>
@ -96,7 +154,12 @@ const Form = (props) => {
<button <button
type="submit" type="submit"
className="btn btn-success btn-standard btn-large" className="btn btn-success btn-standard btn-large"
disabled={submitting || invalid || processing} disabled={
submitting
|| invalid
|| processing
|| (!QUERY_LOG_INTERVALS_DAYS.includes(interval) && !customInterval)
}
> >
<Trans>save_btn</Trans> <Trans>save_btn</Trans>
</button> </button>
@ -121,8 +184,22 @@ Form.propTypes = {
processing: PropTypes.bool.isRequired, processing: PropTypes.bool.isRequired,
processingClear: PropTypes.bool.isRequired, processingClear: PropTypes.bool.isRequired,
t: PropTypes.func.isRequired, t: PropTypes.func.isRequired,
interval: PropTypes.number,
customInterval: PropTypes.number,
dispatch: PropTypes.func.isRequired,
}; };
const selector = formValueSelector(FORM_NAME.LOG_CONFIG);
Form = connect((state) => {
const interval = selector(state, 'interval');
const customInterval = selector(state, CUSTOM_INTERVAL);
return {
interval,
customInterval,
};
})(Form);
export default flow([ export default flow([
withTranslation(), withTranslation(),
reduxForm({ form: FORM_NAME.LOG_CONFIG }), reduxForm({ form: FORM_NAME.LOG_CONFIG }),

View File

@ -4,15 +4,22 @@ import { withTranslation } from 'react-i18next';
import Card from '../../ui/Card'; import Card from '../../ui/Card';
import Form from './Form'; import Form from './Form';
import { HOUR } from '../../../helpers/constants';
class LogsConfig extends Component { class LogsConfig extends Component {
handleFormSubmit = (values) => { handleFormSubmit = (values) => {
const { t, interval: prevInterval } = this.props; const { t, interval: prevInterval } = this.props;
const { interval } = values; const { interval, customInterval, ...rest } = values;
const data = { ...values, ignored: values.ignored ? values.ignored.split('\n') : [] }; const newInterval = customInterval ? customInterval * HOUR : interval;
if (interval !== prevInterval) { const data = {
...rest,
ignored: values.ignored ? values.ignored.split('\n') : [],
interval: newInterval,
};
if (newInterval < prevInterval) {
// eslint-disable-next-line no-alert // eslint-disable-next-line no-alert
if (window.confirm(t('query_log_retention_confirm'))) { if (window.confirm(t('query_log_retention_confirm'))) {
this.props.setLogsConfig(data); this.props.setLogsConfig(data);
@ -32,7 +39,14 @@ class LogsConfig extends Component {
render() { render() {
const { const {
t, enabled, interval, processing, processingClear, anonymize_client_ip, ignored, t,
enabled,
interval,
processing,
processingClear,
anonymize_client_ip,
ignored,
customInterval,
} = this.props; } = this.props;
return ( return (
@ -46,6 +60,7 @@ class LogsConfig extends Component {
initialValues={{ initialValues={{
enabled, enabled,
interval, interval,
customInterval,
anonymize_client_ip, anonymize_client_ip,
ignored: ignored.join('\n'), ignored: ignored.join('\n'),
}} }}
@ -62,6 +77,7 @@ class LogsConfig extends Component {
LogsConfig.propTypes = { LogsConfig.propTypes = {
interval: PropTypes.number.isRequired, interval: PropTypes.number.isRequired,
customInterval: PropTypes.number,
enabled: PropTypes.bool.isRequired, enabled: PropTypes.bool.isRequired,
anonymize_client_ip: PropTypes.bool.isRequired, anonymize_client_ip: PropTypes.bool.isRequired,
processing: PropTypes.bool.isRequired, processing: PropTypes.bool.isRequired,

View File

@ -18,6 +18,11 @@
font-size: 14px; font-size: 14px;
} }
.form__group--input {
max-width: 300px;
margin: 0 1.5rem 10px;
}
.form__group--checkbox { .form__group--checkbox {
margin-bottom: 25px; margin-bottom: 25px;
} }

View File

@ -1,32 +1,44 @@
import React from 'react'; import React, { useEffect } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { Field, reduxForm } from 'redux-form'; import {
change, Field, formValueSelector, reduxForm,
} from 'redux-form';
import { Trans, withTranslation } from 'react-i18next'; import { Trans, withTranslation } from 'react-i18next';
import flow from 'lodash/flow'; import flow from 'lodash/flow';
import { connect } from 'react-redux';
import { import {
renderRadioField, renderRadioField,
toNumber, toNumber,
CheckboxField, CheckboxField,
renderTextareaField, renderTextareaField,
toFloatNumber,
renderInputField,
} from '../../../helpers/form'; } from '../../../helpers/form';
import { import {
FORM_NAME, FORM_NAME,
STATS_INTERVALS_DAYS, STATS_INTERVALS_DAYS,
DAY, DAY,
RETENTION_CUSTOM,
RETENTION_CUSTOM_INPUT,
CUSTOM_INTERVAL,
RETENTION_RANGE,
} from '../../../helpers/constants'; } from '../../../helpers/constants';
import '../FormButton.css'; import '../FormButton.css';
const getIntervalTitle = (intervalMs, t) => { const getIntervalTitle = (intervalMs, t) => {
switch (intervalMs / DAY) { switch (intervalMs) {
case 1: case RETENTION_CUSTOM:
return t('settings_custom');
case DAY:
return t('interval_24_hour'); return t('interval_24_hour');
default: default:
return t('interval_days', { count: intervalMs / DAY }); return t('interval_days', { count: intervalMs / DAY });
} }
}; };
const Form = (props) => { let Form = (props) => {
const { const {
handleSubmit, handleSubmit,
processing, processing,
@ -35,8 +47,17 @@ const Form = (props) => {
handleReset, handleReset,
processingReset, processingReset,
t, t,
interval,
customInterval,
dispatch,
} = props; } = props;
useEffect(() => {
if (STATS_INTERVALS_DAYS.includes(interval)) {
dispatch(change(FORM_NAME.STATS_CONFIG, CUSTOM_INTERVAL, null));
}
}, [interval]);
return ( return (
<form onSubmit={handleSubmit}> <form onSubmit={handleSubmit}>
<div className="form__group form__group--settings"> <div className="form__group form__group--settings">
@ -56,6 +77,37 @@ const Form = (props) => {
</div> </div>
<div className="form__group form__group--settings mt-2"> <div className="form__group form__group--settings mt-2">
<div className="custom-controls-stacked"> <div className="custom-controls-stacked">
<Field
key={RETENTION_CUSTOM}
name="interval"
type="radio"
component={renderRadioField}
value={STATS_INTERVALS_DAYS.includes(interval)
? RETENTION_CUSTOM
: interval
}
placeholder={getIntervalTitle(RETENTION_CUSTOM, t)}
normalize={toFloatNumber}
disabled={processing}
/>
{!STATS_INTERVALS_DAYS.includes(interval) && (
<div className="form__group--input">
<div className="form__desc form__desc--top">
{t('custom_retention_input')}
</div>
<Field
key={RETENTION_CUSTOM_INPUT}
name={CUSTOM_INTERVAL}
type="number"
className="form-control"
component={renderInputField}
disabled={processing}
normalize={toFloatNumber}
min={RETENTION_RANGE.MIN}
max={RETENTION_RANGE.MAX}
/>
</div>
)}
{STATS_INTERVALS_DAYS.map((interval) => ( {STATS_INTERVALS_DAYS.map((interval) => (
<Field <Field
key={interval} key={interval}
@ -90,7 +142,12 @@ const Form = (props) => {
<button <button
type="submit" type="submit"
className="btn btn-success btn-standard btn-large" className="btn btn-success btn-standard btn-large"
disabled={submitting || invalid || processing} disabled={
submitting
|| invalid
|| processing
|| (!STATS_INTERVALS_DAYS.includes(interval) && !customInterval)
}
> >
<Trans>save_btn</Trans> <Trans>save_btn</Trans>
</button> </button>
@ -116,8 +173,22 @@ Form.propTypes = {
processing: PropTypes.bool.isRequired, processing: PropTypes.bool.isRequired,
processingReset: PropTypes.bool.isRequired, processingReset: PropTypes.bool.isRequired,
t: PropTypes.func.isRequired, t: PropTypes.func.isRequired,
interval: PropTypes.number,
customInterval: PropTypes.number,
dispatch: PropTypes.func.isRequired,
}; };
const selector = formValueSelector(FORM_NAME.STATS_CONFIG);
Form = connect((state) => {
const interval = selector(state, 'interval');
const customInterval = selector(state, CUSTOM_INTERVAL);
return {
interval,
customInterval,
};
})(Form);
export default flow([ export default flow([
withTranslation(), withTranslation(),
reduxForm({ form: FORM_NAME.STATS_CONFIG }), reduxForm({ form: FORM_NAME.STATS_CONFIG }),

View File

@ -4,13 +4,18 @@ import { withTranslation } from 'react-i18next';
import Card from '../../ui/Card'; import Card from '../../ui/Card';
import Form from './Form'; import Form from './Form';
import { HOUR } from '../../../helpers/constants';
class StatsConfig extends Component { class StatsConfig extends Component {
handleFormSubmit = ({ enabled, interval, ignored }) => { handleFormSubmit = ({
enabled, interval, ignored, customInterval,
}) => {
const { t, interval: prevInterval } = this.props; const { t, interval: prevInterval } = this.props;
const newInterval = customInterval ? customInterval * HOUR : interval;
const config = { const config = {
enabled, enabled,
interval, interval: newInterval,
ignored: ignored ? ignored.split('\n') : [], ignored: ignored ? ignored.split('\n') : [],
}; };
@ -33,7 +38,13 @@ class StatsConfig extends Component {
render() { render() {
const { const {
t, interval, processing, processingReset, ignored, enabled, t,
interval,
customInterval,
processing,
processingReset,
ignored,
enabled,
} = this.props; } = this.props;
return ( return (
@ -46,6 +57,7 @@ class StatsConfig extends Component {
<Form <Form
initialValues={{ initialValues={{
interval, interval,
customInterval,
enabled, enabled,
ignored: ignored.join('\n'), ignored: ignored.join('\n'),
}} }}
@ -62,6 +74,7 @@ class StatsConfig extends Component {
StatsConfig.propTypes = { StatsConfig.propTypes = {
interval: PropTypes.number.isRequired, interval: PropTypes.number.isRequired,
customInterval: PropTypes.number,
ignored: PropTypes.array.isRequired, ignored: PropTypes.array.isRequired,
enabled: PropTypes.bool.isRequired, enabled: PropTypes.bool.isRequired,
processing: PropTypes.bool.isRequired, processing: PropTypes.bool.isRequired,

View File

@ -124,6 +124,7 @@ class Settings extends Component {
enabled={queryLogs.enabled} enabled={queryLogs.enabled}
ignored={queryLogs.ignored} ignored={queryLogs.ignored}
interval={queryLogs.interval} interval={queryLogs.interval}
customInterval={queryLogs.customInterval}
anonymize_client_ip={queryLogs.anonymize_client_ip} anonymize_client_ip={queryLogs.anonymize_client_ip}
processing={queryLogs.processingSetConfig} processing={queryLogs.processingSetConfig}
processingClear={queryLogs.processingClear} processingClear={queryLogs.processingClear}
@ -134,6 +135,7 @@ class Settings extends Component {
<div className="col-md-12"> <div className="col-md-12">
<StatsConfig <StatsConfig
interval={stats.interval} interval={stats.interval}
customInterval={stats.customInterval}
ignored={stats.ignored} ignored={stats.ignored}
enabled={stats.enabled} enabled={stats.enabled}
processing={stats.processingSetConfig} processing={stats.processingSetConfig}
@ -166,6 +168,7 @@ Settings.propTypes = {
stats: PropTypes.shape({ stats: PropTypes.shape({
processingGetConfig: PropTypes.bool, processingGetConfig: PropTypes.bool,
interval: PropTypes.number, interval: PropTypes.number,
customInterval: PropTypes.number,
enabled: PropTypes.bool, enabled: PropTypes.bool,
ignored: PropTypes.array, ignored: PropTypes.array,
processingSetConfig: PropTypes.bool, processingSetConfig: PropTypes.bool,
@ -174,6 +177,7 @@ Settings.propTypes = {
queryLogs: PropTypes.shape({ queryLogs: PropTypes.shape({
enabled: PropTypes.bool, enabled: PropTypes.bool,
interval: PropTypes.number, interval: PropTypes.number,
customInterval: PropTypes.number,
anonymize_client_ip: PropTypes.bool, anonymize_client_ip: PropTypes.bool,
processingSetConfig: PropTypes.bool, processingSetConfig: PropTypes.bool,
processingClear: PropTypes.bool, processingClear: PropTypes.bool,

View File

@ -220,6 +220,12 @@ export const STATS_INTERVALS_DAYS = [DAY, DAY * 7, DAY * 30, DAY * 90];
export const QUERY_LOG_INTERVALS_DAYS = [HOUR * 6, DAY, DAY * 7, DAY * 30, DAY * 90]; export const QUERY_LOG_INTERVALS_DAYS = [HOUR * 6, DAY, DAY * 7, DAY * 30, DAY * 90];
export const RETENTION_CUSTOM = 1;
export const RETENTION_CUSTOM_INPUT = 'custom_retention_input';
export const CUSTOM_INTERVAL = 'customInterval';
export const FILTERS_INTERVALS_HOURS = [0, 1, 12, 24, 72, 168]; export const FILTERS_INTERVALS_HOURS = [0, 1, 12, 24, 72, 168];
// Note that translation strings contain these modes (blocking_mode_CONSTANT) // Note that translation strings contain these modes (blocking_mode_CONSTANT)
@ -462,6 +468,11 @@ export const UINT32_RANGE = {
MAX: 4294967295, MAX: 4294967295,
}; };
export const RETENTION_RANGE = {
MIN: 1,
MAX: 365 * 24,
};
export const DHCP_VALUES_PLACEHOLDERS = { export const DHCP_VALUES_PLACEHOLDERS = {
ipv4: { ipv4: {
subnet_mask: '255.255.255.0', subnet_mask: '255.255.255.0',

View File

@ -1,7 +1,9 @@
import { handleActions } from 'redux-actions'; import { handleActions } from 'redux-actions';
import * as actions from '../actions/queryLogs'; import * as actions from '../actions/queryLogs';
import { DEFAULT_LOGS_FILTER, DAY } from '../helpers/constants'; import {
DEFAULT_LOGS_FILTER, DAY, QUERY_LOG_INTERVALS_DAYS, HOUR,
} from '../helpers/constants';
const queryLogs = handleActions( const queryLogs = handleActions(
{ {
@ -59,6 +61,9 @@ const queryLogs = handleActions(
[actions.getLogsConfigSuccess]: (state, { payload }) => ({ [actions.getLogsConfigSuccess]: (state, { payload }) => ({
...state, ...state,
...payload, ...payload,
customInterval: !QUERY_LOG_INTERVALS_DAYS.includes(payload.interval)
? payload.interval / HOUR
: null,
processingGetConfig: false, processingGetConfig: false,
}), }),
@ -95,6 +100,7 @@ const queryLogs = handleActions(
anonymize_client_ip: false, anonymize_client_ip: false,
isDetailed: true, isDetailed: true,
isEntireLog: false, isEntireLog: false,
customInterval: null,
}, },
); );

View File

@ -1,6 +1,6 @@
import { handleActions } from 'redux-actions'; import { handleActions } from 'redux-actions';
import { normalizeTopClients } from '../helpers/helpers'; import { normalizeTopClients } from '../helpers/helpers';
import { DAY } from '../helpers/constants'; import { DAY, HOUR, STATS_INTERVALS_DAYS } from '../helpers/constants';
import * as actions from '../actions/stats'; import * as actions from '../actions/stats';
@ -27,6 +27,9 @@ const stats = handleActions(
[actions.getStatsConfigSuccess]: (state, { payload }) => ({ [actions.getStatsConfigSuccess]: (state, { payload }) => ({
...state, ...state,
...payload, ...payload,
customInterval: !STATS_INTERVALS_DAYS.includes(payload.interval)
? payload.interval / HOUR
: null,
processingGetConfig: false, processingGetConfig: false,
}), }),
@ -93,6 +96,7 @@ const stats = handleActions(
processingStats: true, processingStats: true,
processingReset: false, processingReset: false,
interval: DAY, interval: DAY,
customInterval: null,
...defaultStats, ...defaultStats,
}, },
); );