mirror of
https://github.com/xorgy/mediafire-fuse
synced 2026-01-13 13:14:29 -08:00
- don't use folder/get_content but rely on parent information - add many more debugging information and sanity checks - ignore trash - correctly update local device revision using device_response - remove "visited" member and change housekeeping function to check for consistency of parents and children (and retrieve fix from remote if necessary) - print key and parent key in the debugging function - check for remote updates on every fs function - introduce global defines for key and filename length - add parent member, getters and setters to file.h
212 lines
7.0 KiB
C
212 lines
7.0 KiB
C
/*
|
|
* Copyright (C) 2013 Bryan Christ <bryan.christ@mediafire.com>
|
|
* 2014 Johannes Schauer <j.schauer@email.de>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2, as published by
|
|
* the Free Software Foundation.
|
|
*
|
|
* 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.
|
|
*
|
|
*/
|
|
|
|
#include <jansson.h>
|
|
#include <stdlib.h>
|
|
#include <inttypes.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
#include <stddef.h>
|
|
#include <stdio.h>
|
|
|
|
#include "../../utils/http.h"
|
|
#include "../../utils/json.h"
|
|
#include "../mfconn.h"
|
|
#include "../apicalls.h" // IWYU pragma: keep
|
|
|
|
static int _decode_device_get_changes(mfhttp * conn, void *data);
|
|
|
|
/*
|
|
* the mfconn_device_change array will be realloc'ed so it must either point
|
|
* to a malloc'ed memory or be NULL
|
|
*
|
|
* the returned array will be sorted by revision
|
|
*/
|
|
int mfconn_api_device_get_changes(mfconn * conn, uint64_t revision,
|
|
struct mfconn_device_change **changes)
|
|
{
|
|
const char *api_call;
|
|
int retval;
|
|
mfhttp *http;
|
|
|
|
if (conn == NULL)
|
|
return -1;
|
|
|
|
api_call = mfconn_create_signed_get(conn, 0, "device/get_changes.php",
|
|
"?revision=%" PRIu64
|
|
"&response_format=json", revision);
|
|
|
|
http = http_create();
|
|
retval =
|
|
http_get_buf(http, api_call, _decode_device_get_changes,
|
|
(void *)changes);
|
|
http_destroy(http);
|
|
|
|
free((void *)api_call);
|
|
|
|
return retval;
|
|
}
|
|
|
|
static void aux(json_t * key, json_t * parent, json_t * revision,
|
|
enum mfconn_device_change_type change,
|
|
struct mfconn_device_change **changes, size_t * len_changes)
|
|
{
|
|
struct mfconn_device_change *tmp_change;
|
|
|
|
if (key == NULL || revision == NULL || parent == NULL) {
|
|
fprintf(stderr, "change without either key, revision or parent");
|
|
return;
|
|
}
|
|
(*len_changes)++;
|
|
*changes = (struct mfconn_device_change *)
|
|
realloc(*changes,
|
|
(*len_changes) * sizeof(struct mfconn_device_change));
|
|
tmp_change = *changes + (*len_changes) - 1;
|
|
tmp_change->change = change;
|
|
strncpy(tmp_change->key, json_string_value(key), sizeof(tmp_change->key));
|
|
strncpy(tmp_change->parent, json_string_value(parent),
|
|
sizeof(tmp_change->parent));
|
|
tmp_change->revision = atoll(json_string_value(revision));
|
|
}
|
|
|
|
static int change_compare(const void *a, const void *b)
|
|
{
|
|
return ((struct mfconn_device_change *)a)->revision
|
|
- ((struct mfconn_device_change *)b)->revision;
|
|
}
|
|
|
|
static int _decode_device_get_changes(mfhttp * conn, void *user_ptr)
|
|
{
|
|
json_error_t error;
|
|
json_t *root;
|
|
json_t *node;
|
|
json_t *data;
|
|
|
|
json_t *obj_array;
|
|
json_t *key;
|
|
json_t *parent;
|
|
json_t *revision;
|
|
json_t *device_revision;
|
|
|
|
int array_sz;
|
|
int i = 0;
|
|
|
|
struct mfconn_device_change **changes;
|
|
size_t len_changes;
|
|
|
|
changes = (struct mfconn_device_change **)user_ptr;
|
|
if (changes == NULL)
|
|
return -1;
|
|
|
|
root = http_parse_buf_json(conn, 0, &error);
|
|
|
|
node = json_object_by_path(root, "response");
|
|
|
|
device_revision = json_object_get(node, "device_revision");
|
|
if (device_revision == NULL) {
|
|
fprintf(stderr,
|
|
"response/device_revision is not part of the result\n");
|
|
return -1;
|
|
}
|
|
|
|
len_changes = 0;
|
|
|
|
node = json_object_by_path(root, "response/updated");
|
|
|
|
obj_array = json_object_get(node, "files");
|
|
if (json_is_array(obj_array)) {
|
|
array_sz = json_array_size(obj_array);
|
|
for (i = 0; i < array_sz; i++) {
|
|
data = json_array_get(obj_array, i);
|
|
if (!json_is_object(data))
|
|
continue;
|
|
key = json_object_get(data, "quickkey");
|
|
parent = json_object_get(data, "parent_folderkey");
|
|
revision = json_object_get(data, "revision");
|
|
aux(key, parent, revision, MFCONN_DEVICE_CHANGE_UPDATED_FILE,
|
|
changes, &len_changes);
|
|
}
|
|
}
|
|
|
|
obj_array = json_object_get(node, "folders");
|
|
if (json_is_array(obj_array)) {
|
|
array_sz = json_array_size(obj_array);
|
|
for (i = 0; i < array_sz; i++) {
|
|
data = json_array_get(obj_array, i);
|
|
if (!json_is_object(data))
|
|
continue;
|
|
key = json_object_get(data, "folderkey");
|
|
parent = json_object_get(data, "parent_folderkey");
|
|
revision = json_object_get(data, "revision");
|
|
aux(key, parent, revision, MFCONN_DEVICE_CHANGE_UPDATED_FOLDER,
|
|
changes, &len_changes);
|
|
}
|
|
}
|
|
|
|
node = json_object_by_path(root, "response/deleted");
|
|
|
|
obj_array = json_object_get(node, "files");
|
|
if (json_is_array(obj_array)) {
|
|
array_sz = json_array_size(obj_array);
|
|
for (i = 0; i < array_sz; i++) {
|
|
data = json_array_get(obj_array, i);
|
|
if (!json_is_object(data))
|
|
continue;
|
|
key = json_object_get(data, "quickkey");
|
|
parent = json_object_get(data, "parent_folderkey");
|
|
revision = json_object_get(data, "revision");
|
|
aux(key, parent, revision, MFCONN_DEVICE_CHANGE_DELETED_FILE,
|
|
changes, &len_changes);
|
|
}
|
|
}
|
|
|
|
obj_array = json_object_get(node, "folders");
|
|
if (json_is_array(obj_array)) {
|
|
array_sz = json_array_size(obj_array);
|
|
for (i = 0; i < array_sz; i++) {
|
|
data = json_array_get(obj_array, i);
|
|
if (!json_is_object(data))
|
|
continue;
|
|
key = json_object_get(data, "folderkey");
|
|
parent = json_object_get(data, "parent_folderkey");
|
|
revision = json_object_get(data, "revision");
|
|
aux(key, parent, revision, MFCONN_DEVICE_CHANGE_DELETED_FOLDER,
|
|
changes, &len_changes);
|
|
}
|
|
}
|
|
// sort
|
|
qsort(*changes, len_changes, sizeof(struct mfconn_device_change),
|
|
change_compare);
|
|
|
|
// put an entry with change type MFCONN_DEVICE_CHANGE_END at the end
|
|
// encode the current device revision as its revision
|
|
len_changes++;
|
|
*changes = (struct mfconn_device_change *)
|
|
realloc(*changes, len_changes * sizeof(struct mfconn_device_change));
|
|
memset(*changes + len_changes - 1, 0, sizeof(struct mfconn_device_change));
|
|
(*changes)[len_changes - 1].change = MFCONN_DEVICE_CHANGE_END;
|
|
(*changes)[len_changes - 1].revision =
|
|
atoll(json_string_value(device_revision));
|
|
|
|
if (root != NULL)
|
|
json_decref(root);
|
|
|
|
return 0;
|
|
}
|