summaryrefslogtreecommitdiff
path: root/modules
diff options
context:
space:
mode:
authorTim Almdal <tnalmdal@shaw.ca>2009-12-07 18:11:26 -0800
committerTim Almdal <tnalmdal@shaw.ca>2009-12-07 18:11:26 -0800
commit71be6cf239fba5718cd6336403df602b05c21c7d (patch)
treeb1ca3a6d145340e46796553b1a4a3317ef2e9c28 /modules
parentcfbbc22f7c49db18cf11908337c00a61146ab61b (diff)
The rest framework that the new gallery3 remote interface will be built on. At the moment, there are no handlers to perform any functionality.
Diffstat (limited to 'modules')
-rw-r--r--modules/rest/config/routes.php23
-rw-r--r--modules/rest/controllers/rest.php111
-rw-r--r--modules/rest/helpers/rest_event.php75
-rw-r--r--modules/rest/helpers/rest_installer.php37
-rw-r--r--modules/rest/libraries/Form_Label.php45
-rw-r--r--modules/rest/models/rest_key.php21
-rw-r--r--modules/rest/module.info4
-rw-r--r--modules/rest/tests/Rest_Controller_Test.php220
8 files changed, 536 insertions, 0 deletions
diff --git a/modules/rest/config/routes.php b/modules/rest/config/routes.php
new file mode 100644
index 00000000..ec65fda8
--- /dev/null
+++ b/modules/rest/config/routes.php
@@ -0,0 +1,23 @@
+<?php defined("SYSPATH") or die("No direct script access.");
+/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+
+// Redirect module REST requests to the REST Controller
+$config["^(\w+)/rest/(.*)$"] = "rest/$1/$2";
+
diff --git a/modules/rest/controllers/rest.php b/modules/rest/controllers/rest.php
new file mode 100644
index 00000000..d16d2316
--- /dev/null
+++ b/modules/rest/controllers/rest.php
@@ -0,0 +1,111 @@
+<?php defined("SYSPATH") or die("No direct script access.");/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class Rest_Controller extends Controller {
+ public function access_key() {
+ $request = json_decode($this->input->post("request"));
+ if (empty($request->user) || empty($request->password)) {
+ print json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ $user = identity::lookup_user_by_name($request->user);
+ if (empty($user)) {
+ print json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ if (!identity::is_correct_password($user, $request->password)) {
+ print json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed")));
+ return;
+ }
+ $key = ORM::factory("rest_key")
+ ->where("user_id", $user->id)
+ ->find();
+ if (!$key->loaded) {
+ $key->user_id = $user->id;
+ $key->access_key = md5($user->name . rand());
+ $key->save();
+ Kohana::log("alert", Kohana::debug($key->as_array()));
+ }
+ print json_encode(array("status" => "OK", "token" => $key->access_key));
+ }
+
+ public function __call($function, $args) {
+ $access_token = $this->input->get("request_key");
+ $request = $this->input->post("request", null);
+
+ if (empty($access_token)) {
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ if (!empty($request)) {
+ $method = strtolower($this->input->server("HTTP_X_HTTP_METHOD_OVERRIDE", "POST"));
+ $request = json_decode($request);
+ } else {
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ try {
+ $key = ORM::factory("rest_key")
+ ->where("access_key", $access_token)
+ ->find();
+
+ if (!$key->loaded) {
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ $user = identity::lookup_user($key->user_id);
+ if (empty($user)) {
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Authorization failed")));
+ return;
+ }
+
+ if (empty($args[0])) {
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Invalid request parameters")));
+ return;
+ }
+
+ $handler_class = "{$function}_rest";
+ $handler_method = "{$method}_{$args[0]}";
+
+ if (!method_exists($handler_class, $handler_method)) {
+ Kohana::log("error", "$handler_class::$handler_method is not implemented");
+ print json_encode(array("status" => "ERROR",
+ "message" => (string)t("Service not implemented")));
+ return;
+ }
+
+ $response = call_user_func(array($handler_class, $handler_method), $request);
+
+ print json_encode($response);
+ } catch (Exception $e) {
+ Kohana::log("error", $e->__toString());
+ print json_encode(array("status" => "ERROR", "message" => (string)t("Internal error")));
+ }
+ }
+
+} \ No newline at end of file
diff --git a/modules/rest/helpers/rest_event.php b/modules/rest/helpers/rest_event.php
new file mode 100644
index 00000000..fd1c25be
--- /dev/null
+++ b/modules/rest/helpers/rest_event.php
@@ -0,0 +1,75 @@
+<?php defined("SYSPATH") or die("No direct script access.");/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class rest_event {
+ /**
+ * Called just before a user is deleted. This will remove the user from
+ * the user_homes directory.
+ */
+ static function user_before_delete($user) {
+ ORM::factory("rest_key")
+ ->where("id", $user->id)
+ ->delete_all();
+ }
+
+ /**
+ * Called after a user has been added. Just add a remote access key
+ * on every add.
+ */
+ static function user_add_form_admin_completed($user, $form) {
+ $key = ORM::factory("rest_key");
+ $key->user_id = $user->id;
+ $key->access_key = md5($user->name . rand());
+ $key->save();
+ }
+
+ /**
+ * Called when admin is editing a user
+ */
+ static function user_edit_form_admin($user, $form) {
+ self::_get_access_key_form($user, $form);
+ }
+
+ /**
+ * Called when user is editing their own form
+ */
+ static function user_edit_form($user, $form) {
+ self::_get_access_key_form($user, $form);
+ }
+
+ /**
+ * Get the form fields for user edit
+ */
+ static function _get_access_key_form($user, $form) {
+ $key = ORM::factory("rest_key")
+ ->where("user_id", $user->id)
+ ->find();
+
+ if (!$key->loaded) {
+ $key->user_id = $user->id;
+ $key->access_key = md5($user->name . rand());
+ $key->save();
+ }
+
+ $form->edit_user->input("access_key")
+ ->value($key->access_key)
+ ->readonly("readonly")
+ ->class("g-form-static")
+ ->label(t("Remote access key"));
+ }
+}
diff --git a/modules/rest/helpers/rest_installer.php b/modules/rest/helpers/rest_installer.php
new file mode 100644
index 00000000..274002c0
--- /dev/null
+++ b/modules/rest/helpers/rest_installer.php
@@ -0,0 +1,37 @@
+<?php defined("SYSPATH") or die("No direct script access.");
+/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class rest_installer {
+ static function install() {
+ Database::instance()
+ ->query("CREATE TABLE {rest_keys} (
+ `id` int(9) NOT NULL auto_increment,
+ `user_id` int(9) NOT NULL,
+ `access_key` char(32) NOT NULL,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY(`access_key`),
+ UNIQUE KEY(`user_id`))
+ DEFAULT CHARSET=utf8;");
+ module::set_version("rest", 1);
+ }
+
+ static function uninstall() {
+ Database::instance()->query("DROP TABLE IF EXISTS {rest_keys}");
+ }
+}
diff --git a/modules/rest/libraries/Form_Label.php b/modules/rest/libraries/Form_Label.php
new file mode 100644
index 00000000..315ff510
--- /dev/null
+++ b/modules/rest/libraries/Form_Label.php
@@ -0,0 +1,45 @@
+<?php defined("SYSPATH") or die("No direct script access.");
+/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class Form_Label_Core extends Form_Input {
+ protected $data = array(
+ "type" => "label",
+ "text" => "");
+
+ public function __construct($label) {
+ $this->data["text"] = $label;
+ }
+
+ public function __get($key) {
+ return isset($this->data[$key]) ? $this->data[$key] : null;
+ }
+
+ // In this element we never want print any html so make sure
+ // render and ultimately html_element only return the empty string
+ public function render() {
+ return $this->html_element();
+ }
+
+ public function html_element() {
+ $data = $this->data;
+ unset($data["text"]);
+ return "<p " . html::attributes() . ">{$this->data['text']}</p>";
+ }
+
+} // End Form Script \ No newline at end of file
diff --git a/modules/rest/models/rest_key.php b/modules/rest/models/rest_key.php
new file mode 100644
index 00000000..4dee8b65
--- /dev/null
+++ b/modules/rest/models/rest_key.php
@@ -0,0 +1,21 @@
+<?php defined("SYSPATH") or die("No direct script access.");
+/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class Rest_Key_Model extends ORM {
+}
diff --git a/modules/rest/module.info b/modules/rest/module.info
new file mode 100644
index 00000000..5576ec81
--- /dev/null
+++ b/modules/rest/module.info
@@ -0,0 +1,4 @@
+name = REST Access Module
+description = "The RESTful implementation/interface to Gallery3"
+
+version = 1
diff --git a/modules/rest/tests/Rest_Controller_Test.php b/modules/rest/tests/Rest_Controller_Test.php
new file mode 100644
index 00000000..16c5177b
--- /dev/null
+++ b/modules/rest/tests/Rest_Controller_Test.php
@@ -0,0 +1,220 @@
+<?php defined("SYSPATH") or die("No direct script access.");
+/**
+ * Gallery - a web based photo album viewer and editor
+ * Copyright (C) 2000-2009 Bharat Mediratta
+ *
+ * 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.
+ */
+class Rest_Controller_Test extends Unit_Test_Case {
+ public function setup() {
+ $this->_save = array($_GET, $_POST, $_SERVER);
+ $this->_user = identity::create_user("access_test", "Access Test", "password");
+ $key = ORM::factory("rest_key");
+ $this->_access_key = $key->access_key = md5($this->_user->name . rand());
+ $key->user_id = $this->_user->id;
+ $key->save();
+
+ $root = ORM::factory("item", 1);
+ $this->_album = album::create($root, "album", "Test Album", rand());
+ $this->_child = album::create($this->_album, "child", "Test Child Album", rand());
+
+ $filename = MODPATH . "gallery/tests/test.jpg";
+ $rand = rand();
+ $this->_photo = photo::create($this->_child, $filename, "$rand.jpg", $rand);
+ $this->_path = $this->_photo->relative_path();
+ }
+
+ public function teardown() {
+ list($_GET, $_POST, $_SERVER) = $this->_save;
+
+ try {
+ if (!empty($this->_user)) {
+ $this->_user->delete();
+ }
+ if (!empty($this->_album)) {
+ $this->_album->delete();
+ }
+ } catch (Exception $e) { }
+ }
+
+ public function rest_access_key_exists_test() {
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_POST["request"] = json_encode(array("user" => "access_test", "password" => "password"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "OK", "token" => $this->_access_key)),
+ $this->_call_controller());
+ }
+
+ public function rest_access_key_generated_test() {
+ ORM::factory("rest_key")
+ ->where("access_key", $this->_access_key)
+ ->delete();
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_POST["request"] = json_encode(array("user" => "access_test", "password" => "password"));
+
+ $results = json_decode($this->_call_controller());
+
+ $this->assert_equal("OK", $results->status);
+ $this->assert_false(empty($results->token));
+ }
+
+ public function rest_access_key_no_parameters_test() {
+ $_SERVER["REQUEST_METHOD"] = "POST";
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller());
+ }
+
+ public function rest_access_key_user_not_found_test() {
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_POST["request"] = json_encode(array("user" => "access_test2", "password" => "password"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller());
+ }
+
+ public function rest_access_key_invalid_password_test() {
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_POST["request"] = json_encode(array("user" => "access_test", "password" => "invalid"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller());
+ }
+
+ public function rest_get_album_no_request_key_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_POST["request"] = json_encode(array("path" => "/test_album"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller("rest"));
+ }
+
+ public function rest_get_album_no_request_content_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_GET["request_key"] = $this->_access_key;
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller("rest"));
+ }
+
+ public function rest_get_album_invalid_key_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+
+ $_SERVER["REQUEST_METHOD"] = "POST";
+ $_GET["request_key"] = md5($this->_access_key); // screw up the access key
+ $_POST["request"] = json_encode(array("path" => "/test_album"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller());
+ }
+
+ public function rest_get_album_no_user_for_key_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+ $_SERVER["REQUEST_METHOD"] = "POST";
+
+ $_GET["request_key"] = $this->_access_key;
+ $_POST["request"] = json_encode(array("path" => "/test_album"));
+
+ $this->_user->delete();
+ unset($this->_user);
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Authorization failed"))),
+ $this->_call_controller("rest"));
+ }
+
+ public function rest_get_album_no_resource_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+ $_SERVER["REQUEST_METHOD"] = "POST";
+
+ $_GET["request_key"] = $this->_access_key;
+ $_POST["request"] = json_encode(array("path" => "/test_album"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Invalid request parameters"))),
+ $this->_call_controller("rest"));
+ }
+
+ public function rest_get_album_no_handler_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+ $_SERVER["REQUEST_METHOD"] = "POST";
+
+ $_GET["request_key"] = $this->_access_key;
+ $_POST["request"] = json_encode(array("path" => "/test_album"));
+
+ $this->assert_equal(
+ json_encode(array("status" => "ERROR", "message" => (string)t("Service not implemented"))),
+ $this->_call_controller("rest", "album"));
+ }
+
+ public function rest_get_album_test() {
+ $_SERVER["HTTP_X_HTTP_METHOD_OVERRIDE"] = "GET";
+ $_SERVER["REQUEST_METHOD"] = "POST";
+
+ $_GET["request_key"] = $this->_access_key;
+ $_POST["request"] = json_encode(array("path" => $this->_path));
+
+ $this->assert_equal(
+ json_encode(array("status" => "OK", "message" => (string)t("Processed"),
+ "item" => array("path" => $this->_photo->relative_path_cache,
+ "title" => $this->_photo->title,
+ "thumb_url" => $this->_photo->thumb_url(),
+ "description" => $this->_photo->description,
+ "internet_address" => $this->_photo->slug,
+ "type" => $this->_photo->type))),
+ $this->_call_controller("rest", "photo"));
+ }
+
+ private function _call_controller($method="access_key", $arg=null) {
+ $controller = new Rest_Controller();
+
+ ob_start();
+ call_user_func(array($controller, $method), $arg);
+ $results = ob_get_contents();
+ ob_end_clean();
+
+ return $results;
+ }
+}
+
+class rest_rest {
+ static $request = null;
+
+ static function get_photo($request) {
+ self::$request = $request;
+ $item = ORM::factory("item")
+ ->where("relative_path_cache", $request->path)
+ ->find();
+ $response["path"] = $item->relative_path_cache;
+ $response["title"] = $item->title;
+ $response["thumb_url"] = $item->thumb_url();
+ $response["description"] = $item->description;
+ $response["internet_address"] = $item->slug;
+ $response["type"] = $item->type;
+ return array("status" => "OK", "message" => (string)t("Processed"), "item" => $response);
+ }
+
+}