diff --git a/web/src/beta/features/Notification/hooks.ts b/web/src/beta/features/Notification/hooks.ts index fab6de4de6..426c6498a5 100644 --- a/web/src/beta/features/Notification/hooks.ts +++ b/web/src/beta/features/Notification/hooks.ts @@ -1,9 +1,6 @@ import { useT, useLang } from "@reearth/services/i18n"; -import { - useError, - useNotification, - Notification -} from "@reearth/services/state"; +import { useNotification, Notification } from "@reearth/services/state"; +import { useErrors } from "@reearth/services/state/gqlErrorHandling"; import { useState, useEffect, useCallback, useMemo } from "react"; export type PolicyItems = @@ -26,7 +23,7 @@ const policyItems: PolicyItems[] = [ export default () => { const t = useT(); const currentLanguage = useLang(); - const [error, setError] = useError(); + const [errors, setErrors] = useErrors(); const [notification, setNotification] = useNotification(); const [visible, changeVisibility] = useState(false); @@ -54,42 +51,44 @@ export default () => { }, []); useEffect(() => { - if (!error) return; - if (error.message?.includes("policy violation") && error.message) { - const limitedItem = policyItems.find((i) => error.message?.includes(i)); - const policyItem = - limitedItem && policyLimitNotifications - ? policyLimitNotifications[limitedItem] - : undefined; - const message = policyItem - ? typeof policyItem === "string" - ? policyItem - : policyItem[currentLanguage] - : t( - "You have reached a policy limit. Please contact an administrator of your Re:Earth system." - ); + if (errors.length === 0) return; + errors.forEach((error) => { + if (error.message?.includes("policy violation") && error.message) { + const limitedItem = policyItems.find((i) => error.message?.includes(i)); + const policyItem = + limitedItem && policyLimitNotifications + ? policyLimitNotifications[limitedItem] + : undefined; + const message = policyItem + ? typeof policyItem === "string" + ? policyItem + : policyItem[currentLanguage] + : t( + "You have reached a policy limit. Please contact an administrator of your Re:Earth system." + ); - setNotification({ - type: "info", - heading: noticeHeading, - text: message, - duration: "persistent" - }); - } else { - setNotification({ - type: "error", - heading: errorHeading, - text: t("Something went wrong. Please try again later.") - }); - } - setError(undefined); + setNotification({ + type: "info", + heading: noticeHeading, + text: message, + duration: "persistent" + }); + } else { + setNotification({ + type: "error", + heading: errorHeading, + text: error.description || error.message || "" + }); + } + }); + setErrors([]); }, [ - error, + errors, currentLanguage, policyLimitNotifications, errorHeading, noticeHeading, - setError, + setErrors, setNotification, t ]); diff --git a/web/src/sentry.ts b/web/src/sentry.ts index e303896863..95b4d9ed5d 100644 --- a/web/src/sentry.ts +++ b/web/src/sentry.ts @@ -12,14 +12,16 @@ export const initialize = () => { } }; -export const reportError = (error: ReportError) => { - if (error instanceof Error) { - Sentry.captureException(error); - } else { - Sentry.captureException( - new Error( - `${error.type || "Unknown"}: ${error.message || "No message provided"}` - ) - ); - } +export const reportError = (errors: ReportError[]) => { + errors.forEach((error) => { + if (error instanceof Error) { + Sentry.captureException(error); + } else { + Sentry.captureException( + new Error( + `${error.type || "Unknown"}: ${error.message || "No message provided"}` + ) + ); + } + }); }; diff --git a/web/src/services/gql/provider/index.tsx b/web/src/services/gql/provider/index.tsx index 4e865afda7..9f9b1a4e5d 100644 --- a/web/src/services/gql/provider/index.tsx +++ b/web/src/services/gql/provider/index.tsx @@ -14,6 +14,7 @@ import { useCallback, type ReactNode } from "react"; import fragmentMatcher from "../__gen__/fragmentMatcher.json"; import { authLink, sentryLink, errorLink, uploadLink, taskLink } from "./links"; +import langLink from "./links/langLink"; import { paginationMerge } from "./pagination"; const Provider: React.FC<{ children?: ReactNode }> = ({ children }) => { @@ -90,6 +91,7 @@ const Provider: React.FC<{ children?: ReactNode }> = ({ children }) => { errorLink(), sentryLink(endpoint), authLink(), + langLink(), // https://github.com/apollographql/apollo-client/issues/6011#issuecomment-619468320 uploadLink(endpoint) as unknown as ApolloLink ]), diff --git a/web/src/services/gql/provider/links/errorLink.ts b/web/src/services/gql/provider/links/errorLink.ts index 6ce70b57e2..6c0114fdb8 100644 --- a/web/src/services/gql/provider/links/errorLink.ts +++ b/web/src/services/gql/provider/links/errorLink.ts @@ -1,25 +1,33 @@ import { onError } from "@apollo/client/link/error"; import { reportError } from "@reearth/sentry"; import { useSetError } from "@reearth/services/state"; +import { GQLError } from "@reearth/services/state/gqlErrorHandling"; export default () => { - const { setError } = useSetError(); + const { setErrors } = useSetError(); return onError(({ graphQLErrors, networkError }) => { if (!networkError && !graphQLErrors) return; - let error: { type?: string; message?: string } | undefined; - + let errors: GQLError[] = []; + console.log(graphQLErrors); if (networkError?.message) { - error = { message: networkError?.message }; + errors = [ + { message: networkError?.message, description: networkError.message } + ]; } else { - error = { - type: graphQLErrors?.[0].path?.[0].toString(), - message: graphQLErrors?.[0].message - }; + errors = + graphQLErrors?.map((gqlError) => { + return { + type: gqlError.path?.[0].toString(), + message: gqlError.message, + code: gqlError.extensions?.code as string, + description: gqlError.extensions?.description as string + }; + }) ?? []; } - if (error) { - setError(error); - reportError(error); + if (errors.length > 0) { + setErrors(errors); + reportError(errors); } }); }; diff --git a/web/src/services/gql/provider/links/langLink.ts b/web/src/services/gql/provider/links/langLink.ts new file mode 100644 index 0000000000..18d5063531 --- /dev/null +++ b/web/src/services/gql/provider/links/langLink.ts @@ -0,0 +1,14 @@ +import { setContext } from "@apollo/client/link/context"; +import i18n from "@reearth/services/i18n/i18n"; + +export default () => { + return setContext(async (_, { headers }) => { + const local = i18n.language.split("-")[0]; + return { + headers: { + ...headers, + lang: local + } + }; + }); +}; diff --git a/web/src/services/state/gqlErrorHandling.ts b/web/src/services/state/gqlErrorHandling.ts index 27b4a6d996..b4d8e4f744 100644 --- a/web/src/services/state/gqlErrorHandling.ts +++ b/web/src/services/state/gqlErrorHandling.ts @@ -1,12 +1,16 @@ import { atom, useAtom, useSetAtom } from "jotai"; // useError is needed for Apollo provider error only. Handle other errors with useNotification directly. -type GQLError = { type?: string; message?: string }; -const error = atom(undefined); - -export const useError = () => useAtom(error); +export type GQLError = { + type?: string; + message?: string; + code?: string; + description?: string; +}; +const errors = atom([]); +export const useErrors = () => useAtom(errors); export default () => { - const setError = useSetAtom(error); - return { setError }; + const setErrors = useSetAtom(errors); + return { setErrors }; }; diff --git a/web/src/services/state/index.ts b/web/src/services/state/index.ts index 39725f1af9..d45e863bda 100644 --- a/web/src/services/state/index.ts +++ b/web/src/services/state/index.ts @@ -5,7 +5,7 @@ import { TeamMember } from "../gql"; export * from "./devPlugins"; -export { default as useSetError, useError } from "./gqlErrorHandling"; +export { default as useSetError } from "./gqlErrorHandling"; export type WidgetAreaState = { zone: "inner" | "outer";