// Copyright (C) The Arvados Authors. All rights reserved. // // SPDX-License-Identifier: AGPL-3.0 import React from 'react'; import { Dialog, DialogActions, DialogTitle, DialogContent, WithStyles, withStyles, StyleRulesCallback, Button, Typography } from '@material-ui/core'; import CopyToClipboard from 'react-copy-to-clipboard'; import { ArvadosTheme } from 'common/custom-theme'; import { withDialog } from 'store/dialog/with-dialog'; import { WithDialogProps } from 'store/dialog/with-dialog'; import { connect, DispatchProp } from 'react-redux'; import { TokenDialogData, getTokenDialogData, TOKEN_DIALOG_NAME, } from 'store/token-dialog/token-dialog-actions'; import { DefaultCodeSnippet } from 'components/default-code-snippet/default-code-snippet'; import { snackbarActions, SnackbarKind } from 'store/snackbar/snackbar-actions'; import { getNewExtraToken } from 'store/auth/auth-action'; import { DetailsAttributeComponent } from 'components/details-attribute/details-attribute'; import moment from 'moment'; type CssRules = 'link' | 'paper' | 'button' | 'actionButton' | 'codeBlock'; const styles: StyleRulesCallback<CssRules> = (theme: ArvadosTheme) => ({ link: { color: theme.palette.primary.main, textDecoration: 'none', margin: '0px 4px' }, paper: { padding: theme.spacing.unit, marginBottom: theme.spacing.unit * 2, backgroundColor: theme.palette.grey["200"], border: `1px solid ${theme.palette.grey["300"]}` }, button: { fontSize: '0.8125rem', fontWeight: 600 }, actionButton: { boxShadow: 'none', marginTop: theme.spacing.unit * 2, marginBottom: theme.spacing.unit * 2, marginRight: theme.spacing.unit * 2, }, codeBlock: { fontSize: '0.8125rem', }, }); type TokenDialogProps = TokenDialogData & WithDialogProps<{}> & WithStyles<CssRules> & DispatchProp; export class TokenDialogComponent extends React.Component<TokenDialogProps> { onCopy = (message: string) => { this.props.dispatch(snackbarActions.OPEN_SNACKBAR({ message, hideDuration: 2000, kind: SnackbarKind.SUCCESS })); } onGetNewToken = async () => { const newToken = await this.props.dispatch<any>(getNewExtraToken()); if (newToken) { this.props.dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'New token retrieved', hideDuration: 2000, kind: SnackbarKind.SUCCESS })); } else { this.props.dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Creating new tokens is not allowed', hideDuration: 2000, kind: SnackbarKind.WARNING })); } } getSnippet = ({ apiHost, token }: TokenDialogData) => `HISTIGNORE=$HISTIGNORE:'export ARVADOS_API_TOKEN=*' export ARVADOS_API_TOKEN=${token} export ARVADOS_API_HOST=${apiHost} unset ARVADOS_API_HOST_INSECURE` render() { const { classes, open, closeDialog, ...data } = this.props; const tokenExpiration = data.tokenExpiration ? `${data.tokenExpiration.toLocaleString()} (${moment(data.tokenExpiration).fromNow()})` : `This token does not have an expiration date`; return <Dialog open={open} onClose={closeDialog} fullWidth={true} maxWidth='md'> <DialogTitle>Get API Token</DialogTitle> <DialogContent> <Typography paragraph={true}> The Arvados API token is a secret key that enables the Arvados SDKs to access Arvados with the proper permissions. <Typography component='span'> For more information see <a href='http://doc.arvados.org/user/reference/api-tokens.html' target='blank' className={classes.link}> Getting an API token. </a> </Typography> </Typography> <DetailsAttributeComponent label='API Host' value={data.apiHost} copyValue={data.apiHost} onCopy={this.onCopy} /> <DetailsAttributeComponent label='API Token' value={data.token} copyValue={data.token} onCopy={this.onCopy} /> <DetailsAttributeComponent label='Token expiration' value={tokenExpiration} /> { this.props.canCreateNewTokens && <Button onClick={() => this.onGetNewToken()} color="primary" size="small" variant="contained" className={classes.actionButton} > GET NEW TOKEN </Button> } <Typography paragraph={true}> Paste the following lines at a shell prompt to set up the necessary environment for Arvados SDKs to authenticate to your account. </Typography> <DefaultCodeSnippet className={classes.codeBlock} lines={[this.getSnippet(data)]} /> <CopyToClipboard text={this.getSnippet(data)} onCopy={() => this.onCopy('Shell code block copied')}> <Button color="primary" size="small" variant="contained" className={classes.actionButton} > COPY TO CLIPBOARD </Button> </CopyToClipboard> <Typography> Arvados <a href='http://doc.arvados.org/user/reference/api-tokens.html' target='blank' className={classes.link}>virtual machines</a> do this for you automatically. This setup is needed only when you use the API remotely (e.g., from your own workstation). </Typography> </DialogContent> <DialogActions> <Button onClick={closeDialog} className={classes.button} color="primary">CLOSE</Button> </DialogActions> </Dialog>; } } export const TokenDialog = withStyles(styles)( connect(getTokenDialogData)( withDialog(TOKEN_DIALOG_NAME)(TokenDialogComponent)));