Skip to content

Commit 4eb1ee9

Browse files
committed
intial impl of jira sync - need to make it more configurable
1 parent 7d41c15 commit 4eb1ee9

File tree

277 files changed

+36592
-223
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

277 files changed

+36592
-223
lines changed

cli/cmds.go

+11-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ package cli
33
import (
44
"fmt"
55

6-
"github.com/katbyte/ghp-repo-sync/version" // todo - should we rename this (again) to ghp-sync ? if it can do project <> project
6+
"github.com/katbyte/ghp-repo-sync/version" // todo - should we rename this (again) to ghp-sync ? if it can do project <> project & jira <> gh TODO yes we should
77
"github.com/spf13/cobra"
88
"github.com/spf13/viper"
99
)
@@ -71,7 +71,16 @@ func Make(cmdName string) (*cobra.Command, error) {
7171
RunE: CmdSync,
7272
})
7373

74-
// TODO add CLEAR command to reset a project?
74+
root.AddCommand(&cobra.Command{
75+
Use: "jira",
76+
Short: "sync from jira to gh project",
77+
Args: cobra.NoArgs,
78+
SilenceErrors: true,
79+
PreRunE: ValidateParams([]string{"token", "project-owner", "project-number", "jira-url", "jira-user", "jira-token", "jira-jql"}),
80+
RunE: CmdJIRA,
81+
})
82+
83+
// TODO add CLEAR command to reset a project? other commands to cleanup project?
7584

