17415: Removed console.log, used goToPath
[arvados-workbench2.git] / src / views-components / webdav-s3-dialog / webdav-s3-dialog.tsx
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
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";
13
14 export type CssRules = 'details' | 'downloadButton';
15
16 const styles: StyleRulesCallback<CssRules> = theme => ({
17     details: {
18         marginLeft: theme.spacing.unit * 3,
19         marginRight: theme.spacing.unit * 3,
20     },
21     downloadButton: {
22         marginTop: theme.spacing.unit * 2,
23     }
24 });
25
26 interface TabPanelData {
27     children: React.ReactElement<any>[];
28     value: number;
29     index: number;
30 }
31
32 function TabPanel(props: TabPanelData) {
33     const { children, value, index } = props;
34
35     return (
36         <div
37             role="tabpanel"
38             hidden={value !== index}
39             id={`simple-tabpanel-${index}`}
40             aria-labelledby={`simple-tab-${index}`}
41         >
42             {value === index && children}
43         </div>
44     );
45 }
46
47 const isValidIpAddress = (ipAddress: string): Boolean => {
48     if (/^(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/.test(ipAddress)) {
49         return true;
50     }
51
52     return false;
53 };
54
55 const mountainduckTemplate = ({
56     uuid, 
57     username,
58     cyberDavStr,
59     collectionsUrl
60 }: any) => {
61     
62     return `<?xml version="1.0" encoding="UTF-8"?>
63         <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
64         <plist version="1.0">
65         <dict>
66             <key>Protocol</key>
67             <string>davs</string>
68             <key>Provider</key>
69             <string>iterate GmbH</string>
70             <key>UUID</key>
71             <string>${uuid}</string>
72             <key>Hostname</key>
73             <string>${collectionsUrl.replace('https://', ``).replace('*', uuid).split(':')[0]}</string>
74             <key>Port</key>
75             <string>${(cyberDavStr.split(':')[2] || '443').split('/')[0]}</string>
76             <key>Username</key>
77             <string>${username}</string>${isValidIpAddress(collectionsUrl.replace('https://', ``).split(':')[0])? 
78             `
79             <key>Path</key>
80             <string>/c=${uuid}</string>` : ''}
81             <key>Labels</key>
82             <array>
83             </array>
84         </dict>
85         </plist>`.split(/\r?\n/).join('\n');
86 };
87
88 const downloadMountainduckFileHandler = (filename: string, text: string) => {
89     const element = document.createElement('a');
90     element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(text));
91     element.setAttribute('download', filename);
92   
93     element.style.display = 'none';
94     document.body.appendChild(element);
95   
96     element.click();
97   
98     document.body.removeChild(element);
99 };
100
101 export const WebDavS3InfoDialog = compose(
102     withDialog(COLLECTION_WEBDAV_S3_DIALOG_NAME),
103     withStyles(styles),
104 )(
105     (props: WithDialogProps<WebDavS3InfoDialogData> & WithStyles<CssRules>) => {
106         if (!props.data.downloadUrl) { return null; }
107
108         let winDav;
109         let cyberDav;
110
111         if (props.data.collectionsUrl.indexOf("*") > -1) {
112             const withuuid = props.data.collectionsUrl.replace("*", props.data.uuid);
113             winDav = new URL(withuuid);
114             cyberDav = new URL(withuuid);
115         } else {
116             winDav = new URL(props.data.downloadUrl);
117             cyberDav = new URL(props.data.downloadUrl);
118             winDav.pathname = `/by_id/${props.data.uuid}`;
119             cyberDav.pathname = `/by_id/${props.data.uuid}`;
120         }
121
122         cyberDav.username = props.data.username;
123         const cyberDavStr = "dav" + cyberDav.toString().slice(4);
124
125         const s3endpoint = new URL(props.data.collectionsUrl.replace(/\/\*(--[^.]+)?\./, "/"));
126
127         const sp = props.data.token.split("/");
128         let tokenUuid;
129         let tokenSecret;
130         if (sp.length === 3 && sp[0] === "v2" && sp[1].slice(0, 5) === props.data.localCluster) {
131             tokenUuid = sp[1];
132             tokenSecret = sp[2];
133         } else {
134             tokenUuid = props.data.token.replace(/\//g, "_");
135             tokenSecret = tokenUuid;
136         }
137
138         const supportsWebdav = (props.data.uuid.indexOf("-4zz18-") === 5);
139
140         let activeTab = props.data.activeTab;
141         if (!supportsWebdav) {
142             activeTab = 2;
143         }
144
145         return <Dialog
146             open={props.open}
147             maxWidth="md"
148             onClose={props.closeDialog}
149             style={{ alignSelf: 'stretch' }}>
150             <CardHeader
151                 title={`Open as Network Folder or S3 Bucket`} />
152             <div className={props.classes.details} >
153                 <Tabs value={activeTab} onChange={props.data.setActiveTab}>
154                     {supportsWebdav && <Tab value={0} key="cyberduck" label="Cyberduck/Mountain Duck or Gnome Files" />}
155                     {supportsWebdav && <Tab value={1} key="windows" label="Windows or MacOS" />}
156                     <Tab value={2} key="s3" label="S3 bucket" />
157                 </Tabs>
158
159                 <TabPanel index={1} value={activeTab}>
160                     <h2>Settings</h2>
161
162                     <DetailsAttribute
163                         label='Internet address'
164                         value={<a href={winDav.toString()} target="_blank">{winDav.toString()}</a>}
165                         copyValue={winDav.toString()} />
166
167                     <DetailsAttribute
168                         label='Username'
169                         value={props.data.username}
170                         copyValue={props.data.username} />
171
172                     <DetailsAttribute
173                         label='Password'
174                         value={props.data.token}
175                         copyValue={props.data.token} />
176
177                     <h3>Windows</h3>
178                     <ol>
179                         <li>Open File Explorer</li>
180                         <li>Click on "This PC", then go to Computer &rarr; Add a Network Location</li>
181                         <li>Click Next, then choose "Add a custom network location", then click Next</li>
182                     </ol>
183
184                     <h3>MacOS</h3>
185                     <ol>
186                         <li>Open Finder</li>
187                         <li>Click Go &rarr; Connect to server</li>
188                     </ol>
189                 </TabPanel>
190
191                 <TabPanel index={0} value={activeTab}>
192                     <DetailsAttribute
193                         label='Server'
194                         value={<a href={cyberDavStr} target="_blank">{cyberDavStr}</a>}
195                         copyValue={cyberDavStr} />
196
197                     <DetailsAttribute
198                         label='Username'
199                         value={props.data.username}
200                         copyValue={props.data.username} />
201
202                     <DetailsAttribute
203                         label='Password'
204                         value={props.data.token}
205                         copyValue={props.data.token} />
206
207                     <Button
208                         data-cy='download-button'
209                         className={props.classes.downloadButton}
210                         onClick={() => downloadMountainduckFileHandler(`${props.data.collectionName || props.data.uuid}.duck`, mountainduckTemplate({ ...props.data, cyberDavStr }))}
211                         variant='contained'
212                         color='primary'
213                         size='small'>
214                         <DownloadIcon />
215                         Download Cyber/Mountain Duck bookmark
216                     </Button>
217
218                     <h3>Gnome</h3>
219                     <ol>
220                         <li>Open Files</li>
221                         <li>Select +Other Locations</li>
222                         <li>Connect to Server &rarr; Enter server address</li>
223                     </ol>
224
225                 </TabPanel>
226
227                 <TabPanel index={2} value={activeTab}>
228                     <DetailsAttribute
229                         label='Endpoint'
230                         value={s3endpoint.host}
231                         copyValue={s3endpoint.host} />
232
233                     <DetailsAttribute
234                         label='Bucket'
235                         value={props.data.uuid}
236                         copyValue={props.data.uuid} />
237
238                     <DetailsAttribute
239                         label='Access Key'
240                         value={tokenUuid}
241                         copyValue={tokenUuid} />
242
243                     <DetailsAttribute
244                         label='Secret Key'
245                         value={tokenSecret}
246                         copyValue={tokenSecret} />
247
248                 </TabPanel>
249
250             </div>
251             <DialogActions>
252                 <Button
253                     variant='text'
254                     color='primary'
255                     onClick={props.closeDialog}>
256                     Close
257                 </Button>
258             </DialogActions>
259
260         </Dialog >;
261     }
262 );