2016-09-20 14:27:54 +02:00
|
|
|
package jira
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
2016-10-03 13:33:46 +02:00
|
|
|
|
|
|
|
"github.com/trivago/tgo/tcontainer"
|
2016-09-20 14:27:54 +02:00
|
|
|
)
|
|
|
|
|
2016-10-23 14:51:29 +02:00
|
|
|
// CreateMetaInfo contains information about fields and their attributed to create a ticket.
|
2016-09-20 14:27:54 +02:00
|
|
|
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
|
2016-10-04 15:22:03 +02:00
|
|
|
IssueTypes []*MetaIssueType `json:"issuetypes,omitempty"`
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
|
|
|
|
2016-10-05 11:02:15 +02:00
|
|
|
// MetaIssueType represents the different issue types a project has.
|
2016-09-20 14:27:54 +02:00
|
|
|
//
|
|
|
|
// 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.
|
2016-10-04 15:22:03 +02:00
|
|
|
type MetaIssueType struct {
|
2017-02-28 10:02:18 +01:00
|
|
|
Self string `json:"self,omitempty"`
|
2016-09-23 16:19:07 +02:00
|
|
|
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"`
|
2017-02-28 10:02:18 +01:00
|
|
|
Expand string `json:"expand,omitempty"`
|
2016-09-23 16:19:07 +02:00
|
|
|
Fields tcontainer.MarshalMap `json:"fields,omitempty"`
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetCreateMeta makes the api call to get the meta information required to create a ticket
|
|
|
|
func (s *IssueService) GetCreateMeta(projectkey string) (*CreateMetaInfo, *Response, error) {
|
|
|
|
|
2017-02-20 16:16:34 +01:00
|
|
|
apiEndpoint := fmt.Sprintf("rest/api/2/issue/createmeta?projectKeys=%s&expand=projects.issuetypes.fields", projectkey)
|
2016-09-20 14:27:54 +02:00
|
|
|
|
|
|
|
req, err := s.client.NewRequest("GET", apiEndpoint, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
2016-10-03 13:33:46 +02:00
|
|
|
|
2016-09-20 14:27:54 +02:00
|
|
|
meta := new(CreateMetaInfo)
|
|
|
|
resp, err := s.client.Do(req, meta)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, resp, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return meta, resp, nil
|
|
|
|
}
|
|
|
|
|
2017-05-01 14:59:27 +02:00
|
|
|
// GetProjectWithName returns a project with "name" from the meta information received. If not found, this returns nil.
|
2017-05-01 15:03:03 +02:00
|
|
|
// The comparison of the name is case insensitive.
|
2016-09-20 14:27:54 +02:00
|
|
|
func (m *CreateMetaInfo) GetProjectWithName(name string) *MetaProject {
|
|
|
|
for _, m := range m.Projects {
|
|
|
|
if strings.ToLower(m.Name) == strings.ToLower(name) {
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-05-01 14:59:27 +02:00
|
|
|
// GetProjectWithKey returns a project with "name" from the meta information received. If not found, this returns nil.
|
2017-05-01 15:03:03 +02:00
|
|
|
// The comparison of the name is case insensitive.
|
2016-10-05 10:45:38 +02:00
|
|
|
func (m *CreateMetaInfo) GetProjectWithKey(key string) *MetaProject {
|
|
|
|
for _, m := range m.Projects {
|
|
|
|
if strings.ToLower(m.Key) == strings.ToLower(key) {
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-10-23 14:51:29 +02:00
|
|
|
// GetIssueTypeWithName returns an IssueType with name from a given MetaProject. If not found, this returns nil.
|
2017-05-01 15:03:03 +02:00
|
|
|
// The comparison of the name is case insensitive
|
2016-10-04 15:22:03 +02:00
|
|
|
func (p *MetaProject) GetIssueTypeWithName(name string) *MetaIssueType {
|
2016-09-20 14:27:54 +02:00
|
|
|
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.
|
2016-12-11 11:55:43 +01:00
|
|
|
// if a field returned by the api was:
|
2016-09-20 14:27:54 +02:00
|
|
|
// "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.
|
2016-10-04 15:22:03 +02:00
|
|
|
func (t *MetaIssueType) GetMandatoryFields() (map[string]string, error) {
|
2016-09-20 14:27:54 +02:00
|
|
|
ret := make(map[string]string)
|
2016-10-23 14:51:29 +02:00
|
|
|
for key := range t.Fields {
|
2016-09-23 16:19:07 +02:00
|
|
|
required, err := t.Fields.Bool(key + "/required")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if required {
|
|
|
|
name, err := t.Fields.String(key + "/name")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
ret[name] = key
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
|
|
|
}
|
2016-09-23 16:19:07 +02:00
|
|
|
return ret, nil
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
2016-10-04 15:22:03 +02:00
|
|
|
func (t *MetaIssueType) GetAllFields() (map[string]string, error) {
|
2016-09-20 14:27:54 +02:00
|
|
|
ret := make(map[string]string)
|
2016-10-23 14:51:29 +02:00
|
|
|
for key := range t.Fields {
|
2016-09-23 16:19:07 +02:00
|
|
|
|
|
|
|
name, err := t.Fields.String(key + "/name")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
ret[name] = key
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
2016-09-23 16:19:07 +02:00
|
|
|
return ret, nil
|
2016-09-20 14:27:54 +02:00
|
|
|
}
|
2016-10-05 18:04:48 +02:00
|
|
|
|
|
|
|
// CheckCompleteAndAvailable checks if the given fields satisfies the mandatory field required to create a issue for the given type
|
|
|
|
// And also if the given fields are available.
|
|
|
|
func (t *MetaIssueType) CheckCompleteAndAvailable(config map[string]string) (bool, error) {
|
|
|
|
mandatory, err := t.GetMandatoryFields()
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
all, err := t.GetAllFields()
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// check templateconfig against mandatory fields
|
|
|
|
for key := range mandatory {
|
|
|
|
if _, okay := config[key]; !okay {
|
|
|
|
var requiredFields []string
|
|
|
|
for name := range mandatory {
|
|
|
|
requiredFields = append(requiredFields, name)
|
|
|
|
}
|
|
|
|
return false, fmt.Errorf("Required field not found in provided jira.fields. Required are: %#v", requiredFields)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// check templateConfig against all fields to verify they are available
|
|
|
|
for key := range config {
|
|
|
|
if _, okay := all[key]; !okay {
|
|
|
|
var availableFields []string
|
|
|
|
for name := range all {
|
|
|
|
availableFields = append(availableFields, name)
|
|
|
|
}
|
|
|
|
return false, fmt.Errorf("Fields in jira.fields are not available in jira. Available are: %#v", availableFields)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
}
|