mirror of
				https://github.com/grafana/grafana.git
				synced 2025-02-25 18:55:37 -06:00 
			
		
		
		
	* fix: email and username trimming and invitation validation * Trim leading and trailing whitespaces from email and username on signup * Check whether the provided email address is the same as where the invitation sent * Align tests Co-authored-by: Mihaly Gyongyosi <mgyongyosi@users.noreply.github.com>
		
			
				
	
	
		
			112 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			112 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import React, { FC, useState } from 'react';
 | |
| import { useAsync } from 'react-use';
 | |
| 
 | |
| import { getBackendSrv } from '@grafana/runtime';
 | |
| import { Button, Field, Form, Input } from '@grafana/ui';
 | |
| import { Page } from 'app/core/components/Page/Page';
 | |
| import { getConfig } from 'app/core/config';
 | |
| import { contextSrv } from 'app/core/core';
 | |
| import { GrafanaRouteComponentProps } from 'app/core/navigation/types';
 | |
| 
 | |
| import { w3cStandardEmailValidator } from '../admin/utils';
 | |
| 
 | |
| interface FormModel {
 | |
|   email: string;
 | |
|   name?: string;
 | |
|   username: string;
 | |
|   password?: string;
 | |
| }
 | |
| 
 | |
| const navModel = {
 | |
|   main: {
 | |
|     icon: 'grafana' as const,
 | |
|     text: 'Invite',
 | |
|     subTitle: 'Register your Grafana account',
 | |
|     breadcrumbs: [{ title: 'Login', url: 'login' }],
 | |
|   },
 | |
|   node: {
 | |
|     text: '',
 | |
|   },
 | |
| };
 | |
| 
 | |
| export interface Props extends GrafanaRouteComponentProps<{ code: string }> {}
 | |
| 
 | |
| export const SignupInvitedPage: FC<Props> = ({ match }) => {
 | |
|   const code = match.params.code;
 | |
|   const [initFormModel, setInitFormModel] = useState<FormModel>();
 | |
|   const [greeting, setGreeting] = useState<string>();
 | |
|   const [invitedBy, setInvitedBy] = useState<string>();
 | |
| 
 | |
|   useAsync(async () => {
 | |
|     const invite = await getBackendSrv().get(`/api/user/invite/${code}`);
 | |
| 
 | |
|     setInitFormModel({
 | |
|       email: invite.email,
 | |
|       name: invite.name,
 | |
|       username: invite.email,
 | |
|     });
 | |
| 
 | |
|     setGreeting(invite.name || invite.email || invite.username);
 | |
|     setInvitedBy(invite.invitedBy);
 | |
|   }, [code]);
 | |
| 
 | |
|   const onSubmit = async (formData: FormModel) => {
 | |
|     await getBackendSrv().post('/api/user/invite/complete', { ...formData, inviteCode: code });
 | |
|     window.location.href = getConfig().appSubUrl + '/';
 | |
|   };
 | |
| 
 | |
|   if (!initFormModel) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <Page navModel={navModel}>
 | |
|       <Page.Contents>
 | |
|         <h3 className="page-sub-heading">Hello {greeting || 'there'}.</h3>
 | |
| 
 | |
|         <div className="modal-tagline p-b-2">
 | |
|           <em>{invitedBy || 'Someone'}</em> has invited you to join Grafana and the organization{' '}
 | |
|           <span className="highlight-word">{contextSrv.user.orgName}</span>
 | |
|           <br />
 | |
|           Please complete the following and choose a password to accept your invitation and continue:
 | |
|         </div>
 | |
|         <Form defaultValues={initFormModel} onSubmit={onSubmit}>
 | |
|           {({ register, errors }) => (
 | |
|             <>
 | |
|               <Field invalid={!!errors.email} error={errors.email && errors.email.message} label="Email">
 | |
|                 <Input
 | |
|                   placeholder="email@example.com"
 | |
|                   {...register('email', {
 | |
|                     required: 'Email is required',
 | |
|                     pattern: {
 | |
|                       value: w3cStandardEmailValidator,
 | |
|                       message: 'Email is invalid',
 | |
|                     },
 | |
|                   })}
 | |
|                 />
 | |
|               </Field>
 | |
|               <Field invalid={!!errors.name} error={errors.name && errors.name.message} label="Name">
 | |
|                 <Input placeholder="Name (optional)" {...register('name')} />
 | |
|               </Field>
 | |
|               <Field invalid={!!errors.username} error={errors.username && errors.username.message} label="Username">
 | |
|                 <Input {...register('username', { required: 'Username is required' })} placeholder="Username" />
 | |
|               </Field>
 | |
|               <Field invalid={!!errors.password} error={errors.password && errors.password.message} label="Password">
 | |
|                 <Input
 | |
|                   {...register('password', { required: 'Password is required' })}
 | |
|                   type="password"
 | |
|                   placeholder="Password"
 | |
|                 />
 | |
|               </Field>
 | |
| 
 | |
|               <Button type="submit">Sign up</Button>
 | |
|             </>
 | |
|           )}
 | |
|         </Form>
 | |
|       </Page.Contents>
 | |
|     </Page>
 | |
|   );
 | |
| };
 | |
| 
 | |
| export default SignupInvitedPage;
 |