mirror of
https://github.com/YunoHost-Apps/ffsync_ynh.git
synced 2024-09-03 18:26:38 +02:00
330 lines
10 KiB
PHP
330 lines
10 KiB
PHP
|
<?php
|
||
|
|
||
|
# ***** BEGIN LICENSE BLOCK *****
|
||
|
# Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
||
|
#
|
||
|
# The contents of this file are subject to the Mozilla Public License Version
|
||
|
# 1.1 (the "License"); you may not use this file except in compliance with
|
||
|
# the License. You may obtain a copy of the License at
|
||
|
# http://www.mozilla.org/MPL/
|
||
|
#
|
||
|
# Software distributed under the License is distributed on an "AS IS" basis,
|
||
|
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
||
|
# for the specific language governing rights and limitations under the
|
||
|
# License.
|
||
|
#
|
||
|
# The Original Code is Weave Minimal Server
|
||
|
#
|
||
|
# The Initial Developer of the Original Code is
|
||
|
# Mozilla Labs.
|
||
|
# Portions created by the Initial Developer are Copyright (C) 2008
|
||
|
# the Initial Developer. All Rights Reserved.
|
||
|
#
|
||
|
# Contributor(s):
|
||
|
# Toby Elliott (telliott@mozilla.com)
|
||
|
# Luca Tettamanti
|
||
|
#
|
||
|
# Alternatively, the contents of this file may be used under the terms of
|
||
|
# either the GNU General Public License Version 2 or later (the "GPL"), or
|
||
|
# the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
||
|
# in which case the provisions of the GPL or the LGPL are applicable instead
|
||
|
# of those above. If you wish to allow use of your version of this file only
|
||
|
# under the terms of either the GPL or the LGPL, and not to allow others to
|
||
|
# use your version of this file under the terms of the MPL, indicate your
|
||
|
# decision by deleting the provisions above and replace them with the notice
|
||
|
# and other provisions required by the GPL or the LGPL. If you do not delete
|
||
|
# the provisions above, a recipient may use your version of this file under
|
||
|
# the terms of any one of the MPL, the GPL or the LGPL.
|
||
|
#
|
||
|
# ***** END LICENSE BLOCK *****
|
||
|
|
||
|
if ( ! file_exists("settings.php") && file_exists("setup.php") ) {
|
||
|
require_once "setup.php";
|
||
|
exit;
|
||
|
|
||
|
} else if ( ! file_exists("settings.php") ) {
|
||
|
echo "<hr><h2>Maybe the setup is not completed, missing settings.php</h2><hr>";
|
||
|
exit;
|
||
|
|
||
|
} else if ( file_exists("setup.php") ) {
|
||
|
echo "<hr><h2>Maybe the setup is not completed, else please delete setup.php</h2><hr>";
|
||
|
exit;
|
||
|
}
|
||
|
|
||
|
require_once 'weave_storage.php';
|
||
|
require_once 'weave_basic_object.php';
|
||
|
require_once 'weave_utils.php';
|
||
|
require_once 'weave_hash.php';
|
||
|
|
||
|
require_once "WBOJsonOutput.php";
|
||
|
//header("Content-type: application/json");
|
||
|
|
||
|
$server_time = round(microtime(1), 2);
|
||
|
header("X-Weave-Timestamp: " . $server_time);
|
||
|
|
||
|
# Basic path extraction and validation. No point in going on if these are missing
|
||
|
$path = '/';
|
||
|
if (!empty($_SERVER['PATH_INFO']))
|
||
|
$path = $_SERVER['PATH_INFO'];
|
||
|
else if (!empty($_SERVER['ORIG_PATH_INFO']))
|
||
|
$path = $_SERVER['ORIG_PATH_INFO'];
|
||
|
else if (!empty($_SERVER["REQUEST_URI"]))
|
||
|
{
|
||
|
log_error("experimental path");
|
||
|
# this is kind of an experimental try, i needed it so i build it,
|
||
|
# but that doesent mean that it does work... well it works for me
|
||
|
# and it shouldnt break anything...
|
||
|
$path = $_SERVER["REQUEST_URI"];
|
||
|
$lastfolder = substr(FSYNCMS_ROOT,strrpos(FSYNCMS_ROOT, "/",-2));
|
||
|
$path = substr($path, (strpos($path,$lastfolder) + strlen($lastfolder)-1)); #chop the lead slash
|
||
|
if(strpos($path,'?') != false)
|
||
|
$path = substr($path, 0, strpos($path,'?')); //remove php arguments
|
||
|
log_error("path_exp:".$path);
|
||
|
}
|
||
|
else
|
||
|
report_problem("No path found", 404);
|
||
|
|
||
|
$path = substr($path, 1); #chop the lead slash
|
||
|
log_error("start request_____" . $path);
|
||
|
// ensure that we got a valid request
|
||
|
if ( !$path )
|
||
|
report_problem("Invalid request, this was not a firefox sync request!", 400);
|
||
|
|
||
|
// split path into parts and make sure that all values are properly initialized
|
||
|
list($version, $username, $function, $collection, $id) = array_pad(explode('/', $path.'///'), 5, '');
|
||
|
|
||
|
if($version == 'user' || $version == 'misc')
|
||
|
{
|
||
|
//asking for userApi -> user.php
|
||
|
$include = true;
|
||
|
require 'user.php';
|
||
|
exit(); // should not get here, but how knows
|
||
|
}
|
||
|
|
||
|
header("Content-type: application/json");
|
||
|
|
||
|
if ($version != '1.0' && $version != '1.1')
|
||
|
report_problem('Function not found', 404);
|
||
|
|
||
|
if ($function != "info" && $function != "storage")
|
||
|
report_problem(WEAVE_ERROR_FUNCTION_NOT_SUPPORTED, 400);
|
||
|
|
||
|
if (!validate_username($username))
|
||
|
report_problem(WEAVE_ERROR_INVALID_USERNAME, 400);
|
||
|
|
||
|
#only a delete has meaning without a collection
|
||
|
if ($collection)
|
||
|
{
|
||
|
if (!validate_collection($collection))
|
||
|
report_problem(WEAVE_ERROR_INVALID_COLLECTION, 400);
|
||
|
}
|
||
|
else if ($_SERVER['REQUEST_METHOD'] != 'DELETE')
|
||
|
report_problem(WEAVE_ERROR_INVALID_PROTOCOL, 400);
|
||
|
|
||
|
|
||
|
#quick check to make sure that any non-storage function calls are just using GET
|
||
|
if ($function != 'storage' && $_SERVER['REQUEST_METHOD'] != 'GET')
|
||
|
report_problem(WEAVE_ERROR_INVALID_PROTOCOL, 400);
|
||
|
|
||
|
|
||
|
|
||
|
#user passes preliminaries, connections made, onto actually getting the data
|
||
|
try
|
||
|
{
|
||
|
$db = new WeaveStorage($username);
|
||
|
|
||
|
#Auth the user
|
||
|
verify_user($username, $db);
|
||
|
|
||
|
#user passes preliminaries, connections made, onto actually getting the data
|
||
|
if ($_SERVER['REQUEST_METHOD'] == 'GET')
|
||
|
{
|
||
|
if ($function == 'info')
|
||
|
{
|
||
|
switch ($collection)
|
||
|
{
|
||
|
case 'quota':
|
||
|
exit(json_encode(array($db->get_storage_total())));
|
||
|
case 'collections':
|
||
|
exit(json_encode($db->get_collection_list_with_timestamps()));
|
||
|
case 'collection_counts':
|
||
|
exit(json_encode($db->get_collection_list_with_counts()));
|
||
|
case 'collection_usage':
|
||
|
$results = $db->get_collection_storage_totals();
|
||
|
foreach (array_keys($results) as $collection)
|
||
|
{
|
||
|
$results[$collection] = ceil($results[$collection] / 1024); #converting to k from bytes
|
||
|
}
|
||
|
exit(json_encode($results));
|
||
|
default:
|
||
|
report_problem(WEAVE_ERROR_INVALID_PROTOCOL, 400);
|
||
|
}
|
||
|
}
|
||
|
elseif ($function == 'storage')
|
||
|
{
|
||
|
log_error("function storage");
|
||
|
if ($id) #retrieve a single record
|
||
|
{
|
||
|
$wbo = $db->retrieve_objects($collection, $id, 1); #get the full contents of one record
|
||
|
if (count($wbo) > 0)
|
||
|
{
|
||
|
$item = array_shift($wbo);
|
||
|
echo $item->json();
|
||
|
}
|
||
|
else
|
||
|
report_problem("record not found", 404);
|
||
|
}
|
||
|
else #retrieve a batch of records. Sadly, due to potential record sizes, have the storage object stream the output...
|
||
|
{
|
||
|
log_error("retrieve a batch");
|
||
|
$full = array_key_exists('full', $_GET) && $_GET['full'];
|
||
|
|
||
|
$outputter = new WBOJsonOutput($full);
|
||
|
|
||
|
$params = validate_search_params();
|
||
|
|
||
|
$ids = $db->retrieve_objects($collection, null, $full, $outputter,
|
||
|
$params['parentid'], $params['predecessorid'],
|
||
|
$params['newer'], $params['older'],
|
||
|
$params['sort'],
|
||
|
$params['limit'], $params['offset'],
|
||
|
$params['ids'],
|
||
|
$params['index_above'], $params['index_below'], $params['depth']
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if ($_SERVER['REQUEST_METHOD'] == 'PUT') #add a single record to the server
|
||
|
{
|
||
|
$wbo = new wbo();
|
||
|
if (!$wbo->extract_json(get_json()))
|
||
|
report_problem(WEAVE_ERROR_JSON_PARSE, 400);
|
||
|
|
||
|
check_quota($db);
|
||
|
check_timestamp($collection, $db);
|
||
|
|
||
|
#use the url if the json object doesn't have an id
|
||
|
if (!$wbo->id() && $id) { $wbo->id($id); }
|
||
|
|
||
|
$wbo->collection($collection);
|
||
|
$wbo->modified($server_time); #current microtime
|
||
|
|
||
|
if ($wbo->validate())
|
||
|
{
|
||
|
#if there's no payload (as opposed to blank), then update the metadata
|
||
|
if ($wbo->payload_exists())
|
||
|
$db->store_object($wbo);
|
||
|
else
|
||
|
$db->update_object($wbo);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
report_problem(WEAVE_ERROR_INVALID_WBO, 400);
|
||
|
}
|
||
|
echo json_encode($server_time);
|
||
|
}
|
||
|
else if ($_SERVER['REQUEST_METHOD'] == 'POST')
|
||
|
{
|
||
|
$json = get_json();
|
||
|
|
||
|
check_quota($db);
|
||
|
check_timestamp($collection, $db);
|
||
|
|
||
|
$success_ids = array();
|
||
|
$failed_ids = array();
|
||
|
|
||
|
$db->begin_transaction();
|
||
|
|
||
|
foreach ($json as $wbo_data)
|
||
|
{
|
||
|
$wbo = new wbo();
|
||
|
|
||
|
if (!$wbo->extract_json($wbo_data))
|
||
|
{
|
||
|
$failed_ids[$wbo->id()] = $wbo->get_error();
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$wbo->collection($collection);
|
||
|
$wbo->modified($server_time);
|
||
|
|
||
|
|
||
|
if ($wbo->validate())
|
||
|
{
|
||
|
#if there's no payload (as opposed to blank), then update the metadata
|
||
|
if ($wbo->payload_exists())
|
||
|
{
|
||
|
$db->store_object($wbo);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$db->update_object($wbo);
|
||
|
}
|
||
|
$success_ids[] = $wbo->id();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
$failed_ids[$wbo->id()] = $wbo->get_error();
|
||
|
}
|
||
|
}
|
||
|
$db->commit_transaction();
|
||
|
|
||
|
echo json_encode(array('success' => $success_ids, 'failed' => $failed_ids));
|
||
|
}
|
||
|
else if ($_SERVER['REQUEST_METHOD'] == 'DELETE')
|
||
|
{
|
||
|
check_timestamp($collection, $db);
|
||
|
|
||
|
if ($id)
|
||
|
{
|
||
|
$db->delete_object($collection, $id);
|
||
|
}
|
||
|
else if ($collection)
|
||
|
{
|
||
|
$params = validate_search_params();
|
||
|
|
||
|
$db->delete_objects($collection, null,
|
||
|
$params['parentid'], $params['predecessorid'],
|
||
|
$params['newer'], $params['older'],
|
||
|
$params['sort'],
|
||
|
$params['limit'], $params['offset'],
|
||
|
$params['ids'],
|
||
|
$params['index_above'], $params['index_below']
|
||
|
);
|
||
|
}
|
||
|
else if($function == 'storage') // ich vermute mal storage reinigen
|
||
|
{
|
||
|
if (!array_key_exists('HTTP_X_CONFIRM_DELETE', $_SERVER))
|
||
|
report_problem(WEAVE_ERROR_NO_OVERWRITE, 412);
|
||
|
$db->delete_storage($username);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (!array_key_exists('HTTP_X_CONFIRM_DELETE', $_SERVER))
|
||
|
report_problem(WEAVE_ERROR_NO_OVERWRITE, 412);
|
||
|
log_error("delete "."Server ".print_r( $_SERVER, true));
|
||
|
$db->delete_user($username);
|
||
|
}
|
||
|
|
||
|
echo json_encode($server_time);
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
#bad protocol. There are protocols left? HEAD, I guess.
|
||
|
report_problem(1, 400);
|
||
|
}
|
||
|
}
|
||
|
catch(Exception $e)
|
||
|
{
|
||
|
report_problem($e->getMessage(), $e->getCode());
|
||
|
}
|
||
|
|
||
|
|
||
|
#The datasets we might be dealing with here are too large for sticking it all into an array, so
|
||
|
#we need to define a direct-output method for the storage class to use. If we start producing multiples
|
||
|
#(unlikely), we can put them in their own class.
|
||
|
|
||
|
#include_once "WBOJsonOutput.php";
|
||
|
?>
|