blob: 89f2943c0d70c12533627404134dc7f9fbf65a00 [file] [log] [blame]
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
package logging
import (
"context"
"errors"
"fmt"
"log"
"github.com/hashicorp/terraform-provider-google-beta/google-beta/tpgresource"
transport_tpg "github.com/hashicorp/terraform-provider-google-beta/google-beta/transport"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
)
const nonUniqueWriterAccount = "serviceAccount:cloud-logs@system.gserviceaccount.com"
func ResourceLoggingProjectSink() *schema.Resource {
schm := &schema.Resource{
Create: resourceLoggingProjectSinkAcquireOrCreate,
Read: resourceLoggingProjectSinkRead,
Delete: resourceLoggingProjectSinkDelete,
Update: resourceLoggingProjectSinkUpdate,
Schema: resourceLoggingSinkSchema(),
CustomizeDiff: resourceLoggingProjectSinkCustomizeDiff,
Importer: &schema.ResourceImporter{
State: resourceLoggingSinkImportState("project"),
},
UseJSONNumber: true,
}
schm.Schema["project"] = &schema.Schema{
Type: schema.TypeString,
Optional: true,
Computed: true,
ForceNew: true,
Description: `The ID of the project to create the sink in. If omitted, the project associated with the provider is used.`,
}
schm.Schema["unique_writer_identity"] = &schema.Schema{
Type: schema.TypeBool,
Optional: true,
Default: true,
Description: `Whether or not to create a unique identity associated with this sink. If false (the legacy behavior), then the writer_identity used is serviceAccount:cloud-logs@system.gserviceaccount.com. If true (default), then a unique service account is created and used for this sink. If you wish to publish logs across projects, you must set unique_writer_identity to true.`,
}
schm.Schema["custom_writer_identity"] = &schema.Schema{
Type: schema.TypeString,
Optional: true,
Description: `A service account provided by the caller that will be used to write the log entries. The format must be serviceAccount:some@email. This field can only be specified if you are routing logs to a destination outside this sink's project. If not specified, a Logging service account will automatically be generated.`,
}
return schm
}
func resourceLoggingProjectSinkAcquireOrCreate(d *schema.ResourceData, meta interface{}) error {
config := meta.(*transport_tpg.Config)
userAgent, err := tpgresource.GenerateUserAgentString(d, config.UserAgent)
if err != nil {
return err
}
project, err := tpgresource.GetProject(d, config)
if err != nil {
return err
}
id, sink := expandResourceLoggingSink(d, "projects", project)
uniqueWriterIdentity := d.Get("unique_writer_identity").(bool)
customWriterIdentity := d.Get("custom_writer_identity").(string)
log.Printf("[DEBUG] Fetching logging sink config: %#v", id)
res, _ := config.NewLoggingClient(userAgent).Projects.Sinks.Get(id.canonicalId()).Do()
if res == nil {
projectSinkCreateRequest := config.NewLoggingClient(userAgent).Projects.Sinks.Create(id.parent(), sink)
// if custom-sa is specified, use it to write log and it requires uniqueWriterIdentity to be set as well
// otherwise set the uniqueWriter identity
if customWriterIdentity != "" {
projectSinkCreateRequest = projectSinkCreateRequest.UniqueWriterIdentity(uniqueWriterIdentity).CustomWriterIdentity(customWriterIdentity)
} else {
projectSinkCreateRequest = projectSinkCreateRequest.UniqueWriterIdentity(uniqueWriterIdentity)
}
_, err = projectSinkCreateRequest.Do()
if err != nil {
return err
}
d.SetId(id.canonicalId())
return resourceLoggingProjectSinkRead(d, meta)
}
d.SetId(id.canonicalId())
return resourceLoggingProjectSinkUpdate(d, meta)
}
// 1) if bigquery_options is set unique_writer_identity must be true
// 2) taint the value of writer_identity when unique_writer_identity is updating from false -> true
func resourceLoggingProjectSinkCustomizeDiff(ctx context.Context, d *schema.ResourceDiff, meta interface{}) error {
// separate func to allow unit testing
err := resourceLoggingProjectSinkCustomizeDiffFunc(d)
if err != nil {
return err
}
err = resourceLoggingProjectSinkUniqueWriterIdentityCustomizeDiffFunc(d)
if err != nil {
return err
}
return nil
}
func resourceLoggingProjectSinkCustomizeDiffFunc(diff tpgresource.TerraformResourceDiff) error {
if !diff.HasChange("bigquery_options.#") {
return nil
}
bigqueryOptions := diff.Get("bigquery_options.#").(int)
if bigqueryOptions > 0 {
uwi := diff.Get("unique_writer_identity")
if !uwi.(bool) {
return errors.New("unique_writer_identity must be true when bigquery_options is supplied")
}
}
return nil
}
func resourceLoggingProjectSinkUniqueWriterIdentityCustomizeDiffFunc(diff *schema.ResourceDiff) error {
if !diff.HasChange("unique_writer_identity") {
return nil
}
// taint the value of writer_identity when unique_writer_identity is updating from false -> true
if diff.Get("unique_writer_identity").(bool) {
err := diff.SetNewComputed("writer_identity")
if err != nil {
return fmt.Errorf("Error re-setting writer_identity: %s", err)
}
}
return nil
}
func resourceLoggingProjectSinkRead(d *schema.ResourceData, meta interface{}) error {
config := meta.(*transport_tpg.Config)
userAgent, err := tpgresource.GenerateUserAgentString(d, config.UserAgent)
if err != nil {
return err
}
project, err := tpgresource.GetProject(d, config)
if err != nil {
return err
}
sink, err := config.NewLoggingClient(userAgent).Projects.Sinks.Get(d.Id()).Do()
if err != nil {
return transport_tpg.HandleNotFoundError(err, d, fmt.Sprintf("Project Logging Sink %s", d.Get("name").(string)))
}
if err := d.Set("project", project); err != nil {
return fmt.Errorf("Error setting project: %s", err)
}
if err := flattenResourceLoggingSink(d, sink); err != nil {
return err
}
if sink.WriterIdentity != nonUniqueWriterAccount {
if err := d.Set("unique_writer_identity", true); err != nil {
return fmt.Errorf("Error setting unique_writer_identity: %s", err)
}
} else {
if err := d.Set("unique_writer_identity", false); err != nil {
return fmt.Errorf("Error setting unique_writer_identity: %s", err)
}
}
return nil
}
func resourceLoggingProjectSinkUpdate(d *schema.ResourceData, meta interface{}) error {
config := meta.(*transport_tpg.Config)
userAgent, err := tpgresource.GenerateUserAgentString(d, config.UserAgent)
if err != nil {
return err
}
sink, updateMask := expandResourceLoggingSinkForUpdate(d)
uniqueWriterIdentity := d.Get("unique_writer_identity").(bool)
customWriterIdentity := d.Get("custom_writer_identity").(string)
projectSinkUpdateRequest := config.NewLoggingClient(userAgent).Projects.Sinks.Patch(d.Id(), sink).UpdateMask(updateMask)
// if custom-sa is specified, use it to write log and it reqiures uniqueWriterIdentity to be set as well
// otherwise set the uniqueWriter identity
if customWriterIdentity != "" {
projectSinkUpdateRequest = projectSinkUpdateRequest.UniqueWriterIdentity(uniqueWriterIdentity).CustomWriterIdentity(customWriterIdentity)
} else {
projectSinkUpdateRequest = projectSinkUpdateRequest.UniqueWriterIdentity(uniqueWriterIdentity)
}
_, err = projectSinkUpdateRequest.Do()
if err != nil {
return err
}
return resourceLoggingProjectSinkRead(d, meta)
}
func resourceLoggingProjectSinkDelete(d *schema.ResourceData, meta interface{}) error {
name := d.Get("name")
for _, restrictedName := range []string{"_Required", "_Default"} {
if name == restrictedName {
log.Print("[WARN] Default logging sinks cannot be deleted.")
return nil
}
}
config := meta.(*transport_tpg.Config)
userAgent, err := tpgresource.GenerateUserAgentString(d, config.UserAgent)
if err != nil {
return err
}
_, err = config.NewLoggingClient(userAgent).Projects.Sinks.Delete(d.Id()).Do()
if err != nil {
return err
}
d.SetId("")
return nil
}