2018-06-02 00:37:01 +00:00
package jira
import (
"bytes"
2021-10-06 17:33:14 +00:00
"context"
2018-06-02 00:37:01 +00:00
"encoding/json"
"fmt"
"io"
"io/ioutil"
"mime/multipart"
2020-09-14 18:31:56 +00:00
"net/http"
2018-06-02 00:37:01 +00:00
"net/url"
"reflect"
2021-10-06 17:33:14 +00:00
"strconv"
2018-06-02 00:37:01 +00:00
"strings"
"time"
"github.com/fatih/structs"
"github.com/google/go-querystring/query"
"github.com/trivago/tgo/tcontainer"
)
const (
2021-10-06 17:33:14 +00:00
// AssigneeAutomatic represents the value of the "Assignee: Automatic" of Jira
2018-06-02 00:37:01 +00:00
AssigneeAutomatic = "-1"
)
2021-10-06 17:33:14 +00:00
// IssueService handles Issues for the Jira instance / API.
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue
2018-06-02 00:37:01 +00:00
type IssueService struct {
client * Client
}
2020-09-14 18:31:56 +00:00
// UpdateQueryOptions specifies the optional parameters to the Edit issue
type UpdateQueryOptions struct {
NotifyUsers bool ` url:"notifyUsers,omitempty" `
OverrideScreenSecurity bool ` url:"overrideScreenSecurity,omitempty" `
OverrideEditableFlag bool ` url:"overrideEditableFlag,omitempty" `
}
2021-10-06 17:33:14 +00:00
// Issue represents a Jira issue.
2018-06-02 00:37:01 +00:00
type Issue struct {
2020-09-14 18:31:56 +00: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" `
Transitions [ ] Transition ` json:"transitions,omitempty" structs:"transitions,omitempty" `
2021-10-06 17:33:14 +00:00
Names map [ string ] string ` json:"names,omitempty" structs:"names,omitempty" `
2018-06-02 00:37:01 +00: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" `
}
2021-10-06 17:33:14 +00:00
// Attachment represents a Jira attachment
2018-06-02 00:37:01 +00:00
type Attachment struct {
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" `
}
// Epic represents the epic to which an issue is associated
// Not that this struct does not process the returned "color" value
type Epic struct {
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" `
}
2021-10-06 17:33:14 +00:00
// IssueFields represents single fields of a Jira issue.
// Every Jira issue has several fields attached.
2018-06-02 00:37:01 +00:00
type IssueFields struct {
// TODO Missing fields
// * "workratio": -1,
// * "lastViewed": null,
// * "environment": null,
2020-09-14 18:31:56 +00: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" `
Unknowns tcontainer . MarshalMap
2018-06-02 00:37:01 +00:00
}
// MarshalJSON is a custom JSON marshal function for the IssueFields structs.
2021-10-06 17:33:14 +00:00
// It handles Jira custom fields and maps those from / to "Unknowns" key.
2018-06-02 00:37:01 +00:00
func ( i * IssueFields ) MarshalJSON ( ) ( [ ] byte , error ) {
m := structs . Map ( i )
unknowns , okay := m [ "Unknowns" ]
if okay {
// if unknowns present, shift all key value from unknown to a level up
for key , value := range unknowns . ( tcontainer . MarshalMap ) {
m [ key ] = value
}
delete ( m , "Unknowns" )
}
return json . Marshal ( m )
}
// UnmarshalJSON is a custom JSON marshal function for the IssueFields structs.
2021-10-06 17:33:14 +00:00
// It handles Jira custom fields and maps those from / to "Unknowns" key.
2018-06-02 00:37:01 +00: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 {
2021-10-06 17:33:14 +00:00
return fmt . Errorf ( "no tags options found for %s" , field . Name )
2018-06-02 00:37:01 +00:00
}
// 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
}
2021-10-06 17:33:14 +00:00
// IssueRenderedFields represents rendered fields of a Jira issue.
2020-09-14 18:31:56 +00:00
// 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" `
Description string ` json:"description,omitempty" structs:"description,omitempty" `
}
2021-10-06 17:33:14 +00:00
// IssueType represents a type of a Jira issue.
2018-06-02 00:37:01 +00:00
// Typical types are "Request", "Bug", "Story", ...
type IssueType struct {
Self string ` json:"self,omitempty" structs:"self,omitempty" `
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" `
}
2021-10-06 17:33:14 +00:00
// Watches represents a type of how many and which user are "observing" a Jira issue to track the status / updates.
2018-06-02 00:37:01 +00:00
type Watches struct {
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" `
2021-10-06 17:33:14 +00:00
AccountID string ` json:"accountId,omitempty" structs:"accountId,omitempty" `
2018-06-02 00:37:01 +00:00
DisplayName string ` json:"displayName,omitempty" structs:"displayName,omitempty" `
Active bool ` json:"active,omitempty" structs:"active,omitempty" `
}
// AvatarUrls represents different dimensions of avatars / images
type AvatarUrls struct {
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" `
}
2021-10-06 17:33:14 +00:00
// Component represents a "component" of a Jira issue.
// Components can be user defined in every Jira instance.
2018-06-02 00:37:01 +00:00
type Component struct {
2021-10-06 17:33:14 +00: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" `
Description string ` json:"description,omitempty" structs:"description,omitempty" `
2018-06-02 00:37:01 +00:00
}
2021-10-06 17:33:14 +00:00
// Progress represents the progress of a Jira issue.
2018-06-02 00:37:01 +00:00
type Progress struct {
Progress int ` json:"progress" structs:"progress" `
Total int ` json:"total" structs:"total" `
2020-09-14 18:31:56 +00:00
Percent int ` json:"percent" structs:"percent" `
2018-06-02 00:37:01 +00:00
}
2021-10-06 17:33:14 +00:00
// Parent represents the parent of a Jira issue, to be used with subtask issue types.
2018-06-02 00:37:01 +00:00
type Parent struct {
ID string ` json:"id,omitempty" structs:"id" `
Key string ` json:"key,omitempty" structs:"key" `
}
2021-10-06 17:33:14 +00:00
// Time represents the Time definition of Jira as a time.Time of go
2018-06-02 00:37:01 +00:00
type Time time . Time
2020-09-14 18:31:56 +00:00
func ( t Time ) Equal ( u Time ) bool {
return time . Time ( t ) . Equal ( time . Time ( u ) )
}
2021-10-06 17:33:14 +00:00
// Date represents the Date definition of Jira as a time.Time of go
2018-06-02 00:37:01 +00:00
type Date time . Time
// Wrapper struct for search result
type transitionResult struct {
Transitions [ ] Transition ` json:"transitions" structs:"transitions" `
}
2021-10-06 17:33:14 +00:00
// Transition represents an issue transition in Jira
2018-06-02 00:37:01 +00:00
type Transition struct {
ID string ` json:"id" structs:"id" `
Name string ` json:"name" structs:"name" `
To Status ` json:"to" structs:"status" `
Fields map [ string ] TransitionField ` json:"fields" structs:"fields" `
}
// TransitionField represents the value of one Transition
type TransitionField struct {
Required bool ` json:"required" structs:"required" `
}
// CreateTransitionPayload is used for creating new issue transitions
type CreateTransitionPayload struct {
Transition TransitionPayload ` json:"transition" structs:"transition" `
Fields TransitionPayloadFields ` json:"fields" structs:"fields" `
}
// TransitionPayload represents the request payload of Transition calls like DoTransition
type TransitionPayload struct {
ID string ` json:"id" structs:"id" `
}
// TransitionPayloadFields represents the fields that can be set when executing a transition
type TransitionPayloadFields struct {
Resolution * Resolution ` json:"resolution,omitempty" structs:"resolution,omitempty" `
}
// Option represents an option value in a SelectList or MultiSelect
// custom issue field
type Option struct {
Value string ` json:"value" structs:"value" `
}
2021-10-06 17:33:14 +00:00
// UnmarshalJSON will transform the Jira time into a time.Time
// during the transformation of the Jira JSON response
2018-06-02 00:37:01 +00:00
func ( t * Time ) UnmarshalJSON ( b [ ] byte ) error {
// Ignore null, like in the main JSON package.
if string ( b ) == "null" {
return nil
}
ti , err := time . Parse ( "\"2006-01-02T15:04:05.999-0700\"" , string ( b ) )
if err != nil {
return err
}
* t = Time ( ti )
return nil
}
2021-10-06 17:33:14 +00:00
// MarshalJSON will transform the time.Time into a Jira time
// during the creation of a Jira request
2020-09-14 18:31:56 +00:00
func ( t Time ) MarshalJSON ( ) ( [ ] byte , error ) {
2021-10-06 17:33:14 +00:00
return [ ] byte ( time . Time ( t ) . Format ( "\"2006-01-02T15:04:05.000-0700\"" ) ) , nil
2020-09-14 18:31:56 +00:00
}
2021-10-06 17:33:14 +00:00
// UnmarshalJSON will transform the Jira date into a time.Time
// during the transformation of the Jira JSON response
2018-06-02 00:37:01 +00:00
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
}
// MarshalJSON will transform the Date object into a short
2021-10-06 17:33:14 +00:00
// date string as Jira expects during the creation of a
// Jira request
2018-06-02 00:37:01 +00:00
func ( t Date ) MarshalJSON ( ) ( [ ] byte , error ) {
time := time . Time ( t )
return [ ] byte ( time . Format ( "\"2006-01-02\"" ) ) , nil
}
2021-10-06 17:33:14 +00:00
// Worklog represents the work log of a Jira issue.
2018-06-02 00:37:01 +00:00
// One Worklog contains zero or n WorklogRecords
2021-10-06 17:33:14 +00:00
// Jira Wiki: https://confluence.atlassian.com/jira/logging-work-on-an-issue-185729605.html
2018-06-02 00:37:01 +00:00
type Worklog struct {
StartAt int ` json:"startAt" structs:"startAt" `
MaxResults int ` json:"maxResults" structs:"maxResults" `
Total int ` json:"total" structs:"total" `
Worklogs [ ] WorklogRecord ` json:"worklogs" structs:"worklogs" `
}
// WorklogRecord represents one entry of a Worklog
type WorklogRecord struct {
2020-09-14 18:31:56 +00:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Author * User ` json:"author,omitempty" structs:"author,omitempty" `
UpdateAuthor * User ` json:"updateAuthor,omitempty" structs:"updateAuthor,omitempty" `
Comment string ` json:"comment,omitempty" structs:"comment,omitempty" `
Created * Time ` json:"created,omitempty" structs:"created,omitempty" `
Updated * Time ` json:"updated,omitempty" structs:"updated,omitempty" `
Started * Time ` json:"started,omitempty" structs:"started,omitempty" `
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" `
Properties [ ] EntityProperty ` json:"properties,omitempty" `
}
type EntityProperty struct {
Key string ` json:"key" `
Value interface { } ` json:"value" `
2018-06-02 00:37:01 +00:00
}
2021-10-06 17:33:14 +00:00
// TimeTracking represents the timetracking fields of a Jira issue.
2018-06-02 00:37:01 +00:00
type TimeTracking struct {
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" `
}
// Subtasks represents all issues of a parent issue.
type Subtasks struct {
ID string ` json:"id" structs:"id" `
Key string ` json:"key" structs:"key" `
Self string ` json:"self" structs:"self" `
Fields IssueFields ` json:"fields" structs:"fields" `
}
2021-10-06 17:33:14 +00:00
// IssueLink represents a link between two issues in Jira.
2018-06-02 00:37:01 +00:00
type IssueLink struct {
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" `
}
2021-10-06 17:33:14 +00:00
// IssueLinkType represents a type of a link between to issues in Jira.
2018-06-02 00:37:01 +00:00
// Typical issue link types are "Related to", "Duplicate", "Is blocked by", etc.
type IssueLinkType struct {
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" `
}
// Comments represents a list of Comment.
type Comments struct {
Comments [ ] * Comment ` json:"comments,omitempty" structs:"comments,omitempty" `
}
2021-10-06 17:33:14 +00:00
// Comment represents a comment by a person to an issue in Jira.
2018-06-02 00:37:01 +00:00
type Comment struct {
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
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" `
}
// FixVersion represents a software release in which an issue is fixed.
type FixVersion struct {
2020-09-14 18:31:56 +00:00
Self string ` json:"self,omitempty" structs:"self,omitempty" `
2018-06-02 00:37:01 +00:00
ID string ` json:"id,omitempty" structs:"id,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
2020-09-14 18:31:56 +00:00
Description string ` json:"description,omitempty" structs:"description,omitempty" `
Archived * bool ` json:"archived,omitempty" structs:"archived,omitempty" `
2018-06-02 00:37:01 +00:00
Released * bool ` json:"released,omitempty" structs:"released,omitempty" `
2020-09-14 18:31:56 +00:00
ReleaseDate string ` json:"releaseDate,omitempty" structs:"releaseDate,omitempty" `
2018-06-02 00:37:01 +00:00
UserReleaseDate string ` json:"userReleaseDate,omitempty" structs:"userReleaseDate,omitempty" `
2020-09-14 18:31:56 +00: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" `
2018-06-02 00:37:01 +00:00
}
2020-09-14 18:31:56 +00:00
// AffectsVersion represents a software release which is affected by an issue.
type AffectsVersion Version
2018-06-02 00:37:01 +00:00
// CommentVisibility represents he visibility of a comment.
// E.g. Type could be "role" and Value "Administrators"
type CommentVisibility struct {
Type string ` json:"type,omitempty" structs:"type,omitempty" `
Value string ` json:"value,omitempty" structs:"value,omitempty" `
}
// SearchOptions specifies the optional parameters to various List methods that
// support pagination.
2021-10-06 17:33:14 +00:00
// Pagination is used for the Jira REST APIs to conserve server resources and limit
2018-06-02 00:37:01 +00:00
// 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
type SearchOptions struct {
// 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" `
// Expand: Expand specific sections in the returned issues
Expand string ` url:"expand,omitempty" `
Fields [ ] string
// ValidateQuery: The validateQuery param offers control over whether to validate and how strictly to treat the validation. Default: strict.
ValidateQuery string ` url:"validateQuery,omitempty" `
}
// searchResult is only a small wrapper around the Search (with JQL) method
// to be able to parse the results
type searchResult struct {
Issues [ ] Issue ` json:"issues" structs:"issues" `
StartAt int ` json:"startAt" structs:"startAt" `
MaxResults int ` json:"maxResults" structs:"maxResults" `
Total int ` json:"total" structs:"total" `
}
// 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
FieldsByKeys bool ` url:"fieldsByKeys,omitempty" `
UpdateHistory bool ` url:"updateHistory,omitempty" `
ProjectKeys string ` url:"projectKeys,omitempty" `
}
2020-09-14 18:31:56 +00:00
// GetWorklogsQueryOptions specifies the optional parameters for the Get Worklogs method
type GetWorklogsQueryOptions struct {
2021-10-06 17:33:14 +00:00
StartAt int64 ` url:"startAt,omitempty" `
MaxResults int32 ` url:"maxResults,omitempty" `
StartedAfter int64 ` url:"startedAfter,omitempty" `
Expand string ` url:"expand,omitempty" `
2020-09-14 18:31:56 +00:00
}
type AddWorklogQueryOptions struct {
NotifyUsers bool ` url:"notifyUsers,omitempty" `
AdjustEstimate string ` url:"adjustEstimate,omitempty" `
NewEstimate string ` url:"newEstimate,omitempty" `
ReduceBy string ` url:"reduceBy,omitempty" `
Expand string ` url:"expand,omitempty" `
OverrideEditableFlag bool ` url:"overrideEditableFlag,omitempty" `
}
2021-10-06 17:33:14 +00:00
// CustomFields represents custom fields of Jira
// This can heavily differ between Jira instances
2018-06-02 00:37:01 +00:00
type CustomFields map [ string ] string
2020-09-14 18:31:56 +00:00
// RemoteLink represents remote links which linked to issues
type RemoteLink struct {
ID int ` json:"id,omitempty" structs:"id,omitempty" `
Self string ` json:"self,omitempty" structs:"self,omitempty" `
GlobalID string ` json:"globalId,omitempty" structs:"globalId,omitempty" `
Application * RemoteLinkApplication ` json:"application,omitempty" structs:"application,omitempty" `
Relationship string ` json:"relationship,omitempty" structs:"relationship,omitempty" `
Object * RemoteLinkObject ` json:"object,omitempty" structs:"object,omitempty" `
}
// RemoteLinkApplication represents remote links application
type RemoteLinkApplication struct {
Type string ` json:"type,omitempty" structs:"type,omitempty" `
Name string ` json:"name,omitempty" structs:"name,omitempty" `
}
// RemoteLinkObject represents remote link object itself
type RemoteLinkObject struct {
URL string ` json:"url,omitempty" structs:"url,omitempty" `
Title string ` json:"title,omitempty" structs:"title,omitempty" `
Summary string ` json:"summary,omitempty" structs:"summary,omitempty" `
Icon * RemoteLinkIcon ` json:"icon,omitempty" structs:"icon,omitempty" `
Status * RemoteLinkStatus ` json:"status,omitempty" structs:"status,omitempty" `
}
// RemoteLinkIcon represents icon displayed next to link
type RemoteLinkIcon struct {
Url16x16 string ` json:"url16x16,omitempty" structs:"url16x16,omitempty" `
Title string ` json:"title,omitempty" structs:"title,omitempty" `
Link string ` json:"link,omitempty" structs:"link,omitempty" `
}
// RemoteLinkStatus if the link is a resolvable object (issue, epic) - the structure represent its status
type RemoteLinkStatus struct {
2021-10-06 17:33:14 +00:00
Resolved bool ` json:"resolved,omitempty" structs:"resolved,omitempty" `
Icon * RemoteLinkIcon ` json:"icon,omitempty" structs:"icon,omitempty" `
2020-09-14 18:31:56 +00:00
}
2021-10-06 17:33:14 +00:00
// GetWithContext returns a full representation of the issue for the given issue key.
// Jira will attempt to identify the issue by the issueIdOrKey path parameter.
2018-06-02 00:37:01 +00:00
// This can be an issue id, or an issue key.
2021-10-06 17:33:14 +00:00
// 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.
2018-06-02 00:37:01 +00:00
//
// The given options will be appended to the query string
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-getIssue
func ( s * IssueService ) GetWithContext ( ctx context . Context , issueID string , options * GetQueryOptions ) ( * Issue , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2018-06-02 00:37:01 +00:00
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
}
2021-10-06 17:33:14 +00:00
// Get wraps GetWithContext using the background context.
func ( s * IssueService ) Get ( issueID string , options * GetQueryOptions ) ( * Issue , * Response , error ) {
return s . GetWithContext ( context . Background ( ) , issueID , options )
}
// DownloadAttachmentWithContext returns a Response of an attachment for a given attachmentID.
2018-06-02 00:37:01 +00:00
// The attachment is in the Response.Body of the response.
// This is an io.ReadCloser.
// The caller should close the resp.Body.
2021-10-06 17:33:14 +00:00
func ( s * IssueService ) DownloadAttachmentWithContext ( ctx context . Context , attachmentID string ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "secure/attachment/%s/" , attachmentID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
jerr := NewJiraError ( resp , err )
return resp , jerr
}
return resp , nil
}
2021-10-06 17:33:14 +00:00
// DownloadAttachment wraps DownloadAttachmentWithContext using the background context.
func ( s * IssueService ) DownloadAttachment ( attachmentID string ) ( * Response , error ) {
return s . DownloadAttachmentWithContext ( context . Background ( ) , attachmentID )
}
// PostAttachmentWithContext uploads r (io.Reader) as an attachment to a given issueID
func ( s * IssueService ) PostAttachmentWithContext ( ctx context . Context , issueID string , r io . Reader , attachmentName string ) ( * [ ] Attachment , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/attachments" , issueID )
b := new ( bytes . Buffer )
writer := multipart . NewWriter ( b )
fw , err := writer . CreateFormFile ( "file" , attachmentName )
if err != nil {
return nil , nil , err
}
if r != nil {
// Copy the file
if _ , err = io . Copy ( fw , r ) ; err != nil {
return nil , nil , err
}
}
writer . Close ( )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewMultiPartRequestWithContext ( ctx , "POST" , apiEndpoint , b )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
req . Header . Set ( "Content-Type" , writer . FormDataContentType ( ) )
// PostAttachment response returns a JSON array (as multiple attachments can be posted)
attachment := new ( [ ] Attachment )
resp , err := s . client . Do ( req , attachment )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return attachment , resp , nil
}
2021-10-06 17:33:14 +00:00
// PostAttachment wraps PostAttachmentWithContext using the background context.
func ( s * IssueService ) PostAttachment ( issueID string , r io . Reader , attachmentName string ) ( * [ ] Attachment , * Response , error ) {
return s . PostAttachmentWithContext ( context . Background ( ) , issueID , r , attachmentName )
}
// DeleteAttachmentWithContext deletes an attachment of a given attachmentID
func ( s * IssueService ) DeleteAttachmentWithContext ( ctx context . Context , attachmentID string ) ( * Response , error ) {
2020-09-14 18:31:56 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/attachment/%s" , attachmentID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "DELETE" , apiEndpoint , nil )
2020-09-14 18:31:56 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
jerr := NewJiraError ( resp , err )
return resp , jerr
}
return resp , nil
}
2021-10-06 17:33:14 +00:00
// DeleteAttachment wraps DeleteAttachmentWithContext using the background context.
func ( s * IssueService ) DeleteAttachment ( attachmentID string ) ( * Response , error ) {
return s . DeleteAttachmentWithContext ( context . Background ( ) , attachmentID )
}
// DeleteLinkWithContext deletes a link of a given linkID
func ( s * IssueService ) DeleteLinkWithContext ( ctx context . Context , linkID string ) ( * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issueLink/%s" , linkID )
req , err := s . client . NewRequestWithContext ( ctx , "DELETE" , apiEndpoint , nil )
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
jerr := NewJiraError ( resp , err )
return resp , jerr
}
return resp , nil
}
// DeleteLink wraps DeleteLinkWithContext using the background context.
func ( s * IssueService ) DeleteLink ( linkID string ) ( * Response , error ) {
return s . DeleteLinkWithContext ( context . Background ( ) , linkID )
}
// GetWorklogsWithContext gets all the worklogs for an issue.
2018-06-02 00:37:01 +00:00
// 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
2021-10-06 17:33:14 +00:00
func ( s * IssueService ) GetWorklogsWithContext ( ctx context . Context , issueID string , options ... func ( * http . Request ) error ) ( * Worklog , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/worklog" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
2020-09-14 18:31:56 +00:00
for _ , option := range options {
err = option ( req )
if err != nil {
return nil , nil , err
}
}
2018-06-02 00:37:01 +00:00
v := new ( Worklog )
resp , err := s . client . Do ( req , v )
return v , resp , err
}
2021-10-06 17:33:14 +00:00
// GetWorklogs wraps GetWorklogsWithContext using the background context.
func ( s * IssueService ) GetWorklogs ( issueID string , options ... func ( * http . Request ) error ) ( * Worklog , * Response , error ) {
return s . GetWorklogsWithContext ( context . Background ( ) , issueID , options ... )
}
2020-09-14 18:31:56 +00:00
// Applies query options to http request.
// This helper is meant to be used with all "QueryOptions" structs.
func WithQueryOptions ( options interface { } ) func ( * http . Request ) error {
q , err := query . Values ( options )
if err != nil {
return func ( * http . Request ) error {
return err
}
}
return func ( r * http . Request ) error {
r . URL . RawQuery = q . Encode ( )
return nil
}
}
2021-10-06 17:33:14 +00:00
// CreateWithContext creates an issue or a sub-task from a JSON representation.
2018-06-02 00:37:01 +00:00
// 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.
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-createIssues
func ( s * IssueService ) CreateWithContext ( ctx context . Context , issue * Issue ) ( * Issue , * Response , error ) {
2020-09-14 18:31:56 +00:00
apiEndpoint := "rest/api/2/issue"
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , issue )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
// incase of error return the resp for further inspection
return nil , resp , err
}
responseIssue := new ( Issue )
defer resp . Body . Close ( )
data , err := ioutil . ReadAll ( resp . Body )
if err != nil {
2021-10-06 17:33:14 +00:00
return nil , resp , fmt . Errorf ( "could not read the returned data" )
2018-06-02 00:37:01 +00:00
}
err = json . Unmarshal ( data , responseIssue )
if err != nil {
2021-10-06 17:33:14 +00:00
return nil , resp , fmt . Errorf ( "could not unmarshall the data into struct" )
2018-06-02 00:37:01 +00:00
}
return responseIssue , resp , nil
}
2021-10-06 17:33:14 +00:00
// Create wraps CreateWithContext using the background context.
func ( s * IssueService ) Create ( issue * Issue ) ( * Issue , * Response , error ) {
return s . CreateWithContext ( context . Background ( ) , issue )
}
// UpdateWithOptionsWithContext updates an issue from a JSON representation,
// while also specifying query params. The issue is found by key.
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue-editIssue
func ( s * IssueService ) UpdateWithOptionsWithContext ( ctx context . Context , issue * Issue , opts * UpdateQueryOptions ) ( * Issue , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%v" , issue . Key )
2020-09-14 18:31:56 +00:00
url , err := addOptions ( apiEndpoint , opts )
if err != nil {
return nil , nil , err
}
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "PUT" , url , issue )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
// 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
}
2021-10-06 17:33:14 +00:00
// UpdateWithOptions wraps UpdateWithOptionsWithContext using the background context.
func ( s * IssueService ) UpdateWithOptions ( issue * Issue , opts * UpdateQueryOptions ) ( * Issue , * Response , error ) {
return s . UpdateWithOptionsWithContext ( context . Background ( ) , issue , opts )
}
// UpdateWithContext updates an issue from a JSON representation. The issue is found by key.
2020-09-14 18:31:56 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue-editIssue
func ( s * IssueService ) UpdateWithContext ( ctx context . Context , issue * Issue ) ( * Issue , * Response , error ) {
return s . UpdateWithOptionsWithContext ( ctx , issue , nil )
}
// Update wraps UpdateWithContext using the background context.
2020-09-14 18:31:56 +00:00
func ( s * IssueService ) Update ( issue * Issue ) ( * Issue , * Response , error ) {
2021-10-06 17:33:14 +00:00
return s . UpdateWithContext ( context . Background ( ) , issue )
2020-09-14 18:31:56 +00:00
}
2021-10-06 17:33:14 +00:00
// UpdateIssueWithContext updates an issue from a JSON representation. The issue is found by key.
2018-06-02 00:37:01 +00:00
//
// https://docs.atlassian.com/jira/REST/7.4.0/#api/2/issue-editIssue
2021-10-06 17:33:14 +00:00
func ( s * IssueService ) UpdateIssueWithContext ( ctx context . Context , jiraID string , data map [ string ] interface { } ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%v" , jiraID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "PUT" , apiEndpoint , data )
2018-06-02 00:37:01 +00:00
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
}
2021-10-06 17:33:14 +00:00
// UpdateIssue wraps UpdateIssueWithContext using the background context.
func ( s * IssueService ) UpdateIssue ( jiraID string , data map [ string ] interface { } ) ( * Response , error ) {
return s . UpdateIssueWithContext ( context . Background ( ) , jiraID , data )
}
// AddCommentWithContext adds a new comment to issueID.
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-addComment
func ( s * IssueService ) AddCommentWithContext ( ctx context . Context , issueID string , comment * Comment ) ( * Comment , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , comment )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
responseComment := new ( Comment )
resp , err := s . client . Do ( req , responseComment )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return responseComment , resp , nil
}
2021-10-06 17:33:14 +00:00
// AddComment wraps AddCommentWithContext using the background context.
func ( s * IssueService ) AddComment ( issueID string , comment * Comment ) ( * Comment , * Response , error ) {
return s . AddCommentWithContext ( context . Background ( ) , issueID , comment )
}
// UpdateCommentWithContext updates the body of a comment, identified by comment.ID, on the issueID.
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/cloud/#api/2/issue/{issueIdOrKey}/comment-updateComment
func ( s * IssueService ) UpdateCommentWithContext ( ctx context . Context , issueID string , comment * Comment ) ( * Comment , * Response , error ) {
2018-06-02 00:37:01 +00:00
reqBody := struct {
Body string ` json:"body" `
} {
Body : comment . Body ,
}
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment/%s" , issueID , comment . ID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "PUT" , apiEndpoint , reqBody )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
responseComment := new ( Comment )
resp , err := s . client . Do ( req , responseComment )
if err != nil {
return nil , resp , err
}
return responseComment , resp , nil
}
2021-10-06 17:33:14 +00:00
// UpdateComment wraps UpdateCommentWithContext using the background context.
func ( s * IssueService ) UpdateComment ( issueID string , comment * Comment ) ( * Comment , * Response , error ) {
return s . UpdateCommentWithContext ( context . Background ( ) , issueID , comment )
}
// DeleteCommentWithContext Deletes a comment from an issueID.
2020-09-14 18:31:56 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://developer.atlassian.com/cloud/jira/platform/rest/v3/#api-api-3-issue-issueIdOrKey-comment-id-delete
func ( s * IssueService ) DeleteCommentWithContext ( ctx context . Context , issueID , commentID string ) error {
2020-09-14 18:31:56 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/comment/%s" , issueID , commentID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "DELETE" , apiEndpoint , nil )
2020-09-14 18:31:56 +00:00
if err != nil {
return err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
jerr := NewJiraError ( resp , err )
return jerr
}
return nil
}
2021-10-06 17:33:14 +00:00
// DeleteComment wraps DeleteCommentWithContext using the background context.
func ( s * IssueService ) DeleteComment ( issueID , commentID string ) error {
return s . DeleteCommentWithContext ( context . Background ( ) , issueID , commentID )
}
// AddWorklogRecordWithContext adds a new worklog record to issueID.
2018-06-02 00:37:01 +00:00
//
// https://developer.atlassian.com/cloud/jira/platform/rest/#api-api-2-issue-issueIdOrKey-worklog-post
2021-10-06 17:33:14 +00:00
func ( s * IssueService ) AddWorklogRecordWithContext ( ctx context . Context , issueID string , record * WorklogRecord , options ... func ( * http . Request ) error ) ( * WorklogRecord , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/worklog" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , record )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
2020-09-14 18:31:56 +00:00
for _ , option := range options {
err = option ( req )
if err != nil {
return nil , nil , err
}
}
responseRecord := new ( WorklogRecord )
resp , err := s . client . Do ( req , responseRecord )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return responseRecord , resp , nil
}
2021-10-06 17:33:14 +00:00
// AddWorklogRecord wraps AddWorklogRecordWithContext using the background context.
func ( s * IssueService ) AddWorklogRecord ( issueID string , record * WorklogRecord , options ... func ( * http . Request ) error ) ( * WorklogRecord , * Response , error ) {
return s . AddWorklogRecordWithContext ( context . Background ( ) , issueID , record , options ... )
}
// UpdateWorklogRecordWithContext updates a worklog record.
2020-09-14 18:31:56 +00:00
//
// https://docs.atlassian.com/software/jira/docs/api/REST/7.1.2/#api/2/issue-updateWorklog
2021-10-06 17:33:14 +00:00
func ( s * IssueService ) UpdateWorklogRecordWithContext ( ctx context . Context , issueID , worklogID string , record * WorklogRecord , options ... func ( * http . Request ) error ) ( * WorklogRecord , * Response , error ) {
2020-09-14 18:31:56 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/worklog/%s" , issueID , worklogID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "PUT" , apiEndpoint , record )
2020-09-14 18:31:56 +00:00
if err != nil {
return nil , nil , err
}
for _ , option := range options {
err = option ( req )
if err != nil {
return nil , nil , err
}
}
2018-06-02 00:37:01 +00:00
responseRecord := new ( WorklogRecord )
resp , err := s . client . Do ( req , responseRecord )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return responseRecord , resp , nil
}
2021-10-06 17:33:14 +00:00
// UpdateWorklogRecord wraps UpdateWorklogRecordWithContext using the background context.
func ( s * IssueService ) UpdateWorklogRecord ( issueID , worklogID string , record * WorklogRecord , options ... func ( * http . Request ) error ) ( * WorklogRecord , * Response , error ) {
return s . UpdateWorklogRecordWithContext ( context . Background ( ) , issueID , worklogID , record , options ... )
}
// AddLinkWithContext adds a link between two issues.
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issueLink
func ( s * IssueService ) AddLinkWithContext ( ctx context . Context , issueLink * IssueLink ) ( * Response , error ) {
apiEndpoint := "rest/api/2/issueLink"
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , issueLink )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2021-10-06 17:33:14 +00:00
// AddLink wraps AddLinkWithContext using the background context.
func ( s * IssueService ) AddLink ( issueLink * IssueLink ) ( * Response , error ) {
return s . AddLinkWithContext ( context . Background ( ) , issueLink )
}
// SearchWithContext will search for tickets according to the jql
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// 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 ) SearchWithContext ( ctx context . Context , jql string , options * SearchOptions ) ( [ ] Issue , * Response , error ) {
u := url . URL {
Path : "rest/api/2/search" ,
}
uv := url . Values { }
if jql != "" {
uv . Add ( "jql" , jql )
}
if options != nil {
2020-09-14 18:31:56 +00:00
if options . StartAt != 0 {
2021-10-06 17:33:14 +00:00
uv . Add ( "startAt" , strconv . Itoa ( options . StartAt ) )
2020-09-14 18:31:56 +00:00
}
if options . MaxResults != 0 {
2021-10-06 17:33:14 +00:00
uv . Add ( "maxResults" , strconv . Itoa ( options . MaxResults ) )
2020-09-14 18:31:56 +00:00
}
if options . Expand != "" {
2021-10-06 17:33:14 +00:00
uv . Add ( "expand" , options . Expand )
2020-09-14 18:31:56 +00:00
}
if strings . Join ( options . Fields , "," ) != "" {
2021-10-06 17:33:14 +00:00
uv . Add ( "fields" , strings . Join ( options . Fields , "," ) )
2020-09-14 18:31:56 +00:00
}
if options . ValidateQuery != "" {
2021-10-06 17:33:14 +00:00
uv . Add ( "validateQuery" , options . ValidateQuery )
2020-09-14 18:31:56 +00:00
}
2018-06-02 00:37:01 +00:00
}
2021-10-06 17:33:14 +00:00
u . RawQuery = uv . Encode ( )
req , err := s . client . NewRequestWithContext ( ctx , "GET" , u . String ( ) , nil )
2018-06-02 00:37:01 +00:00
if err != nil {
return [ ] Issue { } , nil , err
}
v := new ( searchResult )
resp , err := s . client . Do ( req , v )
if err != nil {
err = NewJiraError ( resp , err )
}
return v . Issues , resp , err
}
2021-10-06 17:33:14 +00:00
// Search wraps SearchWithContext using the background context.
func ( s * IssueService ) Search ( jql string , options * SearchOptions ) ( [ ] Issue , * Response , error ) {
return s . SearchWithContext ( context . Background ( ) , jql , options )
}
// SearchPagesWithContext will get issues from all pages in a search
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// 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 ) SearchPagesWithContext ( ctx context . Context , jql string , options * SearchOptions , f func ( Issue ) error ) error {
2018-06-02 00:37:01 +00:00
if options == nil {
options = & SearchOptions {
StartAt : 0 ,
MaxResults : 50 ,
}
}
if options . MaxResults == 0 {
options . MaxResults = 50
}
2021-10-06 17:33:14 +00:00
issues , resp , err := s . SearchWithContext ( ctx , jql , options )
2018-06-02 00:37:01 +00:00
if err != nil {
return err
}
2021-10-06 17:33:14 +00:00
if len ( issues ) == 0 {
return nil
}
2018-06-02 00:37:01 +00:00
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
2021-10-06 17:33:14 +00:00
issues , resp , err = s . SearchWithContext ( ctx , jql , options )
2018-06-02 00:37:01 +00:00
if err != nil {
return err
}
}
}
2021-10-06 17:33:14 +00:00
// SearchPages wraps SearchPagesWithContext using the background context.
func ( s * IssueService ) SearchPages ( jql string , options * SearchOptions , f func ( Issue ) error ) error {
return s . SearchPagesWithContext ( context . Background ( ) , jql , options , f )
}
// GetCustomFieldsWithContext returns a map of customfield_* keys with string values
func ( s * IssueService ) GetCustomFieldsWithContext ( ctx context . Context , issueID string ) ( CustomFields , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
issue := new ( map [ string ] interface { } )
resp , err := s . client . Do ( req , issue )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
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" ) {
if valMap , ok := val . ( map [ string ] interface { } ) ; ok {
if v , ok := valMap [ "value" ] ; ok {
val = v
}
}
cf [ key ] = fmt . Sprint ( val )
}
}
}
return cf , resp , nil
}
2021-10-06 17:33:14 +00:00
// GetCustomFields wraps GetCustomFieldsWithContext using the background context.
func ( s * IssueService ) GetCustomFields ( issueID string ) ( CustomFields , * Response , error ) {
return s . GetCustomFieldsWithContext ( context . Background ( ) , issueID )
}
// GetTransitionsWithContext gets a list of the transitions possible for this issue by the current user,
2018-06-02 00:37:01 +00:00
// along with fields that are required and their types.
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-getTransitions
func ( s * IssueService ) GetTransitionsWithContext ( ctx context . Context , id string ) ( [ ] Transition , * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/transitions?expand=transitions.fields" , id )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , nil , err
}
result := new ( transitionResult )
resp , err := s . client . Do ( req , result )
if err != nil {
err = NewJiraError ( resp , err )
}
return result . Transitions , resp , err
}
2021-10-06 17:33:14 +00:00
// GetTransitions wraps GetTransitionsWithContext using the background context.
func ( s * IssueService ) GetTransitions ( id string ) ( [ ] Transition , * Response , error ) {
return s . GetTransitionsWithContext ( context . Background ( ) , id )
}
// DoTransitionWithContext performs a transition on an issue.
2018-06-02 00:37:01 +00:00
// When performing the transition you can update or set other issue fields.
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-doTransition
func ( s * IssueService ) DoTransitionWithContext ( ctx context . Context , ticketID , transitionID string ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
payload := CreateTransitionPayload {
Transition : TransitionPayload {
ID : transitionID ,
} ,
}
2021-10-06 17:33:14 +00:00
return s . DoTransitionWithPayloadWithContext ( ctx , ticketID , payload )
2018-06-02 00:37:01 +00:00
}
2021-10-06 17:33:14 +00:00
// DoTransition wraps DoTransitionWithContext using the background context.
func ( s * IssueService ) DoTransition ( ticketID , transitionID string ) ( * Response , error ) {
return s . DoTransitionWithContext ( context . Background ( ) , ticketID , transitionID )
}
// DoTransitionWithPayloadWithContext performs a transition on an issue using any payload.
2018-06-02 00:37:01 +00:00
// When performing the transition you can update or set other issue fields.
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-doTransition
func ( s * IssueService ) DoTransitionWithPayloadWithContext ( ctx context . Context , ticketID , payload interface { } ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/transitions" , ticketID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , payload )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2021-10-06 17:33:14 +00:00
// DoTransitionWithPayload wraps DoTransitionWithPayloadWithContext using the background context.
func ( s * IssueService ) DoTransitionWithPayload ( ticketID , payload interface { } ) ( * Response , error ) {
return s . DoTransitionWithPayloadWithContext ( context . Background ( ) , ticketID , payload )
}
2018-06-02 00:37:01 +00: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
// And value is the string value for that particular key.
// 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 {
return nil , fmt . Errorf ( "key %s is not found in the list of fields" , key )
}
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" :
issueFields . Unknowns [ jiraKey ] = [ ] Component { { Name : value } }
case "option" :
issueFields . Unknowns [ jiraKey ] = [ ] map [ string ] string { { "value" : value } }
default :
issueFields . Unknowns [ jiraKey ] = [ ] string { value }
}
case "string" :
issueFields . Unknowns [ jiraKey ] = value
case "date" :
issueFields . Unknowns [ jiraKey ] = value
case "datetime" :
issueFields . Unknowns [ jiraKey ] = value
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 ,
}
case "option" :
issueFields . Unknowns [ jiraKey ] = Option {
Value : value ,
}
default :
2021-10-06 17:33:14 +00:00
return nil , fmt . Errorf ( "unknown issue type encountered: %s for %s" , valueType , key )
2018-06-02 00:37:01 +00:00
}
}
issue . Fields = issueFields
return issue , nil
}
2021-10-06 17:33:14 +00:00
// DeleteWithContext will delete a specified issue.
func ( s * IssueService ) DeleteWithContext ( ctx context . Context , issueID string ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
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 )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "DELETE" , apiEndpoint , content )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
return resp , err
}
2021-10-06 17:33:14 +00:00
// Delete wraps DeleteWithContext using the background context.
func ( s * IssueService ) Delete ( issueID string ) ( * Response , error ) {
return s . DeleteWithContext ( context . Background ( ) , issueID )
}
// GetWatchersWithContext wil return all the users watching/observing the given issue
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-getIssueWatchers
func ( s * IssueService ) GetWatchersWithContext ( ctx context . Context , issueID string ) ( * [ ] User , * Response , error ) {
2018-06-02 00:37:01 +00:00
watchesAPIEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , watchesAPIEndpoint , nil )
2018-06-02 00:37:01 +00: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 { }
for _ , watcher := range watches . Watchers {
2021-10-06 17:33:14 +00:00
var user * User
if watcher . AccountID != "" {
user , resp , err = s . client . User . GetByAccountID ( watcher . AccountID )
if err != nil {
return nil , resp , NewJiraError ( resp , err )
}
2018-06-02 00:37:01 +00:00
}
result = append ( result , * user )
}
return & result , resp , nil
}
2021-10-06 17:33:14 +00:00
// GetWatchers wraps GetWatchersWithContext using the background context.
func ( s * IssueService ) GetWatchers ( issueID string ) ( * [ ] User , * Response , error ) {
return s . GetWatchersWithContext ( context . Background ( ) , issueID )
}
// AddWatcherWithContext adds watcher to the given issue
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-addWatcher
func ( s * IssueService ) AddWatcherWithContext ( ctx context . Context , issueID string , userName string ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndPoint , userName )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2021-10-06 17:33:14 +00:00
// AddWatcher wraps AddWatcherWithContext using the background context.
func ( s * IssueService ) AddWatcher ( issueID string , userName string ) ( * Response , error ) {
return s . AddWatcherWithContext ( context . Background ( ) , issueID , userName )
}
// RemoveWatcherWithContext removes given user from given issue
2018-06-02 00:37:01 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/software/jira/docs/api/REST/latest/#api/2/issue-removeWatcher
func ( s * IssueService ) RemoveWatcherWithContext ( ctx context . Context , issueID string , userName string ) ( * Response , error ) {
2018-06-02 00:37:01 +00:00
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/watchers" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "DELETE" , apiEndPoint , userName )
2018-06-02 00:37:01 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2020-09-14 18:31:56 +00:00
2021-10-06 17:33:14 +00:00
// RemoveWatcher wraps RemoveWatcherWithContext using the background context.
func ( s * IssueService ) RemoveWatcher ( issueID string , userName string ) ( * Response , error ) {
return s . RemoveWatcherWithContext ( context . Background ( ) , issueID , userName )
}
// UpdateAssigneeWithContext updates the user assigned to work on the given issue
2020-09-14 18:31:56 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/software/jira/docs/api/REST/7.10.2/#api/2/issue-assign
func ( s * IssueService ) UpdateAssigneeWithContext ( ctx context . Context , issueID string , assignee * User ) ( * Response , error ) {
2020-09-14 18:31:56 +00:00
apiEndPoint := fmt . Sprintf ( "rest/api/2/issue/%s/assignee" , issueID )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "PUT" , apiEndPoint , assignee )
2020-09-14 18:31:56 +00:00
if err != nil {
return nil , err
}
resp , err := s . client . Do ( req , nil )
if err != nil {
err = NewJiraError ( resp , err )
}
return resp , err
}
2021-10-06 17:33:14 +00:00
// UpdateAssignee wraps UpdateAssigneeWithContext using the background context.
func ( s * IssueService ) UpdateAssignee ( issueID string , assignee * User ) ( * Response , error ) {
return s . UpdateAssigneeWithContext ( context . Background ( ) , issueID , assignee )
}
2020-09-14 18:31:56 +00: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
}
2021-10-06 17:33:14 +00:00
// GetRemoteLinksWithContext gets remote issue links on the issue.
2020-09-14 18:31:56 +00:00
//
2021-10-06 17:33:14 +00:00
// Jira API docs: https://docs.atlassian.com/jira/REST/latest/#api/2/issue-getRemoteIssueLinks
func ( s * IssueService ) GetRemoteLinksWithContext ( ctx context . Context , id string ) ( * [ ] RemoteLink , * Response , error ) {
2020-09-14 18:31:56 +00:00
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/remotelink" , id )
2021-10-06 17:33:14 +00:00
req , err := s . client . NewRequestWithContext ( ctx , "GET" , apiEndpoint , nil )
2020-09-14 18:31:56 +00:00
if err != nil {
return nil , nil , err
}
result := new ( [ ] RemoteLink )
resp , err := s . client . Do ( req , result )
if err != nil {
err = NewJiraError ( resp , err )
}
return result , resp , err
}
2021-10-06 17:33:14 +00:00
// GetRemoteLinks wraps GetRemoteLinksWithContext using the background context.
func ( s * IssueService ) GetRemoteLinks ( id string ) ( * [ ] RemoteLink , * Response , error ) {
return s . GetRemoteLinksWithContext ( context . Background ( ) , id )
}
// AddRemoteLinkWithContext adds a remote link to issueID.
//
// Jira API docs: https://developer.atlassian.com/cloud/jira/platform/rest/v2/#api-rest-api-2-issue-issueIdOrKey-remotelink-post
func ( s * IssueService ) AddRemoteLinkWithContext ( ctx context . Context , issueID string , remotelink * RemoteLink ) ( * RemoteLink , * Response , error ) {
apiEndpoint := fmt . Sprintf ( "rest/api/2/issue/%s/remotelink" , issueID )
req , err := s . client . NewRequestWithContext ( ctx , "POST" , apiEndpoint , remotelink )
if err != nil {
return nil , nil , err
}
responseRemotelink := new ( RemoteLink )
resp , err := s . client . Do ( req , responseRemotelink )
if err != nil {
jerr := NewJiraError ( resp , err )
return nil , resp , jerr
}
return responseRemotelink , resp , nil
}
// AddRemoteLink wraps AddRemoteLinkWithContext using the background context.
func ( s * IssueService ) AddRemoteLink ( issueID string , remotelink * RemoteLink ) ( * RemoteLink , * Response , error ) {
return s . AddRemoteLinkWithContext ( context . Background ( ) , issueID , remotelink )
}