package cmd import ( "errors" "fmt" "github.com/urfave/cli/v2" "heckel.io/ntfy/auth" "heckel.io/ntfy/util" ) const ( userEveryone = "everyone" ) var flagsAllow = append( userCommandFlags(), &cli.BoolFlag{Name: "reset", Aliases: []string{"r"}, Usage: "reset access for user (and topic)"}, ) var cmdAllow = &cli.Command{ Name: "allow", Usage: "Grant a user access to a topic", UsageText: "ntfy allow USERNAME TOPIC [read-write|read-only|write-only|none]", Flags: flagsAllow, Before: initConfigFileInputSource("config", flagsAllow), Action: execUserAllow, Category: categoryServer, } func execUserAllow(c *cli.Context) error { username := c.Args().Get(0) topic := c.Args().Get(1) perms := c.Args().Get(2) reset := c.Bool("reset") if username == "" { return errors.New("username expected, type 'ntfy allow --help' for help") } else if !reset && topic == "" { return errors.New("topic expected, type 'ntfy allow --help' for help") } else if !util.InStringList([]string{"", "read-write", "rw", "read-only", "read", "ro", "write-only", "write", "wo", "none"}, perms) { return errors.New("permission must be one of: read-write, read-only, write-only, or none (or the aliases: read, ro, write, wo)") } if username == userEveryone { username = "" } read := util.InStringList([]string{"", "read-write", "rw", "read-only", "read", "ro"}, perms) write := util.InStringList([]string{"", "read-write", "rw", "write-only", "write", "wo"}, perms) manager, err := createAuthManager(c) if err != nil { return err } if reset { return doAccessReset(c, manager, username, topic) } return doAccessAllow(c, manager, username, topic, read, write) } func doAccessAllow(c *cli.Context, manager auth.Manager, username string, topic string, read bool, write bool) error { if err := manager.AllowAccess(username, topic, read, write); err != nil { return err } if username == "" { if read && write { fmt.Fprintf(c.App.Writer, "Anonymous users granted full access to topic %s\n", topic) } else if read { fmt.Fprintf(c.App.Writer, "Anonymous users granted read-only access to topic %s\n", topic) } else if write { fmt.Fprintf(c.App.Writer, "Anonymous users granted write-only access to topic %s\n", topic) } else { fmt.Fprintf(c.App.Writer, "Revoked all access to topic %s for all anonymous users\n", topic) } } else { if read && write { fmt.Fprintf(c.App.Writer, "User %s now has read-write access to topic %s\n", username, topic) } else if read { fmt.Fprintf(c.App.Writer, "User %s now has read-only access to topic %s\n", username, topic) } else if write { fmt.Fprintf(c.App.Writer, "User %s now has write-only access to topic %s\n", username, topic) } else { fmt.Fprintf(c.App.Writer, "Revoked all access to topic %s for user %s\n", topic, username) } } user, err := manager.User(username) if err != nil { return err } fmt.Fprintln(c.App.Writer) return showUsers(c, []*auth.User{user}) } func doAccessReset(c *cli.Context, manager auth.Manager, username, topic string) error { if err := manager.ResetAccess(username, topic); err != nil { return err } if username == "" { if topic == "" { fmt.Fprintln(c.App.Writer, "Reset access for all anonymous users and all topics") } else { fmt.Fprintf(c.App.Writer, "Reset access to topic %s for all anonymous users\n", topic) } } else { if topic == "" { fmt.Fprintf(c.App.Writer, "Reset access for user %s to all topics\n", username) } else { fmt.Fprintf(c.App.Writer, "Reset access for user %s and topic %s\n", username, topic) } } return nil }