1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import * as React from "react";
6 import { Dialog, DialogActions, Button, StyleRulesCallback, WithStyles, withStyles, CardHeader, Tab, Tabs } from '@material-ui/core';
7 import { withDialog } from "~/store/dialog/with-dialog";
8 import { COLLECTION_WEBDAV_S3_DIALOG_NAME, WebDavS3InfoDialogData } from '~/store/collections/collection-info-actions';
9 import { WithDialogProps } from '~/store/dialog/with-dialog';
10 import { compose } from 'redux';
11 import { DetailsAttribute } from "~/components/details-attribute/details-attribute";
12 import { DownloadIcon } from "~/components/icon/icon";
14 export type CssRules = 'details' | 'downloadButton';
16 const styles: StyleRulesCallback<CssRules> = theme => ({
18 marginLeft: theme.spacing.unit * 3,
19 marginRight: theme.spacing.unit * 3,
22 marginTop: theme.spacing.unit * 2,
26 interface TabPanelData {
27 children: React.ReactElement<any>[];
32 function TabPanel(props: TabPanelData) {
33 const { children, value, index } = props;
38 hidden={value !== index}
39 id={`simple-tabpanel-${index}`}
40 aria-labelledby={`simple-tab-${index}`}
42 {value === index && children}
47 const mountainduckTemplate = ({
52 }: any) => `<?xml version="1.0" encoding="UTF-8"?>
53 <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
59 <string>iterate GmbH</string>
61 <string>${uuid}</string>
63 <string>${collectionsUrl.replace('https://', ``).replace('*', uuid).split(':')[0]}</string>
65 <string>${(cyberDavStr.split(':')[2] || '443').split('/')[0]}</string>
67 <string>${username}</string>
73 `.split(/\r?\n/).join('\n');
75 const downloadMountainduckFileHandler = (filename: string, text: string) => {
76 const element = document.createElement('a');
77 element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(text));
78 element.setAttribute('download', filename);
80 element.style.display = 'none';
81 document.body.appendChild(element);
85 document.body.removeChild(element);
88 export const WebDavS3InfoDialog = compose(
89 withDialog(COLLECTION_WEBDAV_S3_DIALOG_NAME),
92 (props: WithDialogProps<WebDavS3InfoDialogData> & WithStyles<CssRules>) => {
93 if (!props.data.downloadUrl) { return null; }
98 if (props.data.collectionsUrl.indexOf("*") > -1) {
99 const withuuid = props.data.collectionsUrl.replace("*", props.data.uuid);
100 winDav = new URL(withuuid);
101 cyberDav = new URL(withuuid);
103 winDav = new URL(props.data.downloadUrl);
104 cyberDav = new URL(props.data.downloadUrl);
105 winDav.pathname = `/by_id/${props.data.uuid}`;
106 cyberDav.pathname = `/by_id/${props.data.uuid}`;
109 cyberDav.username = props.data.username;
110 const cyberDavStr = "dav" + cyberDav.toString().slice(4);
112 const s3endpoint = new URL(props.data.collectionsUrl.replace(/\/\*(--[^.]+)?\./, "/"));
114 const sp = props.data.token.split("/");
117 if (sp.length === 3 && sp[0] === "v2" && sp[1].slice(0, 5) === props.data.localCluster) {
121 tokenUuid = props.data.token.replace(/\//g, "_");
122 tokenSecret = tokenUuid;
125 const supportsWebdav = (props.data.uuid.indexOf("-4zz18-") === 5);
127 let activeTab = props.data.activeTab;
128 if (!supportsWebdav) {
135 onClose={props.closeDialog}
136 style={{ alignSelf: 'stretch' }}>
138 title={`Open as Network Folder or S3 Bucket`} />
139 <div className={props.classes.details} >
140 <Tabs value={activeTab} onChange={props.data.setActiveTab}>
141 {supportsWebdav && <Tab value={0} key="cyberduck" label="Cyberduck/Mountain Duck or Gnome Files" />}
142 {supportsWebdav && <Tab value={1} key="windows" label="Windows or MacOS" />}
143 <Tab value={2} key="s3" label="S3 bucket" />
146 <TabPanel index={1} value={activeTab}>
150 label='Internet address'
151 value={<a href={winDav.toString()} target="_blank">{winDav.toString()}</a>}
152 copyValue={winDav.toString()} />
156 value={props.data.username}
157 copyValue={props.data.username} />
161 value={props.data.token}
162 copyValue={props.data.token} />
166 <li>Open File Explorer</li>
167 <li>Click on "This PC", then go to Computer → Add a Network Location</li>
168 <li>Click Next, then choose "Add a custom network location", then click Next</li>
174 <li>Click Go → Connect to server</li>
178 <TabPanel index={0} value={activeTab}>
181 value={<a href={cyberDavStr} target="_blank">{cyberDavStr}</a>}
182 copyValue={cyberDavStr} />
186 value={props.data.username}
187 copyValue={props.data.username} />
191 value={props.data.token}
192 copyValue={props.data.token} />
195 data-cy='download-button'
196 className={props.classes.downloadButton}
197 onClick={() => downloadMountainduckFileHandler(`${props.data.collectionName || props.data.uuid}.duck`, mountainduckTemplate({ ...props.data, cyberDavStr }))}
208 <li>Select +Other Locations</li>
209 <li>Connect to Server → Enter server address</li>
214 <TabPanel index={2} value={activeTab}>
217 value={s3endpoint.host}
218 copyValue={s3endpoint.host} />
222 value={props.data.uuid}
223 copyValue={props.data.uuid} />
228 copyValue={tokenUuid} />
233 copyValue={tokenSecret} />
242 onClick={props.closeDialog}>