mirror of
https://github.com/Threnklyn/jira.git
synced 2026-05-18 12:13:29 +02:00
a26683e01d
https://developer.atlassian.com/cloud/jira/platform/deprecation-notice-user-privacy-api-migration-guide/
263 lines
8.2 KiB
Go
263 lines
8.2 KiB
Go
package jiracmd
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/coryb/figtree"
|
|
"github.com/coryb/oreo"
|
|
"github.com/go-jira/jira"
|
|
"github.com/go-jira/jira/jiracli"
|
|
"github.com/go-jira/jira/jiradata"
|
|
"gopkg.in/AlecAivazis/survey.v1"
|
|
kingpin "gopkg.in/alecthomas/kingpin.v2"
|
|
)
|
|
|
|
type EditOptions struct {
|
|
jiracli.CommonOptions `yaml:",inline" json:",inline" figtree:",inline"`
|
|
jiradata.IssueUpdate `yaml:",inline" json:",inline" figtree:",inline"`
|
|
jira.SearchOptions `yaml:",inline" json:",inline" figtree:",inline"`
|
|
Overrides map[string]string `yaml:"overrides,omitempty" json:"overrides,omitempty"`
|
|
Issue string `yaml:"issue,omitempty" json:"issue,omitempty"`
|
|
Queries map[string]string `yaml:"queries,omitempty" json:"queries,omitempty"`
|
|
}
|
|
|
|
func CmdEditRegistry() *jiracli.CommandRegistryEntry {
|
|
opts := EditOptions{
|
|
CommonOptions: jiracli.CommonOptions{
|
|
Template: figtree.NewStringOption("edit"),
|
|
},
|
|
Overrides: map[string]string{},
|
|
}
|
|
|
|
return &jiracli.CommandRegistryEntry{
|
|
"Edit issue details",
|
|
func(fig *figtree.FigTree, cmd *kingpin.CmdClause) error {
|
|
jiracli.LoadConfigs(cmd, fig, &opts)
|
|
return CmdEditUsage(cmd, &opts, fig)
|
|
},
|
|
func(o *oreo.Client, globals *jiracli.GlobalOptions) error {
|
|
opts.Issue = jiracli.FormatIssue(opts.Issue, opts.Project)
|
|
if opts.QueryFields == "" {
|
|
opts.QueryFields = "assignee,created,priority,reporter,status,summary,updated,issuetype,comment,description,votes,created,customfield_10110,components"
|
|
}
|
|
return CmdEdit(o, globals, &opts)
|
|
},
|
|
}
|
|
}
|
|
|
|
func CmdEditUsage(cmd *kingpin.CmdClause, opts *EditOptions, fig *figtree.FigTree) error {
|
|
jiracli.BrowseUsage(cmd, &opts.CommonOptions)
|
|
jiracli.EditorUsage(cmd, &opts.CommonOptions)
|
|
jiracli.TemplateUsage(cmd, &opts.CommonOptions)
|
|
cmd.Flag("noedit", "Disable opening the editor").SetValue(&opts.SkipEditing)
|
|
cmd.Flag("named-query", "The name of a query in the `queries` configuration").Short('n').PreAction(func(ctx *kingpin.ParseContext) error {
|
|
name := jiracli.FlagValue(ctx, "named-query")
|
|
if query, ok := opts.Queries[name]; ok && query != "" {
|
|
var err error
|
|
opts.Query, err = jiracli.ConfigTemplate(fig, query, cmd.FullCommand(), opts)
|
|
return err
|
|
}
|
|
return fmt.Errorf("A valid named-query %q not found in `queries` configuration", name)
|
|
}).String()
|
|
cmd.Flag("query", "Jira Query Language (JQL) expression for the search to edit multiple issues").Short('q').StringVar(&opts.Query)
|
|
cmd.Flag("comment", "Comment message for issue").Short('m').PreAction(func(ctx *kingpin.ParseContext) error {
|
|
opts.Overrides["comment"] = jiracli.FlagValue(ctx, "comment")
|
|
return nil
|
|
}).String()
|
|
cmd.Flag("override", "Set issue property").Short('o').StringMapVar(&opts.Overrides)
|
|
cmd.Arg("ISSUE", "issue id to edit").StringVar(&opts.Issue)
|
|
return nil
|
|
}
|
|
|
|
// Edit will get issue data and send to "edit" template
|
|
func CmdEdit(o *oreo.Client, globals *jiracli.GlobalOptions, opts *EditOptions) error {
|
|
if globals.JiraDeploymentType.Value == "" {
|
|
serverInfo, err := jira.ServerInfo(o, globals.Endpoint.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
globals.JiraDeploymentType.Value = strings.ToLower(serverInfo.DeploymentType)
|
|
}
|
|
|
|
type templateInput struct {
|
|
*jiradata.Issue `yaml:",inline"`
|
|
Meta *jiradata.EditMeta `yaml:"meta" json:"meta"`
|
|
Overrides map[string]string `yaml:"overrides" json:"overrides"`
|
|
}
|
|
if opts.Issue != "" {
|
|
issueData, err := jira.GetIssue(o, globals.Endpoint.Value, opts.Issue, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
editMeta, err := jira.GetIssueEditMeta(o, globals.Endpoint.Value, opts.Issue)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
issueUpdate := jiradata.IssueUpdate{}
|
|
input := templateInput{
|
|
Issue: issueData,
|
|
Meta: editMeta,
|
|
Overrides: opts.Overrides,
|
|
}
|
|
err = jiracli.EditLoop(&opts.CommonOptions, &input, &issueUpdate, func() error {
|
|
if globals.JiraDeploymentType.Value == jiracli.CloudDeploymentType {
|
|
err := fixGDPRUserFields(o, globals.Endpoint.Value, editMeta.Fields, issueUpdate.Fields)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return jira.EditIssue(o, globals.Endpoint.Value, opts.Issue, &issueUpdate)
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !globals.Quiet.Value {
|
|
fmt.Printf("OK %s %s\n", opts.Issue, jira.URLJoin(globals.Endpoint.Value, "browse", opts.Issue))
|
|
}
|
|
if opts.Browse.Value {
|
|
return CmdBrowse(globals, opts.Issue)
|
|
}
|
|
return nil
|
|
}
|
|
results, err := jira.Search(o, globals.Endpoint.Value, opts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for i, issueData := range results.Issues {
|
|
editMeta, err := jira.GetIssueEditMeta(o, globals.Endpoint.Value, issueData.Key)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
issueUpdate := jiradata.IssueUpdate{}
|
|
input := templateInput{
|
|
Issue: issueData,
|
|
Meta: editMeta,
|
|
Overrides: opts.Overrides,
|
|
}
|
|
err = jiracli.EditLoop(&opts.CommonOptions, &input, &issueUpdate, func() error {
|
|
if globals.JiraDeploymentType.Value == jiracli.CloudDeploymentType {
|
|
err := fixGDPRUserFields(o, globals.Endpoint.Value, editMeta.Fields, issueUpdate.Fields)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return jira.EditIssue(o, globals.Endpoint.Value, issueData.Key, &issueUpdate)
|
|
})
|
|
if err == jiracli.EditLoopAbort && len(results.Issues) > i+1 {
|
|
var answer bool
|
|
survey.AskOne(
|
|
&survey.Confirm{
|
|
Message: fmt.Sprintf("Continue to edit next issue %s?", results.Issues[i+1].Key),
|
|
Default: true,
|
|
},
|
|
&answer,
|
|
nil,
|
|
)
|
|
if answer {
|
|
continue
|
|
}
|
|
panic(jiracli.Exit{1})
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !globals.Quiet.Value {
|
|
fmt.Printf("OK %s %s\n", issueData.Key, jira.URLJoin(globals.Endpoint.Value, "browse", issueData.Key))
|
|
}
|
|
if opts.Browse.Value {
|
|
return CmdBrowse(globals, issueData.Key)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func fixUserField(ua jira.HttpClient, endpoint string, userField map[string]interface{}) error {
|
|
if _, ok := userField["accountId"].(string); ok {
|
|
// this field is already GDPR ready
|
|
return nil
|
|
}
|
|
|
|
if username, ok := userField["name"].(string); ok {
|
|
users, err := jira.UserSearch(ua, endpoint, &jira.UserSearchOptions{
|
|
Username: username,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(users) != 1 {
|
|
return fmt.Errorf("Found %d accounts for username %q", len(users), username)
|
|
}
|
|
userField["accountId"] = users[0].AccountID
|
|
return nil
|
|
}
|
|
|
|
queryName, ok := userField["displayName"].(string)
|
|
if !ok {
|
|
queryName, ok = userField["emailAddress"].(string)
|
|
if !ok {
|
|
// no fields to search on, skip user lookup
|
|
return nil
|
|
}
|
|
}
|
|
users, err := jira.UserSearch(ua, endpoint, &jira.UserSearchOptions{
|
|
// Query field will search users displayName and emailAddress
|
|
Query: queryName,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(users) != 1 {
|
|
return fmt.Errorf("Found %d accounts for users with query %q", len(users), queryName)
|
|
}
|
|
userField["accountId"] = users[0].AccountID
|
|
return nil
|
|
}
|
|
|
|
func fixGDPRUserFields(ua jira.HttpClient, endpoint string, meta jiradata.FieldMetaMap, fields map[string]interface{}) error {
|
|
for fieldName, fieldMeta := range meta {
|
|
// check to see if meta-field is in fields data, otherwise skip
|
|
if _, ok := fields[fieldName]; !ok {
|
|
continue
|
|
}
|
|
if fieldMeta.Schema.Type == "user" {
|
|
userField, ok := fields[fieldName].(map[string]interface{})
|
|
if !ok {
|
|
// for some reason the field seems to be the wrong type in the data
|
|
// even though the schema is a "user"
|
|
continue
|
|
}
|
|
err := fixUserField(ua, endpoint, userField)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fields[fieldName] = userField
|
|
}
|
|
if fieldMeta.Schema.Type == "array" && fieldMeta.Schema.Items == "user" {
|
|
listUserField, ok := fields[fieldName].([]interface{})
|
|
if !ok {
|
|
// for some reason the field seems to be the wrong type in the data
|
|
// even though the schema is a list of "user"
|
|
continue
|
|
}
|
|
for i, userFieldItem := range listUserField {
|
|
userField, ok := userFieldItem.(map[string]interface{})
|
|
if !ok {
|
|
// for some reason the field seems to be the wrong type in the data
|
|
// even though the schema is a "user"
|
|
continue
|
|
}
|
|
err := fixUserField(ua, endpoint, userField)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
listUserField[i] = userField
|
|
}
|
|
fields[fieldName] = listUserField
|
|
}
|
|
}
|
|
return nil
|
|
}
|