2016-05-11 13:45:10 +06:00
|
|
|
/* 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.
|
|
|
|
*
|
|
|
|
*/
|
2016-05-15 00:31:02 +06:00
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
2016-05-11 13:45:10 +06:00
|
|
|
|
2016-05-11 20:24:53 +06:00
|
|
|
#include "backends/cloud/dropbox/dropboxstorage.h"
|
2016-05-31 20:54:41 +06:00
|
|
|
#include "backends/cloud/dropbox/dropboxcreatedirectoryrequest.h"
|
2016-05-24 00:14:24 +06:00
|
|
|
#include "backends/cloud/dropbox/dropboxlistdirectoryrequest.h"
|
2016-05-30 02:23:29 +06:00
|
|
|
#include "backends/cloud/dropbox/dropboxuploadrequest.h"
|
2016-06-09 14:23:14 +06:00
|
|
|
#include "backends/cloud/cloudmanager.h"
|
2016-05-18 15:21:09 +06:00
|
|
|
#include "backends/networking/curl/connectionmanager.h"
|
2016-05-17 01:19:49 +06:00
|
|
|
#include "backends/networking/curl/curljsonrequest.h"
|
2016-05-18 14:08:05 +06:00
|
|
|
#include "common/config-manager.h"
|
2016-05-11 13:45:10 +06:00
|
|
|
#include "common/debug.h"
|
2016-05-17 01:19:49 +06:00
|
|
|
#include "common/json.h"
|
2016-05-15 00:31:02 +06:00
|
|
|
#include <curl/curl.h>
|
2016-05-11 01:10:37 +06:00
|
|
|
|
2016-05-12 18:52:57 +06:00
|
|
|
namespace Cloud {
|
|
|
|
namespace Dropbox {
|
2016-05-11 01:10:37 +06:00
|
|
|
|
2016-05-27 22:33:39 +06:00
|
|
|
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;
|
|
|
|
}
|
2016-05-18 14:08:05 +06:00
|
|
|
|
2016-06-10 16:35:23 +06:00
|
|
|
DropboxStorage::DropboxStorage(Common::String accessToken, Common::String userId): _token(accessToken), _uid(userId) {}
|
|
|
|
|
|
|
|
DropboxStorage::DropboxStorage(Common::String code) {
|
|
|
|
getAccessToken(code);
|
|
|
|
}
|
|
|
|
|
|
|
|
DropboxStorage::~DropboxStorage() {}
|
|
|
|
|
|
|
|
void DropboxStorage::getAccessToken(Common::String code) {
|
|
|
|
Networking::JsonCallback callback = new Common::Callback<DropboxStorage, Networking::JsonResponse>(this, &DropboxStorage::codeFlowComplete);
|
|
|
|
Networking::CurlJsonRequest *request = new Networking::CurlJsonRequest(callback, nullptr, "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");
|
|
|
|
addRequest(request);
|
|
|
|
}
|
2016-05-18 14:08:05 +06:00
|
|
|
|
2016-06-10 16:35:23 +06:00
|
|
|
void DropboxStorage::codeFlowComplete(Networking::JsonResponse response) {
|
|
|
|
Common::JSONValue *json = (Common::JSONValue *)response.value;
|
|
|
|
if (json) {
|
2016-05-18 14:08:05 +06:00
|
|
|
Common::JSONObject result = json->asObject();
|
|
|
|
if (!result.contains("access_token") || !result.contains("uid")) {
|
|
|
|
warning("Bad response, no token/uid passed");
|
|
|
|
} else {
|
2016-06-10 16:35:23 +06:00
|
|
|
_token = result.getVal("access_token")->asString();
|
2016-06-11 21:38:20 +02:00
|
|
|
_uid = result.getVal("uid")->asString();
|
2016-05-18 14:08:05 +06:00
|
|
|
ConfMan.removeKey("dropbox_code", "cloud");
|
2016-06-10 16:35:23 +06:00
|
|
|
CloudMan.replaceStorage(this, kStorageDropboxId);
|
2016-05-27 15:21:06 +06:00
|
|
|
ConfMan.flushToDisk();
|
2016-06-10 16:35:23 +06:00
|
|
|
debug("Done! You can use Dropbox now! Look:");
|
|
|
|
CloudMan.testFeature();
|
2016-05-18 14:08:05 +06:00
|
|
|
}
|
|
|
|
|
2016-05-17 01:19:49 +06:00
|
|
|
delete json;
|
|
|
|
} else {
|
2016-06-10 16:35:23 +06:00
|
|
|
debug("DropboxStorage::codeFlowComplete: got NULL instead of JSON!");
|
2016-05-17 01:19:49 +06:00
|
|
|
}
|
2016-05-15 00:31:02 +06:00
|
|
|
}
|
|
|
|
|
2016-06-09 13:49:52 +06:00
|
|
|
void DropboxStorage::saveConfig(Common::String keyPrefix) {
|
2016-05-23 11:23:33 +06:00
|
|
|
ConfMan.set(keyPrefix + "access_token", _token, "cloud");
|
|
|
|
ConfMan.set(keyPrefix + "user_id", _uid, "cloud");
|
|
|
|
}
|
|
|
|
|
2016-06-08 18:51:00 +06:00
|
|
|
Common::String DropboxStorage::name() const {
|
|
|
|
return "Dropbox";
|
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
void DropboxStorage::printFiles(FileArrayResponse response) {
|
2016-05-24 00:14:24 +06:00
|
|
|
debug("files:");
|
2016-05-31 01:51:32 +06:00
|
|
|
Common::Array<StorageFile> &files = response.value;
|
2016-05-24 00:14:24 +06:00
|
|
|
for (uint32 i = 0; i < files.size(); ++i)
|
|
|
|
debug("\t%s", files[i].name().c_str());
|
2016-05-23 12:21:45 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
void DropboxStorage::printBool(BoolResponse response) {
|
|
|
|
debug("bool: %s", (response.value?"true":"false"));
|
2016-05-30 02:23:29 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
void DropboxStorage::printStorageFile(UploadResponse response) {
|
|
|
|
debug("\nuploaded file info:");
|
|
|
|
debug("\tpath: %s", response.value.path().c_str());
|
|
|
|
debug("\tsize: %u", response.value.size());
|
|
|
|
debug("\ttimestamp: %u", response.value.timestamp());
|
2016-05-30 13:35:53 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
Networking::Request *DropboxStorage::listDirectory(Common::String path, ListDirectoryCallback outerCallback, Networking::ErrorCallback errorCallback, bool recursive) {
|
2016-06-01 12:39:10 +06:00
|
|
|
return addRequest(new DropboxListDirectoryRequest(_token, path, outerCallback, errorCallback, recursive));
|
2016-05-31 01:51:32 +06:00
|
|
|
}
|
|
|
|
|
|
|
|
Networking::Request *DropboxStorage::upload(Common::String path, Common::SeekableReadStream *contents, UploadCallback callback, Networking::ErrorCallback errorCallback) {
|
2016-06-01 12:39:10 +06:00
|
|
|
return addRequest(new DropboxUploadRequest(_token, path, contents, callback, errorCallback));
|
2016-05-31 01:51:32 +06:00
|
|
|
}
|
|
|
|
|
2016-06-08 13:02:49 +06:00
|
|
|
Networking::Request *DropboxStorage::streamFileById(Common::String path, Networking::NetworkReadStreamCallback callback, Networking::ErrorCallback errorCallback) {
|
2016-05-24 11:57:49 +06:00
|
|
|
Common::JSONObject jsonRequestParameters;
|
|
|
|
jsonRequestParameters.setVal("path", new Common::JSONValue(path));
|
|
|
|
Common::JSONValue value(jsonRequestParameters);
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
Networking::CurlRequest *request = new Networking::CurlRequest(nullptr, nullptr, "https://content.dropboxapi.com/2/files/download"); //TODO: is it right?
|
2016-05-24 11:57:49 +06:00
|
|
|
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)
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
Networking::NetworkReadStreamResponse response = request->execute();
|
|
|
|
if (callback) (*callback)(response);
|
|
|
|
return response.request;
|
2016-05-24 11:57:49 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 20:54:41 +06:00
|
|
|
Networking::Request *DropboxStorage::createDirectory(Common::String path, BoolCallback callback, Networking::ErrorCallback errorCallback) {
|
|
|
|
if (!errorCallback) errorCallback = getErrorPrintingCallback();
|
2016-06-01 12:39:10 +06:00
|
|
|
return addRequest(new DropboxCreateDirectoryRequest(_token, path, callback, errorCallback));
|
2016-05-31 20:54:41 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
Networking::Request *DropboxStorage::info(StorageInfoCallback outerCallback, Networking::ErrorCallback errorCallback) {
|
2016-05-27 15:21:06 +06:00
|
|
|
Networking::JsonCallback innerCallback = new Common::CallbackBridge<DropboxStorage, StorageInfoResponse, Networking::JsonResponse>(this, &DropboxStorage::infoInnerCallback, outerCallback);
|
2016-05-31 01:51:32 +06:00
|
|
|
Networking::CurlJsonRequest *request = new Networking::CurlJsonRequest(innerCallback, errorCallback, "https://api.dropboxapi.com/1/account/info");
|
2016-05-18 14:08:05 +06:00
|
|
|
request->addHeader("Authorization: Bearer " + _token);
|
2016-06-01 12:39:10 +06:00
|
|
|
return addRequest(request);
|
2016-05-21 23:21:42 +06:00
|
|
|
//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 *)
|
2016-05-18 14:08:05 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 16:23:25 +06:00
|
|
|
Common::String DropboxStorage::savesDirectoryPath() { return "/saves/"; }
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
void DropboxStorage::infoInnerCallback(StorageInfoCallback outerCallback, Networking::JsonResponse response) {
|
|
|
|
Common::JSONValue *json = response.value;
|
2016-05-21 23:21:42 +06:00
|
|
|
if (!json) {
|
|
|
|
warning("NULL passed instead of JSON");
|
|
|
|
delete outerCallback;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-06-09 18:49:17 +06:00
|
|
|
//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();
|
|
|
|
uint64 quotaNormal = quota.getVal("normal")->asIntegerNumber();
|
|
|
|
uint64 quotaShared = quota.getVal("shared")->asIntegerNumber();
|
|
|
|
uint64 quotaAllocated = quota.getVal("quota")->asIntegerNumber();
|
|
|
|
|
|
|
|
CloudMan.setStorageUsername(kStorageDropboxId, email);
|
|
|
|
|
|
|
|
if (outerCallback) {
|
2016-05-27 15:21:06 +06:00
|
|
|
(*outerCallback)(StorageInfoResponse(nullptr, StorageInfo(uid, name, email, quotaNormal+quotaShared, quotaAllocated)));
|
2016-05-21 23:21:42 +06:00
|
|
|
delete outerCallback;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete json;
|
2016-05-21 21:30:25 +06:00
|
|
|
}
|
|
|
|
|
2016-05-31 01:51:32 +06:00
|
|
|
void DropboxStorage::infoMethodCallback(StorageInfoResponse response) {
|
2016-05-23 12:00:12 +06:00
|
|
|
debug("\nStorage info:");
|
2016-05-31 01:51:32 +06:00
|
|
|
debug("User name: %s", response.value.name().c_str());
|
|
|
|
debug("Email: %s", response.value.email().c_str());
|
2016-05-31 19:32:16 +06:00
|
|
|
debug("Disk usage: %u/%u", (uint32)response.value.used(), (uint32)response.value.available());
|
2016-05-21 14:06:50 +06:00
|
|
|
}
|
|
|
|
|
2016-05-23 11:23:33 +06:00
|
|
|
DropboxStorage *DropboxStorage::loadFromConfig(Common::String keyPrefix) {
|
2016-05-27 22:33:39 +06:00
|
|
|
loadKeyAndSecret();
|
2016-05-18 14:08:05 +06:00
|
|
|
|
2016-05-23 11:23:33 +06:00
|
|
|
if (!ConfMan.hasKey(keyPrefix + "access_token", "cloud")) {
|
2016-05-18 14:08:05 +06:00
|
|
|
warning("No access_token found");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-23 11:23:33 +06:00
|
|
|
if (!ConfMan.hasKey(keyPrefix + "user_id", "cloud")) {
|
2016-05-18 14:08:05 +06:00
|
|
|
warning("No user_id found");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-23 11:23:33 +06:00
|
|
|
Common::String accessToken = ConfMan.get(keyPrefix + "access_token", "cloud");
|
|
|
|
Common::String userId = ConfMan.get(keyPrefix + "user_id", "cloud");
|
2016-05-18 14:08:05 +06:00
|
|
|
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
|
2016-05-27 22:33:39 +06:00
|
|
|
url += "&client_id="; url += KEY;
|
2016-05-18 14:08:05 +06:00
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
2016-05-28 20:10:38 +02:00
|
|
|
} // End of namespace Dropbox
|
|
|
|
} // End of namespace Cloud
|