get_storage(); // Update upload ID $this->jobdata_set('upload_id_'.$this->current_file_hash, $uploadid); $this->jobdata_set('upload_offset_'.$this->current_file_hash, $offset); $time_now = microtime(true); $time_since_last_tick = $time_now - $this->upload_tick; $data_since_last_tick = $offset - $this->uploaded_offset; $this->upload_tick = $time_now; $this->uploaded_offset = $offset; // Here we use job-wide data, because we don't expect wildly different performance for different Dropbox accounts $chunk_size = $updraftplus->jobdata_get('dropbox_chunk_size', 1048576); // Don't go beyond 10MB, or change the chunk size after the last segment if ($chunk_size < 10485760 && $this->current_file_size > 0 && $offset < $this->current_file_size) { $job_run_time = $time_now - $updraftplus->job_time_ms; if ($time_since_last_tick < 10) { $upload_rate = $data_since_last_tick / max($time_since_last_tick, 1); $upload_secs = min(floor($job_run_time), 10); if ($job_run_time < 15) $upload_secs = max(6, $job_run_time*0.6); $new_chunk = max(min($upload_secs * $upload_rate * 0.9, 10485760), 1048576); $new_chunk = $new_chunk - ($new_chunk % 524288); $chunk_size = (int) $new_chunk; $storage->setChunkSize($chunk_size); $updraftplus->jobdata_set('dropbox_chunk_size', $chunk_size); } } if ($this->current_file_size > 0) { $percent = round(100*($offset/$this->current_file_size), 1); $updraftplus->record_uploaded_chunk($percent, "$uploadid, $offset, ".round($chunk_size/1024, 1)." KB", $fullpath); } else { $this->log("Chunked Upload: $offset bytes uploaded"); // This act is done by record_uploaded_chunk, and helps prevent overlapping runs if ($fullpath) touch($fullpath); } } public function get_supported_features() { // This options format is handled via only accessing options via $this->get_options() return array('multi_options', 'config_templates', 'multi_storage'); } public function get_default_options() { return array( 'appkey' => '', 'secret' => '', 'folder' => '', 'tk_access_token' => '', ); } /** * Acts as a WordPress options filter * * @param Array $dropbox - An array of Dropbox options * @return Array - the returned array can either be the set of updated Dropbox settings or a WordPress error array */ public function options_filter($dropbox) { global $updraftplus; // Get the current options (and possibly update them to the new format) $opts = UpdraftPlus_Storage_Methods_Interface::update_remote_storage_options_format('dropbox'); if (is_wp_error($opts)) { if ('recursion' !== $opts->get_error_code()) { $msg = "(".$opts->get_error_code()."): ".$opts->get_error_message(); $this->log($msg); error_log("UpdraftPlus: $msg"); } // The saved options had a problem; so, return the new ones return $dropbox; } // If the input is not as expected, then return the current options if (!is_array($dropbox)) return $opts; // Remove instances that no longer exist foreach ($opts['settings'] as $instance_id => $storage_options) { if (!isset($dropbox['settings'][$instance_id])) unset($opts['settings'][$instance_id]); } // Dropbox has a special case where the settings could be empty so we should check for this before if (!empty($dropbox['settings'])) { foreach ($dropbox['settings'] as $instance_id => $storage_options) { if (!empty($opts['settings'][$instance_id]['tk_access_token'])) { $current_app_key = empty($opts['settings'][$instance_id]['appkey']) ? false : $opts['settings'][$instance_id]['appkey']; $new_app_key = empty($storage_options['appkey']) ? false : $storage_options['appkey']; // If a different app key is being used, then wipe the stored token as it cannot belong to the new app if ($current_app_key !== $new_app_key) { unset($opts['settings'][$instance_id]['tk_access_token']); unset($opts['settings'][$instance_id]['ownername']); unset($opts['settings'][$instance_id]['CSRF']); } } // Now loop over the new options, and replace old options with them foreach ($storage_options as $key => $value) { if (null === $value) { unset($opts['settings'][$instance_id][$key]); } else { if (!isset($opts['settings'][$instance_id])) $opts['settings'][$instance_id] = array(); $opts['settings'][$instance_id][$key] = $value; } } if (!empty($opts['settings'][$instance_id]['folder']) && preg_match('#^https?://(www.)dropbox\.com/home/Apps/UpdraftPlus(.Com)?([^/]*)/(.*)$#i', $opts['settings'][$instance_id]['folder'], $matches)) $opts['settings'][$instance_id]['folder'] = $matches[3]; } } return $opts; } public function backup($backup_array) { global $updraftplus; $opts = $this->get_options(); if (empty($opts['tk_access_token'])) { $this->log('You do not appear to be authenticated with Dropbox (1)'); $this->log(__('You do not appear to be authenticated with Dropbox', 'updraftplus'), 'error'); return false; } // 28 September 2017: APIv1 is gone. We'll keep the variable to make life easier if there's ever an APIv3. $use_api_ver = 2; if (empty($opts['tk_request_token'])) { $this->log("begin cloud upload (using API version $use_api_ver with OAuth v2 token)"); } else { $this->log("begin cloud upload (using API version $use_api_ver with OAuth v1 token)"); } $chunk_size = $updraftplus->jobdata_get('dropbox_chunk_size', 1048576); try { $dropbox = $this->bootstrap(); if (false === $dropbox) throw new Exception(__('You do not appear to be authenticated with Dropbox', 'updraftplus')); $this->log("access gained; setting chunk size to: ".round($chunk_size/1024, 1)." KB"); $dropbox->setChunkSize($chunk_size); } catch (Exception $e) { $this->log('error when trying to gain access: '.$e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); $this->log(sprintf(__('error: %s (see log file for more)', 'updraftplus'), $e->getMessage()), 'error'); return false; } $updraft_dir = $updraftplus->backups_dir_location(); $dropbox_folder = trailingslashit($opts['folder']); foreach ($backup_array as $file) { $available_quota = -1; // If we experience any failures collecting account info, then carry on anyway try { /* Quota information is no longer provided with account information a new call to quotaInfo must be made to get this information. */ $quota_info = $dropbox->quotaInfo(); if ("200" != $quota_info['code']) { $message = "account/info did not return HTTP 200; returned: ". $quota_info['code']; } elseif (!isset($quota_info['body'])) { $message = "account/info did not return the expected data"; } else { $body = $quota_info['body']; if (isset($body->quota_info)) { $quota_info = $body->quota_info; $total_quota = $quota_info->quota; $normal_quota = $quota_info->normal; $shared_quota = $quota_info->shared; $available_quota = $total_quota - ($normal_quota + $shared_quota); $message = "quota usage: normal=".round($normal_quota/1048576, 1)." MB, shared=".round($shared_quota/1048576, 1)." MB, total=".round($total_quota/1048576, 1)." MB, available=".round($available_quota/1048576, 1)." MB"; } else { $total_quota = max($body->allocation->allocated, 1); $used = $body->used; /* check here to see if the account is a team account and if so use the other used value This will give us their total usage including their individual account and team account */ if (isset($body->allocation->used)) $used = $body->allocation->used; $available_quota = $total_quota - $used; $message = "quota usage: used=".round($used/1048576, 1)." MB, total=".round($total_quota/1048576, 1)." MB, available=".round($available_quota/1048576, 1)." MB"; } } $this->log($message); } catch (Exception $e) { $this->log("exception (".get_class($e).") occurred whilst getting account info: ".$e->getMessage()); // $this->log(sprintf(__("%s error: %s", 'updraftplus'), 'Dropbox', $e->getMessage()).' ('.$e->getCode().')', 'warning', md5($e->getMessage())); } $file_success = 1; $hash = md5($file); $this->current_file_hash = $hash; $filesize = filesize($updraft_dir.'/'.$file); $this->current_file_size = $filesize; // Into KB $filesize = $filesize/1024; $microtime = microtime(true); if ($upload_id = $this->jobdata_get('upload_id_'.$hash, null, 'updraf_dbid_'.$hash)) { // Resume $offset = $this->jobdata_get('upload_offset_'.$hash, null, 'updraf_dbof_'.$hash); $this->log("This is a resumption: $offset bytes had already been uploaded"); } else { $offset = 0; $upload_id = null; } // We don't actually abort now - there's no harm in letting it try and then fail if (-1 != $available_quota && $available_quota < ($filesize-$offset)) { $this->log("File upload expected to fail: file data remaining to upload ($file) size is ".($filesize-$offset)." b (overall file size; .".($filesize*1024)." b), whereas available quota is only $available_quota b"); // $this->log(sprintf(__("Account full: your %s account has only %d bytes left, but the file to be uploaded has %d bytes remaining (total size: %d bytes)",'updraftplus'),'Dropbox', $available_quota, $filesize-$offset, $filesize), 'warning'); } // Old-style, single file put: $put = $dropbox->putFile($updraft_dir.'/'.$file, $dropbox_folder.$file); $ufile = apply_filters('updraftplus_dropbox_modpath', $file, $this); $this->log("Attempt to upload: $file to: $ufile"); $this->upload_tick = microtime(true); $this->uploaded_offset = $offset; try { $response = $dropbox->chunkedUpload($updraft_dir.'/'.$file, '', $ufile, true, $offset, $upload_id, array($this, 'chunked_callback')); if (empty($response['code']) || "200" != $response['code']) { $this->log('Unexpected HTTP code returned from Dropbox: '.$response['code']." (".serialize($response).")"); if ($response['code'] >= 400) { $this->log(sprintf(__('error: failed to upload file to %s (see log file for more)', 'updraftplus'), $file), 'error'); } else { $this->log(__('did not return the expected response - check your log file for more details', 'updraftplus'), 'warning'); } } } catch (Exception $e) { $this->log("chunked upload exception (".get_class($e)."): ".$e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); if (preg_match("/Submitted input out of alignment: got \[(\d+)\] expected \[(\d+)\]/i", $e->getMessage(), $matches)) { // Try the indicated offset $we_tried = $matches[1]; $dropbox_wanted = (int) $matches[2]; $this->log("not yet aligned: tried=$we_tried, wanted=$dropbox_wanted; will attempt recovery"); $this->uploaded_offset = $dropbox_wanted; try { $dropbox->chunkedUpload($updraft_dir.'/'.$file, '', $ufile, true, $dropbox_wanted, $upload_id, array($this, 'chunked_callback')); } catch (Exception $e) { $msg = $e->getMessage(); if (preg_match('/Upload with upload_id .* already completed/', $msg)) { $this->log('returned an error, but apparently indicating previous success: '.$msg); } else { $this->log($msg.' (line: '.$e->getLine().', file: '.$e->getFile().')'); $this->log(sprintf(__('failed to upload file to %s (see log file for more)', 'updraftplus'), $ufile), 'error'); $file_success = 0; if (strpos($msg, 'select/poll returned error') !== false && $this->upload_tick > 0 && time() - $this->upload_tick > 800) { $updraftplus->reschedule(60); $this->log("Select/poll returned after a long time: scheduling a resumption and terminating for now"); $updraftplus->record_still_alive(); die; } } } } else { $msg = $e->getMessage(); if (preg_match('/Upload with upload_id .* already completed/', $msg)) { $this->log('returned an error, but apparently indicating previous success: '.$msg); } else { $this->log($msg); $this->log(sprintf(__('failed to upload file to %s (see log file for more)', 'updraftplus'), $ufile), 'error'); $file_success = 0; if (strpos($msg, 'select/poll returned error') !== false && $this->upload_tick > 0 && time() - $this->upload_tick > 800) { $updraftplus->reschedule(60); $this->log("Select/poll returned after a long time: scheduling a resumption and terminating for now"); $updraftplus->record_still_alive(); die; } } } } if ($file_success) { $updraftplus->uploaded_file($file); $microtime_elapsed = microtime(true)-$microtime; $speedps = ($microtime_elapsed > 0) ? $filesize/$microtime_elapsed : 0; $speed = sprintf("%.2d", $filesize)." KB in ".sprintf("%.2d", $microtime_elapsed)."s (".sprintf("%.2d", $speedps)." KB/s)"; $this->log("File upload success (".$file."): $speed"); $this->jobdata_delete('upload_id_'.$hash, 'updraf_dbid_'.$hash); $this->jobdata_delete('upload_offset_'.$hash, 'updraf_dbof_'.$hash); } } return null; } /** * This method gets a list of files from the remote stoage that match the string passed in and returns an array of backups * * @param string $match a substring to require (tested via strpos() !== false) * @return array */ public function listfiles($match = 'backup_') { $opts = $this->get_options(); if (empty($opts['tk_access_token'])) return new WP_Error('no_settings', __('No settings were found', 'updraftplus').' (dropbox)'); global $updraftplus; try { $dropbox = $this->bootstrap(); } catch (Exception $e) { $this->log('access error: '.$e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); return new WP_Error('access_error', $e->getMessage()); } $searchpath = '/'.untrailingslashit(apply_filters('updraftplus_dropbox_modpath', '', $this)); try { /* Some users could have a large amount of backups, the max search is 1000 entries we should continue to search until there are no more entries to bring back. */ $start = 0; $matches = array(); while (true) { $search = $dropbox->search($match, $searchpath, 1000, $start); if (empty($search['code']) || 200 != $search['code']) return new WP_Error('response_error', sprintf(__('%s returned an unexpected HTTP response: %s', 'updraftplus'), 'Dropbox', $search['code']), $search['body']); if (empty($search['body'])) return array(); if (isset($search['body']->matches) && is_array($search['body']->matches)) { $matches = array_merge($matches, $search['body']->matches); } elseif (is_array($search['body'])) { $matches = $search['body']; } else { break; } if (isset($search['body']->more) && true == $search['body']->more && isset($search['body']->start)) { $start = $search['body']->start; } else { break; } } } catch (Exception $e) { $this->log($e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); // The most likely cause of a search_error is specifying a non-existent path, which should just result in an empty result set. // return new WP_Error('search_error', $e->getMessage()); return array(); } $results = array(); foreach ($matches as $item) { $item = $item->metadata; if (!is_object($item)) continue; if ((!isset($item->size) || $item->size > 0) && 'folder' != $item->{'.tag'} && !empty($item->path_display) && 0 === strpos($item->path_display, $searchpath)) { $path = substr($item->path_display, strlen($searchpath)); if ('/' == substr($path, 0, 1)) $path = substr($path, 1); // Ones in subfolders are not wanted if (false !== strpos($path, '/')) continue; $result = array('name' => $path); if (!empty($item->size)) $result['size'] = $item->size; $results[] = $result; } } return $results; } public function defaults() { return apply_filters('updraftplus_dropbox_defaults', array('Z3Q3ZmkwbnplNHA0Zzlx', 'bTY0bm9iNmY4eWhjODRt')); } public function delete($files, $data = null, $sizeinfo = array()) { global $updraftplus; if (is_string($files)) $files = array($files); $opts = $this->get_options(); if (empty($opts['tk_access_token'])) { $this->log('You do not appear to be authenticated with Dropbox (3)'); $this->log(sprintf(__('You do not appear to be authenticated with %s (whilst deleting)', 'updraftplus'), 'Dropbox'), 'warning'); return false; } try { $dropbox = $this->bootstrap(); } catch (Exception $e) { $this->log($e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); $this->log(sprintf(__('Failed to access %s when deleting (see log file for more)', 'updraftplus'), 'Dropbox'), 'warning'); return false; } if (false === $dropbox) return false; foreach ($files as $file) { $ufile = apply_filters('updraftplus_dropbox_modpath', $file, $this); $this->log("request deletion: $ufile"); try { $dropbox->delete($ufile); $file_success = 1; } catch (Exception $e) { $this->log($e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); } if (isset($file_success)) { $this->log('delete succeeded'); } else { return false; } } } public function download($file) { global $updraftplus; $opts = $this->get_options(); if (empty($opts['tk_access_token'])) { $this->log('You do not appear to be authenticated with Dropbox (4)'); $this->log(sprintf(__('You do not appear to be authenticated with %s', 'updraftplus'), 'Dropbox'), 'error'); return false; } try { $dropbox = $this->bootstrap(); } catch (Exception $e) { $this->log($e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')'); $this->log($e->getMessage().' (line: '.$e->getLine().', file: '.$e->getFile().')', 'error'); return false; } if (false === $dropbox) return false; $remote_files = $this->listfiles($file); foreach ($remote_files as $file_info) { if ($file_info['name'] == $file) { return $updraftplus->chunked_download($file, $this, $file_info['size'], true, null, 2*1048576); } } $this->log("$file: file not found in listing of remote directory"); return false; } /** * Callback used by by chunked downloading API * * @param String $file - the file (basename) to be downloaded * @param Array $headers - supplied headers * @param Mixed $data - pass-back from our call to the API (which we don't use) * @param resource $fh - the local file handle * * @return String - the data downloaded */ public function chunked_download($file, $headers, $data, $fh) { global $updraftplus; $opts = $this->get_options(); $storage = $this->get_storage(); $updraft_dir = $updraftplus->backups_dir_location(); $microtime = microtime(true); $try_the_other_one = false; $ufile = apply_filters('updraftplus_dropbox_modpath', $file, $this); $options = array(); if (!empty($headers)) $options['headers'] = $headers; try { $get = $storage->download($ufile, $fh, $options); } catch (Exception $e) { // TODO: Remove this October 2013 (we stored in the wrong place for a while...) $try_the_other_one = true; $possible_error = $e->getMessage(); $this->log($e); $get = false; } // TODO: Remove this October 2013 (we stored files in the wrong place for a while...) if ($try_the_other_one) { $dropbox_folder = trailingslashit($opts['folder']); try { $get = $storage->download($dropbox_folder.'/'.$file, $fh, $options); if (isset($get['response']['body'])) { $this->log("downloaded ".round(strlen($get['response']['body'])/1024, 1).' KB'); } } catch (Exception $e) { $this->log($possible_error, 'error'); $this->log($e->getMessage(), 'error'); $get = false; } } return $get; } /** * Get the pre configuration template * * @return String - the template */ public function get_pre_configuration_template() { global $updraftplus_admin; $classes = $this->get_css_classes(false); ?> <?php _e(sprintf(__('%s logo', 'updraftplus'), 'Dropbox')); ?>

