import normalizeUrl from 'sentry/utils/url/normalizeUrl'; import useOrganization from 'sentry/utils/useOrganization'; import {BASE_URL as RESOURCES_BASE_URL} from 'sentry/views/insights/browser/resources/settings'; import {BASE_URL as VITALS_BASE_URL} from 'sentry/views/insights/browser/webVitals/settings'; import {BASE_URL as CACHE_BASE_URL} from 'sentry/views/insights/cache/settings'; import {BASE_URL as DB_BASE_URL} from 'sentry/views/insights/database/settings'; import {BASE_URL as HTTP_BASE_URL} from 'sentry/views/insights/http/settings'; import {BASE_URL as AI_BASE_URL} from 'sentry/views/insights/llmMonitoring/settings'; import {BASE_URL as APP_STARTS_BASE_URL} from 'sentry/views/insights/mobile/appStarts/settings'; import {BASE_URL as SCREEN_LOADS_BASE_URL} from 'sentry/views/insights/mobile/screenload/settings'; import {BASE_URL as SCREEN_RENDERING_BASE_URL} from 'sentry/views/insights/mobile/screenRendering/settings'; import {BASE_URL as MOBILE_SCREENS_BASE_URL} from 'sentry/views/insights/mobile/screens/settings'; import {BASE_URL as MOBILE_UI_BASE_URL} from 'sentry/views/insights/mobile/ui/settings'; import {DOMAIN_VIEW_BASE_URL} from 'sentry/views/insights/pages/settings'; import { type DomainView, useDomainViewFilters, } from 'sentry/views/insights/pages/useFilters'; import {BASE_URL as QUEUE_BASE_URL} from 'sentry/views/insights/queues/settings'; import {INSIGHTS_BASE_URL} from 'sentry/views/insights/settings'; import {ModuleName} from 'sentry/views/insights/types'; export const MODULE_BASE_URLS: Record = { [ModuleName.DB]: DB_BASE_URL, [ModuleName.HTTP]: HTTP_BASE_URL, [ModuleName.CACHE]: CACHE_BASE_URL, [ModuleName.QUEUE]: QUEUE_BASE_URL, [ModuleName.SCREEN_LOAD]: SCREEN_LOADS_BASE_URL, [ModuleName.APP_START]: APP_STARTS_BASE_URL, [ModuleName.VITAL]: VITALS_BASE_URL, [ModuleName.RESOURCE]: RESOURCES_BASE_URL, [ModuleName.AI]: AI_BASE_URL, [ModuleName.MOBILE_UI]: MOBILE_UI_BASE_URL, [ModuleName.MOBILE_SCREENS]: MOBILE_SCREENS_BASE_URL, [ModuleName.SCREEN_RENDERING]: SCREEN_RENDERING_BASE_URL, [ModuleName.OTHER]: '', }; type ModuleNameStrings = `${ModuleName}`; export type RoutableModuleNames = Exclude; export const useModuleURL = ( moduleName: RoutableModuleNames, bare: boolean = false, view?: DomainView ): string => { const forceDomainView = Boolean(view); const builder = useModuleURLBuilder(bare, true, forceDomainView); return builder(moduleName, view); }; type URLBuilder = (moduleName: RoutableModuleNames, domainView?: DomainView) => string; export function useModuleURLBuilder( bare: boolean = false, autoDetectDomainView: boolean = true, forceDomainView?: boolean // TODO - eventually this param will be removed once we don't have modules in two spots ): URLBuilder { const organization = useOrganization({allowNull: true}); // Some parts of the app, like the main sidebar, render even if the organization isn't available (during loading, or at all). const {isInDomainView, view: currentView} = useDomainViewFilters(); if (!organization) { // If there isn't an organization, items that link to modules won't be visible, so this is a fallback just-in-case, and isn't trying too hard to be useful return () => ''; } const {slug} = organization; if ((autoDetectDomainView && isInDomainView) || forceDomainView) { return function (moduleName: RoutableModuleNames, domainView?: DomainView) { const view = domainView ?? currentView; return bare ? `${DOMAIN_VIEW_BASE_URL}/${view}/${MODULE_BASE_URLS[moduleName]}` : normalizeUrl( `/organizations/${slug}/${DOMAIN_VIEW_BASE_URL}/${view}/${MODULE_BASE_URLS[moduleName]}` ); }; } return function (moduleName: RoutableModuleNames) { return bare ? `${INSIGHTS_BASE_URL}/${MODULE_BASE_URLS[moduleName]}` : normalizeUrl( `/organizations/${slug}/${INSIGHTS_BASE_URL}/${MODULE_BASE_URLS[moduleName]}` ); }; }