Upgrade urfave/cli to v2 (#483)

* migrate urfave/ci v1 -> v2
* refactor cli (format flag)
* log error if agent can not listen on port 3000

close #452
This commit is contained in:
6543 2021-10-27 21:03:14 +02:00 committed by GitHub
parent d35f5158bc
commit 798c2bc8b2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
121 changed files with 8407 additions and 1346 deletions

View file

@ -1,12 +1,17 @@
package build package build
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the build command set. // Command exports the build command set.
var Command = cli.Command{ var Command = &cli.Command{
Name: "build", Name: "build",
Usage: "manage builds", Usage: "manage builds",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
buildListCmd, buildListCmd,
buildLastCmd, buildLastCmd,
buildLogsCmd, buildLogsCmd,

View file

@ -4,16 +4,18 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildApproveCmd = cli.Command{ var buildApproveCmd = &cli.Command{
Name: "approve", Name: "approve",
Usage: "approve a build", Usage: "approve a build",
ArgsUsage: "<repo/name> <build>", ArgsUsage: "<repo/name> <build>",
Action: buildApprove, Action: buildApprove,
Flags: common.GlobalFlags,
} }
func buildApprove(c *cli.Context) (err error) { func buildApprove(c *cli.Context) (err error) {

View file

@ -4,16 +4,18 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildDeclineCmd = cli.Command{ var buildDeclineCmd = &cli.Command{
Name: "decline", Name: "decline",
Usage: "decline a build", Usage: "decline a build",
ArgsUsage: "<repo/name> <build>", ArgsUsage: "<repo/name> <build>",
Action: buildDecline, Action: buildDecline,
Flags: common.GlobalFlags,
} }
func buildDecline(c *cli.Context) (err error) { func buildDecline(c *cli.Context) (err error) {

View file

@ -5,23 +5,20 @@ import (
"strconv" "strconv"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildInfoCmd = cli.Command{ var buildInfoCmd = &cli.Command{
Name: "info", Name: "info",
Usage: "show build details", Usage: "show build details",
ArgsUsage: "<repo/name> [build]", ArgsUsage: "<repo/name> [build]",
Action: buildInfo, Action: buildInfo,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplBuildInfo),
Name: "format", ),
Usage: "format output",
Value: tmplBuildInfo,
},
},
} }
func buildInfo(c *cli.Context) error { func buildInfo(c *cli.Context) error {

View file

@ -4,17 +4,19 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildKillCmd = cli.Command{ var buildKillCmd = &cli.Command{
Name: "kill", Name: "kill",
Usage: "force kill a build", Usage: "force kill a build",
ArgsUsage: "<repo/name> <build>", ArgsUsage: "<repo/name> <build>",
Action: buildKill, Action: buildKill,
Hidden: true, Hidden: true,
Flags: common.GlobalFlags,
} }
func buildKill(c *cli.Context) (err error) { func buildKill(c *cli.Context) (err error) {

View file

@ -4,28 +4,25 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildLastCmd = cli.Command{ var buildLastCmd = &cli.Command{
Name: "last", Name: "last",
Usage: "show latest build details", Usage: "show latest build details",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: buildLast, Action: buildLast,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplBuildInfo),
Name: "format", &cli.StringFlag{
Usage: "format output",
Value: tmplBuildInfo,
},
cli.StringFlag{
Name: "branch", Name: "branch",
Usage: "branch name", Usage: "branch name",
Value: "master", Value: "master",
}, },
}, ),
} }
func buildLast(c *cli.Context) error { func buildLast(c *cli.Context) error {

View file

@ -4,40 +4,37 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildListCmd = cli.Command{ var buildListCmd = &cli.Command{
Name: "ls", Name: "ls",
Usage: "show build history", Usage: "show build history",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: buildList, Action: buildList,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplBuildList),
Name: "format", &cli.StringFlag{
Usage: "format output",
Value: tmplBuildList,
},
cli.StringFlag{
Name: "branch", Name: "branch",
Usage: "branch filter", Usage: "branch filter",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "event", Name: "event",
Usage: "event filter", Usage: "event filter",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "status", Name: "status",
Usage: "status filter", Usage: "status filter",
}, },
cli.IntFlag{ &cli.IntFlag{
Name: "limit", Name: "limit",
Usage: "limit the list size", Usage: "limit the list size",
Value: 25, Value: 25,
}, },
}, ),
} }
func buildList(c *cli.Context) error { func buildList(c *cli.Context) error {

View file

@ -3,14 +3,17 @@ package build
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/urfave/cli/v2"
) )
var buildLogsCmd = cli.Command{ var buildLogsCmd = &cli.Command{
Name: "logs", Name: "logs",
Usage: "show build logs", Usage: "show build logs",
ArgsUsage: "<repo/name> [build] [job]", ArgsUsage: "<repo/name> [build] [job]",
Action: buildLogs, Action: buildLogs,
Flags: common.GlobalFlags,
} }
func buildLogs(c *cli.Context) error { func buildLogs(c *cli.Context) error {

View file

@ -5,23 +5,20 @@ import (
"strconv" "strconv"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildPsCmd = cli.Command{ var buildPsCmd = &cli.Command{
Name: "ps", Name: "ps",
Usage: "show build steps", Usage: "show build steps",
ArgsUsage: "<repo/name> [build]", ArgsUsage: "<repo/name> [build]",
Action: buildPs, Action: buildPs,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplBuildPs),
Name: "format", ),
Usage: "format output",
Value: tmplBuildPs,
},
},
} }
func buildPs(c *cli.Context) error { func buildPs(c *cli.Context) error {

View file

@ -5,23 +5,20 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildQueueCmd = cli.Command{ var buildQueueCmd = &cli.Command{
Name: "queue", Name: "queue",
Usage: "show build queue", Usage: "show build queue",
ArgsUsage: " ", ArgsUsage: " ",
Action: buildQueue, Action: buildQueue,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplBuildQueue),
Name: "format", ),
Usage: "format output",
Value: tmplBuildQueue,
},
},
} }
func buildQueue(c *cli.Context) error { func buildQueue(c *cli.Context) error {

View file

@ -5,22 +5,24 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildStartCmd = cli.Command{ var buildStartCmd = &cli.Command{
Name: "start", Name: "start",
Usage: "start a build", Usage: "start a build",
ArgsUsage: "<repo/name> [build]", ArgsUsage: "<repo/name> [build]",
Action: buildStart, Action: buildStart,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "param, p", Name: "param",
Aliases: []string{"p"},
Usage: "custom parameters to be injected into the job environment. Format: KEY=value", Usage: "custom parameters to be injected into the job environment. Format: KEY=value",
}, },
}, ),
} }
func buildStart(c *cli.Context) (err error) { func buildStart(c *cli.Context) (err error) {

View file

@ -4,15 +4,17 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var buildStopCmd = cli.Command{ var buildStopCmd = &cli.Command{
Name: "stop", Name: "stop",
Usage: "stop a build", Usage: "stop a build",
ArgsUsage: "<repo/name> [build] [job]", ArgsUsage: "<repo/name> [build] [job]",
Flags: common.GlobalFlags,
Action: buildStop, Action: buildStop,
} }

61
cli/common/flags.go Normal file
View file

@ -0,0 +1,61 @@
// Copyright 2021 Woodpecker Authors
//
// 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 common
import "github.com/urfave/cli/v2"
var GlobalFlags = []cli.Flag{
&cli.StringFlag{
EnvVars: []string{"WOODPECKER_TOKEN"},
Name: "token",
Aliases: []string{"t"},
Usage: "server auth token",
},
&cli.StringFlag{
EnvVars: []string{"WOODPECKER_SERVER"},
Name: "server",
Aliases: []string{"s"},
Usage: "server address",
},
&cli.BoolFlag{
EnvVars: []string{"WOODPECKER_SKIP_VERIFY"},
Name: "skip-verify",
Usage: "skip ssl verification",
Hidden: true,
},
&cli.StringFlag{
EnvVars: []string{"SOCKS_PROXY"},
Name: "socks-proxy",
Usage: "socks proxy address",
Hidden: true,
},
&cli.BoolFlag{
EnvVars: []string{"SOCKS_PROXY_OFF"},
Name: "socks-proxy-off",
Usage: "socks proxy ignored",
Hidden: true,
},
}
// FormatFlag return format flag with value set based on template
// if hidden value is set, flag will be hidden
func FormatFlag(tmpl string, hidden ...bool) *cli.StringFlag {
return &cli.StringFlag{
Name: "format",
Usage: "format output",
Value: tmpl,
Hidden: len(hidden) != 0,
}
}

View file

@ -6,44 +6,42 @@ import (
"os" "os"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
// Command exports the deploy command. // Command exports the deploy command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "deploy", Name: "deploy",
Usage: "deploy code", Usage: "deploy code",
ArgsUsage: "<repo/name> <build> <environment>", ArgsUsage: "<repo/name> <build> <environment>",
Action: deploy, Action: deploy,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplDeployInfo),
Name: "format", &cli.StringFlag{
Usage: "format output",
Value: tmplDeployInfo,
},
cli.StringFlag{
Name: "branch", Name: "branch",
Usage: "branch filter", Usage: "branch filter",
Value: "master", Value: "master",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "event", Name: "event",
Usage: "event filter", Usage: "event filter",
Value: woodpecker.EventPush, Value: woodpecker.EventPush,
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "status", Name: "status",
Usage: "status filter", Usage: "status filter",
Value: woodpecker.StatusSuccess, Value: woodpecker.StatusSuccess,
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "param, p", Name: "param",
Aliases: []string{"p"},
Usage: "custom parameters to be injected into the job environment. Format: KEY=value", Usage: "custom parameters to be injected into the job environment. Format: KEY=value",
}, },
}, ),
} }
func deploy(c *cli.Context) error { func deploy(c *cli.Context) error {

View file

@ -11,9 +11,9 @@ import (
"strings" "strings"
"github.com/drone/envsubst" "github.com/drone/envsubst"
"github.com/rs/zerolog/log" "github.com/urfave/cli/v2"
"github.com/urfave/cli"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/pipeline" "github.com/woodpecker-ci/woodpecker/pipeline"
"github.com/woodpecker-ci/woodpecker/pipeline/backend" "github.com/woodpecker-ci/woodpecker/pipeline/backend"
"github.com/woodpecker-ci/woodpecker/pipeline/backend/docker" "github.com/woodpecker-ci/woodpecker/pipeline/backend/docker"
@ -27,16 +27,12 @@ import (
) )
// Command exports the exec command. // Command exports the exec command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "exec", Name: "exec",
Usage: "execute a local build", Usage: "execute a local build",
ArgsUsage: "[path/to/.woodpecker.yml]", ArgsUsage: "[path/to/.woodpecker.yml]",
Action: func(c *cli.Context) { Action: exec,
if err := exec(c); err != nil { Flags: append(common.GlobalFlags, flags...),
log.Fatal().Err(err).Msg("")
}
},
Flags: flags,
} }
func exec(c *cli.Context) error { func exec(c *cli.Context) error {

View file

@ -17,46 +17,47 @@ package exec
import ( import (
"time" "time"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
) )
var flags = []cli.Flag{ var flags = []cli.Flag{
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_LOCAL", EnvVars: []string{"WOODPECKER_LOCAL"},
Name: "local", Name: "local",
Usage: "build from local directory", Usage: "build from local directory",
Value: true,
}, },
cli.DurationFlag{ &cli.DurationFlag{
EnvVar: "WOODPECKER_TIMEOUT", EnvVars: []string{"WOODPECKER_TIMEOUT"},
Name: "timeout", Name: "timeout",
Usage: "build timeout", Usage: "build timeout",
Value: time.Hour, Value: time.Hour,
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_VOLUMES", EnvVars: []string{"WOODPECKER_VOLUMES"},
Name: "volumes", Name: "volumes",
Usage: "build volumes", Usage: "build volumes",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_NETWORKS", EnvVars: []string{"WOODPECKER_NETWORKS"},
Name: "network", Name: "network",
Usage: "external networks", Usage: "external networks",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_DOCKER_PREFIX", EnvVars: []string{"WOODPECKER_DOCKER_PREFIX"},
Name: "prefix", Name: "prefix",
Value: "woodpecker", Value: "woodpecker",
Usage: "prefix containers created by woodpecker", Usage: "prefix containers created by woodpecker",
Hidden: true, Hidden: true,
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "privileged", Name: "privileged",
Usage: "privileged plugins", Usage: "privileged plugins",
Value: &cli.StringSlice{ Value: cli.NewStringSlice(
"plugins/docker", "plugins/docker",
"plugins/gcr", "plugins/gcr",
"plugins/ecr", "plugins/ecr",
}, ),
}, },
// //
@ -67,199 +68,199 @@ var flags = []cli.Flag{
// //
// workspace default // workspace default
// //
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_WORKSPACE_BASE", EnvVars: []string{"WOODPECKER_WORKSPACE_BASE"},
Name: "workspace-base", Name: "workspace-base",
Value: "/woodpecker", Value: "/woodpecker",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_WORKSPACE_PATH", EnvVars: []string{"WOODPECKER_WORKSPACE_PATH"},
Name: "workspace-path", Name: "workspace-path",
Value: "src", Value: "src",
}, },
// //
// netrc parameters // netrc parameters
// //
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_NETRC_USERNAME", EnvVars: []string{"WOODPECKER_NETRC_USERNAME"},
Name: "netrc-username", Name: "netrc-username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_NETRC_PASSWORD", EnvVars: []string{"WOODPECKER_NETRC_PASSWORD"},
Name: "netrc-password", Name: "netrc-password",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_NETRC_MACHINE", EnvVars: []string{"WOODPECKER_NETRC_MACHINE"},
Name: "netrc-machine", Name: "netrc-machine",
}, },
// //
// metadata parameters // metadata parameters
// //
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SYSTEM_ARCH", EnvVars: []string{"WOODPECKER_SYSTEM_ARCH"},
Name: "system-arch", Name: "system-arch",
Value: "linux/amd64", Value: "linux/amd64",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SYSTEM_NAME", EnvVars: []string{"WOODPECKER_SYSTEM_NAME"},
Name: "system-name", Name: "system-name",
Value: "pipec", Value: "pipec",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SYSTEM_LINK", EnvVars: []string{"WOODPECKER_SYSTEM_LINK"},
Name: "system-link", Name: "system-link",
Value: "https://github.com/cncd/pipec", Value: "https://github.com/cncd/pipec",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_REPO_NAME", EnvVars: []string{"WOODPECKER_REPO_NAME"},
Name: "repo-name", Name: "repo-name",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_REPO_LINK", EnvVars: []string{"WOODPECKER_REPO_LINK"},
Name: "repo-link", Name: "repo-link",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_REPO_REMOTE", EnvVars: []string{"WOODPECKER_REPO_REMOTE"},
Name: "repo-remote-url", Name: "repo-remote-url",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_REPO_PRIVATE", EnvVars: []string{"WOODPECKER_REPO_PRIVATE"},
Name: "repo-private", Name: "repo-private",
}, },
cli.IntFlag{ &cli.IntFlag{
EnvVar: "WOODPECKER_BUILD_NUMBER", EnvVars: []string{"WOODPECKER_BUILD_NUMBER"},
Name: "build-number", Name: "build-number",
}, },
cli.IntFlag{ &cli.IntFlag{
EnvVar: "WOODPECKER_PARENT_BUILD_NUMBER", EnvVars: []string{"WOODPECKER_PARENT_BUILD_NUMBER"},
Name: "parent-build-number", Name: "parent-build-number",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_BUILD_CREATED", EnvVars: []string{"WOODPECKER_BUILD_CREATED"},
Name: "build-created", Name: "build-created",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_BUILD_STARTED", EnvVars: []string{"WOODPECKER_BUILD_STARTED"},
Name: "build-started", Name: "build-started",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_BUILD_FINISHED", EnvVars: []string{"WOODPECKER_BUILD_FINISHED"},
Name: "build-finished", Name: "build-finished",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BUILD_STATUS", EnvVars: []string{"WOODPECKER_BUILD_STATUS"},
Name: "build-status", Name: "build-status",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BUILD_EVENT", EnvVars: []string{"WOODPECKER_BUILD_EVENT"},
Name: "build-event", Name: "build-event",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BUILD_LINK", EnvVars: []string{"WOODPECKER_BUILD_LINK"},
Name: "build-link", Name: "build-link",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BUILD_TARGET", EnvVars: []string{"WOODPECKER_BUILD_TARGET"},
Name: "build-target", Name: "build-target",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_SHA", EnvVars: []string{"WOODPECKER_COMMIT_SHA"},
Name: "commit-sha", Name: "commit-sha",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_REF", EnvVars: []string{"WOODPECKER_COMMIT_REF"},
Name: "commit-ref", Name: "commit-ref",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_REFSPEC", EnvVars: []string{"WOODPECKER_COMMIT_REFSPEC"},
Name: "commit-refspec", Name: "commit-refspec",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_BRANCH", EnvVars: []string{"WOODPECKER_COMMIT_BRANCH"},
Name: "commit-branch", Name: "commit-branch",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_MESSAGE", EnvVars: []string{"WOODPECKER_COMMIT_MESSAGE"},
Name: "commit-message", Name: "commit-message",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_AUTHOR_NAME", EnvVars: []string{"WOODPECKER_COMMIT_AUTHOR_NAME"},
Name: "commit-author-name", Name: "commit-author-name",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_AUTHOR_AVATAR", EnvVars: []string{"WOODPECKER_COMMIT_AUTHOR_AVATAR"},
Name: "commit-author-avatar", Name: "commit-author-avatar",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_COMMIT_AUTHOR_EMAIL", EnvVars: []string{"WOODPECKER_COMMIT_AUTHOR_EMAIL"},
Name: "commit-author-email", Name: "commit-author-email",
}, },
cli.IntFlag{ &cli.IntFlag{
EnvVar: "WOODPECKER_PREV_BUILD_NUMBER", EnvVars: []string{"WOODPECKER_PREV_BUILD_NUMBER"},
Name: "prev-build-number", Name: "prev-build-number",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_PREV_BUILD_CREATED", EnvVars: []string{"WOODPECKER_PREV_BUILD_CREATED"},
Name: "prev-build-created", Name: "prev-build-created",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_PREV_BUILD_STARTED", EnvVars: []string{"WOODPECKER_PREV_BUILD_STARTED"},
Name: "prev-build-started", Name: "prev-build-started",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_PREV_BUILD_FINISHED", EnvVars: []string{"WOODPECKER_PREV_BUILD_FINISHED"},
Name: "prev-build-finished", Name: "prev-build-finished",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_BUILD_STATUS", EnvVars: []string{"WOODPECKER_PREV_BUILD_STATUS"},
Name: "prev-build-status", Name: "prev-build-status",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_BUILD_EVENT", EnvVars: []string{"WOODPECKER_PREV_BUILD_EVENT"},
Name: "prev-build-event", Name: "prev-build-event",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_BUILD_LINK", EnvVars: []string{"WOODPECKER_PREV_BUILD_LINK"},
Name: "prev-build-link", Name: "prev-build-link",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_SHA", EnvVars: []string{"WOODPECKER_PREV_COMMIT_SHA"},
Name: "prev-commit-sha", Name: "prev-commit-sha",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_REF", EnvVars: []string{"WOODPECKER_PREV_COMMIT_REF"},
Name: "prev-commit-ref", Name: "prev-commit-ref",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_REFSPEC", EnvVars: []string{"WOODPECKER_PREV_COMMIT_REFSPEC"},
Name: "prev-commit-refspec", Name: "prev-commit-refspec",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_BRANCH", EnvVars: []string{"WOODPECKER_PREV_COMMIT_BRANCH"},
Name: "prev-commit-branch", Name: "prev-commit-branch",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_MESSAGE", EnvVars: []string{"WOODPECKER_PREV_COMMIT_MESSAGE"},
Name: "prev-commit-message", Name: "prev-commit-message",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_AUTHOR_NAME", EnvVars: []string{"WOODPECKER_PREV_COMMIT_AUTHOR_NAME"},
Name: "prev-commit-author-name", Name: "prev-commit-author-name",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_AUTHOR_AVATAR", EnvVars: []string{"WOODPECKER_PREV_COMMIT_AUTHOR_AVATAR"},
Name: "prev-commit-author-avatar", Name: "prev-commit-author-avatar",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PREV_COMMIT_AUTHOR_EMAIL", EnvVars: []string{"WOODPECKER_PREV_COMMIT_AUTHOR_EMAIL"},
Name: "prev-commit-author-email", Name: "prev-commit-author-email",
}, },
cli.IntFlag{ &cli.IntFlag{
EnvVar: "WOODPECKER_JOB_NUMBER", EnvVars: []string{"WOODPECKER_JOB_NUMBER"},
Name: "job-number", Name: "job-number",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_ENV", EnvVars: []string{"WOODPECKER_ENV"},
Name: "env", Name: "env",
}, },
} }

View file

