internal/tfimport/importer/google_resource_manager_lien.go (63 lines of code) (raw):

// Copyright 2021 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 // // http://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. package importer import ( "context" "fmt" "os" "strings" "github.com/GoogleCloudPlatform/healthcare-data-protection-suite/internal/terraform" "github.com/ryanuber/columnize" "google.golang.org/api/cloudresourcemanager/v1" ) // ResourceManagerLien defines a struct with the necessary information for a google_resource_manager_lien to be imported. type ResourceManagerLien struct{} // ImportID returns the ID of the resource to use in importing. func (i *ResourceManagerLien) ImportID(rc terraform.ResourceChange, pcv ConfigMap, interactive bool) (string, error) { // Can't import if not interactive, since lien names are system-generated and will never be in the plan. if !interactive { return "", &InsufficientInfoErr{[]string{"name"}, "Lien name is system-generated and will never be in the plan"} } parentI, err := fromConfigValues("parent", rc.Change.After, pcv) if err != nil { return "", err } parentFull := fmt.Sprintf("%s", parentI) // Parent is in the form "project/1234567890", but we need to drop the prefix. parts := strings.Split(parentFull, "/") parent := parts[len(parts)-1] // The name is not set, it is autogenerated by the system. // Try to find existing liens for this parent to offer as choices. liens, err := i.getLiens(parentFull) if err != nil { return "", err } if len(liens) <= 0 { // There are no liens, just return that it doesn't exist so it can be created. return "", &DoesNotExistErr{rc.Address} } // Present the choices, if any // Format the liens more nicely. liensLines := []string{"Name|Origin|Reason|Restrictions"} for _, lien := range liens { // The name is always "liens/<id for import>". lienID := strings.Split(lien.Name, "/")[1] liensLines = append(liensLines, fmt.Sprintf("%v|%v|%v|%v", lienID, lien.Origin, lien.Reason, strings.Join(lien.Restrictions, ", "))) } prompt := fmt.Sprintf("Found the following liens in %v:\n%s", parentFull, columnize.SimpleFormat(liensLines)) // Get the value from the user name, err := fromUser(os.Stdin, "name", prompt) if err != nil { return "", err } return fmt.Sprintf("%v/%v", parent, name), nil } func (i *ResourceManagerLien) getLiens(parent string) (liens []*cloudresourcemanager.Lien, err error) { ctx := context.Background() service, err := cloudresourcemanager.NewService(ctx) if err != nil { return liens, err } liensService := cloudresourcemanager.NewLiensService(service) listCall := liensService.List() listCall.Parent(parent) // Do the call repeatedly, as long as there are more pages. for { resp, err := listCall.Do() if err != nil { return liens, err } liens = append(liens, resp.Liens...) if resp.NextPageToken == "" { // No more pages, break out. break } listCall.PageToken(resp.NextPageToken) } return liens, nil }