mirror of
https://github.com/fafhrd91/actix-web
synced 2025-01-18 22:01:50 +01:00
add wsload tool; optimize ws frame parser
This commit is contained in:
parent
74377ef73d
commit
78da98a16d
@ -60,9 +60,10 @@ url = "1.6"
|
|||||||
cookie = { version="0.10", features=["percent-encode", "secure"] }
|
cookie = { version="0.10", features=["percent-encode", "secure"] }
|
||||||
|
|
||||||
# io
|
# io
|
||||||
mio = "0.6"
|
mio = "^0.6.13"
|
||||||
net2 = "0.2"
|
net2 = "0.2"
|
||||||
bytes = "0.4"
|
bytes = "0.4"
|
||||||
|
byteorder = "1"
|
||||||
futures = "0.1"
|
futures = "0.1"
|
||||||
tokio-io = "0.1"
|
tokio-io = "0.1"
|
||||||
tokio-core = "0.1"
|
tokio-core = "0.1"
|
||||||
@ -108,4 +109,5 @@ members = [
|
|||||||
"examples/websocket",
|
"examples/websocket",
|
||||||
"examples/websocket-chat",
|
"examples/websocket-chat",
|
||||||
"examples/web-cors/backend",
|
"examples/web-cors/backend",
|
||||||
|
"tools/wsload/",
|
||||||
]
|
]
|
||||||
|
@ -4,16 +4,17 @@ use std::fmt::Write;
|
|||||||
use bytes::BufMut;
|
use bytes::BufMut;
|
||||||
use futures::{Async, Poll};
|
use futures::{Async, Poll};
|
||||||
use tokio_io::AsyncWrite;
|
use tokio_io::AsyncWrite;
|
||||||
// use http::header::{HeaderValue, CONNECTION, DATE};
|
|
||||||
|
|
||||||
use body::Binary;
|
use body::Binary;
|
||||||
use server::{WriterState, MAX_WRITE_BUFFER_SIZE};
|
use server::WriterState;
|
||||||
use server::shared::SharedBytes;
|
use server::shared::SharedBytes;
|
||||||
|
|
||||||
use client::ClientRequest;
|
use client::ClientRequest;
|
||||||
|
|
||||||
|
|
||||||
const AVERAGE_HEADER_SIZE: usize = 30; // totally scientific
|
const LOW_WATERMARK: usize = 1024;
|
||||||
|
const HIGH_WATERMARK: usize = 8 * LOW_WATERMARK;
|
||||||
|
const AVERAGE_HEADER_SIZE: usize = 30;
|
||||||
|
|
||||||
bitflags! {
|
bitflags! {
|
||||||
struct Flags: u8 {
|
struct Flags: u8 {
|
||||||
@ -29,6 +30,8 @@ pub(crate) struct HttpClientWriter {
|
|||||||
written: u64,
|
written: u64,
|
||||||
headers_size: u32,
|
headers_size: u32,
|
||||||
buffer: SharedBytes,
|
buffer: SharedBytes,
|
||||||
|
low: usize,
|
||||||
|
high: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl HttpClientWriter {
|
impl HttpClientWriter {
|
||||||
@ -39,6 +42,8 @@ impl HttpClientWriter {
|
|||||||
written: 0,
|
written: 0,
|
||||||
headers_size: 0,
|
headers_size: 0,
|
||||||
buffer: buf,
|
buffer: buf,
|
||||||
|
low: LOW_WATERMARK,
|
||||||
|
high: HIGH_WATERMARK,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -50,6 +55,12 @@ impl HttpClientWriter {
|
|||||||
self.flags.contains(Flags::KEEPALIVE) && !self.flags.contains(Flags::UPGRADE)
|
self.flags.contains(Flags::KEEPALIVE) && !self.flags.contains(Flags::UPGRADE)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set write buffer capacity
|
||||||
|
pub fn set_buffer_capacity(&mut self, low_watermark: usize, high_watermark: usize) {
|
||||||
|
self.low = low_watermark;
|
||||||
|
self.high = high_watermark;
|
||||||
|
}
|
||||||
|
|
||||||
fn write_to_stream<T: AsyncWrite>(&mut self, stream: &mut T) -> io::Result<WriterState> {
|
fn write_to_stream<T: AsyncWrite>(&mut self, stream: &mut T) -> io::Result<WriterState> {
|
||||||
while !self.buffer.is_empty() {
|
while !self.buffer.is_empty() {
|
||||||
match stream.write(self.buffer.as_ref()) {
|
match stream.write(self.buffer.as_ref()) {
|
||||||
@ -61,7 +72,7 @@ impl HttpClientWriter {
|
|||||||
let _ = self.buffer.split_to(n);
|
let _ = self.buffer.split_to(n);
|
||||||
},
|
},
|
||||||
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
||||||
if self.buffer.len() > MAX_WRITE_BUFFER_SIZE {
|
if self.buffer.len() > self.high {
|
||||||
return Ok(WriterState::Pause)
|
return Ok(WriterState::Pause)
|
||||||
} else {
|
} else {
|
||||||
return Ok(WriterState::Done)
|
return Ok(WriterState::Done)
|
||||||
@ -117,7 +128,7 @@ impl HttpClientWriter {
|
|||||||
self.buffer.extend_from_slice(payload.as_ref())
|
self.buffer.extend_from_slice(payload.as_ref())
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.buffer.len() > MAX_WRITE_BUFFER_SIZE {
|
if self.buffer.len() > self.high {
|
||||||
Ok(WriterState::Pause)
|
Ok(WriterState::Pause)
|
||||||
} else {
|
} else {
|
||||||
Ok(WriterState::Done)
|
Ok(WriterState::Done)
|
||||||
@ -125,7 +136,7 @@ impl HttpClientWriter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn write_eof(&mut self) -> io::Result<WriterState> {
|
pub fn write_eof(&mut self) -> io::Result<WriterState> {
|
||||||
if self.buffer.len() > MAX_WRITE_BUFFER_SIZE {
|
if self.buffer.len() > self.high {
|
||||||
Ok(WriterState::Pause)
|
Ok(WriterState::Pause)
|
||||||
} else {
|
} else {
|
||||||
Ok(WriterState::Done)
|
Ok(WriterState::Done)
|
||||||
|
@ -51,6 +51,7 @@ extern crate log;
|
|||||||
extern crate time;
|
extern crate time;
|
||||||
extern crate base64;
|
extern crate base64;
|
||||||
extern crate bytes;
|
extern crate bytes;
|
||||||
|
extern crate byteorder;
|
||||||
extern crate sha1;
|
extern crate sha1;
|
||||||
extern crate regex;
|
extern crate regex;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
|
@ -92,9 +92,9 @@ impl From<HttpResponseParserError> for WsClientError {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// WebSocket client
|
/// `WebSocket` client
|
||||||
///
|
///
|
||||||
/// Example of WebSocket client usage is available in
|
/// Example of `WebSocket` client usage is available in
|
||||||
/// [websocket example](
|
/// [websocket example](
|
||||||
/// https://github.com/actix/actix-web/blob/master/examples/websocket/src/client.rs#L24)
|
/// https://github.com/actix/actix-web/blob/master/examples/websocket/src/client.rs#L24)
|
||||||
pub struct WsClient {
|
pub struct WsClient {
|
||||||
@ -317,7 +317,7 @@ impl Future for WsHandshake {
|
|||||||
return Err(WsClientError::InvalidChallengeResponse)
|
return Err(WsClientError::InvalidChallengeResponse)
|
||||||
}
|
}
|
||||||
|
|
||||||
let inner = Rc::new(UnsafeCell::new(Inner{inner: inner}));
|
let inner = Rc::new(UnsafeCell::new(inner));
|
||||||
Ok(Async::Ready(
|
Ok(Async::Ready(
|
||||||
(WsClientReader{inner: Rc::clone(&inner)},
|
(WsClientReader{inner: Rc::clone(&inner)},
|
||||||
WsClientWriter{inner: inner})))
|
WsClientWriter{inner: inner})))
|
||||||
@ -332,12 +332,8 @@ impl Future for WsHandshake {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
struct Inner {
|
|
||||||
inner: WsInner,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct WsClientReader {
|
pub struct WsClientReader {
|
||||||
inner: Rc<UnsafeCell<Inner>>
|
inner: Rc<UnsafeCell<WsInner>>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Debug for WsClientReader {
|
impl fmt::Debug for WsClientReader {
|
||||||
@ -348,7 +344,7 @@ impl fmt::Debug for WsClientReader {
|
|||||||
|
|
||||||
impl WsClientReader {
|
impl WsClientReader {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn as_mut(&mut self) -> &mut Inner {
|
fn as_mut(&mut self) -> &mut WsInner {
|
||||||
unsafe{ &mut *self.inner.get() }
|
unsafe{ &mut *self.inner.get() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -361,10 +357,10 @@ impl Stream for WsClientReader {
|
|||||||
let inner = self.as_mut();
|
let inner = self.as_mut();
|
||||||
let mut done = false;
|
let mut done = false;
|
||||||
|
|
||||||
match utils::read_from_io(&mut inner.inner.conn, &mut inner.inner.parser_buf) {
|
match utils::read_from_io(&mut inner.conn, &mut inner.parser_buf) {
|
||||||
Ok(Async::Ready(0)) => {
|
Ok(Async::Ready(0)) => {
|
||||||
done = true;
|
done = true;
|
||||||
inner.inner.closed = true;
|
inner.closed = true;
|
||||||
},
|
},
|
||||||
Ok(Async::Ready(_)) | Ok(Async::NotReady) => (),
|
Ok(Async::Ready(_)) | Ok(Async::NotReady) => (),
|
||||||
Err(err) =>
|
Err(err) =>
|
||||||
@ -372,10 +368,10 @@ impl Stream for WsClientReader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// write
|
// write
|
||||||
let _ = inner.inner.writer.poll_completed(&mut inner.inner.conn, false);
|
let _ = inner.writer.poll_completed(&mut inner.conn, false);
|
||||||
|
|
||||||
// read
|
// read
|
||||||
match Frame::parse(&mut inner.inner.parser_buf) {
|
match Frame::parse(&mut inner.parser_buf) {
|
||||||
Ok(Some(frame)) => {
|
Ok(Some(frame)) => {
|
||||||
// trace!("WsFrame {}", frame);
|
// trace!("WsFrame {}", frame);
|
||||||
let (_finished, opcode, payload) = frame.unpack();
|
let (_finished, opcode, payload) = frame.unpack();
|
||||||
@ -385,8 +381,8 @@ impl Stream for WsClientReader {
|
|||||||
OpCode::Bad =>
|
OpCode::Bad =>
|
||||||
Ok(Async::Ready(Some(Message::Error))),
|
Ok(Async::Ready(Some(Message::Error))),
|
||||||
OpCode::Close => {
|
OpCode::Close => {
|
||||||
inner.inner.closed = true;
|
inner.closed = true;
|
||||||
inner.inner.error_sent = true;
|
inner.error_sent = true;
|
||||||
Ok(Async::Ready(Some(Message::Closed)))
|
Ok(Async::Ready(Some(Message::Closed)))
|
||||||
},
|
},
|
||||||
OpCode::Ping =>
|
OpCode::Ping =>
|
||||||
@ -413,9 +409,9 @@ impl Stream for WsClientReader {
|
|||||||
Ok(None) => {
|
Ok(None) => {
|
||||||
if done {
|
if done {
|
||||||
Ok(Async::Ready(None))
|
Ok(Async::Ready(None))
|
||||||
} else if inner.inner.closed {
|
} else if inner.closed {
|
||||||
if !inner.inner.error_sent {
|
if !inner.error_sent {
|
||||||
inner.inner.error_sent = true;
|
inner.error_sent = true;
|
||||||
Ok(Async::Ready(Some(Message::Closed)))
|
Ok(Async::Ready(Some(Message::Closed)))
|
||||||
} else {
|
} else {
|
||||||
Ok(Async::Ready(None))
|
Ok(Async::Ready(None))
|
||||||
@ -425,8 +421,8 @@ impl Stream for WsClientReader {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
inner.inner.closed = true;
|
inner.closed = true;
|
||||||
inner.inner.error_sent = true;
|
inner.error_sent = true;
|
||||||
Err(err.into())
|
Err(err.into())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -434,12 +430,12 @@ impl Stream for WsClientReader {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct WsClientWriter {
|
pub struct WsClientWriter {
|
||||||
inner: Rc<UnsafeCell<Inner>>
|
inner: Rc<UnsafeCell<WsInner>>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WsClientWriter {
|
impl WsClientWriter {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn as_mut(&mut self) -> &mut Inner {
|
fn as_mut(&mut self) -> &mut WsInner {
|
||||||
unsafe{ &mut *self.inner.get() }
|
unsafe{ &mut *self.inner.get() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -449,8 +445,8 @@ impl WsClientWriter {
|
|||||||
/// Write payload
|
/// Write payload
|
||||||
#[inline]
|
#[inline]
|
||||||
fn write<B: Into<Binary>>(&mut self, data: B) {
|
fn write<B: Into<Binary>>(&mut self, data: B) {
|
||||||
if !self.as_mut().inner.closed {
|
if !self.as_mut().closed {
|
||||||
let _ = self.as_mut().inner.writer.write(&data.into());
|
let _ = self.as_mut().writer.write(&data.into());
|
||||||
} else {
|
} else {
|
||||||
warn!("Trying to write to disconnected response");
|
warn!("Trying to write to disconnected response");
|
||||||
}
|
}
|
||||||
|
@ -211,11 +211,8 @@ impl<A, S> ActorHttpContext for WebsocketContext<A, S> where A: Actor<Context=Se
|
|||||||
mem::transmute(self as &mut WebsocketContext<A, S>)
|
mem::transmute(self as &mut WebsocketContext<A, S>)
|
||||||
};
|
};
|
||||||
|
|
||||||
if self.inner.alive() {
|
if self.inner.alive() && self.inner.poll(ctx).is_err() {
|
||||||
match self.inner.poll(ctx) {
|
return Err(ErrorInternalServerError("error").into())
|
||||||
Ok(Async::NotReady) | Ok(Async::Ready(())) => (),
|
|
||||||
Err(_) => return Err(ErrorInternalServerError("error").into()),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// frames
|
// frames
|
||||||
|
@ -2,6 +2,7 @@ use std::{fmt, mem};
|
|||||||
use std::io::{Write, Error, ErrorKind};
|
use std::io::{Write, Error, ErrorKind};
|
||||||
use std::iter::FromIterator;
|
use std::iter::FromIterator;
|
||||||
use bytes::BytesMut;
|
use bytes::BytesMut;
|
||||||
|
use byteorder::{ByteOrder, BigEndian};
|
||||||
|
|
||||||
use body::Binary;
|
use body::Binary;
|
||||||
use ws::proto::{OpCode, CloseCode};
|
use ws::proto::{OpCode, CloseCode};
|
||||||
@ -39,7 +40,6 @@ impl Frame {
|
|||||||
header_length += 8;
|
header_length += 8;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.mask.is_some() {
|
if self.mask.is_some() {
|
||||||
header_length += 4;
|
header_length += 4;
|
||||||
}
|
}
|
||||||
@ -84,74 +84,48 @@ impl Frame {
|
|||||||
/// Parse the input stream into a frame.
|
/// Parse the input stream into a frame.
|
||||||
pub fn parse(buf: &mut BytesMut) -> Result<Option<Frame>, Error> {
|
pub fn parse(buf: &mut BytesMut) -> Result<Option<Frame>, Error> {
|
||||||
let mut idx = 2;
|
let mut idx = 2;
|
||||||
|
|
||||||
let (frame, length) = {
|
|
||||||
let mut size = buf.len();
|
let mut size = buf.len();
|
||||||
|
|
||||||
if size < 2 {
|
if size < 2 {
|
||||||
return Ok(None)
|
return Ok(None)
|
||||||
}
|
}
|
||||||
let mut head = [0u8; 2];
|
|
||||||
size -= 2;
|
size -= 2;
|
||||||
head.copy_from_slice(&buf[..2]);
|
let first = buf[0];
|
||||||
|
let second = buf[1];
|
||||||
trace!("Parsed headers {:?}", head);
|
|
||||||
|
|
||||||
let first = head[0];
|
|
||||||
let second = head[1];
|
|
||||||
trace!("First: {:b}", first);
|
|
||||||
trace!("Second: {:b}", second);
|
|
||||||
|
|
||||||
let finished = first & 0x80 != 0;
|
let finished = first & 0x80 != 0;
|
||||||
|
|
||||||
let rsv1 = first & 0x40 != 0;
|
let rsv1 = first & 0x40 != 0;
|
||||||
let rsv2 = first & 0x20 != 0;
|
let rsv2 = first & 0x20 != 0;
|
||||||
let rsv3 = first & 0x10 != 0;
|
let rsv3 = first & 0x10 != 0;
|
||||||
|
|
||||||
let opcode = OpCode::from(first & 0x0F);
|
let opcode = OpCode::from(first & 0x0F);
|
||||||
trace!("Opcode: {:?}", opcode);
|
|
||||||
|
|
||||||
let masked = second & 0x80 != 0;
|
let masked = second & 0x80 != 0;
|
||||||
trace!("Masked: {:?}", masked);
|
let len = second & 0x7F;
|
||||||
|
|
||||||
let mut header_length = 2;
|
let length = if len == 126 {
|
||||||
let mut length = u64::from(second & 0x7F);
|
|
||||||
|
|
||||||
if length == 126 {
|
|
||||||
if size < 2 {
|
if size < 2 {
|
||||||
return Ok(None)
|
return Ok(None)
|
||||||
}
|
}
|
||||||
let mut length_bytes = [0u8; 2];
|
let len = u64::from(BigEndian::read_u16(&buf[idx..]));
|
||||||
length_bytes.copy_from_slice(&buf[idx..idx+2]);
|
|
||||||
size -= 2;
|
size -= 2;
|
||||||
idx += 2;
|
idx += 2;
|
||||||
|
len
|
||||||
length = u64::from(unsafe{
|
} else if len == 127 {
|
||||||
let mut wide: u16 = mem::transmute(length_bytes);
|
|
||||||
wide = u16::from_be(wide);
|
|
||||||
wide});
|
|
||||||
header_length += 2;
|
|
||||||
} else if length == 127 {
|
|
||||||
if size < 8 {
|
if size < 8 {
|
||||||
return Ok(None)
|
return Ok(None)
|
||||||
}
|
}
|
||||||
let mut length_bytes = [0u8; 8];
|
let len = BigEndian::read_u64(&buf[idx..]);
|
||||||
length_bytes.copy_from_slice(&buf[idx..idx+8]);
|
|
||||||
size -= 8;
|
size -= 8;
|
||||||
idx += 8;
|
idx += 8;
|
||||||
|
len
|
||||||
unsafe { length = mem::transmute(length_bytes); }
|
} else {
|
||||||
length = u64::from_be(length);
|
u64::from(len)
|
||||||
header_length += 8;
|
};
|
||||||
}
|
|
||||||
trace!("Payload length: {}", length);
|
|
||||||
|
|
||||||
let mask = if masked {
|
let mask = if masked {
|
||||||
let mut mask_bytes = [0u8; 4];
|
let mut mask_bytes = [0u8; 4];
|
||||||
if size < 4 {
|
if size < 4 {
|
||||||
return Ok(None)
|
return Ok(None)
|
||||||
} else {
|
} else {
|
||||||
header_length += 4;
|
|
||||||
size -= 4;
|
size -= 4;
|
||||||
mask_bytes.copy_from_slice(&buf[idx..idx+4]);
|
mask_bytes.copy_from_slice(&buf[idx..idx+4]);
|
||||||
idx += 4;
|
idx += 4;
|
||||||
@ -166,10 +140,13 @@ impl Frame {
|
|||||||
return Ok(None)
|
return Ok(None)
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut data = Vec::with_capacity(length);
|
// get body
|
||||||
if length > 0 {
|
buf.split_to(idx);
|
||||||
data.extend_from_slice(&buf[idx..idx+length]);
|
let mut data = if length > 0 {
|
||||||
}
|
buf.split_to(length)
|
||||||
|
} else {
|
||||||
|
BytesMut::new()
|
||||||
|
};
|
||||||
|
|
||||||
// Disallow bad opcode
|
// Disallow bad opcode
|
||||||
if let OpCode::Bad = opcode {
|
if let OpCode::Bad = opcode {
|
||||||
@ -199,7 +176,7 @@ impl Frame {
|
|||||||
apply_mask(&mut data, mask);
|
apply_mask(&mut data, mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
let frame = Frame {
|
Ok(Some(Frame {
|
||||||
finished: finished,
|
finished: finished,
|
||||||
rsv1: rsv1,
|
rsv1: rsv1,
|
||||||
rsv2: rsv2,
|
rsv2: rsv2,
|
||||||
@ -207,13 +184,7 @@ impl Frame {
|
|||||||
opcode: opcode,
|
opcode: opcode,
|
||||||
mask: mask,
|
mask: mask,
|
||||||
payload: data.into(),
|
payload: data.into(),
|
||||||
};
|
}))
|
||||||
|
|
||||||
(frame, header_length + length)
|
|
||||||
};
|
|
||||||
|
|
||||||
buf.split_to(length);
|
|
||||||
Ok(Some(frame))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Write a frame out to a buffer
|
/// Write a frame out to a buffer
|
||||||
|
21
tools/wsload/Cargo.toml
Normal file
21
tools/wsload/Cargo.toml
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
[package]
|
||||||
|
name = "wsclient"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Nikolay Kim <fafhrd91@gmail.com>"]
|
||||||
|
workspace = "../.."
|
||||||
|
|
||||||
|
[[bin]]
|
||||||
|
name = "wsclient"
|
||||||
|
path = "src/wsclient.rs"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
env_logger = "*"
|
||||||
|
futures = "0.1"
|
||||||
|
clap = "2"
|
||||||
|
url = "1.6"
|
||||||
|
rand = "0.4"
|
||||||
|
time = "*"
|
||||||
|
num_cpus = "1"
|
||||||
|
tokio-core = "0.1"
|
||||||
|
actix = { git = "https://github.com/actix/actix.git" }
|
||||||
|
actix-web = { path="../../" }
|
238
tools/wsload/src/wsclient.rs
Normal file
238
tools/wsload/src/wsclient.rs
Normal file
@ -0,0 +1,238 @@
|
|||||||
|
//! Simple websocket client.
|
||||||
|
|
||||||
|
#![allow(unused_variables)]
|
||||||
|
extern crate actix;
|
||||||
|
extern crate actix_web;
|
||||||
|
extern crate env_logger;
|
||||||
|
extern crate futures;
|
||||||
|
extern crate tokio_core;
|
||||||
|
extern crate url;
|
||||||
|
extern crate clap;
|
||||||
|
extern crate rand;
|
||||||
|
extern crate time;
|
||||||
|
extern crate num_cpus;
|
||||||
|
|
||||||
|
use std::time::Duration;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
use futures::Future;
|
||||||
|
use rand::{thread_rng, Rng};
|
||||||
|
|
||||||
|
use actix::prelude::*;
|
||||||
|
use actix_web::ws::{Message, WsClientError, WsClient, WsClientWriter};
|
||||||
|
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
::std::env::set_var("RUST_LOG", "actix_web=info");
|
||||||
|
let _ = env_logger::init();
|
||||||
|
|
||||||
|
let matches = clap::App::new("ws tool")
|
||||||
|
.version("0.1")
|
||||||
|
.about("Applies load to websocket server")
|
||||||
|
.args_from_usage(
|
||||||
|
"<url> 'WebSocket url'
|
||||||
|
[bin]... -b, 'use binary frames'
|
||||||
|
-s, --size=[NUMBER] 'size of PUBLISH packet payload to send in KB'
|
||||||
|
-w, --warm-up=[SECONDS] 'seconds before counter values are considered for reporting'
|
||||||
|
-r, --sample-rate=[SECONDS] 'seconds between average reports'
|
||||||
|
-c, --concurrency=[NUMBER] 'number of websockt connections to open and use concurrently for sending'
|
||||||
|
-t, --threads=[NUMBER] 'number of threads to use'",
|
||||||
|
)
|
||||||
|
.get_matches();
|
||||||
|
|
||||||
|
let bin: bool = matches.value_of("bin").is_some();
|
||||||
|
let ws_url = matches.value_of("url").unwrap().to_owned();
|
||||||
|
let _ = url::Url::parse(&ws_url).map_err(|e| {
|
||||||
|
println!("Invalid url: {}", ws_url);
|
||||||
|
std::process::exit(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
let threads = parse_u64_default(matches.value_of("threads"), num_cpus::get() as u64);
|
||||||
|
let concurrency = parse_u64_default(matches.value_of("concurrency"), 1);
|
||||||
|
let payload_size: usize = match matches.value_of("size") {
|
||||||
|
Some(s) => parse_u64_default(Some(s), 0) as usize * 1024,
|
||||||
|
None => 1024,
|
||||||
|
};
|
||||||
|
let warmup_seconds = parse_u64_default(matches.value_of("warm-up"), 2) as u64;
|
||||||
|
let sample_rate = parse_u64_default(matches.value_of("sample-rate"), 1) as usize;
|
||||||
|
|
||||||
|
let perf_counters = Arc::new(PerfCounters::new());
|
||||||
|
let payload = Arc::new(thread_rng()
|
||||||
|
.gen_ascii_chars()
|
||||||
|
.take(payload_size)
|
||||||
|
.collect::<String>());
|
||||||
|
|
||||||
|
let sys = actix::System::new("ws-client");
|
||||||
|
|
||||||
|
let mut report = true;
|
||||||
|
for t in 0..threads {
|
||||||
|
let pl = payload.clone();
|
||||||
|
let ws = ws_url.clone();
|
||||||
|
let perf = perf_counters.clone();
|
||||||
|
let addr = Arbiter::new(format!("test {}", t));
|
||||||
|
|
||||||
|
addr.send(actix::msgs::Execute::new(move || -> Result<(), ()> {
|
||||||
|
let mut reps = report;
|
||||||
|
for _ in 0..concurrency {
|
||||||
|
let pl2 = pl.clone();
|
||||||
|
let perf2 = perf.clone();
|
||||||
|
|
||||||
|
Arbiter::handle().spawn(
|
||||||
|
WsClient::new(&ws).connect().unwrap()
|
||||||
|
.map_err(|e| {
|
||||||
|
println!("Error: {}", e);
|
||||||
|
Arbiter::system().send(actix::msgs::SystemExit(0));
|
||||||
|
()
|
||||||
|
})
|
||||||
|
.map(move |(reader, writer)| {
|
||||||
|
let addr: SyncAddress<_> = ChatClient::create(move |ctx| {
|
||||||
|
ChatClient::add_stream(reader, ctx);
|
||||||
|
ChatClient{conn: writer,
|
||||||
|
payload: pl2,
|
||||||
|
report: reps,
|
||||||
|
bin: bin,
|
||||||
|
ts: time::precise_time_ns(),
|
||||||
|
perf_counters: perf2,
|
||||||
|
sample_rate_secs: sample_rate,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
);
|
||||||
|
reps = false;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}));
|
||||||
|
report = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let _ = sys.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_u64_default(input: Option<&str>, default: u64) -> u64 {
|
||||||
|
input.map(|v| v.parse().expect(&format!("not a valid number: {}", v)))
|
||||||
|
.unwrap_or(default)
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ChatClient{
|
||||||
|
conn: WsClientWriter,
|
||||||
|
payload: Arc<String>,
|
||||||
|
ts: u64,
|
||||||
|
bin: bool,
|
||||||
|
report: bool,
|
||||||
|
perf_counters: Arc<PerfCounters>,
|
||||||
|
sample_rate_secs: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Actor for ChatClient {
|
||||||
|
type Context = Context<Self>;
|
||||||
|
|
||||||
|
fn started(&mut self, ctx: &mut Context<Self>) {
|
||||||
|
self.send_text();
|
||||||
|
if self.report {
|
||||||
|
self.sample_rate(ctx);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn stopping(&mut self, _: &mut Context<Self>) -> bool {
|
||||||
|
Arbiter::system().send(actix::msgs::SystemExit(0));
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ChatClient {
|
||||||
|
fn sample_rate(&self, ctx: &mut Context<Self>) {
|
||||||
|
ctx.run_later(Duration::new(self.sample_rate_secs as u64, 0), |act, ctx| {
|
||||||
|
let req_count = act.perf_counters.pull_request_count();
|
||||||
|
if req_count != 0 {
|
||||||
|
let latency = act.perf_counters.pull_latency_ns();
|
||||||
|
let latency_max = act.perf_counters.pull_latency_max_ns();
|
||||||
|
println!(
|
||||||
|
"rate: {}, throughput: {:?} kb, latency: {}, latency max: {}",
|
||||||
|
req_count / act.sample_rate_secs,
|
||||||
|
(((req_count * act.payload.len()) as f64) / 1024.0) /
|
||||||
|
act.sample_rate_secs as f64,
|
||||||
|
time::Duration::nanoseconds((latency / req_count as u64) as i64),
|
||||||
|
time::Duration::nanoseconds(latency_max as i64)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
act.sample_rate(ctx);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn send_text(&mut self) {
|
||||||
|
self.ts = time::precise_time_ns();
|
||||||
|
if self.bin {
|
||||||
|
self.conn.binary(&self.payload);
|
||||||
|
} else {
|
||||||
|
self.conn.text(&self.payload);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Handle server websocket messages
|
||||||
|
impl StreamHandler<Message, WsClientError> for ChatClient {
|
||||||
|
|
||||||
|
fn finished(&mut self, ctx: &mut Context<Self>) {
|
||||||
|
ctx.stop()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle(&mut self, msg: Message, ctx: &mut Context<Self>) {
|
||||||
|
match msg {
|
||||||
|
Message::Text(txt) => {
|
||||||
|
if txt == self.payload.as_ref().as_str() {
|
||||||
|
self.perf_counters.register_request();
|
||||||
|
self.perf_counters.register_latency(time::precise_time_ns() - self.ts);
|
||||||
|
self.send_text();
|
||||||
|
} else {
|
||||||
|
println!("not eaqual");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
pub struct PerfCounters {
|
||||||
|
req: AtomicUsize,
|
||||||
|
lat: AtomicUsize,
|
||||||
|
lat_max: AtomicUsize
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PerfCounters {
|
||||||
|
pub fn new() -> PerfCounters {
|
||||||
|
PerfCounters {
|
||||||
|
req: AtomicUsize::new(0),
|
||||||
|
lat: AtomicUsize::new(0),
|
||||||
|
lat_max: AtomicUsize::new(0),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pull_request_count(&self) -> usize {
|
||||||
|
self.req.swap(0, Ordering::SeqCst)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pull_latency_ns(&self) -> u64 {
|
||||||
|
self.lat.swap(0, Ordering::SeqCst) as u64
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pull_latency_max_ns(&self) -> u64 {
|
||||||
|
self.lat_max.swap(0, Ordering::SeqCst) as u64
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn register_request(&self) {
|
||||||
|
self.req.fetch_add(1, Ordering::SeqCst);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn register_latency(&self, nanos: u64) {
|
||||||
|
let nanos = nanos as usize;
|
||||||
|
self.lat.fetch_add(nanos, Ordering::SeqCst);
|
||||||
|
loop {
|
||||||
|
let current = self.lat_max.load(Ordering::SeqCst);
|
||||||
|
if current >= nanos || self.lat_max.compare_and_swap(current, nanos, Ordering::SeqCst) == current {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user