import styled from '@emotion/styled';
import {openInviteMembersModal} from 'sentry/actionCreators/modal';
import {navigateTo} from 'sentry/actionCreators/navigation';
import {Client} from 'sentry/api';
import {taskIsDone} from 'sentry/components/onboardingWizard/utils';
import {filterProjects} from 'sentry/components/performanceOnboarding/utils';
import {sourceMaps} from 'sentry/data/platformCategories';
import {t} from 'sentry/locale';
import pulsingIndicatorStyles from 'sentry/styles/pulsingIndicator';
import space from 'sentry/styles/space';
import {
OnboardingSupplementComponentProps,
OnboardingTask,
OnboardingTaskDescriptor,
OnboardingTaskKey,
Organization,
Project,
} from 'sentry/types';
import EventWaiter from 'sentry/utils/eventWaiter';
import withApi from 'sentry/utils/withApi';
import {OnboardingState} from 'sentry/views/onboarding/types';
import OnboardingProjectsCard from './onboardingCard';
function hasPlatformWithSourceMaps(projects: Project[] | undefined) {
return projects !== undefined
? projects.some(({platform}) => platform && sourceMaps.includes(platform))
: false;
}
type FirstEventWaiterProps = OnboardingSupplementComponentProps & {
api: Client;
};
type Options = {
/**
* The organization to show onboarding tasks for
*/
organization: Organization;
onboardingState?: OnboardingState;
/**
* A list of the organizations projects. This is used for some onboarding
* tasks to show additional task details (such as for suggesting sourcemaps)
*/
projects?: Project[];
};
function getIssueAlertUrl({projects, organization}: Options) {
if (!projects || !projects.length) {
return `/organizations/${organization.slug}/alerts/rules/`;
}
// pick the first project with events if we have that, otherwise just pick the first project
const firstProjectWithEvents = projects.find(project => !!project.firstEvent);
const project = firstProjectWithEvents ?? projects[0];
return `/organizations/${organization.slug}/alerts/${project.slug}/wizard/`;
}
function getMetricAlertUrl({projects, organization}: Options) {
if (!projects || !projects.length) {
return `/organizations/${organization.slug}/alerts/rules/`;
}
// pick the first project with transaction events if we have that, otherwise just pick the first project
const firstProjectWithEvents = projects.find(
project => !!project.firstTransactionEvent
);
const project = firstProjectWithEvents ?? projects[0];
return `/organizations/${organization.slug}/alerts/${project.slug}/wizard/?alert_option=trans_duration`;
}
export function getOnboardingTasks({
organization,
projects,
onboardingState,
}: Options): OnboardingTaskDescriptor[] {
return [
{
task: OnboardingTaskKey.FIRST_PROJECT,
title: t('Create a project'),
description: t(
"Monitor in seconds by adding a simple lines of code to your project. It's as easy as microwaving leftover pizza."
),
skippable: false,
requisites: [],
actionType: 'app',
location: `/organizations/${organization.slug}/projects/new/`,
display: true,
},
{
task: OnboardingTaskKey.FIRST_EVENT,
title: t('Capture your first error'),
description: t(
"Time to test it out. Now that you've created a project, capture your first error. We've got an example you can fiddle with."
),
skippable: false,
requisites: [OnboardingTaskKey.FIRST_PROJECT],
actionType: 'app',
location: `/settings/${organization.slug}/projects/:projectId/install/`,
display: true,
SupplementComponent: withApi(({api, task, onCompleteTask}: FirstEventWaiterProps) =>
!!projects?.length && task.requisiteTasks.length === 0 && !task.completionSeen ? (
!taskIsDone(task) && onCompleteTask()}
>
{() => }
) : null
),
},
{
task: OnboardingTaskKey.INVITE_MEMBER,
title: t('Invite your team'),
description: t(
'Assign issues and comment on shared errors with coworkers so you always know who to blame when sh*t hits the fan.'
),
skippable: true,
requisites: [],
actionType: 'action',
action: () => openInviteMembersModal({source: 'onboarding_widget'}),
display: true,
},
{
task: OnboardingTaskKey.FIRST_INTEGRATION,
title: t('Install any of our 40+ integrations'),
description: t(
'Get alerted in Slack. Two-way sync issues between Sentry and Jira. Notify Sentry of releases from GitHub, Vercel, or Netlify.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'app',
location: `/settings/${organization.slug}/integrations/`,
display: true,
},
{
task: OnboardingTaskKey.SECOND_PLATFORM,
title: t('Create another project'),
description: t(
'Easy, right? Don’t stop at one. Set up another project and send it events to keep things running smoothly in both the frontend and backend.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'app',
location: `/organizations/${organization.slug}/projects/new/`,
display: true,
},
{
task: OnboardingTaskKey.FIRST_TRANSACTION,
title: t('Boost performance'),
description: t(
"Don't keep users waiting. Trace transactions, investigate spans and cross-reference related issues for those mission-critical endpoints."
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT],
actionType: 'action',
action: ({router}) => {
if (!organization.features?.includes('performance-onboarding-checklist')) {
window.open(
'https://docs.sentry.io/product/performance/getting-started/',
'_blank'
);
return;
}
// TODO: add analytics here for this specific action.
if (!projects) {
navigateTo(`/organizations/${organization.slug}/performance/`, router);
return;
}
const {projectsWithoutFirstTransactionEvent, projectsForOnboarding} =
filterProjects(projects);
if (projectsWithoutFirstTransactionEvent.length <= 0) {
navigateTo(`/organizations/${organization.slug}/performance/`, router);
return;
}
if (projectsForOnboarding.length) {
navigateTo(
`/organizations/${organization.slug}/performance/?project=${projectsForOnboarding[0].id}#performance-sidequest`,
router
);
return;
}
navigateTo(
`/organizations/${organization.slug}/performance/?project=${projectsWithoutFirstTransactionEvent[0].id}#performance-sidequest`,
router
);
},
display: true,
SupplementComponent: withApi(({api, task, onCompleteTask}: FirstEventWaiterProps) =>
!!projects?.length && task.requisiteTasks.length === 0 && !task.completionSeen ? (
!taskIsDone(task) && onCompleteTask()}
>
{() => }
) : null
),
},
{
task: OnboardingTaskKey.USER_CONTEXT,
title: t('Get more user context'),
description: t(
'Enable us to pinpoint which users are suffering from that bad code, so you can debug the problem more swiftly and maybe even apologize for it.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'external',
location:
'https://docs.sentry.io/platform-redirect/?next=/enriching-events/identify-user/',
display: true,
},
{
task: OnboardingTaskKey.RELEASE_TRACKING,
title: t('Track releases'),
description: t(
'Take an in-depth look at the health of each and every release with crash analytics, errors, related issues and suspect commits.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'app',
location: `/settings/${organization.slug}/projects/:projectId/release-tracking/`,
display: true,
},
{
task: OnboardingTaskKey.SOURCEMAPS,
title: t('Upload source maps'),
description: t(
"Deminify Javascript source code to debug with context. Seeing code in it's original form will help you debunk the ghosts of errors past."
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'external',
location: 'https://docs.sentry.io/platforms/javascript/sourcemaps/',
display: hasPlatformWithSourceMaps(projects),
},
{
task: OnboardingTaskKey.USER_REPORTS,
title: 'User crash reports',
description: t('Collect user feedback when your application crashes'),
skippable: true,
requisites: [
OnboardingTaskKey.FIRST_PROJECT,
OnboardingTaskKey.FIRST_EVENT,
OnboardingTaskKey.USER_CONTEXT,
],
actionType: 'app',
location: `/settings/${organization.slug}/projects/:projectId/user-reports/`,
display: false,
},
{
task: OnboardingTaskKey.ISSUE_TRACKER,
title: t('Set up issue tracking'),
description: t('Link to Sentry issues within your issue tracker'),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_EVENT],
actionType: 'app',
location: `/settings/${organization.slug}/projects/:projectId/plugins/`,
display: false,
},
{
task: OnboardingTaskKey.ALERT_RULE,
title: t('Configure an Issue Alert'),
description: t(
'We all have issues. Get real-time error notifications by setting up alerts for issues that match your set criteria.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT],
actionType: 'app',
location: getIssueAlertUrl({projects, organization, onboardingState}),
display: true,
},
{
task: OnboardingTaskKey.METRIC_ALERT,
title: t('Create a Performance Alert'),
description: t(
'See slow fast with performance alerts. Set up alerts for notifications about slow page load times, API latency, or when throughput significantly deviates from normal.'
),
skippable: true,
requisites: [OnboardingTaskKey.FIRST_PROJECT, OnboardingTaskKey.FIRST_TRANSACTION],
actionType: 'app',
location: getMetricAlertUrl({projects, organization, onboardingState}),
display: organization.features?.includes('incidents'),
},
{
task: OnboardingTaskKey.USER_SELECTED_PROJECTS,
title: t('Projects to Setup'),
description: '',
skippable: true,
requisites: [],
actionType: 'action',
action: () => {},
display: true,
renderCard: OnboardingProjectsCard,
},
];
}
export function getMergedTasks({organization, projects, onboardingState}: Options) {
const taskDescriptors = getOnboardingTasks({organization, projects, onboardingState});
const serverTasks = organization.onboardingTasks;
// Map server task state (i.e. completed status) with tasks objects
const allTasks = taskDescriptors.map(
desc =>
({
...desc,
...serverTasks.find(
serverTask =>
serverTask.task === desc.task || serverTask.task === desc.serverTask
),
requisiteTasks: [],
} as OnboardingTask)
);
// Map incomplete requisiteTasks as full task objects
return allTasks.map(task => ({
...task,
requisiteTasks: task.requisites
.map(key => allTasks.find(task2 => task2.task === key)!)
.filter(reqTask => reqTask.status !== 'complete'),
}));
}
const PulsingIndicator = styled('div')`
${pulsingIndicatorStyles};
margin-right: ${space(1)};
`;
const EventWaitingIndicator = styled((p: React.HTMLAttributes) => (
))`
display: flex;
align-items: center;
flex-grow: 1;
font-size: ${p => p.theme.fontSizeMedium};
color: ${p => p.theme.pink300};
`;