236 lines
7.9 KiB
Go
Raw Normal View History

package jira
import (
"fmt"
"io"
"strings"
"github.com/andygrunwald/go-jira"
2023-03-10 15:55:54 -07:00
"github.com/trivago/tgo/tcontainer"
2021-09-01 11:43:02 +02:00
"github.com/projectdiscovery/gologger"
"github.com/projectdiscovery/nuclei/v3/pkg/output"
"github.com/projectdiscovery/nuclei/v3/pkg/reporting/exporters/markdown/util"
"github.com/projectdiscovery/nuclei/v3/pkg/reporting/format"
"github.com/projectdiscovery/retryablehttp-go"
)
type Formatter struct {
util.MarkdownFormatter
}
func (jiraFormatter *Formatter) MakeBold(text string) string {
return "*" + text + "*"
}
func (jiraFormatter *Formatter) CreateCodeBlock(title string, content string, _ string) string {
return fmt.Sprintf("\n%s\n{code}\n%s\n{code}\n", jiraFormatter.MakeBold(title), content)
}
func (jiraFormatter *Formatter) CreateTable(headers []string, rows [][]string) (string, error) {
table, err := jiraFormatter.MarkdownFormatter.CreateTable(headers, rows)
if err != nil {
return "", err
}
tableRows := strings.Split(table, "\n")
tableRowsWithoutHeaderSeparator := append(tableRows[:1], tableRows[2:]...)
return strings.Join(tableRowsWithoutHeaderSeparator, "\n"), nil
}
func (jiraFormatter *Formatter) CreateLink(title string, url string) string {
return fmt.Sprintf("[%s|%s]", title, url)
}
// Integration is a client for an issue tracker integration
type Integration struct {
Formatter
jira *jira.Client
options *Options
}
// Options contains the configuration options for jira client
type Options struct {
2021-10-18 20:52:35 +02:00
// Cloud value (optional) is set to true when Jira cloud is used
Cloud bool `yaml:"cloud" json:"cloud"`
2021-10-18 20:52:35 +02:00
// UpdateExisting value (optional) if true, the existing opened issue is updated
UpdateExisting bool `yaml:"update-existing" json:"update_existing"`
2021-10-19 17:15:58 +02:00
// URL is the URL of the jira server
URL string `yaml:"url" json:"url" validate:"required"`
2021-10-19 17:15:58 +02:00
// AccountID is the accountID of the jira user.
AccountID string `yaml:"account-id" json:"account_id" validate:"required"`
2021-10-19 17:15:58 +02:00
// Email is the email of the user for jira instance
Email string `yaml:"email" json:"email" validate:"required,email"`
2021-10-19 17:15:58 +02:00
// Token is the token for jira instance.
Token string `yaml:"token" json:"token" validate:"required"`
2021-10-19 17:15:58 +02:00
// ProjectName is the name of the project.
ProjectName string `yaml:"project-name" json:"project_name" validate:"required"`
2021-10-18 20:52:35 +02:00
// IssueType (optional) is the name of the created issue type
IssueType string `yaml:"issue-type" json:"issue_type"`
2021-10-19 17:15:58 +02:00
// SeverityAsLabel (optional) sends the severity as the label of the created
// issue.
SeverityAsLabel bool `yaml:"severity-as-label" json:"severity_as_label"`
// Severity (optional) is the severity of the issue.
2023-03-10 15:55:54 -07:00
Severity []string `yaml:"severity" json:"severity"`
HttpClient *retryablehttp.Client `yaml:"-" json:"-"`
2023-03-10 15:55:54 -07:00
// for each customfield specified in the configuration options
// we will create a map of customfield name to the value
// that will be used to create the issue
2023-03-20 00:48:39 +01:00
CustomFields map[string]interface{} `yaml:"custom-fields" json:"custom_fields"`
2023-03-10 15:55:54 -07:00
StatusNot string `yaml:"status-not" json:"status_not"`
}
// New creates a new issue tracker integration client based on options.
func New(options *Options) (*Integration, error) {
2021-06-04 13:11:09 +02:00
username := options.Email
if !options.Cloud {
username = options.AccountID
}
tp := jira.BasicAuthTransport{
2021-06-04 13:11:09 +02:00
Username: username,
Password: options.Token,
}
if options.HttpClient != nil {
tp.Transport = options.HttpClient.HTTPClient.Transport
}
jiraClient, err := jira.NewClient(tp.Client(), options.URL)
if err != nil {
return nil, err
}
return &Integration{jira: jiraClient, options: options}, nil
}
// CreateNewIssue creates a new issue in the tracker
func (i *Integration) CreateNewIssue(event *output.ResultEvent) error {
summary := format.Summary(event)
2021-10-18 20:45:46 +02:00
labels := []string{}
2022-03-07 17:18:27 +01:00
severityLabel := fmt.Sprintf("Severity:%s", event.Info.SeverityHolder.Severity.String())
2021-10-19 17:15:58 +02:00
if i.options.SeverityAsLabel && severityLabel != "" {
2021-10-18 20:45:46 +02:00
labels = append(labels, severityLabel)
}
if label := i.options.IssueType; label != "" {
labels = append(labels, label)
}
2023-03-10 15:55:54 -07:00
// for each custom value, take the name of the custom field and
// set the value of the custom field to the value specified in the
// configuration options
customFields := tcontainer.NewMarshalMap()
for name, value := range i.options.CustomFields {
//customFields[name] = map[string]interface{}{"value": value}
if valueMap, ok := value.(map[interface{}]interface{}); ok {
// Iterate over nested map
for nestedName, nestedValue := range valueMap {
2023-03-20 00:48:39 +01:00
fmtNestedValue, ok := nestedValue.(string)
if !ok {
return fmt.Errorf(`couldn't iterate on nested item "%s": %s`, nestedName, nestedValue)
}
if strings.HasPrefix(fmtNestedValue, "$") {
nestedValue = strings.TrimPrefix(fmtNestedValue, "$")
2023-03-10 15:55:54 -07:00
switch nestedValue {
case "CVSSMetrics":
nestedValue = event.Info.Classification.CVSSMetrics
case "CVEID":
nestedValue = event.Info.Classification.CVEID
case "CWEID":
nestedValue = event.Info.Classification.CWEID
case "CVSSScore":
nestedValue = event.Info.Classification.CVSSScore
case "Host":
nestedValue = event.Host
case "Severity":
nestedValue = event.Info.SeverityHolder
case "Name":
nestedValue = event.Info.Name
}
}
switch nestedName {
case "id":
customFields[name] = map[string]interface{}{"id": nestedValue}
case "name":
customFields[name] = map[string]interface{}{"value": nestedValue}
case "freeform":
customFields[name] = nestedValue
}
}
}
}
fields := &jira.IssueFields{
Description: format.CreateReportDescription(event, i),
2023-03-10 15:55:54 -07:00
Unknowns: customFields,
Type: jira.IssueType{Name: i.options.IssueType},
Project: jira.Project{Key: i.options.ProjectName},
Summary: summary,
}
// On-prem version of Jira server does not use AccountID
if !i.options.Cloud {
fields = &jira.IssueFields{
Assignee: &jira.User{Name: i.options.AccountID},
Description: format.CreateReportDescription(event, i),
Type: jira.IssueType{Name: i.options.IssueType},
Project: jira.Project{Key: i.options.ProjectName},
Summary: summary,
Labels: labels,
2023-03-10 15:55:54 -07:00
Unknowns: customFields,
2021-06-04 13:11:09 +02:00
}
}
2021-06-04 17:14:26 +02:00
issueData := &jira.Issue{
Fields: fields,
}
_, resp, err := i.jira.Issue.Create(issueData)
if err != nil {
var data string
if resp != nil && resp.Body != nil {
d, _ := io.ReadAll(resp.Body)
data = string(d)
2021-06-04 17:14:26 +02:00
}
return fmt.Errorf("%w => %s", err, data)
}
return nil
}
// CreateIssue creates an issue in the tracker or updates the existing one
func (i *Integration) CreateIssue(event *output.ResultEvent) error {
if i.options.UpdateExisting {
issueID, err := i.FindExistingIssue(event)
2021-06-04 17:14:26 +02:00
if err != nil {
return err
} else if issueID != "" {
_, _, err = i.jira.Issue.AddComment(issueID, &jira.Comment{
Body: format.CreateReportDescription(event, i),
})
return err
2021-06-04 17:14:26 +02:00
}
2021-06-04 13:11:09 +02:00
}
return i.CreateNewIssue(event)
2021-06-04 17:14:26 +02:00
}
// FindExistingIssue checks if the issue already exists and returns its ID
func (i *Integration) FindExistingIssue(event *output.ResultEvent) (string, error) {
template := format.GetMatchedTemplateName(event)
jql := fmt.Sprintf("summary ~ \"%s\" AND summary ~ \"%s\" AND status != \"%s\" AND project = \"%s\"", template, event.Host, i.options.StatusNot, i.options.ProjectName)
2021-06-04 13:11:09 +02:00
searchOptions := &jira.SearchOptions{
2021-06-04 17:14:26 +02:00
MaxResults: 1, // if any issue exists, then we won't create a new one
}
2021-06-04 17:14:26 +02:00
chunk, resp, err := i.jira.Issue.Search(jql, searchOptions)
if err != nil {
var data string
if resp != nil && resp.Body != nil {
d, _ := io.ReadAll(resp.Body)
data = string(d)
}
return "", fmt.Errorf("%w => %s", err, data)
2021-06-04 17:14:26 +02:00
}
switch resp.Total {
case 0:
return "", nil
case 1:
return chunk[0].ID, nil
default:
gologger.Warning().Msgf("Discovered multiple opened issues %s for the host %s: The issue [%s] will be updated.", template, event.Host, chunk[0].ID)
return chunk[0].ID, nil
}
}