mirror of
https://github.com/dutchcoders/transfer.sh.git
synced 2025-01-04 00:20:18 +01:00
317 lines
7.5 KiB
Go
317 lines
7.5 KiB
Go
|
// Copyright 2018, OpenCensus Authors
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
package ochttp_test
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"net/http"
|
||
|
"net/http/httptest"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"testing"
|
||
|
|
||
|
"go.opencensus.io/plugin/ochttp"
|
||
|
"go.opencensus.io/stats/view"
|
||
|
"go.opencensus.io/trace"
|
||
|
)
|
||
|
|
||
|
const reqCount = 5
|
||
|
|
||
|
func TestClientNew(t *testing.T) {
|
||
|
server := httptest.NewServer(http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
|
||
|
resp.Write([]byte("Hello, world!"))
|
||
|
}))
|
||
|
defer server.Close()
|
||
|
|
||
|
if err := view.Register(
|
||
|
ochttp.ClientSentBytesDistribution,
|
||
|
ochttp.ClientReceivedBytesDistribution,
|
||
|
ochttp.ClientRoundtripLatencyDistribution,
|
||
|
ochttp.ClientCompletedCount,
|
||
|
); err != nil {
|
||
|
t.Fatalf("Failed to register ochttp.DefaultClientViews error: %v", err)
|
||
|
}
|
||
|
|
||
|
views := []string{
|
||
|
"opencensus.io/http/client/sent_bytes",
|
||
|
"opencensus.io/http/client/received_bytes",
|
||
|
"opencensus.io/http/client/roundtrip_latency",
|
||
|
"opencensus.io/http/client/completed_count",
|
||
|
}
|
||
|
for _, name := range views {
|
||
|
v := view.Find(name)
|
||
|
if v == nil {
|
||
|
t.Errorf("view not found %q", name)
|
||
|
continue
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var wg sync.WaitGroup
|
||
|
var tr ochttp.Transport
|
||
|
errs := make(chan error, reqCount)
|
||
|
wg.Add(reqCount)
|
||
|
|
||
|
for i := 0; i < reqCount; i++ {
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
req, err := http.NewRequest("POST", server.URL, strings.NewReader("req-body"))
|
||
|
if err != nil {
|
||
|
errs <- fmt.Errorf("error creating request: %v", err)
|
||
|
}
|
||
|
resp, err := tr.RoundTrip(req)
|
||
|
if err != nil {
|
||
|
errs <- fmt.Errorf("response error: %v", err)
|
||
|
}
|
||
|
if err := resp.Body.Close(); err != nil {
|
||
|
errs <- fmt.Errorf("error closing response body: %v", err)
|
||
|
}
|
||
|
if got, want := resp.StatusCode, 200; got != want {
|
||
|
errs <- fmt.Errorf("resp.StatusCode=%d; wantCount %d", got, want)
|
||
|
}
|
||
|
}()
|
||
|
}
|
||
|
|
||
|
go func() {
|
||
|
wg.Wait()
|
||
|
close(errs)
|
||
|
}()
|
||
|
|
||
|
for err := range errs {
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for _, viewName := range views {
|
||
|
v := view.Find(viewName)
|
||
|
if v == nil {
|
||
|
t.Errorf("view not found %q", viewName)
|
||
|
continue
|
||
|
}
|
||
|
rows, err := view.RetrieveData(v.Name)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
continue
|
||
|
}
|
||
|
if got, want := len(rows), 1; got != want {
|
||
|
t.Errorf("len(%q) = %d; want %d", viewName, got, want)
|
||
|
continue
|
||
|
}
|
||
|
data := rows[0].Data
|
||
|
var count int64
|
||
|
switch data := data.(type) {
|
||
|
case *view.CountData:
|
||
|
count = data.Value
|
||
|
case *view.DistributionData:
|
||
|
count = data.Count
|
||
|
default:
|
||
|
t.Errorf("Unknown data type: %v", data)
|
||
|
continue
|
||
|
}
|
||
|
if got := count; got != reqCount {
|
||
|
t.Fatalf("%s = %d; want %d", viewName, got, reqCount)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestClientOld(t *testing.T) {
|
||
|
server := httptest.NewServer(http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
|
||
|
resp.Write([]byte("Hello, world!"))
|
||
|
}))
|
||
|
defer server.Close()
|
||
|
|
||
|
if err := view.Register(ochttp.DefaultClientViews...); err != nil {
|
||
|
t.Fatalf("Failed to register ochttp.DefaultClientViews error: %v", err)
|
||
|
}
|
||
|
|
||
|
views := []string{
|
||
|
"opencensus.io/http/client/request_count",
|
||
|
"opencensus.io/http/client/latency",
|
||
|
"opencensus.io/http/client/request_bytes",
|
||
|
"opencensus.io/http/client/response_bytes",
|
||
|
}
|
||
|
for _, name := range views {
|
||
|
v := view.Find(name)
|
||
|
if v == nil {
|
||
|
t.Errorf("view not found %q", name)
|
||
|
continue
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var wg sync.WaitGroup
|
||
|
var tr ochttp.Transport
|
||
|
errs := make(chan error, reqCount)
|
||
|
wg.Add(reqCount)
|
||
|
|
||
|
for i := 0; i < reqCount; i++ {
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
req, err := http.NewRequest("POST", server.URL, strings.NewReader("req-body"))
|
||
|
if err != nil {
|
||
|
errs <- fmt.Errorf("error creating request: %v", err)
|
||
|
}
|
||
|
resp, err := tr.RoundTrip(req)
|
||
|
if err != nil {
|
||
|
errs <- fmt.Errorf("response error: %v", err)
|
||
|
}
|
||
|
if err := resp.Body.Close(); err != nil {
|
||
|
errs <- fmt.Errorf("error closing response body: %v", err)
|
||
|
}
|
||
|
if got, want := resp.StatusCode, 200; got != want {
|
||
|
errs <- fmt.Errorf("resp.StatusCode=%d; wantCount %d", got, want)
|
||
|
}
|
||
|
}()
|
||
|
}
|
||
|
|
||
|
go func() {
|
||
|
wg.Wait()
|
||
|
close(errs)
|
||
|
}()
|
||
|
|
||
|
for err := range errs {
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for _, viewName := range views {
|
||
|
v := view.Find(viewName)
|
||
|
if v == nil {
|
||
|
t.Errorf("view not found %q", viewName)
|
||
|
continue
|
||
|
}
|
||
|
rows, err := view.RetrieveData(v.Name)
|
||
|
if err != nil {
|
||
|
t.Error(err)
|
||
|
continue
|
||
|
}
|
||
|
if got, want := len(rows), 1; got != want {
|
||
|
t.Errorf("len(%q) = %d; want %d", viewName, got, want)
|
||
|
continue
|
||
|
}
|
||
|
data := rows[0].Data
|
||
|
var count int64
|
||
|
switch data := data.(type) {
|
||
|
case *view.CountData:
|
||
|
count = data.Value
|
||
|
case *view.DistributionData:
|
||
|
count = data.Count
|
||
|
default:
|
||
|
t.Errorf("Unknown data type: %v", data)
|
||
|
continue
|
||
|
}
|
||
|
if got := count; got != reqCount {
|
||
|
t.Fatalf("%s = %d; want %d", viewName, got, reqCount)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var noTrace = trace.StartOptions{Sampler: trace.NeverSample()}
|
||
|
|
||
|
func BenchmarkTransportNoTrace(b *testing.B) {
|
||
|
benchmarkClientServer(b, &ochttp.Transport{StartOptions: noTrace})
|
||
|
}
|
||
|
|
||
|
func BenchmarkTransport(b *testing.B) {
|
||
|
benchmarkClientServer(b, &ochttp.Transport{})
|
||
|
}
|
||
|
|
||
|
func benchmarkClientServer(b *testing.B, transport *ochttp.Transport) {
|
||
|
b.ReportAllocs()
|
||
|
ts := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
||
|
fmt.Fprintf(rw, "Hello world.\n")
|
||
|
}))
|
||
|
defer ts.Close()
|
||
|
transport.StartOptions.Sampler = trace.AlwaysSample()
|
||
|
var client http.Client
|
||
|
client.Transport = transport
|
||
|
b.ResetTimer()
|
||
|
|
||
|
for i := 0; i < b.N; i++ {
|
||
|
res, err := client.Get(ts.URL)
|
||
|
if err != nil {
|
||
|
b.Fatalf("Get: %v", err)
|
||
|
}
|
||
|
all, err := ioutil.ReadAll(res.Body)
|
||
|
res.Body.Close()
|
||
|
if err != nil {
|
||
|
b.Fatal("ReadAll:", err)
|
||
|
}
|
||
|
body := string(all)
|
||
|
if body != "Hello world.\n" {
|
||
|
b.Fatal("Got body:", body)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func BenchmarkTransportParallel64NoTrace(b *testing.B) {
|
||
|
benchmarkClientServerParallel(b, 64, &ochttp.Transport{StartOptions: noTrace})
|
||
|
}
|
||
|
|
||
|
func BenchmarkTransportParallel64(b *testing.B) {
|
||
|
benchmarkClientServerParallel(b, 64, &ochttp.Transport{})
|
||
|
}
|
||
|
|
||
|
func benchmarkClientServerParallel(b *testing.B, parallelism int, transport *ochttp.Transport) {
|
||
|
b.ReportAllocs()
|
||
|
ts := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
||
|
fmt.Fprintf(rw, "Hello world.\n")
|
||
|
}))
|
||
|
defer ts.Close()
|
||
|
|
||
|
var c http.Client
|
||
|
transport.Base = &http.Transport{
|
||
|
MaxIdleConns: parallelism,
|
||
|
MaxIdleConnsPerHost: parallelism,
|
||
|
}
|
||
|
transport.StartOptions.Sampler = trace.AlwaysSample()
|
||
|
c.Transport = transport
|
||
|
|
||
|
b.ResetTimer()
|
||
|
|
||
|
// TODO(ramonza): replace with b.RunParallel (it didn't work when I tried)
|
||
|
|
||
|
var wg sync.WaitGroup
|
||
|
wg.Add(parallelism)
|
||
|
for i := 0; i < parallelism; i++ {
|
||
|
iterations := b.N / parallelism
|
||
|
if i == 0 {
|
||
|
iterations += b.N % parallelism
|
||
|
}
|
||
|
go func() {
|
||
|
defer wg.Done()
|
||
|
for j := 0; j < iterations; j++ {
|
||
|
res, err := c.Get(ts.URL)
|
||
|
if err != nil {
|
||
|
b.Logf("Get: %v", err)
|
||
|
return
|
||
|
}
|
||
|
all, err := ioutil.ReadAll(res.Body)
|
||
|
res.Body.Close()
|
||
|
if err != nil {
|
||
|
b.Logf("ReadAll: %v", err)
|
||
|
return
|
||
|
}
|
||
|
body := string(all)
|
||
|
if body != "Hello world.\n" {
|
||
|
panic("Got body: " + body)
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
}
|
||
|
wg.Wait()
|
||
|
}
|