If file could be uploaded in one API call, no need to create a session (which requires at least two calls: to start and then to finish it).
297 lines
12 KiB
C++
297 lines
12 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
|
|
|
#include "backends/cloud/dropbox/dropboxstorage.h"
|
|
#include "backends/cloud/dropbox/dropboxlistdirectoryrequest.h"
|
|
#include "backends/cloud/dropbox/dropboxuploadrequest.h"
|
|
#include "backends/cloud/downloadrequest.h"
|
|
#include "backends/cloud/folderdownloadrequest.h"
|
|
#include "backends/networking/curl/connectionmanager.h"
|
|
#include "backends/networking/curl/curljsonrequest.h"
|
|
#include "common/config-manager.h"
|
|
#include "common/debug.h"
|
|
#include "common/file.h"
|
|
#include "common/json.h"
|
|
#include <curl/curl.h>
|
|
|
|
namespace Cloud {
|
|
namespace Dropbox {
|
|
|
|
char *DropboxStorage::KEY; //can't use ConfMan there yet, loading it on instance creation/auth
|
|
char *DropboxStorage::SECRET; //TODO: hide these secrets somehow
|
|
|
|
void DropboxStorage::loadKeyAndSecret() {
|
|
Common::String k = ConfMan.get("DROPBOX_KEY", "cloud");
|
|
KEY = new char[k.size() + 1];
|
|
memcpy(KEY, k.c_str(), k.size());
|
|
KEY[k.size()] = 0;
|
|
|
|
k = ConfMan.get("DROPBOX_SECRET", "cloud");
|
|
SECRET = new char[k.size() + 1];
|
|
memcpy(SECRET, k.c_str(), k.size());
|
|
SECRET[k.size()] = 0;
|
|
}
|
|
|
|
static void saveAccessTokenCallback(Networking::JsonResponse pair) {
|
|
Common::JSONValue *json = (Common::JSONValue *)pair.value;
|
|
if (json) {
|
|
debug("saveAccessTokenCallback:");
|
|
debug("%s", json->stringify(true).c_str());
|
|
|
|
Common::JSONObject result = json->asObject();
|
|
if (!result.contains("access_token") || !result.contains("uid")) {
|
|
warning("Bad response, no token/uid passed");
|
|
} else {
|
|
//we suppose that's the first storage
|
|
ConfMan.set("storages_number", "1", "cloud");
|
|
ConfMan.set("current_storage", "1", "cloud");
|
|
ConfMan.set("storage1_type", "Dropbox", "cloud");
|
|
ConfMan.set("storage1_access_token", result.getVal("access_token")->asString(), "cloud");
|
|
ConfMan.set("storage1_user_id", result.getVal("uid")->asString(), "cloud");
|
|
ConfMan.removeKey("dropbox_code", "cloud");
|
|
ConfMan.flushToDisk();
|
|
debug("Now please restart ScummVM to apply the changes.");
|
|
}
|
|
|
|
delete json;
|
|
} else {
|
|
debug("saveAccessTokenCallback: got NULL instead of JSON!");
|
|
}
|
|
}
|
|
|
|
DropboxStorage::DropboxStorage(Common::String accessToken, Common::String userId): _token(accessToken), _uid(userId) {
|
|
curl_global_init(CURL_GLOBAL_ALL);
|
|
}
|
|
|
|
DropboxStorage::~DropboxStorage() {
|
|
curl_global_cleanup();
|
|
}
|
|
|
|
void DropboxStorage::saveConfig(Common::String keyPrefix) {
|
|
ConfMan.set(keyPrefix + "type", "Dropbox", "cloud");
|
|
ConfMan.set(keyPrefix + "access_token", _token, "cloud");
|
|
ConfMan.set(keyPrefix + "user_id", _uid, "cloud");
|
|
}
|
|
|
|
void DropboxStorage::printFiles(FileArrayResponse pair) {
|
|
debug("files:");
|
|
Common::Array<StorageFile> &files = pair.value;
|
|
for (uint32 i = 0; i < files.size(); ++i)
|
|
debug("\t%s", files[i].name().c_str());
|
|
}
|
|
|
|
void DropboxStorage::printBool(BoolResponse pair) {
|
|
debug("bool: %s", (pair.value?"true":"false"));
|
|
}
|
|
|
|
void DropboxStorage::printUploadStatus(UploadResponse pair) {
|
|
debug(" ");
|
|
UploadStatus status = pair.value;
|
|
if (status.interrupted) {
|
|
debug("upload interrupted by user");
|
|
return;
|
|
}
|
|
if (status.failed) {
|
|
debug("upload failed with following response:");
|
|
debug("%s", status.response.c_str());
|
|
return;
|
|
}
|
|
debug("upload HTTP response code = %ld", status.httpResponseCode);
|
|
if (!status.failed) {
|
|
debug("uploaded file info:");
|
|
debug("\tpath: %s", status.file.path().c_str());
|
|
debug("\tsize: %u", status.file.size());
|
|
debug("\ttimestamp: %u", status.file.timestamp());
|
|
}
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::listDirectory(Common::String path, FileArrayCallback outerCallback, bool recursive) {
|
|
return ConnMan.addRequest(new DropboxListDirectoryRequest(_token, path, outerCallback, recursive));
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::upload(Common::String path, Common::SeekableReadStream *contents, UploadCallback callback) {
|
|
return ConnMan.addRequest(new DropboxUploadRequest(_token, path, contents, callback));
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::upload(Common::String remotePath, Common::String localPath, UploadCallback callback) {
|
|
Common::File *f = new Common::File();
|
|
if (!f->open(localPath)) {
|
|
warning("DropboxStorage: unable to open file to upload from");
|
|
UploadStatus status(false, true, StorageFile(), "", -1);
|
|
if (callback) (*callback)(UploadResponse(nullptr, status));
|
|
delete f;
|
|
return nullptr;
|
|
}
|
|
return upload(remotePath, f, callback);
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::streamFile(Common::String path, Networking::NetworkReadStreamCallback callback) {
|
|
Common::JSONObject jsonRequestParameters;
|
|
jsonRequestParameters.setVal("path", new Common::JSONValue(path));
|
|
Common::JSONValue value(jsonRequestParameters);
|
|
|
|
Networking::CurlRequest *request = new Networking::CurlRequest(0, "https://content.dropboxapi.com/2/files/download");
|
|
request->addHeader("Authorization: Bearer " + _token);
|
|
request->addHeader("Dropbox-API-Arg: " + Common::JSON::stringify(&value));
|
|
request->addHeader("Content-Type: "); //required to be empty (as we do POST, it's usually app/form-url-encoded)
|
|
|
|
Networking::NetworkReadStreamResponse pair = request->execute();
|
|
if (callback) (*callback)(pair);
|
|
return pair.request;
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::download(Common::String remotePath, Common::String localPath, BoolCallback callback) {
|
|
Common::DumpFile *f = new Common::DumpFile();
|
|
if (!f->open(localPath, true)) {
|
|
warning("DropboxStorage: unable to open file to download into");
|
|
if (callback) (*callback)(BoolResponse(nullptr, false));
|
|
delete f;
|
|
return nullptr;
|
|
}
|
|
|
|
return ConnMan.addRequest(new DownloadRequest(this, callback, remotePath, f));
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::downloadFolder(Common::String remotePath, Common::String localPath, FileArrayCallback callback, bool recursive) {
|
|
return ConnMan.addRequest(new FolderDownloadRequest(this, callback, remotePath, localPath, recursive));
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::syncSaves(BoolCallback callback) {
|
|
//this is not the real syncSaves() implementation
|
|
//"" is root in Dropbox, not "/"
|
|
//this must create all these directories:
|
|
//return download("/remote/test.jpg", "local/a/b/c/d/test.jpg", 0);
|
|
/*
|
|
return downloadFolder(
|
|
"/not_flat", "local/not_flat_1_level/",
|
|
new Common::Callback<DropboxStorage, FileArrayResponse>(this, &DropboxStorage::printFiles),
|
|
false
|
|
);
|
|
*/
|
|
return upload("/remote/test4.bmp", "final.bmp", new Common::Callback<DropboxStorage, UploadResponse>(this, &DropboxStorage::printUploadStatus));
|
|
}
|
|
|
|
Networking::Request *DropboxStorage::info(StorageInfoCallback outerCallback) {
|
|
Networking::JsonCallback innerCallback = new Common::CallbackBridge<DropboxStorage, StorageInfoResponse, Networking::JsonResponse>(this, &DropboxStorage::infoInnerCallback, outerCallback);
|
|
Networking::CurlJsonRequest *request = new Networking::CurlJsonRequest(innerCallback, "https://api.dropboxapi.com/1/account/info");
|
|
request->addHeader("Authorization: Bearer " + _token);
|
|
return ConnMan.addRequest(request);
|
|
//that callback bridge wraps the outerCallback (passed in arguments from user) into innerCallback
|
|
//so, when CurlJsonRequest is finished, it calls the innerCallback
|
|
//innerCallback (which is DropboxStorage::infoInnerCallback in this case) processes the void *ptr
|
|
//and then calls the outerCallback (which wants to receive StorageInfo, not void *)
|
|
}
|
|
|
|
void DropboxStorage::infoInnerCallback(StorageInfoCallback outerCallback, Networking::JsonResponse pair) {
|
|
Common::JSONValue *json = pair.value;
|
|
if (!json) {
|
|
warning("NULL passed instead of JSON");
|
|
delete outerCallback;
|
|
return;
|
|
}
|
|
|
|
if (outerCallback) {
|
|
//Dropbox documentation states there is no errors for this API method
|
|
Common::JSONObject info = json->asObject();
|
|
Common::String uid = Common::String::format("%d", (int)info.getVal("uid")->asIntegerNumber());
|
|
Common::String name = info.getVal("display_name")->asString();
|
|
Common::String email = info.getVal("email")->asString();
|
|
Common::JSONObject quota = info.getVal("quota_info")->asObject();
|
|
uint32 quotaNormal = quota.getVal("normal")->asIntegerNumber();
|
|
uint32 quotaShared = quota.getVal("shared")->asIntegerNumber();
|
|
uint32 quotaAllocated = quota.getVal("quota")->asIntegerNumber();
|
|
(*outerCallback)(StorageInfoResponse(nullptr, StorageInfo(uid, name, email, quotaNormal+quotaShared, quotaAllocated)));
|
|
delete outerCallback;
|
|
}
|
|
|
|
delete json;
|
|
}
|
|
|
|
void DropboxStorage::infoMethodCallback(StorageInfoResponse pair) {
|
|
debug("\nStorage info:");
|
|
debug("User name: %s", pair.value.name().c_str());
|
|
debug("Email: %s", pair.value.email().c_str());
|
|
debug("Disk usage: %u/%u", pair.value.used(), pair.value.available());
|
|
}
|
|
|
|
DropboxStorage *DropboxStorage::loadFromConfig(Common::String keyPrefix) {
|
|
loadKeyAndSecret();
|
|
|
|
if (!ConfMan.hasKey(keyPrefix + "access_token", "cloud")) {
|
|
warning("No access_token found");
|
|
return 0;
|
|
}
|
|
|
|
if (!ConfMan.hasKey(keyPrefix + "user_id", "cloud")) {
|
|
warning("No user_id found");
|
|
return 0;
|
|
}
|
|
|
|
Common::String accessToken = ConfMan.get(keyPrefix + "access_token", "cloud");
|
|
Common::String userId = ConfMan.get(keyPrefix + "user_id", "cloud");
|
|
return new DropboxStorage(accessToken, userId);
|
|
}
|
|
|
|
Common::String DropboxStorage::getAuthLink() {
|
|
Common::String url = "https://www.dropbox.com/1/oauth2/authorize";
|
|
url += "?response_type=code";
|
|
url += "&redirect_uri=http://localhost:12345/"; //that's for copy-pasting
|
|
//url += "&redirect_uri=http%3A%2F%2Flocalhost%3A12345%2F"; //that's "http://localhost:12345/" for automatic opening
|
|
url += "&client_id="; url += KEY;
|
|
return url;
|
|
}
|
|
|
|
void DropboxStorage::authThroughConsole() {
|
|
if (!ConfMan.hasKey("DROPBOX_KEY", "cloud") || !ConfMan.hasKey("DROPBOX_SECRET", "cloud")) {
|
|
warning("No Dropbox keys available, cannot do auth");
|
|
return;
|
|
}
|
|
|
|
loadKeyAndSecret();
|
|
|
|
if (ConfMan.hasKey("dropbox_code", "cloud")) {
|
|
//phase 2: get access_token using specified code
|
|
getAccessToken(ConfMan.get("dropbox_code", "cloud"));
|
|
return;
|
|
}
|
|
|
|
debug("Navigate to this URL and press \"Allow\":");
|
|
debug("%s\n", getAuthLink().c_str());
|
|
debug("Then, add dropbox_code key in [cloud] section of configuration file. You should copy the <code> value from URL and put it as value for that key.\n");
|
|
debug("Navigate to this URL to get more information on ScummVM's configuration files:");
|
|
debug("http://wiki.scummvm.org/index.php/User_Manual/Configuring_ScummVM#Using_the_configuration_file_to_configure_ScummVM\n");
|
|
}
|
|
|
|
void DropboxStorage::getAccessToken(Common::String code) {
|
|
Networking::JsonCallback callback = new Common::GlobalFunctionCallback<Networking::JsonResponse>(saveAccessTokenCallback);
|
|
Networking::CurlJsonRequest *request = new Networking::CurlJsonRequest(callback, "https://api.dropboxapi.com/1/oauth2/token");
|
|
request->addPostField("code=" + code);
|
|
request->addPostField("grant_type=authorization_code");
|
|
request->addPostField("client_id=" + Common::String(KEY));
|
|
request->addPostField("client_secret=" + Common::String(SECRET));
|
|
request->addPostField("&redirect_uri=http%3A%2F%2Flocalhost%3A12345%2F");
|
|
ConnMan.addRequest(request);
|
|
}
|
|
|
|
} // End of namespace Dropbox
|
|
} // End of namespace Cloud
|