The main changes are: * The entry point is how `Http`, the implementation of `ServerProto`. This type has a `new` constructor as well as builder methods to configure it. * A high-level entry point of `Http::bind` was added which returns a `Server`. Binding a protocol to a port requires a socket address (where to bind) as well as the instance of `NewService`. Internally this creates a core and a TCP listener. * The returned `Server` has a few methods to learn about itself, e.g. `local_addr` and `handle`, but mainly has two methods: `run` and `run_until`. * The `Server::run` entry point will execute a server infinitely, never having it exit. * The `Server::run_until` method is intended as a graceful shutdown mechanism. When the provided future resolves the server stops accepting connections immediately and then waits for a fixed period of time for all active connections to get torn down, after which the whole server is torn down anyway. * Finally a `Http::bind_connection` method exists as a low-level entry point to spawning a server connection. This is used by `Server::run` as is intended for external use in other event loops if necessary or otherwise low-level needs. BREAKING CHANGE: `Server` is no longer the pimary entry point. Instead, an `Http` type is created and then either `bind` to receiver a `Server`, or it can be passed to other Tokio things.
		
			
				
	
	
		
			55 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			55 lines
		
	
	
		
			1.5 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| //#![deny(warnings)]
 | |
| extern crate futures;
 | |
| extern crate hyper;
 | |
| extern crate pretty_env_logger;
 | |
| #[macro_use]
 | |
| extern crate log;
 | |
| 
 | |
| use hyper::{Get, Post, StatusCode};
 | |
| use hyper::header::ContentLength;
 | |
| use hyper::server::{Http, Service, Request, Response};
 | |
| 
 | |
| static INDEX: &'static [u8] = b"Try POST /echo";
 | |
| 
 | |
| #[derive(Clone, Copy)]
 | |
| struct Echo;
 | |
| 
 | |
| impl Service for Echo {
 | |
|     type Request = Request;
 | |
|     type Response = Response;
 | |
|     type Error = hyper::Error;
 | |
|     type Future = ::futures::Finished<Response, hyper::Error>;
 | |
| 
 | |
|     fn call(&self, req: Request) -> Self::Future {
 | |
|         ::futures::finished(match (req.method(), req.path()) {
 | |
|             (&Get, "/") | (&Get, "/echo") => {
 | |
|                 Response::new()
 | |
|                     .with_header(ContentLength(INDEX.len() as u64))
 | |
|                     .with_body(INDEX)
 | |
|             },
 | |
|             (&Post, "/echo") => {
 | |
|                 let mut res = Response::new();
 | |
|                 if let Some(len) = req.headers().get::<ContentLength>() {
 | |
|                     res.headers_mut().set(len.clone());
 | |
|                 }
 | |
|                 res.with_body(req.body())
 | |
|             },
 | |
|             _ => {
 | |
|                 Response::new()
 | |
|                     .with_status(StatusCode::NotFound)
 | |
|             }
 | |
|         })
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | |
| fn main() {
 | |
|     pretty_env_logger::init().unwrap();
 | |
|     let addr = "127.0.0.1:1337".parse().unwrap();
 | |
| 
 | |
|     let server = Http::new().bind(&addr, || Ok(Echo)).unwrap();
 | |
|     println!("Listening on http://{}", server.local_addr().unwrap());
 | |
|     server.run().unwrap();
 | |
| }
 |