🏗️ Migrate dnshole summary to tRPC

This commit is contained in:
Meier Lukas
2023-06-10 14:04:32 +02:00
parent 1fd93e6f49
commit 9b6ce9e1f6
4 changed files with 111 additions and 20 deletions

View File

@@ -1,10 +1,11 @@
import { z } from 'zod'; import { z } from 'zod';
import { createTRPCRouter, publicProcedure } from '../trpc'; import { findAppProperty } from '~/tools/client/app-properties';
import { getConfig } from '~/tools/config/getConfig'; import { getConfig } from '~/tools/config/getConfig';
import { AdGuard } from '~/tools/server/sdk/adGuard/adGuard'; import { AdGuard } from '~/tools/server/sdk/adGuard/adGuard';
import { ConfigAppType } from '~/types/app';
import { findAppProperty } from '~/tools/client/app-properties';
import { PiHoleClient } from '~/tools/server/sdk/pihole/piHole'; import { PiHoleClient } from '~/tools/server/sdk/pihole/piHole';
import { ConfigAppType } from '~/types/app';
import { AdStatistics } from '~/widgets/dnshole/type';
import { createTRPCRouter, publicProcedure } from '../trpc';
export const dnsHoleRouter = createTRPCRouter({ export const dnsHoleRouter = createTRPCRouter({
control: publicProcedure control: publicProcedure
@@ -32,6 +33,52 @@ export const dnsHoleRouter = createTRPCRouter({
}) })
); );
}), }),
summary: publicProcedure
.input(
z.object({
configName: z.string(),
})
)
.query(async ({ input }) => {
const config = getConfig(input.configName);
const applicableApps = config.apps.filter(
(x) => x.integration?.type && ['pihole', 'adGuardHome'].includes(x.integration?.type)
);
const result = await Promise.all(
applicableApps.map(async (app) =>
app.integration?.type === 'pihole'
? collectPiHoleSummary(app)
: collectAdGuardSummary(app)
)
);
const data = result.reduce(
(prev: AdStatistics, curr) => ({
domainsBeingBlocked: prev.domainsBeingBlocked + curr.domainsBeingBlocked,
adsBlockedToday: prev.adsBlockedToday + curr.adsBlockedToday,
dnsQueriesToday: prev.dnsQueriesToday + curr.dnsQueriesToday,
status: [...prev.status, curr.status],
adsBlockedTodayPercentage: 0,
}),
{
domainsBeingBlocked: 0,
adsBlockedToday: 0,
adsBlockedTodayPercentage: 0,
dnsQueriesToday: 0,
status: [],
}
);
//const data: AdStatistics = ;
data.adsBlockedTodayPercentage = data.adsBlockedToday / data.dnsQueriesToday;
if (Number.isNaN(data.adsBlockedTodayPercentage)) {
data.adsBlockedTodayPercentage = 0;
}
return data;
}),
}); });
const processAdGuard = async (app: ConfigAppType, enable: boolean) => { const processAdGuard = async (app: ConfigAppType, enable: boolean) => {
@@ -59,3 +106,45 @@ const processPiHole = async (app: ConfigAppType, enable: boolean) => {
await pihole.disable(); await pihole.disable();
}; };
const collectPiHoleSummary = async (app: ConfigAppType) => {
const piHole = new PiHoleClient(app.url, findAppProperty(app, 'password'));
const summary = await piHole.getSummary();
return {
domainsBeingBlocked: summary.domains_being_blocked,
adsBlockedToday: summary.ads_blocked_today,
dnsQueriesToday: summary.dns_queries_today,
status: {
status: summary.status,
appId: app.id,
},
adsBlockedTodayPercentage: summary.ads_percentage_today,
};
};
const collectAdGuardSummary = async (app: ConfigAppType) => {
const adGuard = new AdGuard(
app.url,
findAppProperty(app, 'username'),
findAppProperty(app, 'password')
);
const stats = await adGuard.getStats();
const status = await adGuard.getStatus();
const countFilteredDomains = await adGuard.getCountFilteringDomains();
const blockedQueriesToday = stats.blocked_filtering.reduce((prev, sum) => prev + sum, 0);
const queriesToday = stats.dns_queries.reduce((prev, sum) => prev + sum, 0);
return {
domainsBeingBlocked: countFilteredDomains,
adsBlockedToday: blockedQueriesToday,
dnsQueriesToday: queriesToday,
status: {
status: status.protection_enabled ? ('enabled' as const) : ('disabled' as const),
appId: app.id,
},
adsBlockedTodayPercentage: (queriesToday / blockedQueriesToday) * 100,
};
};

View File

@@ -5,10 +5,10 @@ import { useConfigContext } from '../../config/provider';
import { defineWidget } from '../helper'; import { defineWidget } from '../helper';
import { WidgetLoading } from '../loading'; import { WidgetLoading } from '../loading';
import { IWidget } from '../widgets'; import { IWidget } from '../widgets';
import { useDnsHoleSummeryQuery } from './query';
import { PiholeApiSummaryType } from './type'; import { PiholeApiSummaryType } from './type';
import { queryClient } from '../../tools/server/configurations/tanstack/queryClient.tool'; import { queryClient } from '../../tools/server/configurations/tanstack/queryClient.tool';
import { api } from '~/utils/api'; import { api } from '~/utils/api';
import { useDnsHoleSummeryQuery } from './DnsHoleSummary';
const definition = defineWidget({ const definition = defineWidget({
id: 'dns-hole-controls', id: 'dns-hole-controls',
@@ -30,7 +30,7 @@ interface DnsHoleControlsWidgetProps {
} }
function DnsHoleControlsWidgetTile({ widget }: DnsHoleControlsWidgetProps) { function DnsHoleControlsWidgetTile({ widget }: DnsHoleControlsWidgetProps) {
const { isInitialLoading, data, refetch } = useDnsHoleSummeryQuery(); const { isInitialLoading, data } = useDnsHoleSummeryQuery();
const { mutateAsync } = useDnsHoleControlMutation(); const { mutateAsync } = useDnsHoleControlMutation();
const { t } = useTranslation('common'); const { t } = useTranslation('common');

View File

@@ -1,4 +1,3 @@
import { useTranslation } from 'next-i18next';
import { Card, Center, Container, Stack, Text } from '@mantine/core'; import { Card, Center, Container, Stack, Text } from '@mantine/core';
import { import {
IconAd, IconAd,
@@ -7,11 +6,13 @@ import {
IconSearch, IconSearch,
IconWorldWww, IconWorldWww,
} from '@tabler/icons-react'; } from '@tabler/icons-react';
import { useTranslation } from 'next-i18next';
import { useConfigContext } from '~/config/provider';
import { api } from '~/utils/api';
import { formatNumber } from '../../tools/client/math';
import { defineWidget } from '../helper'; import { defineWidget } from '../helper';
import { WidgetLoading } from '../loading'; import { WidgetLoading } from '../loading';
import { IWidget } from '../widgets'; import { IWidget } from '../widgets';
import { formatNumber } from '../../tools/client/math';
import { useDnsHoleSummeryQuery } from './query';
const definition = defineWidget({ const definition = defineWidget({
id: 'dns-hole-summary', id: 'dns-hole-summary',
@@ -179,4 +180,17 @@ function DnsHoleSummaryWidgetTile({ widget }: DnsHoleSummaryWidgetProps) {
); );
} }
export const useDnsHoleSummeryQuery = () => {
const { name: configName } = useConfigContext();
return api.dnsHole.summary.useQuery(
{
configName: configName!,
},
{
refetchInterval: 3 * 60 * 1000,
}
);
};
export default definition; export default definition;

View File

@@ -1,12 +0,0 @@
import { useQuery } from '@tanstack/react-query';
import { AdStatistics } from './type';
export const useDnsHoleSummeryQuery = () =>
useQuery({
queryKey: ['dns-hole-summary'],
queryFn: async () => {
const response = await fetch('/api/modules/dns-hole/summary');
return (await response.json()) as AdStatistics;
},
refetchInterval: 3 * 60 * 1000,
});