types#User TypeScript Examples
The following examples show how to use
types#User.
You can vote up the ones you like or vote down the ones you don't like,
and go to the original project or source file by following the links above each example. You may check out the related API usage on the sidebar.
Example #1
Source File: AuthSlice.tsx From knboard with MIT License | 6 votes |
register = createAsyncThunk<
User,
RegisterProps,
{
rejectValue: ValidationErrors;
}
>("auth/registerStatus", async (credentials, { rejectWithValue }) => {
try {
// Don't POST blank email
if (!credentials["email"]) {
delete credentials["email"];
}
const response = await api.post(API_REGISTER, credentials);
return response.data;
} catch (err) {
const error: AxiosError<ValidationErrors> = err;
if (!error.response) {
throw err;
}
return rejectWithValue(error.response.data);
}
})
Example #2
Source File: AuthSlice.tsx From knboard with MIT License | 6 votes |
guestRegister = createAsyncThunk<User>(
"auth/guestRegisterStatus",
async (_, { dispatch }) => {
try {
const response = await api.post(API_GUEST_REGISTER);
return response.data;
} catch (e) {
dispatch(
createErrorToast("Failed to enter as a guest, try again later.")
);
}
}
)
Example #3
Source File: AuthSlice.tsx From knboard with MIT License | 6 votes |
login = createAsyncThunk<
User,
LoginProps,
{
rejectValue: ValidationErrors;
}
>("auth/loginStatus", async (credentials, { rejectWithValue }) => {
try {
const response = await api.post(API_LOGIN, credentials);
return response.data;
} catch (err) {
const error: AxiosError<ValidationErrors> = err;
if (!error.response) {
throw err;
}
return rejectWithValue(error.response.data);
}
})
Example #4
Source File: Auth.test.tsx From knboard with MIT License | 5 votes |
steveAuthUser: User = {
id: 1,
username: "steve",
photo_url: null,
}
Example #5
Source File: AuthContextProvider.tsx From crossfeed with Creative Commons Zero v1.0 Universal | 4 votes |
AuthContextProvider: React.FC = ({ children }) => {
const [authUser, setAuthUser] = useState<AuthUser | null>(null);
const [token, setToken] = usePersistentState<string | null>('token', null);
const [org, setOrg] = usePersistentState<
Organization | OrganizationTag | null
>('organization', null);
const [showAllOrganizations, setShowAllOrganizations] = usePersistentState<
boolean
>('showAllOrganizations', false);
const [feedbackMessage, setFeedbackMessage] = useState<{
message: string;
type: AlertProps['severity'];
} | null>(null);
const cookies = useMemo(() => new Cookies(), []);
const logout = useCallback(async () => {
const shouldReload = !!token;
localStorage.clear();
await Auth.signOut();
cookies.remove('crossfeed-token', {
domain: process.env.REACT_APP_COOKIE_DOMAIN
});
if (shouldReload) {
// Refresh the page only if the token was previously defined
// (i.e. it is now invalid / has expired now).
window.location.reload();
}
}, [cookies, token]);
const handleError = useCallback(
async (e: Error) => {
if (e.message.includes('401')) {
// Unauthorized, log out user
await logout();
}
},
[logout]
);
const api = useApi(handleError);
const { apiGet, apiPost } = api;
const getProfile = useCallback(async () => {
const user: User = await apiGet<User>('/users/me');
setAuthUser({
...user,
isRegistered: user.firstName !== ''
});
}, [setAuthUser, apiGet]);
const setProfile = useCallback(
async (user: User) => {
setAuthUser({
...user,
isRegistered: user.firstName !== ''
});
},
[setAuthUser]
);
const refreshUser = useCallback(async () => {
if (!token && process.env.REACT_APP_USE_COGNITO) {
const session = await Auth.currentSession();
const { token } = await apiPost<{ token: string; user: User }>(
'/auth/callback',
{
body: {
token: session.getIdToken().getJwtToken()
}
}
);
setToken(token);
}
}, [apiPost, setToken, token]);
const extendedOrg = useMemo(() => {
return getExtendedOrg(org, authUser);
}, [org, authUser]);
const maximumRole = useMemo(() => {
return getMaximumRole(authUser);
}, [authUser]);
const touVersion = useMemo(() => {
return getTouVersion(maximumRole);
}, [maximumRole]);
const userMustSign = useMemo(() => {
return getUserMustSign(authUser, touVersion);
}, [authUser, touVersion]);
useEffect(() => {
refreshUser();
// eslint-disable-next-line
}, []);
useEffect(() => {
if (!token) {
setAuthUser(null);
} else {
getProfile();
}
}, [token, getProfile]);
return (
<AuthContext.Provider
value={{
user: authUser,
token,
setUser: setProfile,
refreshUser,
setOrganization: setOrg,
currentOrganization: extendedOrg,
showAllOrganizations: showAllOrganizations,
setShowAllOrganizations: setShowAllOrganizations,
login: setToken,
logout,
setLoading: () => {},
maximumRole,
touVersion,
userMustSign,
setFeedbackMessage,
...api
}}
>
{api.loading && (
<div className="cisa-crossfeed-loading">
<div></div>
<div></div>
</div>
)}
{feedbackMessage && (
<Snackbar
open={!!feedbackMessage}
autoHideDuration={5000}
onClose={() => setFeedbackMessage(null)}
>
<Alert
onClose={() => setFeedbackMessage(null)}
severity={feedbackMessage.type}
>
{feedbackMessage.message}
</Alert>
</Snackbar>
)}
{children}
</AuthContext.Provider>
);
}
Example #6
Source File: AuthCreateAccount.tsx From crossfeed with Creative Commons Zero v1.0 Universal | 4 votes |
AuthCreateAccount: React.FC = () => {
const history = useHistory();
const [values, setValues] = useState<FormData>({
firstName: '',
lastName: '',
organization: undefined
});
const [errors, setErrors] = useState<Errors>({});
const [publicOrgs, setPublicOrgs] = useState<{ name: string; id: string }[]>(
[]
);
const { user, setUser, apiGet, apiPut } = useAuthContext();
const fetchPublicOrgs = useCallback(async () => {
const publicOrgs = await apiGet('/organizations/public');
publicOrgs.unshift({
name: 'None',
id: ''
});
setPublicOrgs(publicOrgs);
if (publicOrgs.length > 0)
setValues((values) => ({
...values,
organization: publicOrgs[0].id
}));
}, [apiGet, setValues]);
useEffect(() => {
fetchPublicOrgs();
}, [fetchPublicOrgs]);
const onTextChange: React.ChangeEventHandler<
HTMLInputElement | HTMLSelectElement
> = (e) => onChange(e.target.name, e.target.value);
const onChange = (name: string, value: any) => {
setValues((values) => ({
...values,
[name]: value
}));
};
const onSubmit: React.FormEventHandler = async (e) => {
e.preventDefault();
try {
if (!user) throw Error('Unable to register');
const updated: User = await apiPut(`/users/${user.id}`, {
body: values
});
setUser(updated);
history.push('/', {
message: 'Your account has been successfully created.'
});
} catch (e) {
setErrors({
global: e.message ?? e.toString()
});
}
};
return (
<AuthForm onSubmit={onSubmit}>
<h2>Crossfeed is currently in private beta.</h2>
<p>
Your organization admininistrator will need to invite you to Crossfeed
in order for you to begin using it.
</p>
<p>
Is your organization not yet on Crossfeed? Contact{' '}
<a href="# ">[email protected]</a> to learn more.
</p>
{(process.env.NODE_ENV === 'development' ||
(user && user.userType === 'globalAdmin')) && (
<>
<h1>[DEVELOPMENT, FOR GLOBAL ADMINS ONLY] Finish Creating Account</h1>
<p>
We need just a few more details from you in order to finish creating
your account.
</p>
<Label htmlFor="firstName">First Name</Label>
<TextInput
required
id="firstName"
name="firstName"
type="text"
value={values.firstName}
onChange={onTextChange}
/>
<Label htmlFor="lastName">Last Name</Label>
<TextInput
required
id="lastName"
name="lastName"
type="text"
value={values.lastName}
onChange={onTextChange}
/>
<Label htmlFor="email">Email</Label>
<TextInput
disabled
id="email"
name="email"
type="text"
value={user ? user.email : ''}
/>
<Label htmlFor="organization">
Select an organization to join. Your request will need to be
approved before joining.
</Label>
<Dropdown
id="organization"
name="organization"
onChange={onTextChange}
value={values.organization}
>
{publicOrgs.map((org) => {
return (
<option key={org.id} value={org.id}>
{org.name}
</option>
);
})}
</Dropdown>
<div className="width-full display-flex flex-justify-start">
{errors.global && <p className="text-error">{errors.global}</p>}
</div>
<div className="display-flex flex-justify-start flex-align-center width-full margin-top-3">
<Button type="submit">Submit</Button>
</div>
</>
)}
</AuthForm>
);
}
Example #7
Source File: Organization.tsx From crossfeed with Creative Commons Zero v1.0 Universal | 4 votes |
Organization: React.FC = () => {
const {
apiGet,
apiPut,
apiPost,
user,
setFeedbackMessage
} = useAuthContext();
const { organizationId } = useParams<{ organizationId: string }>();
const [organization, setOrganization] = useState<OrganizationType>();
const [tags, setTags] = useState<AutocompleteType[]>([]);
const [userRoles, setUserRoles] = useState<Role[]>([]);
const [scanTasks, setScanTasks] = useState<ScanTask[]>([]);
const [scans, setScans] = useState<Scan[]>([]);
const [scanSchema, setScanSchema] = useState<ScanSchema>({});
const [newUserValues, setNewUserValues] = useState<{
firstName: string;
lastName: string;
email: string;
organization?: OrganizationType;
role: string;
}>({
firstName: '',
lastName: '',
email: '',
role: ''
});
const classes = useStyles();
const [tagValue, setTagValue] = React.useState<AutocompleteType | null>(null);
const [inputValue, setInputValue] = React.useState('');
const [dialog, setDialog] = React.useState<{
open: boolean;
type?: 'rootDomains' | 'ipBlocks' | 'tags';
label?: string;
}>({ open: false });
const dateAccessor = (date?: string) => {
return !date || new Date(date).getTime() === new Date(0).getTime()
? 'None'
: `${formatDistanceToNow(parseISO(date))} ago`;
};
const userRoleColumns: Column<Role>[] = [
{
Header: 'Name',
accessor: ({ user }) => user.fullName,
width: 200,
disableFilters: true,
id: 'name'
},
{
Header: 'Email',
accessor: ({ user }) => user.email,
width: 150,
minWidth: 150,
id: 'email',
disableFilters: true
},
{
Header: 'Role',
accessor: ({ approved, role, user }) => {
if (approved) {
if (user.invitePending) {
return 'Invite pending';
} else if (role === 'admin') {
return 'Administrator';
} else {
return 'Member';
}
}
return 'Pending approval';
},
width: 50,
minWidth: 50,
id: 'approved',
disableFilters: true
},
{
Header: () => {
return (
<div style={{ justifyContent: 'flex-center' }}>
<Button color="secondary" onClick={() => setDialog({ open: true })}>
<ControlPoint style={{ marginRight: '10px' }}></ControlPoint>
Add member
</Button>
</div>
);
},
id: 'action',
Cell: ({ row }: { row: { index: number } }) => {
const isApproved =
!organization?.userRoles[row.index] ||
organization?.userRoles[row.index].approved;
return (
<>
{isApproved ? (
<Button
onClick={() => {
removeUser(row.index);
}}
color="secondary"
>
<p>Remove</p>
</Button>
) : (
<Button
onClick={() => {
approveUser(row.index);
}}
color="secondary"
>
<p>Approve</p>
</Button>
)}
</>
);
},
disableFilters: true
}
];
const scanColumns: Column<Scan>[] = [
{
Header: 'Name',
accessor: 'name',
width: 150,
id: 'name',
disableFilters: true
},
{
Header: 'Description',
accessor: ({ name }) => scanSchema[name] && scanSchema[name].description,
width: 200,
minWidth: 200,
id: 'description',
disableFilters: true
},
{
Header: 'Mode',
accessor: ({ name }) =>
scanSchema[name] && scanSchema[name].isPassive ? 'Passive' : 'Active',
width: 150,
minWidth: 150,
id: 'mode',
disableFilters: true
},
{
Header: 'Action',
id: 'action',
maxWidth: 100,
Cell: ({ row }: { row: { index: number } }) => {
if (!organization) return;
const enabled = organization.granularScans.find(
(scan) => scan.id === scans[row.index].id
);
return (
<Button
type="button"
onClick={() => {
updateScan(scans[row.index], !enabled);
}}
>
{enabled ? 'Disable' : 'Enable'}
</Button>
);
},
disableFilters: true
}
];
const scanTaskColumns: Column<ScanTask>[] = [
{
Header: 'ID',
accessor: 'id',
disableFilters: true
},
{
Header: 'Status',
accessor: 'status',
disableFilters: true
},
{
Header: 'Type',
accessor: 'type',
disableFilters: true
},
{
Header: 'Name',
accessor: ({ scan }) => scan?.name,
disableFilters: true
},
{
Header: 'Created At',
accessor: ({ createdAt }) => dateAccessor(createdAt),
disableFilters: true,
disableSortBy: true
},
{
Header: 'Requested At',
accessor: ({ requestedAt }) => dateAccessor(requestedAt),
disableFilters: true,
disableSortBy: true
},
{
Header: 'Started At',
accessor: ({ startedAt }) => dateAccessor(startedAt),
disableFilters: true,
disableSortBy: true
},
{
Header: 'Finished At',
accessor: ({ finishedAt }) => dateAccessor(finishedAt),
disableFilters: true,
disableSortBy: true
},
{
Header: 'Output',
accessor: 'output',
disableFilters: true
}
];
const fetchOrganization = useCallback(async () => {
try {
const organization = await apiGet<OrganizationType>(
`/organizations/${organizationId}`
);
organization.scanTasks.sort(
(a, b) =>
new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime()
);
setOrganization(organization);
setUserRoles(organization.userRoles);
setScanTasks(organization.scanTasks);
const tags = await apiGet<OrganizationTag[]>(`/organizations/tags`);
setTags(tags);
} catch (e) {
console.error(e);
}
}, [apiGet, setOrganization, organizationId]);
const fetchScans = useCallback(async () => {
try {
const response = await apiGet<{
scans: Scan[];
schema: ScanSchema;
}>('/granularScans/');
let { scans } = response;
const { schema } = response;
if (user?.userType !== 'globalAdmin')
scans = scans.filter(
(scan) =>
scan.name !== 'censysIpv4' && scan.name !== 'censysCertificates'
);
setScans(scans);
setScanSchema(schema);
} catch (e) {
console.error(e);
}
}, [apiGet, user]);
const approveUser = async (user: number) => {
try {
await apiPost(
`/organizations/${organization?.id}/roles/${organization?.userRoles[user].id}/approve`,
{ body: {} }
);
const copy = userRoles.map((role, id) =>
id === user ? { ...role, approved: true } : role
);
setUserRoles(copy);
} catch (e) {
console.error(e);
}
};
const removeUser = async (user: number) => {
try {
await apiPost(
`/organizations/${organization?.id}/roles/${userRoles[user].id}/remove`,
{ body: {} }
);
const copy = userRoles.filter((_, ind) => ind !== user);
setUserRoles(copy);
} catch (e) {
console.error(e);
}
};
const updateOrganization = async (body: any) => {
try {
const org = await apiPut('/organizations/' + organization?.id, {
body: organization
});
setOrganization(org);
setFeedbackMessage({
message: 'Organization successfully updated',
type: 'success'
});
} catch (e) {
setFeedbackMessage({
message:
e.status === 422
? 'Error updating organization'
: e.message ?? e.toString(),
type: 'error'
});
console.error(e);
}
};
const updateScan = async (scan: Scan, enabled: boolean) => {
try {
if (!organization) return;
await apiPost(
`/organizations/${organization?.id}/granularScans/${scan.id}/update`,
{
body: {
enabled
}
}
);
setOrganization({
...organization,
granularScans: enabled
? organization.granularScans.concat([scan])
: organization.granularScans.filter(
(granularScan) => granularScan.id !== scan.id
)
});
} catch (e) {
setFeedbackMessage({
message:
e.status === 422 ? 'Error updating scan' : e.message ?? e.toString(),
type: 'error'
});
console.error(e);
}
};
useEffect(() => {
fetchOrganization();
}, [fetchOrganization]);
const onInviteUserSubmit = async () => {
try {
const body = {
firstName: newUserValues.firstName,
lastName: newUserValues.lastName,
email: newUserValues.email,
organization: organization?.id,
organizationAdmin: newUserValues.role === 'admin'
};
const user: User = await apiPost('/users/', {
body
});
const newRole = user.roles[user.roles.length - 1];
newRole.user = user;
if (userRoles.find((role) => role.user.id === user.id)) {
setUserRoles(
userRoles.map((role) => (role.user.id === user.id ? newRole : role))
);
} else {
setUserRoles(userRoles.concat([newRole]));
}
} catch (e) {
setFeedbackMessage({
message:
e.status === 422 ? 'Error inviting user' : e.message ?? e.toString(),
type: 'error'
});
console.log(e);
}
};
const onInviteUserTextChange: React.ChangeEventHandler<
HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement
> = (e) => onInviteUserChange(e.target.name, e.target.value);
const onInviteUserChange = (name: string, value: any) => {
setNewUserValues((values) => ({
...values,
[name]: value
}));
};
const filter = createFilterOptions<AutocompleteType>();
const ListInput = (props: {
type: 'rootDomains' | 'ipBlocks' | 'tags';
label: string;
}) => {
if (!organization) return null;
const elements: (string | OrganizationTag)[] = organization[props.type];
return (
<div className={classes.headerRow}>
<label>{props.label}</label>
<span>
{elements &&
elements.map((value: string | OrganizationTag, index: number) => (
<Chip
className={classes.chip}
key={index}
label={typeof value === 'string' ? value : value.name}
onDelete={() => {
organization[props.type].splice(index, 1);
setOrganization({ ...organization });
}}
></Chip>
))}
<Chip
label="ADD"
variant="outlined"
color="secondary"
onClick={() => {
setDialog({
open: true,
type: props.type,
label: props.label
});
}}
/>
</span>
</div>
);
};
if (!organization) return null;
const views = [
<Paper className={classes.settingsWrapper} key={0}>
<Dialog
open={dialog.open}
onClose={() => setDialog({ open: false })}
aria-labelledby="form-dialog-title"
maxWidth="xs"
fullWidth
>
<DialogTitle id="form-dialog-title">
Add {dialog.label && dialog.label.slice(0, -1)}
</DialogTitle>
<DialogContent>
{dialog.type === 'tags' ? (
<>
<DialogContentText>
Select an existing tag or add a new one.
</DialogContentText>
<Autocomplete
value={tagValue}
onChange={(event, newValue) => {
if (typeof newValue === 'string') {
setTagValue({
name: newValue
});
} else {
setTagValue(newValue);
}
}}
filterOptions={(options, params) => {
const filtered = filter(options, params);
// Suggest the creation of a new value
if (
params.inputValue !== '' &&
!filtered.find(
(tag) =>
tag.name?.toLowerCase() ===
params.inputValue.toLowerCase()
)
) {
filtered.push({
name: params.inputValue,
title: `Add "${params.inputValue}"`
});
}
return filtered;
}}
selectOnFocus
clearOnBlur
handleHomeEndKeys
options={tags}
getOptionLabel={(option) => {
return option.name ?? '';
}}
renderOption={(option) => {
if (option.title) return option.title;
return option.name ?? '';
}}
fullWidth
freeSolo
renderInput={(params) => (
<TextField {...params} variant="outlined" />
)}
/>
</>
) : (
<TextField
autoFocus
margin="dense"
id="name"
label={dialog.label && dialog.label.slice(0, -1)}
type="text"
fullWidth
onChange={(e) => setInputValue(e.target.value)}
/>
)}
</DialogContent>
<DialogActions>
<Button variant="outlined" onClick={() => setDialog({ open: false })}>
Cancel
</Button>
<Button
variant="contained"
color="primary"
onClick={() => {
if (dialog.type && dialog.type !== 'tags') {
if (inputValue) {
organization[dialog.type].push(inputValue);
setOrganization({ ...organization });
}
} else {
if (tagValue) {
if (!organization.tags) organization.tags = [];
organization.tags.push(tagValue as any);
setOrganization({ ...organization });
}
}
setDialog({ open: false });
setInputValue('');
setTagValue(null);
}}
>
Add
</Button>
</DialogActions>
</Dialog>
<TextField
value={organization.name}
disabled
variant="filled"
InputProps={{
className: classes.orgName
}}
></TextField>
<ListInput label="Root Domains" type="rootDomains"></ListInput>
<ListInput label="IP Blocks" type="ipBlocks"></ListInput>
<ListInput label="Tags" type="tags"></ListInput>
<div className={classes.headerRow}>
<label>Passive Mode</label>
<span>
<SwitchInput
checked={organization.isPassive}
onChange={(event: React.ChangeEvent<HTMLInputElement>) => {
setOrganization({
...organization,
isPassive: event.target.checked
});
}}
color="primary"
/>
</span>
</div>
<div className={classes.buttons}>
<Link to={`/organizations`}>
<Button
variant="outlined"
style={{ marginRight: '10px', color: '#565C65' }}
>
Cancel
</Button>
</Link>
<Button
variant="contained"
onClick={updateOrganization}
style={{ background: '#565C65', color: 'white' }}
>
Save
</Button>
</div>
</Paper>,
<React.Fragment key={1}>
<Table<Role> columns={userRoleColumns} data={userRoles} />
<Dialog
open={dialog.open}
onClose={() => setDialog({ open: false })}
aria-labelledby="form-dialog-title"
maxWidth="xs"
fullWidth
>
<DialogTitle id="form-dialog-title">Add Member</DialogTitle>
<DialogContent>
<p style={{ color: '#3D4551' }}>
Organization members can view Organization-specific vulnerabilities,
domains, and notes. Organization administrators can additionally
manage members and update the organization.
</p>
<TextField
margin="dense"
id="firstName"
name="firstName"
label="First Name"
type="text"
fullWidth
value={newUserValues.firstName}
onChange={onInviteUserTextChange}
variant="filled"
InputProps={{
className: classes.textField
}}
/>
<TextField
margin="dense"
id="lastName"
name="lastName"
label="Last Name"
type="text"
fullWidth
value={newUserValues.lastName}
onChange={onInviteUserTextChange}
variant="filled"
InputProps={{
className: classes.textField
}}
/>
<TextField
margin="dense"
id="email"
name="email"
label="Email"
type="text"
fullWidth
value={newUserValues.email}
onChange={onInviteUserTextChange}
variant="filled"
InputProps={{
className: classes.textField
}}
/>
<br></br>
<br></br>
<FormLabel component="legend">Role</FormLabel>
<RadioGroup
aria-label="role"
name="role"
value={newUserValues.role}
onChange={onInviteUserTextChange}
>
<FormControlLabel
value="standard"
control={<Radio color="primary" />}
label="Standard"
/>
<FormControlLabel
value="admin"
control={<Radio color="primary" />}
label="Administrator"
/>
</RadioGroup>
</DialogContent>
<DialogActions>
<Button variant="outlined" onClick={() => setDialog({ open: false })}>
Cancel
</Button>
<Button
variant="contained"
color="primary"
onClick={async () => {
onInviteUserSubmit();
setDialog({ open: false });
}}
>
Add
</Button>
</DialogActions>
</Dialog>
</React.Fragment>,
<React.Fragment key={2}>
<OrganizationList parent={organization}></OrganizationList>
</React.Fragment>,
<React.Fragment key={3}>
<Table<Scan> columns={scanColumns} data={scans} fetchData={fetchScans} />
<h2>Organization Scan History</h2>
<Table<ScanTask> columns={scanTaskColumns} data={scanTasks} />
</React.Fragment>
];
let navItems = [
{
title: 'Settings',
path: `/organizations/${organizationId}`,
exact: true
},
{
title: 'Members',
path: `/organizations/${organizationId}/members`
}
];
if (!organization.parent) {
navItems = navItems.concat([
// { title: 'Teams', path: `/organizations/${organizationId}/teams` },
{ title: 'Scans', path: `/organizations/${organizationId}/scans` }
]);
}
return (
<div>
<div className={classes.header}>
<h1 className={classes.headerLabel}>
<Link to="/organizations">Organizations</Link>
{organization.parent && (
<>
<ChevronRight></ChevronRight>
<Link to={'/organizations/' + organization.parent.id}>
{organization.parent.name}
</Link>
</>
)}
<ChevronRight
style={{
verticalAlign: 'middle',
lineHeight: '100%',
fontSize: '26px'
}}
></ChevronRight>
<span style={{ color: '#07648D' }}>{organization.name}</span>
</h1>
<Subnav
items={navItems}
styles={{
background: '#F9F9F9'
}}
></Subnav>
</div>
<div className={classes.root}>
<Switch>
<Route
path="/organizations/:organizationId"
exact
render={() => views[0]}
/>
<Route
path="/organizations/:organizationId/members"
render={() => views[1]}
/>
<Route
path="/organizations/:organizationId/teams"
render={() => views[2]}
/>
<Route
path="/organizations/:organizationId/scans"
render={() => views[3]}
/>
</Switch>
</div>
</div>
);
}
Example #8
Source File: TermsOfUse.tsx From crossfeed with Creative Commons Zero v1.0 Universal | 4 votes |
TermsOfUse: React.FC = () => {
const history = useHistory();
const [accepted, setAccepted] = useState<boolean>(false);
const [errors, setErrors] = useState<Errors>({});
const { user, setUser, apiPost, maximumRole, touVersion } = useAuthContext();
const onSubmit: React.FormEventHandler = async (e) => {
e.preventDefault();
try {
if (!accepted) throw Error('Must accept terms');
const updated: User = await apiPost(`/users/me/acceptTerms`, {
body: { version: touVersion }
});
setUser(updated);
history.push('/', {
message: 'Your account has been successfully created.'
});
} catch (e) {
setErrors({
global: e.message ?? e.toString()
});
}
};
return (
<AuthForm onSubmit={onSubmit}>
<h1>Terms of Use</h1>
<p>You must read and sign the Terms of Use before using Crossfeed.</p>
<p>
Crossfeed is a free, self-service tool offered by the Department of
Homeland Security’s Cybersecurity and Infrastructure Security Agency
(CISA). Using both passive and active processes, Crossfeed can
continuously evaluate the cybersecurity posture of your public-facing,
internet-accessible network assets for vulnerabilities or configuration
issues.
</p>
<p>
Crossfeed supports two types of users for your organization:
administrative users or view-only users. Administrative users can
add/delete domains for their organization, schedule and disable scans
for their organization, and invite others to create Crossfeed accounts
to have access to the organization’s data. View-only users can only view
data provided to or collected by Crossfeed.
</p>
{maximumRole === 'admin' && (
<p>
Once you create a Crossfeed administrator account, input the Internet
Protocol (IP) addresses or domains to be continuously evaluated, and
select the scanning/evaluation protocols to be used, Crossfeed will
collect data about the sites you specified from multiple
publicly-available resources, including through active interactions
with your sites, if that option is selected by you. Crossfeed will
also examine any publicly-available, internet-accessible resources
that appear to be related or otherwise associated with IPs or domains
you have provided us to evaluate, presenting you with a list of those
related sites for your awareness.
</p>
)}
<p>
By creating a Crossfeed{' '}
{maximumRole === 'admin' ? 'administrator' : 'view only'} account and
using this service, you request CISA’s technical assistance to detect
vulnerabilities and configuration issues through Crossfeed and agree to
the following:
</p>
<ul>
{maximumRole === 'admin' && (
<>
<li>
You have authority to authorize scanning/evaluation of the
public-facing networks and systems you submit within Crossfeed and
you authorize CISA to conduct any such scans/evaluation through
Crossfeed;
</li>
<li>
You agree to promptly update or change the information used to
identify the public-facing networks and systems to be
scanned/evaluated pursuant to this authorization;
</li>
<li>
You agree to comply with any notification or authorization
requirement that any third party that operates or maintains your
public-facing networks or systems may impose on external
vulnerability scanning services, modifying any Crossfeed scans
associated with your account if external scanning of those
resources is later prohibited;
</li>
<li>
You accept that, while Crossfeed will use best efforts to conduct
scans in a way that minimizes risk to your organization’s systems
and networks, Crossfeed scanning activities creates some risk of
degradation in performance to your organization’s systems and
networks;
</li>
<li>
You agree that CISA may share data gathered by Crossfeed with
other federal agencies with cybersecurity responsibilities, with
the Multi-State Information Sharing and Analysis Center, and with
the Election Infrastructure Information Sharing and Analysis
Center;
</li>
<li>
You are authorized to make the above certifications on your
organization’s behalf;
</li>
</>
)}
<li>
You accept that CISA may modify or discontinue the Crossfeed service
at any time;
</li>
<li>
You acknowledge that use of Crossfeed is governed exclusively by
federal law and that CISA provides no warranties of any kind relating
to any aspect of your use of Crossfeed, including that Crossfeed may
detect only a limited range of vulnerabilities or configuration issues
and that there is no guarantee that Crossfeed will detect any or all
vulnerabilities or configuration issues present in your system;
</li>
<li>
You agree to not:
<ul>
<li>
Use the Crossfeed service in violation of any applicable law;
</li>
<li>
Access or attempt to access any Crossfeed account other than your
own; and
</li>
<li>
Introduce malware to the Crossfeed platform or otherwise impair,
harm, or disrupt the functioning or integrity of the platform in
any way;
</li>
</ul>
</li>
<li>
You accept that, at CISA’s sole discretion, CISA may terminate or
suspend your access to the Crossfeed service due to violation of these
terms or any other reason.
</li>
</ul>
<p>ToU version {touVersion}</p>
<Checkbox
required
id="accept"
name="accept"
label="I accept the above Terms and Conditions."
checked={accepted}
onChange={(e) => setAccepted(e.target.checked)}
/>
<p style={{ marginBottom: 0 }}>
<strong>Name:</strong> {user?.fullName}
</p>
<p>
<strong>Email:</strong> {user?.email}
</p>
<div className="width-full display-flex flex-justify-start">
{errors.global && <p className="text-error">{errors.global}</p>}
</div>
<Button type="submit">Submit</Button>
</AuthForm>
);
}
Example #9
Source File: Users.tsx From crossfeed with Creative Commons Zero v1.0 Universal | 4 votes |
Users: React.FC = () => {
const { apiGet, apiPost, apiDelete } = useAuthContext();
const [showModal, setShowModal] = useState<Boolean>(false);
const [selectedRow, setSelectedRow] = useState<number>(0);
const [users, setUsers] = useState<User[]>([]);
const columns: Column<User>[] = [
{
Header: 'Name',
accessor: 'fullName',
width: 200,
disableFilters: true,
id: 'name'
},
{
Header: 'Email',
accessor: 'email',
width: 150,
minWidth: 150,
id: 'email',
disableFilters: true
},
{
Header: 'Organizations',
accessor: ({ roles }) =>
roles &&
roles
.filter((role) => role.approved)
.map((role) => role.organization.name)
.join(', '),
id: 'organizations',
width: 200,
disableFilters: true
},
{
Header: 'User type',
accessor: ({ userType }) =>
userType === 'standard'
? 'Standard'
: userType === 'globalView'
? 'Global View'
: 'Global Admin',
width: 50,
minWidth: 50,
id: 'userType',
disableFilters: true
},
{
Header: 'Date ToU Signed',
accessor: ({ dateAcceptedTerms }) =>
dateAcceptedTerms
? `${formatDistanceToNow(parseISO(dateAcceptedTerms))} ago`
: 'None',
width: 50,
minWidth: 50,
id: 'dateAcceptedTerms',
disableFilters: true
},
{
Header: 'ToU Version',
accessor: 'acceptedTermsVersion',
width: 50,
minWidth: 50,
id: 'acceptedTermsVersion',
disableFilters: true
},
{
Header: 'Last Logged In',
accessor: ({ lastLoggedIn }) =>
lastLoggedIn
? `${formatDistanceToNow(parseISO(lastLoggedIn))} ago`
: 'None',
width: 50,
minWidth: 50,
id: 'lastLoggedIn',
disableFilters: true
},
{
Header: 'Delete',
id: 'delete',
Cell: ({ row }: { row: { index: number } }) => (
<span
onClick={() => {
setShowModal(true);
setSelectedRow(row.index);
}}
>
<FaTimes />
</span>
),
disableFilters: true
}
];
const [errors, setErrors] = useState<Errors>({});
const [values, setValues] = useState<{
firstName: string;
lastName: string;
email: string;
organization?: Organization;
userType: string;
}>({
firstName: '',
lastName: '',
email: '',
userType: ''
});
const fetchUsers = useCallback(async () => {
try {
const rows = await apiGet<User[]>('/users/');
setUsers(rows);
} catch (e) {
console.error(e);
}
}, [apiGet]);
const deleteRow = async (index: number) => {
try {
const row = users[index];
await apiDelete(`/users/${row.id}`, { body: {} });
setUsers(users.filter((user) => user.id !== row.id));
} catch (e) {
setErrors({
global:
e.status === 422 ? 'Unable to delete user' : e.message ?? e.toString()
});
console.log(e);
}
};
const onSubmit: React.FormEventHandler = async (e) => {
e.preventDefault();
try {
const body = {
firstName: values.firstName,
lastName: values.lastName,
email: values.email,
userType: values.userType
};
const user = await apiPost('/users/', {
body
});
setUsers(users.concat(user));
} catch (e) {
setErrors({
global:
e.status === 422
? 'Error when submitting user entry.'
: e.message ?? e.toString()
});
console.log(e);
}
};
const onTextChange: React.ChangeEventHandler<
HTMLInputElement | HTMLSelectElement
> = (e) => onChange(e.target.name, e.target.value);
const onChange = (name: string, value: any) => {
setValues((values) => ({
...values,
[name]: value
}));
};
React.useEffect(() => {
document.addEventListener('keyup', (e) => {
//Escape
if (e.keyCode === 27) {
setShowModal(false);
}
});
}, [apiGet]);
return (
<div className={classes.root}>
<h1>Users</h1>
<Table<User> columns={columns} data={users} fetchData={fetchUsers} />
<h2>Invite a user</h2>
<form onSubmit={onSubmit} className={classes.form}>
{errors.global && <p className={classes.error}>{errors.global}</p>}
<Label htmlFor="firstName">First Name</Label>
<TextInput
required
id="firstName"
name="firstName"
className={classes.textField}
type="text"
value={values.firstName}
onChange={onTextChange}
/>
<Label htmlFor="lastName">Last Name</Label>
<TextInput
required
id="lastName"
name="lastName"
className={classes.textField}
type="text"
value={values.lastName}
onChange={onTextChange}
/>
<Label htmlFor="email">Email</Label>
<TextInput
required
id="email"
name="email"
className={classes.textField}
type="text"
value={values.email}
onChange={onTextChange}
/>
<Label htmlFor="userType">User Type</Label>
<RadioGroup
aria-label="User Type"
name="userType"
value={values.userType}
onChange={onTextChange}
>
<FormControlLabel
value="standard"
control={<Radio color="primary" />}
label="Standard"
/>
<FormControlLabel
value="globalView"
control={<Radio color="primary" />}
label="Global View"
/>
<FormControlLabel
value="globalAdmin"
control={<Radio color="primary" />}
label="Global Administrator"
/>
</RadioGroup>
<br></br>
<Button type="submit">Invite User</Button>
</form>
<ImportExport<
| User
| {
roles: string;
}
>
name="users"
fieldsToExport={['firstName', 'lastName', 'email', 'roles', 'userType']}
onImport={async (results) => {
// TODO: use a batch call here instead.
const createdUsers = [];
for (const result of results) {
const parsedRoles: {
organization: string;
role: string;
}[] = JSON.parse(result.roles as string);
const body: any = result;
// For now, just create role with the first organization
if (parsedRoles.length > 0) {
body.organization = parsedRoles[0].organization;
body.organizationAdmin = parsedRoles[0].role === 'admin';
}
try {
createdUsers.push(
await apiPost('/users/', {
body
})
);
} catch (e) {
// Just continue when an error occurs
console.error(e);
}
}
setUsers(users.concat(...createdUsers));
}}
getDataToExport={() =>
users.map((user) => ({
...user,
roles: JSON.stringify(
user.roles.map((role) => ({
organization: role.organization.id,
role: role.role
}))
)
}))
}
/>
{showModal && (
<div>
<Overlay />
<ModalContainer>
<Modal
actions={
<>
<Button
outline
type="button"
onClick={() => {
setShowModal(false);
}}
>
Cancel
</Button>
<Button
type="button"
onClick={() => {
deleteRow(selectedRow);
setShowModal(false);
}}
>
Delete
</Button>
</>
}
title={<h2>Delete user?</h2>}
>
<p>
Are you sure you would like to delete{' '}
<code>{users[selectedRow].fullName}</code>?
</p>
</Modal>
</ModalContainer>
</div>
)}
</div>
);
}