Performance in Python applications
@woakas
Ubidots
Gustavo Angulo
February 2020
woakas
Coffee / 5-7
13 years
2 sons
CTO / Co-founder
Ubidots
Ubidots
+ 40k Developers
+ 30 M per day
The problem
- Slowness
- Downs
- I don't know what's happening
- Bad experience for users
The source
Bad code
Database
Physical
Language
Human
Bad code
from flask import Flask, escape, request
import time
app = Flask(__name__)
@app.route("/add")
def add():
## Adding some time
time.sleep(1)
a = float(request.args.get("a", 0))
b = float(request.args.get("b", 0))
return "{}".format(a + b)
WRK
# lua
request = function()
wrk.method = "GET"
return wrk.format(
"GET", '/add?a=' .. math.random(0, 100) .. '&b=' .. math.random(0, 100)
)
end
# bash
$ wrk -s add_random.lua -c 50 -t 1 -d 10 http://localhost:5000
# Flask
FLASK_APP=slow_code.py flask run --reload --without-threads
Big O notation
https://www.geeksforgeeks.org/analysis-algorithms-big-o-analysis/
Bad code
from flask import Flask, escape, request
app = Flask(__name__)
@app.route("/add/cube")
def add():
t = 0
mn, mx = (0, 100)
a = int(request.args.get("size", 0))
for i in range(a):
for j in range(a):
for k in range(a):
t += i + j
return "{}".format(t)
Database
Database
- Limits
- # queries per second
- Joins
- Request / Queries
Django
devices = Device.objects.filter(owner__username='woakas')
for d in devices:
print(d.owner)
# 66 queries
devices = Device.objects.filter(owner__username='woakas').prefetch_related('owner')
for d in devices:
print(d.owner)
# 2 queries
Redis
➜ ~ redis-benchmark -t get
====== GET ======
100000 requests completed in 1.36 seconds
50 parallel clients
3 bytes payload
keep alive: 1
95.94% <= 1 milliseconds
99.97% <= 2 milliseconds
100.00% <= 2 milliseconds
73583.52 requests per second
Physical
Limits
Limits
fs.file-max=65535
net.core.somaxconn = 65536
net.ipv4.tcp_max_tw_buckets = 1440000
net.ipv4.tcp_fin_timeout = 30
net.ipv4.tcp_window_scaling = 1
net.ipv4.tcp_max_syn_backlog = 65536
Limits
Network
CPU
RAM
Disc
IO
Language
Sanic / async
from sanic import Sanic
from sanic.response import json
import time
app = Sanic()
@app.route("/")
async def test(request):
# time.sleep(1)
return json({"hello": "world"})
if __name__ == "__main__":
app.run(host="0.0.0.0", port=8000)
Languages
Human
Developer
Thank you
@woakas
woakas@ubidots.com
pycon2020
By Gustavo Angulo
pycon2020
- 830