import type {Route, RouteComponentProps, RouteContextInterface} from 'react-router'; import type {ChildrenRenderFn} from 'sentry/components/acl/feature'; import type {Guide} from 'sentry/components/assistant/types'; import type {ButtonProps} from 'sentry/components/button'; import {ProductSelectionProps} from 'sentry/components/onboarding/productSelection'; import type SidebarItem from 'sentry/components/sidebar/sidebarItem'; import type DateRange from 'sentry/components/timeRangeSelector/dateRange'; import type SelectorItems from 'sentry/components/timeRangeSelector/selectorItems'; import {SVGIconProps} from 'sentry/icons/svgIcon'; import type {Group} from 'sentry/types'; import {UseExperiment} from 'sentry/utils/useExperiment'; import {OrganizationStatsProps} from 'sentry/views/organizationStats/index'; import {RouteAnalyticsContext} from 'sentry/views/routeAnalyticsContextProvider'; import type {NavigationItem, NavigationSection} from 'sentry/views/settings/types'; import type {ExperimentKey} from './experiments'; import type {Integration, IntegrationProvider} from './integrations'; import type {Member, Organization} from './organization'; import type {Project} from './project'; import type {User} from './user'; // XXX(epurkhiser): A Note about `_`. // // We add the `_: any` type int our hooks list to stop // typescript from doing too much type tightening. We should absolutely revisit // this in the future because all callbacks _should_ be allowed to be // functions, but doing so causes some unexpected issues and makes typescript // not happy. We still get a huge advantage of typing just by having each hook // type here however. /** * The Hooks type mapping is the master interface for all external Hooks into * the sentry frontend application. */ export interface Hooks extends RouteHooks, ComponentHooks, CustomizationHooks, AnalyticsHooks, FeatureDisabledHooks, InterfaceChromeHooks, OnboardingHooks, SettingsHooks, FeatureSpecificHooks, ReactHooks, CallbackHooks { _: any; } export type HookName = keyof Hooks; /** * Route hooks. */ export type RouteHooks = { 'routes:api': RoutesHook; 'routes:organization': RoutesHook; 'routes:root': RoutesHook; }; /** * Component specific hooks for DateRange and SelectorItems * These components have plan specific overrides in getsentry */ type DateRangeProps = React.ComponentProps; type SelectorItemsProps = React.ComponentProps; type DisabledMemberViewProps = RouteComponentProps<{orgId: string}, {}>; type MemberListHeaderProps = { members: Member[]; organization: Organization; }; type DisabledAppStoreConnectMultiple = { children: React.ReactNode; organization: Organization; }; type DisabledCustomSymbolSources = { children: React.ReactNode; organization: Organization; }; type DisabledMemberTooltipProps = {children: React.ReactNode}; type DashboardHeadersProps = {organization: Organization}; type ReplayFeedbackButton = {children: React.ReactNode}; type ReplayListPageHeaderProps = {children?: React.ReactNode}; type ReplayOnboardingAlertProps = {children: React.ReactNode}; type ReplayOnboardingCTAProps = {children: React.ReactNode; organization: Organization}; type ProductUnavailableCTAProps = {organization: Organization}; type ProfilingBetaAlertBannerProps = { organization: Organization; }; type ProfilingUpgradePlanButtonProps = ButtonProps & { children: React.ReactNode; fallback: React.ReactNode; organization: Organization; }; type ProfilingAM1OrMMXUpgradeProps = { fallback: React.ReactNode; organization: Organization; }; type ProductSelectionAvailabilityProps = Pick< ProductSelectionProps, 'lazyLoader' | 'skipLazyLoader' | 'platform' | 'withBottomMargin' > & { organization: Organization; }; type FirstPartyIntegrationAlertProps = { integrations: Integration[]; hideCTA?: boolean; wrapWithContainer?: boolean; }; type FirstPartyIntegrationAdditionalCTAProps = { integrations: Integration[]; }; type AttemptCloseAttemptProps = { handleRemoveAccount: () => void; organizationSlugs: string[]; }; type CodecovLinkProps = { organization: Organization; }; type QualitativeIssueFeedbackProps = { group: Group; organization: Organization; }; type GuideUpdateCallback = (nextGuide: Guide | null, opts: {dismissed?: boolean}) => void; type SentryLogoProps = SVGIconProps & { pride?: boolean; }; /** * Component wrapping hooks */ export type ComponentHooks = { 'component:codecov-integration-settings-link': () => React.ComponentType; 'component:codecov-integration-stacktrace-link': () => React.ComponentType; 'component:confirm-account-close': () => React.ComponentType; 'component:dashboards-header': () => React.ComponentType; 'component:disabled-app-store-connect-multiple': () => React.ComponentType; 'component:disabled-custom-symbol-sources': () => React.ComponentType; 'component:disabled-member': () => React.ComponentType; 'component:disabled-member-tooltip': () => React.ComponentType; 'component:enhanced-org-stats': () => React.ComponentType; 'component:escalating-issues-banner-feedback': () => React.ComponentType; 'component:first-party-integration-additional-cta': () => React.ComponentType; 'component:first-party-integration-alert': () => React.ComponentType; 'component:header-date-range': () => React.ComponentType; 'component:header-selector-items': () => React.ComponentType; 'component:issue-priority-feedback': () => React.ComponentType; 'component:member-list-header': () => React.ComponentType; 'component:org-stats-banner': () => React.ComponentType; 'component:product-selection-availability': () => React.ComponentType; 'component:product-unavailable-cta': () => React.ComponentType; 'component:profiling-am1-or-mmx-upgrade': () => React.ComponentType; 'component:profiling-billing-banner': () => React.ComponentType; 'component:profiling-upgrade-plan-button': () => React.ComponentType; 'component:replay-feedback-button': () => React.ComponentType; 'component:replay-list-page-header': () => React.ComponentType | null; 'component:replay-onboarding-alert': () => React.ComponentType; 'component:replay-onboarding-cta': () => React.ComponentType; 'component:replay-onboarding-cta-button': () => React.ComponentType<{}> | null; 'component:sentry-logo': () => React.ComponentType; 'component:superuser-access-category': React.ComponentType; }; /** * Customization hooks are advanced hooks that return render-prop style * components the allow for specific customizations of components. * * These are very similar to the component wrapping hooks */ export type CustomizationHooks = { 'integrations:feature-gates': IntegrationsFeatureGatesHook; 'member-invite-button:customization': InviteButtonCustomizationHook; 'member-invite-modal:customization': InviteModalCustomizationHook; }; /** * Analytics / tracking / and operational metrics backend hooks. */ export type AnalyticsHooks = { 'analytics:init-user': AnalyticsInitUser; 'analytics:log-experiment': AnalyticsLogExperiment; 'analytics:raw-track-event': AnalyticsRawTrackEvent; 'metrics:event': MetricsEvent; }; /** * feature-disabled: hooks return components that will be * rendered in place for Feature components when the feature is not enabled. */ export type FeatureDisabledHooks = { 'feature-disabled:alert-wizard-performance': FeatureDisabledHook; 'feature-disabled:alerts-page': FeatureDisabledHook; 'feature-disabled:codecov-integration-setting': FeatureDisabledHook; 'feature-disabled:custom-inbound-filters': FeatureDisabledHook; 'feature-disabled:dashboards-edit': FeatureDisabledHook; 'feature-disabled:dashboards-page': FeatureDisabledHook; 'feature-disabled:dashboards-sidebar-item': FeatureDisabledHook; 'feature-disabled:data-forwarding': FeatureDisabledHook; 'feature-disabled:discard-groups': FeatureDisabledHook; 'feature-disabled:discover-page': FeatureDisabledHook; 'feature-disabled:discover-saved-query-create': FeatureDisabledHook; 'feature-disabled:discover-sidebar-item': FeatureDisabledHook; 'feature-disabled:discover2-page': FeatureDisabledHook; 'feature-disabled:discover2-sidebar-item': FeatureDisabledHook; 'feature-disabled:events-page': FeatureDisabledHook; 'feature-disabled:events-sidebar-item': FeatureDisabledHook; 'feature-disabled:grid-editable-actions': FeatureDisabledHook; 'feature-disabled:incidents-sidebar-item': FeatureDisabledHook; 'feature-disabled:open-discover': FeatureDisabledHook; 'feature-disabled:open-in-discover': FeatureDisabledHook; 'feature-disabled:performance-new-project': FeatureDisabledHook; 'feature-disabled:performance-page': FeatureDisabledHook; 'feature-disabled:performance-quick-trace': FeatureDisabledHook; 'feature-disabled:performance-sidebar-item': FeatureDisabledHook; 'feature-disabled:profiling-page': FeatureDisabledHook; 'feature-disabled:profiling-sidebar-item': FeatureDisabledHook; 'feature-disabled:project-performance-score-card': FeatureDisabledHook; 'feature-disabled:project-selector-all-projects': FeatureDisabledHook; 'feature-disabled:project-selector-checkbox': FeatureDisabledHook; 'feature-disabled:rate-limits': FeatureDisabledHook; 'feature-disabled:relay': FeatureDisabledHook; 'feature-disabled:replay-sidebar-item': FeatureDisabledHook; 'feature-disabled:sso-basic': FeatureDisabledHook; 'feature-disabled:sso-saml2': FeatureDisabledHook; 'feature-disabled:starfish-view': FeatureDisabledHook; 'feature-disabled:trace-view-link': FeatureDisabledHook; }; /** * Interface chrome hooks. */ export type InterfaceChromeHooks = { footer: GenericComponentHook; 'help-modal:footer': HelpModalFooterHook; 'organization:header': OrganizationHeaderComponentHook; 'sidebar:bottom-items': SidebarBottomItemsHook; 'sidebar:help-menu': GenericOrganizationComponentHook; 'sidebar:item-label': SidebarItemLabelHook; 'sidebar:organization-dropdown-menu': GenericOrganizationComponentHook; 'sidebar:organization-dropdown-menu-bottom': GenericOrganizationComponentHook; }; /** * Onboarding experience hooks */ export type OnboardingHooks = { 'onboarding-wizard:skip-help': GenericOrganizationComponentHook; 'onboarding:block-hide-sidebar': () => boolean; 'onboarding:extra-chrome': GenericComponentHook; 'onboarding:targeted-onboarding-header': (opts: {source: string}) => React.ReactNode; }; /** * Settings navigation hooks. */ export type SettingsHooks = { 'settings:api-navigation-config': SettingsItemsHook; 'settings:organization-navigation': OrganizationSettingsHook; 'settings:organization-navigation-config': SettingsConfigHook; }; /** * Feature Specific Hooks */ export interface FeatureSpecificHooks extends SpendVisibilityHooks {} /** * Hooks related to Spend Visibitlity * (i.e. Per-Project Spike Protection + Spend Allocations) */ export type SpendVisibilityHooks = { 'spend-visibility:spike-protection-project-settings': GenericProjectComponentHook; }; /** * Hooks that are actually React Hooks as well */ export type ReactHooks = { 'react-hook:route-activated': ( props: RouteContextInterface ) => React.ContextType; 'react-hook:use-button-tracking': (props: ButtonProps) => () => void; 'react-hook:use-experiment': UseExperiment; }; /** * Callback hooks. * These hooks just call a function that has no return value * and perform some sort of callback logic */ type CallbackHooks = { 'callback:on-guide-update': GuideUpdateCallback; }; /** * Renders a React node with no props */ type GenericComponentHook = () => React.ReactNode; /** * A route hook provides an injection point for a list of routes. */ type RoutesHook = () => Route[]; /** * Receives an organization object and should return a React node. */ type GenericOrganizationComponentHook = (opts: { organization: Organization; }) => React.ReactNode; /** * Receives a project object and should return a React node. */ type GenericProjectComponentHook = (opts: {project: Project}) => React.ReactNode; // TODO(ts): We should correct the organization header hook to conform to the // GenericOrganizationComponentHook, passing org as a prop object, not direct // as the only argument. type OrganizationHeaderComponentHook = (org: Organization) => React.ReactNode; /** * A FeatureDisabledHook returns a react element when a feature is not enabled. */ type FeatureDisabledHook = (opts: { /** * Children can either be a node, or a function that accepts a renderDisabled prop containing * a function/component to render when the feature is not enabled. */ children: React.ReactNode | ChildrenRenderFn; /** * The list of features that are controlled by this hook. */ features: string[]; /** * Weather the feature is or is not enabled. */ hasFeature: boolean; /** * The organization that is associated to this feature. */ organization: Organization; /** * The project that is associated to this feature. */ project?: Project; }) => React.ReactNode; /** * Called when the app is mounted. */ type AnalyticsInitUser = (user: User) => void; /** * Trigger analytics tracking in the hook store. */ type AnalyticsRawTrackEvent = ( data: { /** * Arbitrary data to track */ [key: string]: any; /** * The Reload event key. */ eventKey: string; /** * The Amplitude event name. Set to null if event should not go to Amplitude. */ eventName: string | null; /** * Organization to pass in. If full org object not available, pass in just the Id. * If no org, pass in null. */ organization: Organization | string | null; }, options?: { /** * An arbitrary function to map the parameters to new parameters */ mapValuesFn?: (params: Record) => Record; /** * If true, starts an analytics session. This session can be used * to construct funnels. The start of the funnel should have * startSession set to true. */ startSession?: boolean; /** * Optional unix timestamp */ time?: number; } ) => void; /** * Trigger experiment observed logging. */ type AnalyticsLogExperiment = (opts: { /** * The experiment key */ key: ExperimentKey; /** * The organization. Must be provided for organization experiments. */ organization?: Organization; }) => void; /** * Trigger recording a metric in the hook store. */ type MetricsEvent = ( /** * Metric name */ name: string, /** * Value to record for this metric */ value: number, /** * An additional tags object */ tags?: object ) => void; /** * Provides additional navigation components */ type OrganizationSettingsHook = (organization: Organization) => React.ReactElement; /** * Provides additional setting configurations */ type SettingsConfigHook = (organization: Organization) => NavigationSection; /** * Provides additional setting navigation items */ type SettingsItemsHook = (organization?: Organization) => NavigationItem[]; /** * Each sidebar label is wrapped with this hook, to allow sidebar item * augmentation. */ type SidebarItemLabelHook = () => React.ComponentType<{ /** * The item label being wrapped */ children: React.ReactNode; /** * The key of the item label currently being rendered. If no id is provided * the hook will have no effect. */ id?: string; }>; type SidebarProps = Pick< React.ComponentProps, 'orientation' | 'collapsed' | 'hasPanel' >; /** * Returns an additional list of sidebar items. */ type SidebarBottomItemsHook = ( opts: SidebarProps & {organization: Organization} ) => React.ReactNode; /** * Provides augmentation of the help modal footer */ type HelpModalFooterHook = (opts: { closeModal: () => void; organization: Organization; }) => React.ReactNode; /** * The DecoratedIntegrationFeature differs from the IntegrationFeature as it is * expected to have been transformed into marked up content. */ type DecoratedIntegrationFeature = { /** * Marked up description */ description: React.ReactNode; featureGate: string; }; type IntegrationFeatureGroup = { /** * The list of features within this group */ features: DecoratedIntegrationFeature[]; /** * Weather the group has all of the features enabled within this group * or not. */ hasFeatures: boolean; }; type FeatureGateSharedProps = { /** * The list of features, typically this is provided by the backend. */ features: DecoratedIntegrationFeature[]; /** * Organization of the integration we're querying feature gate details for. */ organization: Organization; }; type IntegrationFeaturesProps = FeatureGateSharedProps & { /** * The children function which will be provided with gating details. */ children: (opts: { /** * Is the integration disabled for installation because of feature gating? */ disabled: boolean; /** * The translated reason that the integration is disabled. */ disabledReason: React.ReactNode; /** * Features grouped based on specific gating criteria (for example, in * sentry.io this is features grouped by plans). */ gatedFeatureGroups: IntegrationFeatureGroup[]; /** * This is the list of features which have *not* been gated in any way. */ ungatedFeatures: DecoratedIntegrationFeature[]; }) => React.ReactElement; }; type IntegrationFeatureListProps = FeatureGateSharedProps & { provider: Pick; }; /** * The integration features gate hook provides components to customize * integration feature lists. */ type IntegrationsFeatureGatesHook = () => { /** * This component renders the list of integration features. */ FeatureList: React.ComponentType; /** * This is a render-prop style component that given a set of integration * features will call the children function with gating details about the * features. */ IntegrationFeatures: React.ComponentType; }; /** * Invite Button customization allows for a render-props component to replace * or intercept props of the button element. */ type InviteButtonCustomizationHook = () => React.ComponentType<{ children: (opts: { /** * Whether the Invite Members button is active or not */ disabled: boolean; onTriggerModal: () => void; }) => React.ReactElement; onTriggerModal: () => void; organization: Organization; }>; /** * Invite Modal customization allows for a render-prop component to add * additional react elements into the modal, and add invite-send middleware. */ type InviteModalCustomizationHook = () => React.ComponentType<{ children: (opts: { /** * Indicates that the modal's send invites button should be enabled and * invites may currently be sent. */ canSend: boolean; /** * Trigger sending invites */ sendInvites: () => void; /** * Additional react elements to render in the header of the modal, just * under the description. */ headerInfo?: React.ReactNode; }) => React.ReactElement; /** * When the children's sendInvites renderProp is called, this will also be * triggered. */ onSendInvites: () => void; /** * The organization that members will be invited to. */ organization: Organization; /** * Indicates if clicking 'send invites' will immediately send invites, or * would just create invite requests. */ willInvite: boolean; }>;