Files
fn-serverless/fn/apps.go
Pedro Nasser 5a6ebb4e2f fn should show API errors messages (#450)
* fn should show API errors messages

* update functions_go

* update functions_go lock
2016-12-19 16:30:18 -02:00

301 lines
6.7 KiB
Go

package main
import (
"encoding/json"
"errors"
"fmt"
"net/http"
"os"
"text/tabwriter"
"github.com/iron-io/functions_go"
"github.com/urfave/cli"
)
type appsCmd struct {
*functions.AppsApi
}
func apps() cli.Command {
a := appsCmd{AppsApi: functions.NewAppsApi()}
return cli.Command{
Name: "apps",
Usage: "operate applications",
ArgsUsage: "fn apps",
Subcommands: []cli.Command{
{
Name: "create",
Aliases: []string{"c"},
Usage: "create a new app",
ArgsUsage: "`app`",
Action: a.create,
Flags: []cli.Flag{
cli.StringSliceFlag{
Name: "config",
Usage: "application configuration",
},
},
},
{
Name: "list",
Aliases: []string{"l"},
Usage: "list all apps",
Action: a.list,
},
{
Name: "config",
Usage: "operate an application configuration set",
Subcommands: []cli.Command{
{
Name: "view",
Aliases: []string{"v"},
Usage: "view all configuration keys for this app",
ArgsUsage: "`app`",
Action: a.configList,
Flags: []cli.Flag{
cli.BoolFlag{
Name: "shell,s",
Usage: "output in shell format",
},
cli.BoolFlag{
Name: "json,j",
Usage: "output in JSON format",
},
},
},
{
Name: "set",
Aliases: []string{"s"},
Usage: "store a configuration key for this application",
ArgsUsage: "`app` <key> <value>",
Action: a.configSet,
},
{
Name: "unset",
Aliases: []string{"u"},
Usage: "remove a configuration key for this application",
ArgsUsage: "`app` <key>",
Action: a.configUnset,
},
},
},
{
Name: "delete",
Usage: "delete an app",
Action: a.delete,
},
},
}
}
func (a *appsCmd) list(c *cli.Context) error {
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
wrapper, _, err := a.AppsGet()
if err != nil {
return fmt.Errorf("error getting app: %v", err)
}
if msg := wrapper.Error_.Message; msg != "" {
return errors.New(msg)
}
if len(wrapper.Apps) == 0 {
fmt.Println("no apps found")
return nil
}
for _, app := range wrapper.Apps {
fmt.Println(app.Name)
}
return nil
}
func (a *appsCmd) create(c *cli.Context) error {
if c.Args().First() == "" {
return errors.New("error: app creating takes one argument, an app name")
}
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
body := functions.AppWrapper{App: functions.App{
Name: c.Args().Get(0),
Config: extractEnvConfig(c.StringSlice("config")),
}}
wrapper, _, err := a.AppsPost(body)
if err != nil {
return fmt.Errorf("error creating app: %v", err)
}
if msg := wrapper.Error_.Message; msg != "" {
return errors.New(msg)
}
fmt.Println(wrapper.App.Name, "created")
return nil
}
func (a *appsCmd) configList(c *cli.Context) error {
if c.Args().First() == "" {
return errors.New("error: app description takes one argument, an app name")
}
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
appName := c.Args().Get(0)
wrapper, _, err := a.AppsAppGet(appName)
if err != nil {
return fmt.Errorf("error creating app: %v", err)
}
if msg := wrapper.Error_.Message; msg != "" {
return errors.New(msg)
}
config := wrapper.App.Config
if len(config) == 0 {
return errors.New("this application has no configurations")
}
if c.Bool("json") {
if err := json.NewEncoder(os.Stdout).Encode(config); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
} else if c.Bool("shell") {
for k, v := range wrapper.App.Config {
fmt.Print("export ", k, "=", v, "\n")
}
} else {
fmt.Println(wrapper.App.Name, "configuration:")
w := tabwriter.NewWriter(os.Stdout, 0, 8, 1, ' ', 0)
for k, v := range config {
fmt.Fprint(w, k, ":\t", v, "\n")
}
w.Flush()
}
return nil
}
func (a *appsCmd) configSet(c *cli.Context) error {
if c.Args().Get(0) == "" || c.Args().Get(1) == "" || c.Args().Get(2) == "" {
return errors.New("error: application configuration setting takes three arguments: an app name, a key and a value")
}
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
appName := c.Args().Get(0)
key := c.Args().Get(1)
value := c.Args().Get(2)
wrapper, _, err := a.AppsAppGet(appName)
if err != nil {
return fmt.Errorf("error creating app: %v", err)
}
if msg := wrapper.Error_.Message; msg != "" {
return errors.New(msg)
}
config := wrapper.App.Config
if config == nil {
config = make(map[string]string)
}
config[key] = value
if err := a.storeApp(appName, config); err != nil {
return fmt.Errorf("error updating app configuration: %v", err)
}
fmt.Println(wrapper.App.Name, "updated", key, "with", value)
return nil
}
func (a *appsCmd) configUnset(c *cli.Context) error {
if c.Args().Get(0) == "" || c.Args().Get(1) == "" {
return errors.New("error: application configuration setting takes three arguments: an app name, a key and a value")
}
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
appName := c.Args().Get(0)
key := c.Args().Get(1)
wrapper, _, err := a.AppsAppGet(appName)
if err != nil {
return fmt.Errorf("error creating app: %v", err)
}
if msg := wrapper.Error_.Message; msg != "" {
return errors.New(msg)
}
config := wrapper.App.Config
if config == nil {
config = make(map[string]string)
}
if _, ok := config[key]; !ok {
return fmt.Errorf("configuration key %s not found", key)
}
delete(config, key)
if err := a.storeApp(appName, config); err != nil {
return fmt.Errorf("error updating app configuration: %v", err)
}
fmt.Println(wrapper.App.Name, "removed", key)
return nil
}
func (a *appsCmd) storeApp(appName string, config map[string]string) error {
body := functions.AppWrapper{App: functions.App{
Name: appName,
Config: config,
}}
if _, _, err := a.AppsPost(body); err != nil {
return fmt.Errorf("error updating app configuration: %v", err)
}
return nil
}
func (a *appsCmd) delete(c *cli.Context) error {
appName := c.Args().First()
if appName == "" {
return errors.New("error: deleting an app takes one argument, an app name")
}
if err := resetBasePath(a.Configuration); err != nil {
return fmt.Errorf("error setting endpoint: %v", err)
}
resp, err := a.AppsAppDelete(appName)
if err != nil {
return fmt.Errorf("error deleting app: %v", err)
}
if resp.StatusCode == http.StatusBadRequest {
return errors.New("could not delete this application - pending routes")
}
fmt.Println(appName, "deleted")
return nil
}