mirror of
https://github.com/ajnart/homarr.git
synced 2025-11-10 07:25:48 +01:00
🐛 Fix ping not reading from correct config
This commit is contained in:
@@ -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',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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,
|
||||||
|
|||||||
Reference in New Issue
Block a user