1/** 2 * Copyright (c) 2006-2020, JGraph Ltd 3 * Copyright (c) 2006-2020, draw.io AG 4 */ 5 6//Add a closure to hide the class private variables without changing the code a lot 7(function() 8{ 9 10var _token = null; 11var pickers = {}; 12 13window.DriveClient = function(editorUi, isExtAuth) 14{ 15 if (isExtAuth == null && window.urlParams != null && window.urlParams['extAuth'] == '1') 16 { 17 isExtAuth = true; 18 } 19 20 mxEventSource.call(this); 21 22 DrawioClient.call(this, editorUi, 'gDriveAuthInfo'); 23 24 this.isExtAuth = isExtAuth; 25 /** 26 * Holds a reference to the UI. Needed for the sharing client. 27 */ 28 this.ui = editorUi; 29 30 // New mime type for XML files 31 this.xmlMimeType = 'application/vnd.jgraph.mxfile'; 32 this.mimeType = 'application/vnd.jgraph.mxfile.realtime'; 33 34 // Reading files now possible with no initial click in drive 35 //TODO In teams we do auth using editor app, we need to support viewer only app also 36 if (this.ui.editor.chromeless && !this.ui.editor.editable && urlParams['rt'] != '1' && urlParams['extAuth'] != '1') 37 { 38 // Uses separate name for the viewer auth tokens 39 this.cookieName = 'gDriveViewerAuthInfo'; 40 this.token = this.getPersistentToken(); 41 42 this.appId = window.DRAWIO_GOOGLE_VIEWER_APP_ID || '850530949725'; 43 this.clientId = window.DRAWIO_GOOGLE_VIEWER_CLIENT_ID || '850530949725.apps.googleusercontent.com'; 44 this.scopes = ['https://www.googleapis.com/auth/drive.readonly', 45 'https://www.googleapis.com/auth/userinfo.profile']; 46 } 47 else 48 { 49 this.appId = window.DRAWIO_GOOGLE_APP_ID || '671128082532'; 50 this.clientId = window.DRAWIO_GOOGLE_CLIENT_ID || '671128082532-jhphbq6d0e1gnsus9mn7vf8a6fjn10mp.apps.googleusercontent.com'; 51 } 52 53 this.mimeTypes = this.xmlMimeType + ',application/mxe,application/mxr,' + 54 'application/vnd.jgraph.mxfile.realtime,application/vnd.jgraph.mxfile.rtlegacy'; 55 56 var authInfo = JSON.parse(this.token); 57 58 if (authInfo != null && authInfo.current != null) 59 { 60 this.userId = authInfo.current.userId; 61 this.authCalled = false; 62 } 63}; 64 65// Extends mxEventSource 66mxUtils.extend(DriveClient, mxEventSource); 67 68// Extends DrawioClient 69mxUtils.extend(DriveClient, DrawioClient); 70 71DriveClient.prototype.redirectUri = window.location.protocol + '//' + window.location.host + '/google'; 72DriveClient.prototype.GDriveBaseUrl = 'https://www.googleapis.com/drive/v2'; 73 74/** 75 * OAuth 2.0 scopes for installing Drive Apps. 76 */ 77DriveClient.prototype.scopes = ['https://www.googleapis.com/auth/drive.file', 78 'https://www.googleapis.com/auth/drive.install', 79 'https://www.googleapis.com/auth/userinfo.profile']; 80 81/** 82 * Contains the hostname of the old app. 83 */ 84DriveClient.prototype.allFields = 'kind,id,parents,headRevisionId,etag,title,mimeType,modifiedDate,' + 85 'editable,copyable,canComment,labels,properties,downloadUrl,webContentLink,userPermission,fileSize'; 86 87/** 88 * Fields required for catchin up. 89 * 90 * TODO: Limit to etag and ekey property only 91 */ 92DriveClient.prototype.catchupFields = 'etag,headRevisionId,modifiedDate,properties(key,value)'; 93 94/** 95 * Specifies if thumbnails should be enabled. Default is true. 96 * LATER: If thumbnails are disabled, make sure to replace the 97 * existing thumbnail with the placeholder only once. 98 */ 99DriveClient.prototype.enableThumbnails = true; 100 101/** 102 * Specifies the width for thumbnails. Default is 1000. This value 103 * must be between 220 and 1600. 104 */ 105DriveClient.prototype.thumbnailWidth = 1000; 106 107/** 108 * The maximum number of bytes per thumbnail. Default is 2000000. 109 */ 110DriveClient.prototype.maxThumbnailSize = 2000000; 111 112/** 113 * Defines the base64url PNG to be used if no thumbnail was generated 114 * (including the case where thumbnails are disabled). 115 */ 116DriveClient.prototype.placeholderThumbnail = 'iVBORw0KGgoAAAANSUhEUgAAAJYAAACWCAMAAAAL34HQAAACN1BMVEXwhwXvhgX4iwXzhwXgbQzvhgXhbAzocgzqcwzldAoAAADhbgvjcQnmdgrlbgDwhgXsfwXufgjwhgXwgQfziAXxgADibgz4iwX4jAX3iwTpcwr1igXoewjsfgj3igX4iwXqcQv4jAX3iwXtfQnndQrvhAbibArwhwXgbQz//////v39jwX6jQX+/v7fagHfawzdVQDwhADgbhPgbhXwhwPocQ3uvKvwiA/faQDscgzxiAT97+XgciTgcSP6jAXgbQ3gcCHwiRfpcQzwhwfeXQD77ef74NLvhgTvegD66uPgbAf66+TvfADwjCzgcCfwiSD67ObhcjjwiBHhczvwiyrgbxj///777ujgcSHgcB/xiRzgbhveWgDeVwDhdEDgbRDqfgffYgDfXwD97+bvfQDxiz7//vvwiRr118rrcgztggbfZgDfZAD++PT98+3gbBPsgAb99vD33tPgcB7icAvuhAX//Pn66N/00sTyy7vuuqbjekLwhwzkcgr88er449n++vfutp/kh1vgcBvhbwvmdwnwgwDwgADeWQD87eLxxrTssJjqpIf0roHmjWTkhFP759n63czvvanomnjnlHDhczD22cr4y6/wwa/3xKX2wJ3rqpH0tY7qp4vpnoDymlbjf0vxjjntcwzldAroegj/kgX12s7518PzqnnnkWfynmLieUjpewjrdAD40Lj1uZTzpm3idTbiciLydQzzfwnyiQTsfgD3xqnzp3TxlkzgbCrdTwDdSwBLKUlNAAAAJ3RSTlP8/b2X/YH8wb+FAIuIggJbQin5opAM9+a/ubaubyD78NjSyr2WgRp4sjN4AAAI70lEQVR42u2cZ38SQRDGT8WGvfde4E4BxVMRRaKiUURRlJhQRDCCSgQVO/bee++9994+nMt5ywoezFJd/fm8uITi3p9n5mbYkcCpO6rVnVu2YEXd+3dRIySuo7pLv4GjGNKg7j3UHTl1l14PajmG9OFBnx7Ird4PumpYEtf1QXc112l0M7OGKXEfeg3guo3iNIyJG92Jaz61mYYxcaNacs1H/8f6j6X5j1WI/mMVIsawRFEzI49SjwOqAJa43emclk8Rp2c7AFZ+LDGyvXE2kmO2Q1Lq17RSd6ND48QIwFVuLNHTOPbEpTOz8ujMpccHGz0AV5mxIo4TpwUeUPj0YwfAVVYs0Tn7VZjnBUA8v+n6CyfERY8FR/DEJj7MQ6oL85vOvfDUAsuVC8s19s5yXuAppOPnvPk4EeSCsehCeBVTwVzHfE6RcFUQa4an8Qw91kpbw2oz4aoc1sSxniO0WAI/J24wriabmEpizZtM79bc+fr4/tUarEpiLabGElJYRsOGjbJfjGDpJCxtmosRLOEnVpqLESzZLYlLg65H1rAkLo2GESwcROwXI1jELcS1Y6OGQSzEVaupZQJLDiLhYtCtFBcbbslYhOueqKllDwtzwVhTq4RFuBh0C3EdEBl0C3OBWNUrEISLvSD+5GLQLYmLoSqfwcUiFuaqzhYDxiJc981lxqqdVsCGbHPcQLBgrtK3rwLt9tWqhblKxxI9hW3267U5ZHhuBrCKzXl4NIJTS5FrmbmMWGIEDZIouOp0/O6boYQ2jxBXWcdu13fzRILuF/2Ku+aGr96uBbhALHo5Z38+XcfXyVRZVx/+Ed513ldDCCCu0rFE0Xlo2mu5TAj8ki0XV0q6ePHilhi+d/15b9ACQGGusg3AFzc+XSMBCPzu89+CNlnB7zfD8t1z4iaLXUvDVT6sGdMOnv5pi47f6r9Qk9YF3xZ0l8S11UfMArlgLMpZM6bamYy6rWnta9q7TrZrzZPgPgoqg3atubY8WK6D8lQXHfb4p/wSK7vFfxmxSsAPQ96AlZ4LxoLNeompdkUDGQVznL5mLr4ar5ESD3PBWHA9fbpbjlT4pq1Bm6H6w9dwfOd69ePouNDYt3S3ULPGZ96S3YqtAW/Tepz1E8bgAANc+xEXhAX36ut1cslcd6rJq81SIvgEe7lmL3kY5iqxVYvOI9isswp22KeMOcrriJlWai5giwHl+yec73Ma9Mbfz+qOJndKz6hLpR5V1uPxavFuTTt0K1XfpbNeO0wKeUaR2IPBN5sMRlqu1eY8bsFmPeIFUpi0CjIGTLvSZY2EGeYSi3VL9Dgeb0I+SQl9MlcZT4TObZKzfmfS5NZSx1GsLQ5r+8Sxp7ERR/1TtDlUn2qNuGXCrZGM5URlLDiEVzDVkje5fdjXdDsm27XpXChBz4XG0UpYcDOMYaxjGc3wtyJxFtu1PohaI71f2K2imqEONcN4nrMZ9TWbMf81wg9z3VNwC26Gr3enY4ObobLqbccFefuz5AKONpVfzQp2y3NoVvrN32GLNl9orA22lTiM+Nqg5CJY1DueOjkwsdtNgAP7gidR2SWVhFqt3o9QwoKHIuiwDcwX+xT/UWztSlvCaqXGmtQBY1GadQmfh6anuE0XlkhhRFs3tGGkd+tuIVhiJN0M+brj0mlAu46lX0bcbizVLbgZrgwl4JhYA+NQa9TJQUetsSJYHscJvAVct7eJKoUbQudxPYmdirqzsYsIojhjoitD01yadH287J+vpZF1/uGt2K4ttinjshQo2C2XMzI2U64X6WY4tyZq99a7wZS3eA3BpNyrUPn1x00Z0uM1ACzilOfg7EN3VmRo8dN16WYYerYw6G9qCOSDCjQ0jQkufRbalt65LVyapaA/2mClxhK3Rxy3rsyavDxDR/DL5sMLFiyYu/7sXps7z8VldPv2Xl6PnjlTwOOuJQuytH7CXpvXCOQWoZrYeHWd4nw2Q+v22OLGnFSG0Nk1PCi0xjgjpVvTGi8hht9F+ARBGq8dtXmtOSLoDm1FhUSHnihkTecESalHkPAaWVhtFbA8jqvQGBmbt8fWkKtNn0Xw9GvAWK6DX9bBVHjzqtyvvcG9a+jXyC5oKoKV/a4YFG7Yij2ofszlgtaA3ZoRwW+pIOH3w0qZFURNh3oNtKsDsAr9LNvMC0pj93H6hTPpX9ocg8FIgTVvcgFYC03jFLBMi6ix0MDAoi8/lh7Cgt2q0VfNrSX0ayhjTa2IW0tKdotNrMq4NbPkILKZW+xdiSoGgshogfh7Ul7FcIEoFevfrPLC3+XWf6y/CEvHZoFQqlts9sQigqjLxFpQCJauakFcsqhKPXH79rGb6bE2B5Qmu0b91zn0WJtN8Wys9tgtIqfjEf2SWw7XKI8gHuKQ0X0eDsQSI44TaGBN6dYN5dlI/eFj9I7f8GWtoUJYOIgkiq6Ds/gw5T7dZDUqTrfscbLbB9eIB7JmEKsUgiii/4uO8ToBfJlhfif5tEGWEsGTMT4Mr6HDa0BBlP5Y88lcnkdkCtLhnyjMM0+Gcn2WzW6xnd/J8zn+LZq4SUeEvUBaA8LCs6Tk1p1AetXt3JoMWexWZSyr3RK6vSUGrRHbmkRUVgCLpP1HW/L4tgl5tO140mdKKFFhrkTUdxta4xleA8DCXC6n/vCYvPJFa9zAWL4m6qNaA8IiqjW73lreWnJrSj0AJYFZpvwq6RZRzjVUGEtB5tX7DdoqCXaL+PXHuEjdYsuvVqva4Sqv6NdabdW4YLeIKsoFYzHGhYPIGBd2izGuVpPaSVgAV7VEsOQgsuUXdosxLuwWxLVMW0WRK5ExLiiIpN4vq2YYVTiIbPmFgii5xRiXimCBqmIcVSS3WMqvdMqz5VcKqzdKeca4UrnVT/ryR6bi2Opuf64TwYJlfl4FLqu2Zxeux5BRXZnisvZ8103NqTtzoziuGa24+wZVRdVK9W7wyNSX1nYeOmrU6JSmjp6KhH5BR+kGvk++Ld0c/X66rPH4SEQeGl+kpq8a33eAumPqK347durWpzm9hrWhUevi1Hd4ZzVC+gGMHY0TYnDOYwAAAABJRU5ErkJggg=='.replace(/\+/g, '-').replace(/\//g, '_'); 117 118/** 119 * Mime type for the paceholder thumbnail. 120 */ 121DriveClient.prototype.placeholderMimeType = 'image/png'; 122 123/** 124 * Executes the first step for connecting to Google Drive. 125 */ 126DriveClient.prototype.libraryMimeType = 'application/vnd.jgraph.mxlibrary'; 127 128/** 129 * Contains the hostname of the new app. 130 */ 131DriveClient.prototype.newAppHostname = 'www.draw.io'; 132 133/** 134 * Executes the first step for connecting to Google Drive. 135 */ 136DriveClient.prototype.extension = '.drawio'; 137 138/** 139 * Interval for updating the access token. 140 */ 141DriveClient.prototype.tokenRefreshInterval = 0; 142 143/** 144 * Interval for updating the access token. 145 */ 146DriveClient.prototype.lastTokenRefresh = 0; 147 148/** 149 * Executes the first step for connecting to Google Drive. 150 */ 151DriveClient.prototype.maxRetries = 5; 152 153/** 154 * Executes the first step for connecting to Google Drive. 155 */ 156DriveClient.prototype.staleEtagMaxRetries = 3; 157 158/** 159 * Executes the first step for connecting to Google Drive. 160 */ 161DriveClient.prototype.coolOff = 1000; 162 163/** 164 * Executes the first step for connecting to Google Drive. 165 */ 166DriveClient.prototype.mimeTypeCheckCoolOff = 60000; 167 168/** 169 * Executes the first step for connecting to Google Drive. 170 */ 171DriveClient.prototype.user = null; 172 173/** 174 * Executes auth in same window (no popups) 175 */ 176DriveClient.prototype.sameWinAuthMode = false; 177 178/** 179 * Redirect URL of samw window mode that will get the token 180 */ 181DriveClient.prototype.sameWinRedirectUrl = null; 182 183 184/** 185 * Authorizes the client, gets the userId and calls <open>. 186 */ 187DriveClient.prototype.setUser = function(user) 188{ 189 this.user = user; 190 191 if (this.user == null) 192 { 193 this.userId = null; 194 195 if (this.tokenRefreshThread != null) 196 { 197 window.clearTimeout(this.tokenRefreshThread); 198 this.tokenRefreshThread = null; 199 } 200 } 201 else 202 { 203 this.userId = user.id; 204 } 205 206 this.fireEvent(new mxEventObject('userChanged')); 207}; 208 209DriveClient.prototype.setUserId = function(userId) 210{ 211 this.userId = userId; 212 213 if (this.user != null && this.user.id != this.userId) 214 { 215 this.user = null; 216 } 217}; 218/** 219 * Authorizes the client, gets the userId and calls <open>. 220 */ 221DriveClient.prototype.getUser = function() 222{ 223 return this.user; 224}; 225 226DriveClient.prototype.getUsersList = function() 227{ 228 var users = []; 229 var authInfo = JSON.parse(this.getPersistentToken(true)); 230 var curUserId = null; 231 232 if (authInfo != null) 233 { 234 if (authInfo.current != null) 235 { 236 curUserId = authInfo.current.userId; 237 users.push(authInfo[curUserId].user); 238 users[0].isCurrent = true; 239 240 } 241 242 for (var id in authInfo) 243 { 244 if (id == 'current' || id == curUserId) continue; 245 246 users.push(authInfo[id].user); 247 } 248 } 249 return users; 250}; 251 252DriveClient.prototype.logout = function() 253{ 254 //Send to server to clear refresh token cookie 255 this.ui.editor.loadUrl(this.redirectUri + '?doLogout=1&userId=' + this.userId + '&state=' + encodeURIComponent('cId=' + this.clientId + '&domain=' + window.location.hostname)); 256 this.clearPersistentToken(); 257 this.setUser(null); 258 _token = null; 259}; 260 261/** 262 * Authorizes the client, gets the userId and calls <open>. 263 */ 264DriveClient.prototype.execute = function(fn) 265{ 266 // Handles error in immediate authorize call via callback that shows a 267 // UI with a button that executes the second non-immediate authorize 268 var fallback = mxUtils.bind(this, function(resp) 269 { 270 // Remember is an argument for the callback that executes 271 // when the user clicks the authorize button in the UI and 272 // success executes after successful authorization. 273 this.ui.showAuthDialog(this, true, mxUtils.bind(this, function(remember, success) 274 { 275 this.authorize(false, mxUtils.bind(this, function() 276 { 277 if (success != null) 278 { 279 success(); 280 } 281 282 fn(); 283 }), mxUtils.bind(this, function(resp) 284 { 285 var msg = mxResources.get('cannotLogin'); 286 287 // Handles special domain policy errors 288 if (resp != null && resp.error != null) 289 { 290 if (resp.error.code == 403 && 291 resp.error.data != null && resp.error.data.length > 0 && 292 resp.error.data[0].reason == 'domainPolicy') 293 { 294 msg = resp.error.message; 295 } 296 } 297 298 this.logout(); 299 300 this.ui.showError(mxResources.get('error'), msg, mxResources.get('help'), mxUtils.bind(this, function() 301 { 302 this.ui.openLink('https://www.diagrams.net/doc/faq/gsuite-authorisation-troubleshoot'); 303 }), null, mxResources.get('ok')); 304 }), remember); 305 })); 306 }); 307 308 // First immediate authorize attempt 309 this.authorize(true, fn, fallback); 310}; 311 312/** 313 * Executes the given request. 314 */ 315DriveClient.prototype.executeRequest = function(reqObj, success, error) 316{ 317 try 318 { 319 var acceptResponse = true; 320 var timeoutThread = null; 321 var retryCount = 0; 322 323 // Cancels any pending requests 324 if (this.requestThread != null) 325 { 326 window.clearTimeout(this.requestThread); 327 } 328 329 var fn = mxUtils.bind(this, function() 330 { 331 try 332 { 333 this.requestThread = null; 334 this.currentRequest = reqObj; 335 336 if (timeoutThread != null) 337 { 338 window.clearTimeout(timeoutThread); 339 } 340 341 timeoutThread = window.setTimeout(mxUtils.bind(this, function() 342 { 343 acceptResponse = false; 344 345 if (error != null) 346 { 347 error({code: App.ERROR_TIMEOUT, message: mxResources.get('timeout'), retry: fn}); 348 } 349 }), this.ui.timeout); 350 351 var params = null; 352 var isJSON = false; 353 354 if (typeof reqObj.params === 'string') 355 { 356 params = reqObj.params; 357 } 358 else if (reqObj.params != null) 359 { 360 params = JSON.stringify(reqObj.params); 361 isJSON = true; 362 } 363 364 var url = reqObj.fullUrl || (this.GDriveBaseUrl + reqObj.url); 365 366 if (isJSON) 367 { 368 url += (url.indexOf('?') > 0 ? '&' : '?') + 'alt=json'; 369 } 370 371 var req = new mxXmlRequest(url, params, reqObj.method || 'GET'); 372 373 req.setRequestHeaders = mxUtils.bind(this, function(request, params) 374 { 375 if (reqObj.headers != null) 376 { 377 for (var key in reqObj.headers) 378 { 379 request.setRequestHeader(key, reqObj.headers[key]); 380 } 381 } 382 else if (reqObj.contentType != null) 383 { 384 request.setRequestHeader('Content-Type', reqObj.contentType); 385 } 386 else if (isJSON) 387 { 388 request.setRequestHeader('Content-Type', 'application/json'); 389 } 390 391 request.setRequestHeader('Authorization', 'Bearer ' + _token); 392 }); 393 394 req.send(mxUtils.bind(this, function(req) 395 { 396 try 397 { 398 window.clearTimeout(timeoutThread); 399 400 if (acceptResponse) 401 { 402 var resp; 403 404 try 405 { 406 resp = JSON.parse(req.getText()); 407 } 408 catch(e) 409 { 410 resp = null; 411 } 412 413 if (req.getStatus() >= 200 && req.getStatus() <= 299) 414 { 415 if (success != null) 416 { 417 success(resp); 418 } 419 } 420 else 421 { 422 // Errors for put request are in data instead of errors 423 var data = (resp != null && resp.error != null) ? ((resp.error.data != null) ? 424 resp.error.data : resp.error.errors) : null; 425 var reason = (data != null && data.length > 0) ? data[0].reason : null; 426 427 // Handles special error for saving old file where mime was changed to new 428 // LATER: Check if 403 is never auth error, for now we check the message for a specific 429 // case where the old app mime type was overridden by the new app 430 if (error != null && resp != null && resp.error != null && (resp.error.code == -1 || 431 (resp.error.code == 403 && (reason == 'domainPolicy' || resp.error.message == 432 'The requested mime type change is forbidden.')))) 433 { 434 error(resp); 435 } 436 // Handles authentication error 437 else if (resp != null && resp.error != null && (resp.error.code == 401 || 438 (resp.error.code == 403 && reason != 'rateLimitExceeded'))) 439 { 440 // Shows an error if re-authenticated but the server still doesn't allow it 441 if ((resp.error.code == 403 && this.retryAuth) || 442 (resp.error.code == 401 && this.retryAuth && reason == 'authError')) 443 { 444 if (error != null) 445 { 446 error(resp); 447 } 448 449 this.retryAuth = false; 450 } 451 else 452 { 453 this.retryAuth = true; 454 this.execute(fn); 455 } 456 } 457 // Schedules a retry if no new request was executed 458 else if (resp != null && resp.error != null && resp.error.code != 412 && resp.error.code != 404 && 459 resp.error.code != 400 && this.currentRequest == reqObj && retryCount < this.maxRetries) 460 { 461 retryCount++; 462 var jitter = 1 + 0.1 * (Math.random() - 0.5); 463 this.requestThread = window.setTimeout(fn, 464 Math.round(Math.pow(2, retryCount) * 465 jitter * this.coolOff)); 466 } 467 else if (error != null) 468 { 469 error(resp); 470 } 471 } 472 } 473 } 474 catch (e) 475 { 476 if (error != null) 477 { 478 error(e); 479 } 480 else 481 { 482 throw e; 483 } 484 } 485 })); 486 } 487 catch (e) 488 { 489 if (error != null) 490 { 491 error(e); 492 } 493 else 494 { 495 throw e; 496 } 497 } 498 }); 499 500 // Must get token before first request in this case 501 if (_token == null || !this.authCalled) 502 { 503 this.execute(fn); 504 } 505 else 506 { 507 fn(); 508 } 509 } 510 catch (e) 511 { 512 if (error != null) 513 { 514 error(e); 515 } 516 else 517 { 518 throw e; 519 } 520 } 521}; 522 523DriveClient.prototype.createAuthWin = function(url) 524{ 525 var width = 525, 526 height = 525, 527 screenX = window.screenX, 528 screenY = window.screenY, 529 outerWidth = window.outerWidth, 530 outerHeight = window.outerHeight; 531 532 var left = screenX + Math.max(outerWidth - width, 0) / 2; 533 var top = screenY + Math.max(outerHeight - height, 0) / 2; 534 535 var features = ['width=' + width, 'height=' + height, 536 'top=' + top, 'left=' + left, 537 'status=no', 'resizable=yes', 538 'toolbar=no', 'menubar=no', 539 'scrollbars=yes']; 540 return window.open(url? url : 'about:blank', 'gdauth', features.join(',')); 541}; 542 543/** 544 * Authorizes the client, gets the userId and calls <open>. 545 */ 546DriveClient.prototype.authorize = function(immediate, success, error, remember, popup) 547{ 548 if (this.isExtAuth && !immediate) 549 { 550 window.parent.driveAuth(mxUtils.bind(this, function(newAuthInfo) 551 { 552 this.updateAuthInfo(newAuthInfo, true, true, success, error); 553 }), error); 554 return; 555 } 556 557 var req = new mxXmlRequest(this.redirectUri + '?getState=1', null, 'GET'); 558 559 req.send(mxUtils.bind(this, function(req) 560 { 561 if (req.getStatus() >= 200 && req.getStatus() <= 299) 562 { 563 this.authorizeStep2(req.getText(), immediate, success, error, remember, popup); 564 } 565 else if (error != null) 566 { 567 error(req); 568 } 569 }), error); 570}; 571 572DriveClient.prototype.updateAuthInfo = function (newAuthInfo, remember, forceUserUpdate, success, error) 573{ 574 _token = newAuthInfo.access_token; 575 delete newAuthInfo.access_token; //Don't store access token 576 newAuthInfo.expires = Date.now() + parseInt(newAuthInfo.expires_in) * 1000; 577 newAuthInfo.remember = remember; 578 579 this.resetTokenRefresh(newAuthInfo); 580 this.authCalled = true; 581 582 if (forceUserUpdate || this.user == null) 583 { 584 //IE/Edge security doesn't allow access to newAuthInfo in a callback function (outside this function scope) 585 //So, stringify the object and restore it (parse) in the callback 586 var strAuthInfo = JSON.stringify(newAuthInfo); 587 588 this.updateUser(mxUtils.bind(this, function() 589 { 590 //Restore the auth info object to bypass IE/Edge security 591 var resAuthInfo = JSON.parse(strAuthInfo); 592 //Save user and new token 593 this.setPersistentToken(resAuthInfo, !remember); 594 595 if (success != null) 596 { 597 success(); 598 } 599 }), error); 600 } 601 else if (success != null) 602 { 603 this.setPersistentToken(newAuthInfo, !remember); 604 success(); 605 } 606}; 607 608DriveClient.prototype.authorizeStep2 = function(state, immediate, success, error, remember, popup) 609{ 610 try 611 { 612 // Takes userId from state URL parameter 613 if (this.ui.stateArg != null && this.ui.stateArg.userId != null) 614 { 615 this.userId = this.ui.stateArg.userId; 616 617 if (this.user != null && this.user.id != this.userId) 618 { 619 this.user = null; 620 } 621 } 622 623 if (this.userId == null) 624 { 625 var authInfo = JSON.parse(this.getPersistentToken(true)); 626 627 if (authInfo && authInfo.current != null) 628 { 629 this.userId = authInfo.current.userId; 630 } 631 } 632 633 // Immediate only possible with a refresh token (there is a userId) 634 if (immediate && this.userId == null) 635 { 636 if (error != null) 637 { 638 error(); 639 } 640 } 641 else 642 { 643 //Retry request with refreshed token (in the cookie) 644 if (immediate) //Note, we checked refresh token is not null above 645 { 646 //state is used to identify which app/domain is used 647 var req = new mxXmlRequest(this.redirectUri + '?state=' + encodeURIComponent('cId=' + this.clientId + '&domain=' + window.location.hostname + '&token=' + state) 648 + '&userId=' + this.userId 649 , null, 'GET'); 650 651 req.send(mxUtils.bind(this, function(req) 652 { 653 if (req.getStatus() >= 200 && req.getStatus() <= 299) 654 { 655 var newAuthInfo = JSON.parse(req.getText()); 656 this.updateAuthInfo(newAuthInfo, true, false, success, error); //We set remember to true since we can only have a refresh token if user initially selected remember 657 } 658 else 659 { 660 //When the request fails (e.g, Hibernate on Windows), the status is 0, this doesn't mean the token is invalid 661 if (req.getStatus() != 0) 662 { 663 this.logout(); 664 } 665 666 if (error != null) 667 { 668 error(req); //TODO review this code path and how error is handled 669 } 670 } 671 }), error); 672 } 673 else 674 { 675 var url = 'https://accounts.google.com/o/oauth2/v2/auth?client_id=' + this.clientId + 676 '&redirect_uri=' + encodeURIComponent(this.redirectUri) + 677 '&response_type=code&include_granted_scopes=true' + 678 (remember? '&access_type=offline&prompt=consent%20select_account' : '') + //Ask for consent again to get a new refresh token 679 '&scope=' + encodeURIComponent(this.scopes.join(' ')) + 680 '&state=' + encodeURIComponent('cId=' + this.clientId + '&domain=' + window.location.hostname + '&token=' + state + //To identify which app/domain is used 681 (this.sameWinRedirectUrl? '&redirect=' + this.sameWinRedirectUrl : '')); 682 683 if (this.sameWinAuthMode) 684 { 685 window.location.assign(url); 686 popup = null; //Same window doesn't use onGoogleDriveCallback or popups 687 } 688 else if (popup == null) 689 { 690 popup = this.createAuthWin(url); 691 } 692 else 693 { 694 popup.location = url; 695 } 696 697 if (popup != null) 698 { 699 window.onGoogleDriveCallback = mxUtils.bind(this, function(newAuthInfo, authWindow) 700 { 701 window.onGoogleDriveCallback = null; 702 703 try 704 { 705 if (newAuthInfo == null) 706 { 707 if (error != null) 708 { 709 error({message: mxResources.get('accessDenied')}); //TODO Check this error handling is correct 710 } 711 } 712 else 713 { 714 this.updateAuthInfo(newAuthInfo, remember, true, success, error); 715 } 716 } 717 catch (e) 718 { 719 if (error != null) 720 { 721 error(e); 722 } 723 } 724 finally 725 { 726 if (authWindow != null) 727 { 728 authWindow.close(); 729 } 730 } 731 }); 732 733 popup.focus(); 734 } 735 } 736 } 737 } 738 catch (e) 739 { 740 if (error != null) 741 { 742 error(e); 743 } 744 else 745 { 746 throw e; 747 } 748 } 749}; 750 751/** 752 * Checks if the client is authorized and calls the next step. 753 */ 754DriveClient.prototype.resetTokenRefresh = function(resp) 755{ 756 if (this.tokenRefreshThread != null) 757 { 758 window.clearTimeout(this.tokenRefreshThread); 759 this.tokenRefreshThread = null; 760 } 761 762 // Starts timer to refresh token before it expires 763 if (resp != null && resp.error == null && resp.expires_in > 0) 764 { 765 this.tokenRefreshInterval = parseInt(resp.expires_in) * 1000; 766 this.lastTokenRefresh = new Date().getTime(); 767 768 this.tokenRefreshThread = window.setTimeout(mxUtils.bind(this, function() 769 { 770 this.authorize(true, mxUtils.bind(this, function() 771 { 772 //console.log('tokenRefresh: refreshed', _token); 773 }), mxUtils.bind(this, function() 774 { 775 //console.log('tokenRefresh: error refreshing', _token); 776 })); 777 }), resp.expires_in * 900); 778 } 779}; 780 781/** 782 * Checks if the client is authorized and calls the next step. 783 */ 784DriveClient.prototype.checkToken = function(fn) 785{ 786 var connected = this.lastTokenRefresh > 0; 787 var delta = new Date().getTime() - this.lastTokenRefresh; 788 789 if (delta > this.tokenRefreshInterval || this.tokenRefreshThread == null) 790 { 791 // Uses execute instead of authorize to allow a fallback authorization if cookie was lost 792 this.execute(mxUtils.bind(this, function() 793 { 794 fn(); 795 796 if (connected) 797 { 798 this.fireEvent(new mxEventObject('disconnected')); 799 } 800 })); 801 } 802 else 803 { 804 fn(); 805 } 806}; 807 808/** 809 * Checks if the client is authorized and calls the next step. 810 */ 811DriveClient.prototype.updateUser = function(success, error) 812{ 813 try 814 { 815 var url = 'https://www.googleapis.com/oauth2/v2/userinfo?alt=json'; 816 var headers = {'Authorization': 'Bearer ' + _token}; 817 818 this.ui.editor.loadUrl(url, mxUtils.bind(this, function(data) 819 { 820 var info = JSON.parse(data); 821 822 // Requests more information about the user (email address is sometimes not in info) 823 this.executeRequest({url: '/about'}, mxUtils.bind(this, function(resp) 824 { 825 var email = mxResources.get('notAvailable'); 826 var name = email; 827 var pic = null; 828 829 if (resp != null && resp.user != null) 830 { 831 email = resp.user.emailAddress; 832 name = resp.user.displayName; 833 pic = (resp.user.picture != null) ? resp.user.picture.url : null; 834 } 835 836 this.setUser(new DrawioUser(info.id, email, name, pic, info.locale)); 837 this.userId = info.id; 838 839 if (success != null) 840 { 841 success(); 842 } 843 }), error); 844 }), error, null, null, null, null, headers); 845 } 846 catch (e) 847 { 848 if (error != null) 849 { 850 error(e); 851 } 852 else 853 { 854 throw e; 855 } 856 } 857}; 858 859/** 860 * Translates this point by the given vector. 861 * 862 * @param {number} dx X-coordinate of the translation. 863 * @param {number} dy Y-coordinate of the translation. 864 */ 865DriveClient.prototype.copyFile = function(id, title, success, error) 866{ 867 if (id != null && title != null) 868 { 869 this.executeRequest({url: '/files/' + id + '/copy?fields=' + encodeURIComponent(this.allFields) 870 + '&supportsAllDrives=true&enforceSingleParent=true', //&alt=json 871 method: 'POST', 872 params: {'title': title, 'properties': 873 [{'key': 'channel', 'value': Editor.guid()}]} 874 }, success, error); 875 } 876}; 877 878/** 879 * Translates this point by the given vector. 880 * 881 * @param {number} dx X-coordinate of the translation. 882 * @param {number} dy Y-coordinate of the translation. 883 */ 884DriveClient.prototype.renameFile = function(id, title, success, error) 885{ 886 if (id != null && title != null) 887 { 888 this.executeRequest(this.createDriveRequest( 889 id, {'title' : title}), success, error); 890 } 891}; 892 893/** 894 * Translates this point by the given vector. 895 * 896 * @param {number} dx X-coordinate of the translation. 897 * @param {number} dy Y-coordinate of the translation. 898 */ 899DriveClient.prototype.moveFile = function(id, folderId, success, error) 900{ 901 if (id != null && folderId != null) 902 { 903 this.executeRequest(this.createDriveRequest(id, {'parents': [{'kind': 904 'drive#fileLink', 'id': folderId}]}), success, error); 905 } 906}; 907 908/** 909 * Translates this point by the given vector. 910 * 911 * @param {number} dx X-coordinate of the translation. 912 * @param {number} dy Y-coordinate of the translation. 913 */ 914DriveClient.prototype.createDriveRequest = function(id, body) 915{ 916 return { 917 'url': '/files/' + id + '?uploadType=multipart&supportsAllDrives=true', 918 'method': 'PUT', 919 'contentType': 'application/json; charset=UTF-8', 920 'params': body 921 }; 922}; 923 924/** 925 * Loads the given file as a library file. 926 */ 927DriveClient.prototype.getLibrary = function(id, success, error) 928{ 929 return this.getFile(id, success, error, true, true); 930}; 931 932/** 933 * Loads the descriptorf for the given file ID. 934 */ 935DriveClient.prototype.loadDescriptor = function(id, success, error, fields) 936{ 937 this.executeRequest({ 938 url: '/files/' + id + '?supportsAllDrives=true&fields=' + (fields != null ? fields : this.allFields) 939 }, success, error); 940}; 941 942DriveClient.prototype.listFiles = function(searchStr, afterDate, mineOnly, success, error) 943{ 944 this.executeRequest({ 945 url: '/files?supportsAllDrives=true&includeItemsFromAllDrives=true&q=' + encodeURIComponent('(mimeType contains \'' + this.xmlMimeType + '\') ' + 946 (searchStr? ' and (title contains \'' + searchStr + '\')' : '') + 947 (afterDate? ' and (modifiedDate > \'' + afterDate.toISOString() + '\')' : '') + 948 (mineOnly? ' and (\'me\' in owners)' : '')) + 949 '&orderBy=modifiedDate desc,title' 950 }, success, error); 951}; 952 953/** 954 * Gets the channel ID from the given descriptor. 955 */ 956DriveClient.prototype.getCustomProperty = function(desc, key) 957{ 958 var props = desc.properties; 959 var result = null; 960 961 if (props != null) 962 { 963 for (var i = 0; i < props.length; i++) 964 { 965 if (props[i].key == key) 966 { 967 result = props[i].value; 968 969 break; 970 } 971 } 972 } 973 974 return result; 975}; 976 977/** 978 * Checks if the client is authorized and calls the next step. The optional 979 * readXml argument is used for import. Default is false. The optional 980 * readLibrary argument is used for reading libraries. Default is false. 981 */ 982DriveClient.prototype.getFile = function(id, success, error, readXml, readLibrary) 983{ 984 readXml = (readXml != null) ? readXml : false; 985 readLibrary = (readLibrary != null) ? readLibrary : false; 986 987 if (urlParams['rev'] != null) 988 { 989 this.executeRequest({ 990 url: '/files/' + id + '/revisions/' + urlParams['rev'] + '?supportsAllDrives=true' 991 }, 992 mxUtils.bind(this, function(resp) 993 { 994 // Redirects title to originalFilename to 995 // match expected descriptor interface 996 resp.title = resp.originalFilename; 997 998 // Uses ID of file instead of revision ID in descriptor 999 // to avoid a change of the document hash property 1000 resp.headRevisionId = resp.id; 1001 resp.id = id; 1002 1003 this.getXmlFile(resp, success, error); 1004 }), error); 1005 } 1006 else 1007 { 1008 this.loadDescriptor(id, mxUtils.bind(this, function(resp) 1009 { 1010 try 1011 { 1012 if (this.user != null) 1013 { 1014 var binary = /\.png$/i.test(resp.title); 1015 1016 // Handles .vsdx, .vsd, .vdx, Gliffy and PNG+XML files by creating a temporary file 1017 if (/\.v(dx|sdx?)$/i.test(resp.title) || /\.gliffy$/i.test(resp.title) || 1018 (!this.ui.useCanvasForExport && binary)) 1019 { 1020 var url = resp.downloadUrl; 1021 var headers = {'Authorization': 'Bearer ' + _token}; 1022 1023 this.ui.convertFile(url, resp.title, resp.mimeType, this.extension, success, error, null, headers); 1024 } 1025 else 1026 { 1027 // Handles converted realtime files as XML files 1028 if (readXml || readLibrary || resp.mimeType == this.libraryMimeType || 1029 resp.mimeType == this.xmlMimeType) 1030 { 1031 this.getXmlFile(resp, success, error, true, readLibrary); 1032 } 1033 else 1034 { 1035 this.getXmlFile(resp, success, error); 1036 } 1037 } 1038 } 1039 else 1040 { 1041 error({message: mxResources.get('loggedOut')}); 1042 } 1043 } 1044 catch (e) 1045 { 1046 if (error != null) 1047 { 1048 error(e); 1049 } 1050 else 1051 { 1052 throw e; 1053 } 1054 } 1055 }), error); 1056 } 1057}; 1058 1059/** 1060 * Returns true if the given mime type is for Google Realtime files. 1061 */ 1062DriveClient.prototype.isGoogleRealtimeMimeType = function(mimeType) 1063{ 1064 return mimeType != null && mimeType.substring(0, 30) == 'application/vnd.jgraph.mxfile.'; 1065}; 1066 1067/** 1068 * Checks if the client is authorized and calls the next step. The ignoreMime argument is 1069 * used for import via getFile. Default is false. The optional 1070 * readLibrary argument is used for reading libraries. Default is false. 1071 */ 1072DriveClient.prototype.getXmlFile = function(resp, success, error, ignoreMime, readLibrary) 1073{ 1074 try 1075 { 1076 var headers = {'Authorization': 'Bearer ' + _token}; 1077 var url = resp.downloadUrl; 1078 1079 // Download URL is null if no option to download for viewers 1080 if (url == null) 1081 { 1082 if (error != null) 1083 { 1084 error({message: mxResources.get('exportOptionsDisabledDetails')}); 1085 } 1086 } 1087 else 1088 { 1089 var retryCount = 0; 1090 1091 var fn = mxUtils.bind(this, function() 1092 { 1093 // Loads XML to initialize realtime document if realtime is empty 1094 this.ui.editor.loadUrl(url, mxUtils.bind(this, function(data) 1095 { 1096 try 1097 { 1098 if (data == null) 1099 { 1100 // TODO: Optional redirect to legacy if link is for old file 1101 error({message: mxResources.get('invalidOrMissingFile')}); 1102 } 1103 else if (resp.mimeType == this.libraryMimeType || readLibrary) 1104 { 1105 if (resp.mimeType == this.libraryMimeType && !readLibrary) 1106 { 1107 error({message: mxResources.get('notADiagramFile')}); 1108 } 1109 else 1110 { 1111 success(new DriveLibrary(this.ui, data, resp)); 1112 } 1113 } 1114 else 1115 { 1116 var importFile = false; 1117 1118 if (/\.png$/i.test(resp.title)) 1119 { 1120 var index = data.lastIndexOf(','); 1121 1122 if (index > 0) 1123 { 1124 var xml = this.ui.extractGraphModelFromPng(data); 1125 1126 if (xml != null && xml.length > 0) 1127 { 1128 data = xml; 1129 } 1130 else 1131 { 1132 // Checks if the file contains XML data which can happen when we insert 1133 // the file and then don't post-process it when loaded into the UI which 1134 // is required for creating the images for .PNG and .SVG files. 1135 try 1136 { 1137 var xml = data.substring(index + 1); 1138 var temp = (window.atob && !mxClient.IS_IE && !mxClient.IS_IE11) ? 1139 atob(xml) : Base64.decode(xml); 1140 var node = this.ui.editor.extractGraphModel( 1141 mxUtils.parseXml(temp).documentElement, true); 1142 1143 if (node == null || node.getElementsByTagName('parsererror').length > 0) 1144 { 1145 importFile = true; 1146 } 1147 else 1148 { 1149 data = temp; 1150 } 1151 } 1152 catch (e) 1153 { 1154 importFile = true; 1155 } 1156 } 1157 } 1158 } 1159 else if (/\.pdf$/i.test(resp.title)) 1160 { 1161 var xml = Editor.extractGraphModelFromPdf(data); 1162 1163 if (xml != null && xml.length > 0) 1164 { 1165 importFile = true; 1166 data = xml; 1167 } 1168 } 1169 // Checks for base64 encoded mxfile 1170 else if (data.substring(0, 32) == '') 1171 { 1172 var temp = data.substring(22); 1173 data = (window.atob && !mxClient.IS_SF) ? atob(temp) : Base64.decode(temp); 1174 } 1175 1176 if (Graph.fileSupport && new XMLHttpRequest().upload && this.ui.isRemoteFileFormat(data, url)) 1177 { 1178 this.ui.parseFile(new Blob([data], {type: 'application/octet-stream'}), mxUtils.bind(this, function(xhr) 1179 { 1180 try 1181 { 1182 if (xhr.readyState == 4) 1183 { 1184 if (xhr.status >= 200 && xhr.status <= 299) 1185 { 1186 success(new LocalFile(this.ui, xhr.responseText, resp.title + this.extension, true)); 1187 } 1188 else if (error != null) 1189 { 1190 error({message: mxResources.get('errorLoadingFile')}); 1191 } 1192 } 1193 } 1194 catch (e) 1195 { 1196 if (error != null) 1197 { 1198 error(e); 1199 } 1200 else 1201 { 1202 throw e; 1203 } 1204 } 1205 }), resp.title); 1206 } 1207 else 1208 { 1209 success((importFile) ? new LocalFile(this.ui, data, resp.title, true) : new DriveFile(this.ui, data, resp)); 1210 } 1211 } 1212 } 1213 catch (e) 1214 { 1215 if (error != null) 1216 { 1217 error(e); 1218 } 1219 else 1220 { 1221 throw e; 1222 } 1223 } 1224 }), mxUtils.bind(this, function(e, req) 1225 { 1226 if (retryCount < this.maxRetries && req != null && req.getStatus() == 403) 1227 { 1228 retryCount++; 1229 var jitter = 1 + 0.1 * (Math.random() - 0.5); 1230 var delay = retryCount * 2 * this.coolOff * jitter; 1231 1232 window.setTimeout(fn, delay); 1233 } 1234 else 1235 { 1236 if (error != null) 1237 { 1238 error(e); 1239 } 1240 else 1241 { 1242 throw e; 1243 } 1244 } 1245 }), ((resp.mimeType != null && resp.mimeType.substring(0, 6) == 'image/' && 1246 resp.mimeType.substring(0, 9) != 'image/svg')) || /\.png$/i.test(resp.title) || 1247 /\.jpe?g$/i.test(resp.title) || /\.pdf$/i.test(resp.title), 1248 null, null, null, headers); 1249 }); 1250 1251 fn(); 1252 } 1253 } 1254 catch (e) 1255 { 1256 if (error != null) 1257 { 1258 error(e); 1259 } 1260 else 1261 { 1262 throw e; 1263 } 1264 } 1265}; 1266 1267/** 1268 * Translates this point by the given vector. 1269 * 1270 * @param {number} dx X-coordinate of the translation. 1271 * @param {number} dy Y-coordinate of the translation. 1272 */ 1273DriveClient.prototype.saveFile = function(file, revision, success, errFn, noCheck, unloading, overwrite, properties, secret) 1274{ 1275 try 1276 { 1277 var retryCount = 0; 1278 file.saveLevel = 1; 1279 1280 var error = mxUtils.bind(this, function(e) 1281 { 1282 if (errFn != null) 1283 { 1284 errFn(e); 1285 } 1286 else 1287 { 1288 throw e; 1289 } 1290 1291 // Logs failed save 1292 try 1293 { 1294 if (!file.isConflict(e)) 1295 { 1296 var err = 'sl_' + file.saveLevel + '-error_' + 1297 (file.getErrorMessage(e) || 'unknown'); 1298 1299 if (e != null && e.error != null && e.error.code != null) 1300 { 1301 err += '-code_' + e.error.code; 1302 } 1303 1304 EditorUi.logEvent({category: 'ERROR-SAVE-FILE-' + file.getHash() + '-rev_' + 1305 file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + 1306 '-size_' + file.getSize() + '-mime_' + file.desc.mimeType + 1307 ((this.ui.editor.autosave) ? '' : '-nosave') + 1308 ((file.isAutosave()) ? '' : '-noauto') + 1309 ((file.changeListenerEnabled) ? '' : '-nolisten') + 1310 ((file.inConflictState) ? '-conflict' : '') + 1311 ((file.invalidChecksum) ? '-invalid' : ''), 1312 action: err, label: ((this.user != null) ? ('user_' + this.user.id) : 'nouser') + 1313 ((file.sync != null) ? ('-client_' + file.sync.clientId) : '-nosync')}); 1314 } 1315 } 1316 catch (ex) 1317 { 1318 // ignore 1319 } 1320 }); 1321 1322 var criticalError = mxUtils.bind(this, function(e) 1323 { 1324 error(e); 1325 1326 try 1327 { 1328 EditorUi.logError(e.message, null, null, e); 1329 1330// EditorUi.sendReport('Critical error in DriveClient.saveFile ' + 1331// new Date().toISOString() + ':' + 1332// '\n\nUserAgent=' + navigator.userAgent + 1333// '\nAppVersion=' + navigator.appVersion + 1334// '\nAppName=' + navigator.appName + 1335// '\nPlatform=' + navigator.platform + 1336// '\nFile=' + file.desc.id + '.' + file.desc.headRevisionId + 1337// '\nMime=' + file.desc.mimeType + 1338// '\nSize=' + file.getSize() + 1339// '\nUser=' + ((this.user != null) ? this.user.id : 'nouser') + 1340// ((file.sync != null) ? '-client_' + file.sync.clientId : '-nosync') + 1341// '\nSaveLevel=' + file.saveLevel + 1342// '\nSaveAsPng=' + (this.ui.useCanvasForExport && /(\.png)$/i.test(file.getTitle())) + 1343// '\nRetryCount=' + retryCount + 1344// '\nError=' + e + 1345// '\nMessage=' + e.message + 1346// '\n\nStack:\n' + e.stack); 1347 } 1348 catch (e) 1349 { 1350 // ignore 1351 } 1352 }); 1353 1354 if (file.isEditable() && file.desc != null) 1355 { 1356 var t0 = new Date().getTime(); 1357 var etag0 = file.desc.etag; 1358 var mod0 = file.desc.modifiedDate; 1359 var head0 = file.desc.headRevisionId; 1360 var saveAsPng = this.ui.useCanvasForExport && /(\.png)$/i.test(file.getTitle()); 1361 noCheck = (noCheck != null) ? noCheck : urlParams['ignoremime'] == '1'; 1362 1363 // NOTE: Unloading arg is currently ignored, saving during unload/beforeUnload is not possible using 1364 // asynchronous code, which is needed to create the thumbnail, or asynchronous requests which is the only 1365 // way to execute the gapi request below. 1366 // If no thumbnail is created and noCheck is true (which is always true if unloading is true) in which case 1367 // this code is synchronous, the executeRequest call is reached but the request is still not sent. This is 1368 // true for both, calls from beforeUnload and unload handlers. Note sure how to make the call synchronous 1369 // which is said to fix this when called from beforeUnload. 1370 // However, this would result in a missing thumbnail in most cases so a better solution might be to reduce 1371 // the autosave interval in DriveRealtime, but that would increase the number of requests. 1372 unloading = (unloading != null) ? unloading : false; 1373 var prevDesc = null; 1374 var pinned = false; 1375 var meta = 1376 { 1377 'mimeType': file.desc.mimeType, 1378 'title': file.getTitle() 1379 }; 1380 1381 // Overrides old mime type and creates a revision 1382 if (this.isGoogleRealtimeMimeType(meta.mimeType)) 1383 { 1384 meta.mimeType = this.xmlMimeType; 1385 prevDesc = file.desc; 1386 revision = true; 1387 pinned = true; 1388 } 1389 // Overrides mime type for unknown file type uploads 1390 else if (meta.mimeType == 'application/octet-stream' || 1391 (urlParams['override-mime'] == '1' && 1392 meta.mimeType != this.xmlMimeType)) 1393 { 1394 meta.mimeType = this.xmlMimeType; 1395 } 1396 1397 // Adds optional thumbnail to upload request 1398 var doSave = mxUtils.bind(this, function(thumb, thumbMime, keepExisting) 1399 { 1400 try 1401 { 1402 file.saveLevel = 3; 1403 1404 if (file.constructor == DriveFile) 1405 { 1406 if (properties == null) 1407 { 1408 properties = []; 1409 } 1410 1411 // Channel ID appended to file ID for comms 1412 if (file.getChannelId() == null) 1413 { 1414 properties.push({'key': 'channel', 'value': Editor.guid(32)}); 1415 } 1416 1417 // Key for encryption of comms 1418 if (file.getChannelKey() == null) 1419 { 1420 properties.push({'key': 'key', 'value': Editor.guid(32)}); 1421 } 1422 1423 // Pass to access cache for each etag 1424 properties.push({'key': 'secret', 'value': (secret != null) ? secret : Editor.guid(32)}); 1425 } 1426 1427 // Specifies that no thumbnail should be uploaded in which case the existing thumbnail is used 1428 if (!keepExisting) 1429 { 1430 // Uses placeholder thumbnail to replace existing one except when unloading 1431 // in which case the XML is updated but the existing thumbnail is not in order 1432 // to avoid executing asynchronous code and get the XML to the server instead 1433 if (thumb == null && !unloading) 1434 { 1435 thumb = this.placeholderThumbnail; 1436 thumbMime = this.placeholderMimeType; 1437 } 1438 1439 // Adds metadata for thumbnail 1440 if (thumb != null && thumbMime != null) 1441 { 1442 meta.thumbnail = 1443 { 1444 'image': thumb, 1445 'mimeType': thumbMime 1446 }; 1447 } 1448 } 1449 1450 var savedData = file.getData(); 1451 1452 // Updates saveDelay on drive file 1453 var wrapper = mxUtils.bind(this, function(resp) 1454 { 1455 try 1456 { 1457 file.saveDelay = new Date().getTime() - t0; 1458 file.saveLevel = 11; 1459 1460 if (resp == null) 1461 { 1462 error({message: mxResources.get('errorSavingFile') + ': Empty response'}); 1463 } 1464 else 1465 { 1466 // Checks if modified time is in the future and head revision has changed 1467 var delta = new Date(resp.modifiedDate).getTime() - new Date(mod0).getTime(); 1468 1469 if (delta <= 0 || etag0 == resp.etag || (revision && head0 == resp.headRevisionId)) 1470 { 1471 file.saveLevel = 12; 1472 var reasons = []; 1473 1474 if (delta <= 0) 1475 { 1476 reasons.push('invalid modified time'); 1477 } 1478 1479 if (etag0 == resp.etag) 1480 { 1481 reasons.push('stale etag'); 1482 } 1483 1484 if (revision && head0 == resp.headRevisionId) 1485 { 1486 reasons.push('stale revision'); 1487 } 1488 1489 var temp = reasons.join(', '); 1490 error({message: mxResources.get('errorSavingFile') + ': ' + temp}, resp); 1491 1492 // Logs failed save 1493 try 1494 { 1495 EditorUi.logError('Critical: Error saving to Google Drive ' + file.desc.id, 1496 null, 'from-' + head0 + '.' + mod0 + '-' + this.ui.hashValue(etag0) + 1497 '-to-' + resp.headRevisionId + '.' + resp.modifiedDate + '-' + 1498 this.ui.hashValue(resp.etag) + ((temp.length > 0) ? '-errors-' + temp : ''), 1499 'user-' + ((this.user != null) ? this.user.id : 'nouser') + 1500 ((file.sync != null) ? '-client_' + file.sync.clientId : '-nosync')); 1501 } 1502 catch (e) 1503 { 1504 // ignore 1505 } 1506 } 1507 else 1508 { 1509 file.saveLevel = null; 1510 success(resp, savedData); 1511 1512 if (prevDesc != null) 1513 { 1514 // Pins previous revision 1515 this.executeRequest({ 1516 url: '/files/' + prevDesc.id + '/revisions/' + prevDesc.headRevisionId + '?supportsAllDrives=true' 1517 }, mxUtils.bind(this, mxUtils.bind(this, function(resp) 1518 { 1519 resp.pinned = true; 1520 1521 this.executeRequest({ 1522 url: '/files/' + prevDesc.id + '/revisions/' + prevDesc.headRevisionId, 1523 method: 'PUT', 1524 params: resp 1525 }); 1526 }))); 1527 1528 // Logs conversion 1529 try 1530 { 1531 EditorUi.logEvent({category: file.convertedFrom + '-CONVERT-FILE-' + file.getHash(), 1532 action: 'from_' + prevDesc.id + '.' + prevDesc.headRevisionId + 1533 '-to_' + file.desc.id + '.' + file.desc.headRevisionId, 1534 label: (this.user != null) ? ('user_' + this.user.id) : 'nouser' + 1535 ((file.sync != null) ? '-client_' + file.sync.clientId : 'nosync')}); 1536 } 1537 catch (e) 1538 { 1539 // ignore 1540 } 1541 } 1542 1543 // Logs successful save 1544// try 1545// { 1546// EditorUi.logEvent({category: 'SUCCESS-SAVE-FILE-' + file.getHash() + 1547// '-rev0_' + head0 + '-mod0_' + mod0, 1548// action: 'rev-' + resp.headRevisionId + 1549// '-mod_' + resp.modifiedDate + '-size_' + file.getSize() + 1550// '-mime_' + file.desc.mimeType + 1551// ((this.ui.editor.autosave) ? '' : '-nosave') + 1552// ((file.isAutosave()) ? '' : '-noauto') + 1553// ((file.changeListenerEnabled) ? '' : '-nolisten') + 1554// ((file.inConflictState) ? '-conflict' : '') + 1555// ((file.invalidChecksum) ? '-invalid' : ''), 1556// label: ((this.user != null) ? ('user_' + this.user.id) : 'nouser') + 1557// ((file.sync != null) ? ('-client_' + file.sync.clientId) : '-nosync')}); 1558// } 1559// catch (e) 1560// { 1561// // ignore 1562// } 1563 } 1564 } 1565 } 1566 catch (e) 1567 { 1568 criticalError(e); 1569 } 1570 }); 1571 1572 var doExecuteRequest = mxUtils.bind(this, function(data, binary) 1573 { 1574 file.saveLevel = 4; 1575 1576 try 1577 { 1578 if (properties != null) 1579 { 1580 meta.properties = properties; 1581 } 1582 1583 // Used to check if file was changed externally 1584 var etag = (!overwrite && file.constructor == DriveFile && 1585 (DrawioFile.SYNC == 'manual' || DrawioFile.SYNC == 'auto')) ? 1586 file.getCurrentEtag() : null; 1587 1588 var doExecuteSave = mxUtils.bind(this, function(realOverwrite) 1589 { 1590 file.saveLevel = 5; 1591 1592 try 1593 { 1594 var unknown = file.desc.mimeType != this.xmlMimeType && file.desc.mimeType != this.mimeType && 1595 file.desc.mimeType != this.libraryMimeType; 1596 var acceptResponse = true; 1597 var timeoutThread = null; 1598 1599 // Allow for re-auth flow with 5x timeout 1600 try 1601 { 1602 timeoutThread = window.setTimeout(mxUtils.bind(this, function() 1603 { 1604 acceptResponse = false; 1605 error({code: App.ERROR_TIMEOUT}); 1606 }), 5 * this.ui.timeout); 1607 } 1608 catch (e) 1609 { 1610 // Ignore window closed 1611 } 1612 1613 this.executeRequest(this.createUploadRequest(file.getId(), meta, 1614 data, revision || realOverwrite || unknown, binary, 1615 (realOverwrite) ? null : etag, pinned), mxUtils.bind(this, function(resp) 1616 { 1617 window.clearTimeout(timeoutThread); 1618 1619 if (acceptResponse) 1620 { 1621 wrapper(resp); 1622 } 1623 }), mxUtils.bind(this, function(err) 1624 { 1625 window.clearTimeout(timeoutThread); 1626 1627 if (acceptResponse) 1628 { 1629 file.saveLevel = 6; 1630 1631 try 1632 { 1633 if (!file.isConflict(err)) 1634 { 1635 error(err); 1636 } 1637 else 1638 { 1639 // Workaround for correct etag and Google always returns 412 conflict error (stale etag) 1640 this.executeRequest({ 1641 url: '/files/' + file.getId() + '?supportsAllDrives=true&fields=' + this.catchupFields 1642 }, 1643 mxUtils.bind(this, function(resp) 1644 { 1645 file.saveLevel = 7; 1646 1647 try 1648 { 1649 // Stale etag detected, retry with delay 1650 if (resp != null && resp.etag == etag) 1651 { 1652 if (retryCount < this.staleEtagMaxRetries) 1653 { 1654 retryCount++; 1655 var jitter = 1 + 0.1 * (Math.random() - 0.5); 1656 var delay = retryCount * 2 * this.coolOff * jitter; 1657 window.setTimeout(executeSave, delay); 1658 1659 if (urlParams['test'] == '1') 1660 { 1661 EditorUi.debug('DriveClient: Stale Etag Detected', 1662 'retry', retryCount, 'delay', delay); 1663 } 1664 } 1665 else 1666 { 1667 executeSave(true); 1668 1669 // Logs overwrite 1670 try 1671 { 1672 EditorUi.logEvent({category: 'STALE-ETAG-SAVE-FILE-' + file.getHash(), 1673 action: 'rev_' + file.desc.headRevisionId + '-mod_' + file.desc.modifiedDate + 1674 '-size_' + file.getSize() + '-mime_' + file.desc.mimeType + 1675 ((this.ui.editor.autosave) ? '' : '-nosave') + 1676 ((file.isAutosave()) ? '' : '-noauto') + 1677 ((file.changeListenerEnabled) ? '' : '-nolisten') + 1678 ((file.inConflictState) ? '-conflict' : '') + 1679 ((file.invalidChecksum) ? '-invalid' : ''), 1680 label: ((this.user != null) ? ('user_' + this.user.id) : 'nouser') + 1681 ((file.sync != null) ? ('-client_' + file.sync.clientId) : '-nosync')}); 1682 } 1683 catch (e) 1684 { 1685 // ignore 1686 } 1687 } 1688 } 1689 else 1690 { 1691 1692 if (urlParams['test'] == '1' && resp.headRevisionId == head0) 1693 { 1694 EditorUi.debug('DriveClient: Remote Etag Changed', 1695 'local', etag, 'remote', resp.etag, 1696 'rev', file.desc.headRevisionId, 1697 'response', [resp], 'file', [file]); 1698 } 1699 1700 error(err, resp); 1701 } 1702 } 1703 catch (e) 1704 { 1705 criticalError(e); 1706 } 1707 }), mxUtils.bind(this, function() 1708 { 1709 error(err); 1710 })); 1711 } 1712 } 1713 catch (e) 1714 { 1715 criticalError(e); 1716 } 1717 } 1718 })); 1719 } 1720 catch (e) 1721 { 1722 criticalError(e); 1723 } 1724 }); 1725 1726 // Workaround for Google returning the wrong etag after file save is to 1727 // update the etag before save and check if the headRevisionId changed 1728 var executeSave = mxUtils.bind(this, function(realOverwrite) 1729 { 1730 file.saveLevel = 9; 1731 1732 if (realOverwrite || etag == null) 1733 { 1734 doExecuteSave(realOverwrite); 1735 } 1736 else 1737 { 1738 var acceptResponse = true; 1739 var timeoutThread = null; 1740 1741 // Allow for re-auth flow with 3x timeout 1742 try 1743 { 1744 timeoutThread = window.setTimeout(mxUtils.bind(this, function() 1745 { 1746 acceptResponse = false; 1747 error({code: App.ERROR_TIMEOUT}); 1748 }), 3 * this.ui.timeout); 1749 } 1750 catch (e) 1751 { 1752 // Ignore window closed 1753 } 1754 1755 this.executeRequest({ 1756 url: '/files/' + file.getId() + '?supportsAllDrives=true&fields=' + this.catchupFields 1757 }, 1758 mxUtils.bind(this, function(desc2) 1759 { 1760 window.clearTimeout(timeoutThread); 1761 1762 if (acceptResponse) 1763 { 1764 file.saveLevel = 10; 1765 1766 try 1767 { 1768 // Checks head revision ID and updates etag or returns conflict 1769 if (desc2 != null && desc2.headRevisionId == head0) 1770 { 1771 if (urlParams['test'] == '1' && etag != desc2.etag) 1772 { 1773 EditorUi.debug('DriveClient: Preflight Etag Update', 1774 'from', etag, 'to', desc2.etag, 1775 'rev', file.desc.headRevisionId, 1776 'response', [desc2], 'file', [file]); 1777 } 1778 1779 etag = desc2.etag; 1780 doExecuteSave(realOverwrite); 1781 } 1782 else 1783 { 1784 error({error: {code: 412}}, desc2); 1785 } 1786 } 1787 catch (e) 1788 { 1789 criticalError(e); 1790 } 1791 } 1792 }), mxUtils.bind(this, function(err) 1793 { 1794 // Simulated 1795 window.clearTimeout(timeoutThread); 1796 1797 if (acceptResponse) 1798 { 1799 file.saveLevel = 11; 1800 error(err); 1801 } 1802 })); 1803 } 1804 }); 1805 1806 // Uses saved PNG data for thumbnail 1807 if (saveAsPng && thumb == null) 1808 { 1809 file.saveLevel = 8; 1810 var img = new Image(); 1811 1812 img.onload = mxUtils.bind(this, function() 1813 { 1814 try 1815 { 1816 var s = this.thumbnailWidth / img.width; 1817 1818 var canvas = document.createElement('canvas'); 1819 canvas.width = this.thumbnailWidth; 1820 canvas.height = Math.floor(img.height * s); 1821 1822 var ctx = canvas.getContext('2d'); 1823 ctx.drawImage(img, 0, 0, canvas.width, canvas.height); 1824 1825 var temp = canvas.toDataURL(); 1826 temp = temp.substring(temp.indexOf(',') + 1).replace(/\+/g, '-').replace(/\//g, '_'); 1827 1828 meta.thumbnail = 1829 { 1830 'image': temp, 1831 'mimeType': 'image/png' 1832 }; 1833 1834 executeSave(false); 1835 } 1836 catch (e) 1837 { 1838 try 1839 { 1840 executeSave(false) 1841 } 1842 catch (e2) 1843 { 1844 criticalError(e2); 1845 } 1846 } 1847 }); 1848 1849 img.src = 'data:image/png;base64,' + data; 1850 } 1851 else 1852 { 1853 executeSave(false); 1854 } 1855 } 1856 catch (e) 1857 { 1858 criticalError(e); 1859 } 1860 }); 1861 1862 if (saveAsPng) 1863 { 1864 var p = this.ui.getPngFileProperties(this.ui.fileNode); 1865 1866 this.ui.getEmbeddedPng(mxUtils.bind(this, function(data) 1867 { 1868 doExecuteRequest(data, true); 1869 }), error, (this.ui.getCurrentFile() != file) ? 1870 savedData : null, p.scale, p.border); 1871 } 1872 else 1873 { 1874 doExecuteRequest(savedData, false); 1875 } 1876 } 1877 catch (e) 1878 { 1879 criticalError(e); 1880 } 1881 }); 1882 1883 // Indirection to generate thumbnails if enabled and supported 1884 // (required because generation of thumbnails is asynchronous) 1885 try 1886 { 1887 file.saveLevel = 2; 1888 1889 // NOTE: getThumbnail is asynchronous and returns false if no thumbnails can be created 1890 if (unloading || saveAsPng || file.constructor == DriveLibrary || !this.enableThumbnails || urlParams['thumb'] == '0' || 1891 (meta.mimeType != null && meta.mimeType.substring(0, 29) != 'application/vnd.jgraph.mxfile') || 1892 !this.ui.getThumbnail(this.thumbnailWidth, mxUtils.bind(this, function(canvas) 1893 { 1894 // Callback for getThumbnail 1895 try 1896 { 1897 var thumb = null; 1898 1899 try 1900 { 1901 if (canvas != null) 1902 { 1903 // Security errors are possible 1904 thumb = canvas.toDataURL('image/png'); 1905 } 1906 1907 // Maximum thumbnail size is 2MB 1908 if (thumb != null) 1909 { 1910 if (thumb.length > this.maxThumbnailSize) 1911 { 1912 thumb = null; 1913 } 1914 else 1915 { 1916 // Converts base64 data into required format for Drive (base64url with no prefix) 1917 thumb = thumb.substring(thumb.indexOf(',') + 1).replace(/\+/g, '-').replace(/\//g, '_'); 1918 } 1919 } 1920 } 1921 catch (e) 1922 { 1923 thumb = null; 1924 } 1925 1926 doSave(thumb, 'image/png'); 1927 } 1928 catch (e) 1929 { 1930 criticalError(e); 1931 } 1932 }))) 1933 { 1934 // If-branch 1935 doSave(null, null, file.constructor != DriveLibrary); 1936 } 1937 } 1938 catch (e) 1939 { 1940 criticalError(e); 1941 } 1942 } 1943 else 1944 { 1945 this.ui.editor.graph.reset(); 1946 error({message: mxResources.get('readOnly')}); 1947 } 1948 } 1949 catch (e) 1950 { 1951 criticalError(e); 1952 } 1953}; 1954 1955/** 1956 * Translates this point by the given vector. 1957 * 1958 * @param {number} dx X-coordinate of the translation. 1959 * @param {number} dy Y-coordinate of the translation. 1960 */ 1961DriveClient.prototype.insertFile = function(title, data, folderId, success, error, mimeType, binary) 1962{ 1963 mimeType = (mimeType != null) ? mimeType : this.xmlMimeType; 1964 1965 var metadata = 1966 { 1967 'mimeType': mimeType, 1968 'title': title 1969 }; 1970 1971 if (folderId != null) 1972 { 1973 metadata.parents = [{'kind': 'drive#fileLink', 'id': folderId}]; 1974 } 1975 1976 // NOTE: Cannot create thumbnail on insert since no ui has no current file 1977 this.executeRequest(this.createUploadRequest(null, metadata, data, false, binary), mxUtils.bind(this, function(resp) 1978 { 1979 if (mimeType == this.libraryMimeType) 1980 { 1981 success(new DriveLibrary(this.ui, data, resp)); 1982 } 1983 else if (resp == false) 1984 { 1985 if (error != null) 1986 { 1987 error({message: mxResources.get('errorSavingFile')}); 1988 } 1989 } 1990 else 1991 { 1992 success(new DriveFile(this.ui, data, resp)); 1993 } 1994 }), error); 1995}; 1996 1997/** 1998 * Translates this point by the given vector. 1999 * 2000 * @param {number} dx X-coordinate of the translation. 2001 * @param {number} dy Y-coordinate of the translation. 2002 */ 2003DriveClient.prototype.createUploadRequest = function(id, metadata, data, revision, binary, etag, pinned) 2004{ 2005 binary = (binary != null) ? binary : false; 2006 var bd = '-------314159265358979323846'; 2007 var delim = '\r\n--' + bd + '\r\n'; 2008 var close = '\r\n--' + bd + '--'; 2009 var ctype = 'application/octect-stream'; 2010 2011 var headers = {'Content-Type' : 'multipart/mixed; boundary="' + bd + '"'}; 2012 2013 if (etag != null) 2014 { 2015 headers['If-Match'] = etag; 2016 } 2017 2018 var reqObj = 2019 { 2020 'fullUrl': 'https://content.googleapis.com/upload/drive/v2/files' + (id != null ? '/' + id : '') + 2021 '?uploadType=multipart&supportsAllDrives=true&enforceSingleParent=true&fields=' + this.allFields, 2022 'method': (id != null) ? 'PUT' : 'POST', 2023 'headers': headers, 2024 'params': delim + 'Content-Type: application/json\r\n\r\n' + JSON.stringify(metadata) + delim + 2025 'Content-Type: ' + ctype + '\r\n' + 'Content-Transfer-Encoding: base64\r\n' + '\r\n' + 2026 ((data != null) ? ((binary) ? data : ((window.btoa && !mxClient.IS_IE && !mxClient.IS_IE11) ? 2027 Graph.base64EncodeUnicode(data) : Base64.encode(data))) : '') + close 2028 } 2029 2030 if (!revision) 2031 { 2032 reqObj.fullUrl += '&newRevision=false'; 2033 } 2034 2035 if (pinned) 2036 { 2037 reqObj.fullUrl += '&pinned=true'; 2038 } 2039 2040 return reqObj; 2041}; 2042 2043/** 2044 * Translates this point by the given vector. 2045 * 2046 * @param {number} dx X-coordinate of the translation. 2047 * @param {number} dy Y-coordinate of the translation. 2048 */ 2049DriveClient.prototype.createLinkPicker = function() 2050{ 2051 var name = 'linkPicker'; 2052 var picker = pickers[name]; 2053 2054 if (picker == null || pickers[name + 'Token'] != _token) 2055 { 2056 pickers[name + 'Token'] = _token; 2057 2058 var view = new google.picker.DocsView(google.picker.ViewId.FOLDERS) 2059 .setParent('root') 2060 .setIncludeFolders(true) 2061 .setSelectFolderEnabled(true); 2062 var view2 = new google.picker.DocsView() 2063 .setIncludeFolders(true) 2064 .setSelectFolderEnabled(true); 2065 var view21 = new google.picker.DocsView() 2066 .setIncludeFolders(true) 2067 .setEnableDrives(true) 2068 .setSelectFolderEnabled(true); 2069 picker = new google.picker.PickerBuilder() 2070 .setAppId(this.appId) 2071 .setLocale(mxLanguage) 2072 .setOAuthToken(pickers[name + 'Token']) 2073 .enableFeature(google.picker.Feature.SUPPORT_DRIVES) 2074 .addView(view) 2075 .addView(view2) 2076 .addView(view21) 2077 .addView(google.picker.ViewId.RECENTLY_PICKED); 2078 } 2079 2080 return picker; 2081}; 2082 2083/** 2084 * Translates this point by the given vector. 2085 * 2086 * @param {number} dx X-coordinate of the translation. 2087 * @param {number} dy Y-coordinate of the translation. 2088 */ 2089DriveClient.prototype.pickFile = function(fn, acceptAllFiles, cancelFn) 2090{ 2091 this.filePickerCallback = (fn != null) ? fn : mxUtils.bind(this, function(id) 2092 { 2093 this.ui.loadFile('G' + id); 2094 }); 2095 2096 this.filePicked = mxUtils.bind(this, function(data) 2097 { 2098 if (data.action == google.picker.Action.PICKED) 2099 { 2100 this.filePickerCallback(data.docs[0].id, data.docs[0]); 2101 } 2102 }); 2103 2104 if (this.ui.spinner.spin(document.body, mxResources.get('authorizing'))) 2105 { 2106 this.execute(mxUtils.bind(this, function() 2107 { 2108 try 2109 { 2110 this.ui.spinner.stop(); 2111 2112 // Reuses picker as long as token doesn't change. 2113 var name = (acceptAllFiles) ? 'genericPicker' : 'filePicker'; 2114 2115 // Click on background closes dialog as workaround for blocking dialog 2116 // states such as 401 where the dialog cannot be closed and blocks UI 2117 var exit = mxUtils.bind(this, function(evt) 2118 { 2119 // Workaround for click from appIcon on second call 2120 if (mxEvent.getSource(evt).className == 'picker modal-dialog-bg picker-dialog-bg') 2121 { 2122 mxEvent.removeListener(document, 'click', exit); 2123 this[name].setVisible(false); 2124 2125 if (cancelFn) 2126 { 2127 cancelFn(); 2128 } 2129 } 2130 }); 2131 2132 if (pickers[name] == null || pickers[name + 'Token'] != _token) 2133 { 2134 // FIXME: Dispose not working 2135 // if (pickers[name] != null) 2136 // { 2137 // console.log(name, pickers[name]); 2138 // pickers[name].dispose(); 2139 // } 2140 2141 pickers[name + 'Token'] = _token; 2142 2143 // Pseudo-hierarchical directory view, see 2144 // https://groups.google.com/forum/#!topic/google-picker-api/FSFcuJe7icQ 2145 var view = new google.picker.DocsView(google.picker.ViewId.FOLDERS) 2146 .setParent('root') 2147 .setIncludeFolders(true); 2148 2149 var view2 = new google.picker.DocsView() 2150 .setIncludeFolders(true); 2151 2152 var view3 = new google.picker.DocsView() 2153 .setEnableDrives(true) 2154 .setIncludeFolders(true); 2155 2156 var view4 = new google.picker.DocsUploadView() 2157 .setIncludeFolders(true); 2158 2159 if (!acceptAllFiles) 2160 { 2161 view.setMimeTypes(this.mimeTypes); 2162 view2.setMimeTypes(this.mimeTypes); 2163 view3.setMimeTypes(this.mimeTypes); 2164 } 2165 else 2166 { 2167 view.setMimeTypes('*/*'); 2168 view2.setMimeTypes('*/*'); 2169 view3.setMimeTypes('*/*'); 2170 } 2171 2172 pickers[name] = new google.picker.PickerBuilder() 2173 .setOAuthToken(pickers[name + 'Token']) 2174 .setLocale(mxLanguage) 2175 .setAppId(this.appId) 2176 .enableFeature(google.picker.Feature.SUPPORT_DRIVES) 2177 .addView(view) 2178 .addView(view2) 2179 .addView(view3) 2180 .addView(google.picker.ViewId.RECENTLY_PICKED) 2181 .addView(view4); 2182 2183 if (urlParams['gPickerSize']) 2184 { 2185 var cSize = urlParams['gPickerSize'].split(','); 2186 pickers[name] = pickers[name].setSize(cSize[0], cSize[1]); 2187 } 2188 2189 if (urlParams['topBaseUrl']) 2190 { 2191 pickers[name] = pickers[name].setOrigin(decodeURIComponent(urlParams['topBaseUrl'])); 2192 } 2193 2194 pickers[name] = pickers[name].setCallback(mxUtils.bind(this, function(data) 2195 { 2196 if (data.action == google.picker.Action.PICKED || 2197 data.action == google.picker.Action.CANCEL) 2198 { 2199 mxEvent.removeListener(document, 'click', exit); 2200 2201 if (cancelFn && data.action == google.picker.Action.CANCEL) 2202 { 2203 cancelFn(); 2204 } 2205 } 2206 2207 if (data.action == google.picker.Action.PICKED) 2208 { 2209 this.filePicked(data); 2210 } 2211 })).build(); 2212 } 2213 2214 mxEvent.addListener(document, 'click', exit); 2215 pickers[name].setVisible(true); 2216 } 2217 catch (e) 2218 { 2219 this.ui.spinner.stop(); 2220 this.ui.handleError(e); 2221 } 2222 })); 2223 } 2224}; 2225 2226/** 2227 * Translates this point by the given vector. 2228 * 2229 * @param {number} dx X-coordinate of the translation. 2230 * @param {number} dy Y-coordinate of the translation. 2231 */ 2232DriveClient.prototype.pickFolder = function(fn, force) 2233{ 2234 this.folderPickerCallback = fn; 2235 2236 // Picker is initialized once and points to this function 2237 // which is overridden each time to the picker is shown 2238 var showPicker = mxUtils.bind(this, function() 2239 { 2240 try 2241 { 2242 if (this.ui.spinner.spin(document.body, mxResources.get('authorizing'))) 2243 { 2244 this.execute(mxUtils.bind(this, function() 2245 { 2246 try 2247 { 2248 this.ui.spinner.stop(); 2249 2250 // Reuses picker as long as token doesn't change. 2251 var name = 'folderPicker'; 2252 2253 // Click on background closes dialog as workaround for blocking dialog 2254 // states such as 401 where the dialog cannot be closed and blocks UI 2255 var exit = mxUtils.bind(this, function(evt) 2256 { 2257 // Workaround for click from appIcon on second call 2258 if (mxEvent.getSource(evt).className == 'picker modal-dialog-bg picker-dialog-bg') 2259 { 2260 mxEvent.removeListener(document, 'click', exit); 2261 pickers[name].setVisible(false); 2262 } 2263 }); 2264 2265 if (pickers[name] == null || pickers[name + 'Token'] != _token) 2266 { 2267 // FIXME: Dispose not working 2268 // if (pickers[name] != null) 2269 // { 2270 // console.log(name, pickers[name]); 2271 // pickers[name].dispose(); 2272 // } 2273 2274 pickers[name + 'Token'] = _token; 2275 2276 // Pseudo-hierarchical directory view, see 2277 // https://groups.google.com/forum/#!topic/google-picker-api/FSFcuJe7icQ 2278 var view = new google.picker.DocsView(google.picker.ViewId.FOLDERS) 2279 .setParent('root') 2280 .setIncludeFolders(true) 2281 .setSelectFolderEnabled(true) 2282 .setMimeTypes('application/vnd.google-apps.folder'); 2283 2284 var view2 = new google.picker.DocsView() 2285 .setIncludeFolders(true) 2286 .setSelectFolderEnabled(true) 2287 .setMimeTypes('application/vnd.google-apps.folder'); 2288 2289 var view3 = new google.picker.DocsView() 2290 .setIncludeFolders(true) 2291 .setEnableDrives(true) 2292 .setSelectFolderEnabled(true) 2293 .setMimeTypes('application/vnd.google-apps.folder'); 2294 2295 pickers[name] = new google.picker.PickerBuilder() 2296 .setSelectableMimeTypes('application/vnd.google-apps.folder') 2297 .setOAuthToken(pickers[name + 'Token']) 2298 .setLocale(mxLanguage) 2299 .setAppId(this.appId) 2300 .enableFeature(google.picker.Feature.SUPPORT_DRIVES) 2301 .addView(view) 2302 .addView(view2) 2303 .addView(view3) 2304 .addView(google.picker.ViewId.RECENTLY_PICKED) 2305 .setTitle(mxResources.get('pickFolder')); 2306 2307 if (urlParams['gPickerSize']) 2308 { 2309 var cSize = urlParams['gPickerSize'].split(','); 2310 pickers[name] = pickers[name].setSize(cSize[0], cSize[1]); 2311 } 2312 2313 if (urlParams['topBaseUrl']) 2314 { 2315 pickers[name] = pickers[name].setOrigin(decodeURIComponent(urlParams['topBaseUrl'])); 2316 } 2317 2318 pickers[name] = pickers[name].setCallback(mxUtils.bind(this, function(data) 2319 { 2320 if (data.action == google.picker.Action.PICKED || 2321 data.action == google.picker.Action.CANCEL) 2322 { 2323 mxEvent.removeListener(document, 'click', exit); 2324 } 2325 2326 this.folderPickerCallback(data); 2327 })).build(); 2328 } 2329 2330 mxEvent.addListener(document, 'click', exit); 2331 pickers[name].setVisible(true); 2332 } 2333 catch (e) 2334 { 2335 this.ui.spinner.stop(); 2336 this.ui.handleError(e); 2337 } 2338 })); 2339 } 2340 } 2341 catch (e) 2342 { 2343 this.ui.handleError(e); 2344 } 2345 }); 2346 2347 if (force) 2348 { 2349 showPicker(); 2350 } 2351 else 2352 { 2353 this.ui.confirm(mxResources.get('useRootFolder'), mxUtils.bind(this, function() 2354 { 2355 this.folderPickerCallback({action: google.picker.Action.PICKED, 2356 docs: [{type: 'folder', id: 'root'}]}); 2357 }), mxUtils.bind(this, function() 2358 { 2359 showPicker(); 2360 }), mxResources.get('yes'), mxResources.get('noPickFolder') + '...', true); 2361 } 2362}; 2363 2364/** 2365 * Translates this point by the given vector. 2366 * 2367 * @param {number} dx X-coordinate of the translation. 2368 * @param {number} dy Y-coordinate of the translation. 2369 */ 2370DriveClient.prototype.pickLibrary = function(fn) 2371{ 2372 this.filePickerCallback = fn; 2373 2374 this.filePicked = mxUtils.bind(this, function(data) 2375 { 2376 if (data.action == google.picker.Action.PICKED) 2377 { 2378 this.filePickerCallback(data.docs[0].id); 2379 } 2380 else if (data.action == google.picker.Action.CANCEL && this.ui.getCurrentFile() == null) 2381 { 2382 this.ui.showSplash(); 2383 } 2384 }); 2385 2386 if (this.ui.spinner.spin(document.body, mxResources.get('authorizing'))) 2387 { 2388 this.execute(mxUtils.bind(this, function() 2389 { 2390 try 2391 { 2392 this.ui.spinner.stop(); 2393 2394 // Click on background closes dialog as workaround for blocking dialog 2395 // states such as 401 where the dialog cannot be closed and blocks UI 2396 var exit = mxUtils.bind(this, function(evt) 2397 { 2398 // Workaround for click from appIcon on second call 2399 if (mxEvent.getSource(evt).className == 'picker modal-dialog-bg picker-dialog-bg') 2400 { 2401 mxEvent.removeListener(document, 'click', exit); 2402 pickers.libraryPicker.setVisible(false); 2403 } 2404 }); 2405 2406 // Reuses picker as long as token doesn't change 2407 2408 if (pickers.libraryPicker == null || pickers.libraryPickerToken != _token) 2409 { 2410 // FIXME: Dispose not working 2411 // if (pickers[name] != null) 2412 // { 2413 // console.log(name, pickers[name]); 2414 // pickers[name].dispose(); 2415 // } 2416 2417 pickers.libraryPickerToken = _token; 2418 2419 // Pseudo-hierarchical directory view, see 2420 // https://groups.google.com/forum/#!topic/google-picker-api/FSFcuJe7icQ 2421 var view = new google.picker.DocsView(google.picker.ViewId.FOLDERS) 2422 .setParent('root') 2423 .setIncludeFolders(true) 2424 .setMimeTypes(this.libraryMimeType + ',application/xml,text/plain,application/octet-stream'); 2425 2426 var view2 = new google.picker.DocsView() 2427 .setIncludeFolders(true) 2428 .setMimeTypes(this.libraryMimeType + ',application/xml,text/plain,application/octet-stream'); 2429 2430 var view3 = new google.picker.DocsView() 2431 .setEnableDrives(true) 2432 .setIncludeFolders(true) 2433 .setMimeTypes(this.libraryMimeType + ',application/xml,text/plain,application/octet-stream'); 2434 2435 var view4 = new google.picker.DocsUploadView() 2436 .setIncludeFolders(true); 2437 2438 pickers.libraryPicker = new google.picker.PickerBuilder() 2439 .setOAuthToken(pickers.libraryPickerToken) 2440 .setLocale(mxLanguage) 2441 .setAppId(this.appId) 2442 .enableFeature(google.picker.Feature.SUPPORT_DRIVES) 2443 .addView(view) 2444 .addView(view2) 2445 .addView(view3) 2446 .addView(google.picker.ViewId.RECENTLY_PICKED) 2447 .addView(view4); 2448 2449 if (urlParams['gPickerSize']) 2450 { 2451 var cSize = urlParams['gPickerSize'].split(','); 2452 pickers.libraryPicker = pickers.libraryPicker.setSize(cSize[0], cSize[1]); 2453 } 2454 2455 if (urlParams['topBaseUrl']) 2456 { 2457 pickers.libraryPicker = pickers.libraryPicker.setOrigin(decodeURIComponent(urlParams['topBaseUrl'])); 2458 } 2459 2460 pickers.libraryPicker = pickers.libraryPicker.setCallback(mxUtils.bind(this, function(data) 2461 { 2462 if (data.action == google.picker.Action.PICKED || 2463 data.action == google.picker.Action.CANCEL) 2464 { 2465 mxEvent.removeListener(document, 'click', exit); 2466 } 2467 2468 if (data.action == google.picker.Action.PICKED) 2469 { 2470 this.filePicked(data); 2471 } 2472 })).build(); 2473 } 2474 2475 mxEvent.addListener(document, 'click', exit); 2476 pickers.libraryPicker.setVisible(true); 2477 } 2478 catch (e) 2479 { 2480 this.ui.spinner.stop(); 2481 this.ui.handleError(e); 2482 } 2483 })); 2484 } 2485}; 2486 2487/** 2488 * Translates this point by the given vector. 2489 * 2490 * @param {number} dx X-coordinate of the translation. 2491 * @param {number} dy Y-coordinate of the translation. 2492 */ 2493DriveClient.prototype.showPermissions = function(id) 2494{ 2495 var fallback = mxUtils.bind(this, function() 2496 { 2497 var dlg = new ConfirmDialog(this.ui, mxResources.get('googleSharingNotAvailable'), mxUtils.bind(this, function() 2498 { 2499 this.ui.editor.graph.openLink('https://drive.google.com/open?id=' + id); 2500 }), null, mxResources.get('open'), null, null, null, null, IMAGE_PATH + '/google-share.png'); 2501 this.ui.showDialog(dlg.container, 360, 190, true, true); 2502 dlg.init(); 2503 }); 2504 2505 if (this.sharingFailed) 2506 { 2507 fallback(); 2508 } 2509 else 2510 { 2511 this.checkToken(mxUtils.bind(this, function() 2512 { 2513 try 2514 { 2515 var shareClient = new gapi.drive.share.ShareClient(this.appId); 2516 shareClient.setOAuthToken(_token); 2517 shareClient.setItemIds([id]); 2518 shareClient.showSettingsDialog(); 2519 2520 // Workaround for https://stackoverflow.com/questions/54753169 is to check 2521 // if "sharing is unavailable" is showing and invoke a fallback dialog 2522 if ('MutationObserver' in window) 2523 { 2524 if (this.sharingObserver != null) 2525 { 2526 this.sharingObserver.disconnect(); 2527 this.sharingObserver = null; 2528 } 2529 2530 // Tries again even if observer was still around as the user may have 2531 // closed the dialog while waiting. TODO: Find condition to disconnect 2532 // observer when dialog is closed (use removedNodes?). 2533 this.sharingObserver = new MutationObserver(mxUtils.bind(this, function(mutations) 2534 { 2535 var done = false; 2536 2537 for (var i = 0; i < mutations.length; i++) 2538 { 2539 for (var j = 0; j < mutations[i].addedNodes.length; j++) 2540 { 2541 var child = mutations[i].addedNodes[j]; 2542 2543 if (child.nodeName == 'BUTTON' && child.getAttribute('name') == 'ok' && 2544 child.parentNode != null && child.parentNode.parentNode != null && 2545 child.parentNode.parentNode.getAttribute('role') == 'dialog') 2546 { 2547 this.sharingFailed = true; 2548 child.click(); 2549 fallback(); 2550 done = true; 2551 } 2552 else if (child.nodeName == 'DIV' && child.className == 'shr-q-shr-r-shr-xb') 2553 { 2554 done = true; 2555 } 2556 } 2557 } 2558 2559 if (done) 2560 { 2561 this.sharingObserver.disconnect(); 2562 this.sharingObserver = null; 2563 } 2564 2565 })); 2566 2567 this.sharingObserver.observe(document, {childList: true, subtree: true}); 2568 } 2569 } 2570 catch (e) 2571 { 2572 this.ui.handleError(e); 2573 } 2574 })); 2575 } 2576}; 2577 2578DriveClient.prototype.clearPersistentToken = function() 2579{ 2580 //Since we have multiple accounts now, full deletion is not possible 2581 var authInfo = JSON.parse(this.getPersistentToken(true)) || {}; 2582 2583 //Delete current user info 2584 delete authInfo.current; 2585 delete authInfo[this.userId]; 2586 2587 //Set the next user as current 2588 for (var id in authInfo) 2589 { 2590 authInfo.current = {userId: id, expires: 0}; //An expired token 2591 break; 2592 } 2593 2594 DrawioClient.prototype.setPersistentToken.call(this, JSON.stringify(authInfo)); 2595}; 2596 2597DriveClient.prototype.setPersistentToken = function(userAuthInfo, sessionOnly) 2598{ 2599 var authInfo = JSON.parse(this.getPersistentToken(true)) || {}; 2600 2601 userAuthInfo.userId = this.userId; 2602 authInfo.current = userAuthInfo; 2603 authInfo[this.userId] = { 2604 user: this.user 2605 }; 2606 2607 DrawioClient.prototype.setPersistentToken.call(this, JSON.stringify(authInfo), sessionOnly); 2608}; 2609 2610})();