mirror of
https://github.com/fafhrd91/actix-net
synced 2025-03-20 16:05:18 +01:00
Separate WorkerHandle to two parts (#323)
This commit is contained in:
parent
859f45868d
commit
0a11cf5cba
@ -12,7 +12,7 @@ use slab::Slab;
|
|||||||
use crate::server::Server;
|
use crate::server::Server;
|
||||||
use crate::socket::{MioListener, SocketAddr};
|
use crate::socket::{MioListener, SocketAddr};
|
||||||
use crate::waker_queue::{WakerInterest, WakerQueue, WAKER_TOKEN};
|
use crate::waker_queue::{WakerInterest, WakerQueue, WAKER_TOKEN};
|
||||||
use crate::worker::{Conn, WorkerHandle};
|
use crate::worker::{Conn, WorkerHandleAccept};
|
||||||
use crate::Token;
|
use crate::Token;
|
||||||
|
|
||||||
struct ServerSocketInfo {
|
struct ServerSocketInfo {
|
||||||
@ -66,7 +66,7 @@ impl AcceptLoop {
|
|||||||
pub(crate) fn start(
|
pub(crate) fn start(
|
||||||
&mut self,
|
&mut self,
|
||||||
socks: Vec<(Token, MioListener)>,
|
socks: Vec<(Token, MioListener)>,
|
||||||
handles: Vec<WorkerHandle>,
|
handles: Vec<WorkerHandleAccept>,
|
||||||
) {
|
) {
|
||||||
let srv = self.srv.take().expect("Can not re-use AcceptInfo");
|
let srv = self.srv.take().expect("Can not re-use AcceptInfo");
|
||||||
let poll = self.poll.take().unwrap();
|
let poll = self.poll.take().unwrap();
|
||||||
@ -80,7 +80,7 @@ impl AcceptLoop {
|
|||||||
struct Accept {
|
struct Accept {
|
||||||
poll: Poll,
|
poll: Poll,
|
||||||
waker: WakerQueue,
|
waker: WakerQueue,
|
||||||
handles: Vec<WorkerHandle>,
|
handles: Vec<WorkerHandleAccept>,
|
||||||
srv: Server,
|
srv: Server,
|
||||||
next: usize,
|
next: usize,
|
||||||
backpressure: bool,
|
backpressure: bool,
|
||||||
@ -105,7 +105,7 @@ impl Accept {
|
|||||||
waker: WakerQueue,
|
waker: WakerQueue,
|
||||||
socks: Vec<(Token, MioListener)>,
|
socks: Vec<(Token, MioListener)>,
|
||||||
srv: Server,
|
srv: Server,
|
||||||
handles: Vec<WorkerHandle>,
|
handles: Vec<WorkerHandleAccept>,
|
||||||
) {
|
) {
|
||||||
// Accept runs in its own thread and would want to spawn additional futures to current
|
// Accept runs in its own thread and would want to spawn additional futures to current
|
||||||
// actix system.
|
// actix system.
|
||||||
@ -125,7 +125,7 @@ impl Accept {
|
|||||||
poll: Poll,
|
poll: Poll,
|
||||||
waker: WakerQueue,
|
waker: WakerQueue,
|
||||||
socks: Vec<(Token, MioListener)>,
|
socks: Vec<(Token, MioListener)>,
|
||||||
handles: Vec<WorkerHandle>,
|
handles: Vec<WorkerHandleAccept>,
|
||||||
srv: Server,
|
srv: Server,
|
||||||
) -> (Accept, Slab<ServerSocketInfo>) {
|
) -> (Accept, Slab<ServerSocketInfo>) {
|
||||||
let mut sockets = Slab::new();
|
let mut sockets = Slab::new();
|
||||||
|
@ -19,7 +19,10 @@ use crate::signals::{Signal, Signals};
|
|||||||
use crate::socket::{MioListener, StdSocketAddr, StdTcpListener, ToSocketAddrs};
|
use crate::socket::{MioListener, StdSocketAddr, StdTcpListener, ToSocketAddrs};
|
||||||
use crate::socket::{MioTcpListener, MioTcpSocket};
|
use crate::socket::{MioTcpListener, MioTcpSocket};
|
||||||
use crate::waker_queue::{WakerInterest, WakerQueue};
|
use crate::waker_queue::{WakerInterest, WakerQueue};
|
||||||
use crate::worker::{ServerWorker, ServerWorkerConfig, WorkerAvailability, WorkerHandle};
|
use crate::worker::{
|
||||||
|
ServerWorker, ServerWorkerConfig, WorkerAvailability, WorkerHandleAccept,
|
||||||
|
WorkerHandleServer,
|
||||||
|
};
|
||||||
use crate::{join_all, Token};
|
use crate::{join_all, Token};
|
||||||
|
|
||||||
/// Server builder
|
/// Server builder
|
||||||
@ -27,7 +30,7 @@ pub struct ServerBuilder {
|
|||||||
threads: usize,
|
threads: usize,
|
||||||
token: Token,
|
token: Token,
|
||||||
backlog: u32,
|
backlog: u32,
|
||||||
handles: Vec<(usize, WorkerHandle)>,
|
handles: Vec<(usize, WorkerHandleServer)>,
|
||||||
services: Vec<Box<dyn InternalServiceFactory>>,
|
services: Vec<Box<dyn InternalServiceFactory>>,
|
||||||
sockets: Vec<(Token, String, MioListener)>,
|
sockets: Vec<(Token, String, MioListener)>,
|
||||||
accept: AcceptLoop,
|
accept: AcceptLoop,
|
||||||
@ -280,10 +283,11 @@ impl ServerBuilder {
|
|||||||
// start workers
|
// start workers
|
||||||
let handles = (0..self.threads)
|
let handles = (0..self.threads)
|
||||||
.map(|idx| {
|
.map(|idx| {
|
||||||
let handle = self.start_worker(idx, self.accept.waker_owned());
|
let (handle_accept, handle_server) =
|
||||||
self.handles.push((idx, handle.clone()));
|
self.start_worker(idx, self.accept.waker_owned());
|
||||||
|
self.handles.push((idx, handle_server));
|
||||||
|
|
||||||
handle
|
handle_accept
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
@ -311,7 +315,11 @@ impl ServerBuilder {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn start_worker(&self, idx: usize, waker: WakerQueue) -> WorkerHandle {
|
fn start_worker(
|
||||||
|
&self,
|
||||||
|
idx: usize,
|
||||||
|
waker: WakerQueue,
|
||||||
|
) -> (WorkerHandleAccept, WorkerHandleServer) {
|
||||||
let avail = WorkerAvailability::new(waker);
|
let avail = WorkerAvailability::new(waker);
|
||||||
let services = self.services.iter().map(|v| v.clone_factory()).collect();
|
let services = self.services.iter().map(|v| v.clone_factory()).collect();
|
||||||
|
|
||||||
@ -437,9 +445,10 @@ impl ServerBuilder {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
let handle = self.start_worker(new_idx, self.accept.waker_owned());
|
let (handle_accept, handle_server) =
|
||||||
self.handles.push((new_idx, handle.clone()));
|
self.start_worker(new_idx, self.accept.waker_owned());
|
||||||
self.accept.wake(WakerInterest::Worker(handle));
|
self.handles.push((new_idx, handle_server));
|
||||||
|
self.accept.wake(WakerInterest::Worker(handle_accept));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,7 @@ use std::{
|
|||||||
|
|
||||||
use mio::{Registry, Token as MioToken, Waker};
|
use mio::{Registry, Token as MioToken, Waker};
|
||||||
|
|
||||||
use crate::worker::WorkerHandle;
|
use crate::worker::WorkerHandleAccept;
|
||||||
|
|
||||||
/// Waker token for `mio::Poll` instance.
|
/// Waker token for `mio::Poll` instance.
|
||||||
pub(crate) const WAKER_TOKEN: MioToken = MioToken(usize::MAX);
|
pub(crate) const WAKER_TOKEN: MioToken = MioToken(usize::MAX);
|
||||||
@ -84,6 +84,6 @@ pub(crate) enum WakerInterest {
|
|||||||
Timer,
|
Timer,
|
||||||
/// `Worker` is an interest happen after a worker runs into faulted state(This is determined
|
/// `Worker` is an interest happen after a worker runs into faulted state(This is determined
|
||||||
/// by if work can be sent to it successfully).`Accept` would be waked up and add the new
|
/// by if work can be sent to it successfully).`Accept` would be waked up and add the new
|
||||||
/// `WorkerHandle`.
|
/// `WorkerHandleAccept`.
|
||||||
Worker(WorkerHandle),
|
Worker(WorkerHandleAccept),
|
||||||
}
|
}
|
||||||
|
@ -28,11 +28,9 @@ use crate::socket::MioStream;
|
|||||||
use crate::waker_queue::{WakerInterest, WakerQueue};
|
use crate::waker_queue::{WakerInterest, WakerQueue};
|
||||||
use crate::{join_all, Token};
|
use crate::{join_all, Token};
|
||||||
|
|
||||||
pub(crate) struct WorkerCommand(Conn);
|
/// Stop worker message. Returns `true` on successful graceful shutdown.
|
||||||
|
/// and `false` if some connections still alive when shutdown execute.
|
||||||
/// Stop worker message. Returns `true` on successful shutdown
|
pub(crate) struct Stop {
|
||||||
/// and `false` if some connections still alive.
|
|
||||||
pub(crate) struct StopCommand {
|
|
||||||
graceful: bool,
|
graceful: bool,
|
||||||
tx: oneshot::Sender<bool>,
|
tx: oneshot::Sender<bool>,
|
||||||
}
|
}
|
||||||
@ -43,42 +41,55 @@ pub(crate) struct Conn {
|
|||||||
pub token: Token,
|
pub token: Token,
|
||||||
}
|
}
|
||||||
|
|
||||||
// a handle to worker that can send message to worker and share the availability of worker to other
|
fn handle_pair(
|
||||||
// thread.
|
idx: usize,
|
||||||
#[derive(Clone)]
|
tx1: UnboundedSender<Conn>,
|
||||||
pub(crate) struct WorkerHandle {
|
tx2: UnboundedSender<Stop>,
|
||||||
|
avail: WorkerAvailability,
|
||||||
|
) -> (WorkerHandleAccept, WorkerHandleServer) {
|
||||||
|
let accept = WorkerHandleAccept {
|
||||||
|
idx,
|
||||||
|
tx: tx1,
|
||||||
|
avail,
|
||||||
|
};
|
||||||
|
|
||||||
|
let server = WorkerHandleServer { idx, tx: tx2 };
|
||||||
|
|
||||||
|
(accept, server)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Handle to worker that can send connection message to worker and share the
|
||||||
|
/// availability of worker to other thread.
|
||||||
|
///
|
||||||
|
/// Held by [Accept](crate::accept::Accept).
|
||||||
|
pub(crate) struct WorkerHandleAccept {
|
||||||
pub idx: usize,
|
pub idx: usize,
|
||||||
tx1: UnboundedSender<WorkerCommand>,
|
tx: UnboundedSender<Conn>,
|
||||||
tx2: UnboundedSender<StopCommand>,
|
|
||||||
avail: WorkerAvailability,
|
avail: WorkerAvailability,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WorkerHandle {
|
impl WorkerHandleAccept {
|
||||||
pub fn new(
|
pub(crate) fn send(&self, msg: Conn) -> Result<(), Conn> {
|
||||||
idx: usize,
|
self.tx.send(msg).map_err(|msg| msg.0)
|
||||||
tx1: UnboundedSender<WorkerCommand>,
|
|
||||||
tx2: UnboundedSender<StopCommand>,
|
|
||||||
avail: WorkerAvailability,
|
|
||||||
) -> Self {
|
|
||||||
WorkerHandle {
|
|
||||||
idx,
|
|
||||||
tx1,
|
|
||||||
tx2,
|
|
||||||
avail,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn send(&self, msg: Conn) -> Result<(), Conn> {
|
pub(crate) fn available(&self) -> bool {
|
||||||
self.tx1.send(WorkerCommand(msg)).map_err(|msg| msg.0 .0)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn available(&self) -> bool {
|
|
||||||
self.avail.available()
|
self.avail.available()
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn stop(&self, graceful: bool) -> oneshot::Receiver<bool> {
|
/// Handle to worker than can send stop message to worker.
|
||||||
|
///
|
||||||
|
/// Held by [ServerBuilder](crate::builder::ServerBuilder).
|
||||||
|
pub(crate) struct WorkerHandleServer {
|
||||||
|
pub idx: usize,
|
||||||
|
tx: UnboundedSender<Stop>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl WorkerHandleServer {
|
||||||
|
pub(crate) fn stop(&self, graceful: bool) -> oneshot::Receiver<bool> {
|
||||||
let (tx, rx) = oneshot::channel();
|
let (tx, rx) = oneshot::channel();
|
||||||
let _ = self.tx2.send(StopCommand { graceful, tx });
|
let _ = self.tx.send(Stop { graceful, tx });
|
||||||
rx
|
rx
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -114,8 +125,8 @@ impl WorkerAvailability {
|
|||||||
///
|
///
|
||||||
/// Worker accepts Socket objects via unbounded channel and starts stream processing.
|
/// Worker accepts Socket objects via unbounded channel and starts stream processing.
|
||||||
pub(crate) struct ServerWorker {
|
pub(crate) struct ServerWorker {
|
||||||
rx: UnboundedReceiver<WorkerCommand>,
|
rx: UnboundedReceiver<Conn>,
|
||||||
rx2: UnboundedReceiver<StopCommand>,
|
rx2: UnboundedReceiver<Stop>,
|
||||||
services: Vec<WorkerService>,
|
services: Vec<WorkerService>,
|
||||||
availability: WorkerAvailability,
|
availability: WorkerAvailability,
|
||||||
conns: Counter,
|
conns: Counter,
|
||||||
@ -187,7 +198,7 @@ impl ServerWorker {
|
|||||||
factories: Vec<Box<dyn InternalServiceFactory>>,
|
factories: Vec<Box<dyn InternalServiceFactory>>,
|
||||||
availability: WorkerAvailability,
|
availability: WorkerAvailability,
|
||||||
config: ServerWorkerConfig,
|
config: ServerWorkerConfig,
|
||||||
) -> WorkerHandle {
|
) -> (WorkerHandleAccept, WorkerHandleServer) {
|
||||||
let (tx1, rx) = unbounded_channel();
|
let (tx1, rx) = unbounded_channel();
|
||||||
let (tx2, rx2) = unbounded_channel();
|
let (tx2, rx2) = unbounded_channel();
|
||||||
let avail = availability.clone();
|
let avail = availability.clone();
|
||||||
@ -254,7 +265,7 @@ impl ServerWorker {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
WorkerHandle::new(idx, tx1, tx2, avail)
|
handle_pair(idx, tx1, tx2, avail)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn restart_service(&mut self, token: Token, factory_id: usize) {
|
fn restart_service(&mut self, token: Token, factory_id: usize) {
|
||||||
@ -360,8 +371,7 @@ impl Future for ServerWorker {
|
|||||||
let this = self.as_mut().get_mut();
|
let this = self.as_mut().get_mut();
|
||||||
|
|
||||||
// `StopWorker` message handler
|
// `StopWorker` message handler
|
||||||
if let Poll::Ready(Some(StopCommand { graceful, tx })) =
|
if let Poll::Ready(Some(Stop { graceful, tx })) = Pin::new(&mut this.rx2).poll_recv(cx)
|
||||||
Pin::new(&mut this.rx2).poll_recv(cx)
|
|
||||||
{
|
{
|
||||||
this.availability.set(false);
|
this.availability.set(false);
|
||||||
let num = this.conns.total();
|
let num = this.conns.total();
|
||||||
@ -472,7 +482,7 @@ impl Future for ServerWorker {
|
|||||||
|
|
||||||
match ready!(Pin::new(&mut this.rx).poll_recv(cx)) {
|
match ready!(Pin::new(&mut this.rx).poll_recv(cx)) {
|
||||||
// handle incoming io stream
|
// handle incoming io stream
|
||||||
Some(WorkerCommand(msg)) => {
|
Some(msg) => {
|
||||||
let guard = this.conns.get();
|
let guard = this.conns.get();
|
||||||
let _ = this.services[msg.token.0].service.call((guard, msg.io));
|
let _ = this.services[msg.token.0].service.call((guard, msg.io));
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user