Skip to content

Commit

Permalink
Resolve gun targets in phttp component once before shooting.
Browse files Browse the repository at this point in the history
  • Loading branch information
skipor committed Oct 1, 2017
1 parent bc61bf6 commit 2363e41
Show file tree
Hide file tree
Showing 14 changed files with 602 additions and 32 deletions.
20 changes: 20 additions & 0 deletions components/example/import/import_suite_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package example

import (
"testing"

. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"

"github.com/yandex/pandora/lib/testutil"
)

func TestImport(t *testing.T) {
testutil.RunSuite(t, "Import Suite")
}

var _ = Describe("import", func() {
It("not panics", func() {
Expect(Import).NotTo(Panic())
})
})
27 changes: 12 additions & 15 deletions components/phttp/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@
package phttp

import (
"context"
"crypto/tls"
"net"
"net/http"
Expand All @@ -17,6 +16,7 @@ import (

"github.com/pkg/errors"
"github.com/yandex/pandora/core/config"
"github.com/yandex/pandora/lib/netutil"
)

//go:generate mockery -name=Client -case=underscore -inpkg -testonly
Expand All @@ -43,6 +43,8 @@ func NewDefaultClientConfig() ClientConfig {
// DialerConfig can be mapped on net.Dialer.
// Set net.Dialer for details.
type DialerConfig struct {
DNSCache bool `config:"dns-cache" map:"-"`

Timeout time.Duration `config:"timeout"`
DualStack bool `config:"dual-stack"`

Expand All @@ -54,25 +56,20 @@ type DialerConfig struct {

func NewDefaultDialerConfig() DialerConfig {
return DialerConfig{
DNSCache: true,
DualStack: true,
Timeout: 3 * time.Second,
KeepAlive: 120 * time.Second,
}
}

type Dialer interface {
DialContext(ctx context.Context, network, address string) (net.Conn, error)
}

type DialerFunc func(ctx context.Context, network, address string) (net.Conn, error)

func (f DialerFunc) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
return f(ctx, network, address)
}

func NewDialer(conf DialerConfig) *net.Dialer {
func NewDialer(conf DialerConfig) netutil.Dialer {
d := &net.Dialer{}
config.Map(d, conf)
return d
if !conf.DNSCache {
return d
}
return netutil.NewDNSCachingDialer(d, netutil.DefaultDNSCache)
}

// TransportConfig can be mapped on http.Transport.
Expand All @@ -98,7 +95,7 @@ func NewDefaultTransportConfig() TransportConfig {
}
}

func NewTransport(conf TransportConfig, dial DialerFunc) *http.Transport {
func NewTransport(conf TransportConfig, dial netutil.DialerFunc) *http.Transport {
tr := &http.Transport{}
tr.TLSClientConfig = &tls.Config{
InsecureSkipVerify: true, // We should not spend time for this stuff.
Expand All @@ -109,7 +106,7 @@ func NewTransport(conf TransportConfig, dial DialerFunc) *http.Transport {
return tr
}

func NewHTTP2Transport(conf TransportConfig, dial DialerFunc) *http.Transport {
func NewHTTP2Transport(conf TransportConfig, dial netutil.DialerFunc) *http.Transport {
tr := NewTransport(conf, dial)
err := http2.ConfigureTransport(tr)
if err != nil {
Expand Down
3 changes: 2 additions & 1 deletion components/phttp/connect.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import (
"net/url"

"github.com/pkg/errors"
"github.com/yandex/pandora/lib/netutil"
)

type ConnectGunConfig struct {
Expand Down Expand Up @@ -78,7 +79,7 @@ func newConnectClient(conf ConnectGunConfig) Client {
return newClient(transport, conf.Client.Redirect)
}

func newConnectDialFunc(target string, connectSSL bool, dialer Dialer) DialerFunc {
func newConnectDialFunc(target string, connectSSL bool, dialer netutil.Dialer) netutil.DialerFunc {
return func(ctx context.Context, network, address string) (conn net.Conn, err error) {
// TODO(skipor): make connect sample.
// TODO(skipor): make httptrace callbacks called correctly.
Expand Down
2 changes: 1 addition & 1 deletion components/phttp/http.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ func NewHTTPGun(conf HTTPGunConfig) *HTTPGun {
// NewHTTP2Gun return simple HTTP/2 gun that can shoot sequentially through one connection.
func NewHTTP2Gun(conf HTTP2GunConfig) (*HTTPGun, error) {
if !conf.Gun.SSL {
// Open issue on github if you need this feature.
// Open issue on github if you really need this feature.
return nil, errors.New("HTTP/2.0 over TCP is not supported. Please leave SSL option true by default.")
}
transport := NewHTTP2Transport(conf.Client.Transport, NewDialer(conf.Client.Dialer).DialContext)
Expand Down
58 changes: 51 additions & 7 deletions components/phttp/import/import.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,18 @@
package phttp

import (
"net"

"github.com/spf13/afero"
"go.uber.org/zap"

. "github.com/yandex/pandora/components/phttp"
"github.com/yandex/pandora/components/phttp/ammo/simple/jsonline"
"github.com/yandex/pandora/components/phttp/ammo/simple/raw"
"github.com/yandex/pandora/components/phttp/ammo/simple/uri"
"github.com/yandex/pandora/core"
"github.com/yandex/pandora/core/register"
"github.com/yandex/pandora/lib/netutil"
)

func Import(fs afero.Fs) {
Expand All @@ -29,16 +33,56 @@ func Import(fs afero.Fs) {
return raw.NewProvider(fs, conf)
})

register.Gun("http", func(conf HTTPGunConfig) core.Gun {
return WrapGun(NewHTTPGun(conf))
register.Gun("http", func(conf HTTPGunConfig) func() core.Gun {
preResolveTargetAddr(&conf.Client, &conf.Gun.Target)
return func() core.Gun { return WrapGun(NewHTTPGun(conf)) }
}, NewDefaultHTTPGunConfig)

register.Gun("http2", func(conf HTTP2GunConfig) (core.Gun, error) {
gun, err := NewHTTP2Gun(conf)
return WrapGun(gun), err
register.Gun("http2", func(conf HTTP2GunConfig) func() (core.Gun, error) {
preResolveTargetAddr(&conf.Client, &conf.Gun.Target)
return func() (core.Gun, error) {
gun, err := NewHTTP2Gun(conf)
return WrapGun(gun), err
}
}, NewDefaultHTTP2GunConfig)

register.Gun("connect", func(conf ConnectGunConfig) core.Gun {
return WrapGun(NewConnectGun(conf))
register.Gun("connect", func(conf ConnectGunConfig) func() core.Gun {
preResolveTargetAddr(&conf.Client, &conf.Target)
return func() core.Gun {
return WrapGun(NewConnectGun(conf))
}
}, NewDefaultConnectGunConfig)
}

// DNS resolve optimisation.
// When DNSCache turned off - do nothing extra, host will be resolved on every shoot.
// When using resolved target, don't use DNS caching logic - it is useless.
// If we can resolve accessible target addr - use it as target, not use caching.
// Otherwise just use DNS cache - we should not fail shooting, we should try to
// connect on every shoot. DNS cache will save resolved addr after first successful connect.
func preResolveTargetAddr(clientConf *ClientConfig, target *string) (err error) {
if !clientConf.Dialer.DNSCache {
return
}
if endpointIsResolved(*target) {
clientConf.Dialer.DNSCache = false
return
}
resolved, err := netutil.LookupReachable(*target)
if err != nil {
zap.L().Warn("DNS target pre resolve failed",
zap.String("target", *target), zap.Error(err))
return
}
clientConf.Dialer.DNSCache = false
*target = resolved
return
}

func endpointIsResolved(endpoint string) bool {
host, _, err := net.SplitHostPort(endpoint)
if err != nil {
return false
}
return net.ParseIP(host) != nil
}
72 changes: 72 additions & 0 deletions components/phttp/import/import_suite_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
package phttp

import (
"net"
"strconv"
"testing"

. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/spf13/afero"

. "github.com/yandex/pandora/components/phttp"
"github.com/yandex/pandora/lib/testutil"
)

func TestImport(t *testing.T) {
testutil.RunSuite(t, "phttp Import Suite")
}

var _ = Describe("import", func() {
It("not panics", func() {
Expect(func() {
Import(afero.NewOsFs())
}).NotTo(Panic())
})
})

var _ = Describe("preResolveTargetAddr", func() {
It("host target", func() {
conf := &ClientConfig{}
conf.Dialer.DNSCache = true

listener, err := net.ListenTCP("tcp6", nil)
defer listener.Close()
Expect(err).NotTo(HaveOccurred())

port := strconv.Itoa(listener.Addr().(*net.TCPAddr).Port)
target := "localhost:" + port
expectedResolved := "[::1]:" + port

err = preResolveTargetAddr(conf, &target)
Expect(err).NotTo(HaveOccurred())
Expect(conf.Dialer.DNSCache).To(BeFalse())

Expect(target).To(Equal(expectedResolved))
})

It("ip target", func() {
conf := &ClientConfig{}
conf.Dialer.DNSCache = true

const addr = "127.0.0.1:80"
target := addr
err := preResolveTargetAddr(conf, &target)
Expect(err).NotTo(HaveOccurred())
Expect(conf.Dialer.DNSCache).To(BeFalse())
Expect(target).To(Equal(addr))
})

It("failed", func() {
conf := &ClientConfig{}
conf.Dialer.DNSCache = true

const addr = "localhost:54321"
target := addr
err := preResolveTargetAddr(conf, &target)
Expect(err).To(HaveOccurred())
Expect(conf.Dialer.DNSCache).To(BeTrue())
Expect(target).To(Equal(addr))
})

})
8 changes: 1 addition & 7 deletions core/import/import_suite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ import (

. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"github.com/onsi/gomega/format"
"github.com/spf13/afero"

"github.com/yandex/pandora/core"
Expand All @@ -16,13 +15,8 @@ import (
)

func TestImport(t *testing.T) {
format.UseStringerRepresentation = true
RegisterFailHandler(Fail)

testutil.ReplaceGlobalLogger()
Import(afero.NewOsFs())

RunSpecs(t, "Import Suite")
testutil.RunSuite(t, "Import Suite")
}

var _ = Describe("plugin decode", func() {
Expand Down
Loading

0 comments on commit 2363e41

Please sign in to comment.