mirror of
https://github.com/interviewstreet/go-jira.git
synced 2025-01-24 03:16:18 +02:00
Adds metaissue support.
This commit is contained in:
parent
eb07612cbf
commit
fe6b129172
121
metaissue.go
Normal file
121
metaissue.go
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
package jira
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CreateMeta contains information about fields and their attributed to create a ticket.
|
||||||
|
type CreateMetaInfo struct {
|
||||||
|
Expand string `json:"expand,omitempty"`
|
||||||
|
Projects []*MetaProject `json:"projects,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// MetaProject is the meta information about a project returned from createmeta api
|
||||||
|
type MetaProject struct {
|
||||||
|
Expand string `json:"expand,omitempty"`
|
||||||
|
Self string `json:"self, omitempty"`
|
||||||
|
Id string `json:"id,omitempty"`
|
||||||
|
Key string `json:"key,omitempty"`
|
||||||
|
Name string `json:"name,omitempty"`
|
||||||
|
// omitted avatarUrls
|
||||||
|
IssueTypes []*MetaIssueTypes `json:"issuetypes,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// metaIssueTypes represents the different issue types a project has.
|
||||||
|
//
|
||||||
|
// Note: Fields is interface because this is an object which can
|
||||||
|
// have arbitraty keys related to customfields. It is not possible to
|
||||||
|
// expect these for a general way. This will be returning a map.
|
||||||
|
// Further processing must be done depending on what is required.
|
||||||
|
type MetaIssueTypes struct {
|
||||||
|
Self string `json:"expand,omitempty"`
|
||||||
|
Id string `json:"id,omitempty"`
|
||||||
|
Description string `json:"description,omitempty"`
|
||||||
|
IconUrl string `json:"iconurl,omitempty"`
|
||||||
|
Name string `json:"name,omitempty"`
|
||||||
|
Subtasks bool `json:"subtask,omitempty"`
|
||||||
|
Expand string `json:"expand,omitempty"`
|
||||||
|
Fields map[string]interface{} `json:"fields,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCreateMeta makes the api call to get the meta information required to create a ticket
|
||||||
|
func (s *IssueService) GetCreateMeta(projectkey string) (*CreateMetaInfo, *Response, error) {
|
||||||
|
|
||||||
|
apiEndpoint := fmt.Sprintf("/rest/api/2/issue/createmeta?projectKeys=%s&expand=projects.issuetypes.fields", projectkey)
|
||||||
|
|
||||||
|
req, err := s.client.NewRequest("GET", apiEndpoint, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
fmt.Println(req.URL)
|
||||||
|
meta := new(CreateMetaInfo)
|
||||||
|
resp, err := s.client.Do(req, meta)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return meta, resp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetProjectWithName returns a project with "name" from the meta information recieved. If not found, this returns nil.
|
||||||
|
// The comparision of the name is case insensitive.
|
||||||
|
func (m *CreateMetaInfo) GetProjectWithName(name string) *MetaProject {
|
||||||
|
for _, m := range m.Projects {
|
||||||
|
if strings.ToLower(m.Name) == strings.ToLower(name) {
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetIssueWithName returns an IssueType with name from a given MetaProject. If not found, this returns nil.
|
||||||
|
// The comparision of the name is case insensitive
|
||||||
|
func (p *MetaProject) GetIssueTypeWithName(name string) *MetaIssueTypes {
|
||||||
|
for _, m := range p.IssueTypes {
|
||||||
|
if strings.ToLower(m.Name) == strings.ToLower(name) {
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetMandatoryFields returns a map of all the required fields from the MetaIssueTypes.
|
||||||
|
// if a frield returned by the api was:
|
||||||
|
// "customfield_10806": {
|
||||||
|
// "required": true,
|
||||||
|
// "schema": {
|
||||||
|
// "type": "any",
|
||||||
|
// "custom": "com.pyxis.greenhopper.jira:gh-epic-link",
|
||||||
|
// "customId": 10806
|
||||||
|
// },
|
||||||
|
// "name": "Epic Link",
|
||||||
|
// "hasDefaultValue": false,
|
||||||
|
// "operations": [
|
||||||
|
// "set"
|
||||||
|
// ]
|
||||||
|
// }
|
||||||
|
// the returned map would have "Epic Link" as the key and "customfield_10806" as value.
|
||||||
|
// This choice has been made so that the it is easier to generate the create api request later.
|
||||||
|
func (t *MetaIssueTypes) GetMandatoryFields() map[string]string {
|
||||||
|
ret := make(map[string]string)
|
||||||
|
for key, obj := range t.Fields {
|
||||||
|
details := obj.(map[string]interface{})
|
||||||
|
if details["required"] == true {
|
||||||
|
ret[details["name"].(string)] = key
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAllFields returns a map of all the fields for an IssueType. This includes all required and not required.
|
||||||
|
// The key of the returned map is what you see in the form and the value is how it is representated in the jira schema.
|
||||||
|
func (t *MetaIssueTypes) GetAllFields() map[string]string {
|
||||||
|
ret := make(map[string]string)
|
||||||
|
for key, obj := range t.Fields {
|
||||||
|
details := obj.(map[string]interface{})
|
||||||
|
ret[details["name"].(string)] = key
|
||||||
|
}
|
||||||
|
return ret
|
||||||
|
}
|
379
metaissue_test.go
Normal file
379
metaissue_test.go
Normal file
@ -0,0 +1,379 @@
|
|||||||
|
package jira
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestIssueService_GetCreateMeta_Success(t *testing.T) {
|
||||||
|
setup()
|
||||||
|
defer teardown()
|
||||||
|
|
||||||
|
testApiEndpoint := "/rest/api/2/issue/createmeta"
|
||||||
|
|
||||||
|
testMux.HandleFunc(testApiEndpoint, func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
testMethod(t, r, "GET")
|
||||||
|
testRequestURL(t, r, testApiEndpoint)
|
||||||
|
|
||||||
|
fmt.Fprint(w, `{
|
||||||
|
"expand": "projects",
|
||||||
|
"projects": [{
|
||||||
|
"expand": "issuetypes",
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/project/11300",
|
||||||
|
"id": "11300",
|
||||||
|
"key": "SOP",
|
||||||
|
"name": "DSE - Software Operations",
|
||||||
|
"avatarUrls": {
|
||||||
|
"48x48": "https://tasks.trivago.com/secure/projectavatar?pid=11300&avatarId=14405",
|
||||||
|
"24x24": "https://tasks.trivago.com/secure/projectavatar?size=small&pid=11300&avatarId=14405",
|
||||||
|
"16x16": "https://tasks.trivago.com/secure/projectavatar?size=xsmall&pid=11300&avatarId=14405",
|
||||||
|
"32x32": "https://tasks.trivago.com/secure/projectavatar?size=medium&pid=11300&avatarId=14405"
|
||||||
|
},
|
||||||
|
"issuetypes": [{
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/issuetype/6",
|
||||||
|
"id": "6",
|
||||||
|
"description": "An issue which ideally should be able to be completed in one step",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/secure/viewavatar?size=xsmall&avatarId=14006&avatarType=issuetype",
|
||||||
|
"name": "Request",
|
||||||
|
"subtask": false,
|
||||||
|
"expand": "fields",
|
||||||
|
"fields": {
|
||||||
|
"summary": {
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"system": "summary"
|
||||||
|
},
|
||||||
|
"name": "Summary",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"issuetype": {
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "issuetype",
|
||||||
|
"system": "issuetype"
|
||||||
|
},
|
||||||
|
"name": "Issue Type",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
|
||||||
|
],
|
||||||
|
"allowedValues": [{
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/issuetype/6",
|
||||||
|
"id": "6",
|
||||||
|
"description": "An issue which ideally should be able to be completed in one step",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/secure/viewavatar?size=xsmall&avatarId=14006&avatarType=issuetype",
|
||||||
|
"name": "Request",
|
||||||
|
"subtask": false,
|
||||||
|
"avatarId": 14006
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
"components": {
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": "component",
|
||||||
|
"system": "components"
|
||||||
|
},
|
||||||
|
"name": "Component/s",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"add",
|
||||||
|
"set",
|
||||||
|
"remove"
|
||||||
|
],
|
||||||
|
"allowedValues": [{
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14144",
|
||||||
|
"id": "14144",
|
||||||
|
"name": "Build automation",
|
||||||
|
"description": "Jenkins, webhooks, etc."
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14149",
|
||||||
|
"id": "14149",
|
||||||
|
"name": "Caches and noSQL",
|
||||||
|
"description": "Cassandra, Memcached, Redis, Twemproxy, Xcache"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14152",
|
||||||
|
"id": "14152",
|
||||||
|
"name": "Cloud services",
|
||||||
|
"description": "AWS and similiar services"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14147",
|
||||||
|
"id": "14147",
|
||||||
|
"name": "Code quality tools",
|
||||||
|
"description": "Code sniffer, Arqtig, Sonar"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14156",
|
||||||
|
"id": "14156",
|
||||||
|
"name": "Configuration management and provisioning",
|
||||||
|
"description": "Apache/PHP modules, Consul, Salt"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/13606",
|
||||||
|
"id": "13606",
|
||||||
|
"name": "Cronjobs",
|
||||||
|
"description": "Cronjobs in general"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14150",
|
||||||
|
"id": "14150",
|
||||||
|
"name": "Data pipelines and queues",
|
||||||
|
"description": "Gollum, Kafka, RabbitMq"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14159",
|
||||||
|
"id": "14159",
|
||||||
|
"name": "Database",
|
||||||
|
"description": "MySQL related problems"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14314",
|
||||||
|
"id": "14314",
|
||||||
|
"name": "Documentation"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14151",
|
||||||
|
"id": "14151",
|
||||||
|
"name": "Git",
|
||||||
|
"description": "Bitbucket, GitHub, GitLab, Git in general"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14155",
|
||||||
|
"id": "14155",
|
||||||
|
"name": "HTTP services",
|
||||||
|
"description": "CDN, HaProxy, HTTP, Varnish"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14154",
|
||||||
|
"id": "14154",
|
||||||
|
"name": "Job and service scheduling",
|
||||||
|
"description": "Chronos, Docker, Marathon, Mesos"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14158",
|
||||||
|
"id": "14158",
|
||||||
|
"name": "Legacy",
|
||||||
|
"description": "Everything related to legacy"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14157",
|
||||||
|
"id": "14157",
|
||||||
|
"name": "Monitoring",
|
||||||
|
"description": "Collectd, Nagios, Monitoring in general"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14148",
|
||||||
|
"id": "14148",
|
||||||
|
"name": "Other services"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/13602",
|
||||||
|
"id": "13602",
|
||||||
|
"name": "Package management",
|
||||||
|
"description": "Composer, Medusa, Satis"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14145",
|
||||||
|
"id": "14145",
|
||||||
|
"name": "Release",
|
||||||
|
"description": "Directory config, release queries, rewrite rules"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14146",
|
||||||
|
"id": "14146",
|
||||||
|
"name": "Staging systems and VMs",
|
||||||
|
"description": "Stage, QA machines, KVMs,Vagrant"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14153",
|
||||||
|
"id": "14153",
|
||||||
|
"name": "Techblog"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14143",
|
||||||
|
"id": "14143",
|
||||||
|
"name": "Test automation",
|
||||||
|
"description": "Testing infrastructure in general"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/component/14221",
|
||||||
|
"id": "14221",
|
||||||
|
"name": "Zup"
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
"attachment": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": "attachment",
|
||||||
|
"system": "attachment"
|
||||||
|
},
|
||||||
|
"name": "Attachment",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"duedate": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "date",
|
||||||
|
"system": "duedate"
|
||||||
|
},
|
||||||
|
"name": "Due Date",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"description": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "string",
|
||||||
|
"system": "description"
|
||||||
|
},
|
||||||
|
"name": "Description",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"customfield_10806": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "any",
|
||||||
|
"custom": "com.pyxis.greenhopper.jira:gh-epic-link",
|
||||||
|
"customId": 10806
|
||||||
|
},
|
||||||
|
"name": "Epic Link",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"project": {
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "project",
|
||||||
|
"system": "project"
|
||||||
|
},
|
||||||
|
"name": "Project",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
],
|
||||||
|
"allowedValues": [{
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/project/11300",
|
||||||
|
"id": "11300",
|
||||||
|
"key": "SOP",
|
||||||
|
"name": "DSE - Software Operations",
|
||||||
|
"avatarUrls": {
|
||||||
|
"48x48": "https://tasks.trivago.com/secure/projectavatar?pid=11300&avatarId=14405",
|
||||||
|
"24x24": "https://tasks.trivago.com/secure/projectavatar?size=small&pid=11300&avatarId=14405",
|
||||||
|
"16x16": "https://tasks.trivago.com/secure/projectavatar?size=xsmall&pid=11300&avatarId=14405",
|
||||||
|
"32x32": "https://tasks.trivago.com/secure/projectavatar?size=medium&pid=11300&avatarId=14405"
|
||||||
|
},
|
||||||
|
"projectCategory": {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/projectCategory/10100",
|
||||||
|
"id": "10100",
|
||||||
|
"description": "",
|
||||||
|
"name": "Product & Development"
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
"assignee": {
|
||||||
|
"required": true,
|
||||||
|
"schema": {
|
||||||
|
"type": "user",
|
||||||
|
"system": "assignee"
|
||||||
|
},
|
||||||
|
"name": "Assignee",
|
||||||
|
"autoCompleteUrl": "https://tasks.trivago.com/rest/api/latest/user/assignable/search?issueKey=null&username=",
|
||||||
|
"hasDefaultValue": true,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"priority": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "priority",
|
||||||
|
"system": "priority"
|
||||||
|
},
|
||||||
|
"name": "Priority",
|
||||||
|
"hasDefaultValue": true,
|
||||||
|
"operations": [
|
||||||
|
"set"
|
||||||
|
],
|
||||||
|
"allowedValues": [{
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/1",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/blocker.svg",
|
||||||
|
"name": "Immediate",
|
||||||
|
"id": "1"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/2",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/critical.svg",
|
||||||
|
"name": "Urgent",
|
||||||
|
"id": "2"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/3",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/major.svg",
|
||||||
|
"name": "High",
|
||||||
|
"id": "3"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/6",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/moderate.svg",
|
||||||
|
"name": "Moderate",
|
||||||
|
"id": "6"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/4",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/minor.svg",
|
||||||
|
"name": "Normal",
|
||||||
|
"id": "4"
|
||||||
|
}, {
|
||||||
|
"self": "https://tasks.trivago.com/rest/api/2/priority/5",
|
||||||
|
"iconUrl": "https://tasks.trivago.com/images/icons/priorities/trivial.svg",
|
||||||
|
"name": "Low",
|
||||||
|
"id": "5"
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
"labels": {
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"type": "array",
|
||||||
|
"items": "string",
|
||||||
|
"system": "labels"
|
||||||
|
},
|
||||||
|
"name": "Labels",
|
||||||
|
"autoCompleteUrl": "https://tasks.trivago.com/rest/api/1.0/labels/suggest?query=",
|
||||||
|
"hasDefaultValue": false,
|
||||||
|
"operations": [
|
||||||
|
"add",
|
||||||
|
"set",
|
||||||
|
"remove"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
}]
|
||||||
|
}`)
|
||||||
|
})
|
||||||
|
|
||||||
|
issue, _, err := testClient.Issue.GetCreateMeta("SOP")
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Expected nil error but got %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(issue.Projects) != 1 {
|
||||||
|
t.Errorf("Expected 1 project, got %d", len(issue.Projects))
|
||||||
|
}
|
||||||
|
for _, project := range issue.Projects {
|
||||||
|
if len(project.IssueTypes) != 1 {
|
||||||
|
t.Errorf("Expected 1 issueTypes, got %d", len(project.IssueTypes))
|
||||||
|
}
|
||||||
|
for _, issueTypes := range project.IssueTypes {
|
||||||
|
requiredFields := 0
|
||||||
|
fields := issueTypes.Fields
|
||||||
|
for _, value := range fields {
|
||||||
|
for key, value := range value.(map[string]interface{}) {
|
||||||
|
if key == "required" && value == true {
|
||||||
|
requiredFields = requiredFields + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
if requiredFields != 5 {
|
||||||
|
t.Errorf("Expected 5 required fields from Create Meta information, got %d", requiredFields)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user