git-bug/bridge/jira/export.go

444 lines
13 KiB
Go
Raw Normal View History

2019-11-07 19:01:08 +03:00
package jira
import (
"context"
"encoding/json"
"fmt"
"net/http"
"time"
"github.com/pkg/errors"
"github.com/MichaelMure/git-bug/bridge/core"
"github.com/MichaelMure/git-bug/bug"
"github.com/MichaelMure/git-bug/cache"
"github.com/MichaelMure/git-bug/entity"
)
var (
ErrMissingCredentials = errors.New("missing credentials")
)
2019-11-07 19:01:08 +03:00
// jiraExporter implement the Exporter interface
type jiraExporter struct {
conf core.Configuration
// the current user identity
// NOTE: this is only needed to mock the credentials database in
// getIdentityClient
userIdentity entity.Id
2019-11-07 19:01:08 +03:00
// cache identities clients
identityClient map[entity.Id]*Client
// cache identifiers used to speed up exporting operations
// cleared for each bug
cachedOperationIDs map[entity.Id]string
// cache labels used to speed up exporting labels events
cachedLabels map[string]string
// store JIRA project information
project *Project
}
// Init .
2020-02-09 23:08:47 +03:00
func (je *jiraExporter) Init(repo *cache.RepoCache, conf core.Configuration) error {
je.conf = conf
je.identityClient = make(map[entity.Id]*Client)
je.cachedOperationIDs = make(map[entity.Id]string)
je.cachedLabels = make(map[string]string)
2019-11-07 19:01:08 +03:00
return nil
}
// getIdentityClient return an API client configured with the credentials
// of the given identity. If no client were found it will initialize it from
// the known credentials map and cache it for next use
2020-02-09 23:08:47 +03:00
func (je *jiraExporter) getIdentityClient(ctx context.Context, id entity.Id) (*Client, error) {
client, ok := je.identityClient[id]
2019-11-07 19:01:08 +03:00
if ok {
return client, nil
}
2020-02-09 23:08:47 +03:00
client = NewClient(je.conf[keyServer], ctx)
// NOTE: as a future enhancement, the bridge would ideally be able to generate
// a separate session token for each user that we have stored credentials
// for. However we currently only support a single user.
2020-02-09 23:08:47 +03:00
if id != je.userIdentity {
return nil, ErrMissingCredentials
}
2020-02-09 23:08:47 +03:00
err := client.Login(je.conf)
if err != nil {
return nil, err
}
2020-02-09 23:08:47 +03:00
je.identityClient[id] = client
return client, nil
2019-11-07 19:01:08 +03:00
}
// ExportAll export all event made by the current user to Jira
2020-02-09 23:08:47 +03:00
func (je *jiraExporter) ExportAll(ctx context.Context, repo *cache.RepoCache, since time.Time) (<-chan core.ExportResult, error) {
2019-11-07 19:01:08 +03:00
out := make(chan core.ExportResult)
user, err := repo.GetUserIdentity()
if err != nil {
return nil, err
}
// NOTE: this is currently only need to mock the credentials database in
// getIdentityClient.
2020-02-09 23:08:47 +03:00
je.userIdentity = user.Id()
client, err := je.getIdentityClient(ctx, user.Id())
2019-11-07 19:01:08 +03:00
if err != nil {
return nil, err
}
2020-02-09 23:08:47 +03:00
je.project, err = client.GetProject(je.conf[keyProject])
2019-11-07 19:01:08 +03:00
if err != nil {
return nil, err
}
go func() {
defer close(out)
var allIdentitiesIds []entity.Id
2020-02-09 23:08:47 +03:00
for id := range je.identityClient {
2019-11-07 19:01:08 +03:00
allIdentitiesIds = append(allIdentitiesIds, id)
}
allBugsIds := repo.AllBugsIds()
for _, id := range allBugsIds {
b, err := repo.ResolveBug(id)
if err != nil {
out <- core.NewExportError(errors.Wrap(err, "can't load bug"), id)
return
}
select {
case <-ctx.Done():
// stop iterating if context cancel function is called
return
default:
snapshot := b.Snapshot()
// ignore issues whose last modification date is before the query date
// TODO: compare the Lamport time instead of using the unix time
if snapshot.CreatedAt.Before(since) {
out <- core.NewExportNothing(b.Id(), "bug created before the since date")
continue
}
if snapshot.HasAnyActor(allIdentitiesIds...) {
// try to export the bug and it associated events
2020-02-09 23:08:47 +03:00
err := je.exportBug(ctx, b, since, out)
if err != nil {
out <- core.NewExportError(errors.Wrap(err, "can't export bug"), id)
return
}
2019-11-07 19:01:08 +03:00
} else {
out <- core.NewExportNothing(id, "not an actor")
}
}
}
}()
return out, nil
}
// exportBug publish bugs and related events
2020-02-09 23:08:47 +03:00
func (je *jiraExporter) exportBug(ctx context.Context, b *cache.BugCache, since time.Time, out chan<- core.ExportResult) error {
2019-11-07 19:01:08 +03:00
snapshot := b.Snapshot()
var bugJiraID string
// Special case:
// if a user try to export a bug that is not already exported to jira (or
// imported from jira) and we do not have the token of the bug author,
// there is nothing we can do.
// first operation is always createOp
createOp := snapshot.Operations[0].(*bug.CreateOperation)
author := snapshot.Author
// skip bug if it was imported from some other bug system
2019-12-18 18:49:49 +03:00
origin, ok := snapshot.GetCreateMetadata(core.MetaKeyOrigin)
2019-11-07 19:01:08 +03:00
if ok && origin != target {
out <- core.NewExportNothing(
b.Id(), fmt.Sprintf("issue tagged with origin: %s", origin))
return nil
2019-11-07 19:01:08 +03:00
}
// skip bug if it is a jira bug but is associated with another project
// (one bridge per JIRA project)
project, ok := snapshot.GetCreateMetadata(keyJiraProject)
2020-02-09 23:08:47 +03:00
if ok && !stringInSlice(project, []string{je.project.ID, je.project.Key}) {
2019-11-07 19:01:08 +03:00
out <- core.NewExportNothing(
b.Id(), fmt.Sprintf("issue tagged with project: %s", project))
return nil
2019-11-07 19:01:08 +03:00
}
// get jira bug ID
jiraID, ok := snapshot.GetCreateMetadata(keyJiraID)
if ok {
// will be used to mark operation related to a bug as exported
bugJiraID = jiraID
} else {
// check that we have credentials for operation author
2020-02-09 23:08:47 +03:00
client, err := je.getIdentityClient(ctx, author.Id())
2019-11-07 19:01:08 +03:00
if err != nil {
// if bug is not yet exported and we do not have the author's credentials
// then there is nothing we can do, so just skip this bug
out <- core.NewExportNothing(
b.Id(), fmt.Sprintf("missing author token for user %.8s",
author.Id().String()))
return err
2019-11-07 19:01:08 +03:00
}
// Load any custom fields required to create an issue from the git
// config file.
fields := make(map[string]interface{})
2020-02-09 23:08:47 +03:00
defaultFields, hasConf := je.conf[keyCreateDefaults]
2019-11-07 19:01:08 +03:00
if hasConf {
err = json.Unmarshal([]byte(defaultFields), &fields)
if err != nil {
return err
}
2019-11-07 19:01:08 +03:00
} else {
// If there is no configuration provided, at the very least the
// "issueType" field is always required. 10001 is "story" which I'm
// pretty sure is standard/default on all JIRA instances.
fields["issuetype"] = map[string]interface{}{
"id": "10001",
}
2019-11-07 19:01:08 +03:00
}
2020-02-09 23:08:47 +03:00
bugIDField, hasConf := je.conf[keyCreateGitBug]
2019-11-07 19:01:08 +03:00
if hasConf {
// If the git configuration also indicates it, we can assign the git-bug
// id to a custom field to assist in integrations
fields[bugIDField] = b.Id().String()
}
// create bug
result, err := client.CreateIssue(
2020-02-09 23:08:47 +03:00
je.project.ID, createOp.Title, createOp.Message, fields)
2019-11-07 19:01:08 +03:00
if err != nil {
err := errors.Wrap(err, "exporting jira issue")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
id := result.ID
out <- core.NewExportBug(b.Id())
// mark bug creation operation as exported
err = markOperationAsExported(
2020-02-09 23:08:47 +03:00
b, createOp.Id(), id, je.project.Key, time.Time{})
2019-11-07 19:01:08 +03:00
if err != nil {
err := errors.Wrap(err, "marking operation as exported")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
// commit operation to avoid creating multiple issues with multiple pushes
err = b.CommitAsNeeded()
if err != nil {
err := errors.Wrap(err, "bug commit")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
// cache bug jira ID
bugJiraID = id
}
// cache operation jira id
2020-02-09 23:08:47 +03:00
je.cachedOperationIDs[createOp.Id()] = bugJiraID
2019-11-07 19:01:08 +03:00
// lookup the mapping from git-bug "status" to JIRA "status" id
2020-02-09 23:08:47 +03:00
statusMap, err := getStatusMap(je.conf)
if err != nil {
return err
}
2019-11-07 19:01:08 +03:00
for _, op := range snapshot.Operations[1:] {
// ignore SetMetadata operations
if _, ok := op.(*bug.SetMetadataOperation); ok {
continue
}
// ignore operations already existing in jira (due to import or export)
// cache the ID of already exported or imported issues and events from
// Jira
if id, ok := op.GetMetadata(keyJiraID); ok {
2020-02-09 23:08:47 +03:00
je.cachedOperationIDs[op.Id()] = id
2019-11-07 19:01:08 +03:00
continue
}
opAuthor := op.GetAuthor()
2020-02-09 23:08:47 +03:00
client, err := je.getIdentityClient(ctx, opAuthor.Id())
2019-11-07 19:01:08 +03:00
if err != nil {
out <- core.NewExportError(
fmt.Errorf("missing operation author credentials for user %.8s",
author.Id().String()), op.Id())
2019-11-07 19:01:08 +03:00
continue
}
var id string
var exportTime time.Time
switch opr := op.(type) {
2019-11-07 19:01:08 +03:00
case *bug.AddCommentOperation:
comment, err := client.AddComment(bugJiraID, opr.Message)
if err != nil {
err := errors.Wrap(err, "adding comment")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
id = comment.ID
out <- core.NewExportComment(op.Id())
// cache comment id
2020-02-09 23:08:47 +03:00
je.cachedOperationIDs[op.Id()] = id
2019-11-07 19:01:08 +03:00
case *bug.EditCommentOperation:
if opr.Target == createOp.Id() {
// An EditCommentOpreation with the Target set to the create operation
// encodes a modification to the long-description/summary.
exportTime, err = client.UpdateIssueBody(bugJiraID, opr.Message)
if err != nil {
err := errors.Wrap(err, "editing issue")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
out <- core.NewExportCommentEdition(op.Id())
id = bugJiraID
} else {
// Otherwise it's an edit to an actual comment. A comment cannot be
// edited before it was created, so it must be the case that we have
// already observed and cached the AddCommentOperation.
2020-02-09 23:08:47 +03:00
commentID, ok := je.cachedOperationIDs[opr.Target]
2019-11-07 19:01:08 +03:00
if !ok {
// Since an edit has to come after the creation, we expect we would
// have cached the creation id.
panic("unexpected error: comment id not found")
}
comment, err := client.UpdateComment(bugJiraID, commentID, opr.Message)
if err != nil {
err := errors.Wrap(err, "editing comment")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
out <- core.NewExportCommentEdition(op.Id())
// JIRA doesn't track all comment edits, they will only tell us about
// the most recent one. We must invent a consistent id for the operation
// so we use the comment ID plus the timestamp of the update, as
// reported by JIRA. Note that this must be consistent with the importer
// during ensureComment()
id = getTimeDerivedID(comment.ID, comment.Updated)
2019-11-07 19:01:08 +03:00
}
case *bug.SetStatusOperation:
jiraStatus, hasStatus := statusMap[opr.Status.String()]
if hasStatus {
exportTime, err = UpdateIssueStatus(client, bugJiraID, jiraStatus)
if err != nil {
err := errors.Wrap(err, "editing status")
out <- core.NewExportWarning(err, b.Id())
2019-11-07 19:01:08 +03:00
// Failure to update status isn't necessarily a big error. It's
// possible that we just don't have enough information to make that
// update. In this case, just don't export the operation.
continue
}
out <- core.NewExportStatusChange(op.Id())
id = bugJiraID
} else {
out <- core.NewExportError(fmt.Errorf(
"No jira status mapped for %.8s", opr.Status.String()), b.Id())
2019-11-07 19:01:08 +03:00
}
case *bug.SetTitleOperation:
exportTime, err = client.UpdateIssueTitle(bugJiraID, opr.Title)
if err != nil {
err := errors.Wrap(err, "editing title")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
out <- core.NewExportTitleEdition(op.Id())
id = bugJiraID
case *bug.LabelChangeOperation:
exportTime, err = client.UpdateLabels(
bugJiraID, opr.Added, opr.Removed)
if err != nil {
err := errors.Wrap(err, "updating labels")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
out <- core.NewExportLabelChange(op.Id())
id = bugJiraID
default:
panic("unhandled operation type case")
}
// mark operation as exported
err = markOperationAsExported(
2020-02-09 23:08:47 +03:00
b, op.Id(), id, je.project.Key, exportTime)
2019-11-07 19:01:08 +03:00
if err != nil {
err := errors.Wrap(err, "marking operation as exported")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
// commit at each operation export to avoid exporting same events multiple
// times
err = b.CommitAsNeeded()
if err != nil {
err := errors.Wrap(err, "bug commit")
out <- core.NewExportError(err, b.Id())
return err
2019-11-07 19:01:08 +03:00
}
}
return nil
2019-11-07 19:01:08 +03:00
}
2020-02-09 23:08:47 +03:00
func markOperationAsExported(b *cache.BugCache, target entity.Id, jiraID, jiraProject string, exportTime time.Time) error {
2019-11-07 19:01:08 +03:00
newMetadata := map[string]string{
keyJiraID: jiraID,
keyJiraProject: jiraProject,
}
if !exportTime.IsZero() {
newMetadata[keyJiraExportTime] = exportTime.Format(http.TimeFormat)
}
_, err := b.SetMetadata(target, newMetadata)
return err
}
// UpdateIssueStatus attempts to change the "status" field by finding a
// transition which achieves the desired state and then performing that
// transition
2020-02-09 23:08:47 +03:00
func UpdateIssueStatus(client *Client, issueKeyOrID string, desiredStateNameOrID string) (time.Time, error) {
2019-11-07 19:01:08 +03:00
var responseTime time.Time
tlist, err := client.GetTransitions(issueKeyOrID)
if err != nil {
return responseTime, err
}
transition := getTransitionTo(tlist, desiredStateNameOrID)
if transition == nil {
return responseTime, errTransitionNotFound
}
responseTime, err = client.DoTransition(issueKeyOrID, transition.ID)
if err != nil {
return responseTime, err
}
return responseTime, nil
}