Merge in DNS/adguard-home from 2295-dhcp-windows to master Updates #2295. Squashed commit of the following: commit 3b00a90c3d9bc33e9af478e4062c0f938d4f327d Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Mon Nov 16 16:45:43 2020 +0300 all: use the 501 handlers instead of the real ones, revert other changes commit 0a3b37736a21abd6181e0d28c32069e8d7a576d0 Merge: 45feba7556358240e9
Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Mon Nov 16 15:59:15 2020 +0300 Merge branch 'master' into 2295-dhcp-windows and update commit 45feba755dde37e43cc8075b896e1576157341e6 Merge: cd987d8bca19523b25
Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Mon Nov 16 15:51:16 2020 +0300 Merge branch 'master' into 2295-dhcp-windows commit cd987d8bc2cd524b7454d9037b595069714645f9 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri Nov 13 15:55:23 2020 +0300 all: improve tests and refactor dhcp checking code even more commit 3aad675443f325b5909523bcc1c987aa04ac61d9 Merge: 70c477e6109196118e
Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri Nov 13 14:44:43 2020 +0300 Merge branch 'master' into 2295-dhcp-windows commit 70c477e61cdc1237603918f1c44470c1549f1136 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri Nov 13 14:34:06 2020 +0300 home: fix dhcpd test on windows commit e59597d783fb9304e63f94eee2b5a5d67a5b2169 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri Nov 13 13:38:25 2020 +0300 all: mention the feature in the changelog commit 5555c8d881b1c20b5b0a0cb096a17cf56e209c06 Merge: c3b6a5a93e802e6645
Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri Nov 13 13:35:35 2020 +0300 Merge branch 'master' into 2295-dhcp-windows commit c3b6a5a930693090838eb1ef9f75a09b5b223ba6 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Thu Nov 12 20:37:09 2020 +0300 util: fix comment commit ed92dfdb5d3a6c4ba5d032cbe781e7fd87882813 Author: ArtemBaskal <asbaskal@miem.hse.ru> Date: Thu Nov 12 20:24:14 2020 +0300 Adapt client commit e6f0494c20a4ad5388492af9091568eea5c6e2d6 Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Thu Nov 12 13:35:25 2020 +0300 return 501 when we don't support features
619 lines
22 KiB
JavaScript
619 lines
22 KiB
JavaScript
import { createAction } from 'redux-actions';
|
|
import i18next from 'i18next';
|
|
import axios from 'axios';
|
|
|
|
import endsWith from 'lodash/endsWith';
|
|
import escapeRegExp from 'lodash/escapeRegExp';
|
|
import React from 'react';
|
|
import { compose } from 'redux';
|
|
import { splitByNewLine, sortClients, filterOutComments } from '../helpers/helpers';
|
|
import {
|
|
BLOCK_ACTIONS,
|
|
CHECK_TIMEOUT,
|
|
STATUS_RESPONSE,
|
|
SETTINGS_NAMES,
|
|
FORM_NAME,
|
|
GETTING_STARTED_LINK,
|
|
} from '../helpers/constants';
|
|
import { areEqualVersions } from '../helpers/version';
|
|
import { getTlsStatus } from './encryption';
|
|
import apiClient from '../api/Api';
|
|
import { addErrorToast, addNoticeToast, addSuccessToast } from './toasts';
|
|
import { getFilteringStatus, setRules } from './filtering';
|
|
|
|
export const toggleSettingStatus = createAction('SETTING_STATUS_TOGGLE');
|
|
export const showSettingsFailure = createAction('SETTINGS_FAILURE_SHOW');
|
|
|
|
export const toggleSetting = (settingKey, status) => async (dispatch) => {
|
|
let successMessage = '';
|
|
try {
|
|
switch (settingKey) {
|
|
case SETTINGS_NAMES.safebrowsing:
|
|
if (status) {
|
|
successMessage = 'disabled_safe_browsing_toast';
|
|
await apiClient.disableSafebrowsing();
|
|
} else {
|
|
successMessage = 'enabled_safe_browsing_toast';
|
|
await apiClient.enableSafebrowsing();
|
|
}
|
|
dispatch(toggleSettingStatus({ settingKey }));
|
|
break;
|
|
case SETTINGS_NAMES.parental:
|
|
if (status) {
|
|
successMessage = 'disabled_parental_toast';
|
|
await apiClient.disableParentalControl();
|
|
} else {
|
|
successMessage = 'enabled_parental_toast';
|
|
await apiClient.enableParentalControl();
|
|
}
|
|
dispatch(toggleSettingStatus({ settingKey }));
|
|
break;
|
|
case SETTINGS_NAMES.safesearch:
|
|
if (status) {
|
|
successMessage = 'disabled_safe_search_toast';
|
|
await apiClient.disableSafesearch();
|
|
} else {
|
|
successMessage = 'enabled_save_search_toast';
|
|
await apiClient.enableSafesearch();
|
|
}
|
|
dispatch(toggleSettingStatus({ settingKey }));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
dispatch(addSuccessToast(successMessage));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
}
|
|
};
|
|
|
|
export const initSettingsRequest = createAction('SETTINGS_INIT_REQUEST');
|
|
export const initSettingsFailure = createAction('SETTINGS_INIT_FAILURE');
|
|
export const initSettingsSuccess = createAction('SETTINGS_INIT_SUCCESS');
|
|
|
|
export const initSettings = (settingsList) => async (dispatch) => {
|
|
dispatch(initSettingsRequest());
|
|
try {
|
|
const safebrowsingStatus = await apiClient.getSafebrowsingStatus();
|
|
const parentalStatus = await apiClient.getParentalStatus();
|
|
const safesearchStatus = await apiClient.getSafesearchStatus();
|
|
const {
|
|
safebrowsing,
|
|
parental,
|
|
safesearch,
|
|
} = settingsList;
|
|
const newSettingsList = {
|
|
safebrowsing: {
|
|
...safebrowsing,
|
|
enabled: safebrowsingStatus.enabled,
|
|
},
|
|
parental: {
|
|
...parental,
|
|
enabled: parentalStatus.enabled,
|
|
},
|
|
safesearch: {
|
|
...safesearch,
|
|
enabled: safesearchStatus.enabled,
|
|
},
|
|
};
|
|
dispatch(initSettingsSuccess({ settingsList: newSettingsList }));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(initSettingsFailure());
|
|
}
|
|
};
|
|
|
|
export const toggleProtectionRequest = createAction('TOGGLE_PROTECTION_REQUEST');
|
|
export const toggleProtectionFailure = createAction('TOGGLE_PROTECTION_FAILURE');
|
|
export const toggleProtectionSuccess = createAction('TOGGLE_PROTECTION_SUCCESS');
|
|
|
|
export const toggleProtection = (status) => async (dispatch) => {
|
|
dispatch(toggleProtectionRequest());
|
|
try {
|
|
const successMessage = status ? 'disabled_protection' : 'enabled_protection';
|
|
await apiClient.setDnsConfig({ protection_enabled: !status });
|
|
dispatch(addSuccessToast(successMessage));
|
|
dispatch(toggleProtectionSuccess());
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(toggleProtectionFailure());
|
|
}
|
|
};
|
|
|
|
export const getVersionRequest = createAction('GET_VERSION_REQUEST');
|
|
export const getVersionFailure = createAction('GET_VERSION_FAILURE');
|
|
export const getVersionSuccess = createAction('GET_VERSION_SUCCESS');
|
|
|
|
export const getVersion = (recheck = false) => async (dispatch, getState) => {
|
|
dispatch(getVersionRequest());
|
|
try {
|
|
const data = await apiClient.getGlobalVersion({ recheck_now: recheck });
|
|
dispatch(getVersionSuccess(data));
|
|
|
|
if (recheck) {
|
|
const { dnsVersion } = getState().dashboard;
|
|
const currentVersion = dnsVersion === 'undefined' ? 0 : dnsVersion;
|
|
|
|
if (data && !areEqualVersions(currentVersion, data.new_version)) {
|
|
dispatch(addSuccessToast('updates_checked'));
|
|
} else {
|
|
dispatch(addSuccessToast('updates_version_equal'));
|
|
}
|
|
}
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getVersionFailure());
|
|
}
|
|
};
|
|
|
|
export const getUpdateRequest = createAction('GET_UPDATE_REQUEST');
|
|
export const getUpdateFailure = createAction('GET_UPDATE_FAILURE');
|
|
export const getUpdateSuccess = createAction('GET_UPDATE_SUCCESS');
|
|
|
|
const checkStatus = async (handleRequestSuccess, handleRequestError, attempts = 60) => {
|
|
let timeout;
|
|
|
|
if (attempts === 0) {
|
|
handleRequestError();
|
|
}
|
|
|
|
const rmTimeout = (t) => t && clearTimeout(t);
|
|
|
|
try {
|
|
const response = await axios.get(`${apiClient.baseUrl}/status`);
|
|
rmTimeout(timeout);
|
|
if (response?.status === 200) {
|
|
handleRequestSuccess(response);
|
|
if (response.data.running === false) {
|
|
timeout = setTimeout(
|
|
checkStatus,
|
|
CHECK_TIMEOUT,
|
|
handleRequestSuccess,
|
|
handleRequestError,
|
|
attempts - 1,
|
|
);
|
|
}
|
|
}
|
|
} catch (error) {
|
|
rmTimeout(timeout);
|
|
timeout = setTimeout(
|
|
checkStatus,
|
|
CHECK_TIMEOUT,
|
|
handleRequestSuccess,
|
|
handleRequestError,
|
|
attempts - 1,
|
|
);
|
|
}
|
|
};
|
|
|
|
export const getUpdate = () => async (dispatch, getState) => {
|
|
const { dnsVersion } = getState().dashboard;
|
|
|
|
dispatch(getUpdateRequest());
|
|
const handleRequestError = () => {
|
|
const options = {
|
|
components: {
|
|
a: <a href={GETTING_STARTED_LINK} target="_blank"
|
|
rel="noopener noreferrer" />,
|
|
},
|
|
};
|
|
|
|
dispatch(addNoticeToast({ error: 'update_failed', options }));
|
|
dispatch(getUpdateFailure());
|
|
};
|
|
|
|
const handleRequestSuccess = (response) => {
|
|
const responseVersion = response.data?.version;
|
|
|
|
if (dnsVersion !== responseVersion) {
|
|
dispatch(getUpdateSuccess());
|
|
window.location.reload(true);
|
|
}
|
|
};
|
|
|
|
try {
|
|
await apiClient.getUpdate();
|
|
checkStatus(handleRequestSuccess, handleRequestError);
|
|
} catch (error) {
|
|
handleRequestError();
|
|
}
|
|
};
|
|
|
|
export const getClientsRequest = createAction('GET_CLIENTS_REQUEST');
|
|
export const getClientsFailure = createAction('GET_CLIENTS_FAILURE');
|
|
export const getClientsSuccess = createAction('GET_CLIENTS_SUCCESS');
|
|
|
|
export const getClients = () => async (dispatch) => {
|
|
dispatch(getClientsRequest());
|
|
try {
|
|
const data = await apiClient.getClients();
|
|
const sortedClients = data.clients && sortClients(data.clients);
|
|
const sortedAutoClients = data.auto_clients && sortClients(data.auto_clients);
|
|
|
|
dispatch(getClientsSuccess({
|
|
clients: sortedClients || [],
|
|
autoClients: sortedAutoClients || [],
|
|
supportedTags: data.supported_tags || [],
|
|
}));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getClientsFailure());
|
|
}
|
|
};
|
|
|
|
export const getProfileRequest = createAction('GET_PROFILE_REQUEST');
|
|
export const getProfileFailure = createAction('GET_PROFILE_FAILURE');
|
|
export const getProfileSuccess = createAction('GET_PROFILE_SUCCESS');
|
|
|
|
export const getProfile = () => async (dispatch) => {
|
|
dispatch(getProfileRequest());
|
|
try {
|
|
const profile = await apiClient.getProfile();
|
|
dispatch(getProfileSuccess(profile));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getProfileFailure());
|
|
}
|
|
};
|
|
|
|
export const dnsStatusRequest = createAction('DNS_STATUS_REQUEST');
|
|
export const dnsStatusFailure = createAction('DNS_STATUS_FAILURE');
|
|
export const dnsStatusSuccess = createAction('DNS_STATUS_SUCCESS');
|
|
export const setDnsRunningStatus = createAction('SET_DNS_RUNNING_STATUS');
|
|
|
|
export const getDnsStatus = () => async (dispatch) => {
|
|
dispatch(dnsStatusRequest());
|
|
|
|
const handleRequestError = () => {
|
|
dispatch(addErrorToast({ error: 'dns_status_error' }));
|
|
dispatch(dnsStatusFailure());
|
|
window.location.reload(true);
|
|
};
|
|
|
|
const handleRequestSuccess = (response) => {
|
|
const dnsStatus = response.data;
|
|
const { running } = dnsStatus;
|
|
const runningStatus = dnsStatus && running;
|
|
if (runningStatus === true) {
|
|
dispatch(dnsStatusSuccess(dnsStatus));
|
|
dispatch(getVersion());
|
|
dispatch(getTlsStatus());
|
|
dispatch(getProfile());
|
|
} else {
|
|
dispatch(setDnsRunningStatus(running));
|
|
}
|
|
};
|
|
|
|
try {
|
|
checkStatus(handleRequestSuccess, handleRequestError);
|
|
} catch (error) {
|
|
handleRequestError(error);
|
|
}
|
|
};
|
|
|
|
export const testUpstreamRequest = createAction('TEST_UPSTREAM_REQUEST');
|
|
export const testUpstreamFailure = createAction('TEST_UPSTREAM_FAILURE');
|
|
export const testUpstreamSuccess = createAction('TEST_UPSTREAM_SUCCESS');
|
|
|
|
export const testUpstream = (
|
|
{ bootstrap_dns, upstream_dns }, upstream_dns_file,
|
|
) => async (dispatch) => {
|
|
dispatch(testUpstreamRequest());
|
|
try {
|
|
const removeComments = compose(filterOutComments, splitByNewLine);
|
|
|
|
const config = {
|
|
bootstrap_dns: splitByNewLine(bootstrap_dns),
|
|
...(upstream_dns_file ? null : {
|
|
upstream_dns: removeComments(upstream_dns),
|
|
}),
|
|
};
|
|
|
|
const upstreamResponse = await apiClient.testUpstream(config);
|
|
const testMessages = Object.keys(upstreamResponse)
|
|
.map((key) => {
|
|
const message = upstreamResponse[key];
|
|
if (message !== 'OK') {
|
|
dispatch(addErrorToast({ error: i18next.t('dns_test_not_ok_toast', { key }) }));
|
|
}
|
|
return message;
|
|
});
|
|
|
|
if (testMessages.every((message) => message === 'OK')) {
|
|
dispatch(addSuccessToast('dns_test_ok_toast'));
|
|
}
|
|
|
|
dispatch(testUpstreamSuccess());
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(testUpstreamFailure());
|
|
}
|
|
};
|
|
|
|
export const testUpstreamWithFormValues = () => async (dispatch, getState) => {
|
|
const { upstream_dns_file } = getState().dnsConfig;
|
|
const { bootstrap_dns, upstream_dns } = getState().form[FORM_NAME.UPSTREAM].values;
|
|
return dispatch(testUpstream({ bootstrap_dns, upstream_dns }, upstream_dns_file));
|
|
};
|
|
|
|
export const changeLanguageRequest = createAction('CHANGE_LANGUAGE_REQUEST');
|
|
export const changeLanguageFailure = createAction('CHANGE_LANGUAGE_FAILURE');
|
|
export const changeLanguageSuccess = createAction('CHANGE_LANGUAGE_SUCCESS');
|
|
|
|
export const changeLanguage = (lang) => async (dispatch) => {
|
|
dispatch(changeLanguageRequest());
|
|
try {
|
|
await apiClient.changeLanguage(lang);
|
|
dispatch(changeLanguageSuccess());
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(changeLanguageFailure());
|
|
}
|
|
};
|
|
|
|
export const getLanguageRequest = createAction('GET_LANGUAGE_REQUEST');
|
|
export const getLanguageFailure = createAction('GET_LANGUAGE_FAILURE');
|
|
export const getLanguageSuccess = createAction('GET_LANGUAGE_SUCCESS');
|
|
|
|
export const getLanguage = () => async (dispatch) => {
|
|
dispatch(getLanguageRequest());
|
|
try {
|
|
const language = await apiClient.getCurrentLanguage();
|
|
dispatch(getLanguageSuccess(language));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getLanguageFailure());
|
|
}
|
|
};
|
|
|
|
export const getDhcpStatusRequest = createAction('GET_DHCP_STATUS_REQUEST');
|
|
export const getDhcpStatusSuccess = createAction('GET_DHCP_STATUS_SUCCESS');
|
|
export const getDhcpStatusFailure = createAction('GET_DHCP_STATUS_FAILURE');
|
|
|
|
export const getDhcpStatus = () => async (dispatch) => {
|
|
dispatch(getDhcpStatusRequest());
|
|
try {
|
|
const globalStatus = await apiClient.getGlobalStatus();
|
|
if (globalStatus.dhcp_available) {
|
|
const status = await apiClient.getDhcpStatus();
|
|
status.dhcp_available = globalStatus.dhcp_available;
|
|
dispatch(getDhcpStatusSuccess(status));
|
|
} else {
|
|
dispatch(getDhcpStatusFailure());
|
|
}
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getDhcpStatusFailure());
|
|
}
|
|
};
|
|
|
|
export const getDhcpInterfacesRequest = createAction('GET_DHCP_INTERFACES_REQUEST');
|
|
export const getDhcpInterfacesSuccess = createAction('GET_DHCP_INTERFACES_SUCCESS');
|
|
export const getDhcpInterfacesFailure = createAction('GET_DHCP_INTERFACES_FAILURE');
|
|
|
|
export const getDhcpInterfaces = () => async (dispatch) => {
|
|
dispatch(getDhcpInterfacesRequest());
|
|
try {
|
|
const interfaces = await apiClient.getDhcpInterfaces();
|
|
dispatch(getDhcpInterfacesSuccess(interfaces));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(getDhcpInterfacesFailure());
|
|
}
|
|
};
|
|
|
|
export const findActiveDhcpRequest = createAction('FIND_ACTIVE_DHCP_REQUEST');
|
|
export const findActiveDhcpSuccess = createAction('FIND_ACTIVE_DHCP_SUCCESS');
|
|
export const findActiveDhcpFailure = createAction('FIND_ACTIVE_DHCP_FAILURE');
|
|
|
|
export const findActiveDhcp = (name) => async (dispatch, getState) => {
|
|
dispatch(findActiveDhcpRequest());
|
|
try {
|
|
const activeDhcp = await apiClient.findActiveDhcp(name);
|
|
dispatch(findActiveDhcpSuccess(activeDhcp));
|
|
const { check, interface_name, interfaces } = getState().dhcp;
|
|
const selectedInterface = getState().form[FORM_NAME.DHCP_INTERFACES].values.interface_name;
|
|
const v4 = check?.v4 ?? { static_ip: {}, other_server: {} };
|
|
const v6 = check?.v6 ?? { other_server: {} };
|
|
|
|
let isError = false;
|
|
let isStaticIPError = false;
|
|
|
|
const hasV4Interface = !!interfaces[selectedInterface]?.ipv4_addresses;
|
|
const hasV6Interface = !!interfaces[selectedInterface]?.ipv6_addresses;
|
|
|
|
if (hasV4Interface && v4.other_server.found === STATUS_RESPONSE.ERROR) {
|
|
isError = true;
|
|
if (v4.other_server.error) {
|
|
dispatch(addErrorToast({ error: v4.other_server.error }));
|
|
}
|
|
}
|
|
|
|
if (hasV6Interface && v6.other_server.found === STATUS_RESPONSE.ERROR) {
|
|
isError = true;
|
|
if (v6.other_server.error) {
|
|
dispatch(addErrorToast({ error: v6.other_server.error }));
|
|
}
|
|
}
|
|
|
|
if (hasV4Interface && v4.static_ip.static === STATUS_RESPONSE.ERROR) {
|
|
isStaticIPError = true;
|
|
dispatch(addErrorToast({ error: 'dhcp_static_ip_error' }));
|
|
}
|
|
|
|
|
|
if (isError) {
|
|
dispatch(addErrorToast({ error: 'dhcp_error' }));
|
|
}
|
|
|
|
if (isStaticIPError || isError) {
|
|
// No need to proceed if there was an error discovering DHCP server
|
|
return;
|
|
}
|
|
|
|
if ((hasV4Interface && v4.other_server.found === STATUS_RESPONSE.YES)
|
|
|| (hasV6Interface && v6.other_server.found === STATUS_RESPONSE.YES)) {
|
|
dispatch(addErrorToast({ error: 'dhcp_found' }));
|
|
} else if (hasV4Interface && v4.static_ip.static === STATUS_RESPONSE.NO
|
|
&& v4.static_ip.ip
|
|
&& interface_name) {
|
|
const warning = i18next.t('dhcp_dynamic_ip_found', {
|
|
interfaceName: interface_name,
|
|
ipAddress: v4.static_ip.ip,
|
|
interpolation: {
|
|
prefix: '<0>{{',
|
|
suffix: '}}</0>',
|
|
},
|
|
});
|
|
dispatch(addErrorToast({ error: warning }));
|
|
} else {
|
|
dispatch(addSuccessToast('dhcp_not_found'));
|
|
}
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(findActiveDhcpFailure());
|
|
}
|
|
};
|
|
|
|
export const setDhcpConfigRequest = createAction('SET_DHCP_CONFIG_REQUEST');
|
|
export const setDhcpConfigSuccess = createAction('SET_DHCP_CONFIG_SUCCESS');
|
|
export const setDhcpConfigFailure = createAction('SET_DHCP_CONFIG_FAILURE');
|
|
|
|
export const setDhcpConfig = (values) => async (dispatch) => {
|
|
dispatch(setDhcpConfigRequest());
|
|
try {
|
|
await apiClient.setDhcpConfig(values);
|
|
dispatch(setDhcpConfigSuccess(values));
|
|
dispatch(addSuccessToast('dhcp_config_saved'));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(setDhcpConfigFailure());
|
|
}
|
|
};
|
|
|
|
export const toggleDhcpRequest = createAction('TOGGLE_DHCP_REQUEST');
|
|
export const toggleDhcpFailure = createAction('TOGGLE_DHCP_FAILURE');
|
|
export const toggleDhcpSuccess = createAction('TOGGLE_DHCP_SUCCESS');
|
|
|
|
export const toggleDhcp = (values) => async (dispatch) => {
|
|
dispatch(toggleDhcpRequest());
|
|
let config = {
|
|
...values,
|
|
enabled: false,
|
|
};
|
|
let successMessage = 'disabled_dhcp';
|
|
|
|
if (!values.enabled) {
|
|
config = {
|
|
...values,
|
|
enabled: true,
|
|
};
|
|
successMessage = 'enabled_dhcp';
|
|
}
|
|
|
|
try {
|
|
await apiClient.setDhcpConfig(config);
|
|
dispatch(toggleDhcpSuccess());
|
|
dispatch(addSuccessToast(successMessage));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(toggleDhcpFailure());
|
|
}
|
|
};
|
|
|
|
export const resetDhcpRequest = createAction('RESET_DHCP_REQUEST');
|
|
export const resetDhcpSuccess = createAction('RESET_DHCP_SUCCESS');
|
|
export const resetDhcpFailure = createAction('RESET_DHCP_FAILURE');
|
|
|
|
export const resetDhcp = () => async (dispatch) => {
|
|
dispatch(resetDhcpRequest());
|
|
try {
|
|
const status = await apiClient.resetDhcp();
|
|
dispatch(resetDhcpSuccess(status));
|
|
dispatch(addSuccessToast('dhcp_config_saved'));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(resetDhcpFailure());
|
|
}
|
|
};
|
|
|
|
export const toggleLeaseModal = createAction('TOGGLE_LEASE_MODAL');
|
|
|
|
export const addStaticLeaseRequest = createAction('ADD_STATIC_LEASE_REQUEST');
|
|
export const addStaticLeaseFailure = createAction('ADD_STATIC_LEASE_FAILURE');
|
|
export const addStaticLeaseSuccess = createAction('ADD_STATIC_LEASE_SUCCESS');
|
|
|
|
export const addStaticLease = (config) => async (dispatch) => {
|
|
dispatch(addStaticLeaseRequest());
|
|
try {
|
|
const name = config.hostname || config.ip;
|
|
await apiClient.addStaticLease(config);
|
|
dispatch(addStaticLeaseSuccess(config));
|
|
dispatch(addSuccessToast(i18next.t('dhcp_lease_added', { key: name })));
|
|
dispatch(toggleLeaseModal());
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(addStaticLeaseFailure());
|
|
}
|
|
};
|
|
|
|
export const removeStaticLeaseRequest = createAction('REMOVE_STATIC_LEASE_REQUEST');
|
|
export const removeStaticLeaseFailure = createAction('REMOVE_STATIC_LEASE_FAILURE');
|
|
export const removeStaticLeaseSuccess = createAction('REMOVE_STATIC_LEASE_SUCCESS');
|
|
|
|
export const removeStaticLease = (config) => async (dispatch) => {
|
|
dispatch(removeStaticLeaseRequest());
|
|
try {
|
|
const name = config.hostname || config.ip;
|
|
await apiClient.removeStaticLease(config);
|
|
dispatch(removeStaticLeaseSuccess(config));
|
|
dispatch(addSuccessToast(i18next.t('dhcp_lease_deleted', { key: name })));
|
|
} catch (error) {
|
|
dispatch(addErrorToast({ error }));
|
|
dispatch(removeStaticLeaseFailure());
|
|
}
|
|
};
|
|
|
|
export const removeToast = createAction('REMOVE_TOAST');
|
|
|
|
export const toggleBlocking = (
|
|
type, domain, baseRule, baseUnblocking,
|
|
) => async (dispatch, getState) => {
|
|
const baseBlockingRule = baseRule || `||${domain}^$important`;
|
|
const baseUnblockingRule = baseUnblocking || `@@${baseBlockingRule}`;
|
|
const { userRules } = getState().filtering;
|
|
|
|
const lineEnding = !endsWith(userRules, '\n') ? '\n' : '';
|
|
|
|
const blockingRule = type === BLOCK_ACTIONS.BLOCK ? baseUnblockingRule : baseBlockingRule;
|
|
const unblockingRule = type === BLOCK_ACTIONS.BLOCK ? baseBlockingRule : baseUnblockingRule;
|
|
const preparedBlockingRule = new RegExp(`(^|\n)${escapeRegExp(blockingRule)}($|\n)`);
|
|
const preparedUnblockingRule = new RegExp(`(^|\n)${escapeRegExp(unblockingRule)}($|\n)`);
|
|
|
|
const matchPreparedBlockingRule = userRules.match(preparedBlockingRule);
|
|
const matchPreparedUnblockingRule = userRules.match(preparedUnblockingRule);
|
|
|
|
if (matchPreparedBlockingRule) {
|
|
await dispatch(setRules(userRules.replace(`${blockingRule}`, '')));
|
|
dispatch(addSuccessToast(i18next.t('rule_removed_from_custom_filtering_toast', { rule: blockingRule })));
|
|
} else if (!matchPreparedUnblockingRule) {
|
|
await dispatch(setRules(`${userRules}${lineEnding}${unblockingRule}\n`));
|
|
dispatch(addSuccessToast(i18next.t('rule_added_to_custom_filtering_toast', { rule: unblockingRule })));
|
|
} else if (matchPreparedUnblockingRule) {
|
|
dispatch(addSuccessToast(i18next.t('rule_added_to_custom_filtering_toast', { rule: unblockingRule })));
|
|
return;
|
|
} else if (!matchPreparedBlockingRule) {
|
|
dispatch(addSuccessToast(i18next.t('rule_removed_from_custom_filtering_toast', { rule: blockingRule })));
|
|
return;
|
|
}
|
|
|
|
dispatch(getFilteringStatus());
|
|
};
|
|
|
|
export const toggleBlockingForClient = (type, domain, client) => {
|
|
const baseRule = `||${domain}^$client='${client.replace(/'/g, '/\'')}'`;
|
|
const baseUnblocking = `@@${baseRule}`;
|
|
|
|
return toggleBlocking(type, domain, baseRule, baseUnblocking);
|
|
};
|