2016-04-03 20:43:35 +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/>.
|
|
|
|
|
2016-04-03 22:39:49 +02:00
|
|
|
//! IPC over nanomsg transport
|
2016-04-03 20:43:35 +02:00
|
|
|
|
|
|
|
extern crate ethcore_ipc as ipc;
|
|
|
|
extern crate nanomsg;
|
2016-04-03 23:00:57 +02:00
|
|
|
#[macro_use] extern crate log;
|
2016-04-03 22:39:49 +02:00
|
|
|
|
|
|
|
pub use ipc::*;
|
|
|
|
|
|
|
|
use std::sync::*;
|
2016-04-03 22:58:18 +02:00
|
|
|
use std::io::Write;
|
2016-04-03 22:39:49 +02:00
|
|
|
use nanomsg::{Socket, Protocol};
|
|
|
|
|
|
|
|
pub struct Worker<S> where S: IpcInterface<S> {
|
|
|
|
service: Arc<S>,
|
|
|
|
sockets: Vec<Socket>,
|
2016-04-03 22:58:18 +02:00
|
|
|
method_buf: [u8;2],
|
2016-04-03 22:39:49 +02:00
|
|
|
}
|
|
|
|
|
2016-04-03 23:54:30 +02:00
|
|
|
#[derive(Debug)]
|
2016-04-03 23:33:30 +02:00
|
|
|
pub enum SocketError {
|
|
|
|
DuplexLink
|
|
|
|
}
|
|
|
|
|
2016-04-03 22:39:49 +02:00
|
|
|
impl<S> Worker<S> where S: IpcInterface<S> {
|
2016-04-03 22:58:18 +02:00
|
|
|
pub fn new(service: Arc<S>) -> Worker<S> {
|
2016-04-03 22:39:49 +02:00
|
|
|
Worker::<S> {
|
|
|
|
service: service.clone(),
|
|
|
|
sockets: Vec::new(),
|
2016-04-03 22:58:18 +02:00
|
|
|
method_buf: [0,0]
|
2016-04-03 22:39:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-03 22:58:18 +02:00
|
|
|
pub fn poll(&mut self) {
|
|
|
|
for socket in self.sockets.iter_mut() {
|
2016-04-03 23:33:30 +02:00
|
|
|
// non-blocking read only ok if there is something to read from socket
|
|
|
|
if let Ok(method_sign_len) = socket.nb_read(&mut self.method_buf) {
|
|
|
|
if method_sign_len == 2 {
|
2016-04-03 22:58:18 +02:00
|
|
|
let result = self.service.dispatch_buf(
|
|
|
|
self.method_buf[1] as u16 * 256 + self.method_buf[0] as u16,
|
|
|
|
socket);
|
2016-04-03 23:00:57 +02:00
|
|
|
if let Err(e) = socket.write(&result) {
|
|
|
|
warn!(target: "ipc", "Failed to write response: {:?}", e);
|
|
|
|
}
|
2016-04-03 22:58:18 +02:00
|
|
|
}
|
2016-04-03 23:33:30 +02:00
|
|
|
else {
|
|
|
|
warn!(target: "ipc", "Failed to read method signature from socket: unexpected message length({})", method_sign_len);
|
|
|
|
}
|
2016-04-03 22:58:18 +02:00
|
|
|
}
|
|
|
|
}
|
2016-04-03 22:39:49 +02:00
|
|
|
}
|
2016-04-03 23:33:30 +02:00
|
|
|
|
|
|
|
pub fn add_duplex(&mut self, addr: &str) -> Result<(), SocketError> {
|
|
|
|
let mut socket = try!(Socket::new(Protocol::Pair).map_err(|e| {
|
|
|
|
warn!(target: "ipc", "Failed to create ipc socket: {:?}", e);
|
|
|
|
SocketError::DuplexLink
|
|
|
|
}));
|
|
|
|
|
|
|
|
try!(socket.bind(addr).map_err(|e| {
|
|
|
|
warn!(target: "ipc", "Failed to bind socket to address '{}': {:?}", addr, e);
|
|
|
|
SocketError::DuplexLink
|
|
|
|
}));
|
|
|
|
|
|
|
|
self.sockets.push(socket);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::Worker;
|
|
|
|
use ipc::*;
|
|
|
|
use std::io::Read;
|
2016-04-03 23:54:30 +02:00
|
|
|
use std::sync::{Arc, RwLock};
|
2016-04-03 23:33:30 +02:00
|
|
|
|
2016-04-03 23:54:30 +02:00
|
|
|
struct TestInvoke {
|
|
|
|
method_num: u16,
|
|
|
|
params: Vec<u8>,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct DummyService {
|
|
|
|
methods_stack: RwLock<Vec<TestInvoke>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DummyService {
|
|
|
|
fn new() -> DummyService {
|
|
|
|
DummyService { methods_stack: RwLock::new(Vec::new()) }
|
|
|
|
}
|
|
|
|
}
|
2016-04-03 23:33:30 +02:00
|
|
|
|
|
|
|
impl IpcInterface<DummyService> for DummyService {
|
2016-04-03 23:54:30 +02:00
|
|
|
fn dispatch<R>(&self, _r: &mut R) -> Vec<u8> where R: Read {
|
2016-04-03 23:33:30 +02:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
fn dispatch_buf<R>(&self, method_num: u16, r: &mut R) -> Vec<u8> where R: Read {
|
2016-04-03 23:54:30 +02:00
|
|
|
let mut buf = vec![0u8; 4096];
|
|
|
|
let size = r.read_to_end(&mut buf).unwrap();
|
|
|
|
self.methods_stack.write().unwrap().push(
|
|
|
|
TestInvoke {
|
|
|
|
method_num: method_num,
|
|
|
|
params: unsafe { Vec::from_raw_parts(buf.as_mut_ptr(), size, size) }
|
|
|
|
});
|
2016-04-03 23:33:30 +02:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-03 23:42:00 +02:00
|
|
|
#[test]
|
2016-04-03 23:33:30 +02:00
|
|
|
fn can_create_worker() {
|
2016-04-03 23:54:30 +02:00
|
|
|
let worker = Worker::<DummyService>::new(Arc::new(DummyService::new()));
|
2016-04-03 23:33:30 +02:00
|
|
|
assert_eq!(0, worker.sockets.len());
|
|
|
|
}
|
2016-04-03 23:54:30 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_add_duplex_socket_to_worker() {
|
|
|
|
let mut worker = Worker::<DummyService>::new(Arc::new(DummyService::new()));
|
|
|
|
worker.add_duplex("ipc://tmp/parity/test1").unwrap();
|
|
|
|
assert_eq!(1, worker.sockets.len());
|
|
|
|
}
|
2016-04-03 22:39:49 +02:00
|
|
|
}
|