* feat: add session timeout configs * Regenerate the api file. * Fix bunch of type errors caused by regenerating the api file. --------- Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
274 lines
10 KiB
TypeScript
274 lines
10 KiB
TypeScript
import { PermissionAction } from '@supabase/shared-types/out/constants'
|
|
|
|
import { useParams } from 'common'
|
|
import { observer } from 'mobx-react-lite'
|
|
import { useEffect, useState } from 'react'
|
|
import {
|
|
AlertDescription_Shadcn_,
|
|
AlertTitle_Shadcn_,
|
|
Alert_Shadcn_,
|
|
Button,
|
|
Form,
|
|
IconAlertCircle,
|
|
IconEye,
|
|
IconEyeOff,
|
|
Input,
|
|
InputNumber,
|
|
Radio,
|
|
Toggle,
|
|
} from 'ui'
|
|
import { boolean, number, object, string } from 'yup'
|
|
|
|
import {
|
|
FormActions,
|
|
FormHeader,
|
|
FormPanel,
|
|
FormSection,
|
|
FormSectionContent,
|
|
FormSectionLabel,
|
|
} from 'components/ui/Forms'
|
|
import UpgradeToPro from 'components/ui/UpgradeToPro'
|
|
import { useAuthConfigQuery } from 'data/auth/auth-config-query'
|
|
import { useAuthConfigUpdateMutation } from 'data/auth/auth-config-update-mutation'
|
|
import { useOrgSubscriptionQuery } from 'data/subscriptions/org-subscription-query'
|
|
import { useCheckPermissions, useSelectedOrganization, useStore } from 'hooks'
|
|
|
|
const schema = object({
|
|
DISABLE_SIGNUP: boolean().required(),
|
|
SITE_URL: string().required('Must have a Site URL'),
|
|
SECURITY_CAPTCHA_ENABLED: boolean().required(),
|
|
SECURITY_CAPTCHA_SECRET: string().when('SECURITY_CAPTCHA_ENABLED', {
|
|
is: true,
|
|
then: string().required('Must have a Captcha secret'),
|
|
}),
|
|
SECURITY_CAPTCHA_PROVIDER: string().when('SECURITY_CAPTCHA_ENABLED', {
|
|
is: true,
|
|
then: string()
|
|
.oneOf(['hcaptcha', 'turnstile'])
|
|
.required('Captcha provider must be either hcaptcha or turnstile'),
|
|
}),
|
|
SESSIONS_TIMEBOX: number().min(0, 'Must be a positive number'),
|
|
SESSIONS_INACTIVITY_TIMEOUT: number().min(0, 'Must be a positive number'),
|
|
})
|
|
|
|
function HoursOrNeverText({ value }: { value: number }) {
|
|
if (value === 0) {
|
|
return 'never'
|
|
} else if (value === 1) {
|
|
return 'hour'
|
|
} else {
|
|
return 'hours'
|
|
}
|
|
}
|
|
|
|
const BasicAuthSettingsForm = observer(() => {
|
|
const { ui } = useStore()
|
|
const { ref: projectRef } = useParams()
|
|
const {
|
|
data: authConfig,
|
|
error: authConfigError,
|
|
isLoading,
|
|
isError,
|
|
isSuccess,
|
|
} = useAuthConfigQuery({ projectRef })
|
|
const { mutate: updateAuthConfig, isLoading: isUpdatingConfig } = useAuthConfigUpdateMutation()
|
|
|
|
const formId = 'auth-config-basic-settings'
|
|
const [hidden, setHidden] = useState(true)
|
|
const canUpdateConfig = useCheckPermissions(PermissionAction.UPDATE, 'custom_config_gotrue')
|
|
|
|
const organization = useSelectedOrganization()
|
|
const { data: subscription, isSuccess: isSuccessSubscription } = useOrgSubscriptionQuery({
|
|
orgSlug: organization!.slug,
|
|
})
|
|
|
|
const isProPlanAndUp = isSuccessSubscription && subscription?.plan?.id !== 'free'
|
|
|
|
const INITIAL_VALUES = {
|
|
DISABLE_SIGNUP: !authConfig?.DISABLE_SIGNUP,
|
|
SITE_URL: authConfig?.SITE_URL,
|
|
SECURITY_CAPTCHA_ENABLED: authConfig?.SECURITY_CAPTCHA_ENABLED || false,
|
|
SECURITY_CAPTCHA_SECRET: authConfig?.SECURITY_CAPTCHA_SECRET || '',
|
|
SECURITY_CAPTCHA_PROVIDER: authConfig?.SECURITY_CAPTCHA_PROVIDER || 'hcaptcha',
|
|
SESSIONS_TIMEBOX: authConfig?.SESSIONS_TIMEBOX || 0,
|
|
SESSIONS_INACTIVITY_TIMEOUT: authConfig?.SESSIONS_INACTIVITY_TIMEOUT || 0,
|
|
}
|
|
|
|
const onSubmit = (values: any, { resetForm }: any) => {
|
|
const payload = { ...values }
|
|
payload.DISABLE_SIGNUP = !values.DISABLE_SIGNUP
|
|
|
|
updateAuthConfig(
|
|
{ projectRef: projectRef!, config: payload },
|
|
{
|
|
onError: (error) => {
|
|
ui.setNotification({
|
|
category: 'error',
|
|
message: `Failed to update settings: ${error?.message}`,
|
|
})
|
|
},
|
|
onSuccess: () => {
|
|
ui.setNotification({
|
|
category: 'success',
|
|
message: `Successfully updated settings`,
|
|
})
|
|
resetForm({ values: values, initialValues: values })
|
|
},
|
|
}
|
|
)
|
|
}
|
|
|
|
if (isError) {
|
|
return (
|
|
<Alert_Shadcn_ variant="destructive">
|
|
<IconAlertCircle strokeWidth={2} />
|
|
<AlertTitle_Shadcn_>Failed to retrieve auth configuration</AlertTitle_Shadcn_>
|
|
<AlertDescription_Shadcn_>{authConfigError.message}</AlertDescription_Shadcn_>
|
|
</Alert_Shadcn_>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<Form id={formId} initialValues={INITIAL_VALUES} onSubmit={onSubmit} validationSchema={schema}>
|
|
{({ handleReset, resetForm, values, initialValues }: any) => {
|
|
const hasChanges = JSON.stringify(values) !== JSON.stringify(initialValues)
|
|
|
|
// Form is reset once remote data is loaded in store
|
|
useEffect(() => {
|
|
if (isSuccess) resetForm({ values: INITIAL_VALUES, initialValues: INITIAL_VALUES })
|
|
}, [isSuccess])
|
|
|
|
return (
|
|
<>
|
|
<FormHeader
|
|
title="Auth Settings"
|
|
description="Configure security and user session settings."
|
|
/>
|
|
<FormPanel
|
|
disabled={true}
|
|
footer={
|
|
<div className="flex py-4 px-8">
|
|
<FormActions
|
|
form={formId}
|
|
isSubmitting={isUpdatingConfig}
|
|
hasChanges={hasChanges}
|
|
handleReset={handleReset}
|
|
disabled={!canUpdateConfig}
|
|
helper={
|
|
!canUpdateConfig
|
|
? 'You need additional permissions to update authentication settings'
|
|
: undefined
|
|
}
|
|
/>
|
|
</div>
|
|
}
|
|
>
|
|
<FormSection header={<FormSectionLabel>User Signups</FormSectionLabel>}>
|
|
<FormSectionContent loading={isLoading}>
|
|
<Toggle
|
|
id="DISABLE_SIGNUP"
|
|
size="small"
|
|
label="Allow new users to sign up"
|
|
layout="flex"
|
|
descriptionText="If this is disabled, new users will not be able to sign up to your application."
|
|
disabled={!canUpdateConfig}
|
|
/>
|
|
</FormSectionContent>
|
|
</FormSection>
|
|
<div className="border-t border-muted"></div>
|
|
<FormSection header={<FormSectionLabel>User Sessions</FormSectionLabel>}>
|
|
<FormSectionContent loading={isLoading}>
|
|
{isProPlanAndUp ? (
|
|
<></>
|
|
) : (
|
|
<UpgradeToPro
|
|
primaryText="Upgrade to Pro"
|
|
secondaryText="Configuring user sessions requires the Pro plan."
|
|
projectRef={projectRef!}
|
|
organizationSlug={organization!.slug}
|
|
/>
|
|
)}
|
|
<InputNumber
|
|
id="SESSIONS_TIMEBOX"
|
|
size="small"
|
|
label="Time-box user sessions"
|
|
descriptionText="How long before a user is forced to log in again, use 0 for never"
|
|
actions={
|
|
<span className="mr-3 text-foreground-lighter">
|
|
<HoursOrNeverText value={values.SESSIONS_TIMEBOX} />
|
|
</span>
|
|
}
|
|
disabled={!canUpdateConfig || !isProPlanAndUp}
|
|
/>
|
|
<InputNumber
|
|
id="SESSIONS_INACTIVITY_TIMEOUT"
|
|
size="small"
|
|
label="Inactivity timeout"
|
|
descriptionText="How long users need to be inactive to be forced to log in again, use 0 for never"
|
|
actions={
|
|
<span className="mr-3 text-foreground-lighter">
|
|
<HoursOrNeverText value={values.SESSIONS_INACTIVITY_TIMEOUT} />
|
|
</span>
|
|
}
|
|
disabled={!canUpdateConfig || !isProPlanAndUp}
|
|
/>
|
|
</FormSectionContent>
|
|
</FormSection>
|
|
<div className="border-t border-scale-400"></div>
|
|
<FormSection header={<FormSectionLabel>Bot and Abuse Protection</FormSectionLabel>}>
|
|
<FormSectionContent loading={isLoading}>
|
|
<Toggle
|
|
id="SECURITY_CAPTCHA_ENABLED"
|
|
size="small"
|
|
label="Enable Captcha protection"
|
|
layout="flex"
|
|
descriptionText="Protect authentication endpoints from bots and abuse."
|
|
disabled={!canUpdateConfig}
|
|
/>
|
|
{values.SECURITY_CAPTCHA_ENABLED && (
|
|
<>
|
|
<Radio.Group
|
|
id="SECURITY_CAPTCHA_PROVIDER"
|
|
name="SECURITY_CAPTCHA_PROVIDER"
|
|
label="Choose Captcha Provider"
|
|
>
|
|
<Radio
|
|
label="hCaptcha"
|
|
value="hcaptcha"
|
|
checked={values.SECURITY_CAPTCHA_PROVIDER === 'hcaptcha'}
|
|
/>
|
|
<Radio
|
|
label="Turnstile by Cloudflare"
|
|
value="turnstile"
|
|
checked={values.SECURITY_CAPTCHA_PROVIDER === 'turnstile'}
|
|
/>
|
|
</Radio.Group>
|
|
<Input
|
|
id="SECURITY_CAPTCHA_SECRET"
|
|
type={hidden ? 'password' : 'text'}
|
|
size="small"
|
|
label="Captcha secret"
|
|
descriptionText="Obtain this secret from the provider."
|
|
disabled={!canUpdateConfig}
|
|
actions={
|
|
<Button
|
|
icon={hidden ? <IconEye /> : <IconEyeOff />}
|
|
type="default"
|
|
onClick={() => setHidden(!hidden)}
|
|
/>
|
|
}
|
|
/>
|
|
</>
|
|
)}
|
|
</FormSectionContent>
|
|
</FormSection>
|
|
</FormPanel>
|
|
</>
|
|
)
|
|
}}
|
|
</Form>
|
|
)
|
|
})
|
|
|
|
export default BasicAuthSettingsForm
|