import {css} from '@emotion/react'; import styled from '@emotion/styled'; import sentryPattern from 'sentry-images/pattern/sentry-pattern.png'; import Alert from 'sentry/components/alert'; import {ApiForm} from 'sentry/components/forms'; import SentryDocumentTitle from 'sentry/components/sentryDocumentTitle'; import {t} from 'sentry/locale'; import ConfigStore from 'sentry/stores/configStore'; import space from 'sentry/styles/space'; import AsyncView from 'sentry/views/asyncView'; import {getForm, getOptionDefault, getOptionField} from '../options'; type Props = AsyncView['props'] & { onConfigured: () => void; }; type State = AsyncView['state']; export default class InstallWizard extends AsyncView { getEndpoints(): ReturnType { return [['data', '/internal/options/?query=is:required']]; } renderFormFields() { const options = this.state.data; let missingOptions = new Set( Object.keys(options).filter(option => !options[option].field.isSet) ); // This is to handle the initial installation case. // Even if all options are filled out, we want to prompt to confirm // them. This is a bit of a hack because we're assuming that // the backend only spit back all filled out options for // this case. if (missingOptions.size === 0) { missingOptions = new Set(Object.keys(options)); } // A mapping of option name to Field object const fields = {}; for (const key of missingOptions) { const option = options[key]; if (option.field.disabled) { continue; } fields[key] = getOptionField(key, option.field); } return getForm(fields); } getInitialData() { const options = this.state.data; const data = {}; Object.keys(options).forEach(optionName => { const option = options[optionName]; if (option.field.disabled) { return; } // TODO(dcramer): we need to rethink this logic as doing multiple "is this value actually set" // is problematic // all values to their server-defaults (as client-side defaults don't really work) const displayValue = option.value || getOptionDefault(optionName); if ( // XXX(dcramer): we need the user to explicitly choose beacon.anonymous // vs using an implied default so effectively this is binding optionName !== 'beacon.anonymous' && // XXX(byk): if we don't have a set value but have a default value filled // instead, from the client, set it on the data so it is sent to the server !option.field.isSet && displayValue !== undefined ) { data[optionName] = displayValue; } }); return data; } getTitle() { return t('Setup Sentry'); } render() { const version = ConfigStore.get('version'); return ( {t('Welcome to Sentry')} {version.current} {this.state.loading ? this.renderLoading() : this.state.error ? this.renderError() : this.renderBody()} ); } renderError() { return ( {t( 'We were unable to load the required configuration from the Sentry server. Please take a look at the service logs.' )} ); } renderBody() { return (

{t('Complete setup by filling out the required configuration.')}

{this.renderFormFields()}
); } } const Wrapper = styled('div')` display: flex; justify-content: center; `; const fixedStyle = css` position: fixed; top: 0; right: 0; bottom: 0; left: 0; `; const Pattern = styled('div')` z-index: -1; &::before { ${fixedStyle} content: ''; background-image: linear-gradient( to right, ${p => p.theme.purple200} 0%, ${p => p.theme.purple300} 100% ); background-repeat: repeat-y; } &::after { ${fixedStyle} content: ''; background: url(${sentryPattern}); background-size: 400px; opacity: 0.8; } `; const Heading = styled('h1')` display: grid; gap: ${space(1)}; justify-content: space-between; grid-auto-flow: column; line-height: 36px; `; const Version = styled('small')` font-size: ${p => p.theme.fontSizeExtraLarge}; line-height: inherit; `; const SetupWizard = styled('div')` background: ${p => p.theme.background}; border-radius: ${p => p.theme.borderRadius}; box-shadow: ${p => p.theme.dropShadowHeavy}; margin-top: 40px; padding: 40px 40px 20px; width: 600px; z-index: ${p => p.theme.zIndex.initial}; `;