2018-09-24 20:22:32 +03:00
|
|
|
package github
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2018-09-25 20:10:38 +03:00
|
|
|
"strings"
|
2018-09-24 20:22:32 +03:00
|
|
|
|
|
|
|
"github.com/MichaelMure/git-bug/bridge/core"
|
2018-09-25 20:10:38 +03:00
|
|
|
"github.com/MichaelMure/git-bug/bug"
|
2018-09-24 20:22:32 +03:00
|
|
|
"github.com/MichaelMure/git-bug/cache"
|
2018-10-02 14:02:16 +03:00
|
|
|
"github.com/MichaelMure/git-bug/util/git"
|
2018-09-24 20:22:32 +03:00
|
|
|
"github.com/shurcooL/githubv4"
|
|
|
|
)
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
const keyGithubId = "github-id"
|
|
|
|
const keyGithubUrl = "github-url"
|
|
|
|
|
|
|
|
// githubImporter implement the Importer interface
|
2018-09-24 20:22:32 +03:00
|
|
|
type githubImporter struct{}
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
func (*githubImporter) ImportAll(repo *cache.RepoCache, conf core.Configuration) error {
|
|
|
|
client := buildClient(conf)
|
2018-09-24 20:22:32 +03:00
|
|
|
|
2018-10-02 00:34:45 +03:00
|
|
|
q := &issueTimelineQuery{}
|
2018-09-24 20:22:32 +03:00
|
|
|
variables := map[string]interface{}{
|
2018-10-02 14:02:16 +03:00
|
|
|
"owner": githubv4.String(conf[keyUser]),
|
|
|
|
"name": githubv4.String(conf[keyProject]),
|
|
|
|
"issueFirst": githubv4.Int(1),
|
|
|
|
"issueAfter": (*githubv4.String)(nil),
|
|
|
|
"timelineFirst": githubv4.Int(10),
|
|
|
|
"timelineAfter": (*githubv4.String)(nil),
|
2018-10-02 00:34:45 +03:00
|
|
|
|
|
|
|
// Fun fact, github provide the comment edition in reverse chronological
|
|
|
|
// order, because haha. Look at me, I'm dying of laughter.
|
2018-10-02 14:02:16 +03:00
|
|
|
"issueEditLast": githubv4.Int(10),
|
|
|
|
"issueEditBefore": (*githubv4.String)(nil),
|
|
|
|
"commentEditLast": githubv4.Int(10),
|
|
|
|
"commentEditBefore": (*githubv4.String)(nil),
|
2018-09-24 20:22:32 +03:00
|
|
|
}
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
var b *cache.BugCache
|
|
|
|
|
2018-09-24 20:22:32 +03:00
|
|
|
for {
|
|
|
|
err := client.Query(context.TODO(), &q, variables)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-10-02 00:34:45 +03:00
|
|
|
if len(q.Repository.Issues.Nodes) == 0 {
|
|
|
|
return nil
|
2018-09-25 20:10:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
issue := q.Repository.Issues.Nodes[0]
|
|
|
|
|
|
|
|
if b == nil {
|
2018-10-02 00:34:45 +03:00
|
|
|
b, err = ensureIssue(repo, issue, client, variables)
|
2018-09-25 20:10:38 +03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
for _, itemEdge := range q.Repository.Issues.Nodes[0].Timeline.Edges {
|
|
|
|
ensureTimelineItem(b, itemEdge.Cursor, itemEdge.Node, client, variables)
|
|
|
|
}
|
2018-09-24 20:22:32 +03:00
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
if !issue.Timeline.PageInfo.HasNextPage {
|
|
|
|
err = b.CommitAsNeeded()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
b = nil
|
|
|
|
|
|
|
|
if !q.Repository.Issues.PageInfo.HasNextPage {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
variables["issueAfter"] = githubv4.NewString(q.Repository.Issues.PageInfo.EndCursor)
|
|
|
|
variables["timelineAfter"] = (*githubv4.String)(nil)
|
|
|
|
continue
|
2018-09-24 20:22:32 +03:00
|
|
|
}
|
2018-09-25 20:10:38 +03:00
|
|
|
|
|
|
|
variables["timelineAfter"] = githubv4.NewString(issue.Timeline.PageInfo.EndCursor)
|
2018-09-24 20:22:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (*githubImporter) Import(repo *cache.RepoCache, conf core.Configuration, id string) error {
|
|
|
|
fmt.Println(conf)
|
|
|
|
fmt.Println("IMPORT")
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2018-09-25 20:10:38 +03:00
|
|
|
|
2018-10-02 00:34:45 +03:00
|
|
|
func ensureIssue(repo *cache.RepoCache, issue issueTimeline, client *githubv4.Client, rootVariables map[string]interface{}) (*cache.BugCache, error) {
|
2018-09-25 20:10:38 +03:00
|
|
|
fmt.Printf("import issue: %s\n", issue.Title)
|
|
|
|
|
2018-10-02 00:34:45 +03:00
|
|
|
b, err := repo.ResolveBugCreateMetadata(keyGithubId, parseId(issue.Id))
|
|
|
|
if err != nil && err != bug.ErrBugNotExist {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is no edit, the UserContentEdits given by github is empty. That
|
|
|
|
// means that the original message is given by the issue message.
|
|
|
|
|
|
|
|
// if there is edits, the UserContentEdits given by github contains both the
|
|
|
|
// original message and the following edits. The issue message give the last
|
|
|
|
// version so we don't care about that.
|
|
|
|
|
|
|
|
if len(issue.UserContentEdits.Nodes) == 0 {
|
|
|
|
if err == bug.ErrBugNotExist {
|
|
|
|
b, err = repo.NewBugRaw(
|
|
|
|
makePerson(issue.Author),
|
|
|
|
issue.CreatedAt.Unix(),
|
|
|
|
// Todo: this might not be the initial title, we need to query the
|
|
|
|
// timeline to be sure
|
|
|
|
issue.Title,
|
|
|
|
cleanupText(string(issue.Body)),
|
|
|
|
nil,
|
|
|
|
map[string]string{
|
|
|
|
keyGithubId: parseId(issue.Id),
|
|
|
|
keyGithubUrl: issue.Url.String(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// reverse the order, because github
|
|
|
|
reverseEdits(issue.UserContentEdits.Nodes)
|
|
|
|
|
|
|
|
if err == bug.ErrBugNotExist {
|
|
|
|
firstEdit := issue.UserContentEdits.Nodes[0]
|
|
|
|
|
|
|
|
if firstEdit.Diff == nil {
|
|
|
|
return nil, fmt.Errorf("no diff")
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err = repo.NewBugRaw(
|
|
|
|
makePerson(issue.Author),
|
|
|
|
issue.CreatedAt.Unix(),
|
|
|
|
// Todo: this might not be the initial title, we need to query the
|
|
|
|
// timeline to be sure
|
|
|
|
issue.Title,
|
2018-10-02 14:02:16 +03:00
|
|
|
cleanupText(string(*firstEdit.Diff)),
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
|
|
|
map[string]string{
|
|
|
|
keyGithubId: parseId(issue.Id),
|
|
|
|
keyGithubUrl: issue.Url.String(),
|
|
|
|
},
|
|
|
|
)
|
2018-10-02 14:02:16 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
target, err := b.ResolveTargetWithMetadata(keyGithubId, parseId(issue.Id))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2018-10-02 00:34:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, edit := range issue.UserContentEdits.Nodes {
|
|
|
|
if i == 0 {
|
|
|
|
// The first edit in the github result is the creation itself, we already have that
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
err := ensureCommentEdit(b, target, edit)
|
2018-10-02 00:34:45 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !issue.UserContentEdits.PageInfo.HasNextPage {
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have more edit, querying them
|
|
|
|
|
|
|
|
q := &issueEditQuery{}
|
|
|
|
variables := map[string]interface{}{
|
2018-10-02 14:02:16 +03:00
|
|
|
"owner": rootVariables["owner"],
|
|
|
|
"name": rootVariables["name"],
|
|
|
|
"issueFirst": rootVariables["issueFirst"],
|
|
|
|
"issueAfter": rootVariables["issueAfter"],
|
|
|
|
"issueEditLast": githubv4.Int(10),
|
|
|
|
"issueEditBefore": issue.UserContentEdits.PageInfo.StartCursor,
|
2018-10-02 00:34:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
err := client.Query(context.TODO(), &q, variables)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
edits := q.Repository.Issues.Nodes[0].UserContentEdits
|
|
|
|
|
|
|
|
if len(edits.Nodes) == 0 {
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, edit := range edits.Nodes {
|
|
|
|
if i == 0 {
|
|
|
|
// The first edit in the github result is the creation itself, we already have that
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
err := ensureCommentEdit(b, target, edit)
|
2018-10-02 00:34:45 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !edits.PageInfo.HasNextPage {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
variables["issueEditBefore"] = edits.PageInfo.StartCursor
|
2018-10-02 00:34:45 +03:00
|
|
|
}
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
// TODO: check + import files
|
|
|
|
|
2018-10-02 00:34:45 +03:00
|
|
|
return b, nil
|
2018-09-25 20:10:38 +03:00
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
func ensureTimelineItem(b *cache.BugCache, cursor githubv4.String, item timelineItem, client *githubv4.Client, rootVariables map[string]interface{}) error {
|
|
|
|
fmt.Printf("import %s\n", item.Typename)
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
switch item.Typename {
|
|
|
|
case "IssueComment":
|
2018-10-02 14:02:16 +03:00
|
|
|
return ensureComment(b, cursor, item.IssueComment, client, rootVariables)
|
2018-09-25 20:10:38 +03:00
|
|
|
|
|
|
|
case "LabeledEvent":
|
|
|
|
_, err := b.ChangeLabelsRaw(
|
|
|
|
makePerson(item.LabeledEvent.Actor),
|
|
|
|
item.LabeledEvent.CreatedAt.Unix(),
|
|
|
|
[]string{
|
|
|
|
string(item.LabeledEvent.Label.Name),
|
|
|
|
},
|
|
|
|
nil,
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
2018-09-25 20:10:38 +03:00
|
|
|
)
|
|
|
|
return err
|
|
|
|
|
|
|
|
case "UnlabeledEvent":
|
|
|
|
_, err := b.ChangeLabelsRaw(
|
|
|
|
makePerson(item.UnlabeledEvent.Actor),
|
|
|
|
item.UnlabeledEvent.CreatedAt.Unix(),
|
|
|
|
nil,
|
|
|
|
[]string{
|
|
|
|
string(item.UnlabeledEvent.Label.Name),
|
|
|
|
},
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
2018-09-25 20:10:38 +03:00
|
|
|
)
|
|
|
|
return err
|
|
|
|
|
|
|
|
case "ClosedEvent":
|
|
|
|
return b.CloseRaw(
|
|
|
|
makePerson(item.ClosedEvent.Actor),
|
|
|
|
item.ClosedEvent.CreatedAt.Unix(),
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
2018-09-25 20:10:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
case "ReopenedEvent":
|
|
|
|
return b.OpenRaw(
|
|
|
|
makePerson(item.ReopenedEvent.Actor),
|
|
|
|
item.ReopenedEvent.CreatedAt.Unix(),
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
2018-09-25 20:10:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
case "RenamedTitleEvent":
|
|
|
|
return b.SetTitleRaw(
|
|
|
|
makePerson(item.RenamedTitleEvent.Actor),
|
|
|
|
item.RenamedTitleEvent.CreatedAt.Unix(),
|
|
|
|
string(item.RenamedTitleEvent.CurrentTitle),
|
2018-10-02 00:34:45 +03:00
|
|
|
nil,
|
2018-09-25 20:10:38 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
default:
|
|
|
|
fmt.Println("ignore event ", item.Typename)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-10-02 14:02:16 +03:00
|
|
|
func ensureComment(b *cache.BugCache, cursor githubv4.String, comment issueComment, client *githubv4.Client, rootVariables map[string]interface{}) error {
|
|
|
|
target, err := b.ResolveTargetWithMetadata(keyGithubId, parseId(comment.Id))
|
|
|
|
if err != nil && err != cache.ErrNoMatchingOp {
|
|
|
|
// real error
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is no edit, the UserContentEdits given by github is empty. That
|
|
|
|
// means that the original message is given by the comment message.
|
|
|
|
|
|
|
|
// if there is edits, the UserContentEdits given by github contains both the
|
|
|
|
// original message and the following edits. The comment message give the last
|
|
|
|
// version so we don't care about that.
|
|
|
|
|
|
|
|
if len(comment.UserContentEdits.Nodes) == 0 {
|
|
|
|
if err == cache.ErrNoMatchingOp {
|
|
|
|
err = b.AddCommentRaw(
|
|
|
|
makePerson(comment.Author),
|
|
|
|
comment.CreatedAt.Unix(),
|
|
|
|
cleanupText(string(comment.Body)),
|
|
|
|
nil,
|
|
|
|
map[string]string{
|
|
|
|
keyGithubId: parseId(comment.Id),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// reverse the order, because github
|
|
|
|
reverseEdits(comment.UserContentEdits.Nodes)
|
|
|
|
|
|
|
|
if err == cache.ErrNoMatchingOp {
|
|
|
|
firstEdit := comment.UserContentEdits.Nodes[0]
|
|
|
|
|
|
|
|
if firstEdit.Diff == nil {
|
|
|
|
return fmt.Errorf("no diff")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = b.AddCommentRaw(
|
|
|
|
makePerson(comment.Author),
|
|
|
|
comment.CreatedAt.Unix(),
|
|
|
|
cleanupText(string(*firstEdit.Diff)),
|
|
|
|
nil,
|
|
|
|
map[string]string{
|
|
|
|
keyGithubId: parseId(comment.Id),
|
|
|
|
keyGithubUrl: comment.Url.String(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
target, err = b.ResolveTargetWithMetadata(keyGithubId, parseId(comment.Id))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, edit := range comment.UserContentEdits.Nodes {
|
|
|
|
if i == 0 {
|
|
|
|
// The first edit in the github result is the comment creation itself, we already have that
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
err := ensureCommentEdit(b, target, edit)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !comment.UserContentEdits.PageInfo.HasNextPage {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have more edit, querying them
|
|
|
|
|
|
|
|
q := &commentEditQuery{}
|
|
|
|
variables := map[string]interface{}{
|
|
|
|
"owner": rootVariables["owner"],
|
|
|
|
"name": rootVariables["name"],
|
|
|
|
"issueFirst": rootVariables["issueFirst"],
|
|
|
|
"issueAfter": rootVariables["issueAfter"],
|
|
|
|
"timelineFirst": githubv4.Int(1),
|
|
|
|
"timelineAfter": cursor,
|
|
|
|
"commentEditLast": githubv4.Int(10),
|
|
|
|
"commentEditBefore": comment.UserContentEdits.PageInfo.StartCursor,
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
err := client.Query(context.TODO(), &q, variables)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
edits := q.Repository.Issues.Nodes[0].Timeline.Nodes[0].IssueComment.UserContentEdits
|
|
|
|
|
|
|
|
if len(edits.Nodes) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, edit := range edits.Nodes {
|
|
|
|
if i == 0 {
|
|
|
|
// The first edit in the github result is the creation itself, we already have that
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
err := ensureCommentEdit(b, target, edit)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !edits.PageInfo.HasNextPage {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
variables["commentEditBefore"] = edits.PageInfo.StartCursor
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: check + import files
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ensureCommentEdit(b *cache.BugCache, target git.Hash, edit userContentEdit) error {
|
2018-10-02 00:34:45 +03:00
|
|
|
if edit.Editor == nil {
|
|
|
|
return fmt.Errorf("no editor")
|
|
|
|
}
|
|
|
|
|
|
|
|
if edit.Diff == nil {
|
|
|
|
return fmt.Errorf("no diff")
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := b.ResolveTargetWithMetadata(keyGithubId, parseId(edit.Id))
|
|
|
|
if err == nil {
|
|
|
|
// already imported
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != cache.ErrNoMatchingOp {
|
|
|
|
// real error
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("import edition\n")
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case edit.DeletedAt != nil:
|
|
|
|
// comment deletion, not supported yet
|
|
|
|
|
|
|
|
case edit.DeletedAt == nil:
|
|
|
|
// comment edition
|
|
|
|
err := b.EditCommentRaw(
|
|
|
|
makePerson(*edit.Editor),
|
|
|
|
edit.CreatedAt.Unix(),
|
2018-10-02 14:02:16 +03:00
|
|
|
target,
|
2018-10-02 00:34:45 +03:00
|
|
|
cleanupText(string(*edit.Diff)),
|
|
|
|
map[string]string{
|
|
|
|
keyGithubId: parseId(edit.Id),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-09-25 20:10:38 +03:00
|
|
|
// makePerson create a bug.Person from the Github data
|
2018-10-02 00:34:45 +03:00
|
|
|
func makePerson(actor actor) bug.Person {
|
2018-09-25 20:10:38 +03:00
|
|
|
return bug.Person{
|
|
|
|
Name: string(actor.Login),
|
|
|
|
AvatarUrl: string(actor.AvatarUrl),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// parseId convert the unusable githubv4.ID (an interface{}) into a string
|
|
|
|
func parseId(id githubv4.ID) string {
|
|
|
|
return fmt.Sprintf("%v", id)
|
|
|
|
}
|
|
|
|
|
|
|
|
func cleanupText(text string) string {
|
|
|
|
// windows new line, Github, really ?
|
|
|
|
return strings.Replace(text, "\r\n", "\n", -1)
|
|
|
|
}
|
2018-10-02 00:34:45 +03:00
|
|
|
|
|
|
|
func reverseEdits(edits []userContentEdit) []userContentEdit {
|
|
|
|
for i, j := 0, len(edits)-1; i < j; i, j = i+1, j-1 {
|
|
|
|
edits[i], edits[j] = edits[j], edits[i]
|
|
|
|
}
|
|
|
|
return edits
|
|
|
|
}
|