diff --git a/cmd/apis/bundlecrtapis.go b/cmd/apis/bundlecrtapis.go index 323b72bf9..5adaed46c 100644 --- a/cmd/apis/bundlecrtapis.go +++ b/cmd/apis/bundlecrtapis.go @@ -21,6 +21,7 @@ import ( "path/filepath" "internal/apiclient" + "internal/clilog" proxybundle "internal/bundlegen/proxybundle" @@ -32,7 +33,7 @@ import ( var BundleCreateCmd = &cobra.Command{ Use: "bundle", Short: "Creates an API proxy from an Zip or folder", - Long: "Creates an API proxy from an Zip or folder", + Long: "Creates an API proxy from an Zip or folder; Optionally deploy the API to an env", Args: func(cmd *cobra.Command, args []string) (err error) { if proxyZip != "" && proxyFolder != "" { return fmt.Errorf("proxy bundle (zip) and folder to an API proxy cannot be combined") @@ -45,11 +46,15 @@ var BundleCreateCmd = &cobra.Command{ return err } } + if env != "" { + apiclient.SetApigeeEnv(env) + } return apiclient.SetApigeeOrg(org) }, RunE: func(cmd *cobra.Command, args []string) (err error) { + var respBody []byte if proxyZip != "" { - _, err = apis.CreateProxy(name, proxyZip) + respBody, err = apis.CreateProxy(name, proxyZip) } else if proxyFolder != "" { if stat, err := os.Stat(folder); err == nil && !stat.IsDir() { return fmt.Errorf("supplied path is not a folder") @@ -68,11 +73,25 @@ var BundleCreateCmd = &cobra.Command{ if err = proxybundle.GenerateArchiveBundle(proxyFolder, proxyBundlePath, false); err != nil { return err } - if _, err = apis.CreateProxy(name, proxyBundlePath); err != nil { + if respBody, err = apis.CreateProxy(name, proxyBundlePath); err != nil { + return err + } + if err = os.Remove(proxyBundlePath); err != nil { return err } - - return os.Remove(proxyBundlePath) + } + if env != "" { + clilog.Info.Printf("Deploying the API Proxy %s to environment %s\n", name, env) + if revision, err = GetRevision(respBody); err != nil { + return err + } + if _, err = apis.DeployProxy(name, revision, overrides, + sequencedRollout, safeDeploy, serviceAccountName); err != nil { + return err + } + if wait { + return Wait(name, revision) + } } return err }, @@ -89,5 +108,19 @@ func init() { BundleCreateCmd.Flags().StringVarP(&proxyFolder, "proxy-folder", "f", "", "Path to the Proxy Bundle; ex: ./test/apiproxy") + BundleCreateCmd.Flags().StringVarP(&env, "env", "e", + "", "Name of the environment to deploy the proxy") + BundleCreateCmd.Flags().BoolVarP(&overrides, "ovr", "r", + false, "Forces deployment of the new revision") + BundleCreateCmd.Flags().BoolVarP(&wait, "wait", "", + false, "Waits for the deployment to finish, with success or error") + BundleCreateCmd.Flags().BoolVarP(&sequencedRollout, "sequencedrollout", "", + false, "If set to true, the routing rules will be rolled out in a safe order; default is false") + BundleCreateCmd.Flags().BoolVarP(&safeDeploy, "safedeploy", "", + true, "When set to true, generateDeployChangeReport will be executed and "+ + "deployment will proceed if there are no conflicts; default is true") + BundleCreateCmd.Flags().StringVarP(&serviceAccountName, "sa", "s", + "", "The format must be {ACCOUNT_ID}@{PROJECT}.iam.gserviceaccount.com.") + _ = BundleCreateCmd.MarkFlagRequired("name") } diff --git a/cmd/apis/common.go b/cmd/apis/common.go new file mode 100644 index 000000000..ea16c87b7 --- /dev/null +++ b/cmd/apis/common.go @@ -0,0 +1,77 @@ +// Copyright 2023 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 apis + +import ( + "encoding/json" + "fmt" + "strconv" + "time" + + "internal/apiclient" + "internal/client/apis" + "internal/clilog" +) + +func GetRevision(respBody []byte) (revision int, err error) { + var apiProxyRevResp map[string]interface{} + + err = json.Unmarshal(respBody, &apiProxyRevResp) + if err != nil { + return -1, err + } + apiProxyRev, err := strconv.Atoi(fmt.Sprintf("%v", apiProxyRevResp["revision"])) + if err != nil { + return -1, err + } + return apiProxyRev, nil +} + +func Wait(name string, revision int) error { + var err error + + clilog.Info.Printf("Checking deployment status in %d seconds\n", interval) + + apiclient.DisableCmdPrintHttpResponse() + + stop := apiclient.Every(interval*time.Second, func(time.Time) bool { + var respBody []byte + respMap := make(map[string]interface{}) + if respBody, err = apis.ListProxyRevisionDeployments(name, revision); err != nil { + clilog.Error.Printf("Error fetching proxy revision status: %v", err) + return false + } + + if err = json.Unmarshal(respBody, &respMap); err != nil { + return true + } + + switch respMap["state"] { + case "PROGRESSING": + clilog.Info.Printf("Proxy deployment status is: %s. Waiting %d seconds.\n", respMap["state"], interval) + return true + case "READY": + clilog.Info.Println("Proxy deployment completed with status: ", respMap["state"]) + default: + clilog.Info.Println("Proxy deployment failed with status: ", respMap["state"]) + } + + return false + }) + + <-stop + + return err +} diff --git a/cmd/apis/depapi.go b/cmd/apis/depapi.go index d812271f7..b9669ef61 100644 --- a/cmd/apis/depapi.go +++ b/cmd/apis/depapi.go @@ -15,11 +15,7 @@ package apis import ( - "encoding/json" - "time" - "internal/apiclient" - "internal/clilog" "internal/client/apis" @@ -51,37 +47,9 @@ var DepCmd = &cobra.Command{ return err } - apiclient.DisableCmdPrintHttpResponse() - if wait { - clilog.Info.Printf("Checking deployment status in %d seconds\n", interval) - - stop := apiclient.Every(interval*time.Second, func(time.Time) bool { - var respBody []byte - respMap := make(map[string]interface{}) - if respBody, err = apis.ListProxyRevisionDeployments(name, revision); err != nil { - clilog.Error.Printf("Error fetching proxy revision status: %v", err) - return false - } - - if err = json.Unmarshal(respBody, &respMap); err != nil { - return true - } - - if respMap["state"] == "PROGRESSING" { - clilog.Info.Printf("Proxy deployment status is: %s. Waiting %d seconds.\n", respMap["state"], interval) - return true - } else if respMap["state"] == "READY" { - clilog.Info.Println("Proxy deployment completed with status: ", respMap["state"]) - } else { - clilog.Info.Println("Proxy deployment failed with status: ", respMap["state"]) - } - return false - }) - - <-stop + err = Wait(name, revision) } - return err }, } diff --git a/cmd/sharedflows/bundlecrtsf.go b/cmd/sharedflows/bundlecrtsf.go index e606ac313..697d5cc74 100644 --- a/cmd/sharedflows/bundlecrtsf.go +++ b/cmd/sharedflows/bundlecrtsf.go @@ -21,6 +21,7 @@ import ( "path/filepath" "internal/apiclient" + "internal/clilog" "internal/bundlegen/proxybundle" @@ -33,7 +34,7 @@ import ( var BundleCreateCmd = &cobra.Command{ Use: "bundle", Short: "Creates a sharedflow in an Apigee Org", - Long: "Creates a sharedflow in an Apigee Org", + Long: "Creates a sharedflow in an Apigee Org; Optionally deploy the sharedflow to an env", Args: func(cmd *cobra.Command, args []string) (err error) { apiclient.SetApigeeEnv(env) if sfZip != "" && sfFolder != "" { @@ -47,11 +48,15 @@ var BundleCreateCmd = &cobra.Command{ return err } } + if env != "" { + apiclient.SetApigeeEnv(env) + } return apiclient.SetApigeeOrg(org) }, RunE: func(cmd *cobra.Command, args []string) (err error) { + var respBody []byte if sfZip != "" { - _, err = sharedflows.Create(name, sfZip) + respBody, err = sharedflows.Create(name, sfZip) } else if sfFolder != "" { if stat, err := os.Stat(folder); err == nil && !stat.IsDir() { return fmt.Errorf("supplied path is not a folder") @@ -70,10 +75,24 @@ var BundleCreateCmd = &cobra.Command{ if err = proxybundle.GenerateArchiveBundle(sfFolder, sfBundlePath, true); err != nil { return err } - if _, err = sharedflows.Create(name, sfBundlePath); err != nil { + if respBody, err = sharedflows.Create(name, sfBundlePath); err != nil { + return err + } + if err = os.Remove(sfBundlePath); err != nil { return err } - return os.Remove(sfBundlePath) + } + if env != "" { + clilog.Info.Printf("Deploying the Sharedflow %s to environment %s\n", name, env) + if revision, err = GetRevision(respBody); err != nil { + return err + } + if _, err = sharedflows.Deploy(name, revision, overrides, serviceAccountName); err != nil { + return err + } + if wait { + return Wait(name, revision) + } } return err }, @@ -88,6 +107,14 @@ func init() { "", "Path to the Sharedflow bundle/zip file") BundleCreateCmd.Flags().StringVarP(&sfFolder, "sf-folder", "f", "", "Path to the Sharedflow Bundle; ex: ./test/sharedflowbundle") + BundleCreateCmd.Flags().StringVarP(&env, "env", "e", + "", "Apigee environment name") + BundleCreateCmd.Flags().BoolVarP(&overrides, "ovr", "r", + false, "Forces deployment of the new revision") + BundleCreateCmd.Flags().BoolVarP(&wait, "wait", "", + false, "Waits for the deployment to finish, with success or error") + BundleCreateCmd.Flags().StringVarP(&serviceAccountName, "sa", "s", + "", "The format must be {ACCOUNT_ID}@{PROJECT}.iam.gserviceaccount.com.") _ = BundleCreateCmd.MarkFlagRequired("name") } diff --git a/cmd/sharedflows/common.go b/cmd/sharedflows/common.go new file mode 100644 index 000000000..d86653034 --- /dev/null +++ b/cmd/sharedflows/common.go @@ -0,0 +1,76 @@ +// Copyright 2023 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 sharedflows + +import ( + "encoding/json" + "fmt" + "strconv" + "time" + + "internal/apiclient" + "internal/client/sharedflows" + "internal/clilog" +) + +func GetRevision(respBody []byte) (revision int, err error) { + var apiProxyRevResp map[string]interface{} + + err = json.Unmarshal(respBody, &apiProxyRevResp) + if err != nil { + return -1, err + } + apiProxyRev, err := strconv.Atoi(fmt.Sprintf("%v", apiProxyRevResp["revision"])) + if err != nil { + return -1, err + } + return apiProxyRev, nil +} + +func Wait(name string, revision int) error { + var err error + + apiclient.DisableCmdPrintHttpResponse() + + clilog.Info.Printf("Checking deployment status in %d seconds\n", interval) + + stop := apiclient.Every(interval*time.Second, func(time.Time) bool { + var respBody []byte + respMap := make(map[string]interface{}) + if respBody, err = sharedflows.ListRevisionDeployments(name, revision); err != nil { + clilog.Error.Printf("Error fetching sharedflow revision status: %v", err) + return false + } + + if err = json.Unmarshal(respBody, &respMap); err != nil { + return true + } + + switch respMap["state"] { + case "PROGRESSING": + clilog.Info.Printf("Sharedflow deployment status is: %s. Waiting %d seconds.\n", respMap["state"], interval) + return true + case "READY": + clilog.Info.Println("Sharedflow deployment completed with status: ", respMap["state"]) + default: + clilog.Info.Println("Sharedflow deployment failed with status: ", respMap["state"]) + } + return false + }) + + <-stop + + return err +} diff --git a/cmd/sharedflows/depsf.go b/cmd/sharedflows/depsf.go index 2295b12e3..289c73b42 100644 --- a/cmd/sharedflows/depsf.go +++ b/cmd/sharedflows/depsf.go @@ -15,11 +15,7 @@ package sharedflows import ( - "encoding/json" - "time" - "internal/apiclient" - "internal/clilog" "internal/client/sharedflows" @@ -45,32 +41,7 @@ var DepCmd = &cobra.Command{ apiclient.DisableCmdPrintHttpResponse() if wait { - clilog.Info.Printf("Checking deployment status in %d seconds\n", interval) - - stop := apiclient.Every(interval*time.Second, func(time.Time) bool { - var respBody []byte - respMap := make(map[string]interface{}) - if respBody, err = sharedflows.ListRevisionDeployments(name, revision); err != nil { - clilog.Error.Printf("Error fetching sharedflow revision status: %v", err) - return false - } - - if err = json.Unmarshal(respBody, &respMap); err != nil { - return true - } - - if respMap["state"] == "PROGRESSING" { - clilog.Info.Printf("Sharedflow deployment status is: %s. Waiting %d seconds.\n", respMap["state"], interval) - return true - } else if respMap["state"] == "READY" { - clilog.Info.Println("Sharedflow deployment completed with status: ", respMap["state"]) - } else { - clilog.Info.Println("Sharedflow deployment failed with status: ", respMap["state"]) - } - return false - }) - - <-stop + err = Wait(name, revision) } return err diff --git a/cmd/targetservers/impts.go b/cmd/targetservers/impts.go index c6caa98d0..8ee5c749c 100644 --- a/cmd/targetservers/impts.go +++ b/cmd/targetservers/impts.go @@ -40,7 +40,7 @@ var filePath string func init() { ImpCmd.Flags().StringVarP(&filePath, "file", "f", - "", "File containing API Products") + "", "Path to a file containing Target Servers") ImpCmd.Flags().IntVarP(&conn, "conn", "c", 4, "Number of connections") diff --git a/internal/apiclient/bundles.go b/internal/apiclient/bundles.go index fe16b6892..59d449697 100644 --- a/internal/apiclient/bundles.go +++ b/internal/apiclient/bundles.go @@ -235,15 +235,14 @@ func FetchBundle(entityType string, folder string, name string, revision string, func ImportBundleAsync(entityType string, name string, bundlePath string, wg *sync.WaitGroup) { defer wg.Done() - _ = ImportBundle(entityType, name, bundlePath) + _, _ = ImportBundle(entityType, name, bundlePath) } // ImportBundle imports a sharedflow or api proxy bundle -func ImportBundle(entityType string, name string, bundlePath string) error { - err := ReadBundle(bundlePath) - if err != nil { +func ImportBundle(entityType string, name string, bundlePath string) (respBody []byte, err error) { + if err = ReadBundle(bundlePath); err != nil { clilog.Error.Println(err) - return err + return nil, err } // when importing from a folder, proxy name = file name @@ -265,14 +264,13 @@ func ImportBundle(entityType string, name string, bundlePath string) error { "proxy": bundlePath, } - _, err = PostHttpOctet(false, u.String(), formParams) - if err != nil { + if respBody, err = PostHttpOctet(false, u.String(), formParams); err != nil { clilog.Error.Println(err) - return err + return nil, err } clilog.Debug.Printf("Completed entity: %s", u.String()) - return nil + return respBody, err } func FolderExists(folder string) (err error) { diff --git a/internal/client/apis/apis.go b/internal/client/apis/apis.go index 61e999d2c..7a94ba873 100644 --- a/internal/client/apis/apis.go +++ b/internal/client/apis/apis.go @@ -83,7 +83,7 @@ type conflictingdeployment struct { // CreateProxy func CreateProxy(name string, proxy string) (respBody []byte, err error) { if proxy != "" { - err = apiclient.ImportBundle("apis", name, proxy) + respBody, err = apiclient.ImportBundle("apis", name, proxy) return respBody, err } u, _ := url.Parse(apiclient.BaseURL) diff --git a/internal/client/sharedflows/sharedflows.go b/internal/client/sharedflows/sharedflows.go index 8f5117297..0340ac3b9 100644 --- a/internal/client/sharedflows/sharedflows.go +++ b/internal/client/sharedflows/sharedflows.go @@ -52,7 +52,7 @@ type revision struct { // Create func Create(name string, proxy string) (respBody []byte, err error) { if proxy != "" { - err = apiclient.ImportBundle("sharedflows", name, proxy) + respBody, err = apiclient.ImportBundle("sharedflows", name, proxy) return respBody, err } u, _ := url.Parse(apiclient.BaseURL)