summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTim Almdal <tnalmdal@shaw.ca>2009-11-25 13:02:14 -0800
committerTim Almdal <tnalmdal@shaw.ca>2009-11-25 13:02:14 -0800
commitdc67cf64813361b34c366123f37d88ef6988fcc8 (patch)
treed4f50f537fd5138b72d919e75747b9ead497b671
parent15cf6870d5a4fc4dae3c5b61b1cb16ccdc226821 (diff)
Remove the REST_Controller and assorted baggage. Completes ticket #917
-rw-r--r--modules/gallery/controllers/rest.php183
-rw-r--r--modules/gallery/helpers/rest.php116
-rw-r--r--modules/gallery/tests/Controller_Auth_Test.php16
-rw-r--r--modules/gallery/tests/REST_Controller_Test.php197
-rw-r--r--modules/gallery/tests/REST_Helper_Test.php45
-rw-r--r--modules/gallery/tests/controller_auth_data.txt11
-rw-r--r--modules/rss/controllers/rss.php2
-rw-r--r--modules/rss/helpers/rss.php2
8 files changed, 3 insertions, 569 deletions
diff --git a/modules/gallery/controllers/rest.php b/modules/gallery/controllers/rest.php
deleted file mode 100644
index 2edf079f..00000000
--- a/modules/gallery/controllers/rest.php
+++ /dev/null
@@ -1,183 +0,0 @@
-<?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.
- */
-/**
- * This abstract controller makes it easy to create a RESTful controller. To use it, create a
- * subclass which defines the resource type and implements get/post/put/delete methods, like this:
- *
- * class Comment_Controller extends REST_Controller {
- * protected $resource_type = "comment"; // this tells REST which model to use
- *
- * public function _index() {
- * // Handle GET request to /controller
- * }
- *
- * public function _show(ORM $comment) {
- * // Handle GET request to /comments/{comment_id}
- * }
- *
- * public function _update(ORM $comment) {
- * // Handle PUT request to /comments/{comment_id}
- * }
- *
- * public function _create(ORM $comment) {
- * // Handle POST request to /comments
- * }
- *
- * public function _delete(ORM $comment) {
- * // Handle DELETE request to /comments/{comments_id}
- * }
- *
- * public function _form_add($parameters) {
- * // Handle GET request to /form/add/comments
- * // Show a form for creating a new comment
- * }
- *
- * public function _form_edit(ORM $comment) {
- * // Handle GET request to /form/edit/comments
- * // Show a form for editing an existing comment
- * }
- *
- * A request to http://example.com/gallery3/comments/3 will result in a call to
- * REST_Controller::__call(3) which will load up the comment associated with id 3. If there's
- * no such comment, it returns a 404. Otherwise, it will then delegate to
- * Comment_Controller::get() with the ORM instance as an argument.
- */
-class REST_Controller extends Controller {
- protected $resource_type = null;
-
- public function __construct() {
- if ($this->resource_type == null) {
- throw new Exception("@todo ERROR_MISSING_RESOURCE_TYPE");
- }
- parent::__construct();
- }
-
- /**
- * Handle dispatching for all REST controllers.
- */
- public function __call($function, $args) {
- // If no parameter was provided after the controller name (eg "/albums") then $function will
- // be set to "index". Otherwise, $function is the first parameter, and $args are all
- // subsequent parameters.
- $request_method = rest::request_method();
- if ($function == "index" && $request_method == "get") {
- return $this->_index();
- }
-
- $resource = ORM::factory($this->resource_type, (int)$function);
- if (!$resource->loaded && $request_method != "post") {
- return Kohana::show_404();
- }
-
- switch ($request_method) {
- case "get":
- return $this->_show($resource);
-
- case "put":
- access::verify_csrf();
- return $this->_update($resource);
-
- case "delete":
- access::verify_csrf();
- return $this->_delete($resource);
-
- case "post":
- access::verify_csrf();
- return $this->_create($resource);
- }
- }
-
- /* We're editing an existing item, load it from the database. */
- public function form_edit($resource_id) {
- if ($this->resource_type == null) {
- throw new Exception("@todo ERROR_MISSING_RESOURCE_TYPE");
- }
-
- $resource = ORM::factory($this->resource_type, $resource_id);
- if (!$resource->loaded) {
- return Kohana::show_404();
- }
-
- // Security checks must be performed in _form_edit
- return $this->_form_edit($resource);
- }
-
- /* We're adding a new item, pass along any additional parameters. */
- public function form_add($parameters) {
- // Security checks must be performed in _form_add
- return $this->_form_add($parameters);
- }
-
- /**
- * Perform a GET request on the controller root
- * (e.g. http://www.example.com/gallery3/comments)
- */
- public function _index() {
- throw new Exception("@todo _create NOT IMPLEMENTED");
- }
-
- /**
- * Perform a POST request on this resource
- * @param ORM $resource the instance of this resource type
- */
- public function _create($resource) {
- throw new Exception("@todo _create NOT IMPLEMENTED");
- }
-
- /**
- * Perform a GET request on this resource
- * @param ORM $resource the instance of this resource type
- */
- public function _show($resource) {
- throw new Exception("@todo _show NOT IMPLEMENTED");
- }
-
- /**
- * Perform a PUT request on this resource
- * @param ORM $resource the instance of this resource type
- */
- public function _update($resource) {
- throw new Exception("@todo _update NOT IMPLEMENTED");
- }
-
- /**
- * Perform a DELETE request on this resource
- * @param ORM $resource the instance of this resource type
- */
- public function _delete($resource) {
- throw new Exception("@todo _delete NOT IMPLEMENTED");
- }
-
- /**
- * Present a form for adding a new resource
- * @param string part of the URI after the controller name
- */
- public function _form_add($parameter) {
- throw new Exception("@todo _form_add NOT IMPLEMENTED");
- }
-
- /**
- * Present a form for editing an existing resource
- * @param ORM $resource the resource container for instances of this resource type
- */
- public function _form_edit($resource) {
- throw new Exception("@todo _form_edit NOT IMPLEMENTED");
- }
-}
diff --git a/modules/gallery/helpers/rest.php b/modules/gallery/helpers/rest.php
deleted file mode 100644
index a63b94c8..00000000
--- a/modules/gallery/helpers/rest.php
+++ /dev/null
@@ -1,116 +0,0 @@
-<?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_Core {
- const OK = "200 OK";
- const CREATED = "201 Created";
- const ACCEPTED = "202 Accepted";
- const NO_CONTENT = "204 No Content";
- const RESET_CONTENT = "205 Reset Content";
- const PARTIAL_CONTENT = "206 Partial Content";
- const MOVED_PERMANENTLY = "301 Moved Permanently";
- const FOUND = "302 Found";
- const SEE_OTHER = "303 See Other";
- const NOT_MODIFIED = "304 Not Modified";
- const TEMPORARY_REDIRECT = "307 Temporary Redirect";
- const BAD_REQUEST = "400 Bad Request";
- const UNAUTHORIZED = "401 Unauthorized";
- const FORBIDDEN = "403 Forbidden";
- const NOT_FOUND = "404 Not Found";
- const METHOD_NOT_ALLOWED = "405 Method Not Allowed";
- const NOT_ACCEPTABLE = "406 Not Acceptable";
- const CONFLICT = "409 Conflict";
- const GONE = "410 Gone";
- const LENGTH_REQUIRED = "411 Length Required";
- const PRECONDITION_FAILED = "412 Precondition Failed";
- const UNSUPPORTED_MEDIA_TYPE = "415 Unsupported Media Type";
- const EXPECTATION_FAILED = "417 Expectation Failed";
- const INTERNAL_SERVER_ERROR = "500 Internal Server Error";
- const SERVICE_UNAVAILABLE = "503 Service Unavailable";
-
- const XML = "application/xml";
- const ATOM = "application/atom+xml";
- const RSS = "application/rss+xml";
- const JSON = "application/json";
- const HTML = "text/html";
-
- /**
- * We're expecting to run in an environment that only supports GET/POST, so expect to tunnel
- * PUT and DELETE through POST.
- *
- * Returns the HTTP request method taking into consideration PUT/DELETE tunneling.
- * @return string HTTP request method
- */
- static function request_method() {
- if (request::method() == "get") {
- return "get";
- } else {
- $input = Input::instance();
- switch (strtolower($input->post("_method", $input->get("_method", request::method())))) {
- case "put": return "put";
- case "delete": return "delete";
- default: return "post";
- }
- }
- }
-
- /**
- * Choose an output format based on what the client prefers to accept.
- * @return string "html", "xml" or "json"
- */
- static function output_format() {
- // Pick a format, but let it be overridden.
- $input = Input::instance();
- $fmt = $input->get(
- "_format", $input->post(
- "_format", request::preferred_accept(
- array("xhtml", "html", "xml", "json"))));
-
- // Some browsers (Chrome!) prefer xhtml over html, but we'll normalize this to html for now.
- if ($fmt == "xhtml") {
- $fmt = "html";
- }
- return $fmt;
- }
-
- /**
- * Set HTTP response code.
- * @param string Use one of the status code constants defined in this class.
- */
- static function http_status($status_code) {
- header("HTTP/1.1 " . $status_code);
- }
-
- /**
- * Set HTTP Location header.
- * @param string URL
- */
- static function http_location($url) {
- header("Location: " . $url);
- }
-
- /**
- * Set HTTP Content-Type header.
- * @param string content type
- */
- static function http_content_type($type) {
- header("Content-Type: " . $type);
- }
-}
diff --git a/modules/gallery/tests/Controller_Auth_Test.php b/modules/gallery/tests/Controller_Auth_Test.php
index 0a7076c6..124d8b4c 100644
--- a/modules/gallery/tests/Controller_Auth_Test.php
+++ b/modules/gallery/tests/Controller_Auth_Test.php
@@ -18,11 +18,6 @@
* Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
*/
class Controller_Auth_Test extends Unit_Test_Case {
- static $rest_methods = array("_index", "_show", "_form_edit", "_form_add", "_create",
- "_update", "_delete");
-
- static $rest_methods_with_csrf_check = array("_update", "_delete", "_create");
-
public function find_missing_auth_test() {
$found = array();
$controllers = explode("\n", `git ls-files '*/*/controllers/*.php'`);
@@ -46,7 +41,6 @@ class Controller_Auth_Test extends Unit_Test_Case {
}
$is_admin_controller = false;
- $is_rest_controller = false;
$open_braces = 0;
$function = null;
@@ -64,7 +58,6 @@ class Controller_Auth_Test extends Unit_Test_Case {
$function = null;
} else if ($open_braces == 0) {
$is_admin_controller = false;
- $is_rest_controller = false;
}
} else if ($token == "{") {
$open_braces++;
@@ -75,8 +68,6 @@ class Controller_Auth_Test extends Unit_Test_Case {
if ($open_braces == 0 && $token[0] == T_EXTENDS) {
if (self::_token_matches(array(T_STRING, "Admin_Controller"), $tokens, $token_number + 1)) {
$is_admin_controller = true;
- } else if (self::_token_matches(array(T_STRING, "REST_Controller"), $tokens, $token_number + 1)) {
- $is_rest_controller = true;
}
} else if ($open_braces == 1 && $token[0] == T_FUNCTION) {
$line = $token[2];
@@ -101,13 +92,8 @@ class Controller_Auth_Test extends Unit_Test_Case {
$is_rss_feed = $name == "feed" && strpos(basename($controller), "_rss.php");
- if ((!$is_static || $is_rss_feed) &&
- (!$is_private ||
- ($is_rest_controller && in_array($name, self::$rest_methods)))) {
+ if ((!$is_static || $is_rss_feed) && !$is_private) {
$function = self::_function($name, $line, $is_admin_controller);
- if ($is_rest_controller && in_array($name, self::$rest_methods_with_csrf_check)) {
- $function->checks_csrf(true);
- }
}
}
diff --git a/modules/gallery/tests/REST_Controller_Test.php b/modules/gallery/tests/REST_Controller_Test.php
deleted file mode 100644
index 8fb04d86..00000000
--- a/modules/gallery/tests/REST_Controller_Test.php
+++ /dev/null
@@ -1,197 +0,0 @@
-<?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->_post = $_POST;
- $this->mock_controller = new Mock_RESTful_Controller("mock");
- $this->mock_not_loaded_controller = new Mock_RESTful_Controller("mock_not_loaded");
- $_POST = array();
- }
-
- public function teardown() {
- $_POST = $this->_post;
- }
-
- public function dispatch_index_test() {
- $_SERVER["REQUEST_METHOD"] = "GET";
- $_POST["_method"] = "";
- $this->mock_controller->__call("index", "");
- $this->assert_equal("index", $this->mock_controller->method_called);
- }
-
- public function dispatch_show_test() {
- $_SERVER["REQUEST_METHOD"] = "GET";
- $_POST["_method"] = "";
- $this->mock_controller->__call("3", "");
- $this->assert_equal("show", $this->mock_controller->method_called);
- $this->assert_equal("Mock_Model", get_class($this->mock_controller->resource));
- }
-
- public function dispatch_update_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "PUT";
- $_POST["csrf"] = access::csrf_token();
- $this->mock_controller->__call("3", "");
- $this->assert_equal("update", $this->mock_controller->method_called);
- $this->assert_equal("Mock_Model", get_class($this->mock_controller->resource));
- }
-
- public function dispatch_update_fails_without_csrf_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "PUT";
- try {
- $this->mock_controller->__call("3", "");
- $this->assert_false(true, "this should fail with a forbidden exception");
- } catch (Exception $e) {
- // pass
- }
- }
-
- public function dispatch_delete_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "DELETE";
- $_POST["csrf"] = access::csrf_token();
- $this->mock_controller->__call("3", "");
- $this->assert_equal("delete", $this->mock_controller->method_called);
- $this->assert_equal("Mock_Model", get_class($this->mock_controller->resource));
- }
-
- public function dispatch_delete_fails_without_csrf_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "DELETE";
- try {
- $this->mock_controller->__call("3", "");
- $this->assert_false(true, "this should fail with a forbidden exception");
- } catch (Exception $e) {
- // pass
- }
- }
-
- public function dispatch_404_test() {
- /* The dispatcher should throw a 404 if the resource isn't loaded and the method isn't POST. */
- $methods = array(
- array("GET", ""),
- array("POST", "PUT"),
- array("POST", "DELETE"));
-
- foreach ($methods as $method) {
- $_SERVER["REQUEST_METHOD"] = $method[0];
- $_POST["_method"] = $method[1];
- $exception_caught = false;
- try {
- $this->mock_not_loaded_controller->__call(rand(), "");
- } catch (Kohana_404_Exception $e) {
- $exception_caught = true;
- }
- $this->assert_true($exception_caught, "$method[0], $method[1]");
- }
- }
-
- public function dispatch_create_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "";
- $_POST["csrf"] = access::csrf_token();
- $this->mock_not_loaded_controller->__call("", "");
- $this->assert_equal("create", $this->mock_not_loaded_controller->method_called);
- $this->assert_equal(
- "Mock_Not_Loaded_Model", get_class($this->mock_not_loaded_controller->resource));
- }
-
- public function dispatch_create_fails_without_csrf_test() {
- $_SERVER["REQUEST_METHOD"] = "POST";
- $_POST["_method"] = "";
- try {
- $this->mock_not_loaded_controller->__call("", "");
- $this->assert_false(true, "this should fail with a forbidden exception");
- } catch (Exception $e) {
- // pass
- }
- }
-
- public function dispatch_form_test_add() {
- $this->mock_controller->form_add("args");
- $this->assert_equal("form_add", $this->mock_controller->method_called);
- $this->assert_equal("args", $this->mock_controller->resource);
- }
-
- public function dispatch_form_test_edit() {
- $this->mock_controller->form_edit("1");
- $this->assert_equal("form_edit", $this->mock_controller->method_called);
- $this->assert_equal("Mock_Model", get_class($this->mock_controller->resource));
- }
-
- public function routes_test() {
- $this->assert_equal("mock/form_add/args", router::routed_uri("form/add/mock/args"));
- $this->assert_equal("mock/form_edit/args", router::routed_uri("form/edit/mock/args"));
- $this->assert_equal(null, router::routed_uri("rest/args"));
- }
-}
-
-class Mock_RESTful_Controller extends REST_Controller {
- public $method_called;
- public $resource;
-
- public function __construct($type) {
- $this->resource_type = $type;
- parent::__construct();
- }
-
- public function _index() {
- $this->method_called = "index";
- }
-
- public function _create($resource) {
- $this->method_called = "create";
- $this->resource = $resource;
- }
-
- public function _show($resource) {
- $this->method_called = "show";
- $this->resource = $resource;
- }
-
- public function _update($resource) {
- $this->method_called = "update";
- $this->resource = $resource;
- }
-
- public function _delete($resource) {
- $this->method_called = "delete";
- $this->resource = $resource;
- }
-
- public function _form_add($args) {
- $this->method_called = "form_add";
- $this->resource = $args;
- }
-
- public function _form_edit($resource) {
- $this->method_called = "form_edit";
- $this->resource = $resource;
- }
-}
-
-class Mock_Model {
- public $loaded = true;
-}
-
-class Mock_Not_Loaded_Model {
- public $loaded = false;
-}
diff --git a/modules/gallery/tests/REST_Helper_Test.php b/modules/gallery/tests/REST_Helper_Test.php
deleted file mode 100644
index 1bfc63ab..00000000
--- a/modules/gallery/tests/REST_Helper_Test.php
+++ /dev/null
@@ -1,45 +0,0 @@
-<?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_Helper_Test extends Unit_Test_Case {
- public function setup() {
- $this->_post = $_POST;
- }
-
- public function teardown() {
- $_POST = $this->_post;
- }
-
- public function request_method_test() {
- foreach (array("GET", "POST") as $method) {
- foreach (array("", "PUT", "DELETE") as $tunnel) {
- if ($method == "GET") {
- $expected = "GET";
- } else {
- $expected = $tunnel == "" ? $method : $tunnel;
- }
- $_SERVER["REQUEST_METHOD"] = $method;
- $_POST["_method"] = $tunnel;
-
- $this->assert_equal(strtolower(rest::request_method()), strtolower($expected),
- "Request method: {$method}, tunneled: {$tunnel}");
- }
- }
- }
-}
diff --git a/modules/gallery/tests/controller_auth_data.txt b/modules/gallery/tests/controller_auth_data.txt
index 73950d88..1fe29ffb 100644
--- a/modules/gallery/tests/controller_auth_data.txt
+++ b/modules/gallery/tests/controller_auth_data.txt
@@ -13,17 +13,6 @@ modules/gallery/controllers/login.php html
modules/gallery/controllers/login.php auth_html DIRTY_AUTH
modules/gallery/controllers/logout.php index DIRTY_CSRF|DIRTY_AUTH
modules/gallery/controllers/maintenance.php index DIRTY_AUTH
-modules/gallery/controllers/rest.php __construct DIRTY_AUTH
-modules/gallery/controllers/rest.php __call DIRTY_AUTH
-modules/gallery/controllers/rest.php form_edit DIRTY_AUTH
-modules/gallery/controllers/rest.php form_add DIRTY_AUTH
-modules/gallery/controllers/rest.php _index DIRTY_AUTH
-modules/gallery/controllers/rest.php _create DIRTY_AUTH
-modules/gallery/controllers/rest.php _show DIRTY_AUTH
-modules/gallery/controllers/rest.php _update DIRTY_AUTH
-modules/gallery/controllers/rest.php _delete DIRTY_AUTH
-modules/gallery/controllers/rest.php _form_add DIRTY_AUTH
-modules/gallery/controllers/rest.php _form_edit DIRTY_AUTH
modules/gallery/controllers/simple_uploader.php start DIRTY_AUTH
modules/gallery/controllers/simple_uploader.php finish DIRTY_AUTH
modules/gallery/controllers/upgrader.php index DIRTY_AUTH
diff --git a/modules/rss/controllers/rss.php b/modules/rss/controllers/rss.php
index 1ecec9af..ed2acef8 100644
--- a/modules/rss/controllers/rss.php
+++ b/modules/rss/controllers/rss.php
@@ -62,7 +62,7 @@ class Rss_Controller extends Controller {
url::abs_site(str_replace("&", "&amp;", url::merge(array("page" => $page + 1))));
}
- rest::http_content_type(rest::RSS);
+ header("Content-Type: application/rss+xml");
print $view;
}
} \ No newline at end of file
diff --git a/modules/rss/helpers/rss.php b/modules/rss/helpers/rss.php
index 81ff175f..4260206c 100644
--- a/modules/rss/helpers/rss.php
+++ b/modules/rss/helpers/rss.php
@@ -31,6 +31,6 @@ class rss_Core {
*/
static function feed_link($uri) {
$url = url::site("rss/feed/$uri");
- return "<link rel=\"alternate\" type=\"" . rest::RSS . "\" href=\"$url\" />";
+ return "<link rel=\"alternate\" type=\"application/rss+xml\" href=\"$url\" />";
}
} \ No newline at end of file