Current File : //usr/local/softaculous/lib/aefer/_onedrive.php |
<?php
class onedrive{
var $access_token;
var $refresh_token;
var $path;
var $filename;
var $filesize = 0;
var $complete = 0;
var $offset = 0;
var $tmpsize = 0;
var $chunk = 4194304;
var $range_lower_limit = 0;
var $range_upper_limit = 0;
var $bytes_remaining = 0;
var $num_bytes = 0;
var $num_fragments = 0;
var $chunk_size = 0;
var $onedrive_file_id = '';
var $mode = '';
var $wp = NULL; // Memory Write Pointer
var $upload_url = '';
var $local_dest = '';
var $root_folder_path = 'root:';
var $graph_api_url = 'https://graph.microsoft.com/v1.0/me/drive/';
// APP name is Softaculous Auto Installer and is assigned to developers@softaculous.com Microsoft account
var $app_key = '31422ea4-40e4-4f0a-ab63-0324fe586c62';
var $app_secret = 'PDK8Q~tlOFNlVzR8CAKC8mg._bvzShbBEVBm6dtY';
var $app_dir = 'Softaculous Auto Installer';
var $redirect_uri = 'https://s2.softaculous.com/onedrive/callback.php';
var $scopes = 'files.Read Files.ReadWrite offline_access';
function stream_open($path, $mode, $options, &$opened_path){
global $error, $l, $user;
$stream = parse_url($path);
$this->refresh_token = $stream['host'];
$this->path = $stream['path'];
$this->mode = $mode;
//One Drive access token expires in an hour so we need to refresh
$this->access_token = $this->refresh_token_func($this->refresh_token);
$pathinfo = pathinfo($this->path);
$dirlist = explode('/', $pathinfo['dirname']);
$this->filename = $pathinfo['basename'];
//Local destination where the backup tar file will be stored initially.
$this->local_dest = $GLOBALS['local_dest'].'/'.$this->filename;
return true;
}
function stream_write($data){
global $error, $l;
if(!is_resource($this->wp)){
$this->wp = fopen($this->local_dest, 'a+');
}
$data_size = strlen($data);
//Write into local tar created inside softaculous_backups
fwrite($this->wp, $data);
return $data_size;
}
function stream_close(){
global $error, $l;
if(preg_match('/w|a/is', $this->mode)){
if(empty($GLOBALS['end_file'])){
$this->create_upload_session();
//Delete local tar file
@unlink($this->local_dest);
}
}
return true;
}
//One Drive API to create an Upload Session
function create_upload_session(){
global $error, $l;
$url = $this->graph_api_url.$this->root_folder_path.rawurlencode($this->path).':/createUploadSession';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$data= '{}';
$response = $this->__curl($url, $headers, $data);
if(!empty($response['error'])){
$error[] = $response['error'];
return false;
}
$resp_data = json_decode($response['result'], true);
$this->upload_url = $resp_data['uploadUrl'];
if(empty($this->upload_url)){
$error[] = $l['onedrive_err_upload_url'];
return false;
}
$this->upload_start();
return true;
}
//One Drive API to Upload
function upload_start(){
global $error, $l;
$this->filesize = filesize($this->local_dest);
$this->num_fragments = ceil($this->filesize / $this->chunk);
$this->bytes_remaining = $this->filesize;
while ($this->tmpsize < $this->num_fragments) {
$this->chunk_size = $this->num_bytes = $this->chunk;
$this->range_lower_limit = $this->tmpsize * $this->chunk;
$this->range_upper_limit = $this->tmpsize * $this->chunk + $this->chunk_size - 1;
$this->offset = $this->tmpsize * $this->chunk;
if ($this->bytes_remaining < $this->chunk_size) {
$this->chunk_size = $this->num_bytes = $this->bytes_remaining;
$this->range_upper_limit = $this->filesize - 1;
}
if ($stream = fopen($this->local_dest, 'rb')) {
$data = stream_get_contents($stream, $this->chunk_size, $this->offset);
fclose($stream);
}
//Call upload append function to write the data from Local tar file to One Drive
$retcode = $this->upload_append($this->upload_url, $data, $this->filesize);
if($retcode == '201 Created'){
$this->complete = 1;
}
if(!empty($error)){
break;
}
$this->bytes_remaining = $this->bytes_remaining - $this->chunk_size;
$this->tmpsize++;
}
if(empty($this->complete)){
$error[] = $l['onedrive_err_end'];
return false;
}
return true;
}
//One Drive API to append
function upload_append($upload_url, $data, $final_size, $retry = false){
global $error, $l;
$headers = array(
'Content-Length: '.$this->num_bytes,
'Content-Range: bytes '.$this->range_lower_limit.'-'.$this->range_upper_limit.'/'.$final_size);
$response = $this->__curl($upload_url, $headers, $data, 'PUT');
// Was there any interruption ? Lets retry
if(!empty($response['error']) && preg_match('/SSL_ERROR_SYSCALL/', $response['error']) && empty($retry)){
$retcode = 'SSL_ERROR_SYSCALL';
return $this->upload_append($upload_url, $data, $final_size, true);
}
if(!empty($response['error'])){
$error[] = $response['error'];
return false;
}
//Check for response code
$resp_obj= json_decode($response['result'], true);
$retcode = '404';
if(!empty($resp_obj)){
if(array_key_exists("nextExpectedRanges",$resp_obj)){
$retcode = '308 Resume Incomplete';
}elseif(array_key_exists("id",$resp_obj)){
$retcode = '201 Created';
}else{
$retcode = '416 Requested Range Not Satisfiable';
}
}else{
$retcode = '416 Requested Range Not Satisfiable';
}
if($retcode == '416 Requested Range Not Satisfiable' && empty($retry)){
return $this->upload_append($upload_url, $data, $final_size, true);
}
if($retcode != '308 Resume Incomplete' && $retcode != '201 Created'){
$error[] = $retcode;
return false;
}
if($retcode == '308 Resume Incomplete'){
$this->range_lower_limit = $this->range_upper_limit + 1;
$this->offset = $this->range_upper_limit + 1;
}elseif($retcode == '201 Created'){
$this->onedrive_file_id = $resp_obj['id'];
}
return $retcode;
}
//In response to file_exists(), is_file(), is_dir()
function url_stat($path){
global $error;
$stream = parse_url($path);
$this->refresh_token = $stream['host'];
$file_path = $stream['path'];
$pathinfo = pathinfo($stream['path']);
$filename = $pathinfo['basename'];
//One Drive access token expires in an hour so we need to refresh
if(empty($this->access_token)){
$this->access_token = $this->refresh_token_func($this->refresh_token);
}
//Metadata for the root folder is unsupported
if(!empty($filename)){
$url=$this->graph_api_url.$this->root_folder_path.rawurlencode($file_path).':';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$resp = $this->__curl($url, $headers, '', 'GET');
$data = json_decode($resp['result'], true);
if(array_key_exists("folder",$data)){
$mode = 0040000; //For DIR
}else{
$mode = 0100000; //For File
}
if(!empty($data['id'])){
$stat = array('dev' => 0,
'ino' => 0,
'mode' => $mode,
'nlink' => 0,
'uid' => 0,
'gid' => 0,
'rdev' => 0,
'size' => $data['size'],
'atime' => strtotime($data['createdDateTime']),
'mtime' => strtotime($data['fileSystemInfo']['lastModifiedDateTime']),
'ctime' => strtotime($data['fileSystemInfo']['createdDateTime']),
'blksize' => 0,
'blocks' => 0);
$this->filesize = $stat['size'];
return $stat;
}
}
return false;
}
//Get onedrive file/folder id if exist
function get_onedrive_file_id($filename, $refresh_token = ''){
global $error, $l;
if(!empty($refresh_token)){
$this->refresh_token = $refresh_token;
}
//One Drive access token expires in an hour so we need to refresh
if(empty($this->access_token)){
$this->access_token = $this->refresh_token_func($this->refresh_token);
}
$url=$this->graph_api_url.$this->root_folder_path.rawurlencode($filename).':';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$response = $this->__curl($url, $headers, '', 'GET');
if(!empty($response['error'])){
//$error[] = $response['error'];
return false;
}
$data = json_decode($response['result'], true);
if(!empty($data['error'])){
return false;
}
$this->onedrive_file_id = $data['id'];
return $this->onedrive_file_id;
}
//Download Backup File from One Drive to local server
function download_file_loop($source, $dest, $startpos = 0){
global $error;
$stream = parse_url($source);
$this->refresh_token = $stream['host'];
$path = $stream['path'];
//One Drive access token expires in an hour so we need to refresh
if(empty($this->access_token)){
$this->access_token = $this->refresh_token_func($this->refresh_token);
}
$this->get_onedrive_file_id($path);
$file_stats = $this->url_stat($source);
$this->filesize = $file_stats['size'];
$url = $this->graph_api_url.$this->root_folder_path.rawurlencode($path).':';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$resp = $this->__curl($url, $headers, '', 'GET');
//Fetch download URL
$object = json_decode($resp['result'], true);
$download_url = $object['@microsoft.graph.downloadUrl'];
$this->range_lower_limit = $startpos;
$this->range_upper_limit = ($this->range_lower_limit + $this->chunk) - 1;
$fp = @fopen($dest, "ab");
while(!$this->__eof()){
if(time() >= $GLOBALS['end']){
$GLOBALS['l_readbytes'] = filesize($dest);
break;
}
if($this->range_upper_limit >= $this->filesize){
$this->range_upper_limit = $this->filesize - 1;
}
$block = $this->__read($download_url, $this->range_lower_limit, $this->range_upper_limit);
fwrite($fp, $block);
$this->offset = $this->range_upper_limit + 1;
$this->range_lower_limit = $this->range_upper_limit + 1;
$this->range_upper_limit = ($this->range_lower_limit + $this->chunk) - 1;
}
fclose($fp);
}
function __read($download_url, $lower_limit, $upper_limit){
global $error;
$headers = array('Range: bytes='.$lower_limit.'-'.$upper_limit);
$resp = $this->__curl($download_url, $headers, '', 'GET');
if(!empty($resp['error'])){
$error[] = $resp['error'];
}
return $resp['result'];
}
function __eof(){
if($this->offset < $this->filesize){
return false;
}
return true;
}
//Delete the backup from One Drive
function unlink($path){
global $error, $l;
$stream = parse_url($path);
$this->refresh_token = $stream['host'];
$pathinfo = pathinfo($stream['path']);
$filename = $pathinfo['basename'];
$file_path = $stream['path'];
//One Drive access token expires in an hour so we need to refresh
if(empty($this->access_token)){
$this->access_token = $this->refresh_token_func($this->refresh_token);
}
if(empty($this->onedrive_file_id)){
$this->get_onedrive_file_id($file_path);
}
$url=$this->graph_api_url.'items/'.$this->onedrive_file_id.':';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$resp = $this->__curl($url, $headers, '', 'DELETE');
if(!empty($resp['error'])){
$error[] = $resp['error'];
return false;
}
return true;
}
//Rename the backup file
function rename($from, $to){
global $error;
$stream_from = parse_url($from);
$this->refresh_token = $stream_from['host'];
$from_path = $stream_from['path'];
$from_pathinfo = pathinfo($stream_from['path']);
$from_file = $from_pathinfo['basename'];
$stream_to = parse_url($to);
$to_path = trim($stream_to['path'], '/\\');
$to_pathinfo = pathinfo($stream_to['path']);
$to_file = $to_pathinfo['basename'];
//One Drive access token expires in an hour so we need to refresh
if(empty($this->access_token)){
$this->access_token = $this->refresh_token_func($this->refresh_token);
}
$this->get_onedrive_file_id($from_path);
$url = $url=$this->graph_api_url.'items/'.$this->onedrive_file_id.':';
$data= '{
"name": "'.$to_file.'"
}';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$resp = $this->__curl($url, $headers, $data, 'PATCH');
if(!empty($resp['error'])){
$error[] = $resp['error'];
return false;
}
return $resp['result'];
}
/**
* Generate One Drive Refresh and Access Token from the Authorization Code provided
*
* @package softaculous
* @author Pratik Jaiswal
* @param string $auth_code The authorization code generated by user during access grant process
* @return string $data One Drive Refresh and Access Token which we can use to create backup files
* @since 5.7.1
*/
function generate_onedrive_token($auth_code){
global $error, $l, $onedrive;
$refresh_token = rawurldecode($refresh_token);
$url = 'https://login.microsoftonline.com/common/oauth2/v2.0/token';
$headers = array("Content-Type: application/x-www-form-urlencoded");
$post = http_build_query(array(
'client_id' => $this->app_key,
'redirect_uri' => $this->redirect_uri,
'client_secret' => $this->app_secret,
'code' => $auth_code,
'grant_type' => 'authorization_code',
'scope' => $this->scopes));
$resp = $this->__curl($url, $headers, $post);
if(!empty($resp['error'])){
$error[] = $resp['error'];
return false;
}
$data = json_decode($resp['result'], true);
if(!empty($data['error'])){
if(is_array($data['error'])){
$error[] = $data['error']['code'].' : '.$data['error']['message'];
}else{
$error[] = $data['error'].' : '.$data['error_description'];
}
return false;
}
return $data;
}
/**
* Generate a new Access Token or Refresh Token from the previous Refresh Token.
*
* @package softaculous
* @author Pratik Jaiswal
* @param string $refresh_token The refresh token generated by user during access grant process
* @return string $token One Drive Access Token which we can use for authentication in behalf of user
* @since 5.7.1
*/
function refresh_token_func($refresh_token){
global $error, $l, $onedrive;
$refresh_token = rawurldecode($refresh_token);
$url = 'https://login.microsoftonline.com/common/oauth2/v2.0/token';
$headers = array("Content-Type: application/x-www-form-urlencoded");
$post = http_build_query(array(
'client_id' => $this->app_key,
'redirect_uri' => $this->redirect_uri,
'client_secret' => $this->app_secret,
'refresh_token' => $refresh_token,
'grant_type' => 'refresh_token'));
$resp = $this->__curl($url, $headers, $post);
if(!empty($resp['error'])){
$error[] = $resp['error'];
return false;
}
$data = json_decode($resp['result'], true);
if(!empty($data['error'])){
if(is_array($data['error'])){
$error[] = $data['error']['code'].' : '.$data['error']['message'];
}else{
$error[] = $data['error'].' : '.$data['error_description'];
}
return false;
}
return $data['access_token'];
}
/**
* Create Softaculous App Directory in user's One Drive account
*
* @package softaculous
* @author Pratik jaiswal
* @param string $refresh_token Refresh Token of user's One Drive account to generate the access token
* @since 5.7.1
*/
function create_onedrive_app_dir($refresh_token){
global $globals, $onedrive, $error;
$file_id = $this->get_onedrive_file_id($this->app_dir, $refresh_token);
if(empty($file_id)){
$this->create_dir($this->app_dir);
}
}
function create_dir($dirname){
global $error, $l, $onedrive;
$url = $this->graph_api_url.'root/children';
$data= '{
"name": "'.$dirname.'",
"folder": { }
}';
$headers = array('Content-Type: application/json',
"Cache-Control: no-cache",
"Pragma: no-cache",
"Authorization: bearer ".$this->access_token);
$resp = $this->__curl($url, $headers, $data, 'POST');
if(!empty($resp['error'])){
$error[] = $resp['error'];
return false;
}
$data = json_decode($resp['result'], true);
if(!empty($data['error'])){
if(is_array($data['error'])){
$error[] = $data['error']['code'].' : '.$data['error']['message'];
}else{
$error[] = $data['error'].' : '.$data['error_description'];
}
return false;
}
return $data['id'];
}
function __curl($url, $headers = '', $post = '', $request_type = 'POST'){
global $error, $l;
// Set the curl parameters.
$ch = curl_init($url);
curl_setopt($ch, CURLOPT_URL, $url);
if(!empty($headers)){
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
}
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $request_type);
//We are setting this as on some servers, the default HTTP version was taken as 2.0 by curl, causing issue
curl_setopt($ch, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
if(!empty($post)){
curl_setopt($ch, CURLOPT_POST, TRUE);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post);
}
// Turn off the server and peer verification (TrustManager Concept).
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
//Get response from the server.
$resp = array();
$resp['result'] = curl_exec($ch);
$resp['error'] = curl_error($ch);
curl_close($ch);
return $resp;
}
}