@ -4,25 +4,21 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
// Command exports the info command. // Command exports the info command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "info", Name: "info",
Usage: "show information about the current user", Usage: "show information about the current user",
ArgsUsage: " ", ArgsUsage: " ",
Action: info, Action: info,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplInfo, true),
Name: "format", ),
Usage: "format output",
Value: tmplInfo,
Hidden: true,
},
},
} }
func info(c *cli.Context) error { func info(c *cli.Context) error {

View file

@ -7,7 +7,7 @@ import (
"strings" "strings"
"github.com/jackspirou/syscerts" "github.com/jackspirou/syscerts"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"golang.org/x/net/proxy" "golang.org/x/net/proxy"
"golang.org/x/oauth2" "golang.org/x/oauth2"
@ -17,11 +17,11 @@ import (
// NewClient returns a new client from the CLI context. // NewClient returns a new client from the CLI context.
func NewClient(c *cli.Context) (woodpecker.Client, error) { func NewClient(c *cli.Context) (woodpecker.Client, error) {
var ( var (
skip = c.GlobalBool("skip-verify") skip = c.Bool("skip-verify")
socks = c.GlobalString("socks-proxy") socks = c.String("socks-proxy")
socksoff = c.GlobalBool("socks-proxy-off") socksoff = c.Bool("socks-proxy-off")
token = c.GlobalString("token") token = c.String("token")
server = c.GlobalString("server") server = c.String("server")
) )
server = strings.TrimRight(server, "/") server = strings.TrimRight(server, "/")

View file

@ -6,18 +6,19 @@ import (
"path/filepath" "path/filepath"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/pipeline/schema" "github.com/woodpecker-ci/woodpecker/pipeline/schema"
) )
// Command exports the info command. // Command exports the info command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "lint", Name: "lint",
Usage: "lint a pipeline configuration file", Usage: "lint a pipeline configuration file",
ArgsUsage: "[path/to/.woodpecker.yml]", ArgsUsage: "[path/to/.woodpecker.yml]",
Action: lint, Action: lint,
Flags: []cli.Flag{}, Flags: common.GlobalFlags,
} }
func lint(c *cli.Context) error { func lint(c *cli.Context) error {

View file

@ -1,12 +1,17 @@
package log package log
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the build command set. // Command exports the build command set.
var Command = cli.Command{ var Command = &cli.Command{
Name: "log", Name: "log",
Usage: "manage logs", Usage: "manage logs",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
logPurgeCmd, logPurgeCmd,
}, },
} }

View file

@ -4,16 +4,18 @@ import (
"fmt" "fmt"
"strconv" "strconv"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var logPurgeCmd = cli.Command{ var logPurgeCmd = &cli.Command{
Name: "purge", Name: "purge",
Usage: "purge a log", Usage: "purge a log",
ArgsUsage: "<repo/name> <build>", ArgsUsage: "<repo/name> <build>",
Action: logPurge, Action: logPurge,
Flags: common.GlobalFlags,
} }
func logPurge(c *cli.Context) (err error) { func logPurge(c *cli.Context) (err error) {

View file

@ -3,18 +3,20 @@ package loglevel
import ( import (
"github.com/rs/zerolog" "github.com/rs/zerolog"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
// Command exports the log-level command used to change the servers log-level. // Command exports the log-level command used to change the servers log-level.
var Command = cli.Command{ var Command = &cli.Command{
Name: "log-level", Name: "log-level",
ArgsUsage: "[level]", ArgsUsage: "[level]",
Usage: "get the logging level of the server, or set it with [level]", Usage: "get the logging level of the server, or set it with [level]",
Action: logLevel, Action: logLevel,
Flags: common.GlobalFlags,
} }
func logLevel(c *cli.Context) error { func logLevel(c *cli.Context) error {

View file

@ -1,12 +1,17 @@
package registry package registry
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the registry command set. // Command exports the registry command set.
var Command = cli.Command{ var Command = &cli.Command{
Name: "registry", Name: "registry",
Usage: "manage registries", Usage: "manage registries",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
registryCreateCmd, registryCreateCmd,
registryDeleteCmd, registryDeleteCmd,
registryUpdateCmd, registryUpdateCmd,

View file

@ -4,36 +4,37 @@ import (
"io/ioutil" "io/ioutil"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var registryCreateCmd = cli.Command{ var registryCreateCmd = &cli.Command{
Name: "add", Name: "add",
Usage: "adds a registry", Usage: "adds a registry",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: registryCreate, Action: registryCreate,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "hostname", Name: "hostname",
Usage: "registry hostname", Usage: "registry hostname",
Value: "docker.io", Value: "docker.io",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "username", Name: "username",
Usage: "registry username", Usage: "registry username",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "password", Name: "password",
Usage: "registry password", Usage: "registry password",
}, },
}, ),
} }
func registryCreate(c *cli.Context) error { func registryCreate(c *cli.Context) error {

View file

@ -4,33 +4,29 @@ import (
"html/template" "html/template"
"os" "os"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var registryInfoCmd = cli.Command{ var registryInfoCmd = &cli.Command{
Name: "info", Name: "info",
Usage: "display registry info", Usage: "display registry info",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: registryInfo, Action: registryInfo,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "hostname", Name: "hostname",
Usage: "registry hostname", Usage: "registry hostname",
Value: "docker.io", Value: "docker.io",
}, },
cli.StringFlag{ common.FormatFlag(tmplRegistryList, true),
Name: "format", ),
Usage: "format output",
Value: tmplRegistryList,
Hidden: true,
},
},
} }
func registryInfo(c *cli.Context) error { func registryInfo(c *cli.Context) error {

View file

@ -4,28 +4,24 @@ import (
"html/template" "html/template"
"os" "os"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var registryListCmd = cli.Command{ var registryListCmd = &cli.Command{
Name: "ls", Name: "ls",
Usage: "list registries", Usage: "list registries",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: registryList, Action: registryList,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ common.FormatFlag(tmplRegistryList, true),
Name: "format", ),
Usage: "format output",
Value: tmplRegistryList,
Hidden: true,
},
},
} }
func registryList(c *cli.Context) error { func registryList(c *cli.Context) error {

View file

@ -1,27 +1,28 @@
package registry package registry
import ( import (
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var registryDeleteCmd = cli.Command{ var registryDeleteCmd = &cli.Command{
Name: "rm", Name: "rm",
Usage: "remove a registry", Usage: "remove a registry",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: registryDelete, Action: registryDelete,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "hostname", Name: "hostname",
Usage: "registry hostname", Usage: "registry hostname",
Value: "docker.io", Value: "docker.io",
}, },
}, ),
} }
func registryDelete(c *cli.Context) error { func registryDelete(c *cli.Context) error {

View file

@ -4,36 +4,37 @@ import (
"io/ioutil" "io/ioutil"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var registryUpdateCmd = cli.Command{ var registryUpdateCmd = &cli.Command{
Name: "update", Name: "update",
Usage: "update a registry", Usage: "update a registry",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: registryUpdate, Action: registryUpdate,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "hostname", Name: "hostname",
Usage: "registry hostname", Usage: "registry hostname",
Value: "docker.io", Value: "docker.io",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "username", Name: "username",
Usage: "registry username", Usage: "registry username",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "password", Name: "password",
Usage: "registry password", Usage: "registry password",
}, },
}, ),
} }
func registryUpdate(c *cli.Context) error { func registryUpdate(c *cli.Context) error {

View file

@ -1,12 +1,17 @@
package repo package repo
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the repository command. // Command exports the repository command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "repo", Name: "repo",
Usage: "manage repositories", Usage: "manage repositories",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
repoListCmd, repoListCmd,
repoInfoCmd, repoInfoCmd,
repoAddCmd, repoAddCmd,

View file

@ -3,16 +3,18 @@ package repo
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoAddCmd = cli.Command{ var repoAddCmd = &cli.Command{
Name: "add", Name: "add",
Usage: "add a repository", Usage: "add a repository",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoAdd, Action: repoAdd,
Flags: common.GlobalFlags,
} }
func repoAdd(c *cli.Context) error { func repoAdd(c *cli.Context) error {

View file

@ -3,16 +3,18 @@ package repo
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoChownCmd = cli.Command{ var repoChownCmd = &cli.Command{
Name: "chown", Name: "chown",
Usage: "assume ownership of a repository", Usage: "assume ownership of a repository",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoChown, Action: repoChown,
Flags: common.GlobalFlags,
} }
func repoChown(c *cli.Context) error { func repoChown(c *cli.Context) error {

View file

@ -4,23 +4,20 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoInfoCmd = cli.Command{ var repoInfoCmd = &cli.Command{
Name: "info", Name: "info",
Usage: "show repository details", Usage: "show repository details",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoInfo, Action: repoInfo,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplRepoInfo),
Name: "format", ),
Usage: "format output",
Value: tmplRepoInfo,
},
},
} }
func repoInfo(c *cli.Context) error { func repoInfo(c *cli.Context) error {

View file

@ -4,27 +4,24 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoListCmd = cli.Command{ var repoListCmd = &cli.Command{
Name: "ls", Name: "ls",
Usage: "list all repos", Usage: "list all repos",
ArgsUsage: " ", ArgsUsage: " ",
Action: repoList, Action: repoList,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplRepoList),
Name: "format", &cli.StringFlag{
Usage: "format output",
Value: tmplRepoList,
},
cli.StringFlag{
Name: "org", Name: "org",
Usage: "filter by organization", Usage: "filter by organization",
}, },
}, ),
} }
func repoList(c *cli.Context) error { func repoList(c *cli.Context) error {

View file

@ -1,16 +1,18 @@
package repo package repo
import ( import (
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoRepairCmd = cli.Command{ var repoRepairCmd = &cli.Command{
Name: "repair", Name: "repair",
Usage: "repair repository webhooks", Usage: "repair repository webhooks",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoRepair, Action: repoRepair,
Flags: common.GlobalFlags,
} }
func repoRepair(c *cli.Context) error { func repoRepair(c *cli.Context) error {

View file

@ -3,16 +3,18 @@ package repo
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoRemoveCmd = cli.Command{ var repoRemoveCmd = &cli.Command{
Name: "rm", Name: "rm",
Usage: "remove a repository", Usage: "remove a repository",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoRemove, Action: repoRemove,
Flags: common.GlobalFlags,
} }
func repoRemove(c *cli.Context) error { func repoRemove(c *cli.Context) error {

View file

@ -4,23 +4,20 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var repoSyncCmd = cli.Command{ var repoSyncCmd = &cli.Command{
Name: "sync", Name: "sync",
Usage: "synchronize the repository list", Usage: "synchronize the repository list",
ArgsUsage: " ", ArgsUsage: " ",
Action: repoSync, Action: repoSync,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplRepoList),
Name: "format", ),
Usage: "format output",
Value: tmplRepoList,
},
},
} }
func repoSync(c *cli.Context) error { func repoSync(c *cli.Context) error {

View file

@ -4,47 +4,48 @@ import (
"fmt" "fmt"
"time" "time"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var repoUpdateCmd = cli.Command{ var repoUpdateCmd = &cli.Command{
Name: "update", Name: "update",
Usage: "update a repository", Usage: "update a repository",
ArgsUsage: "<repo/name>", ArgsUsage: "<repo/name>",
Action: repoUpdate, Action: repoUpdate,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.BoolFlag{ &cli.BoolFlag{
Name: "trusted", Name: "trusted",
Usage: "repository is trusted", Usage: "repository is trusted",
}, },
cli.BoolFlag{ &cli.BoolFlag{
Name: "gated", Name: "gated",
Usage: "repository is gated", Usage: "repository is gated",
}, },
cli.DurationFlag{ &cli.DurationFlag{
Name: "timeout", Name: "timeout",
Usage: "repository timeout", Usage: "repository timeout",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "visibility", Name: "visibility",
Usage: "repository visibility", Usage: "repository visibility",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "config", Name: "config",
Usage: "repository configuration path (e.g. .woodpecker.yml)", Usage: "repository configuration path (e.g. .woodpecker.yml)",
}, },
cli.IntFlag{ &cli.IntFlag{
Name: "build-counter", Name: "build-counter",
Usage: "repository starting build number", Usage: "repository starting build number",
}, },
cli.BoolFlag{ &cli.BoolFlag{
Name: "unsafe", Name: "unsafe",
Usage: "validate updating the build-counter is unsafe", Usage: "validate updating the build-counter is unsafe",
}, },
}, ),
} }
func repoUpdate(c *cli.Context) error { func repoUpdate(c *cli.Context) error {

View file

@ -1,12 +1,17 @@
package secret package secret
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the secret command. // Command exports the secret command.
var Command = cli.Command{ var Command = &cli.Command{
Name: "secret", Name: "secret",
Usage: "manage secrets", Usage: "manage secrets",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
secretCreateCmd, secretCreateCmd,
secretDeleteCmd, secretDeleteCmd,
secretUpdateCmd, secretUpdateCmd,

View file

@ -4,39 +4,40 @@ import (
"io/ioutil" "io/ioutil"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var secretCreateCmd = cli.Command{ var secretCreateCmd = &cli.Command{
Name: "add", Name: "add",
Usage: "adds a secret", Usage: "adds a secret",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: secretCreate, Action: secretCreate,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "name", Name: "name",
Usage: "secret name", Usage: "secret name",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "value", Name: "value",
Usage: "secret value", Usage: "secret value",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "event", Name: "event",
Usage: "secret limited to these events", Usage: "secret limited to these events",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "image", Name: "image",
Usage: "secret limited to these images", Usage: "secret limited to these images",
}, },
}, ),
} }
func secretCreate(c *cli.Context) error { func secretCreate(c *cli.Context) error {

View file

@ -4,32 +4,28 @@ import (
"html/template" "html/template"
"os" "os"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var secretInfoCmd = cli.Command{ var secretInfoCmd = &cli.Command{
Name: "info", Name: "info",
Usage: "display secret info", Usage: "display secret info",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: secretInfo, Action: secretInfo,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "name", Name: "name",
Usage: "secret name", Usage: "secret name",
}, },
cli.StringFlag{ common.FormatFlag(tmplSecretList, true),
Name: "format", ),
Usage: "format output",
Value: tmplSecretList,
Hidden: true,
},
},
} }
func secretInfo(c *cli.Context) error { func secretInfo(c *cli.Context) error {

View file

@ -5,28 +5,24 @@ import (
"os" "os"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var secretListCmd = cli.Command{ var secretListCmd = &cli.Command{
Name: "ls", Name: "ls",
Usage: "list secrets", Usage: "list secrets",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: secretList, Action: secretList,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ common.FormatFlag(tmplSecretList, true),
Name: "format", ),
Usage: "format output",
Value: tmplSecretList,
Hidden: true,
},
},
} }
func secretList(c *cli.Context) error { func secretList(c *cli.Context) error {

View file

@ -1,26 +1,27 @@
package secret package secret
import ( import (
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var secretDeleteCmd = cli.Command{ var secretDeleteCmd = &cli.Command{
Name: "rm", Name: "rm",
Usage: "remove a secret", Usage: "remove a secret",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: secretDelete, Action: secretDelete,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "name", Name: "name",
Usage: "secret name", Usage: "secret name",
}, },
}, ),
} }
func secretDelete(c *cli.Context) error { func secretDelete(c *cli.Context) error {

View file

@ -4,39 +4,40 @@ import (
"io/ioutil" "io/ioutil"
"strings" "strings"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var secretUpdateCmd = cli.Command{ var secretUpdateCmd = &cli.Command{
Name: "update", Name: "update",
Usage: "update a secret", Usage: "update a secret",
ArgsUsage: "[repo/name]", ArgsUsage: "[repo/name]",
Action: secretUpdate, Action: secretUpdate,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ &cli.StringFlag{
Name: "repository", Name: "repository",
Usage: "repository name (e.g. octocat/hello-world)", Usage: "repository name (e.g. octocat/hello-world)",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "name", Name: "name",
Usage: "secret name", Usage: "secret name",
}, },
cli.StringFlag{ &cli.StringFlag{
Name: "value", Name: "value",
Usage: "secret value", Usage: "secret value",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "event", Name: "event",
Usage: "secret limited to these events", Usage: "secret limited to these events",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
Name: "image", Name: "image",
Usage: "secret limited to these images", Usage: "secret limited to these images",
}, },
}, ),
} }
func secretUpdate(c *cli.Context) error { func secretUpdate(c *cli.Context) error {

View file

@ -1,12 +1,17 @@
package user package user
import "github.com/urfave/cli" import (
"github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
)
// Command exports the user command set. // Command exports the user command set.
var Command = cli.Command{ var Command = &cli.Command{
Name: "user", Name: "user",
Usage: "manage users", Usage: "manage users",
Subcommands: []cli.Command{ Flags: common.GlobalFlags,
Subcommands: []*cli.Command{
userListCmd, userListCmd,
userInfoCmd, userInfoCmd,
userAddCmd, userAddCmd,

View file

@ -3,17 +3,19 @@ package user
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
"github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker" "github.com/woodpecker-ci/woodpecker/woodpecker-go/woodpecker"
) )
var userAddCmd = cli.Command{ var userAddCmd = &cli.Command{
Name: "add", Name: "add",
Usage: "adds a user", Usage: "adds a user",
ArgsUsage: "<username>", ArgsUsage: "<username>",
Action: userAdd, Action: userAdd,
Flags: common.GlobalFlags,
} }
func userAdd(c *cli.Context) error { func userAdd(c *cli.Context) error {

View file

@ -5,23 +5,20 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var userInfoCmd = cli.Command{ var userInfoCmd = &cli.Command{
Name: "info", Name: "info",
Usage: "show user details", Usage: "show user details",
ArgsUsage: "<username>", ArgsUsage: "<username>",
Action: userInfo, Action: userInfo,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplUserInfo),
Name: "format", ),
Usage: "format output",
Value: tmplUserInfo,
},
},
} }
func userInfo(c *cli.Context) error { func userInfo(c *cli.Context) error {

View file

@ -4,23 +4,20 @@ import (
"os" "os"
"text/template" "text/template"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var userListCmd = cli.Command{ var userListCmd = &cli.Command{
Name: "ls", Name: "ls",
Usage: "list all users", Usage: "list all users",
ArgsUsage: " ", ArgsUsage: " ",
Action: userList, Action: userList,
Flags: []cli.Flag{ Flags: append(common.GlobalFlags,
cli.StringFlag{ common.FormatFlag(tmplUserList),
Name: "format", ),
Usage: "format output",
Value: tmplUserList,
},
},
} }
func userList(c *cli.Context) error { func userList(c *cli.Context) error {

View file

@ -3,16 +3,18 @@ package user
import ( import (
"fmt" "fmt"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/internal" "github.com/woodpecker-ci/woodpecker/cli/internal"
) )
var userRemoveCmd = cli.Command{ var userRemoveCmd = &cli.Command{
Name: "rm", Name: "rm",
Usage: "remove a user", Usage: "remove a user",
ArgsUsage: "<username>", ArgsUsage: "<username>",
Action: userRemove, Action: userRemove,
Flags: common.GlobalFlags,
} }
func userRemove(c *cli.Context) error { func userRemove(c *cli.Context) error {

View file

@ -24,7 +24,7 @@ import (
"github.com/rs/zerolog" "github.com/rs/zerolog"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/tevino/abool" "github.com/tevino/abool"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"google.golang.org/grpc" "google.golang.org/grpc"
grpccredentials "google.golang.org/grpc/credentials" grpccredentials "google.golang.org/grpc/credentials"
"google.golang.org/grpc/keepalive" "google.golang.org/grpc/keepalive"
@ -52,13 +52,13 @@ func loop(c *cli.Context) error {
log.Logger = log.Output( log.Logger = log.Output(
zerolog.ConsoleWriter{ zerolog.ConsoleWriter{
Out: os.Stderr, Out: os.Stderr,
NoColor: c.BoolT("nocolor"), NoColor: c.Bool("nocolor"),
}, },
) )
} }
zerolog.SetGlobalLevel(zerolog.WarnLevel) zerolog.SetGlobalLevel(zerolog.WarnLevel)
if c.BoolT("debug") { if c.Bool("debug") {
if c.IsSet("debug") { if c.IsSet("debug") {
log.Warn().Msg("--debug is deprecated, use --log-level instead") log.Warn().Msg("--debug is deprecated, use --log-level instead")
} }
@ -77,8 +77,12 @@ func loop(c *cli.Context) error {
counter.Polling = c.Int("max-procs") counter.Polling = c.Int("max-procs")
counter.Running = 0 counter.Running = 0
if c.BoolT("healthcheck") { if c.Bool("healthcheck") {
go http.ListenAndServe(":3000", nil) go func() {
if err := http.ListenAndServe(":3000", nil); err != nil {
log.Error().Msgf("can not listen on port 3000: %v", err)
}
}()
} }
// TODO pass version information to grpc server // TODO pass version information to grpc server

View file

@ -17,93 +17,97 @@ package main
import ( import (
"time" "time"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
) )
var flags = []cli.Flag{ var flags = []cli.Flag{
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SERVER", EnvVars: []string{"WOODPECKER_SERVER"},
Name: "server", Name: "server",
Usage: "server address", Usage: "server address",
Value: "localhost:9000", Value: "localhost:9000",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_USERNAME", EnvVars: []string{"WOODPECKER_USERNAME"},
Name: "username", Name: "username",
Usage: "auth username", Usage: "auth username",
Value: "x-oauth-basic", Value: "x-oauth-basic",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_AGENT_SECRET", EnvVars: []string{"WOODPECKER_AGENT_SECRET"},
Name: "password", Name: "password",
Usage: "server-agent shared password", Usage: "server-agent shared password",
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG", EnvVars: []string{"WOODPECKER_DEBUG"},
Name: "debug", Name: "debug",
Usage: "enable agent debug mode", Usage: "enable agent debug mode",
Value: true,
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_LOG_LEVEL", EnvVars: []string{"WOODPECKER_LOG_LEVEL"},
Name: "log-level", Name: "log-level",
Usage: "set logging level", Usage: "set logging level",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG_PRETTY", EnvVars: []string{"WOODPECKER_DEBUG_PRETTY"},
Name: "pretty", Name: "pretty",
Usage: "enable pretty-printed debug output", Usage: "enable pretty-printed debug output",
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG_NOCOLOR", EnvVars: []string{"WOODPECKER_DEBUG_NOCOLOR"},
Name: "nocolor", Name: "nocolor",
Usage: "disable colored debug output", Usage: "disable colored debug output",
Value: true,
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_HOSTNAME", EnvVars: []string{"WOODPECKER_HOSTNAME"},
Name: "hostname", Name: "hostname",
Usage: "agent hostname", Usage: "agent hostname",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PLATFORM", EnvVars: []string{"WOODPECKER_PLATFORM"},
Name: "platform", Name: "platform",
Usage: "restrict builds by platform conditions", Usage: "restrict builds by platform conditions",
Value: "linux/amd64", Value: "linux/amd64",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_FILTER", EnvVars: []string{"WOODPECKER_FILTER"},
Name: "filter", Name: "filter",
Usage: "filter expression to restrict builds by label", Usage: "filter expression to restrict builds by label",
}, },
cli.IntFlag{ &cli.IntFlag{
EnvVar: "WOODPECKER_MAX_PROCS", EnvVars: []string{"WOODPECKER_MAX_PROCS"},
Name: "max-procs", Name: "max-procs",
Usage: "agent parallel builds", Usage: "agent parallel builds",
Value: 1, Value: 1,
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_HEALTHCHECK", EnvVars: []string{"WOODPECKER_HEALTHCHECK"},
Name: "healthcheck", Name: "healthcheck",
Usage: "enable healthcheck endpoint", Usage: "enable healthcheck endpoint",
Value: true,
}, },
cli.DurationFlag{ &cli.DurationFlag{
EnvVar: "WOODPECKER_KEEPALIVE_TIME", EnvVars: []string{"WOODPECKER_KEEPALIVE_TIME"},
Name: "keepalive-time", Name: "keepalive-time",
Usage: "after a duration of this time of no activity, the agent pings the server to check if the transport is still alive", Usage: "after a duration of this time of no activity, the agent pings the server to check if the transport is still alive",
}, },
cli.DurationFlag{ &cli.DurationFlag{
EnvVar: "WOODPECKER_KEEPALIVE_TIMEOUT", EnvVars: []string{"WOODPECKER_KEEPALIVE_TIMEOUT"},
Name: "keepalive-timeout", Name: "keepalive-timeout",
Usage: "after pinging for a keepalive check, the agent waits for a duration of this time before closing the connection if no activity", Usage: "after pinging for a keepalive check, the agent waits for a duration of this time before closing the connection if no activity",
Value: time.Second * 20, Value: time.Second * 20,
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GRPC_SECURE", EnvVars: []string{"WOODPECKER_GRPC_SECURE"},
Name: "secure-grpc", Name: "secure-grpc",
Usage: "should the connection to WOODPECKER_SERVER be made using a secure transport", Usage: "should the connection to WOODPECKER_SERVER be made using a secure transport",
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GRPC_VERIFY", EnvVars: []string{"WOODPECKER_GRPC_VERIFY"},
Name: "skip-insecure-grpc", Name: "skip-insecure-grpc",
Usage: "should the grpc server certificate be verified, only valid when WOODPECKER_GRPC_SECURE is true", Usage: "should the grpc server certificate be verified, only valid when WOODPECKER_GRPC_SECURE is true",
Value: true,
}, },
} }

View file

@ -19,7 +19,7 @@ import (
"fmt" "fmt"
"net/http" "net/http"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/agent" "github.com/woodpecker-ci/woodpecker/agent"
"github.com/woodpecker-ci/woodpecker/version" "github.com/woodpecker-ci/woodpecker/version"

View file

@ -19,7 +19,7 @@ import (
"os" "os"
_ "github.com/joho/godotenv/autoload" _ "github.com/joho/godotenv/autoload"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/version" "github.com/woodpecker-ci/woodpecker/version"
) )
@ -30,7 +30,7 @@ func main() {
app.Version = version.String() app.Version = version.String()
app.Usage = "woodpecker agent" app.Usage = "woodpecker agent"
app.Action = loop app.Action = loop
app.Commands = []cli.Command{ app.Commands = []*cli.Command{
{ {
Name: "ping", Name: "ping",
Usage: "ping the agent", Usage: "ping the agent",

View file

@ -20,9 +20,10 @@ import (
_ "github.com/joho/godotenv/autoload" _ "github.com/joho/godotenv/autoload"
"github.com/rs/zerolog" "github.com/rs/zerolog"
zlog "github.com/rs/zerolog/log" zlog "github.com/rs/zerolog/log"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/cli/build" "github.com/woodpecker-ci/woodpecker/cli/build"
"github.com/woodpecker-ci/woodpecker/cli/common"
"github.com/woodpecker-ci/woodpecker/cli/deploy" "github.com/woodpecker-ci/woodpecker/cli/deploy"
"github.com/woodpecker-ci/woodpecker/cli/exec" "github.com/woodpecker-ci/woodpecker/cli/exec"
"github.com/woodpecker-ci/woodpecker/cli/info" "github.com/woodpecker-ci/woodpecker/cli/info"
@ -42,40 +43,8 @@ func main() {
app.Version = version.String() app.Version = version.String()
app.Usage = "command line utility" app.Usage = "command line utility"
app.EnableBashCompletion = true app.EnableBashCompletion = true
app.Flags = []cli.Flag{ app.Flags = common.GlobalFlags
cli.StringFlag{ app.Commands = []*cli.Command{
EnvVar: "WOODPECKER_TOKEN",
// TODO: rename to `token`
Name: "t, token",
Usage: "server auth token",
},
cli.StringFlag{
EnvVar: "WOODPECKER_SERVER",
// TODO: rename to `server`
Name: "s, server",
Usage: "server address",
},
cli.BoolFlag{
EnvVar: "WOODPECKER_SKIP_VERIFY",
Name: "skip-verify",
Usage: "skip ssl verification",
Hidden: true,
},
cli.StringFlag{
EnvVar: "SOCKS_PROXY",
Name: "socks-proxy",
Usage: "socks proxy address",
Hidden: true,
},
cli.BoolFlag{
EnvVar: "SOCKS_PROXY_OFF",
Name: "socks-proxy-off",
Usage: "socks proxy ignored",
Hidden: true,
},
}
app.Commands = []cli.Command{
build.Command, build.Command,
log.Command, log.Command,
deploy.Command, deploy.Command,

View file

@ -17,165 +17,166 @@ package main
import ( import (
"time" "time"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
) )
var flags = []cli.Flag{ var flags = []cli.Flag{
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG", EnvVars: []string{"WOODPECKER_DEBUG"},
Name: "debug", Name: "debug",
Usage: "enable server debug mode", Usage: "enable server debug mode",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_LOG_LEVEL", EnvVars: []string{"WOODPECKER_LOG_LEVEL"},
Name: "log-level", Name: "log-level",
Usage: "set logging level", Usage: "set logging level",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG_PRETTY", EnvVars: []string{"WOODPECKER_DEBUG_PRETTY"},
Name: "pretty", Name: "pretty",
Usage: "enable pretty-printed debug output", Usage: "enable pretty-printed debug output",
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_DEBUG_NOCOLOR", EnvVars: []string{"WOODPECKER_DEBUG_NOCOLOR"},
Name: "nocolor", Name: "nocolor",
Usage: "disable colored debug output", Usage: "disable colored debug output",
Value: true,
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_HOST", EnvVars: []string{"WOODPECKER_HOST"},
Name: "server-host", Name: "server-host",
Usage: "server fully qualified url (<scheme>://<host>)", Usage: "server fully qualified url (<scheme>://<host>)",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SERVER_ADDR", EnvVars: []string{"WOODPECKER_SERVER_ADDR"},
Name: "server-addr", Name: "server-addr",
Usage: "server address", Usage: "server address",
Value: ":8000", Value: ":8000",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SERVER_CERT", EnvVars: []string{"WOODPECKER_SERVER_CERT"},
Name: "server-cert", Name: "server-cert",
Usage: "server ssl cert path", Usage: "server ssl cert path",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SERVER_KEY", EnvVars: []string{"WOODPECKER_SERVER_KEY"},
Name: "server-key", Name: "server-key",
Usage: "server ssl key path", Usage: "server ssl key path",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GRPC_ADDR", EnvVars: []string{"WOODPECKER_GRPC_ADDR"},
Name: "grpc-addr", Name: "grpc-addr",
Usage: "grpc address", Usage: "grpc address",
Value: ":9000", Value: ":9000",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_LETS_ENCRYPT", EnvVars: []string{"WOODPECKER_LETS_ENCRYPT"},
Name: "lets-encrypt", Name: "lets-encrypt",
Usage: "enable let's encrypt", Usage: "enable let's encrypt",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_QUIC", EnvVars: []string{"WOODPECKER_QUIC"},
Name: "quic", Name: "quic",
Usage: "enable quic", Usage: "enable quic",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_WWW_PROXY", EnvVars: []string{"WOODPECKER_WWW_PROXY"},
Name: "www-proxy", Name: "www-proxy",
Usage: "serve the website by using a proxy (used for development)", Usage: "serve the website by using a proxy (used for development)",
Hidden: true, Hidden: true,
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_ADMIN", EnvVars: []string{"WOODPECKER_ADMIN"},
Name: "admin", Name: "admin",
Usage: "list of admin users", Usage: "list of admin users",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_ORGS", EnvVars: []string{"WOODPECKER_ORGS"},
Name: "orgs", Name: "orgs",
Usage: "list of approved organizations", Usage: "list of approved organizations",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_REPO_OWNERS", EnvVars: []string{"WOODPECKER_REPO_OWNERS"},
Name: "repo-owners", Name: "repo-owners",
Usage: "List of syncable repo owners", Usage: "List of syncable repo owners",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_OPEN", EnvVars: []string{"WOODPECKER_OPEN"},
Name: "open", Name: "open",
Usage: "enable open user registration", Usage: "enable open user registration",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_DOCS", EnvVars: []string{"WOODPECKER_DOCS"},
Name: "docs", Name: "docs",
Usage: "link to user documentation", Usage: "link to user documentation",
Value: "https://woodpecker-ci.org/", Value: "https://woodpecker-ci.org/",
}, },
cli.DurationFlag{ &cli.DurationFlag{
EnvVar: "WOODPECKER_SESSION_EXPIRES", EnvVars: []string{"WOODPECKER_SESSION_EXPIRES"},
Name: "session-expires", Name: "session-expires",
Usage: "session expiration time", Usage: "session expiration time",
Value: time.Hour * 72, Value: time.Hour * 72,
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_ESCALATE", EnvVars: []string{"WOODPECKER_ESCALATE"},
Name: "escalate", Name: "escalate",
Usage: "images to run in privileged mode", Usage: "images to run in privileged mode",
Value: &cli.StringSlice{ Value: cli.NewStringSlice(
"plugins/docker", "plugins/docker",
"plugins/gcr", "plugins/gcr",
"plugins/ecr", "plugins/ecr",
),
}, },
}, &cli.StringSliceFlag{
cli.StringSliceFlag{ EnvVars: []string{"WOODPECKER_VOLUME"},
EnvVar: "WOODPECKER_VOLUME",
Name: "volume", Name: "volume",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_DOCKER_CONFIG", EnvVars: []string{"WOODPECKER_DOCKER_CONFIG"},
Name: "docker-config", Name: "docker-config",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_ENVIRONMENT", EnvVars: []string{"WOODPECKER_ENVIRONMENT"},
Name: "environment", Name: "environment",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_NETWORK", EnvVars: []string{"WOODPECKER_NETWORK"},
Name: "network", Name: "network",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_AGENT_SECRET", EnvVars: []string{"WOODPECKER_AGENT_SECRET"},
Name: "agent-secret", Name: "agent-secret",
Usage: "server-agent shared password", Usage: "server-agent shared password",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_SECRET_ENDPOINT", EnvVars: []string{"WOODPECKER_SECRET_ENDPOINT"},
Name: "secret-service", Name: "secret-service",
Usage: "secret plugin endpoint", Usage: "secret plugin endpoint",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_REGISTRY_ENDPOINT", EnvVars: []string{"WOODPECKER_REGISTRY_ENDPOINT"},
Name: "registry-service", Name: "registry-service",
Usage: "registry plugin endpoint", Usage: "registry plugin endpoint",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GATEKEEPER_ENDPOINT", EnvVars: []string{"WOODPECKER_GATEKEEPER_ENDPOINT"},
Name: "gating-service", Name: "gating-service",
Usage: "gated build endpoint", Usage: "gated build endpoint",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_DATABASE_DRIVER", EnvVars: []string{"WOODPECKER_DATABASE_DRIVER"},
Name: "driver", Name: "driver",
Usage: "database driver", Usage: "database driver",
Value: "sqlite3", Value: "sqlite3",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_DATABASE_DATASOURCE", EnvVars: []string{"WOODPECKER_DATABASE_DATASOURCE"},
Name: "datasource", Name: "datasource",
Usage: "database driver configuration string", Usage: "database driver configuration string",
Value: "drone.sqlite", Value: "drone.sqlite",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_PROMETHEUS_AUTH_TOKEN", EnvVars: []string{"WOODPECKER_PROMETHEUS_AUTH_TOKEN"},
Name: "prometheus-auth-token", Name: "prometheus-auth-token",
Usage: "token to secure prometheus metrics endpoint", Usage: "token to secure prometheus metrics endpoint",
Value: "", Value: "",
@ -183,330 +184,331 @@ var flags = []cli.Flag{
// //
// resource limit parameters // resource limit parameters
// //
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_LIMIT_MEM_SWAP", EnvVars: []string{"WOODPECKER_LIMIT_MEM_SWAP"},
Name: "limit-mem-swap", Name: "limit-mem-swap",
Usage: "maximum swappable memory allowed in bytes", Usage: "maximum swappable memory allowed in bytes",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_LIMIT_MEM", EnvVars: []string{"WOODPECKER_LIMIT_MEM"},
Name: "limit-mem", Name: "limit-mem",
Usage: "maximum memory allowed in bytes", Usage: "maximum memory allowed in bytes",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_LIMIT_SHM_SIZE", EnvVars: []string{"WOODPECKER_LIMIT_SHM_SIZE"},
Name: "limit-shm-size", Name: "limit-shm-size",
Usage: "docker compose /dev/shm allowed in bytes", Usage: "docker compose /dev/shm allowed in bytes",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_LIMIT_CPU_QUOTA", EnvVars: []string{"WOODPECKER_LIMIT_CPU_QUOTA"},
Name: "limit-cpu-quota", Name: "limit-cpu-quota",
Usage: "impose a cpu quota", Usage: "impose a cpu quota",
}, },
cli.Int64Flag{ &cli.Int64Flag{
EnvVar: "WOODPECKER_LIMIT_CPU_SHARES", EnvVars: []string{"WOODPECKER_LIMIT_CPU_SHARES"},
Name: "limit-cpu-shares", Name: "limit-cpu-shares",
Usage: "change the cpu shares", Usage: "change the cpu shares",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_LIMIT_CPU_SET", EnvVars: []string{"WOODPECKER_LIMIT_CPU_SET"},
Name: "limit-cpu-set", Name: "limit-cpu-set",
Usage: "set the cpus allowed to execute containers", Usage: "set the cpus allowed to execute containers",
}, },
// //
// remote parameters // remote parameters
// //
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITHUB", EnvVars: []string{"WOODPECKER_GITHUB"},
Name: "github", Name: "github",
Usage: "github driver is enabled", Usage: "github driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITHUB_URL", EnvVars: []string{"WOODPECKER_GITHUB_URL"},
Name: "github-server", Name: "github-server",
Usage: "github server address", Usage: "github server address",
Value: "https://github.com", Value: "https://github.com",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITHUB_CONTEXT", EnvVars: []string{"WOODPECKER_GITHUB_CONTEXT"},
Name: "github-context", Name: "github-context",
Usage: "github status context", Usage: "github status context",
Value: "continuous-integration/woodpecker", Value: "continuous-integration/woodpecker",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITHUB_CLIENT", EnvVars: []string{"WOODPECKER_GITHUB_CLIENT"},
Name: "github-client", Name: "github-client",
Usage: "github oauth2 client id", Usage: "github oauth2 client id",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITHUB_SECRET", EnvVars: []string{"WOODPECKER_GITHUB_SECRET"},
Name: "github-secret", Name: "github-secret",
Usage: "github oauth2 client secret", Usage: "github oauth2 client secret",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_GITHUB_SCOPE", EnvVars: []string{"WOODPECKER_GITHUB_SCOPE"},
Name: "github-scope", Name: "github-scope",
Usage: "github oauth scope", Usage: "github oauth scope",
Value: &cli.StringSlice{ Value: cli.NewStringSlice(
"repo", "repo",
"repo:status", "repo:status",
"user:email", "user:email",
"read:org", "read:org",
),
}, },
}, &cli.StringFlag{
cli.StringFlag{ EnvVars: []string{"WOODPECKER_GITHUB_GIT_USERNAME"},
EnvVar: "WOODPECKER_GITHUB_GIT_USERNAME",
Name: "github-git-username", Name: "github-git-username",
Usage: "github machine user username", Usage: "github machine user username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITHUB_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_GITHUB_GIT_PASSWORD"},
Name: "github-git-password", Name: "github-git-password",
Usage: "github machine user password", Usage: "github machine user password",
}, },
cli.BoolTFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITHUB_MERGE_REF", EnvVars: []string{"WOODPECKER_GITHUB_MERGE_REF"},
Name: "github-merge-ref", Name: "github-merge-ref",
Usage: "github pull requests use merge ref", Usage: "github pull requests use merge ref",
Value: true,
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITHUB_PRIVATE_MODE", EnvVars: []string{"WOODPECKER_GITHUB_PRIVATE_MODE"},
Name: "github-private-mode", Name: "github-private-mode",
Usage: "github is running in private mode", Usage: "github is running in private mode",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITHUB_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_GITHUB_SKIP_VERIFY"},
Name: "github-skip-verify", Name: "github-skip-verify",
Usage: "github skip ssl verification", Usage: "github skip ssl verification",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GOGS", EnvVars: []string{"WOODPECKER_GOGS"},
Name: "gogs", Name: "gogs",
Usage: "gogs driver is enabled", Usage: "gogs driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GOGS_URL", EnvVars: []string{"WOODPECKER_GOGS_URL"},
Name: "gogs-server", Name: "gogs-server",
Usage: "gogs server address", Usage: "gogs server address",
Value: "https://github.com", Value: "https://github.com",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GOGS_GIT_USERNAME", EnvVars: []string{"WOODPECKER_GOGS_GIT_USERNAME"},
Name: "gogs-git-username", Name: "gogs-git-username",
Usage: "gogs service account username", Usage: "gogs service account username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GOGS_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_GOGS_GIT_PASSWORD"},
Name: "gogs-git-password", Name: "gogs-git-password",
Usage: "gogs service account password", Usage: "gogs service account password",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GOGS_PRIVATE_MODE", EnvVars: []string{"WOODPECKER_GOGS_PRIVATE_MODE"},
Name: "gogs-private-mode", Name: "gogs-private-mode",
Usage: "gogs private mode enabled", Usage: "gogs private mode enabled",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GOGS_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_GOGS_SKIP_VERIFY"},
Name: "gogs-skip-verify", Name: "gogs-skip-verify",
Usage: "gogs skip ssl verification", Usage: "gogs skip ssl verification",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITEA", EnvVars: []string{"WOODPECKER_GITEA"},
Name: "gitea", Name: "gitea",
Usage: "gitea driver is enabled", Usage: "gitea driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_URL", EnvVars: []string{"WOODPECKER_GITEA_URL"},
Name: "gitea-server", Name: "gitea-server",
Usage: "gitea server address", Usage: "gitea server address",
Value: "https://try.gitea.io", Value: "https://try.gitea.io",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_CLIENT", EnvVars: []string{"WOODPECKER_GITEA_CLIENT"},
Name: "gitea-client", Name: "gitea-client",
Usage: "gitea oauth2 client id", Usage: "gitea oauth2 client id",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_SECRET", EnvVars: []string{"WOODPECKER_GITEA_SECRET"},
Name: "gitea-secret", Name: "gitea-secret",
Usage: "gitea oauth2 client secret", Usage: "gitea oauth2 client secret",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_CONTEXT", EnvVars: []string{"WOODPECKER_GITEA_CONTEXT"},
Name: "gitea-context", Name: "gitea-context",
Usage: "gitea status context", Usage: "gitea status context",
Value: "continuous-integration/woodpecker", Value: "continuous-integration/woodpecker",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_GIT_USERNAME", EnvVars: []string{"WOODPECKER_GITEA_GIT_USERNAME"},
Name: "gitea-git-username", Name: "gitea-git-username",
Usage: "gitea service account username", Usage: "gitea service account username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITEA_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_GITEA_GIT_PASSWORD"},
Name: "gitea-git-password", Name: "gitea-git-password",
Usage: "gitea service account password", Usage: "gitea service account password",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITEA_PRIVATE_MODE", EnvVars: []string{"WOODPECKER_GITEA_PRIVATE_MODE"},
Name: "gitea-private-mode", Name: "gitea-private-mode",
Usage: "gitea private mode enabled", Usage: "gitea private mode enabled",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITEA_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_GITEA_SKIP_VERIFY"},
Name: "gitea-skip-verify", Name: "gitea-skip-verify",
Usage: "gitea skip ssl verification", Usage: "gitea skip ssl verification",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_BITBUCKET", EnvVars: []string{"WOODPECKER_BITBUCKET"},
Name: "bitbucket", Name: "bitbucket",
Usage: "bitbucket driver is enabled", Usage: "bitbucket driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BITBUCKET_CLIENT", EnvVars: []string{"WOODPECKER_BITBUCKET_CLIENT"},
Name: "bitbucket-client", Name: "bitbucket-client",
Usage: "bitbucket oauth2 client id", Usage: "bitbucket oauth2 client id",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_BITBUCKET_SECRET", EnvVars: []string{"WOODPECKER_BITBUCKET_SECRET"},
Name: "bitbucket-secret", Name: "bitbucket-secret",
Usage: "bitbucket oauth2 client secret", Usage: "bitbucket oauth2 client secret",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITLAB", EnvVars: []string{"WOODPECKER_GITLAB"},
Name: "gitlab", Name: "gitlab",
Usage: "gitlab driver is enabled", Usage: "gitlab driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITLAB_URL", EnvVars: []string{"WOODPECKER_GITLAB_URL"},
Name: "gitlab-server", Name: "gitlab-server",
Usage: "gitlab server address", Usage: "gitlab server address",
Value: "https://gitlab.com", Value: "https://gitlab.com",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITLAB_CLIENT", EnvVars: []string{"WOODPECKER_GITLAB_CLIENT"},
Name: "gitlab-client", Name: "gitlab-client",
Usage: "gitlab oauth2 client id", Usage: "gitlab oauth2 client id",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITLAB_SECRET", EnvVars: []string{"WOODPECKER_GITLAB_SECRET"},
Name: "gitlab-secret", Name: "gitlab-secret",
Usage: "gitlab oauth2 client secret", Usage: "gitlab oauth2 client secret",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITLAB_GIT_USERNAME", EnvVars: []string{"WOODPECKER_GITLAB_GIT_USERNAME"},
Name: "gitlab-git-username", Name: "gitlab-git-username",
Usage: "gitlab service account username", Usage: "gitlab service account username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_GITLAB_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_GITLAB_GIT_PASSWORD"},
Name: "gitlab-git-password", Name: "gitlab-git-password",
Usage: "gitlab service account password", Usage: "gitlab service account password",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITLAB_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_GITLAB_SKIP_VERIFY"},
Name: "gitlab-skip-verify", Name: "gitlab-skip-verify",
Usage: "gitlab skip ssl verification", Usage: "gitlab skip ssl verification",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_GITLAB_PRIVATE_MODE", EnvVars: []string{"WOODPECKER_GITLAB_PRIVATE_MODE"},
Name: "gitlab-private-mode", Name: "gitlab-private-mode",
Usage: "gitlab is running in private mode", Usage: "gitlab is running in private mode",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_STASH", EnvVars: []string{"WOODPECKER_STASH"},
Name: "stash", Name: "stash",
Usage: "stash driver is enabled", Usage: "stash driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_URL", EnvVars: []string{"WOODPECKER_STASH_URL"},
Name: "stash-server", Name: "stash-server",
Usage: "stash server address", Usage: "stash server address",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_CONSUMER_KEY", EnvVars: []string{"WOODPECKER_STASH_CONSUMER_KEY"},
Name: "stash-consumer-key", Name: "stash-consumer-key",
Usage: "stash oauth1 consumer key", Usage: "stash oauth1 consumer key",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_CONSUMER_RSA", EnvVars: []string{"WOODPECKER_STASH_CONSUMER_RSA"},
Name: "stash-consumer-rsa", Name: "stash-consumer-rsa",
Usage: "stash oauth1 private key file", Usage: "stash oauth1 private key file",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_CONSUMER_RSA_STRING", EnvVars: []string{"WOODPECKER_STASH_CONSUMER_RSA_STRING"},
Name: "stash-consumer-rsa-string", Name: "stash-consumer-rsa-string",
Usage: "stash oauth1 private key string", Usage: "stash oauth1 private key string",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_GIT_USERNAME", EnvVars: []string{"WOODPECKER_STASH_GIT_USERNAME"},
Name: "stash-git-username", Name: "stash-git-username",
Usage: "stash service account username", Usage: "stash service account username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_STASH_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_STASH_GIT_PASSWORD"},
Name: "stash-git-password", Name: "stash-git-password",
Usage: "stash service account password", Usage: "stash service account password",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_STASH_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_STASH_SKIP_VERIFY"},
Name: "stash-skip-verify", Name: "stash-skip-verify",
Usage: "stash skip ssl verification", Usage: "stash skip ssl verification",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_CODING", EnvVars: []string{"WOODPECKER_CODING"},
Name: "coding", Name: "coding",
Usage: "coding driver is enabled", Usage: "coding driver is enabled",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_CODING_URL", EnvVars: []string{"WOODPECKER_CODING_URL"},
Name: "coding-server", Name: "coding-server",
Usage: "coding server address", Usage: "coding server address",
Value: "https://coding.net", Value: "https://coding.net",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_CODING_CLIENT", EnvVars: []string{"WOODPECKER_CODING_CLIENT"},
Name: "coding-client", Name: "coding-client",
Usage: "coding oauth2 client id", Usage: "coding oauth2 client id",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_CODING_SECRET", EnvVars: []string{"WOODPECKER_CODING_SECRET"},
Name: "coding-secret", Name: "coding-secret",
Usage: "coding oauth2 client secret", Usage: "coding oauth2 client secret",
}, },
cli.StringSliceFlag{ &cli.StringSliceFlag{
EnvVar: "WOODPECKER_CODING_SCOPE", EnvVars: []string{"WOODPECKER_CODING_SCOPE"},
Name: "coding-scope", Name: "coding-scope",
Usage: "coding oauth scope", Usage: "coding oauth scope",
Value: &cli.StringSlice{ Value: cli.NewStringSlice(
"user", "user",
"project", "project",
"project:depot", "project:depot",
),
}, },
}, &cli.StringFlag{
cli.StringFlag{ EnvVars: []string{"WOODPECKER_CODING_GIT_MACHINE"},
EnvVar: "WOODPECKER_CODING_GIT_MACHINE",
Name: "coding-git-machine", Name: "coding-git-machine",
Usage: "coding machine name", Usage: "coding machine name",
Value: "git.coding.net", Value: "git.coding.net",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_CODING_GIT_USERNAME", EnvVars: []string{"WOODPECKER_CODING_GIT_USERNAME"},
Name: "coding-git-username", Name: "coding-git-username",
Usage: "coding machine user username", Usage: "coding machine user username",
}, },
cli.StringFlag{ &cli.StringFlag{
EnvVar: "WOODPECKER_CODING_GIT_PASSWORD", EnvVars: []string{"WOODPECKER_CODING_GIT_PASSWORD"},
Name: "coding-git-password", Name: "coding-git-password",
Usage: "coding machine user password", Usage: "coding machine user password",
}, },
cli.BoolFlag{ &cli.BoolFlag{
EnvVar: "WOODPECKER_CODING_SKIP_VERIFY", EnvVars: []string{"WOODPECKER_CODING_SKIP_VERIFY"},
Name: "coding-skip-verify", Name: "coding-skip-verify",
Usage: "coding skip ssl verification", Usage: "coding skip ssl verification",
}, },
cli.DurationFlag{ &cli.DurationFlag{
EnvVar: "WOODPECKER_KEEPALIVE_MIN_TIME", EnvVars: []string{"WOODPECKER_KEEPALIVE_MIN_TIME"},
Name: "keepalive-min-time", Name: "keepalive-min-time",
Usage: "server-side enforcement policy on the minimum amount of time a client should wait before sending a keepalive ping.", Usage: "server-side enforcement policy on the minimum amount of time a client should wait before sending a keepalive ping.",
}, },

View file

@ -20,7 +20,7 @@ import (
"github.com/joho/godotenv" "github.com/joho/godotenv"
_ "github.com/joho/godotenv/autoload" _ "github.com/joho/godotenv/autoload"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/version" "github.com/woodpecker-ci/woodpecker/version"
) )

View file

@ -29,7 +29,7 @@ import (
"github.com/rs/zerolog" "github.com/rs/zerolog"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"golang.org/x/crypto/acme/autocert" "golang.org/x/crypto/acme/autocert"
"golang.org/x/sync/errgroup" "golang.org/x/sync/errgroup"
"google.golang.org/grpc" "google.golang.org/grpc"
@ -55,7 +55,7 @@ func loop(c *cli.Context) error {
log.Logger = log.Output( log.Logger = log.Output(
zerolog.ConsoleWriter{ zerolog.ConsoleWriter{
Out: os.Stderr, Out: os.Stderr,
NoColor: c.BoolT("nocolor"), NoColor: c.Bool("nocolor"),
}, },
) )
} }

View file

@ -22,7 +22,7 @@ import (
"github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promauto" "github.com/prometheus/client_golang/prometheus/promauto"
"github.com/rs/zerolog/log" "github.com/rs/zerolog/log"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"golang.org/x/sync/errgroup" "golang.org/x/sync/errgroup"
"github.com/woodpecker-ci/woodpecker/server" "github.com/woodpecker-ci/woodpecker/server"
@ -180,7 +180,7 @@ func setupGithub(c *cli.Context) (remote.Remote, error) {
Password: c.String("github-git-password"), Password: c.String("github-git-password"),
PrivateMode: c.Bool("github-private-mode"), PrivateMode: c.Bool("github-private-mode"),
SkipVerify: c.Bool("github-skip-verify"), SkipVerify: c.Bool("github-skip-verify"),
MergeRef: c.BoolT("github-merge-ref"), MergeRef: c.Bool("github-merge-ref"),
} }
log.Trace().Msgf("Remote (github) opts: %#v", opts) log.Trace().Msgf("Remote (github) opts: %#v", opts)
return github.New(opts) return github.New(opts)

3
go.mod
View file

@ -9,6 +9,7 @@ require (
github.com/bmatcuk/doublestar/v4 v4.0.2 github.com/bmatcuk/doublestar/v4 v4.0.2
github.com/bradrydzewski/togo v0.0.0-20180401185031-50a0e4726e74 // indirect github.com/bradrydzewski/togo v0.0.0-20180401185031-50a0e4726e74 // indirect
github.com/containerd/containerd v1.5.5 // indirect github.com/containerd/containerd v1.5.5 // indirect
github.com/cpuguy83/go-md2man/v2 v2.0.1 // indirect
github.com/docker/cli v0.0.0-20200303215952-eb310fca4956 github.com/docker/cli v0.0.0-20200303215952-eb310fca4956
github.com/docker/distribution v2.7.1+incompatible github.com/docker/distribution v2.7.1+incompatible
github.com/docker/docker v20.10.8+incompatible github.com/docker/docker v20.10.8+incompatible
@ -46,7 +47,7 @@ require (
github.com/stretchr/testify v1.7.0 github.com/stretchr/testify v1.7.0
github.com/tevino/abool v0.0.0-20170917061928-9b9efcf221b5 github.com/tevino/abool v0.0.0-20170917061928-9b9efcf221b5
github.com/ugorji/go v1.2.6 // indirect github.com/ugorji/go v1.2.6 // indirect
github.com/urfave/cli v1.22.4 github.com/urfave/cli/v2 v2.3.0
github.com/woodpecker-ci/expr v0.0.0-20210628233344-164b8b3d0915 github.com/woodpecker-ci/expr v0.0.0-20210628233344-164b8b3d0915
github.com/woodpecker-ci/togo v0.0.0-20180401185031-50a0e4726e74 github.com/woodpecker-ci/togo v0.0.0-20180401185031-50a0e4726e74
github.com/xanzy/go-gitlab v0.51.1 github.com/xanzy/go-gitlab v0.51.1

13
go.sum
View file

@ -213,8 +213,9 @@ github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSV
github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
github.com/cpuguy83/go-md2man/v2 v2.0.0 h1:EoUDS0afbrsXAZ9YQ9jdu/mZ2sXgT1/2yyNng4PGlyM=
github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
github.com/cpuguy83/go-md2man/v2 v2.0.1 h1:r/myEWzV9lfsM1tFLgDyu0atFtJ1fXn261LKYj/3DxU=
github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY=
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/creack/pty v1.1.11 h1:07n33Z8lZxZ2qwegKbObQohDhXDQxiMMz1NOUGYlesw= github.com/creack/pty v1.1.11 h1:07n33Z8lZxZ2qwegKbObQohDhXDQxiMMz1NOUGYlesw=
@ -619,14 +620,14 @@ github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6po
github.com/rs/xid v1.3.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= github.com/rs/xid v1.3.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
github.com/rs/zerolog v1.25.0 h1:Rj7XygbUHKUlDPcVdoLyR91fJBsduXj5fRxyqIQj/II= github.com/rs/zerolog v1.25.0 h1:Rj7XygbUHKUlDPcVdoLyR91fJBsduXj5fRxyqIQj/II=
github.com/rs/zerolog v1.25.0/go.mod h1:7KHcEGe0QZPOm2IE4Kpb5rTh6n1h2hIgS5OOnu1rUaI= github.com/rs/zerolog v1.25.0/go.mod h1:7KHcEGe0QZPOm2IE4Kpb5rTh6n1h2hIgS5OOnu1rUaI=
github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0RK8m9o+Q=
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk=
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/russross/meddler v1.0.1 h1:JLR7Z4M4iGm1nr7DIURBq18UW8cTrm+qArUFgOhELo8= github.com/russross/meddler v1.0.1 h1:JLR7Z4M4iGm1nr7DIURBq18UW8cTrm+qArUFgOhELo8=
github.com/russross/meddler v1.0.1/go.mod h1:GzGDChbFHuzxlFwt8gnJMRRNyFSQDSudmy2kHh7GYnQ= github.com/russross/meddler v1.0.1/go.mod h1:GzGDChbFHuzxlFwt8gnJMRRNyFSQDSudmy2kHh7GYnQ=
github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4= github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4=
github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0=
github.com/seccomp/libseccomp-golang v0.9.1/go.mod h1:GbW5+tmTXfcxTToHLXlScSlAvWlF4P2Ca7zGrPiEpWo= github.com/seccomp/libseccomp-golang v0.9.1/go.mod h1:GbW5+tmTXfcxTToHLXlScSlAvWlF4P2Ca7zGrPiEpWo=
github.com/shurcooL/sanitized_anchor_name v1.0.0 h1:PdmoCO6wvbs+7yrJyMORt4/BmY5IYyJwS/kOiWx8mHo=
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
github.com/sirupsen/logrus v1.0.4-0.20170822132746-89742aefa4b2/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc= github.com/sirupsen/logrus v1.0.4-0.20170822132746-89742aefa4b2/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc=
github.com/sirupsen/logrus v1.0.6/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc= github.com/sirupsen/logrus v1.0.6/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc=
@ -686,9 +687,10 @@ github.com/ugorji/go/codec v1.2.6/go.mod h1:V6TCNZ4PHqoHGFZuSG1W8nrCzzdgA2DozYxW
github.com/urfave/cli v0.0.0-20171014202726-7bc6a0acffa5/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= github.com/urfave/cli v0.0.0-20171014202726-7bc6a0acffa5/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA=
github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA=
github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0=
github.com/urfave/cli v1.22.2 h1:gsqYFH8bb9ekPA12kRo0hfjngWQjkJPlN9R0N78BoUo=
github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0=
github.com/urfave/cli v1.22.4 h1:u7tSpNPPswAFymm8IehJhy4uJMlUuU/GmqSkvJ1InXA= github.com/urfave/cli/v2 v2.3.0 h1:qph92Y649prgesehzOrQjdWyxFOp/QVM+6imKHad91M=
github.com/urfave/cli v1.22.4/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= github.com/urfave/cli/v2 v2.3.0/go.mod h1:LJmUH05zAU44vOAcrfzZQKsZbVcdbOG8rtL3/XcUArI=
github.com/vishvananda/netlink v0.0.0-20181108222139-023a6dafdcdf/go.mod h1:+SR5DhBJrl6ZM7CoCKvpw5BKroDKQ+PJqOg65H/2ktk= github.com/vishvananda/netlink v0.0.0-20181108222139-023a6dafdcdf/go.mod h1:+SR5DhBJrl6ZM7CoCKvpw5BKroDKQ+PJqOg65H/2ktk=
github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE=
github.com/vishvananda/netlink v1.1.1-0.20201029203352-d40f9887b852/go.mod h1:twkDnbuQxJYemMlGd4JFIcuhgX83tXhKS2B/PRMpOho= github.com/vishvananda/netlink v1.1.1-0.20201029203352-d40f9887b852/go.mod h1:twkDnbuQxJYemMlGd4JFIcuhgX83tXhKS2B/PRMpOho=
@ -1061,6 +1063,7 @@ gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWD
gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74=
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=

View file

@ -16,7 +16,7 @@ package middleware
import ( import (
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/server/model" "github.com/woodpecker-ci/woodpecker/server/model"
) )

View file

@ -16,7 +16,7 @@ package middleware
import ( import (
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/urfave/cli" "github.com/urfave/cli/v2"
"github.com/woodpecker-ci/woodpecker/server/store" "github.com/woodpecker-ci/woodpecker/server/store"
) )

View file

@ -15,7 +15,7 @@ type roffRenderer struct {
extensions blackfriday.Extensions extensions blackfriday.Extensions
listCounters []int listCounters []int
firstHeader bool firstHeader bool
defineTerm bool firstDD bool
listDepth int listDepth int
} }
@ -42,7 +42,8 @@ const (
quoteCloseTag = "\n.RE\n" quoteCloseTag = "\n.RE\n"
listTag = "\n.RS\n" listTag = "\n.RS\n"
listCloseTag = "\n.RE\n" listCloseTag = "\n.RE\n"
arglistTag = "\n.TP\n" dtTag = "\n.TP\n"
dd2Tag = "\n"
tableStart = "\n.TS\nallbox;\n" tableStart = "\n.TS\nallbox;\n"
tableEnd = ".TE\n" tableEnd = ".TE\n"
tableCellStart = "T{\n" tableCellStart = "T{\n"
@ -90,7 +91,7 @@ func (r *roffRenderer) RenderNode(w io.Writer, node *blackfriday.Node, entering
switch node.Type { switch node.Type {
case blackfriday.Text: case blackfriday.Text:
r.handleText(w, node, entering) escapeSpecialChars(w, node.Literal)
case blackfriday.Softbreak: case blackfriday.Softbreak:
out(w, crTag) out(w, crTag)
case blackfriday.Hardbreak: case blackfriday.Hardbreak:
@ -150,40 +151,21 @@ func (r *roffRenderer) RenderNode(w io.Writer, node *blackfriday.Node, entering
out(w, codeCloseTag) out(w, codeCloseTag)
case blackfriday.Table: case blackfriday.Table:
r.handleTable(w, node, entering) r.handleTable(w, node, entering)
case blackfriday.TableCell:
r.handleTableCell(w, node, entering)
case blackfriday.TableHead: case blackfriday.TableHead:
case blackfriday.TableBody: case blackfriday.TableBody:
case blackfriday.TableRow: case blackfriday.TableRow:
// no action as cell entries do all the nroff formatting // no action as cell entries do all the nroff formatting
return blackfriday.GoToNext return blackfriday.GoToNext
case blackfriday.TableCell:
r.handleTableCell(w, node, entering)
case blackfriday.HTMLSpan:
// ignore other HTML tags
default: default:
fmt.Fprintln(os.Stderr, "WARNING: go-md2man does not handle node type "+node.Type.String()) fmt.Fprintln(os.Stderr, "WARNING: go-md2man does not handle node type "+node.Type.String())
} }
return walkAction return walkAction
} }
func (r *roffRenderer) handleText(w io.Writer, node *blackfriday.Node, entering bool) {
var (
start, end string
)
// handle special roff table cell text encapsulation
if node.Parent.Type == blackfriday.TableCell {
if len(node.Literal) > 30 {
start = tableCellStart
end = tableCellEnd
} else {
// end rows that aren't terminated by "tableCellEnd" with a cr if end of row
if node.Parent.Next == nil && !node.Parent.IsHeader {
end = crTag
}
}
}
out(w, start)
escapeSpecialChars(w, node.Literal)
out(w, end)
}
func (r *roffRenderer) handleHeading(w io.Writer, node *blackfriday.Node, entering bool) { func (r *roffRenderer) handleHeading(w io.Writer, node *blackfriday.Node, entering bool) {
if entering { if entering {
switch node.Level { switch node.Level {
@ -230,15 +212,20 @@ func (r *roffRenderer) handleItem(w io.Writer, node *blackfriday.Node, entering
if node.ListFlags&blackfriday.ListTypeOrdered != 0 { if node.ListFlags&blackfriday.ListTypeOrdered != 0 {
out(w, fmt.Sprintf(".IP \"%3d.\" 5\n", r.listCounters[len(r.listCounters)-1])) out(w, fmt.Sprintf(".IP \"%3d.\" 5\n", r.listCounters[len(r.listCounters)-1]))
r.listCounters[len(r.listCounters)-1]++ r.listCounters[len(r.listCounters)-1]++
} else if node.ListFlags&blackfriday.ListTypeTerm != 0 {
// DT (definition term): line just before DD (see below).
out(w, dtTag)
r.firstDD = true
} else if node.ListFlags&blackfriday.ListTypeDefinition != 0 { } else if node.ListFlags&blackfriday.ListTypeDefinition != 0 {
// state machine for handling terms and following definitions // DD (definition description): line that starts with ": ".
// since blackfriday does not distinguish them properly, nor //
// does it seperate them into separate lists as it should // We have to distinguish between the first DD and the
if !r.defineTerm { // subsequent ones, as there should be no vertical
out(w, arglistTag) // whitespace between the DT and the first DD.
r.defineTerm = true if r.firstDD {
r.firstDD = false
} else { } else {
r.defineTerm = false out(w, dd2Tag)
} }
} else { } else {
out(w, ".IP \\(bu 2\n") out(w, ".IP \\(bu 2\n")
@ -261,28 +248,41 @@ func (r *roffRenderer) handleTable(w io.Writer, node *blackfriday.Node, entering
} }
func (r *roffRenderer) handleTableCell(w io.Writer, node *blackfriday.Node, entering bool) { func (r *roffRenderer) handleTableCell(w io.Writer, node *blackfriday.Node, entering bool) {
var (
start, end string
)
if node.IsHeader {
start = codespanTag
end = codespanCloseTag
}
if entering { if entering {
var start string
if node.Prev != nil && node.Prev.Type == blackfriday.TableCell { if node.Prev != nil && node.Prev.Type == blackfriday.TableCell {
out(w, "\t"+start) start = "\t"
} else {
out(w, start)
} }
if node.IsHeader {
start += codespanTag
} else if nodeLiteralSize(node) > 30 {
start += tableCellStart
}
out(w, start)
} else { } else {
// need to carriage return if we are at the end of the header row var end string
if node.IsHeader && node.Next == nil { if node.IsHeader {
end = end + crTag end = codespanCloseTag
} else if nodeLiteralSize(node) > 30 {
end = tableCellEnd
}
if node.Next == nil && end != tableCellEnd {
// Last cell: need to carriage return if we are at the end of the
// header row and content isn't wrapped in a "tablecell"
end += crTag
} }
out(w, end) out(w, end)
} }
} }
func nodeLiteralSize(node *blackfriday.Node) int {
total := 0
for n := node.FirstChild; n != nil; n = n.FirstChild {
total += len(n.Literal)
}
return total
}
// because roff format requires knowing the column count before outputting any table // because roff format requires knowing the column count before outputting any table
// data we need to walk a table tree and count the columns // data we need to walk a table tree and count the columns
func countColumns(node *blackfriday.Node) int { func countColumns(node *blackfriday.Node) int {
@ -309,15 +309,6 @@ func out(w io.Writer, output string) {
io.WriteString(w, output) // nolint: errcheck io.WriteString(w, output) // nolint: errcheck
} }
func needsBackslash(c byte) bool {
for _, r := range []byte("-_&\\~") {
if c == r {
return true
}
}
return false
}
func escapeSpecialChars(w io.Writer, text []byte) { func escapeSpecialChars(w io.Writer, text []byte) {
for i := 0; i < len(text); i++ { for i := 0; i < len(text); i++ {
// escape initial apostrophe or period // escape initial apostrophe or period
@ -328,7 +319,7 @@ func escapeSpecialChars(w io.Writer, text []byte) {
// directly copy normal characters // directly copy normal characters
org := i org := i
for i < len(text) && !needsBackslash(text[i]) { for i < len(text) && text[i] != '\\' {
i++ i++
} }
if i > org { if i > org {

View file

@ -1,4 +1,6 @@
Blackfriday [![Build Status](https://travis-ci.org/russross/blackfriday.svg?branch=master)](https://travis-ci.org/russross/blackfriday) Blackfriday
[![Build Status][BuildV2SVG]][BuildV2URL]
[![PkgGoDev][PkgGoDevV2SVG]][PkgGoDevV2URL]
=========== ===========
Blackfriday is a [Markdown][1] processor implemented in [Go][2]. It Blackfriday is a [Markdown][1] processor implemented in [Go][2]. It
@ -16,19 +18,21 @@ It started as a translation from C of [Sundown][3].
Installation Installation
------------ ------------
Blackfriday is compatible with any modern Go release. With Go 1.7 and git Blackfriday is compatible with modern Go releases in module mode.
installed: With Go installed:
go get gopkg.in/russross/blackfriday.v2 go get github.com/russross/blackfriday/v2
will download, compile, and install the package into your `$GOPATH` will resolve and add the package to the current development module,
directory hierarchy. Alternatively, you can achieve the same if you then build and install it. Alternatively, you can achieve the same
import it into a project: if you import it in a package:
import "gopkg.in/russross/blackfriday.v2" import "github.com/russross/blackfriday/v2"
and `go get` without parameters. and `go get` without parameters.
Legacy GOPATH mode is unsupported.
Versions Versions
-------- --------
@ -36,13 +40,9 @@ Versions
Currently maintained and recommended version of Blackfriday is `v2`. It's being Currently maintained and recommended version of Blackfriday is `v2`. It's being
developed on its own branch: https://github.com/russross/blackfriday/tree/v2 and the developed on its own branch: https://github.com/russross/blackfriday/tree/v2 and the
documentation is available at documentation is available at
https://godoc.org/gopkg.in/russross/blackfriday.v2. https://pkg.go.dev/github.com/russross/blackfriday/v2.
It is `go get`-able via via [gopkg.in][6] at `gopkg.in/russross/blackfriday.v2`, It is `go get`-able in module mode at `github.com/russross/blackfriday/v2`.
but we highly recommend using package management tool like [dep][7] or
[Glide][8] and make use of semantic versioning. With package management you
should import `github.com/russross/blackfriday` and specify that you're using
version 2.0.0.
Version 2 offers a number of improvements over v1: Version 2 offers a number of improvements over v1:
@ -62,6 +62,11 @@ Potential drawbacks:
v2. See issue [#348](https://github.com/russross/blackfriday/issues/348) for v2. See issue [#348](https://github.com/russross/blackfriday/issues/348) for
tracking. tracking.
If you are still interested in the legacy `v1`, you can import it from
`github.com/russross/blackfriday`. Documentation for the legacy v1 can be found
here: https://pkg.go.dev/github.com/russross/blackfriday.
Usage Usage
----- -----
@ -91,7 +96,7 @@ Here's an example of simple usage of Blackfriday together with Bluemonday:
```go ```go
import ( import (
"github.com/microcosm-cc/bluemonday" "github.com/microcosm-cc/bluemonday"
"github.com/russross/blackfriday" "github.com/russross/blackfriday/v2"
) )
// ... // ...
@ -104,6 +109,8 @@ html := bluemonday.UGCPolicy().SanitizeBytes(unsafe)
If you want to customize the set of options, use `blackfriday.WithExtensions`, If you want to customize the set of options, use `blackfriday.WithExtensions`,
`blackfriday.WithRenderer` and `blackfriday.WithRefOverride`. `blackfriday.WithRenderer` and `blackfriday.WithRefOverride`.
### `blackfriday-tool`
You can also check out `blackfriday-tool` for a more complete example You can also check out `blackfriday-tool` for a more complete example
of how to use it. Download and install it using: of how to use it. Download and install it using:
@ -114,7 +121,7 @@ markdown file using a standalone program. You can also browse the
source directly on github if you are just looking for some example source directly on github if you are just looking for some example
code: code:
* <http://github.com/russross/blackfriday-tool> * <https://github.com/russross/blackfriday-tool>
Note that if you have not already done so, installing Note that if you have not already done so, installing
`blackfriday-tool` will be sufficient to download and install `blackfriday-tool` will be sufficient to download and install
@ -123,6 +130,22 @@ installed in `$GOPATH/bin`. This is a statically-linked binary that
can be copied to wherever you need it without worrying about can be copied to wherever you need it without worrying about
dependencies and library versions. dependencies and library versions.
### Sanitized anchor names
Blackfriday includes an algorithm for creating sanitized anchor names
corresponding to a given input text. This algorithm is used to create
anchors for headings when `AutoHeadingIDs` extension is enabled. The
algorithm has a specification, so that other packages can create
compatible anchor names and links to those anchors.
The specification is located at https://pkg.go.dev/github.com/russross/blackfriday/v2#hdr-Sanitized_Anchor_Names.
[`SanitizedAnchorName`](https://pkg.go.dev/github.com/russross/blackfriday/v2#SanitizedAnchorName) exposes this functionality, and can be used to
create compatible links to the anchor names generated by blackfriday.
This algorithm is also implemented in a small standalone package at
[`github.com/shurcooL/sanitized_anchor_name`](https://pkg.go.dev/github.com/shurcooL/sanitized_anchor_name). It can be useful for clients
that want a small package and don't need full functionality of blackfriday.
Features Features
-------- --------
@ -199,6 +222,15 @@ implements the following extensions:
You can use 3 or more backticks to mark the beginning of the You can use 3 or more backticks to mark the beginning of the
block, and the same number to mark the end of the block. block, and the same number to mark the end of the block.
To preserve classes of fenced code blocks while using the bluemonday
HTML sanitizer, use the following policy:
```go
p := bluemonday.UGCPolicy()
p.AllowAttrs("class").Matching(regexp.MustCompile("^language-[a-zA-Z0-9]+$")).OnElements("code")
html := p.SanitizeBytes(unsafe)
```
* **Definition lists**. A simple definition list is made of a single-line * **Definition lists**. A simple definition list is made of a single-line
term followed by a colon and the definition for that term. term followed by a colon and the definition for that term.
@ -250,7 +282,7 @@ Other renderers
Blackfriday is structured to allow alternative rendering engines. Here Blackfriday is structured to allow alternative rendering engines. Here
are a few of note: are a few of note:
* [github_flavored_markdown](https://godoc.org/github.com/shurcooL/github_flavored_markdown): * [github_flavored_markdown](https://pkg.go.dev/github.com/shurcooL/github_flavored_markdown):
provides a GitHub Flavored Markdown renderer with fenced code block provides a GitHub Flavored Markdown renderer with fenced code block
highlighting, clickable heading anchor links. highlighting, clickable heading anchor links.
@ -261,20 +293,28 @@ are a few of note:
* [markdownfmt](https://github.com/shurcooL/markdownfmt): like gofmt, * [markdownfmt](https://github.com/shurcooL/markdownfmt): like gofmt,
but for markdown. but for markdown.
* [LaTeX output](https://github.com/Ambrevar/Blackfriday-LaTeX): * [LaTeX output](https://gitlab.com/ambrevar/blackfriday-latex):
renders output as LaTeX. renders output as LaTeX.
* [bfchroma](https://github.com/Depado/bfchroma/): provides convenience
integration with the [Chroma](https://github.com/alecthomas/chroma) code
highlighting library. bfchroma is only compatible with v2 of Blackfriday and
provides a drop-in renderer ready to use with Blackfriday, as well as
options and means for further customization.
* [Blackfriday-Confluence](https://github.com/kentaro-m/blackfriday-confluence): provides a [Confluence Wiki Markup](https://confluence.atlassian.com/doc/confluence-wiki-markup-251003035.html) renderer. * [Blackfriday-Confluence](https://github.com/kentaro-m/blackfriday-confluence): provides a [Confluence Wiki Markup](https://confluence.atlassian.com/doc/confluence-wiki-markup-251003035.html) renderer.
* [Blackfriday-Slack](https://github.com/karriereat/blackfriday-slack): converts markdown to slack message style
Todo
TODO
---- ----
* More unit testing * More unit testing
* Improve unicode support. It does not understand all unicode * Improve Unicode support. It does not understand all Unicode
rules (about what constitutes a letter, a punctuation symbol, rules (about what constitutes a letter, a punctuation symbol,
etc.), so it may fail to detect word boundaries correctly in etc.), so it may fail to detect word boundaries correctly in
some instances. It is safe on all utf-8 input. some instances. It is safe on all UTF-8 input.
License License
@ -286,6 +326,10 @@ License
[1]: https://daringfireball.net/projects/markdown/ "Markdown" [1]: https://daringfireball.net/projects/markdown/ "Markdown"
[2]: https://golang.org/ "Go Language" [2]: https://golang.org/ "Go Language"
[3]: https://github.com/vmg/sundown "Sundown" [3]: https://github.com/vmg/sundown "Sundown"
[4]: https://godoc.org/gopkg.in/russross/blackfriday.v2#Parse "Parse func" [4]: https://pkg.go.dev/github.com/russross/blackfriday/v2#Parse "Parse func"
[5]: https://github.com/microcosm-cc/bluemonday "Bluemonday" [5]: https://github.com/microcosm-cc/bluemonday "Bluemonday"
[6]: https://labix.org/gopkg.in "gopkg.in"
[BuildV2SVG]: https://travis-ci.org/russross/blackfriday.svg?branch=v2
[BuildV2URL]: https://travis-ci.org/russross/blackfriday
[PkgGoDevV2SVG]: https://pkg.go.dev/badge/github.com/russross/blackfriday/v2
[PkgGoDevV2URL]: https://pkg.go.dev/github.com/russross/blackfriday/v2

View file

@ -18,8 +18,7 @@ import (
"html" "html"
"regexp" "regexp"
"strings" "strings"
"unicode"
"github.com/shurcooL/sanitized_anchor_name"
) )
const ( const (
@ -259,7 +258,7 @@ func (p *Markdown) prefixHeading(data []byte) int {
} }
if end > i { if end > i {
if id == "" && p.extensions&AutoHeadingIDs != 0 { if id == "" && p.extensions&AutoHeadingIDs != 0 {
id = sanitized_anchor_name.Create(string(data[i:end])) id = SanitizedAnchorName(string(data[i:end]))
} }
block := p.addBlock(Heading, data[i:end]) block := p.addBlock(Heading, data[i:end])
block.HeadingID = id block.HeadingID = id
@ -673,6 +672,7 @@ func (p *Markdown) fencedCodeBlock(data []byte, doRender bool) int {
if beg == 0 || beg >= len(data) { if beg == 0 || beg >= len(data) {
return 0 return 0
} }
fenceLength := beg - 1
var work bytes.Buffer var work bytes.Buffer
work.Write([]byte(info)) work.Write([]byte(info))
@ -706,6 +706,7 @@ func (p *Markdown) fencedCodeBlock(data []byte, doRender bool) int {
if doRender { if doRender {
block := p.addBlock(CodeBlock, work.Bytes()) // TODO: get rid of temp buffer block := p.addBlock(CodeBlock, work.Bytes()) // TODO: get rid of temp buffer
block.IsFenced = true block.IsFenced = true
block.FenceLength = fenceLength
finalizeCodeBlock(block) finalizeCodeBlock(block)
} }
@ -1503,7 +1504,7 @@ func (p *Markdown) paragraph(data []byte) int {
id := "" id := ""
if p.extensions&AutoHeadingIDs != 0 { if p.extensions&AutoHeadingIDs != 0 {
id = sanitized_anchor_name.Create(string(data[prev:eol])) id = SanitizedAnchorName(string(data[prev:eol]))
} }
block := p.addBlock(Heading, data[prev:eol]) block := p.addBlock(Heading, data[prev:eol])
@ -1588,3 +1589,24 @@ func skipUntilChar(text []byte, start int, char byte) int {
} }
return i return i
} }
// SanitizedAnchorName returns a sanitized anchor name for the given text.
//
// It implements the algorithm specified in the package comment.
func SanitizedAnchorName(text string) string {
var anchorName []rune
futureDash := false
for _, r := range text {
switch {
case unicode.IsLetter(r) || unicode.IsNumber(r):
if futureDash && len(anchorName) > 0 {
anchorName = append(anchorName, '-')
}
futureDash = false
anchorName = append(anchorName, unicode.ToLower(r))
default:
futureDash = true
}
}
return string(anchorName)
}

View file

@ -15,4 +15,32 @@
// //
// If you're interested in calling Blackfriday from command line, see // If you're interested in calling Blackfriday from command line, see
// https://github.com/russross/blackfriday-tool. // https://github.com/russross/blackfriday-tool.
//
// Sanitized Anchor Names
//
// Blackfriday includes an algorithm for creating sanitized anchor names
// corresponding to a given input text. This algorithm is used to create
// anchors for headings when AutoHeadingIDs extension is enabled. The
// algorithm is specified below, so that other packages can create
// compatible anchor names and links to those anchors.
//
// The algorithm iterates over the input text, interpreted as UTF-8,
// one Unicode code point (rune) at a time. All runes that are letters (category L)
// or numbers (category N) are considered valid characters. They are mapped to
// lower case, and included in the output. All other runes are considered
// invalid characters. Invalid characters that precede the first valid character,
// as well as invalid character that follow the last valid character
// are dropped completely. All other sequences of invalid characters
// between two valid characters are replaced with a single dash character '-'.
//
// SanitizedAnchorName exposes this functionality, and can be used to
// create compatible links to the anchor names generated by blackfriday.
// This algorithm is also implemented in a small standalone package at
// github.com/shurcooL/sanitized_anchor_name. It can be useful for clients
// that want a small package and don't need full functionality of blackfriday.
package blackfriday package blackfriday
// NOTE: Keep Sanitized Anchor Name algorithm in sync with package
// github.com/shurcooL/sanitized_anchor_name.
// Otherwise, users of sanitized_anchor_name will get anchor names
// that are incompatible with those generated by blackfriday.

2236
vendor/github.com/russross/blackfriday/v2/entities.go generated vendored Normal file

File diff suppressed because it is too large Load diff

View file

@ -13,14 +13,28 @@ var htmlEscaper = [256][]byte{
} }
func escapeHTML(w io.Writer, s []byte) { func escapeHTML(w io.Writer, s []byte) {
escapeEntities(w, s, false)
}
func escapeAllHTML(w io.Writer, s []byte) {
escapeEntities(w, s, true)
}
func escapeEntities(w io.Writer, s []byte, escapeValidEntities bool) {
var start, end int var start, end int
for end < len(s) { for end < len(s) {
escSeq := htmlEscaper[s[end]] escSeq := htmlEscaper[s[end]]
if escSeq != nil { if escSeq != nil {
isEntity, entityEnd := nodeIsEntity(s, end)
if isEntity && !escapeValidEntities {
w.Write(s[start : entityEnd+1])
start = entityEnd + 1
} else {
w.Write(s[start:end]) w.Write(s[start:end])
w.Write(escSeq) w.Write(escSeq)
start = end + 1 start = end + 1
} }
}
end++ end++
} }
if start < len(s) && end <= len(s) { if start < len(s) && end <= len(s) {
@ -28,6 +42,28 @@ func escapeHTML(w io.Writer, s []byte) {
} }
} }
func nodeIsEntity(s []byte, end int) (isEntity bool, endEntityPos int) {
isEntity = false
endEntityPos = end + 1
if s[end] == '&' {
for endEntityPos < len(s) {
if s[endEntityPos] == ';' {
if entities[string(s[end:endEntityPos+1])] {
isEntity = true
break
}
}
if !isalnum(s[endEntityPos]) && s[endEntityPos] != '&' && s[endEntityPos] != '#' {
break
}
endEntityPos++
}
}
return isEntity, endEntityPos
}
func escLink(w io.Writer, text []byte) { func escLink(w io.Writer, text []byte) {
unesc := html.UnescapeString(string(text)) unesc := html.UnescapeString(string(text))
escapeHTML(w, []byte(unesc)) escapeHTML(w, []byte(unesc))

View file

@ -132,7 +132,10 @@ func NewHTMLRenderer(params HTMLRendererParameters) *HTMLRenderer {
} }
if params.FootnoteReturnLinkContents == "" { if params.FootnoteReturnLinkContents == "" {
params.FootnoteReturnLinkContents = `<sup>[return]</sup>` // U+FE0E is VARIATION SELECTOR-15.
// It suppresses automatic emoji presentation of the preceding
// U+21A9 LEFTWARDS ARROW WITH HOOK on iOS and iPadOS.
params.FootnoteReturnLinkContents = "<span aria-label='Return'>↩\ufe0e</span>"
} }
return &HTMLRenderer{ return &HTMLRenderer{
@ -616,7 +619,7 @@ func (r *HTMLRenderer) RenderNode(w io.Writer, node *Node, entering bool) WalkSt
} }
case Code: case Code:
r.out(w, codeTag) r.out(w, codeTag)
escapeHTML(w, node.Literal) escapeAllHTML(w, node.Literal)
r.out(w, codeCloseTag) r.out(w, codeCloseTag)
case Document: case Document:
break break
@ -762,7 +765,7 @@ func (r *HTMLRenderer) RenderNode(w io.Writer, node *Node, entering bool) WalkSt
r.cr(w) r.cr(w)
r.out(w, preTag) r.out(w, preTag)
r.tag(w, codeTag[:len(codeTag)-1], attrs) r.tag(w, codeTag[:len(codeTag)-1], attrs)
escapeHTML(w, node.Literal) escapeAllHTML(w, node.Literal)
r.out(w, codeCloseTag) r.out(w, codeCloseTag)
r.out(w, preCloseTag) r.out(w, preCloseTag)
if node.Parent.Type != Item { if node.Parent.Type != Item {

View file

@ -278,7 +278,7 @@ func link(p *Markdown, data []byte, offset int) (int, *Node) {
case data[i] == '\n': case data[i] == '\n':
textHasNl = true textHasNl = true
case data[i-1] == '\\': case isBackslashEscaped(data, i):
continue continue
case data[i] == '[': case data[i] == '[':

View file

@ -199,7 +199,8 @@ func (n *Node) InsertBefore(sibling *Node) {
} }
} }
func (n *Node) isContainer() bool { // IsContainer returns true if 'n' can contain children.
func (n *Node) IsContainer() bool {
switch n.Type { switch n.Type {
case Document: case Document:
fallthrough fallthrough
@ -238,6 +239,11 @@ func (n *Node) isContainer() bool {
} }
} }
// IsLeaf returns true if 'n' is a leaf node.
func (n *Node) IsLeaf() bool {
return !n.IsContainer()
}
func (n *Node) canContain(t NodeType) bool { func (n *Node) canContain(t NodeType) bool {
if n.Type == List { if n.Type == List {
return t == Item return t == Item
@ -309,11 +315,11 @@ func newNodeWalker(root *Node) *nodeWalker {
} }
func (nw *nodeWalker) next() { func (nw *nodeWalker) next() {
if (!nw.current.isContainer() || !nw.entering) && nw.current == nw.root { if (!nw.current.IsContainer() || !nw.entering) && nw.current == nw.root {
nw.current = nil nw.current = nil
return return
} }
if nw.entering && nw.current.isContainer() { if nw.entering && nw.current.IsContainer() {
if nw.current.FirstChild != nil { if nw.current.FirstChild != nil {
nw.current = nw.current.FirstChild nw.current = nw.current.FirstChild
nw.entering = true nw.entering = true

View file

@ -1,16 +0,0 @@
sudo: false
language: go
go:
- 1.x
- master
matrix:
allow_failures:
- go: master
fast_finish: true
install:
- # Do nothing. This is needed to prevent default install action "go get -t -v ./..." from happening here (we want it to happen inside script step).
script:
- go get -t -v ./...
- diff -u <(echo -n) <(gofmt -d -s .)
- go tool vet .
- go test -v -race ./...

View file

@ -1,36 +0,0 @@
sanitized_anchor_name
=====================
[![Build Status](https://travis-ci.org/shurcooL/sanitized_anchor_name.svg?branch=master)](https://travis-ci.org/shurcooL/sanitized_anchor_name) [![GoDoc](https://godoc.org/github.com/shurcooL/sanitized_anchor_name?status.svg)](https://godoc.org/github.com/shurcooL/sanitized_anchor_name)
Package sanitized_anchor_name provides a func to create sanitized anchor names.
Its logic can be reused by multiple packages to create interoperable anchor names
and links to those anchors.
At this time, it does not try to ensure that generated anchor names
are unique, that responsibility falls on the caller.
Installation
------------
```bash
go get -u github.com/shurcooL/sanitized_anchor_name
```
Example
-------
```Go
anchorName := sanitized_anchor_name.Create("This is a header")
fmt.Println(anchorName)
// Output:
// this-is-a-header
```
License
-------
- [MIT License](LICENSE)

View file

@ -1 +0,0 @@
module github.com/shurcooL/sanitized_anchor_name

View file

@ -1,29 +0,0 @@
// Package sanitized_anchor_name provides a func to create sanitized anchor names.
//
// Its logic can be reused by multiple packages to create interoperable anchor names
// and links to those anchors.
//
// At this time, it does not try to ensure that generated anchor names
// are unique, that responsibility falls on the caller.
package sanitized_anchor_name // import "github.com/shurcooL/sanitized_anchor_name"
import "unicode"
// Create returns a sanitized anchor name for the given text.
func Create(text string) string {
var anchorName []rune
var futureDash = false
for _, r := range text {
switch {
case unicode.IsLetter(r) || unicode.IsNumber(r):
if futureDash && len(anchorName) > 0 {
anchorName = append(anchorName, '-')
}
futureDash = false
anchorName = append(anchorName, unicode.ToLower(r))
default:
futureDash = true
}
}
return string(anchorName)
}

View file

@ -1,4 +1,3 @@
*.coverprofile *.coverprofile
node_modules/ node_modules/
vendor vendor
.idea

35
vendor/github.com/urfave/cli/.travis.yml generated vendored Normal file
View file

@ -0,0 +1,35 @@
language: go
sudo: false
dist: bionic
osx_image: xcode10
go:
- 1.11.x
- 1.12.x
- 1.13.x
os:
- linux
- osx
env:
GO111MODULE=on
GOPROXY=https://proxy.golang.org
cache:
directories:
- node_modules
before_script:
- go get github.com/urfave/gfmrun/cmd/gfmrun
- go get golang.org/x/tools/cmd/goimports
- npm install markdown-toc
- go mod tidy
script:
- go run build.go vet
- go run build.go test
- go run build.go gfmrun docs/v1/manual.md
- go run build.go toc docs/v1/manual.md
after_success:
- bash <(curl -s https://codecov.io/bash)

View file

@ -121,6 +121,7 @@ func NewApp() *App {
HelpName: filepath.Base(os.Args[0]), HelpName: filepath.Base(os.Args[0]),
Usage: "A new cli application", Usage: "A new cli application",
UsageText: "", UsageText: "",
Version: "0.0.0",
BashComplete: DefaultAppComplete, BashComplete: DefaultAppComplete,
Action: helpCommand.Action, Action: helpCommand.Action,
Compiled: compileTime(), Compiled: compileTime(),
@ -158,10 +159,6 @@ func (a *App) Setup() {
} }
} }
if a.Version == "" {
a.HideVersion = true
}
if !a.HideVersion { if !a.HideVersion {
a.appendFlag(VersionFlag) a.appendFlag(VersionFlag)
} }
@ -207,7 +204,7 @@ func (a *App) Run(arguments []string) (err error) {
return err return err
} }
err = parseIter(set, a, arguments[1:], shellComplete) err = parseIter(set, a, arguments[1:])
nerr := normalizeFlags(a.Flags, set) nerr := normalizeFlags(a.Flags, set)
context := NewContext(a, set, nil) context := NewContext(a, set, nil)
if nerr != nil { if nerr != nil {
@ -330,7 +327,7 @@ func (a *App) RunAsSubcommand(ctx *Context) (err error) {
return err return err
} }
err = parseIter(set, a, ctx.Args().Tail(), ctx.shellComplete) err = parseIter(set, a, ctx.Args().Tail())
nerr := normalizeFlags(a.Flags, set) nerr := normalizeFlags(a.Flags, set)
context := NewContext(a, set, ctx) context := NewContext(a, set, ctx)

View file

@ -114,7 +114,7 @@ func (c Command) Run(ctx *Context) (err error) {
c.UseShortOptionHandling = true c.UseShortOptionHandling = true
} }
set, err := c.parseFlags(ctx.Args().Tail(), ctx.shellComplete) set, err := c.parseFlags(ctx.Args().Tail())
context := NewContext(ctx.App, set, ctx) context := NewContext(ctx.App, set, ctx)
context.Command = c context.Command = c
@ -179,7 +179,7 @@ func (c Command) Run(ctx *Context) (err error) {
return err return err
} }
func (c *Command) parseFlags(args Args, shellComplete bool) (*flag.FlagSet, error) { func (c *Command) parseFlags(args Args) (*flag.FlagSet, error) {
if c.SkipFlagParsing { if c.SkipFlagParsing {
set, err := c.newFlagSet() set, err := c.newFlagSet()
if err != nil { if err != nil {
@ -198,7 +198,7 @@ func (c *Command) parseFlags(args Args, shellComplete bool) (*flag.FlagSet, erro
return nil, err return nil, err
} }
err = parseIter(set, c, args, shellComplete) err = parseIter(set, c, args)
if err != nil { if err != nil {
return nil, err return nil, err
} }

View file

@ -87,14 +87,6 @@ func (c *Context) IsSet(name string) bool {
for _, f := range flags { for _, f := range flags {
eachName(f.GetName(), func(name string) { eachName(f.GetName(), func(name string) {
if isSet, ok := c.setFlags[name]; isSet || !ok { if isSet, ok := c.setFlags[name]; isSet || !ok {
// Check if a flag is set
if isSet {
// If the flag is set, also set its other aliases
eachName(f.GetName(), func(name string) {
c.setFlags[name] = true
})
}
return return
} }

View file

@ -22,12 +22,7 @@ func (f *Int64Slice) Set(value string) error {
// String returns a readable representation of this value (for usage defaults) // String returns a readable representation of this value (for usage defaults)
func (f *Int64Slice) String() string { func (f *Int64Slice) String() string {
slice := make([]string, len(*f)) return fmt.Sprintf("%#v", *f)
for i, v := range *f {
slice[i] = strconv.FormatInt(v, 10)
}
return strings.Join(slice, ",")
} }
// Value returns the slice of ints set by this flag // Value returns the slice of ints set by this flag
@ -115,7 +110,6 @@ func (f Int64SliceFlag) ApplyWithError(set *flag.FlagSet) error {
} }
set.Var(f.Value, name, f.Usage) set.Var(f.Value, name, f.Usage)
}) })
return nil return nil
} }
@ -137,61 +131,11 @@ func (c *Context) GlobalInt64Slice(name string) []int64 {
func lookupInt64Slice(name string, set *flag.FlagSet) []int64 { func lookupInt64Slice(name string, set *flag.FlagSet) []int64 {
f := set.Lookup(name) f := set.Lookup(name)
if f != nil { if f != nil {
value, ok := f.Value.(*Int64Slice) parsed, err := (f.Value.(*Int64Slice)).Value(), error(nil)
if !ok {
return nil
}
// extract the slice from asserted value
parsed := value.Value()
// extract default value from the flag
var defaultVal []int64
for _, v := range strings.Split(f.DefValue, ",") {
if v != "" {
int64Value, err := strconv.ParseInt(v, 10, 64)
if err != nil { if err != nil {
panic(err) return nil
}
defaultVal = append(defaultVal, int64Value)
}
}
// if the current value is not equal to the default value
// remove the default values from the flag
if !isInt64SliceEqual(parsed, defaultVal) {
for _, v := range defaultVal {
parsed = removeFromInt64Slice(parsed, v)
}
} }
return parsed return parsed
} }
return nil return nil
} }
func removeFromInt64Slice(slice []int64, val int64) []int64 {
for i, v := range slice {
if v == val {
return append(slice[:i], slice[i+1:]...)
}
}
return slice
}
func isInt64SliceEqual(newValue, defaultValue []int64) bool {
// If one is nil, the other must also be nil.
if (newValue == nil) != (defaultValue == nil) {
return false
}
if len(newValue) != len(defaultValue) {
return false
}
for i, v := range newValue {
if v != defaultValue[i] {
return false
}
}
return true
}

View file

@ -22,12 +22,7 @@ func (f *IntSlice) Set(value string) error {
// String returns a readable representation of this value (for usage defaults) // String returns a readable representation of this value (for usage defaults)
func (f *IntSlice) String() string { func (f *IntSlice) String() string {
slice := make([]string, len(*f)) return fmt.Sprintf("%#v", *f)
for i, v := range *f {
slice[i] = strconv.Itoa(v)
}
return strings.Join(slice, ",")
} }
// Value returns the slice of ints set by this flag // Value returns the slice of ints set by this flag
@ -137,60 +132,11 @@ func (c *Context) GlobalIntSlice(name string) []int {
func lookupIntSlice(name string, set *flag.FlagSet) []int { func lookupIntSlice(name string, set *flag.FlagSet) []int {
f := set.Lookup(name) f := set.Lookup(name)
if f != nil { if f != nil {
value, ok := f.Value.(*IntSlice) parsed, err := (f.Value.(*IntSlice)).Value(), error(nil)
if !ok {
return nil
}
// extract the slice from asserted value
slice := value.Value()
// extract default value from the flag
var defaultVal []int
for _, v := range strings.Split(f.DefValue, ",") {
if v != "" {
intValue, err := strconv.Atoi(v)
if err != nil { if err != nil {
panic(err) return nil
} }
defaultVal = append(defaultVal, intValue) return parsed
}
}
// if the current value is not equal to the default value
// remove the default values from the flag
if !isIntSliceEqual(slice, defaultVal) {
for _, v := range defaultVal {
slice = removeFromIntSlice(slice, v)
}
}
return slice
} }
return nil return nil
} }
func removeFromIntSlice(slice []int, val int) []int {
for i, v := range slice {
if v == val {
return append(slice[:i], slice[i+1:]...)
}
}
return slice
}
func isIntSliceEqual(newValue, defaultValue []int) bool {
// If one is nil, the other must also be nil.
if (newValue == nil) != (defaultValue == nil) {
return false
}
if len(newValue) != len(defaultValue) {
return false
}
for i, v := range newValue {
if v != defaultValue[i] {
return false
}
}
return true
}

View file

@ -17,7 +17,7 @@ func (f *StringSlice) Set(value string) error {
// String returns a readable representation of this value (for usage defaults) // String returns a readable representation of this value (for usage defaults)
func (f *StringSlice) String() string { func (f *StringSlice) String() string {
return strings.Join(*f, ",") return fmt.Sprintf("%s", *f)
} }
// Value returns the slice of strings set by this flag // Value returns the slice of strings set by this flag
@ -128,55 +128,11 @@ func (c *Context) GlobalStringSlice(name string) []string {
func lookupStringSlice(name string, set *flag.FlagSet) []string { func lookupStringSlice(name string, set *flag.FlagSet) []string {
f := set.Lookup(name) f := set.Lookup(name)
if f != nil { if f != nil {
value, ok := f.Value.(*StringSlice) parsed, err := (f.Value.(*StringSlice)).Value(), error(nil)
if !ok { if err != nil {
return nil return nil
} }
// extract the slice from asserted value return parsed
slice := value.Value()
// extract default value from the flag
var defaultVal []string
for _, v := range strings.Split(f.DefValue, ",") {
defaultVal = append(defaultVal, v)
}
// if the current value is not equal to the default value
// remove the default values from the flag
if !isStringSliceEqual(slice, defaultVal) {
for _, v := range defaultVal {
slice = removeFromStringSlice(slice, v)
}
}
return slice
} }
return nil return nil
} }
func removeFromStringSlice(slice []string, val string) []string {
for i, v := range slice {
if v == val {
return append(slice[:i], slice[i+1:]...)
}
}
return slice
}
func isStringSliceEqual(newValue, defaultValue []string) bool {
// If one is nil, the other must also be nil.
if (newValue == nil) != (defaultValue == nil) {
return false
}
if len(newValue) != len(defaultValue) {
return false
}
for i, v := range newValue {
if v != defaultValue[i] {
return false
}
}
return true
}

View file

@ -14,15 +14,10 @@ type iterativeParser interface {
// iteratively catch parsing errors. This way we achieve LR parsing without // iteratively catch parsing errors. This way we achieve LR parsing without
// transforming any arguments. Otherwise, there is no way we can discriminate // transforming any arguments. Otherwise, there is no way we can discriminate
// combined short options from common arguments that should be left untouched. // combined short options from common arguments that should be left untouched.
// Pass `shellComplete` to continue parsing options on failure during shell func parseIter(set *flag.FlagSet, ip iterativeParser, args []string) error {
// completion when, the user-supplied options may be incomplete.
func parseIter(set *flag.FlagSet, ip iterativeParser, args []string, shellComplete bool) error {
for { for {
err := set.Parse(args) err := set.Parse(args)
if !ip.useShortOptionHandling() || err == nil { if !ip.useShortOptionHandling() || err == nil {
if shellComplete {
return nil
}
return err return err
} }

2
vendor/github.com/urfave/cli/v2/.flake8 generated vendored Normal file
View file

@ -0,0 +1,2 @@
[flake8]
max-line-length = 120

7
vendor/github.com/urfave/cli/v2/.gitignore generated vendored Normal file
View file

@ -0,0 +1,7 @@
*.coverprofile
*.orig
node_modules/
vendor
.idea
internal/*/built-example
coverage.txt

74
vendor/github.com/urfave/cli/v2/CODE_OF_CONDUCT.md generated vendored Normal file
View file

@ -0,0 +1,74 @@
# Contributor Covenant Code of Conduct
## Our Pledge
In the interest of fostering an open and welcoming environment, we as
contributors and maintainers pledge to making participation in our project and
our community a harassment-free experience for everyone, regardless of age, body
size, disability, ethnicity, gender identity and expression, level of experience,
education, socio-economic status, nationality, personal appearance, race,
religion, or sexual identity and orientation.
## Our Standards
Examples of behavior that contributes to creating a positive environment
include:
* Using welcoming and inclusive language
* Being respectful of differing viewpoints and experiences
* Gracefully accepting constructive criticism
* Focusing on what is best for the community
* Showing empathy towards other community members
Examples of unacceptable behavior by participants include:
* The use of sexualized language or imagery and unwelcome sexual attention or
advances
* Trolling, insulting/derogatory comments, and personal or political attacks
* Public or private harassment
* Publishing others' private information, such as a physical or electronic
address, without explicit permission
* Other conduct which could reasonably be considered inappropriate in a
professional setting
## Our Responsibilities
Project maintainers are responsible for clarifying the standards of acceptable
behavior and are expected to take appropriate and fair corrective action in
response to any instances of unacceptable behavior.
Project maintainers have the right and responsibility to remove, edit, or
reject comments, commits, code, wiki edits, issues, and other contributions
that are not aligned to this Code of Conduct, or to ban temporarily or
permanently any contributor for other behaviors that they deem inappropriate,
threatening, offensive, or harmful.
## Scope
This Code of Conduct applies both within project spaces and in public spaces
when an individual is representing the project or its community. Examples of
representing a project or community include using an official project e-mail
address, posting via an official social media account, or acting as an appointed
representative at an online or offline event. Representation of a project may be
further defined and clarified by project maintainers.
## Enforcement
Instances of abusive, harassing, or otherwise unacceptable behavior may be
reported by contacting Dan Buch at dan@meatballhat.com. All complaints will be
reviewed and investigated and will result in a response that is deemed necessary
and appropriate to the circumstances. The project team is obligated to maintain
confidentiality with regard to the reporter of an incident. Further details of
specific enforcement policies may be posted separately.
Project maintainers who do not follow or enforce the Code of Conduct in good
faith may face temporary or permanent repercussions as determined by other
members of the project's leadership.
## Attribution
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html
[homepage]: https://www.contributor-covenant.org

View file

@ -1,6 +1,6 @@
MIT License MIT License
Copyright (c) 2015 Dmitri Shuralyov Copyright (c) 2016 Jeremy Saenz & Contributors
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal

70
vendor/github.com/urfave/cli/v2/README.md generated vendored Normal file
View file

@ -0,0 +1,70 @@
cli
===
[![GoDoc](https://godoc.org/github.com/urfave/cli?status.svg)](https://godoc.org/github.com/urfave/cli)
[![codebeat](https://codebeat.co/badges/0a8f30aa-f975-404b-b878-5fab3ae1cc5f)](https://codebeat.co/projects/github-com-urfave-cli)
[![Go Report Card](https://goreportcard.com/badge/urfave/cli)](https://goreportcard.com/report/urfave/cli)
[![codecov](https://codecov.io/gh/urfave/cli/branch/master/graph/badge.svg)](https://codecov.io/gh/urfave/cli)
cli is a simple, fast, and fun package for building command line apps in Go. The
goal is to enable developers to write fast and distributable command line
applications in an expressive way.
## Usage Documentation
Usage documentation exists for each major version. Don't know what version you're on? You're probably using the version from the `master` branch, which is currently `v2`.
- `v2` - [./docs/v2/manual.md](./docs/v2/manual.md)
- `v1` - [./docs/v1/manual.md](./docs/v1/manual.md)
Guides for migrating to newer versions:
- `v1-to-v2` - [./docs/migrate-v1-to-v2.md](./docs/migrate-v1-to-v2.md)
## Installation
Using this package requires a working Go environment. [See the install instructions for Go](http://golang.org/doc/install.html).
Go Modules are required when using this package. [See the go blog guide on using Go Modules](https://blog.golang.org/using-go-modules).
### Using `v2` releases
```
$ GO111MODULE=on go get github.com/urfave/cli/v2
```
```go
...
import (
"github.com/urfave/cli/v2" // imports as package "cli"
)
...
```
### Using `v1` releases
```
$ GO111MODULE=on go get github.com/urfave/cli
```
```go
...
import (
"github.com/urfave/cli"
)
...
```
### GOPATH
Make sure your `PATH` includes the `$GOPATH/bin` directory so your commands can
be easily used:
```
export PATH=$PATH:$GOPATH/bin
```
### Supported platforms
cli is tested against multiple versions of Go on Linux, and against the latest
released version of Go on OS X and Windows. This project uses Github Actions for
builds. To see our currently supported go versions and platforms, look at the [./.github/workflows/cli.yml](https://github.com/urfave/cli/blob/master/.github/workflows/cli.yml).

540
vendor/github.com/urfave/cli/v2/app.go generated vendored Normal file
View file

@ -0,0 +1,540 @@
package cli
import (
"context"
"flag"
"fmt"
"io"
"os"
"path/filepath"
"sort"
"time"
)
var (
changeLogURL = "https://github.com/urfave/cli/blob/master/docs/CHANGELOG.md"
appActionDeprecationURL = fmt.Sprintf("%s#deprecated-cli-app-action-signature", changeLogURL)
contactSysadmin = "This is an error in the application. Please contact the distributor of this application if this is not you."
errInvalidActionType = NewExitError("ERROR invalid Action type. "+
fmt.Sprintf("Must be `func(*Context`)` or `func(*Context) error). %s", contactSysadmin)+
fmt.Sprintf("See %s", appActionDeprecationURL), 2)
)
// App is the main structure of a cli application. It is recommended that
// an app be created with the cli.NewApp() function
type App struct {
// The name of the program. Defaults to path.Base(os.Args[0])
Name string
// Full name of command for help, defaults to Name
HelpName string
// Description of the program.
Usage string
// Text to override the USAGE section of help
UsageText string
// Description of the program argument format.
ArgsUsage string
// Version of the program
Version string
// Description of the program
Description string
// List of commands to execute
Commands []*Command
// List of flags to parse
Flags []Flag
// Boolean to enable bash completion commands
EnableBashCompletion bool
// Boolean to hide built-in help command and help flag
HideHelp bool
// Boolean to hide built-in help command but keep help flag.
// Ignored if HideHelp is true.
HideHelpCommand bool
// Boolean to hide built-in version flag and the VERSION section of help
HideVersion bool
// categories contains the categorized commands and is populated on app startup
categories CommandCategories
// An action to execute when the shell completion flag is set
BashComplete BashCompleteFunc
// An action to execute before any subcommands are run, but after the context is ready
// If a non-nil error is returned, no subcommands are run
Before BeforeFunc
// An action to execute after any subcommands are run, but after the subcommand has finished
// It is run even if Action() panics
After AfterFunc
// The action to execute when no subcommands are specified
Action ActionFunc
// Execute this function if the proper command cannot be found
CommandNotFound CommandNotFoundFunc
// Execute this function if a usage error occurs
OnUsageError OnUsageErrorFunc
// Compilation date
Compiled time.Time
// List of all authors who contributed
Authors []*Author
// Copyright of the binary if any
Copyright string
// Reader reader to write input to (useful for tests)
Reader io.Reader
// Writer writer to write output to
Writer io.Writer
// ErrWriter writes error output
ErrWriter io.Writer
// ExitErrHandler processes any error encountered while running an App before
// it is returned to the caller. If no function is provided, HandleExitCoder
// is used as the default behavior.
ExitErrHandler ExitErrHandlerFunc
// Other custom info
Metadata map[string]interface{}
// Carries a function which returns app specific info.
ExtraInfo func() map[string]string
// CustomAppHelpTemplate the text template for app help topic.
// cli.go uses text/template to render templates. You can
// render custom help text by setting this variable.
CustomAppHelpTemplate string
// Boolean to enable short-option handling so user can combine several
// single-character bool arguments into one
// i.e. foobar -o -v -> foobar -ov
UseShortOptionHandling bool
didSetup bool
}
// Tries to find out when this binary was compiled.
// Returns the current time if it fails to find it.
func compileTime() time.Time {
info, err := os.Stat(os.Args[0])
if err != nil {
return time.Now()
}
return info.ModTime()
}
// NewApp creates a new cli Application with some reasonable defaults for Name,
// Usage, Version and Action.
func NewApp() *App {
return &App{
Name: filepath.Base(os.Args[0]),
HelpName: filepath.Base(os.Args[0]),
Usage: "A new cli application",
UsageText: "",
BashComplete: DefaultAppComplete,
Action: helpCommand.Action,
Compiled: compileTime(),
Reader: os.Stdin,
Writer: os.Stdout,
ErrWriter: os.Stderr,
}
}
// Setup runs initialization code to ensure all data structures are ready for
// `Run` or inspection prior to `Run`. It is internally called by `Run`, but
// will return early if setup has already happened.
func (a *App) Setup() {
if a.didSetup {
return
}
a.didSetup = true
if a.Name == "" {
a.Name = filepath.Base(os.Args[0])
}
if a.HelpName == "" {
a.HelpName = filepath.Base(os.Args[0])
}
if a.Usage == "" {
a.Usage = "A new cli application"
}
if a.Version == "" {
a.HideVersion = true
}
if a.BashComplete == nil {
a.BashComplete = DefaultAppComplete
}
if a.Action == nil {
a.Action = helpCommand.Action
}
if a.Compiled == (time.Time{}) {
a.Compiled = compileTime()
}
if a.Reader == nil {
a.Reader = os.Stdin
}
if a.Writer == nil {
a.Writer = os.Stdout
}
if a.ErrWriter == nil {
a.ErrWriter = os.Stderr
}
var newCommands []*Command
for _, c := range a.Commands {
if c.HelpName == "" {
c.HelpName = fmt.Sprintf("%s %s", a.HelpName, c.Name)
}
newCommands = append(newCommands, c)
}
a.Commands = newCommands
if a.Command(helpCommand.Name) == nil && !a.HideHelp {
if !a.HideHelpCommand {
a.appendCommand(helpCommand)
}
if HelpFlag != nil {
a.appendFlag(HelpFlag)
}
}
if !a.HideVersion {
a.appendFlag(VersionFlag)
}
a.categories = newCommandCategories()
for _, command := range a.Commands {
a.categories.AddCommand(command.Category, command)
}
sort.Sort(a.categories.(*commandCategories))
if a.Metadata == nil {
a.Metadata = make(map[string]interface{})
}
}
func (a *App) newFlagSet() (*flag.FlagSet, error) {
return flagSet(a.Name, a.Flags)
}
func (a *App) useShortOptionHandling() bool {
return a.UseShortOptionHandling
}
// Run is the entry point to the cli app. Parses the arguments slice and routes
// to the proper flag/args combination
func (a *App) Run(arguments []string) (err error) {
return a.RunContext(context.Background(), arguments)
}
// RunContext is like Run except it takes a Context that will be
// passed to its commands and sub-commands. Through this, you can
// propagate timeouts and cancellation requests
func (a *App) RunContext(ctx context.Context, arguments []string) (err error) {
a.Setup()
// handle the completion flag separately from the flagset since
// completion could be attempted after a flag, but before its value was put
// on the command line. this causes the flagset to interpret the completion
// flag name as the value of the flag before it which is undesirable
// note that we can only do this because the shell autocomplete function
// always appends the completion flag at the end of the command
shellComplete, arguments := checkShellCompleteFlag(a, arguments)
set, err := a.newFlagSet()
if err != nil {
return err
}
err = parseIter(set, a, arguments[1:], shellComplete)
nerr := normalizeFlags(a.Flags, set)
context := NewContext(a, set, &Context{Context: ctx})
if nerr != nil {
_, _ = fmt.Fprintln(a.Writer, nerr)
_ = ShowAppHelp(context)
return nerr
}
context.shellComplete = shellComplete
if checkCompletions(context) {
return nil
}
if err != nil {
if a.OnUsageError != nil {
err := a.OnUsageError(context, err, false)
a.handleExitCoder(context, err)
return err
}
_, _ = fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error())
_ = ShowAppHelp(context)
return err
}
if !a.HideHelp && checkHelp(context) {
_ = ShowAppHelp(context)
return nil
}
if !a.HideVersion && checkVersion(context) {
ShowVersion(context)
return nil
}
cerr := checkRequiredFlags(a.Flags, context)
if cerr != nil {
_ = ShowAppHelp(context)
return cerr
}
if a.After != nil {
defer func() {
if afterErr := a.After(context); afterErr != nil {
if err != nil {
err = newMultiError(err, afterErr)
} else {
err = afterErr
}
}
}()
}
if a.Before != nil {
beforeErr := a.Before(context)
if beforeErr != nil {
a.handleExitCoder(context, beforeErr)
err = beforeErr
return err
}
}
args := context.Args()
if args.Present() {
name := args.First()
c := a.Command(name)
if c != nil {
return c.Run(context)
}
}
if a.Action == nil {
a.Action = helpCommand.Action
}
// Run default Action
err = a.Action(context)
a.handleExitCoder(context, err)
return err
}
// RunAndExitOnError calls .Run() and exits non-zero if an error was returned
//
// Deprecated: instead you should return an error that fulfills cli.ExitCoder
// to cli.App.Run. This will cause the application to exit with the given eror
// code in the cli.ExitCoder
func (a *App) RunAndExitOnError() {
if err := a.Run(os.Args); err != nil {
_, _ = fmt.Fprintln(a.ErrWriter, err)
OsExiter(1)
}
}
// RunAsSubcommand invokes the subcommand given the context, parses ctx.Args() to
// generate command-specific flags
func (a *App) RunAsSubcommand(ctx *Context) (err error) {
// Setup also handles HideHelp and HideHelpCommand
a.Setup()
var newCmds []*Command
for _, c := range a.Commands {
if c.HelpName == "" {
c.HelpName = fmt.Sprintf("%s %s", a.HelpName, c.Name)
}
newCmds = append(newCmds, c)
}
a.Commands = newCmds
set, err := a.newFlagSet()
if err != nil {
return err
}
err = parseIter(set, a, ctx.Args().Tail(), ctx.shellComplete)
nerr := normalizeFlags(a.Flags, set)
context := NewContext(a, set, ctx)
if nerr != nil {
_, _ = fmt.Fprintln(a.Writer, nerr)
_, _ = fmt.Fprintln(a.Writer)
if len(a.Commands) > 0 {
_ = ShowSubcommandHelp(context)
} else {
_ = ShowCommandHelp(ctx, context.Args().First())
}
return nerr
}
if checkCompletions(context) {
return nil
}
if err != nil {
if a.OnUsageError != nil {
err = a.OnUsageError(context, err, true)
a.handleExitCoder(context, err)
return err
}
_, _ = fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error())
_ = ShowSubcommandHelp(context)
return err
}
if len(a.Commands) > 0 {
if checkSubcommandHelp(context) {
return nil
}
} else {
if checkCommandHelp(ctx, context.Args().First()) {
return nil
}
}
cerr := checkRequiredFlags(a.Flags, context)
if cerr != nil {
_ = ShowSubcommandHelp(context)
return cerr
}
if a.After != nil {
defer func() {
afterErr := a.After(context)
if afterErr != nil {
a.handleExitCoder(context, err)
if err != nil {
err = newMultiError(err, afterErr)
} else {
err = afterErr
}
}
}()
}
if a.Before != nil {
beforeErr := a.Before(context)
if beforeErr != nil {
a.handleExitCoder(context, beforeErr)
err = beforeErr
return err
}
}
args := context.Args()
if args.Present() {
name := args.First()
c := a.Command(name)
if c != nil {
return c.Run(context)
}
}
// Run default Action
err = a.Action(context)
a.handleExitCoder(context, err)
return err
}
// Command returns the named command on App. Returns nil if the command does not exist
func (a *App) Command(name string) *Command {
for _, c := range a.Commands {
if c.HasName(name) {
return c
}
}
return nil
}
// VisibleCategories returns a slice of categories and commands that are
// Hidden=false
func (a *App) VisibleCategories() []CommandCategory {
ret := []CommandCategory{}
for _, category := range a.categories.Categories() {
if visible := func() CommandCategory {
if len(category.VisibleCommands()) > 0 {
return category
}
return nil
}(); visible != nil {
ret = append(ret, visible)
}
}
return ret
}
// VisibleCommands returns a slice of the Commands with Hidden=false
func (a *App) VisibleCommands() []*Command {
var ret []*Command
for _, command := range a.Commands {
if !command.Hidden {
ret = append(ret, command)
}
}
return ret
}
// VisibleFlags returns a slice of the Flags with Hidden=false
func (a *App) VisibleFlags() []Flag {
return visibleFlags(a.Flags)
}
func (a *App) appendFlag(fl Flag) {
if !hasFlag(a.Flags, fl) {
a.Flags = append(a.Flags, fl)
}
}
func (a *App) appendCommand(c *Command) {
if !hasCommand(a.Commands, c) {
a.Commands = append(a.Commands, c)
}
}
func (a *App) handleExitCoder(context *Context, err error) {
if a.ExitErrHandler != nil {
a.ExitErrHandler(context, err)
} else {
HandleExitCoder(err)
}
}
// Author represents someone who has contributed to a cli project.
type Author struct {
Name string // The Authors name
Email string // The Authors email
}
// String makes Author comply to the Stringer interface, to allow an easy print in the templating process
func (a *Author) String() string {
e := ""
if a.Email != "" {
e = " <" + a.Email + ">"
}
return fmt.Sprintf("%v%v", a.Name, e)
}
// HandleAction attempts to figure out which Action signature was used. If
// it's an ActionFunc or a func with the legacy signature for Action, the func
// is run!
func HandleAction(action interface{}, context *Context) (err error) {
switch a := action.(type) {
case ActionFunc:
return a(context)
case func(*Context) error:
return a(context)
case func(*Context): // deprecated function signature
a(context)
return nil
}
return errInvalidActionType
}

54
vendor/github.com/urfave/cli/v2/args.go generated vendored Normal file
View file

@ -0,0 +1,54 @@
package cli
type Args interface {
// Get returns the nth argument, or else a blank string
Get(n int) string
// First returns the first argument, or else a blank string
First() string
// Tail returns the rest of the arguments (not the first one)
// or else an empty string slice
Tail() []string
// Len returns the length of the wrapped slice
Len() int
// Present checks if there are any arguments present
Present() bool
// Slice returns a copy of the internal slice
Slice() []string
}
type args []string
func (a *args) Get(n int) string {
if len(*a) > n {
return (*a)[n]
}
return ""
}
func (a *args) First() string {
return a.Get(0)
}
func (a *args) Tail() []string {
if a.Len() >= 2 {
tail := []string((*a)[1:])
ret := make([]string, len(tail))
copy(ret, tail)
return ret
}
return []string{}
}
func (a *args) Len() int {
return len(*a)
}
func (a *args) Present() bool {
return a.Len() != 0
}
func (a *args) Slice() []string {
ret := make([]string, len(*a))
copy(ret, *a)
return ret
}

79
vendor/github.com/urfave/cli/v2/category.go generated vendored Normal file
View file

@ -0,0 +1,79 @@
package cli
// CommandCategories interface allows for category manipulation
type CommandCategories interface {
// AddCommand adds a command to a category, creating a new category if necessary.
AddCommand(category string, command *Command)
// categories returns a copy of the category slice
Categories() []CommandCategory
}
type commandCategories []*commandCategory
func newCommandCategories() CommandCategories {
ret := commandCategories([]*commandCategory{})
return &ret
}
func (c *commandCategories) Less(i, j int) bool {
return lexicographicLess((*c)[i].Name(), (*c)[j].Name())
}
func (c *commandCategories) Len() int {
return len(*c)
}
func (c *commandCategories) Swap(i, j int) {
(*c)[i], (*c)[j] = (*c)[j], (*c)[i]
}
func (c *commandCategories) AddCommand(category string, command *Command) {
for _, commandCategory := range []*commandCategory(*c) {
if commandCategory.name == category {
commandCategory.commands = append(commandCategory.commands, command)
return
}
}
newVal := append(*c,
&commandCategory{name: category, commands: []*Command{command}})
*c = newVal
}
func (c *commandCategories) Categories() []CommandCategory {
ret := make([]CommandCategory, len(*c))
for i, cat := range *c {
ret[i] = cat
}
return ret
}
// CommandCategory is a category containing commands.
type CommandCategory interface {
// Name returns the category name string
Name() string
// VisibleCommands returns a slice of the Commands with Hidden=false
VisibleCommands() []*Command
}
type commandCategory struct {
name string
commands []*Command
}
func (c *commandCategory) Name() string {
return c.name
}
func (c *commandCategory) VisibleCommands() []*Command {
if c.commands == nil {
c.commands = []*Command{}
}
var ret []*Command
for _, command := range c.commands {
if !command.Hidden {
ret = append(ret, command)
}
}
return ret
}

23
vendor/github.com/urfave/cli/v2/cli.go generated vendored Normal file
View file

@ -0,0 +1,23 @@
// Package cli provides a minimal framework for creating and organizing command line
// Go applications. cli is designed to be easy to understand and write, the most simple
// cli application can be written as follows:
// func main() {
// (&cli.App{}).Run(os.Args)
// }
//
// Of course this application does not do much, so let's make this an actual application:
// func main() {
// app := &cli.App{
// Name: "greet",
// Usage: "say a greeting",
// Action: func(c *cli.Context) error {
// fmt.Println("Greetings")
// return nil
// },
// }
//
// app.Run(os.Args)
// }
package cli
//go:generate go run flag-gen/main.go flag-gen/assets_vfsdata.go

300
vendor/github.com/urfave/cli/v2/command.go generated vendored Normal file
View file

@ -0,0 +1,300 @@
package cli
import (
"flag"
"fmt"
"sort"
"strings"
)
// Command is a subcommand for a cli.App.
type Command struct {
// The name of the command
Name string
// A list of aliases for the command
Aliases []string
// A short description of the usage of this command
Usage string
// Custom text to show on USAGE section of help
UsageText string
// A longer explanation of how the command works
Description string
// A short description of the arguments of this command
ArgsUsage string
// The category the command is part of
Category string
// The function to call when checking for bash command completions
BashComplete BashCompleteFunc
// An action to execute before any sub-subcommands are run, but after the context is ready
// If a non-nil error is returned, no sub-subcommands are run
Before BeforeFunc
// An action to execute after any subcommands are run, but after the subcommand has finished
// It is run even if Action() panics
After AfterFunc
// The function to call when this command is invoked
Action ActionFunc
// Execute this function if a usage error occurs.
OnUsageError OnUsageErrorFunc
// List of child commands
Subcommands []*Command
// List of flags to parse
Flags []Flag
// Treat all flags as normal arguments if true
SkipFlagParsing bool
// Boolean to hide built-in help command and help flag
HideHelp bool
// Boolean to hide built-in help command but keep help flag
// Ignored if HideHelp is true.
HideHelpCommand bool
// Boolean to hide this command from help or completion
Hidden bool
// Boolean to enable short-option handling so user can combine several
// single-character bool arguments into one
// i.e. foobar -o -v -> foobar -ov
UseShortOptionHandling bool
// Full name of command for help, defaults to full command name, including parent commands.
HelpName string
commandNamePath []string
// CustomHelpTemplate the text template for the command help topic.
// cli.go uses text/template to render templates. You can
// render custom help text by setting this variable.
CustomHelpTemplate string
}
type Commands []*Command
type CommandsByName []*Command
func (c CommandsByName) Len() int {
return len(c)
}
func (c CommandsByName) Less(i, j int) bool {
return lexicographicLess(c[i].Name, c[j].Name)
}
func (c CommandsByName) Swap(i, j int) {
c[i], c[j] = c[j], c[i]
}
// FullName returns the full name of the command.
// For subcommands this ensures that parent commands are part of the command path
func (c *Command) FullName() string {
if c.commandNamePath == nil {
return c.Name
}
return strings.Join(c.commandNamePath, " ")
}
// Run invokes the command given the context, parses ctx.Args() to generate command-specific flags
func (c *Command) Run(ctx *Context) (err error) {
if len(c.Subcommands) > 0 {
return c.startApp(ctx)
}
if !c.HideHelp && HelpFlag != nil {
// append help to flags
c.appendFlag(HelpFlag)
}
if ctx.App.UseShortOptionHandling {
c.UseShortOptionHandling = true
}
set, err := c.parseFlags(ctx.Args(), ctx.shellComplete)
context := NewContext(ctx.App, set, ctx)
context.Command = c
if checkCommandCompletions(context, c.Name) {
return nil
}
if err != nil {
if c.OnUsageError != nil {
err = c.OnUsageError(context, err, false)
context.App.handleExitCoder(context, err)
return err
}
_, _ = fmt.Fprintln(context.App.Writer, "Incorrect Usage:", err.Error())
_, _ = fmt.Fprintln(context.App.Writer)
_ = ShowCommandHelp(context, c.Name)
return err
}
if checkCommandHelp(context, c.Name) {
return nil
}
cerr := checkRequiredFlags(c.Flags, context)
if cerr != nil {
_ = ShowCommandHelp(context, c.Name)
return cerr
}
if c.After != nil {
defer func() {
afterErr := c.After(context)
if afterErr != nil {
context.App.handleExitCoder(context, err)
if err != nil {
err = newMultiError(err, afterErr)
} else {
err = afterErr
}
}
}()
}
if c.Before != nil {
err = c.Before(context)
if err != nil {
context.App.handleExitCoder(context, err)
return err
}
}
if c.Action == nil {
c.Action = helpSubcommand.Action
}
context.Command = c
err = c.Action(context)
if err != nil {
context.App.handleExitCoder(context, err)
}
return err
}
func (c *Command) newFlagSet() (*flag.FlagSet, error) {
return flagSet(c.Name, c.Flags)
}
func (c *Command) useShortOptionHandling() bool {
return c.UseShortOptionHandling
}
func (c *Command) parseFlags(args Args, shellComplete bool) (*flag.FlagSet, error) {
set, err := c.newFlagSet()
if err != nil {
return nil, err
}
if c.SkipFlagParsing {
return set, set.Parse(append([]string{"--"}, args.Tail()...))
}
err = parseIter(set, c, args.Tail(), shellComplete)
if err != nil {
return nil, err
}
err = normalizeFlags(c.Flags, set)
if err != nil {
return nil, err
}
return set, nil
}
// Names returns the names including short names and aliases.
func (c *Command) Names() []string {
return append([]string{c.Name}, c.Aliases...)
}
// HasName returns true if Command.Name matches given name
func (c *Command) HasName(name string) bool {
for _, n := range c.Names() {
if n == name {
return true
}
}
return false
}
func (c *Command) startApp(ctx *Context) error {
app := &App{
Metadata: ctx.App.Metadata,
Name: fmt.Sprintf("%s %s", ctx.App.Name, c.Name),
}
if c.HelpName == "" {
app.HelpName = c.HelpName
} else {
app.HelpName = app.Name
}
app.Usage = c.Usage
app.Description = c.Description
app.ArgsUsage = c.ArgsUsage
// set CommandNotFound
app.CommandNotFound = ctx.App.CommandNotFound
app.CustomAppHelpTemplate = c.CustomHelpTemplate
// set the flags and commands
app.Commands = c.Subcommands
app.Flags = c.Flags
app.HideHelp = c.HideHelp
app.HideHelpCommand = c.HideHelpCommand
app.Version = ctx.App.Version
app.HideVersion = true
app.Compiled = ctx.App.Compiled
app.Writer = ctx.App.Writer
app.ErrWriter = ctx.App.ErrWriter
app.ExitErrHandler = ctx.App.ExitErrHandler
app.UseShortOptionHandling = ctx.App.UseShortOptionHandling
app.categories = newCommandCategories()
for _, command := range c.Subcommands {
app.categories.AddCommand(command.Category, command)
}
sort.Sort(app.categories.(*commandCategories))
// bash completion
app.EnableBashCompletion = ctx.App.EnableBashCompletion
if c.BashComplete != nil {
app.BashComplete = c.BashComplete
}
// set the actions
app.Before = c.Before
app.After = c.After
if c.Action != nil {
app.Action = c.Action
} else {
app.Action = helpSubcommand.Action
}
app.OnUsageError = c.OnUsageError
for index, cc := range app.Commands {
app.Commands[index].commandNamePath = []string{c.Name, cc.Name}
}
return app.RunAsSubcommand(ctx)
}
// VisibleFlags returns a slice of the Flags with Hidden=false
func (c *Command) VisibleFlags() []Flag {
return visibleFlags(c.Flags)
}
func (c *Command) appendFlag(fl Flag) {
if !hasFlag(c.Flags, fl) {
c.Flags = append(c.Flags, fl)
}
}
func hasCommand(commands []*Command, command *Command) bool {
for _, existing := range commands {
if command == existing {
return true
}
}
return false
}

273
vendor/github.com/urfave/cli/v2/context.go generated vendored Normal file
View file

@ -0,0 +1,273 @@
package cli
import (
"context"
"errors"
"flag"
"fmt"
"strings"
)
// Context is a type that is passed through to
// each Handler action in a cli application. Context
// can be used to retrieve context-specific args and
// parsed command-line options.
type Context struct {
context.Context
App *App
Command *Command
shellComplete bool
flagSet *flag.FlagSet
parentContext *Context
}
// NewContext creates a new context. For use in when invoking an App or Command action.
func NewContext(app *App, set *flag.FlagSet, parentCtx *Context) *Context {
c := &Context{App: app, flagSet: set, parentContext: parentCtx}
if parentCtx != nil {
c.Context = parentCtx.Context
c.shellComplete = parentCtx.shellComplete
if parentCtx.flagSet == nil {
parentCtx.flagSet = &flag.FlagSet{}
}
}
c.Command = &Command{}
if c.Context == nil {
c.Context = context.Background()
}
return c
}
// NumFlags returns the number of flags set
func (c *Context) NumFlags() int {
return c.flagSet.NFlag()
}
// Set sets a context flag to a value.
func (c *Context) Set(name, value string) error {
return c.flagSet.Set(name, value)
}
// IsSet determines if the flag was actually set
func (c *Context) IsSet(name string) bool {
if fs := lookupFlagSet(name, c); fs != nil {
if fs := lookupFlagSet(name, c); fs != nil {
isSet := false
fs.Visit(func(f *flag.Flag) {
if f.Name == name {
isSet = true
}
})
if isSet {
return true
}
}
f := lookupFlag(name, c)
if f == nil {
return false
}
return f.IsSet()
}
return false
}
// LocalFlagNames returns a slice of flag names used in this context.
func (c *Context) LocalFlagNames() []string {
var names []string
c.flagSet.Visit(makeFlagNameVisitor(&names))
return names
}
// FlagNames returns a slice of flag names used by the this context and all of
// its parent contexts.
func (c *Context) FlagNames() []string {
var names []string
for _, ctx := range c.Lineage() {
ctx.flagSet.Visit(makeFlagNameVisitor(&names))
}
return names
}
// Lineage returns *this* context and all of its ancestor contexts in order from
// child to parent
func (c *Context) Lineage() []*Context {
var lineage []*Context
for cur := c; cur != nil; cur = cur.parentContext {
lineage = append(lineage, cur)
}
return lineage
}
// Value returns the value of the flag corresponding to `name`
func (c *Context) Value(name string) interface{} {
return c.flagSet.Lookup(name).Value.(flag.Getter).Get()
}
// Args returns the command line arguments associated with the context.
func (c *Context) Args() Args {
ret := args(c.flagSet.Args())
return &ret
}
// NArg returns the number of the command line arguments.
func (c *Context) NArg() int {
return c.Args().Len()
}
func lookupFlag(name string, ctx *Context) Flag {
for _, c := range ctx.Lineage() {
if c.Command == nil {
continue
}
for _, f := range c.Command.Flags {
for _, n := range f.Names() {
if n == name {
return f
}
}
}
}
if ctx.App != nil {
for _, f := range ctx.App.Flags {
for _, n := range f.Names() {
if n == name {
return f
}
}
}
}
return nil
}
func lookupFlagSet(name string, ctx *Context) *flag.FlagSet {
for _, c := range ctx.Lineage() {
if f := c.flagSet.Lookup(name); f != nil {
return c.flagSet
}
}
return nil
}
func copyFlag(name string, ff *flag.Flag, set *flag.FlagSet) {
switch ff.Value.(type) {
case Serializer:
_ = set.Set(name, ff.Value.(Serializer).Serialize())
default:
_ = set.Set(name, ff.Value.String())
}
}
func normalizeFlags(flags []Flag, set *flag.FlagSet) error {
visited := make(map[string]bool)
set.Visit(func(f *flag.Flag) {
visited[f.Name] = true
})
for _, f := range flags {
parts := f.Names()
if len(parts) == 1 {
continue
}
var ff *flag.Flag
for _, name := range parts {
name = strings.Trim(name, " ")
if visited[name] {
if ff != nil {
return errors.New("Cannot use two forms of the same flag: " + name + " " + ff.Name)
}
ff = set.Lookup(name)
}
}
if ff == nil {
continue
}
for _, name := range parts {
name = strings.Trim(name, " ")
if !visited[name] {
copyFlag(name, ff, set)
}
}
}
return nil
}
func makeFlagNameVisitor(names *[]string) func(*flag.Flag) {
return func(f *flag.Flag) {
nameParts := strings.Split(f.Name, ",")
name := strings.TrimSpace(nameParts[0])
for _, part := range nameParts {
part = strings.TrimSpace(part)
if len(part) > len(name) {
name = part
}
}
if name != "" {
*names = append(*names, name)
}
}
}
type requiredFlagsErr interface {
error
getMissingFlags() []string
}
type errRequiredFlags struct {
missingFlags []string
}
func (e *errRequiredFlags) Error() string {
numberOfMissingFlags := len(e.missingFlags)
if numberOfMissingFlags == 1 {
return fmt.Sprintf("Required flag %q not set", e.missingFlags[0])
}
joinedMissingFlags := strings.Join(e.missingFlags, ", ")
return fmt.Sprintf("Required flags %q not set", joinedMissingFlags)
}
func (e *errRequiredFlags) getMissingFlags() []string {
return e.missingFlags
}
func checkRequiredFlags(flags []Flag, context *Context) requiredFlagsErr {
var missingFlags []string
for _, f := range flags {
if rf, ok := f.(RequiredFlag); ok && rf.IsRequired() {
var flagPresent bool
var flagName string
for _, key := range f.Names() {
if len(key) > 1 {
flagName = key
}
if context.IsSet(strings.TrimSpace(key)) {
flagPresent = true
}
}
if !flagPresent && flagName != "" {
missingFlags = append(missingFlags, flagName)
}
}
}
if len(missingFlags) != 0 {
return &errRequiredFlags{missingFlags: missingFlags}
}
return nil
}

148
vendor/github.com/urfave/cli/v2/docs.go generated vendored Normal file
View file

@ -0,0 +1,148 @@
package cli
import (
"bytes"
"fmt"
"io"
"sort"
"strings"
"text/template"
"github.com/cpuguy83/go-md2man/v2/md2man"
)
// ToMarkdown creates a markdown string for the `*App`
// The function errors if either parsing or writing of the string fails.
func (a *App) ToMarkdown() (string, error) {
var w bytes.Buffer
if err := a.writeDocTemplate(&w); err != nil {
return "", err
}
return w.String(), nil
}
// ToMan creates a man page string for the `*App`
// The function errors if either parsing or writing of the string fails.
func (a *App) ToMan() (string, error) {
var w bytes.Buffer
if err := a.writeDocTemplate(&w); err != nil {
return "", err
}
man := md2man.Render(w.Bytes())
return string(man), nil
}
type cliTemplate struct {
App *App
Commands []string
GlobalArgs []string
SynopsisArgs []string
}
func (a *App) writeDocTemplate(w io.Writer) error {
const name = "cli"
t, err := template.New(name).Parse(MarkdownDocTemplate)
if err != nil {
return err
}
return t.ExecuteTemplate(w, name, &cliTemplate{
App: a,
Commands: prepareCommands(a.Commands, 0),
GlobalArgs: prepareArgsWithValues(a.VisibleFlags()),
SynopsisArgs: prepareArgsSynopsis(a.VisibleFlags()),
})
}
func prepareCommands(commands []*Command, level int) []string {
var coms []string
for _, command := range commands {
if command.Hidden {
continue
}
usage := ""
if command.Usage != "" {
usage = command.Usage
}
prepared := fmt.Sprintf("%s %s\n\n%s\n",
strings.Repeat("#", level+2),
strings.Join(command.Names(), ", "),
usage,
)
flags := prepareArgsWithValues(command.Flags)
if len(flags) > 0 {
prepared += fmt.Sprintf("\n%s", strings.Join(flags, "\n"))
}
coms = append(coms, prepared)
// recursevly iterate subcommands
if len(command.Subcommands) > 0 {
coms = append(
coms,
prepareCommands(command.Subcommands, level+1)...,
)
}
}
return coms
}
func prepareArgsWithValues(flags []Flag) []string {
return prepareFlags(flags, ", ", "**", "**", `""`, true)
}
func prepareArgsSynopsis(flags []Flag) []string {
return prepareFlags(flags, "|", "[", "]", "[value]", false)
}
func prepareFlags(
flags []Flag,
sep, opener, closer, value string,
addDetails bool,
) []string {
args := []string{}
for _, f := range flags {
flag, ok := f.(DocGenerationFlag)
if !ok {
continue
}
modifiedArg := opener
for _, s := range flag.Names() {
trimmed := strings.TrimSpace(s)
if len(modifiedArg) > len(opener) {
modifiedArg += sep
}
if len(trimmed) > 1 {
modifiedArg += fmt.Sprintf("--%s", trimmed)
} else {
modifiedArg += fmt.Sprintf("-%s", trimmed)
}
}
modifiedArg += closer
if flag.TakesValue() {
modifiedArg += fmt.Sprintf("=%s", value)
}
if addDetails {
modifiedArg += flagDetails(flag)
}
args = append(args, modifiedArg+"\n")
}
sort.Strings(args)
return args
}
// flagDetails returns a string containing the flags metadata
func flagDetails(flag DocGenerationFlag) string {
description := flag.GetUsage()
value := flag.GetValue()
if value != "" {
description += " (default: " + value + ")"
}
return ": " + description
}

141
vendor/github.com/urfave/cli/v2/errors.go generated vendored Normal file
View file

@ -0,0 +1,141 @@
package cli
import (
"fmt"
"io"
"os"
"strings"
)
// OsExiter is the function used when the app exits. If not set defaults to os.Exit.
var OsExiter = os.Exit
// ErrWriter is used to write errors to the user. This can be anything
// implementing the io.Writer interface and defaults to os.Stderr.
var ErrWriter io.Writer = os.Stderr
// MultiError is an error that wraps multiple errors.
type MultiError interface {
error
Errors() []error
}
// newMultiError creates a new MultiError. Pass in one or more errors.
func newMultiError(err ...error) MultiError {
ret := multiError(err)
return &ret
}
type multiError []error
// Error implements the error interface.
func (m *multiError) Error() string {
errs := make([]string, len(*m))
for i, err := range *m {
errs[i] = err.Error()
}
return strings.Join(errs, "\n")
}
// Errors returns a copy of the errors slice
func (m *multiError) Errors() []error {
errs := make([]error, len(*m))
for _, err := range *m {
errs = append(errs, err)
}
return errs
}
// ErrorFormatter is the interface that will suitably format the error output
type ErrorFormatter interface {
Format(s fmt.State, verb rune)
}
// ExitCoder is the interface checked by `App` and `Command` for a custom exit
// code
type ExitCoder interface {
error
ExitCode() int
}
type exitError struct {
exitCode int
message interface{}
}
// NewExitError calls Exit to create a new ExitCoder.
//
// Deprecated: This function is a duplicate of Exit and will eventually be removed.
func NewExitError(message interface{}, exitCode int) ExitCoder {
return Exit(message, exitCode)
}
// Exit wraps a message and exit code into an error, which by default is
// handled with a call to os.Exit during default error handling.
//
// This is the simplest way to trigger a non-zero exit code for an App without
// having to call os.Exit manually. During testing, this behavior can be avoided
// by overiding the ExitErrHandler function on an App or the package-global
// OsExiter function.
func Exit(message interface{}, exitCode int) ExitCoder {
return &exitError{
message: message,
exitCode: exitCode,
}
}
func (ee *exitError) Error() string {
return fmt.Sprintf("%v", ee.message)
}
func (ee *exitError) ExitCode() int {
return ee.exitCode
}
// HandleExitCoder handles errors implementing ExitCoder by printing their
// message and calling OsExiter with the given exit code.
//
// If the given error instead implements MultiError, each error will be checked
// for the ExitCoder interface, and OsExiter will be called with the last exit
// code found, or exit code 1 if no ExitCoder is found.
//
// This function is the default error-handling behavior for an App.
func HandleExitCoder(err error) {
if err == nil {
return
}
if exitErr, ok := err.(ExitCoder); ok {
if err.Error() != "" {
if _, ok := exitErr.(ErrorFormatter); ok {
_, _ = fmt.Fprintf(ErrWriter, "%+v\n", err)
} else {
_, _ = fmt.Fprintln(ErrWriter, err)
}
}
OsExiter(exitErr.ExitCode())
return
}
if multiErr, ok := err.(MultiError); ok {
code := handleMultiError(multiErr)
OsExiter(code)
return
}
}
func handleMultiError(multiErr MultiError) int {
code := 1
for _, merr := range multiErr.Errors() {
if multiErr2, ok := merr.(MultiError); ok {
code = handleMultiError(multiErr2)
} else if merr != nil {
fmt.Fprintln(ErrWriter, merr)
if exitErr, ok := merr.(ExitCoder); ok {
code = exitErr.ExitCode()
}
}
}
return code
}

196
vendor/github.com/urfave/cli/v2/fish.go generated vendored Normal file
View file

@ -0,0 +1,196 @@
package cli
import (
"bytes"
"fmt"
"io"
"strings"
"text/template"
)
// ToFishCompletion creates a fish completion string for the `*App`
// The function errors if either parsing or writing of the string fails.
func (a *App) ToFishCompletion() (string, error) {
var w bytes.Buffer
if err := a.writeFishCompletionTemplate(&w); err != nil {
return "", err
}
return w.String(), nil
}
type fishCompletionTemplate struct {
App *App
Completions []string
AllCommands []string
}
func (a *App) writeFishCompletionTemplate(w io.Writer) error {
const name = "cli"
t, err := template.New(name).Parse(FishCompletionTemplate)
if err != nil {
return err
}
allCommands := []string{}
// Add global flags
completions := a.prepareFishFlags(a.VisibleFlags(), allCommands)
// Add help flag
if !a.HideHelp {
completions = append(
completions,
a.prepareFishFlags([]Flag{HelpFlag}, allCommands)...,
)
}
// Add version flag
if !a.HideVersion {
completions = append(
completions,
a.prepareFishFlags([]Flag{VersionFlag}, allCommands)...,
)
}
// Add commands and their flags
completions = append(
completions,
a.prepareFishCommands(a.VisibleCommands(), &allCommands, []string{})...,
)
return t.ExecuteTemplate(w, name, &fishCompletionTemplate{
App: a,
Completions: completions,
AllCommands: allCommands,
})
}
func (a *App) prepareFishCommands(commands []*Command, allCommands *[]string, previousCommands []string) []string {
completions := []string{}
for _, command := range commands {
if command.Hidden {
continue
}
var completion strings.Builder
completion.WriteString(fmt.Sprintf(
"complete -r -c %s -n '%s' -a '%s'",
a.Name,
a.fishSubcommandHelper(previousCommands),
strings.Join(command.Names(), " "),
))
if command.Usage != "" {
completion.WriteString(fmt.Sprintf(" -d '%s'",
escapeSingleQuotes(command.Usage)))
}
if !command.HideHelp {
completions = append(
completions,
a.prepareFishFlags([]Flag{HelpFlag}, command.Names())...,
)
}
*allCommands = append(*allCommands, command.Names()...)
completions = append(completions, completion.String())
completions = append(
completions,
a.prepareFishFlags(command.Flags, command.Names())...,
)
// recursevly iterate subcommands
if len(command.Subcommands) > 0 {
completions = append(
completions,
a.prepareFishCommands(
command.Subcommands, allCommands, command.Names(),
)...,
)
}
}
return completions
}
func (a *App) prepareFishFlags(flags []Flag, previousCommands []string) []string {
completions := []string{}
for _, f := range flags {
flag, ok := f.(DocGenerationFlag)
if !ok {
continue
}
completion := &strings.Builder{}
completion.WriteString(fmt.Sprintf(
"complete -c %s -n '%s'",
a.Name,
a.fishSubcommandHelper(previousCommands),
))
fishAddFileFlag(f, completion)
for idx, opt := range flag.Names() {
if idx == 0 {
completion.WriteString(fmt.Sprintf(
" -l %s", strings.TrimSpace(opt),
))
} else {
completion.WriteString(fmt.Sprintf(
" -s %s", strings.TrimSpace(opt),
))
}
}
if flag.TakesValue() {
completion.WriteString(" -r")
}
if flag.GetUsage() != "" {
completion.WriteString(fmt.Sprintf(" -d '%s'",
escapeSingleQuotes(flag.GetUsage())))
}
completions = append(completions, completion.String())
}
return completions
}
func fishAddFileFlag(flag Flag, completion *strings.Builder) {
switch f := flag.(type) {
case *GenericFlag:
if f.TakesFile {
return
}
case *StringFlag:
if f.TakesFile {
return
}
case *StringSliceFlag:
if f.TakesFile {
return
}
case *PathFlag:
if f.TakesFile {
return
}
}
completion.WriteString(" -f")
}
func (a *App) fishSubcommandHelper(allCommands []string) string {
fishHelper := fmt.Sprintf("__fish_%s_no_subcommand", a.Name)
if len(allCommands) > 0 {
fishHelper = fmt.Sprintf(
"__fish_seen_subcommand_from %s",
strings.Join(allCommands, " "),
)
}
return fishHelper
}
func escapeSingleQuotes(input string) string {
return strings.Replace(input, `'`, `\'`, -1)
}

388
vendor/github.com/urfave/cli/v2/flag.go generated vendored Normal file
View file

@ -0,0 +1,388 @@
package cli
import (
"flag"
"fmt"
"io/ioutil"
"reflect"
"regexp"
"runtime"
"strconv"
"strings"
"syscall"
"time"
)
const defaultPlaceholder = "value"
var (
slPfx = fmt.Sprintf("sl:::%d:::", time.Now().UTC().UnixNano())
commaWhitespace = regexp.MustCompile("[, ]+.*")
)
// BashCompletionFlag enables bash-completion for all commands and subcommands
var BashCompletionFlag Flag = &BoolFlag{
Name: "generate-bash-completion",
Hidden: true,
}
// VersionFlag prints the version for the application
var VersionFlag Flag = &BoolFlag{
Name: "version",
Aliases: []string{"v"},
Usage: "print the version",
}
// HelpFlag prints the help for all commands and subcommands.
// Set to nil to disable the flag. The subcommand
// will still be added unless HideHelp or HideHelpCommand is set to true.
var HelpFlag Flag = &BoolFlag{
Name: "help",
Aliases: []string{"h"},
Usage: "show help",
}
// FlagStringer converts a flag definition to a string. This is used by help
// to display a flag.
var FlagStringer FlagStringFunc = stringifyFlag
// Serializer is used to circumvent the limitations of flag.FlagSet.Set
type Serializer interface {
Serialize() string
}
// FlagNamePrefixer converts a full flag name and its placeholder into the help
// message flag prefix. This is used by the default FlagStringer.
var FlagNamePrefixer FlagNamePrefixFunc = prefixedNames
// FlagEnvHinter annotates flag help message with the environment variable
// details. This is used by the default FlagStringer.
var FlagEnvHinter FlagEnvHintFunc = withEnvHint
// FlagFileHinter annotates flag help message with the environment variable
// details. This is used by the default FlagStringer.
var FlagFileHinter FlagFileHintFunc = withFileHint
// FlagsByName is a slice of Flag.
type FlagsByName []Flag
func (f FlagsByName) Len() int {
return len(f)
}
func (f FlagsByName) Less(i, j int) bool {
if len(f[j].Names()) == 0 {
return false
} else if len(f[i].Names()) == 0 {
return true
}
return lexicographicLess(f[i].Names()[0], f[j].Names()[0])
}
func (f FlagsByName) Swap(i, j int) {
f[i], f[j] = f[j], f[i]
}
// Flag is a common interface related to parsing flags in cli.
// For more advanced flag parsing techniques, it is recommended that
// this interface be implemented.
type Flag interface {
fmt.Stringer
// Apply Flag settings to the given flag set
Apply(*flag.FlagSet) error
Names() []string
IsSet() bool
}
// RequiredFlag is an interface that allows us to mark flags as required
// it allows flags required flags to be backwards compatible with the Flag interface
type RequiredFlag interface {
Flag
IsRequired() bool
}
// DocGenerationFlag is an interface that allows documentation generation for the flag
type DocGenerationFlag interface {
Flag
// TakesValue returns true if the flag takes a value, otherwise false
TakesValue() bool
// GetUsage returns the usage string for the flag
GetUsage() string
// GetValue returns the flags value as string representation and an empty
// string if the flag takes no value at all.
GetValue() string
}
func flagSet(name string, flags []Flag) (*flag.FlagSet, error) {
set := flag.NewFlagSet(name, flag.ContinueOnError)
for _, f := range flags {
if err := f.Apply(set); err != nil {
return nil, err
}
}
set.SetOutput(ioutil.Discard)
return set, nil
}
func visibleFlags(fl []Flag) []Flag {
var visible []Flag
for _, f := range fl {
field := flagValue(f).FieldByName("Hidden")
if !field.IsValid() || !field.Bool() {
visible = append(visible, f)
}
}
return visible
}
func prefixFor(name string) (prefix string) {
if len(name) == 1 {
prefix = "-"
} else {
prefix = "--"
}
return
}
// Returns the placeholder, if any, and the unquoted usage string.
func unquoteUsage(usage string) (string, string) {
for i := 0; i < len(usage); i++ {
if usage[i] == '`' {
for j := i + 1; j < len(usage); j++ {
if usage[j] == '`' {
name := usage[i+1 : j]
usage = usage[:i] + name + usage[j+1:]
return name, usage
}
}
break
}
}
return "", usage
}
func prefixedNames(names []string, placeholder string) string {
var prefixed string
for i, name := range names {
if name == "" {
continue
}
prefixed += prefixFor(name) + name
if placeholder != "" {
prefixed += " " + placeholder
}
if i < len(names)-1 {
prefixed += ", "
}
}
return prefixed
}
func withEnvHint(envVars []string, str string) string {
envText := ""
if envVars != nil && len(envVars) > 0 {
prefix := "$"
suffix := ""
sep := ", $"
if runtime.GOOS == "windows" {
prefix = "%"
suffix = "%"
sep = "%, %"
}
envText = fmt.Sprintf(" [%s%s%s]", prefix, strings.Join(envVars, sep), suffix)
}
return str + envText
}
func flagNames(name string, aliases []string) []string {
var ret []string
for _, part := range append([]string{name}, aliases...) {
// v1 -> v2 migration warning zone:
// Strip off anything after the first found comma or space, which
// *hopefully* makes it a tiny bit more obvious that unexpected behavior is
// caused by using the v1 form of stringly typed "Name".
ret = append(ret, commaWhitespace.ReplaceAllString(part, ""))
}
return ret
}
func flagStringSliceField(f Flag, name string) []string {
fv := flagValue(f)
field := fv.FieldByName(name)
if field.IsValid() {
return field.Interface().([]string)
}
return []string{}
}
func withFileHint(filePath, str string) string {
fileText := ""
if filePath != "" {
fileText = fmt.Sprintf(" [%s]", filePath)
}
return str + fileText
}
func flagValue(f Flag) reflect.Value {
fv := reflect.ValueOf(f)
for fv.Kind() == reflect.Ptr {
fv = reflect.Indirect(fv)
}
return fv
}
func formatDefault(format string) string {
return " (default: " + format + ")"
}
func stringifyFlag(f Flag) string {
fv := flagValue(f)
switch f := f.(type) {
case *IntSliceFlag:
return withEnvHint(flagStringSliceField(f, "EnvVars"),
stringifyIntSliceFlag(f))
case *Int64SliceFlag:
return withEnvHint(flagStringSliceField(f, "EnvVars"),
stringifyInt64SliceFlag(f))
case *Float64SliceFlag:
return withEnvHint(flagStringSliceField(f, "EnvVars"),
stringifyFloat64SliceFlag(f))
case *StringSliceFlag:
return withEnvHint(flagStringSliceField(f, "EnvVars"),
stringifyStringSliceFlag(f))
}
placeholder, usage := unquoteUsage(fv.FieldByName("Usage").String())
needsPlaceholder := false
defaultValueString := ""
val := fv.FieldByName("Value")
if val.IsValid() {
needsPlaceholder = val.Kind() != reflect.Bool
defaultValueString = fmt.Sprintf(formatDefault("%v"), val.Interface())
if val.Kind() == reflect.String && val.String() != "" {
defaultValueString = fmt.Sprintf(formatDefault("%q"), val.String())
}
}
helpText := fv.FieldByName("DefaultText")
if helpText.IsValid() && helpText.String() != "" {
needsPlaceholder = val.Kind() != reflect.Bool
defaultValueString = fmt.Sprintf(formatDefault("%s"), helpText.String())
}
if defaultValueString == formatDefault("") {
defaultValueString = ""
}
if needsPlaceholder && placeholder == "" {
placeholder = defaultPlaceholder
}
usageWithDefault := strings.TrimSpace(usage + defaultValueString)
return withEnvHint(flagStringSliceField(f, "EnvVars"),
fmt.Sprintf("%s\t%s", prefixedNames(f.Names(), placeholder), usageWithDefault))
}
func stringifyIntSliceFlag(f *IntSliceFlag) string {
var defaultVals []string
if f.Value != nil && len(f.Value.Value()) > 0 {
for _, i := range f.Value.Value() {
defaultVals = append(defaultVals, strconv.Itoa(i))
}
}
return stringifySliceFlag(f.Usage, f.Names(), defaultVals)
}
func stringifyInt64SliceFlag(f *Int64SliceFlag) string {
var defaultVals []string
if f.Value != nil && len(f.Value.Value()) > 0 {
for _, i := range f.Value.Value() {
defaultVals = append(defaultVals, strconv.FormatInt(i, 10))
}
}
return stringifySliceFlag(f.Usage, f.Names(), defaultVals)
}
func stringifyFloat64SliceFlag(f *Float64SliceFlag) string {
var defaultVals []string
if f.Value != nil && len(f.Value.Value()) > 0 {
for _, i := range f.Value.Value() {
defaultVals = append(defaultVals, strings.TrimRight(strings.TrimRight(fmt.Sprintf("%f", i), "0"), "."))
}
}
return stringifySliceFlag(f.Usage, f.Names(), defaultVals)
}
func stringifyStringSliceFlag(f *StringSliceFlag) string {
var defaultVals []string
if f.Value != nil && len(f.Value.Value()) > 0 {
for _, s := range f.Value.Value() {
if len(s) > 0 {
defaultVals = append(defaultVals, strconv.Quote(s))
}
}
}
return stringifySliceFlag(f.Usage, f.Names(), defaultVals)
}
func stringifySliceFlag(usage string, names, defaultVals []string) string {
placeholder, usage := unquoteUsage(usage)
if placeholder == "" {
placeholder = defaultPlaceholder
}
defaultVal := ""
if len(defaultVals) > 0 {
defaultVal = fmt.Sprintf(formatDefault("%s"), strings.Join(defaultVals, ", "))
}
usageWithDefault := strings.TrimSpace(fmt.Sprintf("%s%s", usage, defaultVal))
return fmt.Sprintf("%s\t%s", prefixedNames(names, placeholder), usageWithDefault)
}
func hasFlag(flags []Flag, fl Flag) bool {
for _, existing := range flags {
if fl == existing {
return true
}
}
return false
}
func flagFromEnvOrFile(envVars []string, filePath string) (val string, ok bool) {
for _, envVar := range envVars {
envVar = strings.TrimSpace(envVar)
if val, ok := syscall.Getenv(envVar); ok {
return val, true
}
}
for _, fileVar := range strings.Split(filePath, ",") {
if data, err := ioutil.ReadFile(fileVar); err == nil {
return string(data), true
}
}
return "", false
}

106
vendor/github.com/urfave/cli/v2/flag_bool.go generated vendored Normal file
View file

@ -0,0 +1,106 @@
package cli
import (
"flag"
"fmt"
"strconv"
)
// BoolFlag is a flag with type bool
type BoolFlag struct {
Name string
Aliases []string
Usage string
EnvVars []string
FilePath string
Required bool
Hidden bool
Value bool
DefaultText string
Destination *bool
HasBeenSet bool
}
// IsSet returns whether or not the flag has been set through env or file
func (f *BoolFlag) IsSet() bool {
return f.HasBeenSet
}
// String returns a readable representation of this value
// (for usage defaults)
func (f *BoolFlag) String() string {
return FlagStringer(f)
}
// Names returns the names of the flag
func (f *BoolFlag) Names() []string {
return flagNames(f.Name, f.Aliases)
}
// IsRequired returns whether or not the flag is required
func (f *BoolFlag) IsRequired() bool {
return f.Required
}
// TakesValue returns true of the flag takes a value, otherwise false
func (f *BoolFlag) TakesValue() bool {
return false
}
// GetUsage returns the usage string for the flag
func (f *BoolFlag) GetUsage() string {
return f.Usage
}
// GetValue returns the flags value as string representation and an empty
// string if the flag takes no value at all.
func (f *BoolFlag) GetValue() string {
return ""
}
// Apply populates the flag given the flag set and environment
func (f *BoolFlag) Apply(set *flag.FlagSet) error {
if val, ok := flagFromEnvOrFile(f.EnvVars, f.FilePath); ok {
if val != "" {
valBool, err := strconv.ParseBool(val)
if err != nil {
return fmt.Errorf("could not parse %q as bool value for flag %s: %s", val, f.Name, err)
}
f.Value = valBool
f.HasBeenSet = true
}
}
for _, name := range f.Names() {
if f.Destination != nil {
set.BoolVar(f.Destination, name, f.Value, f.Usage)
continue
}
set.Bool(name, f.Value, f.Usage)
}
return nil
}
// Bool looks up the value of a local BoolFlag, returns
// false if not found
func (c *Context) Bool(name string) bool {
if fs := lookupFlagSet(name, c); fs != nil {
return lookupBool(name, fs)
}
return false
}
func lookupBool(name string, set *flag.FlagSet) bool {
f := set.Lookup(name)
if f != nil {
parsed, err := strconv.ParseBool(f.Value.String())
if err != nil {
return false
}
return parsed
}
return false
}

105
vendor/github.com/urfave/cli/v2/flag_duration.go generated vendored Normal file
View file

@ -0,0 +1,105 @@
package cli
import (
"flag"
"fmt"
"time"
)
// DurationFlag is a flag with type time.Duration (see https://golang.org/pkg/time/#ParseDuration)
type DurationFlag struct {
Name string
Aliases []string
Usage string
EnvVars []string
FilePath string
Required bool
Hidden bool
Value time.Duration
DefaultText string
Destination *time.Duration
HasBeenSet bool
}
// IsSet returns whether or not the flag has been set through env or file
func (f *DurationFlag) IsSet() bool {
return f.HasBeenSet
}
// String returns a readable representation of this value
// (for usage defaults)
func (f *DurationFlag) String() string {
return FlagStringer(f)
}
// Names returns the names of the flag
func (f *DurationFlag) Names() []string {
return flagNames(f.Name, f.Aliases)
}
// IsRequired returns whether or not the flag is required
func (f *DurationFlag) IsRequired() bool {
return f.Required
}
// TakesValue returns true of the flag takes a value, otherwise false
func (f *DurationFlag) TakesValue() bool {
return true
}
// GetUsage returns the usage string for the flag
func (f *DurationFlag) GetUsage() string {
return f.Usage
}
// GetValue returns the flags value as string representation and an empty
// string if the flag takes no value at all.
func (f *DurationFlag) GetValue() string {
return f.Value.String()
}
// Apply populates the flag given the flag set and environment
func (f *DurationFlag) Apply(set *flag.FlagSet) error {
if val, ok := flagFromEnvOrFile(f.EnvVars, f.FilePath); ok {
if val != "" {
valDuration, err := time.ParseDuration(val)
if err != nil {
return fmt.Errorf("could not parse %q as duration value for flag %s: %s", val, f.Name, err)
}
f.Value = valDuration
f.HasBeenSet = true
}
}
for _, name := range f.Names() {
if f.Destination != nil {
set.DurationVar(f.Destination, name, f.Value, f.Usage)
continue
}
set.Duration(name, f.Value, f.Usage)
}
return nil
}
// Duration looks up the value of a local DurationFlag, returns
// 0 if not found
func (c *Context) Duration(name string) time.Duration {
if fs := lookupFlagSet(name, c); fs != nil {
return lookupDuration(name, fs)
}
return 0
}
func lookupDuration(name string, set *flag.FlagSet) time.Duration {
f := set.Lookup(name)
if f != nil {
parsed, err := time.ParseDuration(f.Value.String())
if err != nil {
return 0
}
return parsed
}
return 0
}

Some files were not shown because too many files have changed in this diff Show more