diff --git a/package.json b/package.json index 5503530175de..a4dc90df92ac 100644 --- a/package.json +++ b/package.json @@ -36,8 +36,6 @@ "@nestjs/serve-static": "^4.0.1", "@nestjs/terminus": "^9.2.2", "@nestjs/typeorm": "^10.0.0", - "@nivo/calendar": "^0.84.0", - "@nivo/core": "^0.84.0", "@nx/eslint-plugin": "^17.2.8", "@octokit/graphql": "^7.0.2", "@ptc-org/nestjs-query-core": "^4.2.0", diff --git a/packages/twenty-front/package.json b/packages/twenty-front/package.json index 90bd99f40175..b7c173b7fed0 100644 --- a/packages/twenty-front/package.json +++ b/packages/twenty-front/package.json @@ -30,6 +30,9 @@ "workerDirectory": "public" }, "dependencies": { + "@nivo/calendar": "^0.87.0", + "@nivo/core": "^0.87.0", + "@nivo/line": "^0.87.0", "@xyflow/react": "^12.0.4", "transliteration": "^2.3.5" } diff --git a/packages/twenty-front/src/modules/app/components/SettingsRoutes.tsx b/packages/twenty-front/src/modules/app/components/SettingsRoutes.tsx index 6f37c0595e12..5eede03959b5 100644 --- a/packages/twenty-front/src/modules/app/components/SettingsRoutes.tsx +++ b/packages/twenty-front/src/modules/app/components/SettingsRoutes.tsx @@ -65,7 +65,7 @@ const SettingsDevelopersApiKeysNew = lazy(() => const SettingsDevelopersWebhooksNew = lazy(() => import( - '~/pages/settings/developers/webhooks/SettingsDevelopersWebhooksNew' + '~/pages/settings/developers/webhooks/components/SettingsDevelopersWebhooksNew' ).then((module) => ({ default: module.SettingsDevelopersWebhooksNew, })), @@ -165,7 +165,7 @@ const SettingsObjects = lazy(() => const SettingsDevelopersWebhooksDetail = lazy(() => import( - '~/pages/settings/developers/webhooks/SettingsDevelopersWebhookDetail' + '~/pages/settings/developers/webhooks/components/SettingsDevelopersWebhookDetail' ).then((module) => ({ default: module.SettingsDevelopersWebhooksDetail, })), diff --git a/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph.tsx b/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph.tsx new file mode 100644 index 000000000000..eb2e359fff16 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph.tsx @@ -0,0 +1,59 @@ +import { webhookGraphDataState } from '@/settings/developers/webhook/states/webhookGraphDataState'; +import styled from '@emotion/styled'; +import { ResponsiveLine } from '@nivo/line'; +import { Section } from '@react-email/components'; +import { useRecoilValue } from 'recoil'; +import { H2Title } from 'twenty-ui'; + +export type NivoLineInput = { + id: string | number; + color?: string; + data: Array<{ + x: number | string | Date; + y: number | string | Date; + }>; +}; +const StyledGraphContainer = styled.div` + height: 200px; + width: 100%; +`; +export const SettingsDeveloppersWebhookUsageGraph = () => { + const webhookGraphData = useRecoilValue(webhookGraphDataState); + + return ( + <> + {webhookGraphData.length ? ( +
+ + + d.color} + margin={{ top: 0, right: 0, bottom: 50, left: 60 }} + xFormat="time:%Y-%m-%d %H:%M%" + xScale={{ + type: 'time', + useUTC: false, + format: '%Y-%m-%d %H:%M:%S', + precision: 'hour', + }} + yScale={{ + type: 'linear', + }} + axisBottom={{ + tickValues: 'every day', + format: '%b %d', + }} + enableTouchCrosshair={true} + enableGridY={false} + enableGridX={false} + enablePoints={false} + /> + +
+ ) : ( + <> + )} + + ); +}; diff --git a/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraphEffect.tsx b/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraphEffect.tsx new file mode 100644 index 000000000000..0c26350243b2 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraphEffect.tsx @@ -0,0 +1,101 @@ +import { NivoLineInput } from '@/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph'; +import { webhookGraphDataState } from '@/settings/developers/webhook/states/webhookGraphDataState'; +import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; +import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; +import { useEffect } from 'react'; +import { useSetRecoilState } from 'recoil'; + +type SettingsDevelopersWebhookUsageGraphEffectProps = { + webhookId: string; +}; + +export const SettingsDevelopersWebhookUsageGraphEffect = ({ + webhookId, +}: SettingsDevelopersWebhookUsageGraphEffectProps) => { + const setWebhookGraphData = useSetRecoilState(webhookGraphDataState); + + const { enqueueSnackBar } = useSnackBar(); + + useEffect(() => { + const fetchData = async () => { + try { + const queryString = new URLSearchParams({ + webhookIdRequest: webhookId, + }).toString(); + const token = 'REPLACE_ME'; + const response = await fetch( + `https://api.eu-central-1.aws.tinybird.co/v0/pipes/getWebhooksAnalytics.json?${queryString}`, + { + headers: { + Authorization: 'Bearer ' + token, + }, + }, + ); + const result = await response.json(); + + if (!response.ok) { + enqueueSnackBar('Something went wrong while fetching webhook usage', { + variant: SnackBarVariant.Error, + }); + return; + } + + const graphInput = result.data + .flatMap( + (dataRow: { + start_interval: string; + failure_count: number; + success_count: number; + }) => [ + { + x: dataRow.start_interval, + y: dataRow.failure_count, + id: 'failure_count', + color: 'red', + }, + { + x: dataRow.start_interval, + y: dataRow.success_count, + id: 'success_count', + color: 'green', + }, + ], + ) + .reduce( + ( + acc: NivoLineInput[], + { + id, + x, + y, + color, + }: { id: string; x: string; y: number; color: string }, + ) => { + const existingGroupIndex = acc.findIndex( + (group) => group.id === id, + ); + const isExistingGroup = existingGroupIndex !== -1; + + if (isExistingGroup) { + return acc.map((group, index) => + index === existingGroupIndex + ? { ...group, data: [...group.data, { x, y }] } + : group, + ); + } else { + return [...acc, { id, color, data: [{ x, y }] }]; + } + }, + [], + ); + setWebhookGraphData(graphInput); + } catch (error) { + enqueueSnackBar('Something went wrong while fetching webhook usage', { + variant: SnackBarVariant.Error, + }); + } + }; + fetchData(); + }, [enqueueSnackBar, setWebhookGraphData, webhookId]); + return <>; +}; diff --git a/packages/twenty-front/src/modules/settings/developers/webhook/states/webhookGraphDataState.ts b/packages/twenty-front/src/modules/settings/developers/webhook/states/webhookGraphDataState.ts new file mode 100644 index 000000000000..bb91864e2782 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/developers/webhook/states/webhookGraphDataState.ts @@ -0,0 +1,7 @@ +import { NivoLineInput } from '@/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph'; +import { createState } from 'twenty-ui'; + +export const webhookGraphDataState = createState({ + key: 'webhookGraphData', + defaultValue: [], +}); diff --git a/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts b/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts index 3abc8299b7f0..9a79efa933bf 100644 --- a/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts +++ b/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts @@ -13,4 +13,5 @@ export type FeatureFlagKey = | 'IS_SEARCH_ENABLED' | 'IS_QUERY_RUNNER_TWENTY_ORM_ENABLED' | 'IS_GMAIL_SEND_EMAIL_SCOPE_ENABLED' - | 'IS_WORKSPACE_MIGRATED_FOR_SEARCH'; + | 'IS_WORKSPACE_MIGRATED_FOR_SEARCH' + | 'IS_ANALYTICS_V2_ENABLED'; diff --git a/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksDetail.stories.tsx b/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksDetail.stories.tsx index ec430fcf3058..a77e38d73a5b 100644 --- a/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksDetail.stories.tsx +++ b/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksDetail.stories.tsx @@ -2,7 +2,7 @@ import { Meta, StoryObj } from '@storybook/react'; import { within } from '@storybook/test'; import { graphql, HttpResponse } from 'msw'; -import { SettingsDevelopersWebhooksDetail } from '~/pages/settings/developers/webhooks/SettingsDevelopersWebhookDetail'; +import { SettingsDevelopersWebhooksDetail } from '~/pages/settings/developers/webhooks/components/SettingsDevelopersWebhookDetail'; import { PageDecorator, PageDecoratorArgs, diff --git a/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksNew.stories.tsx b/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksNew.stories.tsx index dd4adfb71305..430349183258 100644 --- a/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksNew.stories.tsx +++ b/packages/twenty-front/src/pages/settings/developers/__stories__/webhooks/SettingsDevelopersWebhooksNew.stories.tsx @@ -1,7 +1,7 @@ import { Meta, StoryObj } from '@storybook/react'; import { within } from '@storybook/test'; -import { SettingsDevelopersWebhooksNew } from '~/pages/settings/developers/webhooks/SettingsDevelopersWebhooksNew'; +import { SettingsDevelopersWebhooksNew } from '~/pages/settings/developers/webhooks/components/SettingsDevelopersWebhooksNew'; import { PageDecorator, PageDecoratorArgs, diff --git a/packages/twenty-front/src/pages/settings/developers/webhooks/SettingsDevelopersWebhookDetail.tsx b/packages/twenty-front/src/pages/settings/developers/webhooks/components/SettingsDevelopersWebhookDetail.tsx similarity index 91% rename from packages/twenty-front/src/pages/settings/developers/webhooks/SettingsDevelopersWebhookDetail.tsx rename to packages/twenty-front/src/pages/settings/developers/webhooks/components/SettingsDevelopersWebhookDetail.tsx index ffe6d9807c85..f88ba926c6b8 100644 --- a/packages/twenty-front/src/pages/settings/developers/webhooks/SettingsDevelopersWebhookDetail.tsx +++ b/packages/twenty-front/src/pages/settings/developers/webhooks/components/SettingsDevelopersWebhookDetail.tsx @@ -11,6 +11,8 @@ import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; import { Webhook } from '@/settings/developers/types/webhook/Webhook'; +import { SettingsDeveloppersWebhookUsageGraph } from '@/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraph'; +import { SettingsDevelopersWebhookUsageGraphEffect } from '@/settings/developers/webhook/components/SettingsDevelopersWebhookUsageGraphEffect'; import { getSettingsPagePath } from '@/settings/utils/getSettingsPagePath'; import { SettingsPath } from '@/types/SettingsPath'; import { Button } from '@/ui/input/button/components/Button'; @@ -20,6 +22,7 @@ import { TextInput } from '@/ui/input/components/TextInput'; import { ConfirmationModal } from '@/ui/layout/modal/components/ConfirmationModal'; import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; import { Section } from '@/ui/layout/section/components/Section'; +import { useIsFeatureEnabled } from '@/workspace/hooks/useIsFeatureEnabled'; const StyledFilterRow = styled.div` display: flex; @@ -63,6 +66,8 @@ export const SettingsDevelopersWebhooksDetail = () => { navigate(developerPath); }; + const isAnalyticsV2Enabled = useIsFeatureEnabled('IS_ANALYTICS_V2_ENABLED'); + const fieldTypeOptions = [ { value: '*', label: 'All Objects' }, ...objectMetadataItems.map((item) => ({ @@ -173,6 +178,14 @@ export const SettingsDevelopersWebhooksDetail = () => { /> + {isAnalyticsV2Enabled ? ( + <> + + + + ) : ( + <> + )}