Locust to Go, Kubernetes to Load
===
<!-- .slide: data-background="pink" -->
<!-- .slide: data-transition="zoom" -->
"Go" to "load" test :dizzy:
> [name=郭學聰 Hsueh-Tsung Kuo] [time=Sat, 11 Aug 2018] [color=red]
---
<!-- .slide: data-transition="convex" -->
## who am I?
![fieliapm](https://pbs.twimg.com/profile_images/591670980021387264/aZAYLRUe_400x400.png)
----
<!-- .slide: data-transition="convex" -->
* programmer from Rayark, a game company in Taiwan
* backend engineer, focus on common service
* usually develop something related to my work in Python, Ruby, Golang, C#
* built almost entire VOEZ game server by myself only
* supported Sdorica backend development
---
<!-- .slide: data-transition="convex" -->
## outline
----
<!-- .slide: data-transition="convex" -->
4. Locust <- Python
1. Locust?
2. usage
5. Locust -> Go
1. interoperability
2. boomer
6. play http service
7. docker & Kubernetes
1. docker
2. Kubernetes
----
<!-- .slide: data-transition="convex" -->
8. live coding
9. conclusion
10. commercial
11. Q&A
---
<!-- .slide: data-transition="convex" -->
## Locust <- Python
----
<!-- .slide: data-transition="convex" -->
### Locust?
----
<!-- .slide: data-transition="convex" -->
![locust](https://upload.wikimedia.org/wikipedia/commons/thumb/2/25/Locusta-migratoria-wanderheuschrecke.jpg/1280px-Locusta-migratoria-wanderheuschrecke.jpg =800x559)
----
<!-- .slide: data-transition="convex" -->
### Locust?
* official website: https://locust.io/
* Define user behaviour with Python code, and swarm your system with millions of simultaneous users.
----
<!-- .slide: data-transition="convex" -->
#### framework
```graphviz
digraph locust {
nodesep=1.0 // increases the separation between nodes
node [color="red" shape=box]
edge [color="blue" style=dashed]
master [label="master (coordinator)"]
client1 [label="client 1...n" shape="oval"]
client2 [label="client 1...n" shape="oval"]
client3 [label="client 1...n" shape="oval"]
master->slave1
master->slave2
master->slave3
slave1->client1 [style=solid]
slave2->client2 [style=solid]
slave3->client3 [style=solid]
client1->server
client1->server
client1->server
client1->server
client1->server
client2->server
client2->server
client2->server
client2->server
client2->server
client3->server
client3->server
client3->server
client3->server
client3->server
}
```
----
<!-- .slide: data-transition="convex" -->
#### why slave?
* one client vs Google Cloud load balancer
* :x: client lose
* connections limitation
* file descriptors limitation
* many clients vs Google Cloud load balancer
* :o: client may win
* ==**大家一起上!**== <!-- .element: class="fragment" data-fragment-index="2" -->
----
<!-- .slide: data-transition="convex" -->
### usage
----
<!-- .slide: data-transition="convex" -->
#### install
```shell
pip install locustio
```
----
<!-- .slide: data-transition="convex" -->
run Locust distributed across multiple processes/machines
```shell
pip install pyzmq
```
<!-- .element: class="fragment" data-fragment-index="1" -->
----
<!-- .slide: data-transition="convex" -->
#### code (simple)
```python=
from locust import HttpLocust, TaskSet, task
class WebsiteTasks(TaskSet):
def on_start(self):
self.client.post("/login", {
"username": "test_user",
"password": ""
})
@task
def index(self):
self.client.get("/")
@task
def about(self):
self.client.get("/about/")
class WebsiteUser(HttpLocust):
task_set = WebsiteTasks
min_wait = 5000
max_wait = 15000
```
----
<!-- .slide: data-transition="convex" -->
#### code (assign weight)
```python=
from locust import HttpLocust, TaskSet, task
class WebsiteTasks(TaskSet):
def on_start(self):
self.client.post("/login", {
"username": "test_user",
"password": ""
})
@task(100)
def index(self):
self.client.get("/")
@task(7)
def about(self):
self.client.get("/about/")
class WebsiteUser(HttpLocust):
task_set = WebsiteTasks
min_wait = 5000
max_wait = 15000
```
----
<!-- .slide: data-transition="convex" -->
#### build & run
```shell
locust -f locustfile.py --host=http://example.com
```
----
<!-- .slide: data-transition="convex" -->
#### monitoring
![monitoring](https://locust.io/static/img/screenshot.png)
---
<!-- .slide: data-transition="convex" -->
## Locust -> Go
----
<!-- .slide: data-transition="convex" -->
### interoperability
* Python?
* define tasks out of the box (standalone mode)
* foreign language?
* ZeroMQ (master/slave mode) <!-- .element: class="fragment" data-fragment-index="1" -->
----
<!-- .slide: data-transition="convex" -->
```sequence
locust master->locust slave: ZeroMQ: make a request
note left of locust slave: log start time
locust slave->server: HTTP request
server->locust slave: HTTP response
note left of locust slave: log end time
note left of locust slave: detect response status
locust slave->locust master: ZeroMQ: report status
```
----
<!-- .slide: data-transition="convex" -->
#### why Go?
* fast <!-- .element: class="fragment" data-fragment-index="1" -->
* faster <!-- .element: class="fragment" data-fragment-index="2" -->
* fastest <!-- .element: class="fragment" data-fragment-index="3" -->
----
<!-- .slide: data-transition="convex" -->
#### why Go?
* server is written with Go
* client bots written with Python cannot be as fast as Go
----
<!-- .slide: data-transition="convex" -->
### boomer
* boomer: https://github.com/myzhan/boomer
* Go module
* load generator for Locust
* be a slave
* connect with Locust master via ZeroMQ
* dependency:
* https://github.com/zeromq/gomq
----
<!-- .slide: data-transition="convex" -->
#### code (simple)
`dummy.py`
```python=
# -*- coding: utf-8 -*-
from locust import Locust, TaskSet, task
class MyTaskSet(TaskSet):
@task(20)
def hello(self):
pass
class Dummy(Locust):
task_set = MyTaskSet
```
----
<!-- .slide: data-transition="convex" -->
`main.go`
```go=
package main
import "github.com/myzhan/boomer"
import "time"
func foo() {
start := boomer.Now()
time.Sleep(100 * time.Millisecond)
elapsed := boomer.Now() - start
// Report your test result as a success, if you write it in python, it will looks like this
// events.request_success.fire(request_type="http", name="foo", response_time=100, response_length=10)
boomer.Events.Publish("request_success", "http", "foo", elapsed, int64(10))
}
```
----
<!-- .slide: data-transition="convex" -->
`main.go`
```go=
func bar() {
start := boomer.Now()
time.Sleep(100 * time.Millisecond)
elapsed := boomer.Now() - start
// Report your test result as a failure, if you write it in python, it will looks like this
// events.request_failure.fire(request_type="udp", name="bar", response_time=100, exception=Exception("udp error"))
boomer.Events.Publish("request_failure", "udp", "bar", elapsed, "udp error")
}
```
----
<!-- .slide: data-transition="convex" -->
`main.go`
```go=
func main() {
task1 := &boomer.Task{
Name: "foo",
Weight: 10,
Fn: foo,
}
task2 := &boomer.Task{
Name: "bar",
Weight: 20,
Fn: bar,
}
boomer.Run(task1, task2)
}
```
----
<!-- .slide: data-transition="convex" -->
#### finite state machine
* randomized
* finance
* social network
* \.\.\.\.\.\. etc
* ordered
* game
----
<!-- .slide: data-transition="convex" -->
#### code (finite state machine)
`main_seq.go`
```go=
package main
import "github.com/myzhan/boomer"
import "time"
func foobar(){
start := boomer.Now()
time.Sleep(100 * time.Millisecond)
elapsed := boomer.Now() - start
// Report your test result as a success, if you write it in python, it will looks like this
// events.request_success.fire(request_type="http", name="foo", response_time=100, response_length=10)
boomer.Events.Publish("request_success", "http", "foo", elapsed, int64(10))
start = boomer.Now()
time.Sleep(100 * time.Millisecond)
elapsed = boomer.Now() - start
// Report your test result as a failure, if you write it in python, it will looks like this
// events.request_failure.fire(request_type="udp", name="bar", response_time=100, exception=Exception("udp error"))
boomer.Events.Publish("request_failure", "udp", "bar", elapsed, "udp error")
}
```
----
<!-- .slide: data-transition="convex" -->
`main_seq.go`
```go=
func main(){
sequencialTask1 := &boomer.Task{
Name: "foobar1",
Weight: 10,
Fn: foobar,
}
sequencialTask2 := &boomer.Task{
Name: "foobar2",
Weight: 20,
Fn: foobar,
}
boomer.Run(sequencialTask1, sequencialTask2)
}
```
----
<!-- .slide: data-transition="convex" -->
#### run as stand alone process (for debug)
```shell
go build -o a.out main.go
./a.out --run-tasks foo,bar
```
----
<!-- .slide: data-transition="convex" -->
#### run as slave mode
```shell
locust -f dummy.py --master --master-bind-host=127.0.0.1 --master-bind-port=5557
go build -o a.out main.go
./a.out --master-host=127.0.0.1 --master-port=5557 --rpc=zeromq
```
---
<!-- .slide: data-transition="convex" -->
## play http service
----
<!-- .slide: data-transition="convex" -->
```go=
resp, err := http.Get("http://example.com/")
...
resp, err := http.Post("http://example.com/upload", "image/jpeg", &buf)
...
resp, err := http.PostForm("http://example.com/form",
url.Values{"key": {"Value"}, "id": {"123"}})
```
----
<!-- .slide: data-transition="convex" -->
```go=
func exampleGet() {
start := boomer.Now()
resp, err := http.Get("https://dl.google.com/go/go1.10.3.src.tar.gz")
//resp, err := http.Get("https://cache.ruby-lang.org/pub/ruby/2.5/ruby-2.5.1.tar.gz")
if err == nil {
func() {
defer resp.Body.Close()
_, err = ioutil.ReadAll(resp.Body)
}()
}
elapsed := boomer.Now() - start
if err == nil {
boomer.Events.Publish("request_success", "http", "example_get", elapsed, resp.ContentLength)
} else {
boomer.Events.Publish("request_failure", "http", "example_get", elapsed, err.Error())
}
}
```
---
<!-- .slide: data-transition="convex" -->
## docker & Kubernetes
----
<!-- .slide: data-transition="convex" -->
### docker
----
<!-- .slide: data-transition="convex" -->
### docker
* multiple machine
* break connections limitation
* break file descriptors limitation
* multiple docker instance
* break limitation
* **easy scale** <!-- .element: class="fragment" data-fragment-index="1" -->
----
<!-- .slide: data-transition="convex" -->
### Kubernetes
----
<!-- .slide: data-transition="convex" -->
### Kubernetes
* ~~kill~~ control them all!!
----
<!-- .slide: data-transition="convex" -->
### Kubernetes
![kubernetes](https://d33wubrfki0l68.cloudfront.net/1567471e7c58dc9b7d9c65dcd54e60cbf5870daa/a2249/images/flower.png)
----
<!-- .slide: data-transition="convex" -->
#### kubectl scale --replicas=1 locust-slave
```graphviz
digraph locust {
nodesep=1.0 // increases the separation between nodes
node [color="red" shape=box]
edge [color="blue" style=dashed]
master [label="master (coordinator)"]
client1 [label="client 1...n" shape="oval"]
master->slave1
slave1->client1 [style=solid]
client1->server
client1->server
client1->server
client1->server
client1->server
}
```
----
<!-- .slide: data-transition="convex" -->
#### kubectl scale --replicas=2 locust-slave
```graphviz
digraph locust {
nodesep=1.0 // increases the separation between nodes
node [color="red" shape=box]
edge [color="blue" style=dashed]
master [label="master (coordinator)"]
client1 [label="client 1...n" shape="oval"]
client2 [label="client 1...n" shape="oval"]
master->slave1
master->slave2
slave1->client1 [style=solid]
slave2->client2 [style=solid]
client1->server
client1->server
client1->server
client1->server
client1->server
client2->server
client2->server
client2->server
client2->server
client2->server
}
```
----
<!-- .slide: data-transition="convex" -->
#### kubectl scale --replicas=3 locust-slave
```graphviz
digraph locust {
nodesep=1.0 // increases the separation between nodes
node [color="red" shape=box]
edge [color="blue" style=dashed]
master [label="master (coordinator)"]
client1 [label="client 1...n" shape="oval"]
client2 [label="client 1...n" shape="oval"]
client3 [label="client 1...n" shape="oval"]
master->slave1
master->slave2
master->slave3
slave1->client1 [style=solid]
slave2->client2 [style=solid]
slave3->client3 [style=solid]
client1->server
client1->server
client1->server
client1->server
client1->server
client2->server
client2->server
client2->server
client2->server
client2->server
client3->server
client3->server
client3->server
client3->server
client3->server
}
```
---
<!-- .slide: data-transition="convex" -->
## live coding
----
<!-- .slide: data-transition="convex" -->
never live coding
¯\\\_\(ツ\)\_/¯
----
<!-- .slide: data-transition="convex" -->
~~never~~ live coding
( O\_o) ?
----
<!-- .slide: data-transition="convex" -->
重新定義
"living" live coding
( @д@) !
----
<!-- .slide: data-transition="convex" -->
just do it
( >ω•)b
---
<!-- .slide: data-transition="convex" -->
## conclusion
----
<!-- .slide: data-transition="convex" -->
> :hash: "倒站即進化"
> [name=郭學聰 Hsueh-Tsung Kuo] [time=2018_08_11] [color=red]
----
<!-- .slide: data-transition="convex" -->
### special thanks
* backend: yuxioz, bluesky
* SRE: haraguroicha, hyww, mswu
* QA: xible
* others
---
<!-- .slide: data-transition="zoom" -->
## commercial
Sdorica -sunset-
* <small>OP Animation https://youtu.be/ktGRO8tCBqs</small>
* <small>Global Launch Trailer https://youtu.be/2aY7il7zuNk</small>
* <small>Background Story https://youtu.be/9tOmgueeY7I</small>
----
<!-- .slide: data-transition="zoom" -->
{%youtube ktGRO8tCBqs %}
----
<!-- .slide: data-transition="zoom" -->
# Rayark Wants You !
Rayark Careers | Make A Difference
* EN https://careers.rayark.com/jobs/
* ZH https://careers.rayark.com/zh/jobs/
----
<!-- .slide: data-transition="zoom" -->
# Meet Rayark in COSCUP 攤位區
* 產品展示
* 精美小物
---
<!-- .slide: data-transition="zoom" -->
## Q&A
---
<style>
.reveal {
background: #FFDFEF;
color: black;
}
.reveal h2,
.reveal h3,
.reveal h4 {
color: black;
}
.reveal code {
font-size: 14px !important;
line-height: 1.2;
}
.rightpart{
float:right;
width:50%;
}
.leftpart{
margin-right: 50% !important;
height:50%;
}
.reveal section img { background:none; border:none; box-shadow:none; }
p.blo {
font-size: 50px !important;
background:#B6BDBB;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 10px;
box-shadow: 5px 5px 5px #666;
}
p.blo1 {
background: #c7c2bb;
}
p.blo2 {
background: #b8c0c8;
}
p.blo3 {
background: #c7cedd;
}
p.bloT {
font-size: 60px !important;
background:#B6BDD3;
border:1px solid silver;
display:inline-block;
padding:0.5em 0.75em;
border-radius: 8px;
box-shadow: 1px 2px 5px #333;
}
p.bloA {
background: #B6BDE3;
}
p.bloB {
background: #E3BDB3;
}
.slide-number{
margin-bottom:10px !important;
width:100%;
text-align:center;
font-size:25px !important;
background-color:transparent !important;
}
iframe.myclass{
width:100px;
height:100px;
bottom:0;
left:0;
position:fixed;
border:none;
z-index:99999;
}
h1.raw {
color: #fff;
background-image: linear-gradient(90deg,#f35626,#feab3a);
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
animation: hue 5s infinite linear;
}
@keyframes hue {
from {
filter: hue-rotate(0deg);
}
to {
filter: hue-rotate(360deg);
}
}
.progress{
height:14px !important;
}
.progress span{
height:14px !important;
background: url("") repeat-x !important;
}
.progress span:after,
.progress span.nyancat{
content: "";
background: url('') !important;
width: 34px !important;
height: 21px !important;
border: none !important;
float:right;
margin-top:-7px;
margin-right:-10px;
}
</style>
{"metaMigratedAt":"2023-06-14T17:30:59.469Z","metaMigratedFrom":"Content","title":"Locust to Go, Kubernetes to Load","breaks":true,"contributors":"[{\"id\":\"ea27dcd7-a3f2-47c2-b25e-6760e7936c38\",\"add\":23538,\"del\":1175}]"}