/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ #![cfg(test)] #![allow(dead_code)] #[macro_use] extern crate lazy_static; mod cookie; mod cookie_http_state; mod data_loader; mod fetch; mod file_loader; mod filemanager_thread; mod hsts; mod http_cache; mod http_loader; mod mime_classifier; mod resource_thread; mod subresource_integrity; use crossbeam_channel::{unbounded, Sender}; use devtools_traits::DevtoolsControlMsg; use embedder_traits::resources::{self, Resource}; use embedder_traits::{EmbedderProxy, EventLoopWaker}; use futures::{Future, Stream}; use hyper::server::conn::Http; use hyper::server::Server as HyperServer; use hyper::service::service_fn_ok; use hyper::{Body, Request as HyperRequest, Response as HyperResponse}; use net::connector::{create_tls_config, ConnectionCerts, ExtraCerts, ALPN_H2_H1}; use net::fetch::cors_cache::CorsCache; use net::fetch::methods::{self, CancellationListener, FetchContext}; use net::filemanager_thread::FileManager; use net::resource_thread::CoreResourceThreadPool; use net::test::HttpState; use net_traits::filemanager_thread::FileTokenCheck; use net_traits::request::Request; use net_traits::response::Response; use net_traits::{FetchTaskTarget, ResourceFetchTiming, ResourceTimingType}; use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod}; use servo_arc::Arc as ServoArc; use servo_url::ServoUrl; use std::net::TcpListener as StdTcpListener; use std::path::PathBuf; use std::sync::{Arc, Mutex, Weak}; use tokio::net::TcpListener; use tokio::reactor::Handle; use tokio::runtime::Runtime; use tokio_openssl::SslAcceptorExt; lazy_static! { pub static ref HANDLE: Mutex = Mutex::new(Runtime::new().unwrap()); } const DEFAULT_USER_AGENT: &'static str = "Such Browser. Very Layout. Wow."; struct FetchResponseCollector { sender: Sender, } fn create_embedder_proxy() -> EmbedderProxy { let (sender, _) = unbounded(); let event_loop_waker = || { struct DummyEventLoopWaker {} impl DummyEventLoopWaker { fn new() -> DummyEventLoopWaker { DummyEventLoopWaker {} } } impl EventLoopWaker for DummyEventLoopWaker { fn wake(&self) {} fn clone_box(&self) -> Box { Box::new(DummyEventLoopWaker {}) } } Box::new(DummyEventLoopWaker::new()) }; EmbedderProxy { sender: sender, event_loop_waker: event_loop_waker(), } } fn new_fetch_context( dc: Option>, fc: Option, pool_handle: Option>, ) -> FetchContext { let certs = resources::read_string(Resource::SSLCertificates); let tls_config = create_tls_config( &certs, ALPN_H2_H1, ExtraCerts::new(), ConnectionCerts::new(), ); let sender = fc.unwrap_or_else(|| create_embedder_proxy()); FetchContext { state: Arc::new(HttpState::new(tls_config)), user_agent: DEFAULT_USER_AGENT.into(), devtools_chan: dc, filemanager: FileManager::new(sender, pool_handle.unwrap_or_else(|| Weak::new())), file_token: FileTokenCheck::NotRequired, cancellation_listener: Arc::new(Mutex::new(CancellationListener::new(None))), timing: ServoArc::new(Mutex::new(ResourceFetchTiming::new( ResourceTimingType::Navigation, ))), } } impl FetchTaskTarget for FetchResponseCollector { fn process_request_body(&mut self, _: &Request) {} fn process_request_eof(&mut self, _: &Request) {} fn process_response(&mut self, _: &Response) {} fn process_response_chunk(&mut self, _: Vec) {} /// Fired when the response is fully fetched fn process_response_eof(&mut self, response: &Response) { let _ = self.sender.send(response.clone()); } } fn fetch(request: &mut Request, dc: Option>) -> Response { fetch_with_context(request, &mut new_fetch_context(dc, None, None)) } fn fetch_with_context(request: &mut Request, mut context: &mut FetchContext) -> Response { let (sender, receiver) = unbounded(); let mut target = FetchResponseCollector { sender: sender }; methods::fetch(request, &mut target, &mut context); receiver.recv().unwrap() } fn fetch_with_cors_cache(request: &mut Request, cache: &mut CorsCache) -> Response { let (sender, receiver) = unbounded(); let mut target = FetchResponseCollector { sender: sender }; methods::fetch_with_cors_cache( request, cache, &mut target, &mut new_fetch_context(None, None, None), ); receiver.recv().unwrap() } pub(crate) struct Server { pub close_channel: futures::sync::oneshot::Sender<()>, } impl Server { fn close(self) { self.close_channel.send(()).unwrap(); } } fn make_server(handler: H) -> (Server, ServoUrl) where H: Fn(HyperRequest, &mut HyperResponse) + Send + Sync + 'static, { let handler = Arc::new(handler); let listener = StdTcpListener::bind("0.0.0.0:0").unwrap(); let url_string = format!("http://localhost:{}", listener.local_addr().unwrap().port()); let url = ServoUrl::parse(&url_string).unwrap(); let (tx, rx) = futures::sync::oneshot::channel::<()>(); let server = HyperServer::from_tcp(listener) .unwrap() .serve(move || { let handler = handler.clone(); service_fn_ok(move |req: HyperRequest| { let mut response = HyperResponse::new(Vec::::new().into()); handler(req, &mut response); response }) }) .with_graceful_shutdown(rx) .map_err(|_| ()); HANDLE.lock().unwrap().spawn(server); let server = Server { close_channel: tx }; (server, url) } fn make_ssl_server(handler: H, cert_path: PathBuf, key_path: PathBuf) -> (Server, ServoUrl) where H: Fn(HyperRequest, &mut HyperResponse) + Send + Sync + 'static, { let handler = Arc::new(handler); let listener = StdTcpListener::bind("[::0]:0").unwrap(); let listener = TcpListener::from_std(listener, &Handle::default()).unwrap(); let url_string = format!("http://localhost:{}", listener.local_addr().unwrap().port()); let url = ServoUrl::parse(&url_string).unwrap(); let server = listener.incoming().map_err(|_| ()).for_each(move |sock| { let mut tls_server_config = SslAcceptor::mozilla_intermediate_v5(SslMethod::tls()).unwrap(); tls_server_config .set_certificate_file(&cert_path, SslFiletype::PEM) .unwrap(); tls_server_config .set_private_key_file(&key_path, SslFiletype::PEM) .unwrap(); let handler = handler.clone(); tls_server_config .build() .accept_async(sock) .map_err(|_| ()) .and_then(move |ssl| { Http::new() .serve_connection( ssl, service_fn_ok(move |req: HyperRequest| { let mut response = HyperResponse::new(Vec::::new().into()); handler(req, &mut response); response }), ) .map_err(|_| ()) }) }); let (tx, rx) = futures::sync::oneshot::channel::<()>(); let server = server .select(rx.map_err(|_| ())) .map(|_| ()) .map_err(|_| ()); HANDLE.lock().unwrap().spawn(server); let server = Server { close_channel: tx }; (server, url) }