about summary refs log tree commit diff
path: root/src/main.rs
blob: 559db8cc9662e4006dc57461d430160b2e4092ca (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
#[macro_use]
extern crate diesel;

#[macro_use]
extern crate log;

extern crate chrono;
extern crate actix;
extern crate actix_web;
extern crate env_logger;
extern crate r2d2;
extern crate futures;

pub mod schema;
pub mod models;
pub mod db;

use actix::prelude::*;
use actix_web::*;

use diesel::pg::PgConnection;
use diesel::r2d2::{ConnectionManager, Pool};
use std::env;
use db::*;
use futures::Future;
use models::Thread;

/// Represents the state carried by the web server actors.
struct AppState {
    db: Addr<Syn, DbExecutor>,
}

/// Really inefficient renderer example!
fn render_threads(threads: Vec<Thread>) -> String {
    let mut res = String::new();

    for thread in threads {
        res.push_str(&format!("Subject: {}\n", thread.title));
        res.push_str(&format!("Posted at: {}\n\n", thread.posted));
        res.push_str(&format!("{}\n", thread.body));
        res.push_str("-------------------------------");
    }

    res
}

fn forum_index(req: HttpRequest<AppState>) -> FutureResponse<HttpResponse> {
    req.state().db.send(ListThreads)
        .from_err()
        .and_then(|res| match res {
            Ok(threads) => Ok(HttpResponse::from(render_threads(threads))),
            Err(err) => {
                error!("Error loading threads: {}", err);
                Ok(HttpResponse::InternalServerError().into())
            }
        })
        .responder()
}

fn main() {
    env_logger::init();

    info!("Welcome to Converse! Hold on tight while we're getting ready.");
    let sys = actix::System::new("converse");

    info!("Initialising database connection pool ...");
    let db_url = env::var("DATABASE_URL")
        .expect("DATABASE_URL must be set");

    let manager = ConnectionManager::<PgConnection>::new(db_url);
    let pool = Pool::builder().build(manager).expect("Failed to initialise DB pool");

    let db_addr = SyncArbiter::start(2, move || DbExecutor(pool.clone()));

    info!("Initialising HTTP server ...");
    server::new(move || {
        App::with_state(AppState { db: db_addr.clone() })
            .middleware(middleware::Logger::default())
            .route("/", http::Method::GET, &forum_index)
    }).bind("127.0.0.1:4567").unwrap().start();

    let _ = sys.run();
}