🐛 Fix ping not reading from correct config

This commit is contained in:
Meierschlumpf
2023-07-22 18:27:24 +02:00
parent 6658c05e5a
commit 1634510070
2 changed files with 108 additions and 71 deletions

View File

@@ -3,7 +3,7 @@ import { IconCheck, IconLoader, IconX } from '@tabler/icons-react';
import Consola from 'consola'; import Consola from 'consola';
import { TargetAndTransition, Transition, motion } from 'framer-motion'; import { TargetAndTransition, Transition, motion } from 'framer-motion';
import { useTranslation } from 'next-i18next'; import { useTranslation } from 'next-i18next';
import { api } from '~/utils/api'; import { RouterOutputs, api } from '~/utils/api';
import { useConfigContext } from '../../../../config/provider'; import { useConfigContext } from '../../../../config/provider';
import { AppType } from '../../../../types/app'; import { AppType } from '../../../../types/app';
@@ -13,57 +13,19 @@ interface AppPingProps {
} }
export const AppPing = ({ app }: AppPingProps) => { export const AppPing = ({ app }: AppPingProps) => {
const { t } = useTranslation('modules/ping');
const { config } = useConfigContext(); const { config } = useConfigContext();
const active =
(config?.settings.customization.layout.enabledPing && app.network.enabledStatusChecker) ??
false;
const { data, isFetching, isError, error } = api.app.ping.useQuery(app.id, {
retry: false,
enabled: active,
select: (data) => {
const isOk = getIsOk(app, data.status);
if (isOk)
Consola.info(`Ping of app "${app.name}" (${app.url}) returned ${data.status} (Accepted)`);
else
Consola.warn(`Ping of app "${app.name}" (${app.url}) returned ${data.status} (Refused)`);
return {
status: data.status,
state: isOk ? ('online' as const) : ('down' as const),
statusText: data.statusText,
};
},
});
if (!active) return null;
const { data, isFetching, isError, error, isActive } = usePing(app);
const tooltipLabel = useTooltipLabel({isFetching, isError, data, errorMessage: error?.message})
const isOnline = isError ? false : data?.state === 'online'; const isOnline = isError ? false : data?.state === 'online';
const pulse = usePingPulse({isOnline});
if (!isActive) return null;
const disablePulse = config?.settings.customization.accessibility?.disablePingPulse ?? false;
const replaceDotWithIcon = const replaceDotWithIcon =
config?.settings.customization.accessibility?.replacePingDotsWithIcons ?? false; config?.settings.customization.accessibility?.replacePingDotsWithIcons ?? false;
const scaleAnimation = isOnline ? [1, 0.7, 1] : 1;
const animate: TargetAndTransition | undefined = disablePulse
? undefined
: {
scale: scaleAnimation,
};
const transition: Transition | undefined = disablePulse
? undefined
: {
repeat: Infinity,
duration: 2.5,
ease: 'easeInOut',
};
const label = () => {
if (isFetching) return t('states.loading');
if (isError) return error?.message;
if (data?.state === 'online') return t('states.online', { response: data?.status ?? 'N/A' });
return `${data?.statusText}: ${data?.status} (denied)`;
}
return ( return (
<motion.div <motion.div
style={{ style={{
@@ -72,17 +34,17 @@ export const AppPing = ({ app }: AppPingProps) => {
right: replaceDotWithIcon ? 8 : 20, right: replaceDotWithIcon ? 8 : 20,
zIndex: 2, zIndex: 2,
}} }}
animate={animate} animate={pulse.animate}
transition={transition} transition={pulse.transition}
> >
<Tooltip <Tooltip
withinPortal withinPortal
radius="lg" radius="lg"
label={label()} label={tooltipLabel}
> >
{config?.settings.customization.accessibility?.replacePingDotsWithIcons ? ( {replaceDotWithIcon ? (
<Box> <Box>
<AccessibleIndicatorPing isLoading={isFetching} isOnline={isOnline} /> <AccessibleIndicatorPing isFetching={isFetching} isOnline={isOnline} />
</Box> </Box>
) : ( ) : (
<Indicator <Indicator
@@ -96,27 +58,101 @@ export const AppPing = ({ app }: AppPingProps) => {
); );
}; };
const AccessibleIndicatorPing = ({ type AccessibleIndicatorPingProps = {
isLoading,
isOnline,
}: {
isOnline: boolean; isOnline: boolean;
isLoading: boolean; isFetching: boolean;
}) => { }
const AccessibleIndicatorPing = ({
isFetching,
isOnline,
}: AccessibleIndicatorPingProps) => {
if (isOnline) { if (isOnline) {
return <IconCheck color="green" />; return <IconCheck color="green" />;
} }
if (isLoading) { if (isFetching) {
return <IconLoader />; return <IconLoader />;
} }
return <IconX color="red" />; return <IconX color="red" />;
}; };
export const getIsOk = (app: AppType, status: number) => { export const isStatusOk = (app: AppType, status: number) => {
if (app.network.okStatus === undefined || app.network.statusCodes.length >= 1) { if (app.network.okStatus === undefined || app.network.statusCodes.length >= 1) {
return app.network.statusCodes.includes(status.toString()); return app.network.statusCodes.includes(status.toString());
} }
return app.network.okStatus.includes(status); return app.network.okStatus.includes(status);
}; };
type TooltipLabelProps = {
isFetching: boolean;
isError: boolean;
data: RouterOutputs['app']['ping'] | undefined;
errorMessage: string | undefined;
}
const useTooltipLabel = ({isFetching, isError, data, errorMessage}: TooltipLabelProps) => {
const { t } = useTranslation('modules/ping');
if (isFetching) return t('states.loading');
if (isError) return errorMessage;
if (data?.state === 'online') return t('states.online', { response: data?.status ?? 'N/A' });
return `${data?.statusText}: ${data?.status} (denied)`;
}
const usePing = (app: AppType) => {
const { config, name } = useConfigContext();
const isActive = (config?.settings.customization.layout.enabledPing && app.network.enabledStatusChecker) ??
false;
const queryResult = api.app.ping.useQuery({
id: app.id,
configName: name ?? ''
}, {
retry: false,
enabled: isActive,
select: (data) => {
const isOk = isStatusOk(app, data.status);
if (isOk)
Consola.info(`Ping of app "${app.name}" (${app.url}) returned ${data.status} (Accepted)`);
else
Consola.warn(`Ping of app "${app.name}" (${app.url}) returned ${data.status} (Refused)`);
return {
status: data.status,
state: isOk ? ('online' as const) : ('down' as const),
statusText: data.statusText,
};
},
});
return {
...queryResult,
isActive
}
}
type PingPulse = {
animate?: TargetAndTransition;
transition?: Transition;
}
const usePingPulse = ({isOnline}: {isOnline: boolean}): PingPulse => {
const { config } = useConfigContext();
const disablePulse = config?.settings.customization.accessibility?.disablePingPulse ?? false;
if (disablePulse) {
return {};
}
return {
animate: {
scale: isOnline ? [1, 0.7, 1] : 1
},
transition: {
repeat: Infinity,
duration: 2.5,
ease: 'easeInOut',
}
}
}

View File

@@ -1,24 +1,24 @@
import { TRPCError } from '@trpc/server'; import { TRPCError } from '@trpc/server';
import axios, { AxiosError } from 'axios'; import axios, { AxiosError } from 'axios';
import Consola from 'consola'; import Consola from 'consola';
import { getCookie } from 'cookies-next';
import https from 'https'; import https from 'https';
import { z } from 'zod'; import { z } from 'zod';
import { getIsOk } from '~/components/Dashboard/Tiles/Apps/AppPing'; import { isStatusOk } from '~/components/Dashboard/Tiles/Apps/AppPing';
import { getConfig } from '~/tools/config/getConfig'; import { getConfig } from '~/tools/config/getConfig';
import { AppType } from '~/types/app'; import { AppType } from '~/types/app';
import { createTRPCRouter, publicProcedure } from '../trpc'; import { createTRPCRouter, publicProcedure } from '../trpc';
export const appRouter = createTRPCRouter({ export const appRouter = createTRPCRouter({
ping: publicProcedure.input(z.string()).query(async ({ input }) => { ping: publicProcedure.input(z.object({
id: z.string(),
configName: z.string()
})).query(async ({ input }) => {
const agent = new https.Agent({ rejectUnauthorized: false }); const agent = new https.Agent({ rejectUnauthorized: false });
const configName = getCookie('config-name'); const config = getConfig(input.configName);
const config = getConfig(configName?.toString() ?? 'default'); const app = config.apps.find((app) => app.id === input.id);
const app = config.apps.find((app) => app.id === input);
const url = app?.url; if (!app?.url) {
if (url === undefined || !app) {
Consola.error(`App ${input} not found`); Consola.error(`App ${input} not found`);
throw new TRPCError({ throw new TRPCError({
code: 'NOT_FOUND', code: 'NOT_FOUND',
@@ -27,22 +27,23 @@ export const appRouter = createTRPCRouter({
}); });
} }
const res = await axios const res = await axios
.get(url, { httpsAgent: agent, timeout: 2000 }) .get(app.url, { httpsAgent: agent, timeout: 2000 })
.then((response) => ({ .then((response) => ({
status: response.status, status: response.status,
statusText: response.statusText, statusText: response.statusText,
state: isStatusOk(app as AppType, response.status) ? 'online' : 'offline'
})) }))
.catch((error: AxiosError) => { .catch((error: AxiosError) => {
if (error.response) { if (error.response) {
return { return {
state: getIsOk(app as AppType, error.response.status) ? 'online' : 'offline', state: isStatusOk(app as AppType, error.response.status) ? 'online' : 'offline',
status: error.response.status, status: error.response.status,
statusText: error.response.statusText, statusText: error.response.statusText,
}; };
} }
if (error.code === 'ECONNABORTED') { if (error.code === 'ECONNABORTED') {
Consola.error(`Ping timed out for app with id : ${input} (url: ${url})`); Consola.error(`Ping timed out for app with id : ${input} (url: ${app.url})`);
throw new TRPCError({ throw new TRPCError({
code: 'TIMEOUT', code: 'TIMEOUT',
cause: input, cause: input,