From ebcabdc10f005e86073c2cdd8e73aa44196a0933 Mon Sep 17 00:00:00 2001 From: GermaVinsmoke Date: Sun, 8 Jan 2023 20:05:10 +0530 Subject: [PATCH] replaced older IM and circumvention charts (#785) Signed-off-by: GermaVinsmoke Co-authored-by: Maja Komel --- components/{network => }/Chart.js | 43 +-- components/aggregation/mat/XAxis.js | 2 +- components/country/ASNSelector.js | 31 -- components/country/Apps.js | 60 ++-- components/country/AppsStats.js | 77 ----- components/country/AppsStatsChart.js | 111 ------- components/country/AppsStatsCircumvention.js | 73 ----- .../country/AppsStatsCircumventionRow.js | 174 ----------- components/country/AppsStatsRow.js | 179 ----------- .../country/ConfirmedBlockedCategory.js | 123 ++++++++ components/country/CountryContext.js | 18 -- components/country/NetworkProperties.js | 162 ---------- components/country/NetworkStats.js | 91 ------ components/country/Overview.js | 6 - components/country/OverviewCharts.js | 3 +- components/country/PageNavMenu.js | 3 - components/country/PeriodFilter.js | 17 -- components/country/URLChart.js | 287 ------------------ components/country/WebsiteChartLoader.js | 87 ------ components/country/Websites.js | 119 +++----- components/country/WebsitesCharts.js | 165 ---------- components/network/Form.js | 6 +- cypress/support/e2e.js | 10 + pages/country/[countryCode].js | 74 +++-- pages/network/[asn].js | 71 +++-- 25 files changed, 337 insertions(+), 1655 deletions(-) rename components/{network => }/Chart.js (65%) delete mode 100644 components/country/ASNSelector.js delete mode 100644 components/country/AppsStats.js delete mode 100644 components/country/AppsStatsChart.js delete mode 100644 components/country/AppsStatsCircumvention.js delete mode 100644 components/country/AppsStatsCircumventionRow.js delete mode 100644 components/country/AppsStatsRow.js create mode 100644 components/country/ConfirmedBlockedCategory.js delete mode 100644 components/country/NetworkProperties.js delete mode 100644 components/country/NetworkStats.js delete mode 100644 components/country/PeriodFilter.js delete mode 100644 components/country/URLChart.js delete mode 100644 components/country/WebsiteChartLoader.js delete mode 100644 components/country/WebsitesCharts.js diff --git a/components/network/Chart.js b/components/Chart.js similarity index 65% rename from components/network/Chart.js rename to components/Chart.js index b517d15d2..847f781cf 100644 --- a/components/network/Chart.js +++ b/components/Chart.js @@ -1,6 +1,6 @@ import React, { useMemo } from 'react' -import { useIntl } from 'react-intl' import { useRouter } from 'next/router' +import { FormattedMessage } from 'react-intl' import { Heading, Box, Flex } from 'ooni-components' import useSWR from 'swr' import GridChart, { prepareDataForGridChart } from 'components/aggregation/mat/GridChart' @@ -14,33 +14,16 @@ const swrOptions = { } const Chart = React.memo(function Chart({testName, testGroup = null, title, queryParams = {}}) { - const intl = useIntl() - const router = useRouter() - const { query: {since, until, asn} } = router - const name = testName || testGroup.name - const params = useMemo(() => ({ - ...queryParams, - axis_x: 'measurement_start_day' - }), [queryParams]) - - const query = useMemo(() => ({ - ...params, - probe_asn: asn, - since: since, - until: until, - ...testName && {test_name: testName} - }), [since, until, asn, params, testName]) - const apiQuery = useMemo(() => { - const qs = new URLSearchParams(query).toString() + const qs = new URLSearchParams(queryParams).toString() return qs - }, [query]) + }, [queryParams]) const { data, error } = useSWR( - testGroup ? { query: apiQuery, - testNames: testGroup.tests, + testGroup ? { query: apiQuery, + testNames: testGroup.tests, groupKey: name } : apiQuery, testGroup ? MATMultipleFetcher : MATFetcher, @@ -51,23 +34,23 @@ const Chart = React.memo(function Chart({testName, testGroup = null, title, quer return [null, 0] } let chartData = testGroup ? data : data.data - const graphQuery = testGroup ? {...query, axis_y: name} : query - const [reshapedData, rowKeys, rowLabels] = prepareDataForGridChart(chartData, graphQuery, intl.locale) + const graphQuery = testGroup ? {...queryParams, axis_y: name} : queryParams + const [reshapedData, rowKeys, rowLabels] = prepareDataForGridChart(chartData, graphQuery) return [reshapedData, rowKeys, rowLabels] - }, [data, query, name, testGroup, intl]) + }, [data, queryParams, name, testGroup]) const headerOptions = { probe_cc: false, subtitle: false } return ( - + {title} {(!chartData && !error) ? ( -
{intl.formatMessage({id: 'General.Loading'})}
+ ) : ( chartData === null || chartData.length === 0 ? ( - {intl.formatMessage({id: 'General.NoData'})} + ) : (
- {intl.formatMessage({id: 'General.Error'})}: {error.message} + Error: {error.message} {JSON.stringify(error, null, 2)} @@ -95,4 +78,4 @@ const Chart = React.memo(function Chart({testName, testGroup = null, title, quer ) }) -export default Chart \ No newline at end of file +export default Chart diff --git a/components/aggregation/mat/XAxis.js b/components/aggregation/mat/XAxis.js index 36b6ac095..0b38ce8eb 100644 --- a/components/aggregation/mat/XAxis.js +++ b/components/aggregation/mat/XAxis.js @@ -45,4 +45,4 @@ export const XAxis = ({ data }) => { ) -} \ No newline at end of file +} diff --git a/components/country/ASNSelector.js b/components/country/ASNSelector.js deleted file mode 100644 index d318ba2de..000000000 --- a/components/country/ASNSelector.js +++ /dev/null @@ -1,31 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import { Select } from 'ooni-components' -import styled from 'styled-components' - -const StyledSelect = styled(Select)` - font-family: 'Fira Sans'; -` - -const ASNSelector = ({ networks, onNetworkChange, selectedNetwork }) => ( - onNetworkChange(e.target.value)} defaultValue={selectedNetwork}> - { - networks.map((network, index) => ( - - )) - } - -) - -ASNSelector.propTypes = { - networks: PropTypes.arrayOf(PropTypes.shape({ - probe_asn: PropTypes.number, - count: PropTypes.number - })), - onNetworkChange: PropTypes.func, - selectedNetwork: PropTypes.number -} - -export default ASNSelector diff --git a/components/country/Apps.js b/components/country/Apps.js index 16c446692..04368d9c6 100644 --- a/components/country/Apps.js +++ b/components/country/Apps.js @@ -1,13 +1,50 @@ -import React from 'react' -import { FormattedMessage } from 'react-intl' +import { useMemo } from 'react' +import { FormattedMessage, useIntl } from 'react-intl' import { Text } from 'ooni-components' import SectionHeader from './SectionHeader' import { SimpleBox } from './boxes' -// import PeriodFilter from './PeriodFilter' -import AppsStatsGroup from './AppsStats' -import AppsStatsCircumvention from './AppsStatsCircumvention' +import Chart from 'components/Chart' import FormattedMarkdown from '../FormattedMarkdown' +import { useRouter } from 'next/router' + +const messagingTestNames = ['signal', 'telegram', 'whatsapp', 'facebook_messenger'] +const circumventionTestNames = ['vanilla_tor', 'psiphon', 'tor', 'torsf'] + +const ChartsContainer = () => { + const intl = useIntl() + const router = useRouter() + const { query: { since, until, countryCode } } = router + + const queryMessagingApps = useMemo(() => ({ + axis_x: 'measurement_start_day', + probe_cc: countryCode, + since, + until, + }), [countryCode, since, until]) + + const queryCircumventionTools = useMemo(() => ({ + axis_x: 'measurement_start_day', + probe_cc: countryCode, + since, + until, + }), [countryCode, since, until]) + + return ( + <> + + + + ) +} const AppsSection = () => ( <> @@ -15,24 +52,13 @@ const AppsSection = () => ( - {/* - - */} - {/* App-wise graphs */} - } - testGroup='im' - /> - {} - testGroup='circumvention' - />} + ) diff --git a/components/country/AppsStats.js b/components/country/AppsStats.js deleted file mode 100644 index 60578ac65..000000000 --- a/components/country/AppsStats.js +++ /dev/null @@ -1,77 +0,0 @@ -import React from 'react' -import { Box, Heading, Text } from 'ooni-components' -import styled from 'styled-components' -import axios from 'axios' -import { FormattedMessage } from 'react-intl' - -import { inCountry } from './CountryContext' -import AppsStatRow from './AppsStatsRow' -import { AppSectionLoader } from './WebsiteChartLoader' - -const AppGroupHeading = styled(Box)` - border: 1px solid ${props => props.theme.colors.gray3}; - border-left: 12px solid ${props => props.theme.colors.cyan6}; -` - -const defaultState = { - data: null, - fetching: true -} - -class AppsStatsGroup extends React.Component { - constructor(props) { - super(props) - this.state = { - fetching: false, - ...defaultState - } - } - - componentDidMount() { - this.fetchIMNetworks() - } - - async fetchIMNetworks() { - const { countryCode } = this.props - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/im_networks', { - params: { - probe_cc: countryCode - } - }) - - this.setState({ - data: result.data, - fetching: false - }) - } - - render() { - const { title } = this.props - const { data, fetching } = this.state - if (fetching) { - return ( - - ) - } - return ( - - - {title} - - {data && Object.keys(data).length === 0 && - - - - - - } - {Object.keys(data).map((im, index) => ( - - ))} - - ) - } -} - -export default inCountry(AppsStatsGroup) diff --git a/components/country/AppsStatsChart.js b/components/country/AppsStatsChart.js deleted file mode 100644 index a846d6a83..000000000 --- a/components/country/AppsStatsChart.js +++ /dev/null @@ -1,111 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import axios from 'axios' -import { - VictoryChart, - VictoryBar, - VictoryAxis, - VictoryVoronoiContainer -} from 'victory' -import { theme } from 'ooni-components' - -import { inCountry } from './CountryContext' -import Tooltip from './Tooltip' -import { AppsChartLoader } from './WebsiteChartLoader' - -class AppsStatChart extends React.Component { - constructor(props) { - super(props) - this.state = { - data: null, - fetching: true - } - } - - componentDidMount() { - this.fetchAppNetworkStats() - } - - async fetchAppNetworkStats() { - const { countryCode, app, asn } = this.props - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/im_stats', { - params: { - probe_cc: countryCode, - probe_asn: asn, - test_name: app - } - }) - - this.setState({ - data: result.data.results, - fetching: false - }) - } - - render() { - const { data, fetching } = this.state - - if (fetching) { - return () - } - - const yMax = data.reduce((max, item) => ( - (item.total_count > max) ? item.total_count : max - ), 0) - - return ( - <> - - } - > - {}} - /> - yMax} - style={{ - data: { - fill: theme.colors.gray3, - } - }} - /> - { - let s = `${new Date(d.test_day).toLocaleDateString()}` - s += `\n${d.total_count} Total` - return s - }} - labelComponent={} - /> - - - ) - } -} - -export default inCountry(AppsStatChart) diff --git a/components/country/AppsStatsCircumvention.js b/components/country/AppsStatsCircumvention.js deleted file mode 100644 index 2e191f07a..000000000 --- a/components/country/AppsStatsCircumvention.js +++ /dev/null @@ -1,73 +0,0 @@ -import React from 'react' -import { Box, Heading } from 'ooni-components' -import styled from 'styled-components' -import axios from 'axios' - -import { inCountry } from './CountryContext' -import AppsStatsRowCircumvention from './AppsStatsCircumventionRow' -import { AppSectionLoader } from './WebsiteChartLoader' - -const AppGroupHeading = styled(Box)` - border: 1px solid ${props => props.theme.colors.gray3}; - border-left: 12px solid ${props => props.theme.colors.cyan6}; -` - -const defaultState = { - data: null, - fetching: true -} - -class AppsStatsCircumvention extends React.Component { - constructor(props) { - super(props) - this.state = { - fetching: false, - ...defaultState - } - } - - componentDidMount() { - this.fetchCircumventionStats() - } - - async fetchCircumventionStats() { - const { countryCode } = this.props - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/vanilla_tor_stats', { - params: { - probe_cc: countryCode - } - }) - - this.setState({ - data: result.data, - fetching: false - }) - } - - static getDerivedStateFromprops() { - return defaultState - } - - render() { - const { title } = this.props - const { data, fetching } = this.state - - if (fetching) { - return ( - - ) - } - - return ( - - - {title} - - - - ) - } -} - -export default inCountry(AppsStatsCircumvention) diff --git a/components/country/AppsStatsCircumventionRow.js b/components/country/AppsStatsCircumventionRow.js deleted file mode 100644 index 0181dd7c6..000000000 --- a/components/country/AppsStatsCircumventionRow.js +++ /dev/null @@ -1,174 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import { FormattedMessage } from 'react-intl' -import { Flex, Box, Link, Text, theme } from 'ooni-components' -import styled from 'styled-components' -import { - NettestVanillaTor -} from 'ooni-components/dist/icons' -import dayjs from 'services/dayjs' - -import { testNames } from '../test-info' -import { CountryContext } from './CountryContext' -import { CollapseTrigger } from '../CollapseTrigger' - -const NETWORK_STATS_PER_PAGE = 4 - -const StyledRow = styled(Box)` - border: 1px solid ${props => props.theme.colors.gray3}; -` - -const NetworkRow = ({ asn, data }) => ( - - - - AS{ asn } - - - {data.last_tested} Last tested - - - {data.failure_count} Not OK - - - {data.success_count} OK - - - {data.total_count} Total - - - - - {data.test_runtime_avg} Runtime avg - - - {data.test_runtime_max} Runtime max - - - {data.test_runtime_min} Runtime min - - - - - -) - -class AppsStatsCircumventionRow extends React.Component { - constructor(props) { - super(props) - this.state = { - minimized: true, - visibleNetworks: 0 - } - this.toggleMinimize = this.toggleMinimize.bind(this) - this.showMore = this.showMore.bind(this) - } - - toggleMinimize() { - const { totalNetworks } = this.state - const networksToShow = Math.min(totalNetworks, NETWORK_STATS_PER_PAGE) - this.setState((state) => ({ - minimized: !state.minimized, - visibleNetworks: state.minimized ? networksToShow : 0 - })) - } - - showMore() { - // Ensure not to add more rows than available - const { visibleNetworks, totalNetworks } = this.state - - let rowsToAdd = NETWORK_STATS_PER_PAGE - - if (visibleNetworks + NETWORK_STATS_PER_PAGE > totalNetworks) { - rowsToAdd = totalNetworks - visibleNetworks - } - - this.setState((state) => ({ - visibleNetworks: state.visibleNetworks + rowsToAdd - })) - } - - - static getDerivedStateFromProps(props, state) { - const totalNetworks = props.data.networks.length - return { - totalNetworks, - ...state - } - } - - renderCharts() { - const { data } = this.props - const { totalNetworks, visibleNetworks } = this.state - const networks = data.networks - const content = [] - - for (let i = 0; i < networks.length && i < visibleNetworks ; i ++) { - content.push() - } - - return ( - <> - - {content} - - {(visibleNetworks < totalNetworks) && - - { - e.preventDefault() - this.showMore() - }}> - - - - } - - ) - } - - render () { - const { data } = this.props - const { minimized, totalNetworks } = this.state - - return ( - - - - - - - Vanilla Tor - - - {data.networks.length === 0 && - - - - } - {data.networks.length > 0 && `${data.networks.length} Networks Tested`} - - {totalNetworks > 0 && - <> - - - {' '} - {dayjs.utc(data.last_tested).fromNow()} - - - - - - } - - {!minimized && this.renderCharts()} - - ) - } -} - -export default AppsStatsCircumventionRow diff --git a/components/country/AppsStatsRow.js b/components/country/AppsStatsRow.js deleted file mode 100644 index 6acc04626..000000000 --- a/components/country/AppsStatsRow.js +++ /dev/null @@ -1,179 +0,0 @@ -import React, { useContext } from 'react' -import PropTypes from 'prop-types' -import { FormattedMessage } from 'react-intl' -import { Flex, Box, Link, theme } from 'ooni-components' -import styled from 'styled-components' -import { - NettestWhatsApp, - NettestTelegram, - NettestFacebookMessenger -} from 'ooni-components/dist/icons' -import dayjs from 'services/dayjs' - -import { testNames } from '../test-info' -import AppsStatChart from './AppsStatsChart' -import { CountryContext } from './CountryContext' -import { CollapseTrigger } from '../CollapseTrigger' - -const NETWORK_STATS_PER_PAGE = 4 - -const AppIcon = ({ app, size }) => { - switch(app) { - case 'whatsapp': - return - case 'telegram': - return - case 'facebook_messenger': - return - default: - return - } -} - -const StyledRow = styled(Box)` - border: 1px solid ${props => props.theme.colors.gray3}; -` - -const NetworkRow = ({ asn, app }) => { - const { countryCode } = useContext(CountryContext) - const until = dayjs.utc().add(1, 'day').format('YYYY-MM-DD') - const since = dayjs.utc().subtract(30, 'day').format('YYYY-MM-DD') - - const linkToMeasurements = `/search?probe_cc=${countryCode}&probe_asn=AS${asn}&test_name=${app}&since=${since}&until=${until}` - - return ( - - - - - - - AS{ asn } - - - - - - - - - - - - - - - ) -} - -class AppsStatRow extends React.Component { - constructor(props) { - super(props) - this.state = { - minimized: true, - visibleNetworks: 0 - } - this.toggleMinimize = this.toggleMinimize.bind(this) - this.showMore = this.showMore.bind(this) - } - - toggleMinimize() { - const { totalNetworks } = this.state - const networksToShow = Math.min(totalNetworks, NETWORK_STATS_PER_PAGE) - this.setState((state) => ({ - minimized: !state.minimized, - visibleNetworks: state.minimized ? networksToShow : 0 - })) - } - - showMore() { - // Ensure not to add more rows than available - const { visibleNetworks, totalNetworks } = this.state - - let rowsToAdd = NETWORK_STATS_PER_PAGE - - if (visibleNetworks + NETWORK_STATS_PER_PAGE > totalNetworks) { - rowsToAdd = totalNetworks - visibleNetworks - } - - this.setState((state) => ({ - visibleNetworks: state.visibleNetworks + rowsToAdd - })) - } - - - static getDerivedStateFromProps(props, state) { - const totalNetworks = props.data.anomaly_networks.length + props.data.ok_networks.length - return { - totalNetworks, - ...state - } - } - - renderCharts() { - const { data, app } = this.props - const { totalNetworks, visibleNetworks } = this.state - const networks = [...data.anomaly_networks, ...data.ok_networks] - const content = [] - - for (let i = 0; i < networks.length && i < visibleNetworks ; i ++) { - content.push() - } - - return ( - <> - - {content} - - {(visibleNetworks < totalNetworks) && - - {e.preventDefault(); this.showMore()}}> - - - - } - - ) - } - - render () { - const { app, data } = this.props - const { minimized } = this.state - return ( - - - - - - - {testNames[app].name} - - - {`${data.anomaly_networks.length + data.ok_networks.length} Networks Tested`} - - - - {' '} - {dayjs.utc(data.last_tested).fromNow()} - - - - - - {!minimized && this.renderCharts()} - - ) - } -} - -export default AppsStatRow diff --git a/components/country/ConfirmedBlockedCategory.js b/components/country/ConfirmedBlockedCategory.js new file mode 100644 index 000000000..1458335be --- /dev/null +++ b/components/country/ConfirmedBlockedCategory.js @@ -0,0 +1,123 @@ +import React, { useMemo } from 'react' +import { useRouter } from 'next/router' +import { FormattedMessage } from 'react-intl' +import { Heading, Box, Flex, Text, theme } from 'ooni-components' +import useSWR from 'swr' +import { DetailsBox } from 'components/measurement/DetailsBox' +import { MATFetcher } from 'services/fetchers' +import * as icons from 'ooni-components/dist/icons' +import Badge from 'components/Badge' +import { getCategoryCodesMap } from 'components/utils/categoryCodes' + +const swrOptions = { + revalidateOnFocus: false, + dedupingInterval: 10 * 60 * 1000, +} + +const ConfirmedBlockedCategory = React.memo(function Chart({testName, title, queryParams = {}}) { + const router = useRouter() + const { query: { countryCode } } = router + + const categoryCodeMap = getCategoryCodesMap() + + const params = useMemo(() => ({ + ...queryParams, + axis_x: 'category_code' + }), [queryParams]) + + const query = useMemo(() => ({ + ...params, + probe_cc: countryCode, + ...testName && {test_name: testName} + }), [countryCode, params, testName]) + + const apiQuery = useMemo(() => { + const qs = new URLSearchParams(query).toString() + return qs + }, [query]) + + const prepareDataForBadge = (categoriesData) => { + return categoriesData.filter(category => category.confirmed_count > 0) + } + + const { data, error } = useSWR( + apiQuery, + MATFetcher, + swrOptions + ) + + const blockedCategoriesData = useMemo(() => { + if (!data) { + return null + } + + const categoriesData = prepareDataForBadge(data.data) + + return categoriesData + }, [data]) + + return ( + + {title} + + {(!blockedCategoriesData && !error) ? ( +
Loading ...
+ ) : ( + blockedCategoriesData === null || blockedCategoriesData.length === 0 ? ( + + ) : ( + + {blockedCategoriesData && blockedCategoriesData.map(category => ( + + ))} + + ) + )} +
+ {error && + +
+ Error: {error.message} + + {JSON.stringify(error, null, 2)} + +
+ }/> + } + +
+ ) +}) + +const CategoryBadge = ({ categoryCode, categoryCodeMap, confirmedCount }) => { + const categoryDesc = categoryCodeMap.get(categoryCode) + const CategoryIcon = icons[`CategoryCode${categoryCode}`] + + if (categoryDesc === undefined && confirmedCount === 0) { + return null + } + + if (CategoryIcon === undefined) { + return null + } + + return ( + + + + + + + + + + + ) +} + +export default ConfirmedBlockedCategory diff --git a/components/country/CountryContext.js b/components/country/CountryContext.js index 7c501a794..21c8ac17c 100644 --- a/components/country/CountryContext.js +++ b/components/country/CountryContext.js @@ -25,24 +25,6 @@ CountryContextProvider.propTypes = { children: PropTypes.any } - -/* HoC to inject Country context into wrapped components */ -export const inCountry = (WrappedComponent) => { - return function InjectCountry(props) { - return ( - - {({countryCode, countryName}) => ( - - )} - - ) - } -} - /* Custom Hook to use CountryContext */ export const useCountry = () => { return useContext(CountryContext) diff --git a/components/country/NetworkProperties.js b/components/country/NetworkProperties.js deleted file mode 100644 index 9c0137d12..000000000 --- a/components/country/NetworkProperties.js +++ /dev/null @@ -1,162 +0,0 @@ -import React from 'react' -import { FormattedMessage } from 'react-intl' -import { Flex, Box, Heading, Text, Link } from 'ooni-components' -import axios from 'axios' - -import SectionHeader from './SectionHeader' -import { SimpleBox } from './boxes' -// import PeriodFilter from './PeriodFilter' -import NetworkStats from './NetworkStats' -import SpinLoader from '../vendor/SpinLoader' -import FormattedMarkdown from '../FormattedMarkdown' - -const NETWORK_STATS_PER_PAGE = 4 - -class NetworkPropertiesSection extends React.Component { - constructor(props) { - super(props) - this.state = { - fetching: true, - visibleNetworks: 0, - data: [], - currentPage: 0 - } - this.showMoreNetworks = this.showMoreNetworks.bind(this) - } - - showMoreNetworks() { - // Ensure not to add more rows than available - const { visibleNetworks, totalNetworks } = this.state - - let rowsToAdd = NETWORK_STATS_PER_PAGE - - if (visibleNetworks + NETWORK_STATS_PER_PAGE > totalNetworks) { - rowsToAdd = totalNetworks - visibleNetworks - } - - this.setState((state) => ({ - visibleNetworks: state.visibleNetworks + rowsToAdd - })) - } - - componentDidMount() { - this.fetchNetworkStats() - } - - componentDidUpdate(prevProps, prevState) { - const { data, totalNetworks } = this.state - if (prevState.data.length === data.length && data.length < totalNetworks) { - this.fetchNetworkStats() - } - } - - async fetchNetworkStats() { - const { countryCode } = this.props - const { currentPage } = this.state - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/network_stats', { - params: { - probe_cc: countryCode, - limit: NETWORK_STATS_PER_PAGE, - offset: (currentPage > 0 ? currentPage : 0) * NETWORK_STATS_PER_PAGE - } - }) - - this.setState((state) => ({ - data: [...state.data, ...result.data.results], - totalNetworks: result.data.metadata.total_count, - currentPage: result.data.metadata.current_page, - fetching: false, - visibleNetworks: state.data.length + result.data.results.length - })) - } - - renderStats() { - const { fetching, visibleNetworks, totalNetworks, data } = this.state - - if (fetching) { - return () - } - - if (data.length === 0) { - return ( - - - - - - ) - } - - const content = [] - - for ( let i = 0; i < data.length && i < visibleNetworks; i++) { - content.push( - - ) - } - return ( - <> - {content} - {(visibleNetworks < totalNetworks) && - - { - e.preventDefault() - this.showMoreNetworks() - }}> - - - - } - - ) - } - - render() { - return ( - <> - - - - - {/* - {}} /> - */} - - - - - - - {/* Country Level Summary - - - - - - - - - - - - */} - {/* Network-wise infoboxes */} - - {this.renderStats()} - - - - ) - } -} - -export default NetworkPropertiesSection diff --git a/components/country/NetworkStats.js b/components/country/NetworkStats.js deleted file mode 100644 index 8aec06c84..000000000 --- a/components/country/NetworkStats.js +++ /dev/null @@ -1,91 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import { Flex, Box, theme } from 'ooni-components' -import { - NettestGroupMiddleBoxes, -} from 'ooni-components/dist/icons' -import { MdFileDownload, MdFileUpload } from 'react-icons/md' -import { FormattedMessage } from 'react-intl' -import styled from 'styled-components' - -import { testGroups } from '../test-info' - - -const BorderedBox = styled(Flex)` - border: 1px solid ${props => props.theme.colors.gray3}; - padding: 12px; -` - -const StyledStat = styled(Flex)` - font-size: 28px; - font-weight: 300; - line-height: 40px; -` - -const StatBox = ({ - label = 'Average Download', - value = '50 Mbit/s', - ...props -}) => ( - - - {value} - - - {label} - - -) - -const NetworkStats = ({ - asn, - asnName, - avgDownload, - avgUpload, - avgPing, - middleboxes -}) => ( - <> - - AS{asn} - {asnName} - - - } - value={( - - - {avgDownload} - - - )} - /> - } - value={( - - - {avgUpload} - - - )} - /> - } - value={( - - {avgPing} - - - )} - /> - } - value={} - /> - - -) - -export default NetworkStats diff --git a/components/country/Overview.js b/components/country/Overview.js index da2f6728f..5b2b9f226 100644 --- a/components/country/Overview.js +++ b/components/country/Overview.js @@ -7,12 +7,6 @@ import { BoxWithTitle } from './boxes' import TestsByGroup from './OverviewCharts' import FormattedMarkdown from '../FormattedMarkdown' import { useCountry } from './CountryContext' -import { - NettestGroupWebsites, - NettestGroupInstantMessaging, - NettestGroupMiddleBoxes, -} from 'ooni-components/dist/icons' - const NwInterferenceStatus = styled(Box)` color: ${props => props.color || props.theme.colors.gray5}; diff --git a/components/country/OverviewCharts.js b/components/country/OverviewCharts.js index 97a18f4ff..00136942e 100644 --- a/components/country/OverviewCharts.js +++ b/components/country/OverviewCharts.js @@ -1,6 +1,6 @@ import React from 'react' import styled from 'styled-components' -import { Heading, Button, Flex, Box, Text, theme } from 'ooni-components' +import { Flex, Box, theme } from 'ooni-components' import { VictoryChart, VictoryBar, @@ -11,7 +11,6 @@ import { VictoryVoronoiContainer } from 'victory' import { FormattedMessage, injectIntl } from 'react-intl' -import NLink from 'next/link' import Tooltip from './Tooltip' import VictoryTheme from '../VictoryTheme' diff --git a/components/country/PageNavMenu.js b/components/country/PageNavMenu.js index c9a91de64..8f4cdcba3 100644 --- a/components/country/PageNavMenu.js +++ b/components/country/PageNavMenu.js @@ -55,9 +55,6 @@ const PageNavMenu = ({ countryCode }) => { - - - } diff --git a/components/country/PeriodFilter.js b/components/country/PeriodFilter.js deleted file mode 100644 index 25ec9b7bc..000000000 --- a/components/country/PeriodFilter.js +++ /dev/null @@ -1,17 +0,0 @@ -import React from 'react' -import { Flex, Select } from 'ooni-components' -import { FormattedMessage } from 'react-intl' - -const PeriodFilter = () => ( - - {(msg) => } - - -) - -export default PeriodFilter diff --git a/components/country/URLChart.js b/components/country/URLChart.js deleted file mode 100644 index fca084f7a..000000000 --- a/components/country/URLChart.js +++ /dev/null @@ -1,287 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import axios from 'axios' -import { Flex, Box, Link } from 'ooni-components' -import { - VictoryChart, - VictoryStack, - VictoryBar, - VictoryAxis, - VictoryVoronoiContainer -} from 'victory' -import { FormattedMessage } from 'react-intl' -import styled from 'styled-components' -import dayjs from 'services/dayjs' - -import { - colorNormal, - colorError, - colorConfirmed, - colorAnomaly, - colorEmpty -} from '../colors' - -import Tooltip from './Tooltip' -import { WebsiteChartLoader } from './WebsiteChartLoader' - -const Circle = styled.div` - position: relative; - top: 0; - right: 0; - background-color: ${props => props.theme.colors.gray3}; - padding: 6px; - border-radius: 50%; - :hover { - background-color: ${props => props.theme.colors.gray4}; - } -` -/* CSS Triangle from CSS-Tricks: https://css-tricks.com/snippets/css/css-triangle/#article-header-id-1 */ -// TOOD: Improve toggle using transforms and animation -const Triangle = styled.div` - width: 0; - height: 0; - border-left: 6px solid transparent; - border-right: 6px solid transparent; - border-top: ${props => props.down ? '12px solid ' + props.theme.colors.gray7 : 'none'}; - border-bottom: ${props => !props.down ? '12px solid ' + props.theme.colors.gray7 : 'none'}; -` - -const WrappedText = styled.div` - overflow-wrap: break-word; - min-height: 2em; -` - -const TruncatedURL = ({ url }) => { - const MAX_URL_LENGTH = 60 - try { - const urlObj = new URL(url) - const domain = urlObj.origin - const path = urlObj.pathname - let endOfPath = path.split('/').pop() - if (domain.length + endOfPath.length > MAX_URL_LENGTH) { - endOfPath = endOfPath.substring(0, MAX_URL_LENGTH - domain.length) + '...' - } - return ( - - - {`${domain}${endOfPath.length > 5 ? '/...' : ''}/${endOfPath}`} - - - ) - } catch (e) { - return ( - - {url} - - ) - } -} - -const StyledChartRow = styled(Flex)` - border: 1px solid ${props => props.theme.colors.gray3}; - border-radius: 5px; -` - -const ToggleMinimizeButton = ({ minimized, onToggle }) => ( - -) - -const defaultState = { - data: null, - minimized: true, - fetching: true -} - -class URLChart extends React.Component { - constructor(props) { - super(props) - this.state = defaultState - this.onToggleMinimize = this.onToggleMinimize.bind(this) - } - - onToggleMinimize() { - this.setState((state) => ({ - minimized: !state.minimized - })) - } - - componentDidMount() { - this.fetchURLChartData() - } - - componentDidUpdate(prevProps, prevState) { - if(this.state.data === null) { - this.fetchURLChartData() - } - } - - async fetchURLChartData() { - const { metadata, network, countryCode } = this.props - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/website_stats', { - params: { - probe_cc: countryCode, - probe_asn: network, - input: metadata.input - } - }) - // HACK: Temporary fix to workaround backend bug showing wrong anomaly and confirmed counts - const fixedData = result.data.results.map(d => { - d.anomaly_count = d.anomaly_count - d.confirmed_count - return d - }) - this.setState({ - data: result.data.results, - fetching: false - }) - } - - static getDerivedStateFromProps(props, state) { - if (props.metadata.input !== state.prevTestUrl) { - return { - prevTestUrl: props.metadata.input, - ...defaultState - } - } - return null - } - - render() { - const { metadata, countryCode, network } = this.props - const { data, minimized, fetching } = this.state - const dataColorMap = { - total_count: colorNormal, - confirmed_count: colorConfirmed, - anomaly_count: colorAnomaly, - failure_count: colorError, - empty: colorEmpty - } - - if (fetching) { - return ( - - ) - } - - const until = dayjs.utc().add(1, 'day').format('YYYY-MM-DD') - const since30days = dayjs.utc().subtract(30, 'days').format('YYYY-MM-DD') - - const yMax = data.reduce((max, item) => ( - (item.total_count > max) ? item.total_count : max - ), 0) - - const domainToExplore = new URL(metadata.input).hostname - - return ( - - - - - - - - - {/* TODO: Show percentages - - - - - - */} - - - { - data && - - } - > - {}} - /> - yMax} - style={{ - data: { - fill: dataColorMap.empty - } - }} - /> - - { - let s = `${new Date(d.test_day).toLocaleDateString()}` - if (d.confirmed_count > 0) { - s += `\n${d.confirmed_count} Confirmed` - } - if (d.anomaly_count > 0) { - s += `\n${d.anomaly_count} Anomalies` - } - if (d.failure_count > 0) { - s += `\n${d.failure_count} Failures` - } - s += `\n${d.total_count} Total` - return s - }} - labelComponent={} - data={data} - x='test_day' - y={(d) => (d.total_count - d.confirmed_count - d.anomaly_count - d.failure_count)} - style={{ - data: { - fill: dataColorMap.total_count, - } - }} - /> - { - ['confirmed_count', 'anomaly_count', 'failure_count'].map((type, index) => ( - - )) - } - - - } - - - - {/* - - - - */} - - ) - } -} - -export default URLChart diff --git a/components/country/WebsiteChartLoader.js b/components/country/WebsiteChartLoader.js deleted file mode 100644 index 6022d15fd..000000000 --- a/components/country/WebsiteChartLoader.js +++ /dev/null @@ -1,87 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import ContentLoader from 'react-content-loader' -import { theme } from 'ooni-components' - -export const WebsiteChartLoader = (props) => { - const random = Math.floor(Math.random() * (20 - 14) + 14) - return ( - - - {Array(random).fill('').map((e, i) => ( - - )) - } - - ) -} - -export const WebsiteSectionLoader = ({ rows = 5 }) => ( - <> - {Array(rows) - .fill('') - .map((e, i) => ( - - )) - } - -) - -WebsiteSectionLoader.propTypes = { - rows: PropTypes.number -} - -export const AppsChartLoader = ({xOffset = 50, barWidth = 10, barHeight = 30, ...props}) => { - const random = Math.floor(Math.random() * (20 - 16) + 16) - return ( - - {Array(random).fill('').map((e, i) => ( - - )) - } - - ) -} - -export const AppSectionLoader = ({ rows = 1 }) => { - const Row = ({ y }) => ( - [ - , - , - , - - ] - ) - return ( - - - {Array(rows).fill('').map((e, i) => ( - - ))} - - ) -} - -AppSectionLoader.propTypes = { - rows: PropTypes.number -} diff --git a/components/country/Websites.js b/components/country/Websites.js index 51e65351e..363618007 100644 --- a/components/country/Websites.js +++ b/components/country/Websites.js @@ -1,88 +1,47 @@ -import React from 'react' -import { inCountry } from './CountryContext' -import { FormattedMessage } from 'react-intl' -import axios from 'axios' -import { Flex, Box, Heading, Text, Input } from 'ooni-components' - +import React, {useCallback, useMemo} from 'react' +import { useIntl, FormattedMessage } from 'react-intl' +import { Box, Text } from 'ooni-components' +import ChartCountry from 'components/Chart' import SectionHeader from './SectionHeader' import { SimpleBox } from './boxes' -// import PeriodFilter from './PeriodFilter' -import TestsByCategoryInNetwork from './WebsitesCharts' import FormattedMarkdown from '../FormattedMarkdown' +import ConfirmedBlockedCategory from './ConfirmedBlockedCategory' +import { useRouter } from 'next/router' -class WebsitesSection extends React.Component { - constructor(props) { - super(props) - this.state = { - noData: false, - selectedNetwork: null, - networks: [] - } - this.onNetworkChange = this.onNetworkChange.bind(this) - } - - onNetworkChange(asn) { - this.setState({ - selectedNetwork: Number(asn) - }) - } - - async componentDidMount() { - const { countryCode } = this.props - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/website_networks', { - params: { - probe_cc: countryCode - } - }) - if (result.data.results.length > 0) { - this.setState({ - networks: result.data.results, - selectedNetwork: Number(result.data.results[0].probe_asn) - }) - } else { - this.setState({ - noData: true, - networks: null - }) - } - } +const WebsitesSection = ({ countryCode }) => { + const router = useRouter() + const { query: { since, until } } = router - render () { - const { onPeriodChange, countryCode } = this.props - const { noData, selectedNetwork } = this.state - return ( - <> - - - - - - {/* - - */} - - - - - + const query = useMemo(() => ({ + axis_y: 'domain', + axis_x: 'measurement_start_day', + probe_cc: countryCode, + since, + until, + test_name: 'web_connectivity', + }), [countryCode, since, until]) - - {noData && - - - - } - - - - ) - } + return ( + <> + + + + + + + + + + + + + + + + ) } -export default inCountry(WebsitesSection) +export default WebsitesSection \ No newline at end of file diff --git a/components/country/WebsitesCharts.js b/components/country/WebsitesCharts.js deleted file mode 100644 index 005da24e0..000000000 --- a/components/country/WebsitesCharts.js +++ /dev/null @@ -1,165 +0,0 @@ -import React from 'react' -import PropTypes from 'prop-types' -import { FormattedMessage } from 'react-intl' -import { Flex, Box, Heading, Text, Link } from 'ooni-components' -import axios from 'axios' -import URLChart from './URLChart' -import ASNSelector from './ASNSelector' -import { WebsiteSectionLoader, WebsiteChartLoader } from './WebsiteChartLoader' - -const defaultState = { - resultsPerPage: 5, - testedUrlsCount: 0, - testedUrls: null, - fetching: true -} - -class TestsByCategoryInNetwork extends React.Component { - constructor(props) { - super(props) - this.state = defaultState - } - - // This is dead code now. Made so to ensure the loader - // is rendered even when is not ready to render because - // list of networks is still being fetched by the parent component. - // This prevents the jump in the layout. - // - // componentDidMount() { - // if (this.props.network !== null) { - // this.fetchUrlsInNetwork() - // } - // } - - componentDidUpdate(prevProps) { - if (prevProps.networks !== this.props.networks && this.props.networks === null) { - this.setState({ - fetching: false - }) - } else if (this.state.testedUrls === null && this.props.network !== null) { - this.fetchUrlsInNetwork() - } - } - - async fetchUrlsInNetwork() { - const { network, countryCode } = this.props - const { resultsPerPage, currentPage } = this.state - const client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line - const result = await client.get('/api/_/website_urls', { - params: { - probe_cc: countryCode, - probe_asn: network, - limit: resultsPerPage, - offset: (currentPage > 0 ? (currentPage - 1) : 0) * resultsPerPage - } - }) - this.setState({ - testedUrlsCount: result.data.metadata.total_count, - testedUrls: result.data.results, - currentPage: result.data.metadata.current_page, - fetching: false - }) - } - - prevPage() { - this.setState((state) => ({ - currentPage: state.currentPage - 1 - })) - } - - nextPage() { - this.setState((state) => ({ - currentPage: state.currentPage + 1 - })) - } - - static getDerivedStateFromProps(props, state) { - if (props.network !== state.prevNetwork) { - return { - prevNetwork: props.network, - currentPage: 1, - prevPage: 1, - ...defaultState - } - } - if (state.currentPage !== state.prevPage) { - return { - prevPage: state.currentPage || 1, - ...defaultState - } - } - return null - } - - render() { - const { network, countryCode, networks, onNetworkChange } = this.props - const { testedUrlsCount, testedUrls, currentPage, resultsPerPage, fetching } = this.state - - const renderLoader = () => ( - - ) - - return ( - <> - {/* */} - {/* {'AS'+network} - }} - /> */} - {/* Category Selection */} - - {(network !== null && networks !== null) ? - <> - - - - - {testedUrlsCount} - - - : - - } - {/* Results per page dropdown - - - - */} - - {/* Hide until API is available - - {(msg) => ( - - )} - - */} - {/* URL-wise barcharts Start */} - {fetching && renderLoader()} - {(!fetching && testedUrls && testedUrls.length === 0) && - - - - - - } - {(!fetching && testedUrls && testedUrls.length > 0) && - testedUrls.map((testedUrl, index) => ( - - ))} - {(!fetching && testedUrlsCount > 0) && - {e.preventDefault(); this.prevPage()}}>{'< '} - {currentPage} of { Math.ceil(testedUrlsCount / resultsPerPage)} pages - {e.preventDefault(); this.nextPage()}}>{' >'} - } - {/* URL-wise barcharts End */} - - ) - } -} - -export default TestsByCategoryInNetwork diff --git a/components/network/Form.js b/components/network/Form.js index 2db164899..aa5c2323c 100644 --- a/components/network/Form.js +++ b/components/network/Form.js @@ -16,7 +16,7 @@ const defaultDefaultValues = { until: tomorrow, } -const Form = ({ onChange, query }) => { +const Form = ({ onSubmit, query }) => { const intl = useIntl() const query2formValues = (query) => { @@ -49,11 +49,11 @@ const Form = ({ onChange, query }) => { const submit = (e) => { e.preventDefault() - onChange({since, until}) + onSubmit({since, until}) } return ( -
+ diff --git a/cypress/support/e2e.js b/cypress/support/e2e.js index d68db96df..5fae3fc7b 100644 --- a/cypress/support/e2e.js +++ b/cypress/support/e2e.js @@ -18,3 +18,13 @@ import './commands' // Alternatively you can use CommonJS syntax: // require('./commands') + +Cypress.on('uncaught:exception', (err, runnable) => { + // we expect a 3rd party library error with message 'ResizeObserver loop limit exceeded' + // and don't want to fail the test so we return false + if (err.message.includes('ResizeObserver loop limit exceeded')) { + return false + } + // we still want to ensure there are no other unexpected + // errors, so we let them fail the test +}) diff --git a/pages/country/[countryCode].js b/pages/country/[countryCode].js index c569c4c04..b6a77d7b6 100644 --- a/pages/country/[countryCode].js +++ b/pages/country/[countryCode].js @@ -1,25 +1,28 @@ /* global process */ -import React, { useCallback, useState } from 'react' +import React, { useCallback, useState, useEffect } from 'react' import axios from 'axios' +import { useRouter } from 'next/router' import { Container, Heading, Flex, Box } from 'ooni-components' import styled from 'styled-components' +import { useIntl } from 'react-intl' import { StickyContainer, Sticky } from 'react-sticky' import { getLocalisedRegionName } from '../../utils/i18nCountries' +import dayjs from 'services/dayjs' -import NavBar from '../../components/NavBar' -import Flag from '../../components/Flag' -import PageNavMenu from '../../components/country/PageNavMenu' -import Overview from '../../components/country/Overview' -import WebsitesSection from '../../components/country/Websites' -import AppsSection from '../../components/country/Apps' -// import NetworkPropertiesSection from '../../components/country/NetworkProperties' -import { CountryContextProvider } from '../../components/country/CountryContext' -import CountryHead from '../../components/country/CountryHead' -import { useIntl } from 'react-intl' +import Form from 'components/network/Form' +import NavBar from 'components/NavBar' +import Flag from 'components/Flag' +import Layout from 'components/Layout' +import PageNavMenu from 'components/country/PageNavMenu' +import Overview from 'components/country/Overview' +import WebsitesSection from 'components/country/Websites' +import AppsSection from 'components/country/Apps' +import { CountryContextProvider } from 'components/country/CountryContext' +import CountryHead from 'components/country/CountryHead' const getCountryReports = (countryCode, data) => { const reports = data.filter((article) => ( @@ -77,14 +80,30 @@ export async function getServerSideProps ({ res, query }) { } } - const Country = ({ countryCode, overviewStats, reports, ...coverageDataSSR }) => { const intl = useIntl() - const [newData, setNewData] = useState(false) const countryName = getLocalisedRegionName(countryCode, intl.locale) + const [newData, setNewData] = useState(false) + const router = useRouter() + const query = router.query + + useEffect(() => { + if (Object.keys(query).length === 1) { + const today = dayjs.utc().add(1, 'day') + const monthAgo = dayjs.utc(today).subtract(1, 'month') + const href = { + pathname: router.pathname, + query: { + since: monthAgo.format('YYYY-MM-DD'), + until: today.format('YYYY-MM-DD'), + countryCode + }, + } + router.replace(href, undefined, { shallow: true }) + } + }, []) const fetchTestCoverageData = useCallback((testGroupList) => { - console.log(testGroupList) const fetcher = async (testGroupList) => { let client = axios.create({baseURL: process.env.NEXT_PUBLIC_OONI_API}) // eslint-disable-line const result = await client.get('/api/_/test_coverage', { @@ -100,9 +119,29 @@ const Country = ({ countryCode, overviewStats, reports, ...coverageDataSSR }) => }) } fetcher(testGroupList) - + }, [countryCode, setNewData]) + // Sync page URL params with changes from form values + const onSubmit = ({ since, until }) => { + const params = { + since, + until, + } + + const href = { + pathname: router.pathname.replace('[countryCode]', countryCode), + query: params, + } + + if (query.since !== since + || query.until !== until + ) { + router.push(href, href, { shallow: true }) + } + + } + const { testCoverage, networkCoverage } = newData !== false ? newData : coverageDataSSR return ( @@ -153,7 +192,8 @@ const Country = ({ countryCode, overviewStats, reports, ...coverageDataSSR }) => fetchTestCoverageData={fetchTestCoverageData} featuredArticles={reports} /> - + + @@ -164,4 +204,4 @@ const Country = ({ countryCode, overviewStats, reports, ...coverageDataSSR }) => ) } -export default Country \ No newline at end of file +export default Country diff --git a/pages/network/[asn].js b/pages/network/[asn].js index 9c2619936..81fb67c53 100644 --- a/pages/network/[asn].js +++ b/pages/network/[asn].js @@ -1,26 +1,21 @@ -import React, { useCallback, useEffect } from 'react' +import React, { useCallback, useEffect, useMemo } from 'react' import { useRouter } from 'next/router' import axios from 'axios' -import { Container, Heading, Box, Flex, Text, Link } from 'ooni-components' +import { Container, Heading, Box, Text, Link } from 'ooni-components' import { useIntl } from 'react-intl' import NLink from 'next/link' -import styled from 'styled-components' import dayjs from 'services/dayjs' import Layout from 'components/Layout' import NavBar from 'components/NavBar' import { MetaTags } from 'components/dashboard/MetaTags' import Form from 'components/network/Form' -import Chart from 'components/network/Chart' +import Chart from 'components/Chart' import Calendar from 'components/network/Calendar' import FormattedMarkdown from 'components/FormattedMarkdown' import { FormattedMessage } from 'react-intl' import CallToActionBox from 'components/CallToActionBox' import { getLocalisedRegionName } from '../../utils/i18nCountries' -const Bold = styled.span` - font-weight: bold -` - const prepareDataForCalendar = (data) => { return data.map((r) => ({ value: r.measurement_count, @@ -34,18 +29,46 @@ const circumventionTestNames = ['psiphon', 'tor', 'torsf'] const ChartsContainer = () => { const intl = useIntl() + const router = useRouter() + const { query: { since, until, asn } } = router + + const queryWebsites = useMemo(() => ({ + axis_y: 'domain', + axis_x: 'measurement_start_day', + asn, + since, + until, + test_name: 'web_connectivity', + }), [asn, since, until]) + + const queryMessagingApps = useMemo(() => ({ + axis_x: 'measurement_start_day', + asn, + since, + until, + }), [asn, since, until]) + + const queryCircumventionTools = useMemo(() => ({ + axis_x: 'measurement_start_day', + asn, + since, + until, + }), [asn, since, until]) + return ( - <> + <> + queryParams={queryWebsites} /> + title={intl.formatMessage({id: 'Tests.Groups.Instant Messagging.Name'})} + queryParams={queryMessagingApps} /> + title={intl.formatMessage({id: 'Tests.Groups.Circumvention.Name'})} + queryParams={queryCircumventionTools} /> ) } @@ -89,7 +112,7 @@ const NetworkDashboard = ({asn, calendarData = [], measurementsTotal, countriesD const displayASN = asn.replace('AS', '') useEffect(() => { - if (Object.keys(query).length < 3) { + if (Object.keys(query).length < 3) { const today = dayjs.utc().add(1, 'day') const monthAgo = dayjs.utc(today).subtract(1, 'month') const href = { @@ -104,7 +127,7 @@ const NetworkDashboard = ({asn, calendarData = [], measurementsTotal, countriesD }, []) // Sync page URL params with changes from form values - const onChange = useCallback(({ since, until }) => { + const onSubmit = ({ since, until }) => { // since: "2022-01-02", // until: "2022-02-01", const params = { @@ -115,7 +138,7 @@ const NetworkDashboard = ({asn, calendarData = [], measurementsTotal, countriesD if (query.since !== since || query.until !== until) { router.push({ query: params }, undefined, { shallow: true }) } - }, [router, query, asn]) + } return ( <> @@ -125,17 +148,17 @@ const NetworkDashboard = ({asn, calendarData = [], measurementsTotal, countriesD AS{displayASN} {router.isReady && <> - {!!calendarData.length ? + {!!calendarData.length ? <> - + : - } - text={} + text={} /> } @@ -154,8 +177,8 @@ export const getServerSideProps = async (context) => { const measurementsTotal = await client .get(path, {params: {'probe_asn': asn}}) - .then((response)=> response?.data?.result.measurement_count) - + .then((response) => response?.data?.result.measurement_count) + const calendarData = await client.get(path, { params: { probe_asn: asn, since: dayjs.utc().subtract(10, 'year').format('YYYY-MM-DD'), @@ -168,8 +191,8 @@ export const getServerSideProps = async (context) => { axis_x: 'probe_cc' }}).then((response) => (response.data.result.map(res => ({country: res.probe_cc, measurements: res.measurement_count})))) - return { - props: { + return { + props: { asn, calendarData, measurementsTotal, @@ -186,4 +209,4 @@ export const getServerSideProps = async (context) => { } } -export default NetworkDashboard \ No newline at end of file +export default NetworkDashboard