// Copyright 2019 Google LLC. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // +build go1.11,linux package grpc import ( "context" "errors" "fmt" "net" "os" "syscall" "testing" "time" "golang.org/x/oauth2" "golang.org/x/sys/unix" "google.golang.org/api/option" "google.golang.org/grpc" ) func TestDialTCPUserTimeout(t *testing.T) { l, err := net.Listen("tcp", ":3000") if err != nil { t.Fatal(err) } defer l.Close() acceptErrCh := make(chan error, 1) go func() { conn, err := l.Accept() if err != nil { acceptErrCh <- err return } defer conn.Close() if err := conn.Close(); err != nil { acceptErrCh <- err } }() conn, err := dialTCPUserTimeout(context.Background(), ":3000") if err != nil { t.Fatal(err) } defer conn.Close() timeout, err := getTCPUserTimeout(conn) if err != nil { t.Fatal(err) } if timeout != tcpUserTimeoutMilliseconds { t.Fatalf("expected %v, got %v", tcpUserTimeoutMilliseconds, timeout) } select { case err := <-acceptErrCh: t.Fatalf("Accept failed with: %v", err) default: } } func getTCPUserTimeout(conn net.Conn) (int, error) { tcpconn, ok := conn.(*net.TCPConn) if !ok { return 0, fmt.Errorf("conn is not *net.TCPConn. got %T", conn) } rawConn, err := tcpconn.SyscallConn() if err != nil { return 0, err } var timeout int var syscalErr error controlErr := rawConn.Control(func(fd uintptr) { timeout, syscalErr = syscall.GetsockoptInt(int(fd), syscall.IPPROTO_TCP, unix.TCP_USER_TIMEOUT) }) if syscalErr != nil { return 0, syscalErr } if controlErr != nil { return 0, controlErr } return timeout, nil } // Check that tcp timeout dialer overwrites user defined dialer. func TestDialWithDirectPathEnabled(t *testing.T) { os.Setenv("GOOGLE_CLOUD_ENABLE_DIRECT_PATH", "example,other") defer os.Clearenv() ctx, cancel := context.WithTimeout(context.Background(), 10*time.Millisecond) userDialer := grpc.WithDialer(func(addr string, timeout time.Duration) (net.Conn, error) { t.Error("did not expect a call to user dialer, got one") cancel() return nil, errors.New("not expected") }) conn, err := Dial(ctx, option.WithTokenSource(oauth2.StaticTokenSource(nil)), // No creds. option.WithGRPCDialOption(userDialer), option.WithEndpoint("example.google.com:443")) if err != nil { t.Errorf("DialGRPC: error %v, want nil", err) } defer conn.Close() // gRPC doesn't connect before the first call. grpc.Invoke(ctx, "foo", nil, nil, conn) }