diff --git a/connectors/src/api/get_connector_permissions.ts b/connectors/src/api/get_connector_permissions.ts index 71dc3f09b56f..bb3598f1c7fb 100644 --- a/connectors/src/api/get_connector_permissions.ts +++ b/connectors/src/api/get_connector_permissions.ts @@ -7,6 +7,7 @@ import type { Request, Response } from "express"; import { getConnectorManager } from "@connectors/connectors"; import { augmentContentNodesWithParentIds } from "@connectors/lib/api/content_nodes"; +import { ProviderWorkflowError } from "@connectors/lib/error"; import logger from "@connectors/logger/logger"; import { apiError, withLogging } from "@connectors/logger/withlogging"; import { ConnectorResource } from "@connectors/resources/connector_resource"; @@ -85,6 +86,18 @@ const _getConnectorPermissions = async ( }); if (pRes.isErr()) { + if ( + pRes.error instanceof ProviderWorkflowError && + pRes.error.type === "rate_limit_error" + ) { + return apiError(req, res, { + status_code: 429, + api_error: { + type: "connector_rate_limit_error", + message: pRes.error.message, + }, + }); + } return apiError(req, res, { status_code: 500, api_error: { diff --git a/connectors/src/connectors/slack/index.ts b/connectors/src/connectors/slack/index.ts index 3729371c2a30..8efb5cd7b986 100644 --- a/connectors/src/connectors/slack/index.ts +++ b/connectors/src/connectors/slack/index.ts @@ -21,7 +21,10 @@ import { getSlackClient, } from "@connectors/connectors/slack/lib/slack_client"; import { launchSlackSyncWorkflow } from "@connectors/connectors/slack/temporal/client.js"; -import { ExternalOAuthTokenError } from "@connectors/lib/error"; +import { + ExternalOAuthTokenError, + ProviderWorkflowError, +} from "@connectors/lib/error"; import { SlackChannel } from "@connectors/lib/models/slack"; import { terminateAllWorkflowsForConnectorId } from "@connectors/lib/temporal"; import logger from "@connectors/logger/logger"; @@ -404,6 +407,13 @@ export class SlackConnectorManager extends BaseConnectorManager { resources: BaseContentNode[]; isResourcesLoading: boolean; isResourcesError: boolean; + resourcesError?: APIError | null; }; export type ContentNodeTreeItemStatus = { @@ -112,7 +113,7 @@ function ContentNodeTreeChildren({ const { useResourcesHook, emptyComponent } = useContentNodeTreeContext(); - const { resources, isResourcesLoading, isResourcesError } = + const { resources, isResourcesLoading, isResourcesError, resourcesError } = useResourcesHook(parentId); const filteredNodes = resources.filter( @@ -152,7 +153,14 @@ function ContentNodeTreeChildren({ if (isResourcesError) { return (
- Failed to retrieve permissions likely due to a revoked authorization. + {resourcesError?.type === "rate_limit_error" ? ( + <>Connected service's API limit reached. Please retry shortly. + ) : ( + <> + Failed to retrieve permissions likely due to a revoked + authorization. + + )}
); } diff --git a/front/lib/api/data_source_view.ts b/front/lib/api/data_source_view.ts index 5f63b7c691e3..f0be3c5263cd 100644 --- a/front/lib/api/data_source_view.ts +++ b/front/lib/api/data_source_view.ts @@ -109,6 +109,9 @@ async function getContentNodesForManagedDataSourceView( }); if (connectorsRes.isErr()) { + if (connectorsRes.error.type === "connector_rate_limit_error") { + return new Err(new Error(connectorsRes.error.message)); + } return new Err( new Error( "An error occurred while fetching the resources' children content nodes." diff --git a/front/lib/swr/connectors.ts b/front/lib/swr/connectors.ts index 0e13ae266d96..b8b08e134ea5 100644 --- a/front/lib/swr/connectors.ts +++ b/front/lib/swr/connectors.ts @@ -1,5 +1,6 @@ import { useSendNotification } from "@dust-tt/sparkle"; import type { + APIError, ConnectorPermission, ContentNodesViewType, DataSourceType, @@ -21,7 +22,8 @@ import type { GetDataSourcePermissionsResponseBody } from "@app/pages/api/w/[wId interface UseConnectorPermissionsReturn { resources: GetDataSourcePermissionsResponseBody["resources"]; isResourcesLoading: boolean; - isResourcesError: any; + isResourcesError: boolean; + resourcesError: APIError | null; } export function useConnectorPermissions({ @@ -77,6 +79,7 @@ export function useConnectorPermissions({ ), isResourcesLoading: !error && !data, isResourcesError: error, + resourcesError: error ? (error.error as APIError) : null, }; } diff --git a/front/pages/api/w/[wId]/data_sources/[dsId]/managed/permissions/index.ts b/front/pages/api/w/[wId]/data_sources/[dsId]/managed/permissions/index.ts index 33f0ee51e75b..e117819a8a45 100644 --- a/front/pages/api/w/[wId]/data_sources/[dsId]/managed/permissions/index.ts +++ b/front/pages/api/w/[wId]/data_sources/[dsId]/managed/permissions/index.ts @@ -272,6 +272,16 @@ export async function getManagedDataSourcePermissionsHandler( includeParents, }); if (permissionsRes.isErr()) { + if (permissionsRes.error.type === "connector_rate_limit_error") { + return apiError(req, res, { + status_code: 429, + api_error: { + type: "rate_limit_error", + message: + "Rate limit error while retrieving the data source permissions", + }, + }); + } return apiError(req, res, { status_code: 500, api_error: {