mirror of https://github.com/matrix-org/go-neb.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
282 lines
8.0 KiB
282 lines
8.0 KiB
package services
|
|
|
|
import (
|
|
"database/sql"
|
|
"errors"
|
|
"fmt"
|
|
log "github.com/Sirupsen/logrus"
|
|
"github.com/andygrunwald/go-jira"
|
|
"github.com/matrix-org/go-neb/database"
|
|
"github.com/matrix-org/go-neb/matrix"
|
|
"github.com/matrix-org/go-neb/plugin"
|
|
"github.com/matrix-org/go-neb/realms/jira"
|
|
"github.com/matrix-org/go-neb/types"
|
|
"html"
|
|
"net/http"
|
|
"regexp"
|
|
"strings"
|
|
)
|
|
|
|
// Matches alphas then a -, then a number. E.g "FOO-123"
|
|
var issueKeyRegex = regexp.MustCompile("([A-z]+)-([0-9]+)")
|
|
var projectKeyRegex = regexp.MustCompile("^[A-z]+$")
|
|
|
|
type jiraService struct {
|
|
id string
|
|
BotUserID string
|
|
ClientUserID string
|
|
Rooms map[string]struct { // room_id => {}
|
|
RealmID string // Determines the JIRA endpoint
|
|
Projects map[string]struct { // SYN => {}
|
|
Expand bool
|
|
Track bool
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *jiraService) ServiceUserID() string { return s.BotUserID }
|
|
func (s *jiraService) ServiceID() string { return s.id }
|
|
func (s *jiraService) ServiceType() string { return "jira" }
|
|
func (s *jiraService) RoomIDs() []string {
|
|
var keys []string
|
|
for k := range s.Rooms {
|
|
keys = append(keys, k)
|
|
}
|
|
return keys
|
|
}
|
|
func (s *jiraService) Register() error { return nil }
|
|
func (s *jiraService) PostRegister(old types.Service) {}
|
|
|
|
func (s *jiraService) cmdJiraCreate(roomID, userID string, args []string) (interface{}, error) {
|
|
// E.g jira create PROJ "Issue title" "Issue desc"
|
|
if len(args) <= 1 {
|
|
return nil, errors.New("Missing project key (e.g 'ABC') and/or title")
|
|
}
|
|
|
|
if !projectKeyRegex.MatchString(args[0]) {
|
|
return nil, errors.New("Project key must only contain A-Z.")
|
|
}
|
|
|
|
pkey := strings.ToUpper(args[0]) // REST API complains if they are not ALL CAPS
|
|
|
|
title := args[1]
|
|
desc := ""
|
|
if len(args) == 3 {
|
|
desc = args[2]
|
|
} else if len(args) > 3 { // > 3 args is probably a title without quote marks
|
|
joinedTitle := strings.Join(args[1:], " ")
|
|
title = joinedTitle
|
|
}
|
|
|
|
r, err := s.projectToRealm(userID, pkey)
|
|
if err != nil {
|
|
log.WithError(err).Print("Failed to map project key to realm")
|
|
return nil, errors.New("Failed to map project key to a JIRA endpoint.")
|
|
}
|
|
if r == nil {
|
|
return nil, errors.New("No known project exists with that project key.")
|
|
}
|
|
|
|
iss := jira.Issue{
|
|
Fields: &jira.IssueFields{
|
|
Summary: title,
|
|
Description: desc,
|
|
Project: jira.Project{
|
|
Key: pkey,
|
|
},
|
|
// FIXME: This may vary depending on the JIRA install!
|
|
Type: jira.IssueType{
|
|
Name: "Bug",
|
|
},
|
|
},
|
|
}
|
|
cli, err := r.JIRAClient(userID, false)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
i, res, err := cli.Issue.Create(&iss)
|
|
if err != nil {
|
|
log.WithFields(log.Fields{
|
|
log.ErrorKey: err,
|
|
"user_id": userID,
|
|
"project": pkey,
|
|
"realm_id": r.ID(),
|
|
}).Print("Failed to create issue")
|
|
return nil, errors.New("Failed to create issue")
|
|
}
|
|
if res.StatusCode < 200 || res.StatusCode >= 300 {
|
|
return nil, fmt.Errorf("Failed to create issue: JIRA returned %d", res.StatusCode)
|
|
}
|
|
|
|
return &matrix.TextMessage{
|
|
"m.notice",
|
|
fmt.Sprintf("Created issue: %sbrowse/%s", r.JIRAEndpoint, i.Key),
|
|
}, nil
|
|
}
|
|
|
|
func (s *jiraService) expandIssue(roomID, userID, issueKey string) interface{} {
|
|
issueKey = strings.ToUpper(issueKey)
|
|
logger := log.WithField("issue_key", issueKey)
|
|
// [ISSU-123, ISSU, 123]
|
|
groups := issueKeyRegex.FindStringSubmatch(issueKey)
|
|
if len(groups) != 3 {
|
|
logger.Print("Failed to find issue key")
|
|
return nil
|
|
}
|
|
|
|
projectKey := groups[1]
|
|
if !s.Rooms[roomID].Projects[projectKey].Expand {
|
|
return nil
|
|
}
|
|
|
|
r, err := database.GetServiceDB().LoadAuthRealm(s.Rooms[roomID].RealmID)
|
|
if err != nil {
|
|
logger.WithFields(log.Fields{
|
|
"realm_id": s.Rooms[roomID].RealmID,
|
|
log.ErrorKey: err,
|
|
}).Print("Failed to load realm")
|
|
return nil
|
|
}
|
|
jrealm, ok := r.(*realms.JIRARealm)
|
|
if !ok {
|
|
logger.WithField("realm_id", s.Rooms[roomID].RealmID).Print(
|
|
"Realm cannot be typecast to JIRARealm",
|
|
)
|
|
}
|
|
logger.WithFields(log.Fields{
|
|
"room_id": roomID,
|
|
"user_id": s.ClientUserID,
|
|
}).Print("Expanding issue")
|
|
|
|
// Use the person who *provisioned* the service to check for project keys
|
|
// rather than the person who mentioned the issue key, as it is unlikely
|
|
// some random who mentioned the issue will have the intended auth.
|
|
cli, err := jrealm.JIRAClient(s.ClientUserID, false)
|
|
if err != nil {
|
|
logger.WithFields(log.Fields{
|
|
log.ErrorKey: err,
|
|
"user_id": s.ClientUserID,
|
|
}).Print("Failed to retrieve client")
|
|
return nil
|
|
}
|
|
|
|
issue, _, err := cli.Issue.Get(issueKey)
|
|
if err != nil {
|
|
logger.WithError(err).Print("Failed to GET issue")
|
|
return err
|
|
}
|
|
return matrix.GetHTMLMessage(
|
|
"m.notice",
|
|
fmt.Sprintf(
|
|
"%sbrowse/%s : %s",
|
|
jrealm.JIRAEndpoint, issueKey, htmlSummaryForIssue(issue),
|
|
),
|
|
)
|
|
}
|
|
|
|
func (s *jiraService) Plugin(roomID string) plugin.Plugin {
|
|
return plugin.Plugin{
|
|
Commands: []plugin.Command{
|
|
plugin.Command{
|
|
Path: []string{"jira", "create"},
|
|
Command: func(roomID, userID string, args []string) (interface{}, error) {
|
|
return s.cmdJiraCreate(roomID, userID, args)
|
|
},
|
|
},
|
|
},
|
|
Expansions: []plugin.Expansion{
|
|
plugin.Expansion{
|
|
Regexp: issueKeyRegex,
|
|
Expand: func(roomID, userID, issueKey string) interface{} {
|
|
return s.expandIssue(roomID, userID, issueKey)
|
|
},
|
|
},
|
|
},
|
|
}
|
|
}
|
|
func (s *jiraService) OnReceiveWebhook(w http.ResponseWriter, req *http.Request, cli *matrix.Client) {
|
|
w.WriteHeader(200) // Do nothing
|
|
}
|
|
|
|
func (s *jiraService) projectToRealm(userID, pkey string) (*realms.JIRARealm, error) {
|
|
// We don't know which JIRA installation this project maps to, so:
|
|
// - Get all known JIRA realms and f.e query their endpoints with the
|
|
// given user ID's credentials (so if it is a private project they
|
|
// can see it will succeed.)
|
|
// - If there is a matching project with that key, return that realm.
|
|
// We search installations which the user has already OAuthed with first as most likely
|
|
// the project key will be on a JIRA they have access to.
|
|
// TODO: Return whether they have authed or not so they know if they need to make a starter link
|
|
logger := log.WithFields(log.Fields{
|
|
"user_id": userID,
|
|
"project": pkey,
|
|
})
|
|
knownRealms, err := database.GetServiceDB().LoadAuthRealmsByType("jira")
|
|
if err != nil {
|
|
logger.WithError(err).Print("Failed to load jira auth realms")
|
|
return nil, err
|
|
}
|
|
// typecast and move ones which the user has authed with to the front of the queue
|
|
var queue []*realms.JIRARealm
|
|
var unauthRealms []*realms.JIRARealm
|
|
for _, r := range knownRealms {
|
|
jrealm, ok := r.(*realms.JIRARealm)
|
|
if !ok {
|
|
logger.WithField("realm_id", r.ID()).Print(
|
|
"Failed to type-cast 'jira' type realm into JIRARealm",
|
|
)
|
|
continue
|
|
}
|
|
|
|
_, err := database.GetServiceDB().LoadAuthSessionByUser(r.ID(), userID)
|
|
if err != nil {
|
|
if err == sql.ErrNoRows {
|
|
unauthRealms = append(unauthRealms, jrealm)
|
|
} else {
|
|
logger.WithError(err).WithField("realm_id", r.ID()).Print(
|
|
"Failed to load auth sessions for user",
|
|
)
|
|
}
|
|
continue // this may not have been the match anyway so don't give up!
|
|
}
|
|
queue = append(queue, jrealm)
|
|
}
|
|
|
|
// push unauthed realms to the back
|
|
queue = append(queue, unauthRealms...)
|
|
|
|
for _, jr := range queue {
|
|
exists, err := jr.ProjectKeyExists(userID, pkey)
|
|
if err != nil {
|
|
logger.WithError(err).WithField("realm_id", jr.ID()).Print(
|
|
"Failed to check if project key exists on this realm.",
|
|
)
|
|
continue // may not have been found anyway so keep searching!
|
|
}
|
|
if exists {
|
|
logger.Info("Project exists on ", jr.ID())
|
|
return jr, nil
|
|
}
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
func htmlSummaryForIssue(issue *jira.Issue) string {
|
|
// form a summary of the issue being affected e.g:
|
|
// "Flibble Wibble [P1, In Progress]"
|
|
status := html.EscapeString(issue.Fields.Status.Name)
|
|
if issue.Fields.Resolution != nil {
|
|
status = fmt.Sprintf("%s (%s)",
|
|
status, html.EscapeString(issue.Fields.Resolution.Name))
|
|
}
|
|
return fmt.Sprintf("%s [%s, %s]",
|
|
html.EscapeString(issue.Fields.Summary),
|
|
html.EscapeString(issue.Fields.Priority.Name),
|
|
status)
|
|
}
|
|
|
|
func init() {
|
|
types.RegisterService(func(serviceID, webhookEndpointURL string) types.Service {
|
|
return &jiraService{id: serviceID}
|
|
})
|
|
}
|