Iron — это высокоуровневый веб-фреймворк, написанный на языке программирования Rust и построенный на базе другой небезызвестной библиотеки hyper. Iron разработан таким образом, чтобы пользоваться всеми преимуществами, которые нам предоставляет Rust.


Философия


Iron построен на принципе расширяемости настолько, насколько это возможно.
Он вводит понятия для расширения собственного функционала:


  • "промежуточные" типажи — используются для реализации сквозного функционала в обработке запросов;
  • модификаторы — используются для изменения запросов и ответов наиболее эргономичным
    способом.

С базовой частью модификаторов и промежуточных типажей вы познакомитесь в ходе статьи.


Официальный сайт Iron


Создание проекта


Для начала создадим проект с помощью Cargo, используя команду:


cargo new rust-iron-tutorial --bin

Далее добавим в раздел [dependencies] файла Cargo.toml зависимость iron = "0.4.0".


Пишем первую программу с использованием Iron


Напишем первую простенькую программу на Rust с использованием Iron, которая на любые запросы по порту 3000 будет отвечать текстом "Hello habrahabr!".


extern crate iron;
use iron::prelude::*;
use iron::status;
fn main() {
   Iron::new(|_: &mut Request| {
       Ok(Response::with((status::Ok, "Hello habrahabr!\n")))
   }).http("localhost:3000").unwrap();
}

Запустите код при помощи команды cargo run и после того, как компиляция завершится и программа запустится, протестируйте сервис, например, при помощи curl:


[loomaclin@loomaclin ~]$ curl localhost:3000
Hello habrahabr!

Давайте разберём программу, чтобы понимать, что тут происходит. В первой строке программы импортируется пакет iron.
Во второй строке был подключен модуль-прелюдия, содержащий набор наиболее важных типов, таких как Request,
Response, IronRequest, IronResult, IronError и Iron. В третьей строке подключается модуль status, содержащий списки кодов для ответов на запросы. Iron::new создаёт новый инстанс Iron'а, который, в свою очередь, является базовым объектом вашего сервера. Он принимает параметром объект, реализующий типаж Handler. В нашем случае мы передаём замыкание, аргументом которого является изменяемая ссылка на переданный запрос.


Указываем mime-type в заголовке ответа


Чаще всего при построении веб-сервисов (SOAP, REST) требуется отсылать ответы с указанием типа контента, который они содержат. Для этого в Iron предусмотрены специальные средства.


Выполним следующее.


Подключаем соответствующую структуру:


use iron::mime::Mime;

Связываем имя content_type, которое будет хранить распарсенное при помощи подключенного типа Mime значение типа:


let content_type = "application/json".parse::<Mime>().unwrap();

Модифицируем строку ответа на запрос следующим образом:


Ok(Response::with((content_type, status::Ok, "{}")))

Запускаем программу и проверяем работоспособность:


