Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
misc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
misc
Commits
b77b5f81
Commit
b77b5f81
authored
Apr 26, 2016
by
Elvis Pranskevichus
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add support for Unix socket testing, JSON output of benchmarks, selective running
parent
e7fe9dc8
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
344 additions
and
119 deletions
+344
-119
.dockerignore
.dockerignore
+1
-0
Dockerfile
Dockerfile
+3
-1
echo_client
echo_client
+81
-39
entrypoint
entrypoint
+4
-2
run_benchmarks
run_benchmarks
+232
-77
torecho.py
torecho.py
+23
-0
No files found.
.dockerignore
View file @
b77b5f81
.cache
.cache
sockets
Dockerfile
View file @
b77b5f81
...
@@ -13,18 +13,20 @@ RUN mkdir -p /usr/local/python-venvs
...
@@ -13,18 +13,20 @@ RUN mkdir -p /usr/local/python-venvs
RUN
DEBIAN_FRONTEND
=
noninteractive
\
RUN
DEBIAN_FRONTEND
=
noninteractive
\
apt-get update
&&
apt-get
install
-y
\
apt-get update
&&
apt-get
install
-y
\
autoconf automake libtool build-essential
\
autoconf automake libtool build-essential
\
python3 python3-pip git nodejs
python3 python3-pip git nodejs
gosu
RUN
pip3
install
vex
RUN
pip3
install
vex
RUN
vex
--python
=
python3.5
-m
bench pip
install
-U
pip
RUN
vex
--python
=
python3.5
-m
bench pip
install
-U
pip
RUN
mkdir
-p
/var/lib/cache/pip
RUN
mkdir
-p
/var/lib/cache/pip
ADD
http_server.py /tmp/http_server.py
ADD
http_server.py /tmp/http_server.py
ADD
torecho.py /tmp/torecho.py
ADD
requirements.txt /tmp/requirements.txt
ADD
requirements.txt /tmp/requirements.txt
EXPOSE
25000
EXPOSE
25000
VOLUME
/var/lib/cache
VOLUME
/var/lib/cache
VOLUME
/tmp/sockets
ENTRYPOINT
["/entrypoint"]
ENTRYPOINT
["/entrypoint"]
...
...
tcp
_client
→
echo
_client
View file @
b77b5f81
...
@@ -6,6 +6,8 @@
...
@@ -6,6 +6,8 @@
import
argparse
import
argparse
from
concurrent
import
futures
from
concurrent
import
futures
import
json
import
math
import
socket
import
socket
import
time
import
time
...
@@ -26,6 +28,8 @@ if __name__ == '__main__':
...
@@ -26,6 +28,8 @@ if __name__ == '__main__':
help
=
'socket timeout in seconds'
)
help
=
'socket timeout in seconds'
)
parser
.
add_argument
(
'--addr'
,
default
=
'127.0.0.1:25000'
,
type
=
str
,
parser
.
add_argument
(
'--addr'
,
default
=
'127.0.0.1:25000'
,
type
=
str
,
help
=
'server address'
)
help
=
'server address'
)
parser
.
add_argument
(
'--output-format'
,
default
=
'text'
,
type
=
str
,
help
=
'output format'
,
choices
=
[
'text'
,
'json'
])
args
=
parser
.
parse_args
()
args
=
parser
.
parse_args
()
unix
=
False
unix
=
False
...
@@ -49,11 +53,13 @@ if __name__ == '__main__':
...
@@ -49,11 +53,13 @@ if __name__ == '__main__':
else
:
else
:
sock
=
socket
.
socket
(
socket
.
AF_INET
,
socket
.
SOCK_STREAM
)
sock
=
socket
.
socket
(
socket
.
AF_INET
,
socket
.
SOCK_STREAM
)
sock
.
settimeout
(
5
)
sock
.
settimeout
(
1
5
)
sock
.
connect
(
addr
)
sock
.
connect
(
addr
)
n
=
0
n
=
0
latency_stats
=
np
.
zeros
((
timeout
*
10
,))
latency_stats
=
np
.
zeros
((
timeout
*
100
,))
min_latency
=
float
(
'inf'
)
max_latency
=
0.0
while
time
.
monotonic
()
-
start
<
duration
:
while
time
.
monotonic
()
-
start
<
duration
:
req_start
=
time
.
monotonic
()
req_start
=
time
.
monotonic
()
...
@@ -64,49 +70,42 @@ if __name__ == '__main__':
...
@@ -64,49 +70,42 @@ if __name__ == '__main__':
if
not
resp
:
if
not
resp
:
raise
SystemExit
()
raise
SystemExit
()
nrecv
+=
len
(
resp
)
nrecv
+=
len
(
resp
)
req_time
=
round
((
time
.
monotonic
()
-
req_start
)
*
10000
)
req_time
=
round
((
time
.
monotonic
()
-
req_start
)
*
100000
)
if
req_time
>
max_latency
:
max_latency
=
req_time
if
req_time
<
min_latency
:
min_latency
=
req_time
latency_stats
[
req_time
]
+=
1
latency_stats
[
req_time
]
+=
1
n
+=
1
n
+=
1
return
n
,
latency_stats
return
n
,
latency_stats
,
min_latency
,
max_latency
def
weighted_quantile
(
values
,
quantiles
,
sample_weight
=
None
,
def
weighted_quantile
(
values
,
quantiles
,
weights
):
values_sorted
=
False
,
old_style
=
False
):
""" Very close to np.percentile, but supports weights.
""" Very close to np.percentile, but supports weights.
NOTE: quantiles should be in [0, 1]!
:param values: np.array with data
:param values: np.array with data
:param quantiles: array-like with many quantiles needed
:param quantiles: array-like with many quantiles needed,
:param sample_weight: array-like of the same length as `array`
quantiles should be in [0, 1]!
:param values_sorted: bool, if True, then will avoid sorting of initial array
:param weights: array-like of the same length as `array`
:param old_style: if True, will correct output to be consistent with np.percentile.
:return: np.array with computed quantiles.
:return: np.array with computed quantiles.
"""
"""
values
=
np
.
array
(
values
)
values
=
np
.
array
(
values
)
quantiles
=
np
.
array
(
quantiles
)
quantiles
=
np
.
array
(
quantiles
)
if
sample_weight
is
None
:
weights
=
np
.
array
(
weights
)
sample_weight
=
np
.
ones
(
len
(
values
))
sample_weight
=
np
.
array
(
sample_weight
)
assert
np
.
all
(
quantiles
>=
0
)
and
np
.
all
(
quantiles
<=
1
),
\
assert
np
.
all
(
quantiles
>=
0
)
and
np
.
all
(
quantiles
<=
1
),
\
'quantiles should be in [0, 1]'
'quantiles should be in [0, 1]'
if
not
values_sorted
:
weighted_quantiles
=
np
.
cumsum
(
weights
)
-
0.5
*
weights
sorter
=
np
.
argsort
(
values
)
weighted_quantiles
/=
np
.
sum
(
weights
)
values
=
values
[
sorter
]
sample_weight
=
sample_weight
[
sorter
]
weighted_quantiles
=
np
.
cumsum
(
sample_weight
)
-
0.5
*
sample_weight
if
old_style
:
# To be convenient with np.percentile
weighted_quantiles
-=
weighted_quantiles
[
0
]
weighted_quantiles
/=
weighted_quantiles
[
-
1
]
else
:
weighted_quantiles
/=
np
.
sum
(
sample_weight
)
return
np
.
interp
(
quantiles
,
weighted_quantiles
,
values
)
return
np
.
interp
(
quantiles
,
weighted_quantiles
,
values
)
TIMES
=
args
.
times
TIMES
=
args
.
times
N
=
args
.
concurrency
N
=
args
.
concurrency
DURATION
=
args
.
duration
DURATION
=
args
.
duration
min_latency
=
float
(
'inf'
)
max_latency
=
0.0
messages
=
0
messages
=
0
latency_stats
=
None
latency_stats
=
None
start
=
time
.
monotonic
()
start
=
time
.
monotonic
()
...
@@ -118,35 +117,78 @@ if __name__ == '__main__':
...
@@ -118,35 +117,78 @@ if __name__ == '__main__':
res
=
futures
.
wait
(
fs
)
res
=
futures
.
wait
(
fs
)
for
fut
in
res
.
done
:
for
fut
in
res
.
done
:
t_messages
,
t_latency_stats
=
fut
.
result
()
t_messages
,
t_latency_stats
,
t_min_latency
,
t_max_latency
=
\
fut
.
result
()
messages
+=
t_messages
messages
+=
t_messages
if
latency_stats
is
None
:
if
latency_stats
is
None
:
latency_stats
=
t_latency_stats
latency_stats
=
t_latency_stats
else
:
else
:
latency_stats
=
np
.
add
(
latency_stats
,
t_latency_stats
)
latency_stats
=
np
.
add
(
latency_stats
,
t_latency_stats
)
if
t_max_latency
>
max_latency
:
max_latency
=
t_max_latency
if
t_min_latency
<
min_latency
:
min_latency
=
t_min_latency
end
=
time
.
monotonic
()
end
=
time
.
monotonic
()
duration
=
end
-
start
arange
=
np
.
arange
(
len
(
latency_stats
))
arange
=
np
.
arange
(
len
(
latency_stats
))
stddev
=
np
.
std
(
arange
)
weighted_latency
=
np
.
multiply
(
latency_stats
,
arange
)
weighted_latency
=
np
.
multiply
(
latency_stats
,
arange
)
mean_latency
=
np
.
sum
(
weighted_latency
)
/
messages
mean_latency
=
np
.
average
(
arange
,
weights
=
latency_stats
)
variance
=
np
.
average
((
arange
-
mean_latency
)
**
2
,
weights
=
latency_stats
)
latency_std
=
math
.
sqrt
(
variance
)
latency_cv
=
latency_std
/
mean_latency
percentiles
=
[
50
,
75
,
90
,
99
]
percentiles
=
[
50
,
75
,
90
,
99
]
percentile_data
=
[]
percentile_data
=
[]
quantiles
=
weighted_quantile
(
arange
,
[
p
/
100
for
p
in
percentiles
],
quantiles
=
weighted_quantile
(
arange
,
[
p
/
100
for
p
in
percentiles
],
sample_weight
=
latency_stats
,
weights
=
latency_stats
)
values_sorted
=
True
)
for
i
,
percentile
in
enumerate
(
percentiles
):
for
i
,
percentile
in
enumerate
(
percentiles
):
percentile_data
.
append
(
'{}%: {}ms'
.
format
(
percentile_data
.
append
((
percentile
,
round
(
quantiles
[
i
]
/
100
,
3
)))
percentile
,
round
(
quantiles
[
i
],
2
)))
data
=
dict
(
print
(
messages
,
'in'
,
round
(
duration
,
2
))
messages
=
messages
,
print
(
'Latency avg: {}ms'
.
format
(
round
(
mean_latency
,
2
)))
transfer
=
round
((
messages
*
MSGSIZE
/
(
1024
*
1024
))
/
DURATION
,
2
),
print
(
'Latency distribution: {}'
.
format
(
'; '
.
join
(
percentile_data
)))
rps
=
round
(
messages
/
DURATION
,
2
),
print
(
'Requests/sec: {}'
.
format
(
round
(
messages
/
duration
,
2
)))
latency_min
=
round
(
min_latency
/
100
,
3
),
transfer
=
(
messages
*
MSGSIZE
/
(
1024
*
1024
))
/
duration
latency_mean
=
round
(
mean_latency
/
100
,
3
),
print
(
'Transfer/sec: {}MiB'
.
format
(
round
(
transfer
,
2
)))
latency_max
=
round
(
max_latency
/
100
,
3
),
latency_std
=
round
(
latency_std
/
100
,
3
),
latency_cv
=
round
(
latency_cv
*
100
,
2
),
latency_percentiles
=
percentile_data
)
if
args
.
output_format
==
'json'
:
data
[
'latency_percentiles'
]
=
json
.
dumps
(
percentile_data
)
output
=
'''
\
{{
"messages": {messages},
"transfer": {transfer},
"rps": {rps},
"latency_min": {latency_min},
"latency_mean": {latency_mean},
"latency_max": {latency_max},
"latency_std": {latency_std},
"latency_cv": {latency_cv},
"latency_percentiles": {latency_percentiles}
}}'''
.
format
(
**
data
)
else
:
data
[
'latency_percentiles'
]
=
'; '
.
join
(
'{}% under {}ms'
.
format
(
*
v
)
for
v
in
percentile_data
)
output
=
'''
\
{messages} {size}KiB messages in {duration} seconds
Latency: min {latency_min}ms; max {latency_max}ms; mean {latency_mean}ms;
\
std: {latency_std}ms ({latency_cv}%)
Latency distribtion: {latency_percentiles}
Requests/sec: {rps}
Transfer/sec: {transfer}MiB
'''
.
format
(
duration
=
DURATION
,
size
=
round
(
MSGSIZE
/
1024
,
2
),
**
data
)
print
(
output
)
entrypoint
View file @
b77b5f81
...
@@ -10,5 +10,7 @@ git clone https://github.com/dabeaz/curio.git
...
@@ -10,5 +10,7 @@ git clone https://github.com/dabeaz/curio.git
git clone https://github.com/MagicStack/uvloop.git
git clone https://github.com/MagicStack/uvloop.git
cp
/tmp/http_server.py uvloop/examples/bench/
cp
/tmp/http_server.py uvloop/examples/bench/
echo
"Running server on port 25000..."
UID
=
${
UID
:-
0
}
"
$@
"
GID
=
${
GID
:-
0
}
gosu
${
UID
}
:
${
GID
}
"
$@
"
run_benchmarks
View file @
b77b5f81
...
@@ -2,97 +2,170 @@
...
@@ -2,97 +2,170 @@
import
argparse
import
argparse
import
multiprocessing
import
json
import
os
import
os.path
import
os.path
import
socket
import
socket
import
subprocess
import
subprocess
import
sys
import
sys
import
textwrap
import
time
import
time
_dir
=
os
.
path
.
dirname
(
__file__
)
_dir
=
os
.
path
.
dirname
(
__file__
)
_cache
=
os
.
path
.
abspath
(
os
.
path
.
join
(
_dir
,
'.cache'
))
_cache
=
os
.
path
.
abspath
(
os
.
path
.
join
(
_dir
,
'.cache'
))
_socket
=
os
.
path
.
abspath
(
os
.
path
.
join
(
_dir
,
'sockets'
))
http_client
=
"wrk --latency -d 30 -c 200 -t 4 http://127.0.0.1:25000/{msize}"
http_client
=
"wrk --latency -d 30 -c 200 -t 4 http://127.0.0.1:25000/{msize}"
http_server
=
"uvloop/examples/bench/http_server.py --addr='0.0.0.0:25000'"
http_server
=
"uvloop/examples/bench/http_server.py --addr='0.0.0.0:25000'"
server_base
=
[
'docker'
,
'run'
,
'--rm'
,
'-t'
,
'-p'
,
'25000:25000'
,
server_base
=
[
'docker'
,
'run'
,
'--rm'
,
'-t'
,
'-p'
,
'25000:25000'
,
'-e'
,
'UID={}'
.
format
(
os
.
geteuid
()),
'-e'
,
'GID={}'
.
format
(
os
.
getegid
()),
'-v'
,
'{_cache}:/var/lib/cache'
.
format
(
_cache
=
_cache
),
'-v'
,
'{_cache}:/var/lib/cache'
.
format
(
_cache
=
_cache
),
'-v'
,
'{_socket}:/tmp/sockets'
.
format
(
_socket
=
_socket
),
'--name'
,
'magicbench'
,
'magic/benchmark'
]
'--name'
,
'magicbench'
,
'magic/benchmark'
]
python
=
[
'vex'
,
'bench'
,
'python'
]
python
=
[
'vex'
,
'bench'
,
'python'
]
nodejs
=
[
'nodejs'
]
nodejs
=
[
'nodejs'
]
benchmarks
=
[{
echo_variations
=
[{
'title'
:
'TCP echo server (asyncio)'
,
'name'
:
'1kb'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=0.0.0.0:25000'
,
'--streams'
],
'client'
:
[
'./tcp_client'
,
'--addr=127.0.0.1:25000'
,
'--concurrency=4'
],
'warmup'
:
[
'--msize=1024'
,
'--concurrency=4'
,
'--duration=5'
],
'variations'
:
[{
'title'
:
'1kb messages, concurrency 4'
,
'title'
:
'1kb messages, concurrency 4'
,
'args'
:
[
'--msize=1024'
,
'--concurrency=4'
]
'args'
:
[
'--msize=1024'
,
'--concurrency=4'
]
},
{
},
{
'name'
:
'10kb'
,
'title'
:
'10kb messages, concurrency 4'
,
'title'
:
'10kb messages, concurrency 4'
,
'args'
:
[
'--msize=10240'
,
'--concurrency=4'
]
'args'
:
[
'--msize=10240'
,
'--concurrency=4'
]
},
{
},
{
'name'
:
'100kb'
,
'title'
:
'100kb messages, concurrency 4'
,
'title'
:
'100kb messages, concurrency 4'
,
'args'
:
[
'--msize=102400'
,
'--concurrency=4'
]
'args'
:
[
'--msize=102400'
,
'--concurrency=4'
]
}]
}]
},
{
echo_client
=
[
'./echo_client'
,
'--concurrency=4'
,
'--output-format=json'
]
echo_warmup
=
[
'--msize=1024'
,
'--concurrency=4'
,
'--duration=5'
]
tcp_address
=
'127.0.0.1:25000'
unix_address
=
'file:{_socket}/server.sock'
.
format
(
_socket
=
_socket
)
tcp_client
=
echo_client
+
[
'--addr={}'
.
format
(
tcp_address
)]
unix_client
=
echo_client
+
[
'--addr={}'
.
format
(
unix_address
)]
benchmarks
=
[{
'name'
:
'tcpecho-gevent'
,
'title'
:
'TCP echo server (gevent)'
,
'title'
:
'TCP echo server (gevent)'
,
'server'
:
python
+
[
'curio/examples/bench/gevecho.py'
],
'server'
:
python
+
[
'curio/examples/bench/gevecho.py'
],
'client'
:
[
'./tcp_client'
,
'--addr=127.0.0.1:25000'
,
'--concurrency=4'
],
'server_address'
:
tcp_address
,
'warmup'
:
[
'--msize=1024'
,
'--concurrency=4'
,
'--duration=5'
],
'client'
:
tcp_client
,
'warmup'
:
echo_warmup
,
'variations'
:
[{
'variations'
:
echo_variations
,
'title'
:
'1kb messages, concurrency 4'
,
},
{
'args'
:
[
'--msize=1024'
,
'--concurrency=4'
]
'name'
:
'tcpecho-tornado'
,
},
{
'title'
:
'TCP echo server (tornado)'
,
'title'
:
'10kb messages, concurrency 4'
,
'server'
:
python
+
[
'/tmp/torecho.py'
],
'args'
:
[
'--msize=10240'
,
'--concurrency=4'
]
'server_address'
:
tcp_address
,
},
{
'client'
:
tcp_client
,
'title'
:
'100kb messages, concurrency 4'
,
'warmup'
:
echo_warmup
,
'args'
:
[
'--msize=102400'
,
'--concurrency=4'
]
'variations'
:
echo_variations
,
}]
},
{
'name'
:
'tcpecho-curio'
,
'title'
:
'TCP echo server (curio)'
,
'server'
:
python
+
[
'curio/examples/bench/curioecho.py'
],
'server_address'
:
tcp_address
,
'client'
:
tcp_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
},
{
'name'
:
'tcpecho-nodejs'
,
'title'
:
'TCP echo server (nodejs)'
,
'title'
:
'TCP echo server (nodejs)'
,
'server'
:
nodejs
+
[
'curio/examples/bench/nodeecho.js'
],
'server'
:
nodejs
+
[
'curio/examples/bench/nodeecho.js'
],
'client'
:
[
'./tcp_client'
,
'--addr=127.0.0.1:25000'
,
'--concurrency=4'
],
'server_address'
:
tcp_address
,
'warmup'
:
[
'--msize=1024'
,
'--concurrency=4'
,
'--duration=5'
],
'client'
:
tcp_client
,
'warmup'
:
echo_warmup
,
'variations'
:
[{
'variations'
:
echo_variations
,
'title'
:
'1kb messages, concurrency 4'
,
},
{
'args'
:
[
'--msize=1024'
,
'--concurrency=4'
]
'name'
:
'tcpecho-asyncio-stdstreams'
,
},
{
'title'
:
'TCP echo server (asyncio/stdstreams)'
,
'title'
:
'10kb messages, concurrency 4'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'args'
:
[
'--msize=10240'
,
'--concurrency=4'
]
'--addr=0.0.0.0:25000'
,
},
{
'--streams'
],
'title'
:
'100kb messages, concurrency 4'
,
'server_address'
:
tcp_address
,
'args'
:
[
'--msize=102400'
,
'--concurrency=4'
]
'client'
:
tcp_client
,
}]
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
},
{
'title'
:
'TCP echo server (uvloop)'
,
'name'
:
'tcpecho-asyncio-minproto'
,
'title'
:
'TCP echo server (asyncio/minproto)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=0.0.0.0:25000'
,
'--proto'
],
'server_address'
:
tcp_address
,
'client'
:
tcp_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
'name'
:
'unixecho-asyncio-stdstreams'
,
'title'
:
'Unix socket echo server (asyncio/stdstreams)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=file:/tmp/sockets/server.sock'
,
'--streams'
],
'server_address'
:
unix_address
,
'client'
:
unix_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
'name'
:
'unixecho-asyncio-minproto'
,
'title'
:
'Unix socket echo server (asyncio/minproto)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=file:/tmp/sockets/server.sock'
,
'--proto'
],
'server_address'
:
unix_address
,
'client'
:
unix_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
'name'
:
'tcpecho-uvloop-stdstreams'
,
'title'
:
'TCP echo server (uvloop/stdstreams)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=0.0.0.0:25000'
,
'--addr=0.0.0.0:25000'
,
'--streams'
,
'--uvloop'
],
'--streams'
,
'--uvloop'
],
'client'
:
[
'./tcp_client'
,
'--addr=127.0.0.1:25000'
,
'--concurrency=4'
],
'server_address'
:
tcp_address
,
'warmup'
:
[
'--msize=1024'
,
'--concurrency=4'
,
'--duration=5'
],
'client'
:
tcp_client
,
'warmup'
:
echo_warmup
,
'variations'
:
[{
'variations'
:
echo_variations
,
'title'
:
'1kb messages, concurrency 4'
,
},
{
'args'
:
[
'--msize=1024'
,
'--concurrency=4'
]
'name'
:
'tcpecho-uvloop-minproto'
,
},
{
'title'
:
'TCP echo server (uvloop/minproto)'
,
'title'
:
'10kb messages, concurrency 4'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'args'
:
[
'--msize=10240'
,
'--concurrency=4'
]
'--addr=0.0.0.0:25000'
,
},
{
'--proto'
,
'--uvloop'
],
'title'
:
'100kb messages, concurrency 4'
,
'server_address'
:
tcp_address
,
'args'
:
[
'--msize=102400'
,
'--concurrency=4'
]
'client'
:
tcp_client
,
}]
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
'name'
:
'unixecho-uvloop-stdstreams'
,
'title'
:
'Unix socket echo server (uvloop/stdstreams)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=file:/tmp/sockets/server.sock'
,
'--streams'
,
'--uvloop'
],
'server_address'
:
unix_address
,
'client'
:
unix_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
},
{
'name'
:
'unixecho-uvloop-minproto'
,
'title'
:
'Unix socket echo server (uvloop/minproto)'
,
'server'
:
python
+
[
'uvloop/examples/bench/server.py'
,
'--addr=file:/tmp/sockets/server.sock'
,
'--proto'
,
'--uvloop'
],
'server_address'
:
unix_address
,
'client'
:
unix_client
,
'warmup'
:
echo_warmup
,
'variations'
:
echo_variations
,
}]
}]
...
@@ -101,17 +174,30 @@ def abort(msg):
...
@@ -101,17 +174,30 @@ def abort(msg):
sys
.
exit
(
1
)
sys
.
exit
(
1
)
def
start_and_wait_for_server
(
server_cmd
,
timeout
=
60
):
def
start_and_wait_for_server
(
server_cmd
,
address
,
timeout
=
60
):
kill_server
()
server
=
subprocess
.
Popen
(
server_cmd
,
universal_newlines
=
True
,
server
=
subprocess
.
Popen
(
server_cmd
,
universal_newlines
=
True
,
stdout
=
subprocess
.
PIPE
,
stderr
=
subprocess
.
PIPE
)
stdout
=
subprocess
.
PIPE
,
stderr
=
subprocess
.
PIPE
)
start
=
time
.
monotonic
()
start
=
time
.
monotonic
()
if
address
.
startswith
(
'file:'
):
family
=
socket
.
AF_UNIX
addr
=
address
[
5
:]
else
:
family
=
socket
.
AF_INET
addr
=
address
.
split
(
':'
)
addr
[
1
]
=
int
(
addr
[
1
])
addr
=
tuple
(
addr
)
print
(
'Trying to connect to server at address {}'
.
format
(
address
))
while
time
.
monotonic
()
-
start
<
timeout
:
while
time
.
monotonic
()
-
start
<
timeout
:
sock
=
socket
.
socket
(
socket
.
AF_INET
,
socket
.
SOCK_STREAM
)
sock
=
socket
.
socket
(
family
,
socket
.
SOCK_STREAM
)
sock
.
settimeout
(
time
.
monotonic
()
-
start
)
sock
.
settimeout
(
time
.
monotonic
()
-
start
)
try
:
try
:
sock
.
connect
(
(
'127.0.0.1'
,
25000
)
)
sock
.
connect
(
addr
)
sock
.
sendall
(
b'ping'
)
sock
.
sendall
(
b'ping'
)
if
sock
.
recv
(
4
):
if
sock
.
recv
(
4
):
print
(
'Server is up and running.'
)
print
(
'Server is up and running.'
)
...
@@ -126,33 +212,70 @@ def start_and_wait_for_server(server_cmd, timeout=60):
...
@@ -126,33 +212,70 @@ def start_and_wait_for_server(server_cmd, timeout=60):
sock
.
close
()
sock
.
close
()
return
server
return
server
kill_server
(
server
)
kill_server
()
abort
(
'Could not start server
\
n
'
+
abort
(
'Could not start server
\
n
'
+
'----------------------
\
n
'
+
'----------------------
\
n
'
+
'
\
n
\
n
'
.
join
(
server
.
communicate
()))
'
\
n
\
n
'
.
join
(
server
.
communicate
()))
def
kill_server
(
server
):
def
server_is_running
():
if
server
.
returncode
is
None
:
try
:
print
(
'Shutting down server...'
)
ret
=
subprocess
.
check_output
(
subprocess
.
check_output
([
'docker'
,
'stop'
,
'magicbench'
])
[
'docker'
,
'inspect'
,
'--type=container'
,
server
.
wait
()
'--format="{{ .State.Running }}"'
,
'magicbench'
],
stderr
=
subprocess
.
DEVNULL
,
universal_newlines
=
True
)
except
subprocess
.
CalledProcessError
:
return
False
else
:
return
ret
==
'true
\
n
'
def
server_container_exists
():
ret
=
subprocess
.
call
([
'docker'
,
'inspect'
,
'--type=container'
,
ret
=
subprocess
.
call
([
'docker'
,
'inspect'
,
'--type=container'
,
'magicbench'
],
'magicbench'
],
stdout
=
subprocess
.
DEVNULL
,
stdout
=
subprocess
.
DEVNULL
,
stderr
=
subprocess
.
DEVNULL
)
stderr
=
subprocess
.
DEVNULL
)
if
ret
==
0
:
return
ret
==
0
def
kill_server
():
if
server_is_running
():
print
(
'Shutting down server...'
)
subprocess
.
check_output
([
'docker'
,
'stop'
,
'magicbench'
])
if
server_container_exists
():
print
(
'Removing server container...'
)
subprocess
.
check_output
([
'docker'
,
'rm'
,
'magicbench'
])
subprocess
.
check_output
([
'docker'
,
'rm'
,
'magicbench'
])
def
main
():
def
main
():
parser
=
argparse
.
ArgumentParser
()
parser
=
argparse
.
ArgumentParser
()
parser
.
add_argument
(
'--duration'
,
'-
T
'
,
default
=
30
,
type
=
int
,
parser
.
add_argument
(
'--duration'
,
'-
D
'
,
default
=
30
,
type
=
int
,
help
=
'duration of each benchmark in seconds'
)
help
=
'duration of each benchmark in seconds'
)
parser
.
add_argument
(
'--benchmarks'
,
type
=
str
,
help
=
'comma-separated list of benchmarks to run'
)
parser
.
add_argument
(
'--save-json'
,
'-J'
,
default
=
'text'
,
type
=
str
,
help
=
'path to save benchmark results in JSON format'
)
args
=
parser
.
parse_args
()
args
=
parser
.
parse_args
()
if
not
os
.
path
.
exists
(
_socket
):
os
.
mkdir
(
_socket
)
if
args
.
benchmarks
:
benchmarks_to_run
=
args
.
benchmarks
.
split
(
','
)
else
:
benchmarks_to_run
=
{
b
[
'name'
]
for
b
in
benchmarks
}
benchmarks_data
=
[]
for
benchmark
in
benchmarks
:
for
benchmark
in
benchmarks
:
if
benchmark
[
'name'
]
not
in
benchmarks_to_run
:
continue
print
(
benchmark
[
'title'
])
print
(
benchmark
[
'title'
])
print
(
'='
*
len
(
benchmark
[
'title'
]))
print
(
'='
*
len
(
benchmark
[
'title'
]))
print
()
print
()
...
@@ -160,7 +283,7 @@ def main():
...
@@ -160,7 +283,7 @@ def main():
print
(
'Starting server...'
)
print
(
'Starting server...'
)
server_cmd
=
server_base
+
benchmark
[
'server'
]
server_cmd
=
server_base
+
benchmark
[
'server'
]
print
(
' '
+
' '
.
join
(
server_cmd
))
print
(
' '
+
' '
.
join
(
server_cmd
))
s
erver
=
start_and_wait_for_server
(
server_cmd
)
s
tart_and_wait_for_server
(
server_cmd
,
benchmark
[
'server_address'
]
)
print
()
print
()
print
(
'Warming up server...'
)
print
(
'Warming up server...'
)
...
@@ -171,6 +294,13 @@ def main():
...
@@ -171,6 +294,13 @@ def main():
duration
=
args
.
duration
duration
=
args
.
duration
benchmark_data
=
{
'name'
:
benchmark
[
'name'
],
'variations'
:
[]
}
benchmarks_data
.
append
(
benchmark_data
)
try
:
try
:
for
variation
in
benchmark
[
'variations'
]:
for
variation
in
benchmark
[
'variations'
]:
title
=
'BENCHMARK: {}'
.
format
(
variation
[
'title'
])
title
=
'BENCHMARK: {}'
.
format
(
variation
[
'title'
])
...
@@ -179,13 +309,38 @@ def main():
...
@@ -179,13 +309,38 @@ def main():
client_cmd
=
benchmark
[
'client'
]
+
variation
[
'args'
]
client_cmd
=
benchmark
[
'client'
]
+
variation
[
'args'
]
client_cmd
+=
[
'--duration={}'
.
format
(
duration
)]
client_cmd
+=
[
'--duration={}'
.
format
(
duration
)]
print
(
' '
.
join
(
client_cmd
))
print
(
' '
.
join
(
client_cmd
))
subprocess
.
check_call
(
client_cmd
)
output
=
subprocess
.
check_output
(
print
()
client_cmd
,
universal_newlines
=
True
)
data
=
json
.
loads
(
output
)
format_data
=
data
.
copy
()
format_data
[
'latency_percentiles'
]
=
'; '
.
join
(
'{}% under {}ms'
.
format
(
*
v
)
for
v
in
data
[
'latency_percentiles'
])
output
=
textwrap
.
dedent
(
'''
\
{messages} messages in {duration} seconds
Latency: min {latency_min}ms; max {latency_max}ms; mean {latency_mean}ms; std: {latency_std}ms ({latency_cv}%)
Latency distribtion: {latency_percentiles}
Requests/sec: {rps}
Transfer/sec: {transfer}MiB
'''
).
format
(
duration
=
duration
,
**
format_data
)
print
(
output
)
data
[
'name'
]
=
variation
[
'name'
]
benchmark_data
[
'variations'
].
append
(
data
)
finally
:
finally
:
kill_server
(
server
)
kill_server
()
print
()
print
()
if
args
.
save_json
:
with
open
(
args
.
save_json
,
'w'
)
as
f
:
json
.
dump
(
benchmarks_data
,
f
)
if
__name__
==
'__main__'
:
if
__name__
==
'__main__'
:
main
()
main
()
torecho.py
0 → 100644
View file @
b77b5f81
from
tornado.ioloop
import
IOLoop
from
tornado.tcpserver
import
TCPServer
class
StreamHandler
:
def
__init__
(
self
,
stream
):
self
.
_stream
=
stream
self
.
_stream
.
read_until_close
(
None
,
self
.
_handle_read
)
def
_handle_read
(
self
,
data
):
self
.
_stream
.
write
(
data
)
class
EchoServer
(
TCPServer
):
def
handle_stream
(
self
,
stream
,
address
):
StreamHandler
(
stream
)
if
__name__
==
'__main__'
:
server
=
EchoServer
()
server
.
bind
(
25000
)
server
.
start
(
1
)
IOLoop
.
instance
().
start
()
IOLoop
.
instance
().
close
()
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment