mirror of
https://github.com/project-zot/zot.git
synced 2025-01-06 22:40:28 -05:00
zb: pick client IPs from a pool, closes #472
Signed-off-by: Petu Eusebiu <peusebiu@cisco.com>
This commit is contained in:
parent
a5e091e3d2
commit
ca8b866c46
3 changed files with 154 additions and 114 deletions
|
@ -1,11 +1,29 @@
|
||||||
# `zb`
|
# `zb`
|
||||||
|
|
||||||
`zb` is a registry benchmarking tool which can run against any [distribution spec](https://github.com/opencontainers/distribution-spec) comformant registry.
|
## `zb` is a registry benchmarking tool which can run against any [distribution spec](https://github.com/opencontainers/distribution-spec) comformant registry.
|
||||||
|
|
||||||
-n : total number of requests
|
|
||||||
-c : number of concurrent clients performing (n/c) requests per client
|
```
|
||||||
-d : working dir to store test data
|
Usage:
|
||||||
-A : BASIC authentication in `username:passwd` format
|
zb [options] <url> [flags]
|
||||||
|
|
||||||
|
Flags:
|
||||||
|
-A, --auth-creds string Use colon-separated BASIC auth creds
|
||||||
|
-c, --concurrency int Number of multiple requests to make at a time (default 1)
|
||||||
|
-h, --help help for zb
|
||||||
|
-o, --output-format string Output format of test results: stdout (default), json, ci-cd
|
||||||
|
-r, --repo string Use specified repo on remote registry for test data
|
||||||
|
-n, --requests int Number of requests to perform (default 1)
|
||||||
|
-s, --src-cidr string Use specified cidr to obtain ips to make requests from, src-ips and src-cidr are mutually exclusive
|
||||||
|
-i, --src-ips string Use colon-separated ips to make requests from, src-ips and src-cidr are mutually exclusive
|
||||||
|
-v, --version Show the version and exit
|
||||||
|
-d, --working-dir string Use specified directory to store test data
|
||||||
|
```
|
||||||
|
|
||||||
|
## Command example
|
||||||
|
```
|
||||||
|
./bin/zb-linux-amd64 -c 10 -n 100 --src-cidr 127.0.0.0/8 -A user:pass http://localhost:8080
|
||||||
|
```
|
||||||
|
|
||||||
# References
|
# References
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,7 @@ import (
|
||||||
func NewPerfRootCmd() *cobra.Command {
|
func NewPerfRootCmd() *cobra.Command {
|
||||||
showVersion := false
|
showVersion := false
|
||||||
|
|
||||||
var auth, workdir, repo, outFmt string
|
var auth, workdir, repo, outFmt, srcIPs, srcCIDR string
|
||||||
|
|
||||||
var concurrency, requests int
|
var concurrency, requests int
|
||||||
|
|
||||||
|
@ -45,12 +45,16 @@ func NewPerfRootCmd() *cobra.Command {
|
||||||
|
|
||||||
requests = concurrency * (requests / concurrency)
|
requests = concurrency * (requests / concurrency)
|
||||||
|
|
||||||
Perf(workdir, url, auth, repo, concurrency, requests, outFmt)
|
Perf(workdir, url, auth, repo, concurrency, requests, outFmt, srcIPs, srcCIDR)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
rootCmd.Flags().StringVarP(&auth, "auth-creds", "A", "",
|
rootCmd.Flags().StringVarP(&auth, "auth-creds", "A", "",
|
||||||
"Use colon-separated BASIC auth creds")
|
"Use colon-separated BASIC auth creds")
|
||||||
|
rootCmd.Flags().StringVarP(&srcIPs, "src-ips", "i", "",
|
||||||
|
"Use colon-separated ips to make requests from, src-ips and src-cidr are mutually exclusive")
|
||||||
|
rootCmd.Flags().StringVarP(&srcCIDR, "src-cidr", "s", "",
|
||||||
|
"Use specified cidr to obtain ips to make requests from, src-ips and src-cidr are mutually exclusive")
|
||||||
rootCmd.Flags().StringVarP(&workdir, "working-dir", "d", "",
|
rootCmd.Flags().StringVarP(&workdir, "working-dir", "d", "",
|
||||||
"Use specified directory to store test data")
|
"Use specified directory to store test data")
|
||||||
rootCmd.Flags().StringVarP(&repo, "repo", "r", "",
|
rootCmd.Flags().StringVarP(&repo, "repo", "r", "",
|
||||||
|
|
232
cmd/zb/perf.go
232
cmd/zb/perf.go
|
@ -6,7 +6,9 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
|
"math/big"
|
||||||
mrand "math/rand"
|
mrand "math/rand"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
urlparser "net/url"
|
urlparser "net/url"
|
||||||
"os"
|
"os"
|
||||||
|
@ -36,9 +38,13 @@ const (
|
||||||
largeBlob = 100 * MiB
|
largeBlob = 100 * MiB
|
||||||
cicdFmt = "ci-cd"
|
cicdFmt = "ci-cd"
|
||||||
secureProtocol = "https"
|
secureProtocol = "https"
|
||||||
|
httpKeepAlive = 30 * time.Second
|
||||||
|
maxSourceIPs = 1000
|
||||||
|
httpTimeout = 30 * time.Second
|
||||||
|
TLSHandshakeTimeout = 10 * time.Second
|
||||||
)
|
)
|
||||||
|
|
||||||
// nolint:gochecknoglobals // used only in this test
|
// nolint:gochecknoglobals
|
||||||
var blobHash map[string]godigest.Digest = map[string]godigest.Digest{}
|
var blobHash map[string]godigest.Digest = map[string]godigest.Digest{}
|
||||||
|
|
||||||
// nolint:gochecknoglobals // used only in this test
|
// nolint:gochecknoglobals // used only in this test
|
||||||
|
@ -283,26 +289,11 @@ func normalizeProbabilityRange(pbty []float64) []float64 {
|
||||||
|
|
||||||
// test suites/funcs.
|
// test suites/funcs.
|
||||||
|
|
||||||
type testFunc func(workdir, url, auth, repo string, requests int, config testConfig, statsCh chan statsRecord) error
|
type testFunc func(workdir, url, repo string, requests int, config testConfig,
|
||||||
|
statsCh chan statsRecord, client *resty.Client) error
|
||||||
func GetCatalog(workdir, url, auth, repo string, requests int, config testConfig, statsCh chan statsRecord) error {
|
|
||||||
client := resty.New()
|
|
||||||
|
|
||||||
if auth != "" {
|
|
||||||
creds := strings.Split(auth, ":")
|
|
||||||
client.SetBasicAuth(creds[0], creds[1])
|
|
||||||
}
|
|
||||||
|
|
||||||
parsedURL, err := urlparser.Parse(url)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: gosec
|
|
||||||
if parsedURL.Scheme == secureProtocol {
|
|
||||||
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
|
||||||
}
|
|
||||||
|
|
||||||
|
func GetCatalog(workdir, url, repo string, requests int, config testConfig,
|
||||||
|
statsCh chan statsRecord, client *resty.Client) error {
|
||||||
for count := 0; count < requests; count++ {
|
for count := 0; count < requests; count++ {
|
||||||
func() {
|
func() {
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
@ -347,26 +338,8 @@ func GetCatalog(workdir, url, auth, repo string, requests int, config testConfig
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func PushMonolithStreamed(workdir, url, auth, trepo string, requests int,
|
func PushMonolithStreamed(workdir, url, trepo string, requests int, config testConfig,
|
||||||
config testConfig, statsCh chan statsRecord,
|
statsCh chan statsRecord, client *resty.Client) error {
|
||||||
) error {
|
|
||||||
client := resty.New()
|
|
||||||
|
|
||||||
if auth != "" {
|
|
||||||
creds := strings.Split(auth, ":")
|
|
||||||
client.SetBasicAuth(creds[0], creds[1])
|
|
||||||
}
|
|
||||||
|
|
||||||
parsedURL, err := urlparser.Parse(url)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: gosec
|
|
||||||
if parsedURL.Scheme == secureProtocol {
|
|
||||||
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
|
||||||
}
|
|
||||||
|
|
||||||
var repos []string
|
var repos []string
|
||||||
|
|
||||||
if config.mixedSize {
|
if config.mixedSize {
|
||||||
|
@ -379,7 +352,7 @@ func PushMonolithStreamed(workdir, url, auth, trepo string, requests int,
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean up
|
// clean up
|
||||||
err = deleteTestRepo(repos, url, client)
|
err := deleteTestRepo(repos, url, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -387,26 +360,8 @@ func PushMonolithStreamed(workdir, url, auth, trepo string, requests int,
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func PushChunkStreamed(workdir, url, auth, trepo string, requests int,
|
func PushChunkStreamed(workdir, url, trepo string, requests int, config testConfig,
|
||||||
config testConfig, statsCh chan statsRecord,
|
statsCh chan statsRecord, client *resty.Client) error {
|
||||||
) error {
|
|
||||||
client := resty.New()
|
|
||||||
|
|
||||||
if auth != "" {
|
|
||||||
creds := strings.Split(auth, ":")
|
|
||||||
client.SetBasicAuth(creds[0], creds[1])
|
|
||||||
}
|
|
||||||
|
|
||||||
parsedURL, err := urlparser.Parse(url)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: gosec
|
|
||||||
if parsedURL.Scheme == secureProtocol {
|
|
||||||
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
|
||||||
}
|
|
||||||
|
|
||||||
var repos []string
|
var repos []string
|
||||||
|
|
||||||
if config.mixedSize {
|
if config.mixedSize {
|
||||||
|
@ -419,7 +374,7 @@ func PushChunkStreamed(workdir, url, auth, trepo string, requests int,
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean up
|
// clean up
|
||||||
err = deleteTestRepo(repos, url, client)
|
err := deleteTestRepo(repos, url, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -427,26 +382,8 @@ func PushChunkStreamed(workdir, url, auth, trepo string, requests int,
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func Pull(workdir, url, auth, trepo string, requests int,
|
func Pull(workdir, url, trepo string, requests int,
|
||||||
config testConfig, statsCh chan statsRecord,
|
config testConfig, statsCh chan statsRecord, client *resty.Client) error {
|
||||||
) error {
|
|
||||||
client := resty.New()
|
|
||||||
|
|
||||||
if auth != "" {
|
|
||||||
creds := strings.Split(auth, ":")
|
|
||||||
client.SetBasicAuth(creds[0], creds[1])
|
|
||||||
}
|
|
||||||
|
|
||||||
parsedURL, err := urlparser.Parse(url)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: gosec
|
|
||||||
if parsedURL.Scheme == secureProtocol {
|
|
||||||
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
|
||||||
}
|
|
||||||
|
|
||||||
var repos []string
|
var repos []string
|
||||||
|
|
||||||
var manifestHash map[string]string
|
var manifestHash map[string]string
|
||||||
|
@ -465,7 +402,7 @@ func Pull(workdir, url, auth, trepo string, requests int,
|
||||||
largeSizeIdx := 2
|
largeSizeIdx := 2
|
||||||
|
|
||||||
// Push small blob
|
// Push small blob
|
||||||
manifestBySize, repos, err = pushMonolithImage(workdir, url, trepo, repos, smallBlob, client)
|
manifestBySize, repos, err := pushMonolithImage(workdir, url, trepo, repos, smallBlob, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -481,6 +418,7 @@ func Pull(workdir, url, auth, trepo string, requests int,
|
||||||
manifestBySizeHash[mediumSizeIdx] = manifestBySize
|
manifestBySizeHash[mediumSizeIdx] = manifestBySize
|
||||||
|
|
||||||
// Push large blob
|
// Push large blob
|
||||||
|
// nolint: ineffassign, staticcheck, wastedassign
|
||||||
manifestBySize, repos, err = pushMonolithImage(workdir, url, trepo, repos, largeBlob, client)
|
manifestBySize, repos, err = pushMonolithImage(workdir, url, trepo, repos, largeBlob, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -489,6 +427,7 @@ func Pull(workdir, url, auth, trepo string, requests int,
|
||||||
manifestBySizeHash[largeSizeIdx] = manifestBySize
|
manifestBySizeHash[largeSizeIdx] = manifestBySize
|
||||||
} else {
|
} else {
|
||||||
// Push blob given size
|
// Push blob given size
|
||||||
|
var err error
|
||||||
manifestHash, repos, err = pushMonolithImage(workdir, url, trepo, repos, config.size, client)
|
manifestHash, repos, err = pushMonolithImage(workdir, url, trepo, repos, config.size, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -506,7 +445,7 @@ func Pull(workdir, url, auth, trepo string, requests int,
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean up
|
// clean up
|
||||||
err = deleteTestRepo(repos, url, client)
|
err := deleteTestRepo(repos, url, client)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -514,26 +453,8 @@ func Pull(workdir, url, auth, trepo string, requests int,
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func MixedPullAndPush(workdir, url, auth, trepo string, requests int,
|
func MixedPullAndPush(workdir, url, trepo string, requests int,
|
||||||
config testConfig, statsCh chan statsRecord,
|
config testConfig, statsCh chan statsRecord, client *resty.Client) error {
|
||||||
) error {
|
|
||||||
client := resty.New()
|
|
||||||
|
|
||||||
if auth != "" {
|
|
||||||
creds := strings.Split(auth, ":")
|
|
||||||
client.SetBasicAuth(creds[0], creds[1])
|
|
||||||
}
|
|
||||||
|
|
||||||
parsedURL, err := urlparser.Parse(url)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// nolint: gosec
|
|
||||||
if parsedURL.Scheme == secureProtocol {
|
|
||||||
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
|
||||||
}
|
|
||||||
|
|
||||||
var repos []string
|
var repos []string
|
||||||
|
|
||||||
statusRequests = make(map[string]int)
|
statusRequests = make(map[string]int)
|
||||||
|
@ -674,7 +595,8 @@ var testSuite = []testConfig{ // nolint:gochecknoglobals // used only in this te
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt string) {
|
func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt string,
|
||||||
|
srcIPs string, srcCIDR string) {
|
||||||
json := jsoniter.ConfigCompatibleWithStandardLibrary
|
json := jsoniter.ConfigCompatibleWithStandardLibrary
|
||||||
// logging
|
// logging
|
||||||
log.SetFlags(0)
|
log.SetFlags(0)
|
||||||
|
@ -694,6 +616,19 @@ func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt
|
||||||
|
|
||||||
zbError := false
|
zbError := false
|
||||||
|
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// get host ips from command line to make requests from
|
||||||
|
var ips []string
|
||||||
|
if len(srcIPs) > 0 {
|
||||||
|
ips = strings.Split(srcIPs, ",")
|
||||||
|
} else if len(srcCIDR) > 0 {
|
||||||
|
ips, err = getIPsFromCIDR(srcCIDR, maxSourceIPs)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err) //nolint: gocritic
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for _, tconfig := range testSuite {
|
for _, tconfig := range testSuite {
|
||||||
statsCh := make(chan statsRecord, requests)
|
statsCh := make(chan statsRecord, requests)
|
||||||
|
|
||||||
|
@ -710,7 +645,12 @@ func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt
|
||||||
go func() {
|
go func() {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
|
|
||||||
_ = tconfig.tfunc(workdir, url, auth, repo, requests/concurrency, tconfig, statsCh)
|
httpClient, err := getRandomClientIPs(auth, url, ips)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
_ = tconfig.tfunc(workdir, url, repo, requests/concurrency, tconfig, statsCh, httpClient)
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
@ -743,7 +683,7 @@ func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt
|
||||||
if outFmt == cicdFmt {
|
if outFmt == cicdFmt {
|
||||||
jsonOut, err := json.Marshal(cicdSummary)
|
jsonOut, err := json.Marshal(cicdSummary)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err) // nolint:gocritic // file closed on exit
|
log.Fatal(err) // file closed on exit
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := ioutil.WriteFile(fmt.Sprintf("%s.json", outFmt), jsonOut, defaultFilePerms); err != nil {
|
if err := ioutil.WriteFile(fmt.Sprintf("%s.json", outFmt), jsonOut, defaultFilePerms); err != nil {
|
||||||
|
@ -755,3 +695,81 @@ func Perf(workdir, url, auth, repo string, concurrency int, requests int, outFmt
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// getRandomClientIPs returns a resty client with a random bind address from ips slice.
|
||||||
|
func getRandomClientIPs(auth string, url string, ips []string) (*resty.Client, error) {
|
||||||
|
client := resty.New()
|
||||||
|
|
||||||
|
if auth != "" {
|
||||||
|
creds := strings.Split(auth, ":")
|
||||||
|
client.SetBasicAuth(creds[0], creds[1])
|
||||||
|
}
|
||||||
|
|
||||||
|
// get random ip client
|
||||||
|
if len(ips) != 0 {
|
||||||
|
// get random number
|
||||||
|
nBig, err := crand.Int(crand.Reader, big.NewInt(int64(len(ips))))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// get random ip
|
||||||
|
ip := ips[nBig.Int64()]
|
||||||
|
|
||||||
|
// set ip in transport
|
||||||
|
localAddr, err := net.ResolveTCPAddr("tcp", fmt.Sprintf("%s:0", ip))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
transport := &http.Transport{
|
||||||
|
Proxy: http.ProxyFromEnvironment,
|
||||||
|
DialContext: (&net.Dialer{
|
||||||
|
Timeout: httpTimeout,
|
||||||
|
KeepAlive: httpKeepAlive,
|
||||||
|
LocalAddr: localAddr,
|
||||||
|
}).DialContext,
|
||||||
|
TLSHandshakeTimeout: TLSHandshakeTimeout,
|
||||||
|
}
|
||||||
|
|
||||||
|
client.SetTransport(transport)
|
||||||
|
}
|
||||||
|
|
||||||
|
parsedURL, err := urlparser.Parse(url)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// nolint: gosec
|
||||||
|
if parsedURL.Scheme == secureProtocol {
|
||||||
|
client.SetTLSClientConfig(&tls.Config{InsecureSkipVerify: true})
|
||||||
|
}
|
||||||
|
|
||||||
|
return client, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// getIPsFromCIDR returns a list of ips given a cidr.
|
||||||
|
func getIPsFromCIDR(cidr string, maxIPs int) ([]string, error) {
|
||||||
|
// nolint:varnamelen
|
||||||
|
ip, ipnet, err := net.ParseCIDR(cidr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var ips []string
|
||||||
|
for ip := ip.Mask(ipnet.Mask); ipnet.Contains(ip) && len(ips) < maxIPs; inc(ip) {
|
||||||
|
ips = append(ips, ip.String())
|
||||||
|
}
|
||||||
|
// remove network address and broadcast address
|
||||||
|
return ips[1 : len(ips)-1], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://go.dev/play/p/sdzcMvZYWnc
|
||||||
|
func inc(ip net.IP) {
|
||||||
|
for j := len(ip) - 1; j >= 0; j-- {
|
||||||
|
ip[j]++
|
||||||
|
if ip[j] > 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue