From 0ad995f2b406ffb62ffb6ec35545882d55a0f8d5 Mon Sep 17 00:00:00 2001 From: Manfred Touron Date: Thu, 13 Sep 2018 18:14:17 +0200 Subject: [PATCH] feat: refactor cli (#7) --- db.go => cmd_db.go | 0 fetch.go => cmd_pull.go | 56 ++- render.go => cmd_run.go | 54 +-- examples/depviz/Makefile | 8 +- examples/depviz/depviz.svg | 857 +++++++++++++++++++----------------- examples/depviz/orphans.svg | 102 +++++ graphviz.go | 4 +- main.go | 4 +- 8 files changed, 632 insertions(+), 453 deletions(-) rename db.go => cmd_db.go (100%) rename fetch.go => cmd_pull.go (74%) rename render.go => cmd_run.go (64%) create mode 100644 examples/depviz/orphans.svg diff --git a/db.go b/cmd_db.go similarity index 100% rename from db.go rename to cmd_db.go diff --git a/fetch.go b/cmd_pull.go similarity index 74% rename from fetch.go rename to cmd_pull.go index 1fb28fb10..ed80015df 100644 --- a/fetch.go +++ b/cmd_pull.go @@ -7,6 +7,7 @@ import ( "io/ioutil" "log" "os" + "strings" "sync" "github.com/google/go-github/github" @@ -19,53 +20,76 @@ import ( "golang.org/x/oauth2" ) -type fetchOptions struct { +type pullOptions struct { // db DBOpts dbOptions - // fetch + // pull Repos []string GithubToken string `mapstructure:"github-token"` + GitlabToken string `mapstructure:"gitlab-token"` // includeExternalDeps bool + + Targets []string } -func (opts fetchOptions) String() string { +func (opts pullOptions) String() string { out, _ := json.Marshal(opts) return string(out) } -func fetchSetupFlags(flags *pflag.FlagSet, opts *fetchOptions) { - flags.StringSliceVarP(&opts.Repos, "repos", "r", []string{}, "list of repositories to aggregate issues from") // FIXME: get the default value dynamically from .git, if present +func pullSetupFlags(flags *pflag.FlagSet, opts *pullOptions) { flags.StringVarP(&opts.GithubToken, "github-token", "", "", "GitHub Token with 'issues' access") + flags.StringVarP(&opts.GitlabToken, "gitlab-token", "", "", "GitLab Token with 'issues' access") viper.BindPFlags(flags) } -func newFetchCommand() *cobra.Command { - opts := &fetchOptions{} +func newPullCommand() *cobra.Command { + opts := &pullOptions{} cmd := &cobra.Command{ - Use: "fetch", + Use: "pull", RunE: func(cmd *cobra.Command, args []string) error { if err := viper.Unmarshal(opts); err != nil { return err } - return fetch(opts) + return pull(opts) }, } - fetchSetupFlags(cmd.Flags(), opts) + pullSetupFlags(cmd.Flags(), opts) dbSetupFlags(cmd.Flags(), &opts.DBOpts) return cmd } -func fetch(opts *fetchOptions) error { - logger().Debug("fetch", zap.Stringer("opts", *opts)) +func getReposFromTargets(targets []string) []string { + reposMap := map[string]bool{} + + for _, target := range targets { + if _, err := os.Stat(target); err == nil { + logger().Fatal("filesystem target are not yet supported") + } + repo := strings.Split(target, "/issues")[0] + reposMap[repo] = true + } + repos := []string{} + for repo := range reposMap { + repos = append(repos, repo) + } + return repos +} + +func pull(opts *pullOptions) error { + logger().Debug("pull", zap.Stringer("opts", *opts)) var ( wg sync.WaitGroup allIssues []*Issue out = make(chan []*Issue, 100) ) - wg.Add(len(opts.Repos)) - for _, repoURL := range opts.Repos { + + repos := getReposFromTargets(opts.Targets) + + wg.Add(len(repos)) + for _, repoURL := range repos { repo := NewRepo(repoURL) switch repo.Provider() { case GitHubProvider: @@ -107,7 +131,7 @@ func fetch(opts *fetchOptions) error { }(repo) case GitLabProvider: go func(repo Repo) { - client := gitlab.NewClient(nil, os.Getenv("GITLAB_TOKEN")) + client := gitlab.NewClient(nil, opts.GitlabToken) client.SetBaseURL(fmt.Sprintf("%s/api/v4", repo.SiteURL())) //projectID := url.QueryEscape(repo.RepoPath()) @@ -123,7 +147,7 @@ func fetch(opts *fetchOptions) error { for { issues, resp, err := client.Issues.ListProjectIssues(projectID, opts) if err != nil { - logger().Error("failed to fetch issues", zap.Error(err)) + logger().Error("failed to pull issues", zap.Error(err)) return } total += len(issues) diff --git a/render.go b/cmd_run.go similarity index 64% rename from render.go rename to cmd_run.go index 17a68b041..ba5290a50 100644 --- a/render.go +++ b/cmd_run.go @@ -13,31 +13,33 @@ import ( "go.uber.org/zap" ) -type renderOptions struct { - // fetch - FetchOpts fetchOptions - NoFetch bool +type runOptions struct { + // pull + PullOpts pullOptions + NoPull bool // db DBOpts dbOptions - // render - RenderType string + // run + RunType string ShowClosed bool `mapstructure:"show-closed"` ShowOrphans bool EpicLabel string Destination string + + Targets []string //Preview bool } -func (opts renderOptions) String() string { +func (opts runOptions) String() string { out, _ := json.Marshal(opts) return string(out) } -func renderSetupFlags(flags *pflag.FlagSet, opts *renderOptions) { - flags.BoolVarP(&opts.NoFetch, "no-fetch", "f", false, "do not fetch new issues before rendering") - flags.StringVarP(&opts.RenderType, "type", "t", "roadmap", "graph type ('roadmap', 'orphans')") +func runSetupFlags(flags *pflag.FlagSet, opts *runOptions) { + flags.BoolVarP(&opts.NoPull, "no-pull", "f", false, "do not pull new issues before runing") + flags.StringVarP(&opts.RunType, "type", "t", "roadmap", "graph type ('roadmap', 'orphans')") flags.BoolVarP(&opts.ShowClosed, "show-closed", "", false, "show closed issues") flags.BoolVarP(&opts.ShowOrphans, "show-orphans", "", false, "show issues not linked to an epic") flags.StringVarP(&opts.EpicLabel, "epic-label", "", "", "label used for epics (empty means issues with dependencies but without dependants)") @@ -46,34 +48,36 @@ func renderSetupFlags(flags *pflag.FlagSet, opts *renderOptions) { viper.BindPFlags(flags) } -func newRenderCommand() *cobra.Command { - opts := &renderOptions{} +func newRunCommand() *cobra.Command { + opts := &runOptions{} cmd := &cobra.Command{ - Use: "render", + Use: "run", RunE: func(cmd *cobra.Command, args []string) error { if err := viper.Unmarshal(opts); err != nil { return err } - if err := viper.Unmarshal(&opts.FetchOpts); err != nil { + if err := viper.Unmarshal(&opts.PullOpts); err != nil { return err } if err := viper.Unmarshal(&opts.DBOpts); err != nil { return err } - opts.FetchOpts.DBOpts = opts.DBOpts - return render(opts) + opts.PullOpts.DBOpts = opts.DBOpts + opts.PullOpts.Targets = args + opts.Targets = args + return run(opts) }, } - renderSetupFlags(cmd.Flags(), opts) - fetchSetupFlags(cmd.Flags(), &opts.FetchOpts) + runSetupFlags(cmd.Flags(), opts) + pullSetupFlags(cmd.Flags(), &opts.PullOpts) dbSetupFlags(cmd.Flags(), &opts.DBOpts) return cmd } -func render(opts *renderOptions) error { - logger().Debug("render", zap.Stringer("opts", *opts)) - if !opts.NoFetch || !dbExists(&opts.DBOpts) { - if err := fetch(&opts.FetchOpts); err != nil { +func run(opts *runOptions) error { + logger().Debug("run", zap.Stringer("opts", *opts)) + if !opts.NoPull || !dbExists(&opts.DBOpts) { + if err := pull(&opts.PullOpts); err != nil { return err } } @@ -88,16 +92,16 @@ func render(opts *renderOptions) error { } var out string - switch opts.RenderType { + switch opts.RunType { case "roadmap": out, err = roadmapGraph(issues, opts) case "orphans": out, err = orphansGraph(issues, opts) default: - err = fmt.Errorf("unknown graph type: %q", opts.RenderType) + err = fmt.Errorf("unknown graph type: %q", opts.RunType) } if err != nil { - return errors.Wrap(err, "failed to render graph") + return errors.Wrap(err, "failed to run graph") } var dest io.WriteCloser diff --git a/examples/depviz/Makefile b/examples/depviz/Makefile index a6617f082..68e11c50c 100644 --- a/examples/depviz/Makefile +++ b/examples/depviz/Makefile @@ -1,9 +1,9 @@ .PHONY: run run: - depviz render -f | dot -Tsvg > depviz.svg - depviz render -t orphans -f | dot -Tsvg > orphans.svg - cat orphans.svg | grep -s text || rm orphans.svg + depviz run | dot -Tsvg > depviz.svg + depviz run -t orphans | dot -Tsvg > orphans.svg + cat orphans.svg | grep -q text || rm orphans.svg .PHONY: preview preview: - depviz render -f | dot -Tpng | imgcat + depviz run | dot -Tpng | imgcat diff --git a/examples/depviz/depviz.svg b/examples/depviz/depviz.svg index 0380d99ba..2ed52f03d 100644 --- a/examples/depviz/depviz.svg +++ b/examples/depviz/depviz.svg @@ -4,549 +4,598 @@ - - + + G - - + + -moul/depviz#8 - - - -moul/depviz#8: -depviz is easy to -use - - - -epic - - +https://github.com/moul/depviz/issues/2 + + + +moul/depviz#2: Setup +CI + + + +enhancement + + - + -moul/depviz#6 - - - -moul/depviz#6: Web -version - - - -enhancement - - +https://github.com/moul/depviz/issues/5 + + + +moul/depviz#5: Add +Docker image + + + +enhancement + + - + -moul/depviz#8->moul/depviz#6 - +https://github.com/moul/depviz/issues/2->https://github.com/moul/depviz/issues/5 + - + -moul/depviz#7 - - - -moul/depviz#7: -Refactor CLI to be -easier to run -projects without -configuration files - - - -enhancement - - -p0 - - +https://github.com/moul/depviz/issues/36 + + + +moul/depviz#36: +DepViz has "cool" +features + + + +epic + + - - -moul/depviz#8->moul/depviz#7 - - - + -moul/depviz#24 - - - -moul/depviz#24: Add -a issue-focus mode - +https://github.com/moul/depviz/issues/37 + + + +moul/depviz#37: Add +an option to update +label(s) based on +rules + + + +enhancement + + - - -moul/depviz#8->moul/depviz#24 - + + +https://github.com/moul/depviz/issues/36->https://github.com/moul/depviz/issues/37 + - + -moul/depviz#5 - - - -moul/depviz#5: Add -Docker image - - - -enhancement - - +https://github.com/moul/depviz/issues/34 + + + +moul/depviz#34: Add +a warn when two +issues "fixes" the +same one + + + +enhancement + + - - -moul/depviz#8->moul/depviz#5 - + + +https://github.com/moul/depviz/issues/36->https://github.com/moul/depviz/issues/34 + - + -moul/depviz#25 - - - -moul/depviz#25: add -clickable links in -generated graph - +https://github.com/moul/depviz/issues/35 + + + +moul/depviz#35: +Support PRs + + + +enhancement + + - - -moul/depviz#8->moul/depviz#25 - + + +https://github.com/moul/depviz/issues/36->https://github.com/moul/depviz/issues/35 + - + -moul/depviz#12 - - - -moul/depviz#12: I -can use depviz to -manage my company - - - -epic - - +https://github.com/moul/depviz/issues/13 + + + +moul/depviz#13: I +can use depviz to +plan all my projects + + + +epic + + - - -moul/depviz#12->moul/depviz#8 - - - + -moul/depviz#21 - - - -moul/depviz#21: Show -epic's completion -rate - - - -enhancement - - +https://github.com/moul/depviz/issues/12 + + + +moul/depviz#12: I +can use depviz to +manage my company + + + +epic + + - - -moul/depviz#12->moul/depviz#21 - + + +https://github.com/moul/depviz/issues/13->https://github.com/moul/depviz/issues/12 + - + -moul/depviz#32 - - - -moul/depviz#32: -Support GitLab - +https://github.com/moul/depviz/issues/32 + + + +moul/depviz#32: +Support GitLab + + + +enhancement + + - - -moul/depviz#12->moul/depviz#32 - + + +https://github.com/moul/depviz/issues/12->https://github.com/moul/depviz/issues/32 + - + -moul/depviz#10 - - - -moul/depviz#10: -depviz is easy to -discover - - - -epic - - +https://github.com/moul/depviz/issues/21 + + + +moul/depviz#21: Show +epic's completion +rate + + + +enhancement + + - + + +https://github.com/moul/depviz/issues/12->https://github.com/moul/depviz/issues/21 + + + -moul/depviz#3 - - - -moul/depviz#3: Add -examples - +https://github.com/moul/depviz/issues/8 + + + +moul/depviz#8: +depviz is easy to +use + + + +epic + + - - -moul/depviz#10->moul/depviz#3 - + + +https://github.com/moul/depviz/issues/12->https://github.com/moul/depviz/issues/8 + - - -moul/depviz#18 - - - -moul/depviz#18: Add -badges in README - - - -enhancement - - + + +https://github.com/moul/depviz/issues/8->https://github.com/moul/depviz/issues/5 + + + + +https://github.com/moul/depviz/issues/25 + + + +moul/depviz#25: add +clickable links in +generated graph + - - -moul/depviz#10->moul/depviz#18 - + + +https://github.com/moul/depviz/issues/8->https://github.com/moul/depviz/issues/25 + - - -moul/depviz#19 - - - -moul/depviz#19: Add -a logo in the README - - - -enhancement - - + + +https://github.com/moul/depviz/issues/24 + + + +moul/depviz#24: Add +an issue-focus mode + + + +enhancement + + - - -moul/depviz#10->moul/depviz#19 - + + +https://github.com/moul/depviz/issues/8->https://github.com/moul/depviz/issues/24 + - - -moul/depviz#2 - - - -moul/depviz#2: Setup -CI - - - -enhancement - - + + +https://github.com/moul/depviz/issues/7 + + + +moul/depviz#7: +Refactor CLI to be +easier to run +projects without +configuration files + + + +enhancement + + +p0 + + - - -moul/depviz#18->moul/depviz#2 - + + +https://github.com/moul/depviz/issues/8->https://github.com/moul/depviz/issues/7 + - - -moul/depviz#36 - - - -moul/depviz#36: -DepViz has "cool" -features - - - -epic - - + + +https://github.com/moul/depviz/issues/6 + + + +moul/depviz#6: Web +version + + + +enhancement + + - - -moul/depviz#34 - - - -moul/depviz#34: Add -a warn when two -issues "fixes" the -same one - - + + +https://github.com/moul/depviz/issues/8->https://github.com/moul/depviz/issues/6 + + + +https://github.com/moul/depviz/issues/10 + + + +moul/depviz#10: +depviz is easy to +discover + + + +epic + + + - - -moul/depviz#36->moul/depviz#34 - - - -moul/depviz#35 - - - -moul/depviz#35: -Support PRs - + + +https://github.com/moul/depviz/issues/19 + + + +moul/depviz#19: Add +a logo in the README + + + +enhancement + + - - -moul/depviz#36->moul/depviz#35 - + + +https://github.com/moul/depviz/issues/10->https://github.com/moul/depviz/issues/19 + - - -moul/depviz#13 - - - -moul/depviz#13: I -can use depviz to -plan all my projects - - - -epic - - + + +https://github.com/moul/depviz/issues/3 + + + +moul/depviz#3: Add +examples + - - -moul/depviz#13->moul/depviz#12 - + + +https://github.com/moul/depviz/issues/10->https://github.com/moul/depviz/issues/3 + - - -moul/depviz#11 - - - -moul/depviz#11: -depviz is pluggable -with other workflows -/ systems - - - -epic - - + + +https://github.com/moul/depviz/issues/18 + + + +moul/depviz#18: Add +badges in README + + + +enhancement + + - - -moul/depviz#11->moul/depviz#5 - + + +https://github.com/moul/depviz/issues/10->https://github.com/moul/depviz/issues/18 + - - -moul/depviz#4 - - - -moul/depviz#4: -Export to OPML - - - -enhancement - - + + +https://github.com/moul/depviz/issues/18->https://github.com/moul/depviz/issues/2 + + + + +https://github.com/moul/depviz/issues/9 + + + +moul/depviz#9: +depviz is easy to +hack + + + +epic + + - - -moul/depviz#11->moul/depviz#4 - + + +https://github.com/moul/depviz/issues/9->https://github.com/moul/depviz/issues/2 + - - -moul/depviz#2->moul/depviz#5 - + + +https://github.com/moul/depviz/issues/9->https://github.com/moul/depviz/issues/18 + - - -moul/depviz#9 - - - -moul/depviz#9: -depviz is easy to -hack - - - -epic - - + + +https://github.com/moul/depviz/issues/33 + + + +moul/depviz#33: Use +a database to store +issues in a smarter +way + + + +enhancement + + - - -moul/depviz#9->moul/depviz#18 - - - - -moul/depviz#9->moul/depviz#2 - + + +https://github.com/moul/depviz/issues/9->https://github.com/moul/depviz/issues/33 + - - -moul/depviz#33 - - - -moul/depviz#33: Use -a database to store -issues in a smarter -way - + + +https://github.com/moul/depviz/issues/22 + + + +moul/depviz#22: Add +a license + - - -moul/depviz#9->moul/depviz#33 - + + +https://github.com/moul/depviz/issues/9->https://github.com/moul/depviz/issues/22 + - + -moul/depviz#22 - - - -moul/depviz#22: Add -a license - +https://github.com/moul/depviz/issues/11 + + + +moul/depviz#11: +depviz is pluggable +with other workflows +/ systems + + + +epic + + + + + + + +https://github.com/moul/depviz/issues/11->https://github.com/moul/depviz/issues/5 + + + + +https://github.com/moul/depviz/issues/4 + + + +moul/depviz#4: +Export to OPML + + + +enhancement + + - + -moul/depviz#9->moul/depviz#22 - +https://github.com/moul/depviz/issues/11->https://github.com/moul/depviz/issues/4 + - + placeholder_1 -weight=1 +weight=1 - + placeholder_2 -weight=2 +weight=2 - + placeholder_3 -weight=3 +weight=3 - + placeholder_4 -weight=4 +weight=4 - + placeholder_5 -weight=5 +weight=5 - + placeholder_6 -weight=6 +weight=6 - + placeholder_7 -weight=7 +weight=7 - + placeholder_8 -weight=8 +weight=8 - + placeholder_14 -weight=14 +weight=14 - + placeholder_22 -weight=22 +weight=22 diff --git a/examples/depviz/orphans.svg b/examples/depviz/orphans.svg new file mode 100644 index 000000000..9cabdbbbe --- /dev/null +++ b/examples/depviz/orphans.svg @@ -0,0 +1,102 @@ + + + + + + +G + + +cluster_mouldepviz + +/moul/depviz + + + +https://github.com/moul/depviz/issues/39 + + + +moul/depviz#39: make +--fetch the default +behavior + + + + + + +https://github.com/moul/depviz/issues/40 + + + +moul/depviz#40: +automatically detect +the project by +parsing `.git` + + + + + + +https://github.com/moul/depviz/issues/41 + + + +moul/depviz#41: +Support +Cross-Provider +dependencies (GitHub +-> GitLab) + + + + + + +https://github.com/moul/depviz/issues/43 + + + +moul/depviz#43: Add +an option that +automatically +fetches external +repos + + + + + + +https://github.com/moul/depviz/issues/44 + + + +moul/depviz#44: +Support netrc for +authentication + + + + + + +https://github.com/moul/depviz/issues/45 + + + +moul/depviz#45: Use +a caching backend +for client http +requests + + + + + + diff --git a/graphviz.go b/graphviz.go index 58402c3d7..1dc950603 100644 --- a/graphviz.go +++ b/graphviz.go @@ -7,7 +7,7 @@ import ( "github.com/awalterschulze/gographviz" ) -func orphansGraph(issues Issues, opts *renderOptions) (string, error) { +func orphansGraph(issues Issues, opts *runOptions) (string, error) { if !opts.ShowClosed { issues.HideClosed() } @@ -60,7 +60,7 @@ func orphansGraph(issues Issues, opts *renderOptions) (string, error) { return g.String(), nil } -func roadmapGraph(issues Issues, opts *renderOptions) (string, error) { +func roadmapGraph(issues Issues, opts *runOptions) (string, error) { if !opts.ShowClosed { issues.HideClosed() } diff --git a/main.go b/main.go index f5d3237ec..efe49326e 100644 --- a/main.go +++ b/main.go @@ -64,8 +64,8 @@ func newRootCommand() *cobra.Command { return nil } cmd.AddCommand( - newFetchCommand(), - newRenderCommand(), + newPullCommand(), + newRunCommand(), newDBCommand(), ) viper.AutomaticEnv()