mirror of
https://github.com/actix/examples
synced 2024-11-23 22:41:07 +01:00
Add tiny_template as an example. (#358)
This commit is contained in:
parent
ec6e14aacc
commit
adccbe2a13
2
.gitignore
vendored
2
.gitignore
vendored
@ -12,7 +12,7 @@ Cargo.lock
|
||||
**/*.rs.bk
|
||||
|
||||
# intellij files
|
||||
.idea/**
|
||||
**/.idea/**
|
||||
|
||||
.history/
|
||||
|
||||
|
@ -41,6 +41,7 @@ members = [
|
||||
"template_askama",
|
||||
"template_handlebars",
|
||||
"template_tera",
|
||||
"template_tinytemplate",
|
||||
"template_yarte",
|
||||
"todo",
|
||||
"udp-echo",
|
||||
|
@ -5,7 +5,7 @@ use actix_web::{
|
||||
};
|
||||
use diesel::prelude::*;
|
||||
use diesel::PgConnection;
|
||||
use futures::future::{Ready, ok, err};
|
||||
use futures::future::{err, ok, Ready};
|
||||
use serde::Deserialize;
|
||||
|
||||
use crate::errors::ServiceError;
|
||||
|
15
template_tinytemplate/Cargo.toml
Normal file
15
template_tinytemplate/Cargo.toml
Normal file
@ -0,0 +1,15 @@
|
||||
[package]
|
||||
name = "template-tinytemplate"
|
||||
version = "2.0.0"
|
||||
authors = ["Hanif Bin Ariffin <hanif.ariffin.4326@gmail.com>"]
|
||||
edition = "2018"
|
||||
|
||||
[dependencies]
|
||||
env_logger = "0.7"
|
||||
tinytemplate = "1.1"
|
||||
serde = "1"
|
||||
serde_derive = "1"
|
||||
actix-http = "1"
|
||||
actix-web = "2"
|
||||
actix-rt = "1"
|
||||
serde_json = "1"
|
18
template_tinytemplate/README.md
Normal file
18
template_tinytemplate/README.md
Normal file
@ -0,0 +1,18 @@
|
||||
# template_tinytemplate
|
||||
|
||||
See the documentation for the minimalist template engine [tiny_template](https://docs.rs/tinytemplate/1.1.0/tinytemplate/).
|
||||
|
||||
## Usage
|
||||
|
||||
### Server
|
||||
|
||||
```bash
|
||||
cd examples/template_tinytemplate
|
||||
cargo run # (or ``cargo watch -x run``)
|
||||
# Started http server: 127.0.0.1:8080
|
||||
```
|
||||
|
||||
### Web Client
|
||||
|
||||
- [http://localhost:8080](http://localhost:8080)
|
||||
- [http://localhost:8080/non-existing-page](http://localhost:8080/non-existing-page) - 404 page rendered using template
|
101
template_tinytemplate/src/main.rs
Normal file
101
template_tinytemplate/src/main.rs
Normal file
@ -0,0 +1,101 @@
|
||||
use std::collections::HashMap;
|
||||
|
||||
use actix_http::{body::Body, Response};
|
||||
use actix_web::dev::ServiceResponse;
|
||||
use actix_web::http::StatusCode;
|
||||
use actix_web::middleware::errhandlers::{ErrorHandlerResponse, ErrorHandlers};
|
||||
use actix_web::{error, middleware, web, App, Error, HttpResponse, HttpServer, Result};
|
||||
use serde_json::json;
|
||||
use tinytemplate::TinyTemplate;
|
||||
|
||||
// store tiny_template in application state
|
||||
async fn index(
|
||||
tmpl: web::Data<TinyTemplate<'_>>,
|
||||
query: web::Query<HashMap<String, String>>,
|
||||
) -> Result<HttpResponse, Error> {
|
||||
let s = if let Some(name) = query.get("name") {
|
||||
// submitted form
|
||||
let ctx = json!({
|
||||
"name" : name.to_owned(),
|
||||
"text" : "Welcome!".to_owned()
|
||||
});
|
||||
tmpl.render("user.html", &ctx)
|
||||
.map_err(|_| error::ErrorInternalServerError("Template error"))?
|
||||
} else {
|
||||
tmpl.render("index.html", &serde_json::Value::Null)
|
||||
.map_err(|_| error::ErrorInternalServerError("Template error"))?
|
||||
};
|
||||
Ok(HttpResponse::Ok().content_type("text/html").body(s))
|
||||
}
|
||||
|
||||
#[actix_rt::main]
|
||||
async fn main() -> std::io::Result<()> {
|
||||
std::env::set_var("RUST_LOG", "actix_web=info");
|
||||
env_logger::init();
|
||||
|
||||
HttpServer::new(|| {
|
||||
let mut tt = TinyTemplate::new();
|
||||
tt.add_template("index.html", INDEX).unwrap();
|
||||
tt.add_template("user.html", USER).unwrap();
|
||||
tt.add_template("error.html", ERROR).unwrap();
|
||||
|
||||
App::new()
|
||||
.data(tt)
|
||||
.wrap(middleware::Logger::default()) // enable logger
|
||||
.service(web::resource("/").route(web::get().to(index)))
|
||||
.service(web::scope("").wrap(error_handlers()))
|
||||
})
|
||||
.bind("127.0.0.1:8080")?
|
||||
.run()
|
||||
.await
|
||||
}
|
||||
|
||||
// Custom error handlers, to return HTML responses when an error occurs.
|
||||
fn error_handlers() -> ErrorHandlers<Body> {
|
||||
ErrorHandlers::new().handler(StatusCode::NOT_FOUND, not_found)
|
||||
}
|
||||
|
||||
// Error handler for a 404 Page not found error.
|
||||
fn not_found<B>(res: ServiceResponse<B>) -> Result<ErrorHandlerResponse<B>> {
|
||||
let response = get_error_response(&res, "Page not found");
|
||||
Ok(ErrorHandlerResponse::Response(
|
||||
res.into_response(response.into_body()),
|
||||
))
|
||||
}
|
||||
|
||||
// Generic error handler.
|
||||
fn get_error_response<B>(res: &ServiceResponse<B>, error: &str) -> Response<Body> {
|
||||
let request = res.request();
|
||||
|
||||
// Provide a fallback to a simple plain text response in case an error occurs during the
|
||||
// rendering of the error page.
|
||||
let fallback = |e: &str| {
|
||||
Response::build(res.status())
|
||||
.content_type("text/plain")
|
||||
.body(e.to_string())
|
||||
};
|
||||
|
||||
let tt = request
|
||||
.app_data::<web::Data<TinyTemplate<'_>>>()
|
||||
.map(|t| t.get_ref());
|
||||
match tt {
|
||||
Some(tt) => {
|
||||
let mut context = std::collections::HashMap::new();
|
||||
context.insert("error", error.to_owned());
|
||||
context.insert("status_code", res.status().as_str().to_owned());
|
||||
let body = tt.render("error.html", &context);
|
||||
|
||||
match body {
|
||||
Ok(body) => Response::build(res.status())
|
||||
.content_type("text/html")
|
||||
.body(body),
|
||||
Err(_) => fallback(error),
|
||||
}
|
||||
}
|
||||
None => fallback(error),
|
||||
}
|
||||
}
|
||||
|
||||
static ERROR: &str = include_str!("../templates/error.html");
|
||||
static INDEX: &str = include_str!("../templates/index.html");
|
||||
static USER: &str = include_str!("../templates/user.html");
|
10
template_tinytemplate/templates/error.html
Normal file
10
template_tinytemplate/templates/error.html
Normal file
@ -0,0 +1,10 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<title>{error}</title>
|
||||
</head>
|
||||
<body>
|
||||
<h1>{status_code} {error}</h1>
|
||||
</body>
|
||||
</html>
|
17
template_tinytemplate/templates/index.html
Normal file
17
template_tinytemplate/templates/index.html
Normal file
@ -0,0 +1,17 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<title>Actix web</title>
|
||||
</head>
|
||||
<body>
|
||||
<h1>Welcome!</h1>
|
||||
<p>
|
||||
<h3>What is your name?</h3>
|
||||
<form>
|
||||
<input type="text" name="name" /><br/>
|
||||
<p><input type="submit"></p>
|
||||
</form>
|
||||
</p>
|
||||
</body>
|
||||
</html>
|
11
template_tinytemplate/templates/user.html
Normal file
11
template_tinytemplate/templates/user.html
Normal file
@ -0,0 +1,11 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8" />
|
||||
<title>Actix web</title>
|
||||
</head>
|
||||
<body>
|
||||
<h1>Hi, {name}!</h1>
|
||||
<p>{text}</p>
|
||||
</body>
|
||||
</html>
|
@ -43,14 +43,14 @@ async fn main() {
|
||||
);
|
||||
let (sink, stream) = UdpFramed::new(sock, BytesCodec::new()).split();
|
||||
UdpActor::create(|ctx| {
|
||||
ctx.add_stream(
|
||||
stream.filter_map(
|
||||
|item: Result<(BytesMut, SocketAddr)>| async {
|
||||
item.map(|(data, sender)| UdpPacket(data, sender)).ok()
|
||||
},
|
||||
),
|
||||
);
|
||||
UdpActor { sink: SinkWrite::new(sink, ctx), }
|
||||
ctx.add_stream(stream.filter_map(
|
||||
|item: Result<(BytesMut, SocketAddr)>| async {
|
||||
item.map(|(data, sender)| UdpPacket(data, sender)).ok()
|
||||
},
|
||||
));
|
||||
UdpActor {
|
||||
sink: SinkWrite::new(sink, ctx),
|
||||
}
|
||||
});
|
||||
|
||||
actix_rt::Arbiter::local_join().await;
|
||||
|
Loading…
Reference in New Issue
Block a user