[loomaclin@loomaclin ~]$ curl -v localhost:3000
* Rebuilt URL to: localhost:3000/
*   Trying ::1...
* Connected to localhost (::1) port 3000 (#0)
> GET / HTTP/1.1
> Host: localhost:3000
> User-Agent: curl/7.49.1
> Accept: */*
>
< HTTP/1.1 200 OK
< Content-Type: application/json
< Date: Tue, 12 Jul 2016 19:53:21 GMT
< Content-Length: 2
<
* Connection #0 to host localhost left intact
{}

Управление статус-кодами ответов


В перечислении StatusCode, расположенном в модуле status, располагаются всевозможные статус-коды. Давайте воспользуемся этим и вернём "клиенту" ошибку 404 — NotFound, изменив строку с формированием ответа на запрос:


Ok(Response::with((content_type, status::NotFound)))

Проверка:


[loomaclin@loomaclin ~]$ curl -v localhost:3000
* Rebuilt URL to: localhost:3000/
*   Trying ::1...
* Connected to localhost (::1) port 3000 (#0)
> GET / HTTP/1.1
> Host: localhost:3000
> User-Agent: curl/7.49.1
> Accept: */*
>
< HTTP/1.1 404 Not Found
< Content-Length: 2
< Content-Type: application/json
< Date: Tue, 12 Jul 2016 20:55:40 GMT
<
* Connection #0 to host localhost left intact

Примечание: по сути, весь модуль status является обёрткой для соответствующих перечислений в библиотеке hyper, на которой базируется iron.


Перенаправление запросов


Для редиректа в iron используется структура Redirect из модуля modifiers (не путать с modifier). Она состоит из URL цели, куда необходимо будет произвести перенаправление.
Попробуем её применить, проделав следующие изменения:


Подключаем структуру Redirect:


use iron::modifiers::Redirect;

К подключению модуля status добавляем подключение модуля Url:


use iron::{Url, status};

Связываем имя url, которое будет хранить распарсенное значение адреса редиректа:


let url = Url::parse("https://habrahabr.ru/").unwrap();

Меняем блок инициализации Iron следующим образом:


   Iron::new(move |_: &mut Request | {
       Ok(Response::with((status::Found, Redirect(url.clone()))))
   }).http("localhost:3000").unwrap();

Проверяем результат:


[loomaclin@loomaclin ~]$ curl -v localhost:3000
* Rebuilt URL to: localhost:3000/
*   Trying ::1...
* Connected to localhost (::1) port 3000 (#0)
> GET / HTTP/1.1
> Host: localhost:3000
> User-Agent: curl/7.49.1
> Accept: */*
>
< HTTP/1.1 302 Found
< Location: https://habrahabr.ru/
< Date: Tue, 12 Jul 2016 21:39:24 GMT
< Content-Length: 0
<
* Connection #0 to host localhost left intact

Также вы можете воспользоваться ещё одной структурой RedirectRaw из модуля modifiers, для конструирования которой требуется лишь строка.


Работа с типом http-запроса


У структуры Request есть поле method, позволяющее определять тип пришедшего http-запроса.
Напишем сервис, который будет сохранять в файл данные, переданные в теле запроса с типом Put,
считывать данные из файла и передавать их в ответе на запрос с типом Get.


Аннотируем импортируемый контейнер iron атрибутом macro_use, чтобы в дальнейшем использовать макросы iexpect и itry для обработки ошибочных ситуаций:


#[macro_use]
extern crate iron;

Подключаем модули для работы с файловой системой и вводом/выводом из стандартной библиотеки:


use std::io;
use std::fs;

Подключаем модуль method, содержащий список типов http-запросов:


use iron::method;

Меняем блок инициализации Iron таким образом, чтобы связать полученный запрос с именем req:


Iron::new(|req: &mut Request| {
   ...
   ...
   ...
}.http("localhost:3000").unwrap();

Добавляем в обработку запроса сопоставление с образцом поля method для двух типов запросов Get и Put, а для остальных будем использовать ответ в виде статус-кода BadRequest:


       Ok(match req.method {
           method::Get => {
               let f = iexpect!(fs::File::open("foo.txt").ok(), (status::Ok, ""));
               Response::with((status::Ok, f))
           },
           method::Put => {
               let mut f = itry!(fs::File::create("foo.txt"));
               itry!(io::copy(&mut req.body, &mut f));
               Response::with(status::Created)
           },
           _ => Response::with(status::BadRequest)
       }

В Iron макрос iexcept используется для разворачивания переданного в него объекта типа Option и в случае, если Option содержит None макрос возвращает Ok(Response::new()) с модификатором по умолчанию status::BadRequest.
Макрос itry используется для оборачивания ошибки в IronError.


Пробуем запустить и проверить работоспособность.


PUT:


[loomaclin@loomaclin ~]$ curl -X PUT -d my_file_content localhost:3000
[loomaclin@loomaclin ~]$ cat ~/IdeaProjects/cycle/foo.txt
my_file_content

GET:


[loomaclin@loomaclin ~]$ curl localhost:3000
my_file_content

POST:


[loomaclin@loomaclin ~]$ curl -X POST -v localhost:3000
* Rebuilt URL to: localhost:3000/
*   Trying ::1...
* Connected to localhost (::1) port 3000 (#0)
> POST / HTTP/1.1
> Host: localhost:3000
> User-Agent: curl/7.49.1
> Accept: */*
>
< HTTP/1.1 400 Bad Request
< Content-Length: 0
< Date: Tue, 12 Jul 2016 22:29:58 GMT
<
* Connection #0 to host localhost left intact

Реализация сквозного функционала при помощи пре- и пост-процессинга


В iron также присутствуют типажи BeforeMiddleware,AfterMiddleware и AroundMiddleware для сквозного функционала,
позволяющие реализовывать логику по обработке запроса до того, как она началась в основном обработчике, и после того, как она там завершилась.


Напишем пример использования AOP'о подобных указанных типажей:


Код примера
extern crate iron;

use iron::prelude::*;
use iron::{BeforeMiddleware, AfterMiddleware, AroundMiddleware, Handler};

struct SampleStruct;
struct SampleStructAroundHandler<H:Handler> { logger: SampleStruct, handler: H}

impl BeforeMiddleware for SampleStruct {
   fn before(&self, req: &mut Request) -> IronResult<()> {
       println!("До обработки запроса.");
       Ok(())
   }
}

impl AfterMiddleware for SampleStruct {
   fn after(&self, req: &mut Request, res: Response) -> IronResult<Response> {
       println!("После обработки запроса.");
       Ok(res)
   }
}

impl<H: Handler> Handler for SampleStructAroundHandler<H> {
   fn handle(&self, req: &mut Request) -> IronResult<Response> {
       println!("А я ещё один обработчик запроса.");
       let res = self.handler.handle(req);
       res
   }
}

impl AroundMiddleware for SampleStruct {
   fn around(self, handler: Box<Handler>) -> Box<Handler> {
       Box::new(SampleStructAroundHandler {
           logger: self,
           handler: handler
       }) as Box<Handler>
   }
}

fn sample_of_middlewares(_:&mut Request) -> IronResult<Response> {
   println!("В основном обработчике запроса.");
   Ok(Response::with((iron::status::Ok, "Привет, я ответ на запрос!")))
}

fn main() {
   let mut chain = Chain::new(sample_of_middlewares);
   chain.link_before(SampleStruct);
   chain.link_after(SampleStruct);
   chain.link_around(SampleStruct);
   Iron::new(chain).http("localhost:3000").unwrap();
}

В этом примере вводится структура SampleStruct, для которой реализуются типажи BeforeMiddleware с функцией before и AfterMiddleware с функцией after. С их помощью может быть реализована вся сквозная логика. Типаж AroundMiddleware используется совместно с типажом Handler для добавления дополнительного обработчика. Добавление всех реализованных
обработчиков в жизненный цикл обработки запроса производится с помощью специального типажа Chain, позволяющего формировать цепь вызовов пре- и пост-обработчиков.


Протестируем программу.
В консоли:


[loomaclin@loomaclin ~]$ curl localhost:3000
Привет, я ответ на запрос!

В выводе программы:


   Running `target/debug/cycle`
До обработки запроса.
А я ещё один обработчик запроса.
В основном обработчике запроса.
После обработки запроса.

Роутинг


Какое серверное API может обойтись без роутинга? Добавим его =) Модифицируем наш базовый пример из начала статьи следующим образом.


Подключаем коллекцию из стандартной библиотеки:


use std::collections:HashMap;

Объявим структуру для хранения коллекции вида "путь — обработчик" и опишем для этой структуры конструктор, который будет производить инициализацию этой коллекции, и функцию для добавления в коллекцию новых роутов с их обработчиками:


struct Router {
    routes: HashMap<String, Box<Handler>>
}

impl Router {

    fn new() -> Self {
        Router {
            routes: HashMap::new()
        }
    }

    fn add_route<H>(&mut self, path: String, handler: H) where H: Handler {
        self.routes.insert(path, Box::new(handler));
    }
}

Для использования нашей структуры в связке с Iron необходимо реализовать для неё типаж Handler с функцией handle:


impl Handler for Router {
    fn handle(&self, req: &mut Request) -> IronResult<Response> {
        match self.routes.get(&req.url.path().join("/")) {
            Some(handler) => handler.handle(req),
            None => Ok(Response::with(status::NotFound))
        }
    }
}

В функции handle мы по переданному в запросе пути находим соответствующий обработчик в коллекции и вызываем обработчик этого пути с передачей в него запроса. В случае если переданный в запросе путь не "зарегистрирован" в коллекции — возвращается ответ с кодом ошибки NotFound.


Последнее, что осталось реализовать, — это инициализация нашего роутера и регистрация в нём необходимых нам путей с их обработчиками:


fn main() {
   let mut router = Router::new();

   router.add_route("hello_habrahabr".to_string(), |_: &mut Request| {
       Ok(Response::with((status::Ok, "Hello Loo Maclin!\n")))
   });

   router.add_route("hello_habrahabr/again".to_string(), |_: &mut Request| {
       Ok(Response::with((status::Ok, "Ты повторяешься!\n")))
   });

   router.add_route("error".to_string(), |_: &mut Request| {
       Ok(Response::with(status::BadRequest))
   });
...

Добавление новых путей происходит путём вызова реалиованной выше функции.
Инициализируем инстанс Iron с использованием нашего роутера:


Iron::new(router).http("localhost:3000").unwrap();

Тестируем:


[loomaclin@loomaclin ~]$ curl localhost:3000/hello_habrahabr
Hello Loo Maclin!
[loomaclin@loomaclin ~]$ curl localhost:3000/hello_habrahabr/again
Ты повторяешься!
[loomaclin@loomaclin ~]$ curl -v localhost:3000/error
*   Trying ::1...
* Connected to localhost (::1) port 3000 (#0)
> GET /error HTTP/1.1
> Host: localhost:3000
> User-Agent: curl/7.49.1
> Accept: */*
>
< HTTP/1.1 400 Bad Request
< Date: Wed, 13 Jul 2016 21:29:20 GMT
< Content-Length: 0
<
* Connection #0 to host localhost left intact

Заключение


На этом статья подходит к концу.
Помимо рассмотренного функционала, Iron выносит большую часть типичной для веб-фреймворка функциональности по базовым расширениям:



Статья предназначена для базового ознакомления с Iron, и хочется надеяться, что она справляется с этой целью.


Спасибо за внимание!

Поделиться с друзьями
-->

Комментарии (11)


  1. aka-demik
    15.07.2016 09:47

    А ничего не известно, когда планируется релиз hyper? Ведь текущий master уже использует mio + rotor.
    И интересно как это воспримет iron. У них в git-е не видно ничего про переход на новый API.


    1. LooMaclin
      15.07.2016 10:03
      +1

      Насколько мне известно был заведён запрос пользователем bfrog в репозитории Iron в котором и обсуждается переход на "mio-ветку" hyper. Судя по последним комментариям, будет выпущен ещё как минимум один релиз (финальный) с поддержкой синхронного hyper. После чего он возможно будет поддерживаться некоторое время, а все последующие — будут основаны на асинхронной ветке hyper.
      Вот ссылка на запрос: https://github.com/iron/iron/issues/446.
      Примечательно то, что Станислав Панферов создал похожий запрос в репозитории rustless (ещё одна библиотека базирующаяся на hyper, но нацеленная на построение REST API).


  1. tgz
    15.07.2016 11:40

    А мне вот как-то никель больше приглянулся. Хорошо когда есть из чего выбрать.


  1. kstep
    15.07.2016 12:43
    +3

    Всё хорошо, только вот в статье перидочески путаются типажи и типы.
    Request, Response, IronRequest, IronResult, IronError, Iron, Mime, Chain — это всё типы, а не типажи.
    Это очень важное отличие, типажи нельзя инстанциировать, они определяют только поведение, интерфейс.


    1. LooMaclin
      15.07.2016 13:10

      Вы правы, это типы, а не типажи. Спасибо за комментарий! Поправил.


  1. 4p4
    15.07.2016 12:52

    Получается, это что-то вроде node для v8 или rails для ruby?


    1. abasovar
      15.07.2016 13:17
      +1

      Скорее тогда как Express для Node


    1. LooMaclin
      15.07.2016 13:28
      +2

      Сравнение с NodeJS для V8 не совсем верное. NodeJS это всё таки программная платформа для запуска приложений написанных на javascript. А вот с Rails вы уже ближе, но он полноценный веб-фреймворк, использующий MVC в отличии от Iron. Вместе с расширениями возможно вы получите некое подобие "рельс", если захотите.


  1. moveax3
    15.07.2016 18:21

    а ORM имеется?


    1. LooMaclin
      15.07.2016 22:37
      +1

      ORM отсутствует, так как предназначение — работа с сетью. Но если вас интересует ORM имеет смысл посмотреть на Diesel. Вот ссылка на его сайт: http://diesel.rs/


      1. moveax3
        16.07.2016 08:25

        спасибо