import 'prism-sentry/index.css';
import {Fragment, useCallback, useEffect, useState} from 'react';
import {browserHistory} from 'react-router';
import {css} from '@emotion/react';
import styled from '@emotion/styled';
import {AnimatePresence, motion} from 'framer-motion';
import * as qs from 'query-string';
import {loadDocs} from 'sentry/actionCreators/projects';
import Alert, {alertStyles} from 'sentry/components/alert';
import Button from 'sentry/components/button';
import ExternalLink from 'sentry/components/links/externalLink';
import LoadingError from 'sentry/components/loadingError';
import {PlatformKey} from 'sentry/data/platformCategories';
import platforms from 'sentry/data/platforms';
import {IconChevron} from 'sentry/icons';
import {t, tct} from 'sentry/locale';
import space from 'sentry/styles/space';
import {Organization, Project} from 'sentry/types';
import {logExperiment} from 'sentry/utils/analytics';
import trackAdvancedAnalyticsEvent from 'sentry/utils/analytics/trackAdvancedAnalyticsEvent';
import getDynamicText from 'sentry/utils/getDynamicText';
import {platformToIntegrationMap} from 'sentry/utils/integrationUtil';
import {Theme} from 'sentry/utils/theme';
import useApi from 'sentry/utils/useApi';
import withProjects from 'sentry/utils/withProjects';
import FirstEventFooter from './components/firstEventFooter';
import FullIntroduction from './components/fullIntroduction';
import ProjectSidebarSection from './components/projectSidebarSection';
import IntegrationSetup from './integrationSetup';
import {StepProps} from './types';
import {usePersistedOnboardingState} from './utils';
/**
* The documentation will include the following string should it be missing the
* verification example, which currently a lot of docs are.
*/
const INCOMPLETE_DOC_FLAG = 'TODO-ADD-VERIFICATION-EXAMPLE';
type PlatformDoc = {html: string; link: string; wizardSetup: string};
type Props = {
projects: Project[];
search: string;
loadingProjects?: boolean;
} & StepProps;
function ProjecDocs(props: {
hasError: boolean;
onRetry: () => void;
organization: Organization;
platform: PlatformKey | null;
platformDocs: PlatformDoc | null;
project: Project;
}) {
const testOnlyAlert = (
Platform documentation is not rendered in for tests in CI
);
const missingExampleWarning = () => {
const missingExample =
props.platformDocs && props.platformDocs.html.includes(INCOMPLETE_DOC_FLAG);
if (!missingExample) {
return null;
}
return (
{tct(
`Looks like this getting started example is still undergoing some
work and doesn't include an example for triggering an event quite
yet. If you have trouble sending your first event be sure to consult
the [docsLink:full documentation] for [platform].`,
{
docsLink: ,
platform: platforms.find(p => p.id === props.platform)?.name,
}
)}
);
};
useEffect(() => {
props.platformDocs?.wizardSetup &&
logExperiment({
key: 'OnboardingHighlightWizardExperiment',
organization: props.organization,
});
}, [props.organization, props.platformDocs?.wizardSetup]);
const showWizardSetup =
props.organization.experiments.OnboardingHighlightWizardExperiment;
const [wizardSetupDetailsCollapsed, setWizardSetupDetailsCollapsed] = useState(true);
const [interacted, setInteracted] = useState(false);
const docs =
props.platformDocs !== null &&
(showWizardSetup && props.platformDocs.wizardSetup ? (
{
!interacted &&
trackAdvancedAnalyticsEvent('growth.onboarding_wizard_interacted', {
organization: props.organization,
project_id: props.project.id,
platform: props.platform || 'unknown',
wizard_instructions: true,
});
setInteracted(true);
}}
/>
{!wizardSetupDetailsCollapsed && (
{missingExampleWarning()}
)}
) : (
{
!interacted &&
trackAdvancedAnalyticsEvent('growth.onboarding_wizard_interacted', {
organization: props.organization,
project_id: props.project.id,
platform: props.platform || undefined,
wizard_instructions: false,
});
setInteracted(true);
}}
/>
{missingExampleWarning()}
));
const loadingError = (
);
const currentPlatform = props.platform ?? props.project?.platform ?? 'other';
return (
{getDynamicText({
value: !props.hasError ? docs : loadingError,
fixed: testOnlyAlert,
})}
);
}
function SetupDocs({
organization,
projects: rawProjects,
search,
loadingProjects,
}: Props) {
const api = useApi();
const [clientState, setClientState] = usePersistedOnboardingState();
const selectedPlatforms = clientState?.selectedPlatforms || [];
const platformToProjectIdMap = clientState?.platformToProjectIdMap || {};
// id is really slug here
const projectSlugs = selectedPlatforms
.map(platform => platformToProjectIdMap[platform])
.filter((slug): slug is string => slug !== undefined);
const selectedProjectsSet = new Set(projectSlugs);
// get projects in the order they appear in selectedPlatforms
const projects = projectSlugs
.map(slug => rawProjects.find(project => project.slug === slug))
.filter((project): project is Project => project !== undefined);
// SDK instrumentation
const [hasError, setHasError] = useState(false);
const [platformDocs, setPlatformDocs] = useState(null);
const [loadedPlatform, setLoadedPlatform] = useState(null);
// store what projects have sent first event in state based project.firstEvent
const [hasFirstEventMap, setHasFirstEventMap] = useState>(
projects.reduce((accum, project: Project) => {
accum[project.id] = !!project.firstEvent;
return accum;
}, {} as Record)
);
const checkProjectHasFirstEvent = (project: Project) => {
return !!hasFirstEventMap[project.id];
};
const {project_id: rawProjectId} = qs.parse(search);
const rawProjectIndex = projects.findIndex(p => p.id === rawProjectId);
const firstProjectNoError = projects.findIndex(
p => selectedProjectsSet.has(p.slug) && !checkProjectHasFirstEvent(p)
);
// Select a project based on search params. If non exist, use the first project without first event.
const projectIndex = rawProjectIndex >= 0 ? rawProjectIndex : firstProjectNoError;
const project = projects[projectIndex];
// find the next project that doesn't have a first event
const nextProject = projects.find(
(p, i) => i > projectIndex && !checkProjectHasFirstEvent(p)
);
const integrationSlug = project?.platform && platformToIntegrationMap[project.platform];
const [integrationUseManualSetup, setIntegrationUseManualSetup] = useState(false);
useEffect(() => {
// should not redirect if we don't have an active client state or projects aren't loaded
if (!clientState || loadingProjects) {
return;
}
if (
// If no projects remaining, then we can leave
!project
) {
browserHistory.push('/');
}
});
const currentPlatform = loadedPlatform ?? project?.platform ?? 'other';
const fetchData = useCallback(async () => {
// TODO: add better error handling logic
if (!project?.platform) {
return;
}
if (integrationSlug && !integrationUseManualSetup) {
setLoadedPlatform(project.platform);
setPlatformDocs(null);
setHasError(false);
return;
}
try {
const loadedDocs = await loadDocs(
api,
organization.slug,
project.slug,
project.platform
);
setPlatformDocs(loadedDocs);
setLoadedPlatform(project.platform);
setHasError(false);
} catch (error) {
setHasError(error);
throw error;
}
}, [project, api, organization, integrationSlug, integrationUseManualSetup]);
useEffect(() => {
fetchData();
}, [fetchData]);
if (!project) {
return null;
}
const setNewProject = (newProjectId: string) => {
setLoadedPlatform(null);
setPlatformDocs(null);
setHasError(false);
setIntegrationUseManualSetup(false);
const searchParams = new URLSearchParams({
sub_step: 'project',
project_id: newProjectId,
});
browserHistory.push(`${window.location.pathname}?${searchParams}`);
clientState &&
setClientState({
...clientState,
state: 'projects_selected',
url: `setup-docs/?${searchParams}`,
});
};
const selectProject = (newProjectId: string) => {
const matchedProject = projects.find(p => p.id === newProjectId);
trackAdvancedAnalyticsEvent('growth.onboarding_clicked_project_in_sidebar', {
organization,
platform: matchedProject?.platform || 'unknown',
});
setNewProject(newProjectId);
};
return (
[
platform,
platformToProjectIdMap[platform],
])
)}
activeProject={project}
{...{checkProjectHasFirstEvent, selectProject}}
/>
{integrationSlug && !integrationUseManualSetup ? (
{
setIntegrationUseManualSetup(true);
}}
/>
) : (
)}
{project && (
{
const orgIssuesURL = `/organizations/${organization.slug}/issues/?project=${project.id}`;
trackAdvancedAnalyticsEvent(
'growth.onboarding_clicked_setup_platform_later',
{
organization,
platform: currentPlatform,
project_index: projectIndex,
}
);
if (!project.platform || !clientState) {
browserHistory.push(orgIssuesURL);
return;
}
// if we have a next project, switch to that
if (nextProject) {
setNewProject(nextProject.id);
} else {
setClientState({
...clientState,
state: 'finished',
});
browserHistory.push(orgIssuesURL);
}
}}
handleFirstIssueReceived={() => {
const newHasFirstEventMap = {...hasFirstEventMap, [project.id]: true};
setHasFirstEventMap(newHasFirstEventMap);
}}
/>
)}
);
}
export default withProjects(SetupDocs);
type AlertType = React.ComponentProps['type'];
const getAlertSelector = (type: AlertType) =>
type === 'muted' ? null : `.alert[level="${type}"], .alert-${type}`;
const mapAlertStyles = (p: {theme: Theme}, type: AlertType) =>
css`
${getAlertSelector(type)} {
${alertStyles({theme: p.theme, type})};
display: block;
}
`;
const AnimatedContentWrapper = styled(motion.div)`
overflow: hidden;
`;
AnimatedContentWrapper.defaultProps = {
initial: {
height: 0,
},
animate: {
height: 'auto',
},
exit: {
height: 0,
},
};
const Content = styled(motion.div)`
h1,
h2,
h3,
h4,
h5,
h6,
p {
margin-bottom: 18px;
}
div[data-language] {
margin-bottom: ${space(2)};
}
code {
font-size: 87.5%;
color: ${p => p.theme.pink300};
}
pre code {
color: inherit;
font-size: inherit;
white-space: pre;
}
h2 {
font-size: 1.4em;
}
.alert h5 {
font-size: 1em;
margin-bottom: 0.625rem;
}
/**
* XXX(epurkhiser): This comes from the doc styles and avoids bottom margin issues in alerts
*/
.content-flush-bottom *:last-child {
margin-bottom: 0;
}
${p => Object.keys(p.theme.alert).map(type => mapAlertStyles(p, type as AlertType))}
`;
const DocsWrapper = styled(motion.div)``;
DocsWrapper.defaultProps = {
initial: {opacity: 0, y: 40},
animate: {opacity: 1, y: 0},
exit: {opacity: 0},
};
const Wrapper = styled('div')`
display: flex;
flex-direction: row;
margin: ${space(2)};
justify-content: center;
`;
const MainContent = styled('div')`
max-width: 850px;
min-width: 0;
flex-grow: 1;
`;
// the number icon will be space(2) + 30px to the left of the margin of center column
// so we need to offset the right margin by that much
// also hide the sidebar if the screen is too small
const SidebarWrapper = styled('div')`
margin: ${space(1)} calc(${space(2)} + 30px + ${space(4)}) 0 ${space(2)};
@media (max-width: 1150px) {
display: none;
}
flex-basis: 240px;
flex-grow: 0;
flex-shrink: 0;
min-width: 240px;
`;