2016-04-13 18:03:57 +02:00
|
|
|
// Copyright 2015, 2016 Ethcore (UK) Ltd.
|
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
//! Parity interprocess hypervisor module
|
|
|
|
|
2016-04-21 16:32:44 +02:00
|
|
|
#![cfg_attr(feature="dev", allow(used_underscore_binding))]
|
2016-04-14 19:56:06 +02:00
|
|
|
|
2016-07-09 17:18:34 +02:00
|
|
|
extern crate ethcore_ipc as ipc;
|
|
|
|
extern crate ethcore_ipc_nano as nanoipc;
|
|
|
|
extern crate semver;
|
|
|
|
#[macro_use] extern crate log;
|
2016-04-14 20:45:53 +02:00
|
|
|
|
2016-07-07 09:37:31 +02:00
|
|
|
pub mod service;
|
2016-04-14 20:45:53 +02:00
|
|
|
|
2016-07-07 09:37:31 +02:00
|
|
|
/// Default value for hypervisor ipc listener
|
|
|
|
pub const HYPERVISOR_IPC_URL: &'static str = "ipc:///tmp/parity-internal-hyper-status.ipc";
|
|
|
|
|
2016-07-09 17:18:34 +02:00
|
|
|
use std::sync::{Arc,RwLock};
|
|
|
|
use service::{HypervisorService, IpcModuleId};
|
|
|
|
use std::process::{Command,Child};
|
|
|
|
use std::collections::HashMap;
|
|
|
|
|
2016-07-15 15:32:29 +02:00
|
|
|
pub use service::{HypervisorServiceClient, CLIENT_MODULE_ID, SYNC_MODULE_ID};
|
2016-07-09 17:18:34 +02:00
|
|
|
|
2016-07-15 19:50:17 +02:00
|
|
|
pub type BinaryId = &'static str;
|
2016-04-13 18:03:57 +02:00
|
|
|
|
|
|
|
pub struct Hypervisor {
|
2016-04-14 20:45:53 +02:00
|
|
|
ipc_addr: String,
|
2016-04-14 17:22:31 +02:00
|
|
|
service: Arc<HypervisorService>,
|
2016-04-13 18:03:57 +02:00
|
|
|
ipc_worker: RwLock<nanoipc::Worker<HypervisorService>>,
|
2016-04-14 20:45:53 +02:00
|
|
|
processes: RwLock<HashMap<BinaryId, Child>>,
|
2016-07-20 18:13:56 +02:00
|
|
|
modules: HashMap<IpcModuleId, (BinaryId, BootArgs)>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Boot arguments for binary
|
|
|
|
pub struct BootArgs {
|
|
|
|
cli: Option<Vec<String>>,
|
|
|
|
stdin: Option<Vec<u8>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BootArgs {
|
|
|
|
/// New empty boot arguments
|
|
|
|
pub fn new() -> BootArgs {
|
|
|
|
BootArgs {
|
|
|
|
cli: None,
|
|
|
|
stdin: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set command-line arguments for boot
|
|
|
|
pub fn cli(mut self, cli: Vec<String>) -> BootArgs {
|
|
|
|
self.cli = Some(cli);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set std-in stream for boot
|
|
|
|
pub fn stdin(mut self, stdin: Vec<u8>) -> BootArgs {
|
|
|
|
self.stdin = Some(stdin);
|
|
|
|
self
|
|
|
|
}
|
2016-04-21 15:56:35 +02:00
|
|
|
}
|
|
|
|
|
2016-04-13 18:03:57 +02:00
|
|
|
impl Hypervisor {
|
|
|
|
/// initializes the Hypervisor service with the open ipc socket for incoming clients
|
2016-07-15 19:50:17 +02:00
|
|
|
pub fn new() -> Hypervisor {
|
|
|
|
Hypervisor::with_url(HYPERVISOR_IPC_URL)
|
|
|
|
}
|
|
|
|
|
2016-07-20 18:13:56 +02:00
|
|
|
pub fn module(mut self, module_id: IpcModuleId, binary_id: BinaryId, args: BootArgs) -> Hypervisor {
|
2016-07-15 19:50:17 +02:00
|
|
|
self.modules.insert(module_id, (binary_id, args));
|
|
|
|
self.service.add_module(module_id);
|
|
|
|
self
|
2016-04-14 17:22:31 +02:00
|
|
|
}
|
|
|
|
|
2016-07-15 19:50:17 +02:00
|
|
|
pub fn local_module(self, module_id: IpcModuleId) -> Hypervisor {
|
|
|
|
self.service.add_module(module_id);
|
|
|
|
self
|
2016-04-14 17:49:25 +02:00
|
|
|
}
|
|
|
|
|
2016-04-14 20:50:35 +02:00
|
|
|
/// Starts with the specified address for the ipc listener and
|
|
|
|
/// the specified list of modules in form of created service
|
2016-07-15 19:50:17 +02:00
|
|
|
pub fn with_url(addr: &str) -> Hypervisor {
|
|
|
|
let service = HypervisorService::new();
|
2016-04-14 20:45:53 +02:00
|
|
|
let worker = nanoipc::Worker::new(&service);
|
|
|
|
Hypervisor{
|
|
|
|
ipc_addr: addr.to_owned(),
|
2016-04-14 17:22:31 +02:00
|
|
|
service: service,
|
|
|
|
ipc_worker: RwLock::new(worker),
|
2016-04-14 20:45:53 +02:00
|
|
|
processes: RwLock::new(HashMap::new()),
|
2016-07-15 19:50:17 +02:00
|
|
|
modules: HashMap::new(),
|
2016-04-14 20:45:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Since one binary can host multiple modules
|
|
|
|
/// we match binaries
|
2016-07-20 18:13:56 +02:00
|
|
|
fn match_module(&self, module_id: &IpcModuleId) -> Option<&(BinaryId, BootArgs)> {
|
2016-07-15 19:50:17 +02:00
|
|
|
self.modules.get(module_id)
|
2016-04-14 20:45:53 +02:00
|
|
|
}
|
|
|
|
|
2016-04-14 20:50:35 +02:00
|
|
|
/// Creates IPC listener and starts all binaries
|
2016-07-09 17:18:34 +02:00
|
|
|
pub fn start(&self) {
|
2016-04-14 20:45:53 +02:00
|
|
|
let mut worker = self.ipc_worker.write().unwrap();
|
|
|
|
worker.add_reqrep(&self.ipc_addr).unwrap_or_else(|e| panic!("Hypervisor ipc worker can not start - critical! ({:?})", e));
|
|
|
|
|
|
|
|
for module_id in self.service.module_ids() {
|
|
|
|
self.start_module(module_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-14 20:50:35 +02:00
|
|
|
/// Start binary for the specified module
|
|
|
|
/// Does nothing when it is already started on module is inside the
|
|
|
|
/// main binary
|
2016-04-14 20:45:53 +02:00
|
|
|
fn start_module(&self, module_id: IpcModuleId) {
|
2016-07-20 18:13:56 +02:00
|
|
|
use std::io::Write;
|
|
|
|
|
2016-07-15 19:50:17 +02:00
|
|
|
self.match_module(&module_id).map(|&(ref binary_id, ref binary_args)| {
|
2016-04-15 07:13:38 +02:00
|
|
|
let mut processes = self.processes.write().unwrap();
|
|
|
|
{
|
|
|
|
if processes.get(binary_id).is_some() {
|
|
|
|
// already started for another module
|
|
|
|
return;
|
2016-04-14 20:45:53 +02:00
|
|
|
}
|
2016-04-15 07:13:38 +02:00
|
|
|
}
|
2016-07-09 17:18:34 +02:00
|
|
|
|
|
|
|
let mut executable_path = std::env::current_exe().unwrap();
|
|
|
|
executable_path.pop();
|
|
|
|
executable_path.push(binary_id);
|
|
|
|
|
2016-07-20 18:13:56 +02:00
|
|
|
let executable_path = executable_path.to_str().unwrap();
|
|
|
|
let mut command = Command::new(&executable_path);
|
|
|
|
command.stderr(std::process::Stdio::inherit());
|
|
|
|
|
|
|
|
if let Some(ref cli_args) = binary_args.cli {
|
|
|
|
for arg in cli_args { command.arg(arg); }
|
|
|
|
}
|
|
|
|
|
|
|
|
command.stdin(std::process::Stdio::piped());
|
2016-07-15 19:50:17 +02:00
|
|
|
|
|
|
|
trace!(target: "hypervisor", "Spawn executable: {:?}", command);
|
|
|
|
|
2016-07-20 18:13:56 +02:00
|
|
|
let mut child = command.spawn().unwrap_or_else(
|
2016-07-09 17:18:34 +02:00
|
|
|
|e| panic!("Hypervisor cannot start binary ({:?}): {}", executable_path, e));
|
2016-07-20 18:13:56 +02:00
|
|
|
|
|
|
|
if let Some(ref std_in) = binary_args.stdin {
|
|
|
|
trace!(target: "hypervisor", "Pushing std-in payload...");
|
|
|
|
child.stdin.as_mut()
|
|
|
|
.expect("std-in should be piped above")
|
|
|
|
.write(std_in)
|
|
|
|
.unwrap_or_else(|e| panic!(format!("Error trying to pipe stdin for {}: {:?}", &executable_path, e)));
|
|
|
|
drop(child.stdin.take());
|
|
|
|
}
|
|
|
|
|
2016-04-15 07:13:38 +02:00
|
|
|
processes.insert(binary_id, child);
|
|
|
|
});
|
2016-04-13 18:03:57 +02:00
|
|
|
}
|
|
|
|
|
2016-04-14 20:50:35 +02:00
|
|
|
/// Reports if all modules are checked in
|
2016-04-14 17:22:31 +02:00
|
|
|
pub fn modules_ready(&self) -> bool {
|
|
|
|
self.service.unchecked_count() == 0
|
|
|
|
}
|
|
|
|
|
2016-04-13 18:03:57 +02:00
|
|
|
/// Waits for every required module to check in
|
2016-04-14 17:22:31 +02:00
|
|
|
pub fn wait_for_startup(&self) {
|
|
|
|
let mut worker = self.ipc_worker.write().unwrap();
|
|
|
|
while !self.modules_ready() {
|
|
|
|
worker.poll()
|
|
|
|
}
|
|
|
|
}
|
2016-07-09 17:18:34 +02:00
|
|
|
|
2016-07-20 18:13:56 +02:00
|
|
|
/// Shutdown the ipc and all managed child processes
|
2016-07-09 17:18:34 +02:00
|
|
|
pub fn shutdown(&self, wait_time: Option<std::time::Duration>) {
|
|
|
|
if wait_time.is_some() { std::thread::sleep(wait_time.unwrap()) }
|
|
|
|
|
|
|
|
let mut childs = self.processes.write().unwrap();
|
|
|
|
for (ref mut binary, ref mut child) in childs.iter_mut() {
|
2016-07-15 19:50:17 +02:00
|
|
|
trace!(target: "hypervisor", "Stopping process module: {}", binary);
|
2016-07-09 17:18:34 +02:00
|
|
|
child.kill().unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Hypervisor {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.shutdown(Some(std::time::Duration::new(1, 0)));
|
|
|
|
}
|
2016-04-14 17:22:31 +02:00
|
|
|
}
|
|
|
|
|
2016-04-21 15:56:35 +02:00
|
|
|
#[cfg(test)]
|
2016-04-14 17:22:31 +02:00
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2016-04-14 17:49:25 +02:00
|
|
|
use std::sync::atomic::{AtomicBool,Ordering};
|
|
|
|
use std::sync::Arc;
|
|
|
|
use nanoipc;
|
2016-04-14 17:22:31 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_init() {
|
2016-04-14 19:43:14 +02:00
|
|
|
let url = "ipc:///tmp/test-parity-hypervisor-10.ipc";
|
2016-04-14 19:56:06 +02:00
|
|
|
let test_module_id = 8080u64;
|
2016-04-14 17:49:25 +02:00
|
|
|
|
2016-07-15 19:50:17 +02:00
|
|
|
let hypervisor = Hypervisor::with_url(url).local_module(test_module_id);
|
2016-04-14 17:22:31 +02:00
|
|
|
assert_eq!(false, hypervisor.modules_ready());
|
|
|
|
}
|
2016-04-13 18:03:57 +02:00
|
|
|
|
2016-04-14 17:22:31 +02:00
|
|
|
#[test]
|
|
|
|
fn can_wait_for_startup() {
|
2016-04-14 19:43:14 +02:00
|
|
|
let url = "ipc:///tmp/test-parity-hypervisor-20.ipc";
|
2016-04-14 17:49:25 +02:00
|
|
|
let test_module_id = 8080u64;
|
|
|
|
|
|
|
|
let hypervisor_ready = Arc::new(AtomicBool::new(false));
|
|
|
|
let hypervisor_ready_local = hypervisor_ready.clone();
|
|
|
|
|
|
|
|
::std::thread::spawn(move || {
|
|
|
|
while !hypervisor_ready.load(Ordering::Relaxed) { }
|
|
|
|
|
|
|
|
let client = nanoipc::init_client::<HypervisorServiceClient<_>>(url).unwrap();
|
2016-04-14 19:43:14 +02:00
|
|
|
client.handshake().unwrap();
|
2016-04-14 17:49:25 +02:00
|
|
|
client.module_ready(test_module_id);
|
|
|
|
});
|
|
|
|
|
2016-07-15 19:50:17 +02:00
|
|
|
let hypervisor = Hypervisor::with_url(url).local_module(test_module_id);
|
2016-04-14 20:45:53 +02:00
|
|
|
hypervisor.start();
|
2016-04-14 17:49:25 +02:00
|
|
|
hypervisor_ready_local.store(true, Ordering::Relaxed);
|
2016-04-14 17:22:31 +02:00
|
|
|
hypervisor.wait_for_startup();
|
2016-04-14 17:49:25 +02:00
|
|
|
|
|
|
|
assert_eq!(true, hypervisor.modules_ready());
|
2016-04-13 18:03:57 +02:00
|
|
|
}
|
|
|
|
}
|