// 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 monitoring_test

import (
	"context"

	monitoring "cloud.google.com/go/monitoring/apiv3"
	"google.golang.org/api/iterator"
	monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"
)

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

func ExampleNotificationChannelClient_ListNotificationChannelDescriptors() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"
	// import "google.golang.org/api/iterator"

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

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

func ExampleNotificationChannelClient_GetNotificationChannelDescriptor() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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

func ExampleNotificationChannelClient_ListNotificationChannels() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"
	// import "google.golang.org/api/iterator"

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

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

func ExampleNotificationChannelClient_GetNotificationChannel() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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

func ExampleNotificationChannelClient_CreateNotificationChannel() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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

func ExampleNotificationChannelClient_UpdateNotificationChannel() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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

func ExampleNotificationChannelClient_DeleteNotificationChannel() {
	ctx := context.Background()
	c, err := monitoring.NewNotificationChannelClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

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

func ExampleNotificationChannelClient_SendNotificationChannelVerificationCode() {
	ctx := context.Background()
	c, err := monitoring.NewNotificationChannelClient(ctx)
	if err != nil {
		// TODO: Handle error.
	}

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

func ExampleNotificationChannelClient_GetNotificationChannelVerificationCode() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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

func ExampleNotificationChannelClient_VerifyNotificationChannel() {
	// import monitoringpb "google.golang.org/genproto/googleapis/monitoring/v3"

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

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