// Copyright 2020 Google LLC
//
// 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
//
//     https://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.

// Code generated by protoc-gen-go_gapic. DO NOT EDIT.

package connection_test

import (
	"context"

	connection "cloud.google.com/go/bigquery/connection/apiv1"
	"google.golang.org/api/iterator"
	connectionpb "google.golang.org/genproto/googleapis/cloud/bigquery/connection/v1"
	iampb "google.golang.org/genproto/googleapis/iam/v1"
)

func ExampleNewClient() {
	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use client.
	_ = c
}

func ExampleClient_CreateConnection() {
	// import connectionpb "google.golang.org/genproto/googleapis/cloud/bigquery/connection/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &connectionpb.CreateConnectionRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.CreateConnection(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}

func ExampleClient_GetConnection() {
	// import connectionpb "google.golang.org/genproto/googleapis/cloud/bigquery/connection/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &connectionpb.GetConnectionRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.GetConnection(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}

func ExampleClient_ListConnections() {
	// import connectionpb "google.golang.org/genproto/googleapis/cloud/bigquery/connection/v1"
	// import "google.golang.org/api/iterator"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &connectionpb.ListConnectionsRequest{
		// TODO: Fill request struct fields.
	}
	it := c.ListConnections(ctx, req)
	for {
		resp, err := it.Next()
		if err == iterator.Done {
			break
		}
		if err != nil {
			// TODO: Handle error.
		}
		// TODO: Use resp.
		_ = resp
	}
}

func ExampleClient_UpdateConnection() {
	// import connectionpb "google.golang.org/genproto/googleapis/cloud/bigquery/connection/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &connectionpb.UpdateConnectionRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.UpdateConnection(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}

func ExampleClient_DeleteConnection() {
	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &connectionpb.DeleteConnectionRequest{
		// TODO: Fill request struct fields.
	}
	err = c.DeleteConnection(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
}

func ExampleClient_GetIamPolicy() {
	// import iampb "google.golang.org/genproto/googleapis/iam/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &iampb.GetIamPolicyRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.GetIamPolicy(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}

func ExampleClient_SetIamPolicy() {
	// import iampb "google.golang.org/genproto/googleapis/iam/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &iampb.SetIamPolicyRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.SetIamPolicy(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}

func ExampleClient_TestIamPermissions() {
	// import iampb "google.golang.org/genproto/googleapis/iam/v1"

	ctx := context.Background()
	c, err := connection.NewClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

	req := &iampb.TestIamPermissionsRequest{
		// TODO: Fill request struct fields.
	}
	resp, err := c.TestIamPermissions(ctx, req)
	if err != nil {
		// TODO: Handle error.
	}
	// TODO: Use resp.
	_ = resp
}