7685
if err := configureFlags(root); err != nil {
7786
return nil, fmt.Errorf("unable to configure flags: %w", err)

cli/cmds_issues.go

+1
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ func CmdIssues(_ *cobra.Command, _ []string) error {
110110
}
111111
c.Printf("<magenta>%s</>", *iid)
112112

113+
// TODO switch to gh.UpdateItem
113114
q := `query=
114115
mutation (
115116
$project:ID!, $item:ID!,

cli/cmds_jira.go

+165
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,165 @@
1+
package cli
2+
3+
import (
4+
"fmt"
5+
"github.com/ctreminiom/go-atlassian/v2/pkg/infra/models"
6+
"github.com/katbyte/ghp-repo-sync/lib/gh"
7+
"github.com/katbyte/ghp-repo-sync/lib/j"
8+
"github.com/spf13/cobra"
9+
"strings"
10+
"time"
11+
12+
//nolint:misspell
13+
c "github.com/gookit/color"
14+
)
15+
16+
func CmdJIRA(_ *cobra.Command, _ []string) error {
17+
f := GetFlags()
18+
19+
p := gh.NewProject(f.ProjectOwner, f.ProjectNumber, f.Token)
20+
jira := j.NewInstance(f.Jira.Url, f.Jira.User, f.Jira.Token)
21+
22+
ghc, ctx := p.NewClient()
23+
24+
c.Printf("Looking up project details for <green>%s</>/<lightGreen>%d</>...\n", f.ProjectOwner, f.ProjectNumber)
25+
err := p.LoadDetails()
26+
if err != nil {
27+
c.Printf("\n\n <red>ERROR!!</> %s", err)
28+
return nil
29+
}
30+
c.Printf(" ID: <magenta>%s</>\n", p.ID)
31+
32+
// todo we can probably remove this? its just for printing the fields (we do this multiple times so maybe just a helper)
33+
for _, f := range p.Fields {
34+
c.Printf(" <lightBlue>%s</> <> <lightCyan>%s</>\n", f.Name, f.ID)
35+
36+
if f.Name == "Status" {
37+
for _, s := range f.Options {
38+
c.Printf(" <blue>%s</> <> <cyan>%s</>\n", s.Name, s.ID)
39+
}
40+
}
41+
}
42+
fmt.Println()
43+
44+
c.Printf("Retrieving all issues matching <white>%s</> from <cyan>%s</>...\n", f.Jira.JQL, f.Jira.Url)
45+
c.Printf(" Fields %s\n", strings.Join(f.Jira.Fields, ", "))
46+
c.Printf(" Expand %s\n", strings.Join(f.Jira.Expand, ", "))
47+
48+
//custom fields, this needs to be configurable but how? TODO
49+
// customfield_10089 -> type -> project_field ?
50+
cfIssueLink := "customfield_10089"
51+
cfSE := "customfield_10582"
52+
cfACV := "customfield_10134"
53+
54+
// temp fields to get the jira issues
55+
fields := []string{"status", "summary", "created", "parent", "IssueLinks", "Issue Link", "Solution Engineer", cfIssueLink, cfSE, cfACV}
56+
expand := []string{}
57+
58+
n := 0
59+
err = jira.ListAllIssues(f.Jira.JQL, &fields, &expand, func(results *models.IssueSearchScheme, resp *models.ResponseScheme) error {
60+
c.Printf("<magenta>%d</>-<lightMagenta>%d</> <darkGray>of %d</>\n", results.StartAt, results.MaxResults, results.Total)
61+
62+
//custom fields need to be loaded from the response
63+
issueLinks, err := models.ParseStringCustomFields(resp.Bytes, cfIssueLink)
64+
if err != nil {
65+
return fmt.Errorf("failed to parse issue links: %w", err)
66+
}
67+
68+
solutionEngineers, err := models.ParseUserPickerCustomFields(resp.Bytes, cfSE)
69+
if err != nil {
70+
return fmt.Errorf("failed to parse solution engineers: %w", err)
71+
}
72+
73+
acvs, err := models.ParseFloatCustomFields(resp.Bytes, cfACV)
74+
if err != nil {
75+
return fmt.Errorf("failed to parse acvs: %w", err)
76+
}
77+
78+
for _, jiraIssue := range results.Issues {
79+
n++
80+
81+
keyColour := "lightGreen"
82+
if jiraIssue.Fields.Status.Name == "Closed" {
83+
keyColour = "green"
84+
}
85+
86+
// lets collect everything we need
87+
key := jiraIssue.Key
88+
url := fmt.Sprintf("%s/browse/%s", jira.URL, key)
89+
summary := jiraIssue.Fields.Summary
90+
status := jiraIssue.Fields.Status.Name
91+
ghLink := issueLinks[jiraIssue.Key]
92+
created := time.Time(*jiraIssue.Fields.Created)
93+
94+
parent := ""
95+
if jiraIssue.Fields.Parent != nil {
96+
parent = jiraIssue.Fields.Parent.Fields.Summary
97+
}
98+
99+
se := ""
100+
if solutionEngineers[jiraIssue.Key] != nil {
101+
se = solutionEngineers[jiraIssue.Key].DisplayName
102+
}
103+
104+
//c.Printf("<darkGray>%03d/%d</> <%s>%s</><darkGray>@%s</> - %s\n", n, results.Total, keyColour, jiraIssue.Key, parsedDate.Format("2006-01-02"), jiraIssue.Fields.Summary)
105+
c.Printf("<darkGray>%03d/%d</> <%s>%s</><darkGray>@%v</> - %s\n", n, results.Total, keyColour, key, created, summary)
106+
107+
owner, name, _, number, err := gh.ParseGitHubURL(ghLink)
108+
if err != nil {
109+
c.Printf("\n\n <red>ERROR!!</> parsing gh url %s: %s", ghLink, err)
110+
return nil
111+
}
112+
113+
ghIssue, _, err := ghc.Issues.Get(ctx, owner, name, number)
114+
if err != nil {
115+
c.Printf("\n\n <red>ERROR!!</> %s", err)
116+
return nil
117+
}
118+
119+
// TODO output all the information we got and are adding to the project in a nice manner
120+
fmt.Println(" Status: ", status)
121+
fmt.Println(" Parent: ", parent)
122+
fmt.Println(" Issue Links: ", ghLink)
123+
fmt.Println(" Solution Engineers: ", se)
124+
125+
iid, err := p.AddItem(*ghIssue.NodeID)
126+
if err != nil {
127+
c.Printf("\n\n <red>ERROR!!</> %s", err)
128+
continue
129+
}
130+
131+
c.Printf("\n")
132+
133+
fields := []gh.ProjectItemField{
134+
{Name: "key", FieldID: p.FieldIDs["KEY"], Type: gh.ItemValueTypeText, Value: key},
135+
{Name: "url", FieldID: p.FieldIDs["JIRA"], Type: gh.ItemValueTypeText, Value: url},
136+
{Name: "summary", FieldID: p.FieldIDs["Title (JIRA)"], Type: gh.ItemValueTypeText, Value: summary},
137+
{Name: "status", FieldID: p.FieldIDs["Status (JIRA)"], Type: gh.ItemValueTypeText, Value: status},
138+
{Name: "parent", FieldID: p.FieldIDs["EPIC"], Type: gh.ItemValueTypeText, Value: parent},
139+
{Name: "se", FieldID: p.FieldIDs["SE"], Type: gh.ItemValueTypeText, Value: se},
140+
{Name: "age", FieldID: p.FieldIDs["Age (days)"], Type: gh.ItemValueTypeNumber, Value: int(time.Since(created).Hours() / 24)},
141+
{Name: "number", FieldID: p.FieldIDs["#"], Type: gh.ItemValueTypeNumber, Value: *ghIssue.Number},
142+
}
143+
144+
if v, ok := acvs[jiraIssue.Key]; ok {
145+
fields = append(fields, gh.ProjectItemField{Name: "acv", FieldID: p.FieldIDs["ACV"], Type: gh.ItemValueTypeNumber, Value: int(v)})
146+
}
147+
148+
err = p.UpdateItem(*iid, fields)
149+
if err != nil {
150+
c.Printf("\n\n <red>ERROR!!</> %s", err)
151+
continue
152+
}
153+
154+
c.Printf("\n")
155+
}
156+
157+
return nil
158+
})
159+
160+
if err != nil {
161+
return fmt.Errorf("failed to list issues for %s @ %s: %w", jira.URL, f.Jira.JQL, err)
162+
}
163+
164+
return nil
165+
}

cli/cmds_project.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -112,7 +112,7 @@ func CmdSync(_ *cobra.Command, args []string) error {
112112
c.Printf("<blue>updating</>...")
113113

114114
//update status to "Unclaimed PR" & update request type to
115-
// TODO we can loop through the fields and build a more dynamic query from a function p.UpdateItemFields()
115+
// TODO switch to gh.UpdateItem
116116
q := `query=
117117
mutation (
118118
$project:ID!, $item:ID!,

cli/cmds_prs.go

+1
Original file line numberDiff line numberDiff line change
@@ -158,6 +158,7 @@ func CmdPRs(_ *cobra.Command, _ []string) error {
158158

159159
c.Printf(" open %d days, waiting %d days\n", daysOpen, daysWaiting)
160160

161+
// TODO switch to gh.UpdateItem
161162
q := `query=
162163
mutation (
163164
$project:ID!, $item:ID!,

cli/flags.go

+48-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ type FlagData struct {
1616
IncludeClosed bool
1717
DryRun bool
1818
Filters Filters
19+
Jira Jira
1920
}
2021

2122
type Filters struct {
@@ -25,6 +26,15 @@ type Filters struct {
2526
LabelsAnd []string
2627
}
2728

29+
type Jira struct {
30+
Url string
31+
User string
32+
Token string
33+
JQL string
34+
Fields []string
35+
Expand []string
36+
}
37+
2838
func configureFlags(root *cobra.Command) error {
2939
flags := FlagData{}
3040
pflags := root.PersistentFlags()
@@ -34,10 +44,19 @@ func configureFlags(root *cobra.Command) error {
3444
pflags.StringVarP(&flags.ProjectOwner, "project-owner", "o", "", "github project owner (GITHUB_PROJECT_OWNER)")
3545
pflags.IntVarP(&flags.ProjectNumber, "project-number", "p", 0, "github project number (GITHUB_PROJECT_NUMBER)")
3646
pflags.BoolVarP(&flags.IncludeClosed, "include-closed", "c", false, "include closed prs/issues")
47+
48+
pflags.StringVarP(&flags.Jira.Url, "jira-url", "", "", "jira instance url")
49+
pflags.StringVarP(&flags.Jira.User, "jira-user", "", "", "jira user")
50+
pflags.StringVarP(&flags.Jira.Token, "jira-token", "", "", "jira oauth token (JIRA_TOKEN)")
51+
pflags.StringVarP(&flags.Jira.JQL, "jira-jql", "", "", "jira jql query to list all issues")
52+
pflags.StringSliceVarP(&flags.Jira.Fields, "jira-fields", "", nil, "jira fields to fetch seperated by commas")
53+
pflags.StringSliceVarP(&flags.Jira.Expand, "jira-expand", "", nil, "jira fields to expand seperated by commas")
54+
3755
pflags.StringSliceVarP(&flags.Filters.Authors, "authors", "a", []string{}, "only sync prs by these authors. ie 'katbyte,author2,author3'")
3856
pflags.StringSliceVarP(&flags.Filters.Assignees, "assignees", "", []string{}, "sync prs assigned to these users. ie 'katbyte,assignee2,assignee3'")
3957
pflags.StringSliceVarP(&flags.Filters.LabelsOr, "labels-or", "l", []string{}, "filter that match any label conditions. ie 'label1,label2,-not-this-label'")
4058
pflags.StringSliceVarP(&flags.Filters.LabelsAnd, "labels-and", "", []string{}, "filter that match all label conditions. ie 'label1,label2,-not-this-label'")
59+
4160
pflags.BoolVarP(&flags.DryRun, "dry-run", "d", false, "dry run, don't actually add issues/prs to project")
4261

4362
// binding map for viper/pflag -> env
@@ -47,6 +66,12 @@ func configureFlags(root *cobra.Command) error {
4766
"project-owner": "GITHUB_PROJECT_OWNER",
4867
"project-number": "GITHUB_PROJECT_NUMBER",
4968
"include-closed": "GITHUB_INCLUDE_CLOSED",
69+
"jira-url": "JIRA_URL",
70+
"jira-user": "JIRA_USER",
71+
"jira-jql": "JIRA_JQL",
72+
"jira-token": "JIRA_TOKEN",
73+
"jira-fields": "JIRA_FIELDS",
74+
"jira-expand": "JIRA_EXPAND",
5075
"authors": "GITHUB_AUTHORS",
5176
"assignees": "GITHUB_ASSIGNEES",
5277
"labels-or": "GITHUB_LABELS_OR",
@@ -72,6 +97,17 @@ func configureFlags(root *cobra.Command) error {
7297
func GetFlags() FlagData {
7398

7499
// TODO BUG for some reason it is not correctly splitting on ,? so hack this in
100+
101+
jiraFields := viper.GetStringSlice("jira-fields")
102+
if len(jiraFields) > 0 {
103+
jiraFields = strings.Split(jiraFields[0], ",")
104+
}
105+
106+
jiraExpand := viper.GetStringSlice("jira-expand")
107+
if len(jiraExpand) > 0 {
108+
jiraExpand = strings.Split(jiraExpand[0], ",")
109+
}
110+
75111
authors := viper.GetStringSlice("authors")
76112
if len(authors) > 0 {
77113
authors = strings.Split(authors[0], ",")
@@ -92,7 +128,18 @@ func GetFlags() FlagData {
92128
ProjectNumber: viper.GetInt("project-number"),
93129
ProjectOwner: viper.GetString("project-owner"),
94130
IncludeClosed: viper.GetBool("include-closed"),
95-
DryRun: viper.GetBool("dry-run"),
131+
132+
DryRun: viper.GetBool("dry-run"),
133+
134+
Jira: Jira{
135+
Url: viper.GetString("jira-url"),
136+
User: viper.GetString("jira-user"),
137+
Token: viper.GetString("jira-token"),
138+
JQL: viper.GetString("jira-jql"),
139+
Fields: jiraFields,
140+
Expand: jiraExpand,
141+
},
142+
96143
Filters: Filters{
97144
Authors: authors,
98145
Assignees: assignees,

ghp-pr-sync

-11.2 MB
Binary file not shown.

go.mod

+6-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,9 @@
11
module github.com/katbyte/ghp-repo-sync
22

3-
go 1.18
3+
go 1.23
44

55
require (
6+
github.com/ctreminiom/go-atlassian/v2 v2.1.2
67
github.com/google/go-github/v45 v45.2.0
78
github.com/gookit/color v1.5.0
89
github.com/hashicorp/go-retryablehttp v0.7.4
@@ -13,6 +14,7 @@ require (
1314
)
1415

1516
require (
17+
dario.cat/mergo v1.0.1 // indirect
1618
github.com/fsnotify/fsnotify v1.5.1 // indirect
1719
github.com/golang/protobuf v1.5.2 // indirect
1820
github.com/google/go-querystring v1.1.0 // indirect
@@ -27,6 +29,9 @@ require (
2729
github.com/spf13/jwalterweatherman v1.1.0 // indirect
2830
github.com/spf13/pflag v1.0.5 // indirect
2931
github.com/subosito/gotenv v1.2.0 // indirect
32+
github.com/tidwall/gjson v1.18.0 // indirect
33+
github.com/tidwall/match v1.1.1 // indirect
34+
github.com/tidwall/pretty v1.2.0 // indirect
3035
github.com/xo/terminfo v0.0.0-20210125001918-ca9a967f8778 // indirect
3136
golang.org/x/crypto v0.0.0-20210817164053-32db794688a5 // indirect
3237
golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d // indirect

0 commit comments

Comments
 (0)