2015-09-03 12:25:21 +02:00
package jira
import (
2016-05-19 23:11:21 +02:00
"bytes"
2016-09-23 16:19:07 +02:00
"encoding/json"
2015-09-03 12:25:21 +02:00
"fmt"
2016-05-19 23:11:21 +02:00
"io"
2016-10-05 14:53:30 +02:00
"io/ioutil"
2016-05-19 23:11:21 +02:00
"mime/multipart"
2016-05-29 17:30:45 +02:00
"net/url"
2016-09-23 16:19:07 +02:00
"reflect"
2016-05-29 17:30:45 +02:00
"strings"
"time"
2016-10-03 13:33:46 +02:00
"github.com/fatih/structs"
2017-01-28 00:04:08 +02:00
"github.com/google/go-querystring/query"
2016-10-03 13:33:46 +02:00
"github.com/trivago/tgo/tcontainer"
2015-09-03 12:25:21 +02:00
)
const (
// AssigneeAutomatic represents the value of the "Assignee: Automatic" of JIRA
AssigneeAutomatic = "-1"
)
// IssueService handles Issues for the JIRA instance / API.
//
2016-03-27 14:24:48 +02:00
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue
2015-09-03 12:25:21 +02:00
type IssueService struct {
client * Client
}
2018-10-17 21:44:13 +02:00
// UpdateQueryOptions specifies the optional parameters to the Edit issue
type UpdateQueryOptions struct {
2018-10-17 22:25:06 +02:00
NotifyUsers bool ` url:"notifyUsers,omitempty" `
OverrideScreenSecurity bool ` url:"overrideScreenSecurity,omitempty" `
OverrideEditableFlag bool ` url:"overrideEditableFlag,omitempty" `
2015-09-03 12:25:21 +02:00
}
// Issue represents a JIRA issue.
type Issue struct {
2018-06-03 23:42:41 +02:00
Expand string ` json:"expand,omitempty" structs:"expand,omitempty" `
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Key string ` json:"key,omitempty" structs:"key,omitempty" `
Fields * IssueFields ` json:"fields,omitempty" structs:"fields,omitempty" `
RenderedFields * IssueRenderedFields ` json:"renderedFields,omitempty" structs:"renderedFields,omitempty" `
Changelog * Changelog ` json:"changelog,omitempty" structs:"changelog,omitempty" `
2017-01-28 00:04:08 +02:00
}
// ChangelogItems reflects one single changelog item of a history item
type ChangelogItems struct {
Field string ` json:"field" structs:"field" `
FieldType string ` json:"fieldtype" structs:"fieldtype" `
From interface { } ` json:"from" structs:"from" `
FromString string ` json:"fromString" structs:"fromString" `
To interface { } ` json:"to" structs:"to" `
ToString string ` json:"toString" structs:"toString" `
}
// ChangelogHistory reflects one single changelog history entry
type ChangelogHistory struct {
Id string ` json:"id" structs:"id" `
Author User ` json:"author" structs:"author" `
Created string ` json:"created" structs:"created" `
Items [ ] ChangelogItems ` json:"items" structs:"items" `
}
// Changelog reflects the change log of an issue
type Changelog struct {
Histories [ ] ChangelogHistory ` json:"histories,omitempty" `
2015-09-03 12:25:21 +02:00
}
2016-05-19 23:11:21 +02:00
// Attachment represents a JIRA attachment
type Attachment struct {
2016-09-23 16:19:07 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Filename string ` json:"filename,omitempty" structs:"filename,omitempty" `
Author * User ` json:"author,omitempty" structs:"author,omitempty" `
Created string ` json:"created,omitempty" structs:"created,omitempty" `
Size int ` json:"size,omitempty" structs:"size,omitempty" `
MimeType string ` json:"mimeType,omitempty" structs:"mimeType,omitempty" `
Content string ` json:"content,omitempty" structs:"content,omitempty" `
Thumbnail string ` json:"thumbnail,omitempty" structs:"thumbnail,omitempty" `
2016-05-19 23:11:21 +02:00
}
2016-05-29 17:30:45 +02:00
2016-07-28 22:06:52 +02:00
// Epic represents the epic to which an issue is associated
// Not that this struct does not process the returned "color" value
type Epic struct {
2016-09-23 16:19:07 +02:00
ID int ` json:"id" structs:"id" `
Key string ` json:"key" structs:"key" `
Self string ` json:"self" structs:"self" `
Name string ` json:"name" structs:"name" `
Summary string ` json:"summary" structs:"summary" `
Done bool ` json:"done" structs:"done" `
2016-07-28 22:06:52 +02:00
}
2015-09-03 12:25:21 +02:00
// IssueFields represents single fields of a JIRA issue.
// Every JIRA issue has several fields attached.
type IssueFields struct {
// TODO Missing fields
2016-10-24 11:06:37 +02:00
// * "workratio": -1,
// * "lastViewed": null,
// * "environment": null,
2019-04-28 19:52:48 +02:00
Expand string ` json:"expand,omitempty" structs:"expand,omitempty" `
Type IssueType ` json:"issuetype,omitempty" structs:"issuetype,omitempty" `
Project Project ` json:"project,omitempty" structs:"project,omitempty" `
Resolution * Resolution ` json:"resolution,omitempty" structs:"resolution,omitempty" `
Priority * Priority ` json:"priority,omitempty" structs:"priority,omitempty" `
Resolutiondate Time ` json:"resolutiondate,omitempty" structs:"resolutiondate,omitempty" `
Created Time ` json:"created,omitempty" structs:"created,omitempty" `
Duedate Date ` json:"duedate,omitempty" structs:"duedate,omitempty" `
Watches * Watches ` json:"watches,omitempty" structs:"watches,omitempty" `
Assignee * User ` json:"assignee,omitempty" structs:"assignee,omitempty" `
Updated Time ` json:"updated,omitempty" structs:"updated,omitempty" `
Description string ` json:"description,omitempty" structs:"description,omitempty" `
Summary string ` json:"summary,omitempty" structs:"summary,omitempty" `
Creator * User ` json:"Creator,omitempty" structs:"Creator,omitempty" `
Reporter * User ` json:"reporter,omitempty" structs:"reporter,omitempty" `
Components [ ] * Component ` json:"components,omitempty" structs:"components,omitempty" `
Status * Status ` json:"status,omitempty" structs:"status,omitempty" `
Progress * Progress ` json:"progress,omitempty" structs:"progress,omitempty" `
AggregateProgress * Progress ` json:"aggregateprogress,omitempty" structs:"aggregateprogress,omitempty" `
TimeTracking * TimeTracking ` json:"timetracking,omitempty" structs:"timetracking,omitempty" `
TimeSpent int ` json:"timespent,omitempty" structs:"timespent,omitempty" `
TimeEstimate int ` json:"timeestimate,omitempty" structs:"timeestimate,omitempty" `
TimeOriginalEstimate int ` json:"timeoriginalestimate,omitempty" structs:"timeoriginalestimate,omitempty" `
Worklog * Worklog ` json:"worklog,omitempty" structs:"worklog,omitempty" `
IssueLinks [ ] * IssueLink ` json:"issuelinks,omitempty" structs:"issuelinks,omitempty" `
Comments * Comments ` json:"comment,omitempty" structs:"comment,omitempty" `
FixVersions [ ] * FixVersion ` json:"fixVersions,omitempty" structs:"fixVersions,omitempty" `
AffectsVersions [ ] * AffectsVersion ` json:"versions,omitempty" structs:"versions,omitempty" `
Labels [ ] string ` json:"labels,omitempty" structs:"labels,omitempty" `
Subtasks [ ] * Subtasks ` json:"subtasks,omitempty" structs:"subtasks,omitempty" `
Attachments [ ] * Attachment ` json:"attachment,omitempty" structs:"attachment,omitempty" `
Epic * Epic ` json:"epic,omitempty" structs:"epic,omitempty" `
Sprint * Sprint ` json:"sprint,omitempty" structs:"sprint,omitempty" `
Parent * Parent ` json:"parent,omitempty" structs:"parent,omitempty" `
AggregateTimeOriginalEstimate int ` json:"aggregatetimeoriginalestimate,omitempty" structs:"aggregatetimeoriginalestimate,omitempty" `
AggregateTimeSpent int ` json:"aggregatetimespent,omitempty" structs:"aggregatetimespent,omitempty" `
AggregateTimeEstimate int ` json:"aggregatetimeestimate,omitempty" structs:"aggregatetimeestimate,omitempty" `
2018-07-20 14:20:54 +02:00
Unknowns tcontainer . MarshalMap
2016-09-23 16:19:07 +02:00
}
2016-10-03 13:33:46 +02:00
// MarshalJSON is a custom JSON marshal function for the IssueFields structs.
// It handles JIRA custom fields and maps those from / to "Unknowns" key.
2016-09-23 16:19:07 +02:00
func ( i * IssueFields ) MarshalJSON ( ) ( [ ] byte , error ) {
m := structs . Map ( i )
unknowns , okay := m [ "Unknowns" ]
if okay {
2017-05-01 15:03:03 +02:00
// if unknowns present, shift all key value from unknown to a level up
2016-09-23 16:19:07 +02:00
for key , value := range unknowns . ( tcontainer . MarshalMap ) {
m [ key ] = value
}
delete ( m , "Unknowns" )
}
return json . Marshal ( m )
}
2016-10-03 13:33:46 +02:00
// UnmarshalJSON is a custom JSON marshal function for the IssueFields structs.
// It handles JIRA custom fields and maps those from / to "Unknowns" key.
2016-09-23 16:19:07 +02:00
func ( i * IssueFields ) UnmarshalJSON ( data [ ] byte ) error {
// Do the normal unmarshalling first
// Details for this way: http://choly.ca/post/go-json-marshalling/
type Alias IssueFields
aux := & struct {
* Alias
} {
Alias : ( * Alias ) ( i ) ,
}
if err := json . Unmarshal ( data , & aux ) ; err != nil {
return err
}
totalMap := tcontainer . NewMarshalMap ( )
err := json . Unmarshal ( data , & totalMap )
if err != nil {
return err
}
t := reflect . TypeOf ( * i )
for i := 0 ; i < t . NumField ( ) ; i ++ {
field := t . Field ( i )
tagDetail := field . Tag . Get ( "json" )
if tagDetail == "" {
// ignore if there are no tags
continue
}
options := strings . Split ( tagDetail , "," )
if len ( options ) == 0 {
return fmt . Errorf ( "No tags options found for %s" , field . Name )
}
// the first one is the json tag
key := options [ 0 ]
if _ , okay := totalMap . Value ( key ) ; okay {
delete ( totalMap , key )
}
}
i = ( * IssueFields ) ( aux . Alias )
// all the tags found in the struct were removed. Whatever is left are unknowns to struct
i . Unknowns = totalMap
return nil
2015-09-03 12:25:21 +02:00
}
2018-06-03 23:42:41 +02:00
// IssueRenderedFields represents rendered fields of a JIRA issue.
// Not all IssueFields are rendered.
type IssueRenderedFields struct {
// TODO Missing fields
// * "aggregatetimespent": null,
// * "workratio": -1,
// * "lastViewed": null,
// * "aggregatetimeoriginalestimate": null,
// * "aggregatetimeestimate": null,
// * "environment": null,
Resolutiondate string ` json:"resolutiondate,omitempty" structs:"resolutiondate,omitempty" `
Created string ` json:"created,omitempty" structs:"created,omitempty" `
Duedate string ` json:"duedate,omitempty" structs:"duedate,omitempty" `
Updated string ` json:"updated,omitempty" structs:"updated,omitempty" `
Comments * Comments ` json:"comment,omitempty" structs:"comment,omitempty" `
2019-04-22 13:43:06 +02:00
Description string ` json:"description,omitempty" structs:"description,omitempty" `
2018-06-03 23:42:41 +02:00
}
2015-09-03 12:25:21 +02:00
// IssueType represents a type of a JIRA issue.
// Typical types are "Request", "Bug", "Story", ...
type IssueType struct {
2016-09-23 16:19:07 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
2016-10-07 10:52:40 +02:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Description string ` json:"description,omitempty" structs:"description,omitempty" `
IconURL string ` json:"iconUrl,omitempty" structs:"iconUrl,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
Subtask bool ` json:"subtask,omitempty" structs:"subtask,omitempty" `
AvatarID int ` json:"avatarId,omitempty" structs:"avatarId,omitempty" `
2015-09-03 12:25:21 +02:00
}
2017-12-11 15:16:13 +02:00
// Watches represents a type of how many and which user are "observing" a JIRA issue to track the status / updates.
2015-09-03 12:25:21 +02:00
type Watches struct {
2017-12-11 15:16:13 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
WatchCount int ` json:"watchCount,omitempty" structs:"watchCount,omitempty" `
IsWatching bool ` json:"isWatching,omitempty" structs:"isWatching,omitempty" `
Watchers [ ] * Watcher ` json:"watchers,omitempty" structs:"watchers,omitempty" `
}
// Watcher represents a simplified user that "observes" the issue
type Watcher struct {
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
DisplayName string ` json:"displayName,omitempty" structs:"displayName,omitempty" `
Active bool ` json:"active,omitempty" structs:"active,omitempty" `
2016-06-03 23:14:27 +02:00
}
2016-06-03 23:25:18 +02:00
// AvatarUrls represents different dimensions of avatars / images
2016-06-03 23:14:27 +02:00
type AvatarUrls struct {
2016-09-23 16:19:07 +02:00
Four8X48 string ` json:"48x48,omitempty" structs:"48x48,omitempty" `
Two4X24 string ` json:"24x24,omitempty" structs:"24x24,omitempty" `
One6X16 string ` json:"16x16,omitempty" structs:"16x16,omitempty" `
Three2X32 string ` json:"32x32,omitempty" structs:"32x32,omitempty" `
2015-09-03 12:25:21 +02:00
}
// Component represents a "component" of a JIRA issue.
// Components can be user defined in every JIRA instance.
type Component struct {
2016-09-23 16:19:07 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
2015-09-03 12:25:21 +02:00
}
// Status represents the current status of a JIRA issue.
// Typical status are "Open", "In Progress", "Closed", ...
// Status can be user defined in every JIRA instance.
type Status struct {
2016-09-23 16:19:07 +02:00
Self string ` json:"self" structs:"self" `
Description string ` json:"description" structs:"description" `
IconURL string ` json:"iconUrl" structs:"iconUrl" `
Name string ` json:"name" structs:"name" `
ID string ` json:"id" structs:"id" `
StatusCategory StatusCategory ` json:"statusCategory" structs:"statusCategory" `
2015-09-03 12:25:21 +02:00
}
// Progress represents the progress of a JIRA issue.
type Progress struct {
2016-09-23 16:19:07 +02:00
Progress int ` json:"progress" structs:"progress" `
Total int ` json:"total" structs:"total" `
2018-07-20 14:26:26 +02:00
Percent int ` json:"percent" structs:"percent" `
2015-09-03 12:25:21 +02:00
}
2016-11-01 02:31:51 +02:00
// Parent represents the parent of a JIRA issue, to be used with subtask issue types.
type Parent struct {
2016-11-04 21:44:23 +02:00
ID string ` json:"id,omitempty" structs:"id" `
Key string ` json:"key,omitempty" structs:"key" `
2015-09-03 12:25:21 +02:00
}
2016-06-04 10:41:34 +02:00
// Time represents the Time definition of JIRA as a time.Time of go
type Time time . Time
2016-05-29 21:10:19 +02:00
2018-01-16 15:38:17 +02:00
// Date represents the Date definition of JIRA as a time.Time of go
type Date time . Time
2016-07-17 11:23:49 +02:00
// Wrapper struct for search result
type transitionResult struct {
2016-09-23 16:19:07 +02:00
Transitions [ ] Transition ` json:"transitions" structs:"transitions" `
2016-07-17 11:23:49 +02:00
}
// Transition represents an issue transition in JIRA
type Transition struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id" structs:"id" `
Name string ` json:"name" structs:"name" `
2018-01-10 18:21:25 +02:00
To Status ` json:"to" structs:"status" `
2016-09-23 16:19:07 +02:00
Fields map [ string ] TransitionField ` json:"fields" structs:"fields" `
2016-07-17 11:23:49 +02:00
}
2017-05-01 15:03:03 +02:00
// TransitionField represents the value of one Transition
2016-07-17 11:23:49 +02:00
type TransitionField struct {
2016-09-23 16:19:07 +02:00
Required bool ` json:"required" structs:"required" `
2016-07-17 11:23:49 +02:00
}
2016-07-17 11:41:50 +02:00
// CreateTransitionPayload is used for creating new issue transitions
2016-07-17 11:23:49 +02:00
type CreateTransitionPayload struct {
2018-01-10 18:12:02 +02:00
Transition TransitionPayload ` json:"transition" structs:"transition" `
Fields TransitionPayloadFields ` json:"fields" structs:"fields" `
2016-07-17 11:23:49 +02:00
}
2017-05-01 15:03:03 +02:00
// TransitionPayload represents the request payload of Transition calls like DoTransition
2016-07-17 11:23:49 +02:00
type TransitionPayload struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id" structs:"id" `
2016-07-17 11:23:49 +02:00
}
2018-01-10 18:12:02 +02:00
// TransitionPayloadFields represents the fields that can be set when executing a transition
type TransitionPayloadFields struct {
Resolution * Resolution ` json:"resolution,omitempty" structs:"resolution,omitempty" `
}
2017-04-27 07:06:10 +02:00
// Option represents an option value in a SelectList or MultiSelect
// custom issue field
type Option struct {
Value string ` json:"value" structs:"value" `
2016-07-17 11:23:49 +02:00
}
2016-06-04 10:41:34 +02:00
// UnmarshalJSON will transform the JIRA time into a time.Time
// during the transformation of the JIRA JSON response
func ( t * Time ) UnmarshalJSON ( b [ ] byte ) error {
2018-01-10 15:51:30 +02:00
// Ignore null, like in the main JSON package.
if string ( b ) == "null" {
2018-01-16 15:38:17 +02:00
return nil
2018-01-10 15:51:30 +02:00
}
2016-05-29 17:30:45 +02:00
ti , err := time . Parse ( "\"2006-01-02T15:04:05.999-0700\"" , string ( b ) )
if err != nil {
return err
}
2016-06-04 10:41:34 +02:00
* t = Time ( ti )
2016-05-29 17:30:45 +02:00
return nil
2015-09-03 12:25:21 +02:00
}
2018-06-27 13:32:48 +02:00
// MarshalJSON will transform the time.Time into a JIRA time
// during the creation of a JIRA request
func ( t Time ) MarshalJSON ( ) ( [ ] byte , error ) {
return [ ] byte ( time . Time ( t ) . Format ( "\"2006-01-02T15:04:05.999-0700\"" ) ) , nil
}
2018-01-16 15:38:17 +02:00
// UnmarshalJSON will transform the JIRA date into a time.Time
// during the transformation of the JIRA JSON response
func ( t * Date ) UnmarshalJSON ( b [ ] byte ) error {
// Ignore null, like in the main JSON package.
if string ( b ) == "null" {
return nil
}
ti , err := time . Parse ( "\"2006-01-02\"" , string ( b ) )
if err != nil {
return err
}
* t = Date ( ti )
return nil
}
2018-01-19 18:57:08 +02:00
// MarshalJSON will transform the Date object into a short
// date string as JIRA expects during the creation of a
// JIRA request
func ( t Date ) MarshalJSON ( ) ( [ ] byte , error ) {
time := time . Time ( t )
return [ ] byte ( time . Format ( "\"2006-01-02\"" ) ) , nil
}
2015-09-03 12:25:21 +02:00
// Worklog represents the work log of a JIRA issue.
2016-06-03 23:25:18 +02:00
// One Worklog contains zero or n WorklogRecords
2015-09-03 12:25:21 +02:00
// JIRA Wiki: https://confluence.atlassian.com/jira/logging-work-on-an-issue-185729605.html
type Worklog struct {
2016-09-23 16:19:07 +02:00
StartAt int ` json:"startAt" structs:"startAt" `
MaxResults int ` json:"maxResults" structs:"maxResults" `
Total int ` json:"total" structs:"total" `
Worklogs [ ] WorklogRecord ` json:"worklogs" structs:"worklogs" `
2016-06-03 23:14:27 +02:00
}
2016-06-03 23:25:18 +02:00
// WorklogRecord represents one entry of a Worklog
2016-06-03 23:14:27 +02:00
type WorklogRecord struct {
2018-02-17 14:09:23 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
2018-03-01 08:46:54 +02:00
Author * User ` json:"author,omitempty" structs:"author,omitempty" `
UpdateAuthor * User ` json:"updateAuthor,omitempty" structs:"updateAuthor,omitempty" `
2018-02-17 14:09:23 +02:00
Comment string ` json:"comment,omitempty" structs:"comment,omitempty" `
2018-03-01 08:46:54 +02:00
Created * Time ` json:"created,omitempty" structs:"created,omitempty" `
Updated * Time ` json:"updated,omitempty" structs:"updated,omitempty" `
Started * Time ` json:"started,omitempty" structs:"started,omitempty" `
2018-02-17 14:09:23 +02:00
TimeSpent string ` json:"timeSpent,omitempty" structs:"timeSpent,omitempty" `
TimeSpentSeconds int ` json:"timeSpentSeconds,omitempty" structs:"timeSpentSeconds,omitempty" `
ID string ` json:"id,omitempty" structs:"id,omitempty" `
IssueID string ` json:"issueId,omitempty" structs:"issueId,omitempty" `
2016-05-29 18:42:38 +02:00
}
2016-10-24 11:06:37 +02:00
// TimeTracking represents the timetracking fields of a JIRA issue.
type TimeTracking struct {
2016-10-23 14:50:36 +02:00
OriginalEstimate string ` json:"originalEstimate,omitempty" structs:"originalEstimate,omitempty" `
RemainingEstimate string ` json:"remainingEstimate,omitempty" structs:"remainingEstimate,omitempty" `
TimeSpent string ` json:"timeSpent,omitempty" structs:"timeSpent,omitempty" `
OriginalEstimateSeconds int ` json:"originalEstimateSeconds,omitempty" structs:"originalEstimateSeconds,omitempty" `
RemainingEstimateSeconds int ` json:"remainingEstimateSeconds,omitempty" structs:"remainingEstimateSeconds,omitempty" `
TimeSpentSeconds int ` json:"timeSpentSeconds,omitempty" structs:"timeSpentSeconds,omitempty" `
2016-05-29 18:42:38 +02:00
}
2016-06-03 23:25:18 +02:00
// Subtasks represents all issues of a parent issue.
2016-05-29 18:42:38 +02:00
type Subtasks struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id" structs:"id" `
Key string ` json:"key" structs:"key" `
Self string ` json:"self" structs:"self" `
Fields IssueFields ` json:"fields" structs:"fields" `
2015-09-03 12:25:21 +02:00
}
// IssueLink represents a link between two issues in JIRA.
type IssueLink struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Type IssueLinkType ` json:"type" structs:"type" `
OutwardIssue * Issue ` json:"outwardIssue" structs:"outwardIssue" `
InwardIssue * Issue ` json:"inwardIssue" structs:"inwardIssue" `
Comment * Comment ` json:"comment,omitempty" structs:"comment,omitempty" `
2015-09-03 12:25:21 +02:00
}
// IssueLinkType represents a type of a link between to issues in JIRA.
// Typical issue link types are "Related to", "Duplicate", "Is blocked by", etc.
type IssueLinkType struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Name string ` json:"name" structs:"name" `
Inward string ` json:"inward" structs:"inward" `
Outward string ` json:"outward" structs:"outward" `
2015-09-03 12:25:21 +02:00
}
2016-07-08 01:21:14 +02:00
// Comments represents a list of Comment.
type Comments struct {
2016-09-23 16:19:07 +02:00
Comments [ ] * Comment ` json:"comments,omitempty" structs:"comments,omitempty" `
2016-07-08 01:21:14 +02:00
}
2015-09-03 12:25:21 +02:00
// Comment represents a comment by a person to an issue in JIRA.
type Comment struct {
2016-09-23 16:19:07 +02:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
2016-10-23 14:50:36 +02:00
Name string ` json:"name,omitempty" structs:"name,omitempty" `
2016-09-23 16:19:07 +02:00
Author User ` json:"author,omitempty" structs:"author,omitempty" `
Body string ` json:"body,omitempty" structs:"body,omitempty" `
UpdateAuthor User ` json:"updateAuthor,omitempty" structs:"updateAuthor,omitempty" `
Updated string ` json:"updated,omitempty" structs:"updated,omitempty" `
Created string ` json:"created,omitempty" structs:"created,omitempty" `
Visibility CommentVisibility ` json:"visibility,omitempty" structs:"visibility,omitempty" `
2015-09-03 12:25:21 +02:00
}
2016-02-14 05:15:32 +02:00
// FixVersion represents a software release in which an issue is fixed.
type FixVersion struct {
2019-04-28 19:10:41 +02:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
2016-09-23 16:19:07 +02:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
2019-04-28 19:10:41 +02:00
Description string ` json:"description,omitempty" structs:"name,omitempty" `
Archived * bool ` json:"archived,omitempty" structs:"archived,omitempty" `
2016-09-23 16:19:07 +02:00
Released * bool ` json:"released,omitempty" structs:"released,omitempty" `
2019-04-28 19:10:41 +02:00
ReleaseDate string ` json:"releaseDate,omitempty" structs:"releaseDate,omitempty" `
2016-09-23 16:19:07 +02:00
UserReleaseDate string ` json:"userReleaseDate,omitempty" structs:"userReleaseDate,omitempty" `
2019-04-28 19:10:41 +02:00
ProjectID int ` json:"projectId,omitempty" structs:"projectId,omitempty" ` // Unlike other IDs, this is returned as a number
StartDate string ` json:"startDate,omitempty" structs:"startDate,omitempty" `
2016-02-14 05:15:32 +02:00
}
2019-04-28 19:52:48 +02:00
// AffectsVersion represents a software release which is affected by an issue.
type AffectsVersion Version
2016-03-27 14:03:40 +02:00
// CommentVisibility represents he visibility of a comment.
// E.g. Type could be "role" and Value "Administrators"
type CommentVisibility struct {
2016-09-23 16:19:07 +02:00
Type string ` json:"type,omitempty" structs:"type,omitempty" `
Value string ` json:"value,omitempty" structs:"value,omitempty" `
2016-05-29 17:30:45 +02:00
}
2016-06-19 15:08:53 +02:00
// SearchOptions specifies the optional parameters to various List methods that
// support pagination.
// Pagination is used for the JIRA REST APIs to conserve server resources and limit
// response size for resources that return potentially large collection of items.
// A request to a pages API will result in a values array wrapped in a JSON object with some paging metadata
// Default Pagination options
2016-06-16 11:10:30 +02:00
type SearchOptions struct {
2016-06-19 15:08:53 +02:00
// StartAt: The starting index of the returned projects. Base index: 0.
StartAt int ` url:"startAt,omitempty" `
// MaxResults: The maximum number of projects to return per page. Default: 50.
MaxResults int ` url:"maxResults,omitempty" `
2017-01-28 00:04:08 +02:00
// Expand: Expand specific sections in the returned issues
2017-02-24 02:48:06 +02:00
Expand string ` url:"expand,omitempty" `
2017-04-13 21:02:50 +02:00
Fields [ ] string
2018-03-15 15:52:16 +02:00
// ValidateQuery: The validateQuery param offers control over whether to validate and how strictly to treat the validation. Default: strict.
ValidateQuery string ` url:"validateQuery,omitempty" `
2016-06-16 11:10:30 +02:00
}
2017-01-29 18:28:04 +02:00
// searchResult is only a small wrapper around the Search (with JQL) method
2016-06-04 10:41:34 +02:00
// to be able to parse the results
type searchResult struct {
2016-09-23 16:19:07 +02:00
Issues [ ] Issue ` json:"issues" structs:"issues" `
StartAt int ` json:"startAt" structs:"startAt" `
MaxResults int ` json:"maxResults" structs:"maxResults" `
Total int ` json:"total" structs:"total" `
2016-06-04 10:41:34 +02:00
}
2017-01-28 00:04:08 +02:00
// GetQueryOptions specifies the optional parameters for the Get Issue methods
type GetQueryOptions struct {
// Fields is the list of fields to return for the issue. By default, all fields are returned.
Fields string ` url:"fields,omitempty" `
Expand string ` url:"expand,omitempty" `
// Properties is the list of properties to return for the issue. By default no properties are returned.
Properties string ` url:"properties,omitempty" `
// FieldsByKeys if true then fields in issues will be referenced by keys instead of ids
2018-03-05 15:56:57 +02:00
FieldsByKeys bool ` url:"fieldsByKeys,omitempty" `
UpdateHistory bool ` url:"updateHistory,omitempty" `
ProjectKeys string ` url:"projectKeys,omitempty" `
2016-06-04 10:41:34 +02:00
}
// CustomFields represents custom fields of JIRA
// This can heavily differ between JIRA instances
type CustomFields map [ string ] string
2015-09-03 12:25:21 +02:00
// Get returns a full representation of the issue for the given issue key.
// JIRA will attempt to identify the issue by the issueIdOrKey path parameter.
// This can be an issue id, or an issue key.
// If the issue cannot be found via an exact match, JIRA will also look for the issue in a case-insensitive way, or by looking to see if the issue was moved.
//
2017-01-28 00:04:08 +02:00
// The given options will be appended to the query string
//
2016-03-27 14:24:48 +02:00
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-getIssue
2017-01-28 00:04:08 +02:00
func ( s * IssueService ) Get ( issueID string , options * GetQueryOptions ) ( * Issue , * Response , error ) {
2018-03-13 11:49:27 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s" , issueID )
req , err := s . client . NewRequest ( "GET" , apiEndpoint , nil )
if err != nil {
return nil , nil , err
}
if options != nil {
q , err := query . Values ( options )
if err != nil {
return nil , nil , err
}
req . URL . RawQuery = q . Encode ( )
}
issue := new ( Issue )
resp , err := s . client . Do ( req , issue )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return issue , resp , nil
}
2016-06-15 17:09:13 +02:00
// DownloadAttachment returns a Response of an attachment for a given attachmentID.
// The attachment is in the Response.Body of the response.
2016-05-27 14:14:09 +02:00
// This is an io.ReadCloser.
// The caller should close the resp.Body.
2016-06-15 17:09:13 +02:00
func ( s * IssueService ) DownloadAttachment ( attachmentID string ) ( * Response , error ) {
2016-05-19 23:11:21 +02:00
apiEndpoint := fmt . Sprintf ( "secure/attachment/%s/" , attachmentID )
2016-05-29 17:30:45 +02:00
req , err := s . client . NewRequest ( "GET" , apiEndpoint , nil )
if err != nil {
2016-05-19 23:11:21 +02:00
return nil , err
2016-05-29 17:30:45 +02:00
}
2016-05-27 14:14:09 +02:00
resp , err := s . client . Do ( req , nil )
2016-05-29 17:30:45 +02:00
if err != nil {
2017-11-04 00:22:32 +02:00
jerr := NewJiraError ( resp , err )
return resp , jerr
2016-05-29 17:30:45 +02:00
}
2016-05-19 23:11:21 +02:00
return resp , nil
}
2017-05-06 14:09:22 +02:00
// PostAttachment uploads r (io.Reader) as an attachment to a given issueID
func ( s * IssueService ) PostAttachment ( issueID string , r io . Reader , attachmentName string ) ( * [ ] Attachment , * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/attachments" , issueID )
2016-05-19 23:11:21 +02:00
b := new ( bytes . Buffer )
writer := multipart . NewWriter ( b )
fw , err := writer . CreateFormFile ( "file" , attachmentName )
if err != nil {
return nil , nil , err
2016-05-29 17:30:45 +02:00
}
2016-05-25 09:47:10 +02:00
if r != nil {
// Copy the file
if _ , err = io . Copy ( fw , r ) ; err != nil {
return nil , nil , err
2016-05-29 17:30:45 +02:00
}
}
2016-05-19 23:11:21 +02:00
writer . Close ( )
req , err := s . client . NewMultiPartRequest ( "POST" , apiEndpoint , b )
if err != nil {
return nil , nil , err
}
req . Header . Set ( "Content-Type" , writer . FormDataContentType ( ) )
2016-05-25 09:04:23 +02:00
// PostAttachment response returns a JSON array (as multiple attachments can be posted)
attachment := new ( [ ] Attachment )
resp , err := s . client . Do ( req , attachment )
2016-05-19 23:11:21 +02:00
if err != nil {
2017-11-04 00:22:32 +02:00
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
2016-05-19 23:11:21 +02:00
}
2016-05-25 09:04:23 +02:00
return attachment , resp , nil
2016-05-29 17:30:45 +02:00
}
2017-10-27 23:21:47 +02:00
// GetWorklogs gets all the worklogs for an issue.
// This method is especially important if you need to read all the worklogs, not just the first page.
//
// https://docs.atlassian.com/jira/REST/cloud/#api/2/issue/{issueIdOrKey}/worklog-getIssueWorklog
func ( s * IssueService ) GetWorklogs ( issueID string ) ( * Worklog , * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/worklog" , issueID )
req , err := s . client . NewRequest ( "GET" , apiEndpoint , nil )
if err != nil {
return nil , nil , err
}
v := new ( Worklog )
resp , err := s . client . Do ( req , v )
return v , resp , err
}
2015-09-03 12:25:21 +02:00
// Create creates an issue or a sub-task from a JSON representation.
// Creating a sub-task is similar to creating a regular issue, with two important differences:
// The issueType field must correspond to a sub-task issue type and you must provide a parent field in the issue create request containing the id or key of the parent issue.
//
2016-03-27 14:24:48 +02:00
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-createIssues
2016-06-15 17:09:13 +02:00
func ( s * IssueService ) Create ( issue * Issue ) ( * Issue , * Response , error ) {
2018-08-10 17:11:53 +02:00
apiEndpoint := "rest/api/2/issue"
2015-09-03 12:25:21 +02:00
req , err := s . client . NewRequest ( "POST" , apiEndpoint , issue )
if err != nil {
return nil , nil , err
}
2016-10-05 14:53:30 +02:00
resp , err := s . client . Do ( req , nil )
2015-09-03 12:25:21 +02:00
if err != nil {
2016-10-05 14:53:30 +02:00
// incase of error return the resp for further inspection
2015-09-03 12:25:21 +02:00
return nil , resp , err
}
responseIssue := new ( Issue )
2016-10-05 14:53:30 +02:00
defer resp . Body . Close ( )
data , err := ioutil . ReadAll ( resp . Body )
if err != nil {
return nil , resp , fmt . Errorf ( "Could not read the returned data" )
}
err = json . Unmarshal ( data , responseIssue )
if err != nil {
return nil , resp , fmt . Errorf ( "Could not unmarshall the data into struct" )
}
2015-09-03 12:25:21 +02:00
return responseIssue , resp , nil
}
2016-01-05 16:22:21 +02:00
2018-10-17 21:17:08 +02:00
// UpdateWithOptions updates an issue from a JSON representation,
// while also specifiying query params. The issue is found by key.
2017-06-09 21:46:55 +02:00
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue-editIssue
2018-10-17 21:44:13 +02:00
func ( s * IssueService ) UpdateWithOptions ( issue * Issue , opts * UpdateQueryOptions ) ( * Issue , * Response , error ) {
2017-06-26 16:10:19 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%v" , issue . Key )
2018-10-17 21:17:08 +02:00
url , err := addOptions ( apiEndpoint , opts )
if err != nil {
return nil , nil , err
}
req , err := s . client . NewRequest ( "PUT" , url , issue )
2017-06-09 21:46:55 +02:00
if err != nil {
return nil , nil , err
}
resp , err := s . client . Do ( req , nil )
2015-09-03 12:25:21 +02:00
if err != nil {
2017-11-04 00:22:32 +02:00
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
2015-09-03 12:25:21 +02:00
}
2017-06-09 21:46:55 +02:00
// This is just to follow the rest of the API's convention of returning an issue.
// Returning the same pointer here is pointless, so we return a copy instead.
ret := * issue
return & ret , resp , nil
}
2017-06-26 16:10:19 +02:00
// Update updates an issue from a JSON representation. The issue is found by key.
2017-06-09 21:46:55 +02:00
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue-editIssue
func ( s * IssueService ) Update ( issue * Issue ) ( * Issue , * Response , error ) {
2018-10-18 20:31:40 +02:00
return s . UpdateWithOptions ( issue , nil )
2017-06-09 21:46:55 +02:00
}
2018-03-01 08:46:54 +02:00
// UpdateIssue updates an issue from a JSON representation. The issue is found by key.
2017-08-03 11:57:24 +02:00
//
// https://docs.atlassian.com/jira/REST/7.4.0/#api/2/issue-editIssue
2018-03-01 08:46:54 +02:00
func ( s * IssueService ) UpdateIssue ( jiraID string , data map [ string ] interface { } ) ( * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%v" , jiraID )
2017-08-03 11:57:24 +02:00
req , err := s . client . NewRequest ( "PUT" , apiEndpoint , data )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
return resp , err
}
// This is just to follow the rest of the API's convention of returning an issue.
// Returning the same pointer here is pointless, so we return a copy instead.
return resp , nil
2015-09-03 12:25:21 +02:00
}
2016-01-05 16:22:21 +02:00
2016-03-27 14:03:40 +02:00
// AddComment adds a new comment to issueID.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-addComment
2016-06-15 17:09:13 +02:00
func ( s * IssueService ) AddComment ( issueID string , comment * Comment ) ( * Comment , * Response , error ) {
2016-01-05 16:22:21 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment" , issueID )
2016-03-27 14:03:40 +02:00
req , err := s . client . NewRequest ( "POST" , apiEndpoint , comment )
2016-01-05 16:22:21 +02:00
if err != nil {
return nil , nil , err
}
2016-03-27 14:03:40 +02:00
responseComment := new ( Comment )
2016-02-29 09:29:44 +02:00
resp , err := s . client . Do ( req , responseComment )
if err != nil {
2017-11-04 00:22:32 +02:00
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
2016-02-29 09:29:44 +02:00
}
2016-01-05 16:22:21 +02:00
2016-03-27 14:03:40 +02:00
return responseComment , resp , nil
}
2016-02-29 09:29:44 +02:00
2017-08-31 23:46:04 +02:00
// UpdateComment updates the body of a comment, identified by comment.ID, on the issueID.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue/{issueIdOrKey}/comment-updateComment
func ( s * IssueService ) UpdateComment ( issueID string , comment * Comment ) ( * Comment , * Response , error ) {
reqBody := struct {
Body string ` json:"body" `
} {
Body : comment . Body ,
}
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment/%s" , issueID , comment . ID )
2018-08-10 05:08:51 +02:00
req , err := s . client . NewRequest ( "PUT" , apiEndpoint , reqBody )
2017-08-31 23:46:04 +02:00
if err != nil {
return nil , nil , err
}
responseComment := new ( Comment )
resp , err := s . client . Do ( req , responseComment )
2016-02-29 09:29:44 +02:00
if err != nil {
return nil , resp , err
}
2016-01-05 16:22:21 +02:00
2016-03-27 14:03:40 +02:00
return responseComment , resp , nil
}
2016-02-29 09:29:44 +02:00
2018-09-14 21:11:08 +02:00
// DeleteComment Deletes a comment from an issueID.
//
// JIRA API docs: https://developer.atlassian.com/cloud/jira/platform/rest/v3/#api-api-3-issue-issueIdOrKey-comment-id-delete
func ( s * IssueService ) DeleteComment ( issueID , commentID string ) error {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment/%s" , issueID , commentID )
req , err := s . client . NewRequest ( "DELETE" , apiEndpoint , nil )
if err != nil {
return err
}
2018-09-17 17:44:35 +02:00
resp , err := s . client . Do ( req , nil )
2018-09-14 21:11:08 +02:00
if err != nil {
2018-09-17 17:44:35 +02:00
jerr := NewJiraError ( resp , err )
return jerr
2018-09-14 21:11:08 +02:00
}
return nil
}
2018-02-17 14:09:23 +02:00
// AddWorklogRecord adds a new worklog record to issueID.
//
// https://developer.atlassian.com/cloud/jira/platform/rest/#api-api-2-issue-issueIdOrKey-worklog-post
func ( s * IssueService ) AddWorklogRecord ( issueID string , record * WorklogRecord ) ( * WorklogRecord , * Response , error ) {
2018-03-01 08:46:54 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/worklog" , issueID )
req , err := s . client . NewRequest ( "POST" , apiEndpoint , record )
if err != nil {
return nil , nil , err
}
2018-02-17 14:09:23 +02:00
2018-03-01 08:46:54 +02:00
responseRecord := new ( WorklogRecord )
resp , err := s . client . Do ( req , responseRecord )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
2018-02-17 14:09:23 +02:00
2018-03-01 08:46:54 +02:00
return responseRecord , resp , nil
2018-02-17 14:09:23 +02:00
}
2016-02-29 09:29:44 +02:00
// AddLink adds a link between two issues.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issueLink
2016-06-15 17:09:13 +02:00
func ( s * IssueService ) AddLink ( issueLink * IssueLink ) ( * Response , error ) {
2016-02-29 09:29:44 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issueLink" )
req , err := s . client . NewRequest ( "POST" , apiEndpoint , issueLink )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
2017-11-04 00:22:32 +02:00
if err != nil {
err = NewJiraError ( resp , err )
}
2016-02-29 09:29:44 +02:00
return resp , err
}
2016-05-29 17:30:45 +02:00
2016-06-04 10:41:34 +02:00
// Search will search for tickets according to the jql
//
2016-05-29 17:30:45 +02:00
// JIRA API docs: https://developer.atlassian.com/jiradev/jira-apis/jira-rest-apis/jira-rest-api-tutorials/jira-rest-api-example-query-issues
2016-06-16 11:10:30 +02:00
func ( s * IssueService ) Search ( jql string , options * SearchOptions ) ( [ ] Issue , * Response , error ) {
var u string
if options == nil {
u = fmt . Sprintf ( "rest/api/2/search?jql=%s" , url . QueryEscape ( jql ) )
} else {
2018-03-15 15:52:16 +02:00
u = fmt . Sprintf ( "rest/api/2/search?jql=%s&startAt=%d&maxResults=%d&expand=%s&fields=%s&validateQuery=%s" , url . QueryEscape ( jql ) ,
options . StartAt , options . MaxResults , options . Expand , strings . Join ( options . Fields , "," ) , options . ValidateQuery )
2016-06-16 11:10:30 +02:00
}
2016-06-04 10:41:34 +02:00
req , err := s . client . NewRequest ( "GET" , u , nil )
2016-05-29 17:30:45 +02:00
if err != nil {
2016-06-04 10:41:34 +02:00
return [ ] Issue { } , nil , err
2016-05-29 17:30:45 +02:00
}
2016-06-04 02:51:44 +02:00
2016-06-04 10:41:34 +02:00
v := new ( searchResult )
resp , err := s . client . Do ( req , v )
2017-11-04 00:22:32 +02:00
if err != nil {
err = NewJiraError ( resp , err )
}
2016-06-04 10:41:34 +02:00
return v . Issues , resp , err
}
2016-06-04 02:51:44 +02:00
2017-11-01 18:19:19 +02:00
// SearchPages will get issues from all pages in a search
//
// JIRA API docs: https://developer.atlassian.com/jiradev/jira-apis/jira-rest-apis/jira-rest-api-tutorials/jira-rest-api-example-query-issues
func ( s * IssueService ) SearchPages ( jql string , options * SearchOptions , f func ( Issue ) error ) error {
if options == nil {
options = & SearchOptions {
StartAt : 0 ,
MaxResults : 50 ,
}
}
if options . MaxResults == 0 {
options . MaxResults = 50
}
issues , resp , err := s . Search ( jql , options )
if err != nil {
return err
}
for {
for _ , issue := range issues {
err = f ( issue )
if err != nil {
return err
}
}
if resp . StartAt + resp . MaxResults >= resp . Total {
return nil
}
options . StartAt += resp . MaxResults
issues , resp , err = s . Search ( jql , options )
if err != nil {
return err
}
}
}
2016-06-04 10:41:34 +02:00
// GetCustomFields returns a map of customfield_* keys with string values
2016-06-15 17:09:13 +02:00
func ( s * IssueService ) GetCustomFields ( issueID string ) ( CustomFields , * Response , error ) {
2016-06-04 02:51:44 +02:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s" , issueID )
req , err := s . client . NewRequest ( "GET" , apiEndpoint , nil )
if err != nil {
return nil , nil , err
}
issue := new ( map [ string ] interface { } )
resp , err := s . client . Do ( req , issue )
if err != nil {
2017-11-04 00:22:32 +02:00
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
2016-06-04 02:51:44 +02:00
}
2016-06-04 10:41:34 +02:00
2016-06-04 02:51:44 +02:00
m := * issue
f := m [ "fields" ]
cf := make ( CustomFields )
if f == nil {
return cf , resp , nil
}
if rec , ok := f . ( map [ string ] interface { } ) ; ok {
for key , val := range rec {
if strings . Contains ( key , "customfield" ) {
2016-09-07 01:14:15 +02:00
if valMap , ok := val . ( map [ string ] interface { } ) ; ok {
if v , ok := valMap [ "value" ] ; ok {
val = v
}
}
2016-06-04 02:51:44 +02:00
cf [ key ] = fmt . Sprint ( val )
}
}
}
return cf , resp , nil
2016-05-29 17:30:45 +02:00
}
2016-07-17 11:23:49 +02:00
// GetTransitions gets a list of the transitions possible for this issue by the current user,
// along with fields that are required and their types.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-getTransitions
func ( s * IssueService ) GetTransitions ( id string ) ( [ ] Transition , * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/transitions?expand=transitions.fields" , id )
req , err := s . client . NewRequest ( "GET" , apiEndpoint , nil )
if err != nil {
return nil , nil , err
}
result := new ( transitionResult )
resp , err := s . client . Do ( req , result )
2017-11-04 00:22:32 +02:00
if err != nil {
err = NewJiraError ( resp , err )
}
2016-07-17 11:23:49 +02:00
return result . Transitions , resp , err
}
// DoTransition performs a transition on an issue.
// When performing the transition you can update or set other issue fields.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-doTransition
func ( s * IssueService ) DoTransition ( ticketID , transitionID string ) ( * Response , error ) {
payload := CreateTransitionPayload {
Transition : TransitionPayload {
ID : transitionID ,
} ,
}
2017-05-04 15:45:34 +02:00
return s . DoTransitionWithPayload ( ticketID , payload )
}
// DoTransitionWithPayload performs a transition on an issue using any payload.
// When performing the transition you can update or set other issue fields.
//
// JIRA API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-doTransition
func ( s * IssueService ) DoTransitionWithPayload ( ticketID , payload interface { } ) ( * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/transitions" , ticketID )
2016-07-17 11:23:49 +02:00
req , err := s . client . NewRequest ( "POST" , apiEndpoint , payload )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
2017-11-04 00:22:32 +02:00
err = NewJiraError ( resp , err )
2016-07-17 11:23:49 +02:00
}
2017-11-04 00:22:32 +02:00
return resp , err
2016-07-17 11:24:23 +02:00
}
2016-10-05 18:04:48 +02:00
// InitIssueWithMetaAndFields returns Issue with with values from fieldsConfig properly set.
// * metaProject should contain metaInformation about the project where the issue should be created.
// * metaIssuetype is the MetaInformation about the Issuetype that needs to be created.
// * fieldsConfig is a key->value pair where key represents the name of the field as seen in the UI
2017-04-27 07:06:10 +02:00
// And value is the string value for that particular key.
2016-10-05 18:04:48 +02:00
// Note: This method doesn't verify that the fieldsConfig is complete with mandatory fields. The fieldsConfig is
// supposed to be already verified with MetaIssueType.CheckCompleteAndAvailable. It will however return
// error if the key is not found.
// All values will be packed into Unknowns. This is much convenient. If the struct fields needs to be
// configured as well, marshalling and unmarshalling will set the proper fields.
func InitIssueWithMetaAndFields ( metaProject * MetaProject , metaIssuetype * MetaIssueType , fieldsConfig map [ string ] string ) ( * Issue , error ) {
issue := new ( Issue )
issueFields := new ( IssueFields )
issueFields . Unknowns = tcontainer . NewMarshalMap ( )
// map the field names the User presented to jira's internal key
allFields , _ := metaIssuetype . GetAllFields ( )
for key , value := range fieldsConfig {
jiraKey , found := allFields [ key ]
if ! found {
2018-03-01 08:46:54 +02:00
return nil , fmt . Errorf ( "key %s is not found in the list of fields" , key )
2016-10-05 18:04:48 +02:00
}
valueType , err := metaIssuetype . Fields . String ( jiraKey + "/schema/type" )
if err != nil {
return nil , err
}
switch valueType {
case "array" :
elemType , err := metaIssuetype . Fields . String ( jiraKey + "/schema/items" )
if err != nil {
return nil , err
}
switch elemType {
case "component" :
2017-05-01 15:06:18 +02:00
issueFields . Unknowns [ jiraKey ] = [ ] Component { { Name : value } }
2017-10-27 23:22:58 +02:00
case "option" :
issueFields . Unknowns [ jiraKey ] = [ ] map [ string ] string { { "value" : value } }
2016-10-05 18:04:48 +02:00
default :
issueFields . Unknowns [ jiraKey ] = [ ] string { value }
}
case "string" :
issueFields . Unknowns [ jiraKey ] = value
case "date" :
issueFields . Unknowns [ jiraKey ] = value
2017-07-31 09:03:34 +02:00
case "datetime" :
issueFields . Unknowns [ jiraKey ] = value
2016-10-05 18:04:48 +02:00
case "any" :
// Treat any as string
issueFields . Unknowns [ jiraKey ] = value
case "project" :
issueFields . Unknowns [ jiraKey ] = Project {
Name : metaProject . Name ,
ID : metaProject . Id ,
}
case "priority" :
issueFields . Unknowns [ jiraKey ] = Priority { Name : value }
case "user" :
issueFields . Unknowns [ jiraKey ] = User {
Name : value ,
}
case "issuetype" :
issueFields . Unknowns [ jiraKey ] = IssueType {
Name : value ,
}
2017-04-27 07:06:10 +02:00
case "option" :
issueFields . Unknowns [ jiraKey ] = Option {
Value : value ,
}
2016-10-05 18:04:48 +02:00
default :
return nil , fmt . Errorf ( "Unknown issue type encountered: %s for %s" , valueType , key )
}
}
issue . Fields = issueFields
return issue , nil
}
2017-02-24 03:41:52 +02:00
// Delete will delete a specified issue.
func ( s * IssueService ) Delete ( issueID string ) ( * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s" , issueID )
// to enable deletion of subtasks; without this, the request will fail if the issue has subtasks
deletePayload := make ( map [ string ] interface { } )
deletePayload [ "deleteSubtasks" ] = "true"
content , _ := json . Marshal ( deletePayload )
req , err := s . client . NewRequest ( "DELETE" , apiEndpoint , content )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
return resp , err
}
2017-12-11 15:16:13 +02:00
// GetWatchers wil return all the users watching/observing the given issue
2017-12-18 12:30:59 +02:00
//
// JIRA API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-getIssueWatchers
2017-12-11 15:16:13 +02:00
func ( s * IssueService ) GetWatchers ( issueID string ) ( * [ ] User , * Response , error ) {
2018-03-01 08:46:54 +02:00
watchesAPIEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
2017-12-11 15:16:13 +02:00
2018-03-01 08:46:54 +02:00
req , err := s . client . NewRequest ( "GET" , watchesAPIEndpoint , nil )
2017-12-11 15:16:13 +02:00
if err != nil {
return nil , nil , err
}
watches := new ( Watches )
resp , err := s . client . Do ( req , watches )
if err != nil {
return nil , nil , NewJiraError ( resp , err )
}
result := [ ] User { }
user := new ( User )
for _ , watcher := range watches . Watchers {
user , resp , err = s . client . User . Get ( watcher . Name )
if err != nil {
return nil , resp , NewJiraError ( resp , err )
}
result = append ( result , * user )
}
return & result , resp , nil
}
2018-03-01 08:46:54 +02:00
// AddWatcher adds watcher to the given issue
2017-12-18 12:30:59 +02:00
//
// JIRA API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-addWatcher
2017-12-11 15:16:13 +02:00
func ( s * IssueService ) AddWatcher ( issueID string , userName string ) ( * Response , error ) {
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
req , err := s . client . NewRequest ( "POST" , apiEndPoint , userName )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
// RemoveWatcher removes given user from given issue
2017-12-18 12:30:59 +02:00
//
// JIRA API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-removeWatcher
2017-12-11 15:16:13 +02:00
func ( s * IssueService ) RemoveWatcher ( issueID string , userName string ) ( * Response , error ) {
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
req , err := s . client . NewRequest ( "DELETE" , apiEndPoint , userName )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2018-06-28 17:28:06 +02:00
// UpdateAssignee updates the user assigned to work on the given issue
//
// JIRA API docs: https://docs.atlassian.com/software/jira/docs/api/REST/7.10.2/#api/2/issue-assign
func ( s * IssueService ) UpdateAssignee ( issueID string , assignee * User ) ( * Response , error ) {
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/assignee" , issueID )
req , err := s . client . NewRequest ( "PUT" , apiEndPoint , assignee )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2018-09-03 21:51:52 +02:00
2018-07-21 22:42:58 +02:00
func ( c ChangelogHistory ) CreatedTime ( ) ( time . Time , error ) {
var t time . Time
// Ignore null
if string ( c . Created ) == "null" {
return t , nil
}
t , err := time . Parse ( "2006-01-02T15:04:05.999-0700" , c . Created )
return t , err
}