curl_check('Dropbox', false, 'dropbox'); ?>

'.__('this privacy policy', 'updraftplus').'', 'Dropbox');?>

get_css_classes(); $defmsg = ''.__('Need to use sub-folders?', 'updraftplus').' '.__('Backups are saved in', 'updraftplus').' apps/UpdraftPlus. '.__('If you backup several sites into the same Dropbox and want to organize with sub-folders, then ', 'updraftplus').''.__("there's an add-on for that.", 'updraftplus').''; $defmsg = ''.__('Need to use sub-folders?', 'updraftplus').' '.__('Backups are saved in', 'updraftplus').' apps/UpdraftPlus. '.__('If you backup several sites into the same Dropbox and want to organize with sub-folders, then ', 'updraftplus').''.__("there's an add-on for that.", 'updraftplus').''; $extra_config = apply_filters('updraftplus_dropbox_extra_config_template', $defmsg, $this); echo $extra_config; ?> : {{#if is_authenticated}} ".__('(You appear to be already authenticated).', 'updraftplus').""; $this->get_deauthentication_link(); echo '

'; ?> {{/if}} {{#if ownername_sentence}}
{{ownername_sentence}} {{/if}} '; $this->get_authentication_link(); echo '

'; ?> {{!-- Legacy: only show this next setting to old users who had a setting stored --}} {{#if old_user_settings}} '.htmlspecialchars(__('You must add the following as the authorised redirect URI in your Dropbox console (under "API Settings") when asked', 'updraftplus')).': '.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-dropbox-auth

'; ?> Your Dropbox App Key: output_settings_field_name_and_id('appkey');?> value="{{appkey}}" /> Your Dropbox App Secret: output_settings_field_name_and_id('secret');?> value="{{secret}}" /> {{else}} output_settings_field_name_and_id('tk_access_token');?> value="0"> {{/if}} action_authenticate_storage(); return; } elseif ('deauth' == $_GET['updraftplus_dropboxauth']) { $this->action_deauthenticate_storage(); return; } } elseif (isset($_REQUEST['state'])) { if ('POST' == $_SERVER['REQUEST_METHOD']) { $raw_state = urldecode($_POST['state']); if (isset($_POST['code'])) $raw_code = urldecode($_POST['code']); } else { $raw_state = $_GET['state']; if (isset($_GET['code'])) $raw_code = $_GET['code']; } // Get the CSRF from setting and check it matches the one returned if it does no CSRF attack has happened $opts = $this->get_options(); $csrf = $opts['CSRF']; $state = stripslashes($raw_state); // Check the state to see if an instance_id has been attached and if it has then extract the state $parts = explode(':', $state); $state = $parts[0]; if (strcmp($csrf, $state) == 0) { $opts['CSRF'] = ''; if (isset($raw_code)) { // set code so it can be accessed in the next authentication step $opts['code'] = stripslashes($raw_code); $this->set_options($opts, true); $this->auth_token(); } } else { error_log("UpdraftPlus: CSRF comparison failure: $csrf != $state"); } } try { $this->auth_request(); } catch (Exception $e) { global $updraftplus; $this->log(sprintf(__("%s error: %s", 'updraftplus'), sprintf(__("%s authentication", 'updraftplus'), 'Dropbox'), $e->getMessage()), 'error'); } } /** * This method will reset any saved options and start the bootstrap process for an authentication * * @param String $instance_id - the instance id of the settings we want to authenticate */ public function do_authenticate_storage($instance_id) { try { // Clear out the existing credentials $opts = $this->get_options(); $opts['tk_access_token'] = ''; unset($opts['tk_request_token']); $opts['ownername'] = ''; $this->set_options($opts, true); $this->set_instance_id($instance_id); $this->bootstrap(false); } catch (Exception $e) { global $updraftplus; $this->log(sprintf(__("%s error: %s", 'updraftplus'), sprintf(__("%s authentication", 'updraftplus'), 'Dropbox'), $e->getMessage()), 'error'); } } /** * This method will start the bootstrap process for a de-authentication * * @param String $instance_id - the instance id of the settings we want to de-authenticate */ public function do_deauthenticate_storage($instance_id) { try { $this->set_instance_id($instance_id); $this->bootstrap(true); } catch (Exception $e) { global $updraftplus; $this->log(sprintf(__("%s error: %s", 'updraftplus'), sprintf(__("%s de-authentication", 'updraftplus'), 'Dropbox'), $e->getMessage()), 'error'); } } public function show_authed_admin_warning() { global $updraftplus_admin, $updraftplus; $dropbox = $this->bootstrap(); if (false === $dropbox) return false; try { $account_info = $dropbox->accountInfo(); } catch (Exception $e) { $accountinfo_err = sprintf(__("%s error: %s", 'updraftplus'), 'Dropbox', $e->getMessage()).' ('.$e->getCode().')'; } $message = "".__('Success:', 'updraftplus').' '.sprintf(__('you have authenticated your %s account', 'updraftplus'), 'Dropbox'); // We log, because otherwise people get confused by the most recent log message of 'Parameter not found: oauth_token' and raise support requests $this->log(__('Success:', 'updraftplus').' '.sprintf(__('you have authenticated your %s account', 'updraftplus'), 'Dropbox')); if (empty($account_info['code']) || "200" != $account_info['code']) { $message .= " (".__('though part of the returned information was not as expected - your mileage may vary', 'updraftplus').") ". $account_info['code']; if (!empty($accountinfo_err)) $message .= "
".htmlspecialchars($accountinfo_err); } else { $body = $account_info['body']; $name = ''; if (isset($body->display_name)) { $name = $body->display_name; } else { $name = $body->name->display_name; } $message .= ".
".sprintf(__('Your %s account name: %s', 'updraftplus'), 'Dropbox', htmlspecialchars($name)); $opts = $this->get_options(); $opts['ownername'] = $name; $this->set_options($opts, true); try { /** * Quota information is no longer provided with account information a new call to qoutaInfo must be made to get this information. */ $quota_info = $dropbox->quotaInfo(); if (empty($quota_info['code']) || "200" != $quota_info['code']) { $message .= " (".__('though part of the returned information was not as expected - your mileage may vary', 'updraftplus').")". $quota_info['code']; if (!empty($accountinfo_err)) $message .= "
".htmlspecialchars($accountinfo_err); } else { $body = $quota_info['body']; if (isset($body->quota_info)) { $quota_info = $body->quota_info; $total_quota = max($quota_info->quota, 1); $normal_quota = $quota_info->normal; $shared_quota = $quota_info->shared; $available_quota =$total_quota - ($normal_quota + $shared_quota); $used_perc = round(($normal_quota + $shared_quota)*100/$total_quota, 1); $message .= '
'.sprintf(__('Your %s quota usage: %s %% used, %s available', 'updraftplus'), 'Dropbox', $used_perc, round($available_quota/1048576, 1).' MB'); } else { $total_quota = max($body->allocation->allocated, 1); $used = $body->used; /* check here to see if the account is a team account and if so use the other used value This will give us their total usage including their individual account and team account */ if (isset($body->allocation->used)) $used = $body->allocation->used; $available_quota =$total_quota - $used; $used_perc = round($used*100/$total_quota, 1); $message .= '
'.sprintf(__('Your %s quota usage: %s %% used, %s available', 'updraftplus'), 'Dropbox', $used_perc, round($available_quota/1048576, 1).' MB'); } } } catch (Exception $e) { // Catch } } $updraftplus_admin->show_admin_warning($message); } public function auth_token() { $this->bootstrap(); $opts = $this->get_options(); if (!empty($opts['tk_access_token'])) { add_action('all_admin_notices', array($this, 'show_authed_admin_warning')); } } /** * Acquire single-use authorization code */ public function auth_request() { $this->bootstrap(); } /** * This basically reproduces the relevant bits of bootstrap.php from the SDK * * @param boolean $deauthenticate indicates if we should bootstrap for a deauth or auth request * @return object */ public function bootstrap($deauthenticate = false) { $storage = $this->get_storage(); if (!empty($storage) && !is_wp_error($storage)) return $storage; // Dropbox APIv1 is dead, but we'll keep the variable in case v3 is ever announced $dropbox_api = 'Dropbox2'; include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/API.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/Exception.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Consumer/ConsumerAbstract.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Storage/StorageInterface.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Storage/Encrypter.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Storage/WordPress.php'); include_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Consumer/Curl.php'); // require_once(UPDRAFTPLUS_DIR.'/includes/'.$dropbox_api.'/OAuth/Consumer/WordPress.php'); $opts = $this->get_options(); $key = empty($opts['secret']) ? '' : $opts['secret']; $sec = empty($opts['appkey']) ? '' : $opts['appkey']; $oauth2_id = base64_decode('aXA3NGR2Zm1sOHFteTA5'); // Set the callback URL $callbackhome = UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-dropbox-auth'; $callback = defined('UPDRAFTPLUS_DROPBOX_AUTH_RETURN_URL') ? UPDRAFTPLUS_DROPBOX_AUTH_RETURN_URL : 'https://auth.updraftplus.com/auth/dropbox/'; $instance_id = $this->get_instance_id(); // Instantiate the Encrypter and storage objects $encrypter = new Dropbox_Encrypter('ThisOneDoesNotMatterBeyondLength'); // Instantiate the storage $dropbox_storage = new Dropbox_WordPress($encrypter, "tk_", 'updraft_dropbox', $this); // WordPress consumer does not yet work // $oauth = new Dropbox_ConsumerWordPress($sec, $key, $dropbox_storage, $callback); // Get the DropBox API access details list($d2, $d1) = $this->defaults(); if (empty($sec)) { $sec = base64_decode($d1); } if (empty($key)) { $key = base64_decode($d2); } $root = 'sandbox'; if ('dropbox:' == substr($sec, 0, 8)) { $sec = substr($sec, 8); $root = 'dropbox'; } try { $oauth = new Dropbox_Curl($sec, $oauth2_id, $key, $dropbox_storage, $callback, $callbackhome, $deauthenticate, $instance_id); } catch (Exception $e) { global $updraftplus; $this->log("Curl error: ".$e->getMessage()); $this->log(sprintf(__("%s error: %s", 'updraftplus'), "Dropbox/Curl", $e->getMessage().' ('.get_class($e).') (line: '.$e->getLine().', file: '.$e->getFile()).')', 'error'); return false; } if ($deauthenticate) return true; $storage = new UpdraftPlus_Dropbox_API($oauth, $root); $this->set_storage($storage); return $storage; } }