Skip to content

Commit

Permalink
feat: optimize issues fetching
Browse files Browse the repository at this point in the history
  • Loading branch information
moul committed Sep 3, 2019
1 parent 1cb68fe commit 951b22e
Show file tree
Hide file tree
Showing 10 changed files with 153 additions and 43 deletions.
64 changes: 64 additions & 0 deletions compute/db.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
package compute

import (
"github.com/jinzhu/gorm"
"go.uber.org/zap"
"moul.io/depviz/sql"
"moul.io/multipmuri"
)

type multipmuriRepo interface {
RepoEntity() multipmuri.Entity
}

type multipmuriOwner interface {
OwnerEntity() multipmuri.Entity
}

type multipmuriService interface {
ServiceEntity() multipmuri.Entity
}

// FIXME: loadIssuesByAuthor
// FIXME: handle github search

func LoadIssuesByTargets(db *gorm.DB, targets []multipmuri.Entity) (*Computed, error) {
byRepo := []string{}
byOwner := []string{}
byService := []string{}
useFilters := true
for _, target := range targets {
switch v := target.(type) {
case multipmuriRepo:
byRepo = append(byRepo, v.RepoEntity().String())
case multipmuriOwner:
byOwner = append(byOwner, v.OwnerEntity().String())
case multipmuriService:
byService = append(byService, v.ServiceEntity().String())
default:
zap.L().Warn("unsupported target filter", zap.Any("target", target))
useFilters = false
}
}

// FIXME: add a owner field on issue
filteredDB := db
if useFilters {
filteredDB = filteredDB.Where(
"repository_id IN (?) OR repository_owner_id IN (?) OR service_id IN (?)",
byRepo,
byOwner,
byService,
)
}

allIssues, err := sql.LoadAllIssues(filteredDB)
if err != nil {
return nil, err
}

computed := Compute(allIssues)
computed.FilterByTargets(targets) // in most cases, this step is optional as we are already filtering by targets when querying the database

return &computed, nil
}
6 changes: 3 additions & 3 deletions github/github.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,14 +17,14 @@ import (
func Pull(input multipmuri.Entity, wg *sync.WaitGroup, token string, db *gorm.DB, out chan<- []*model.Issue) {
defer wg.Done()
type multipmuriMinimalInterface interface {
RepoEntity() *multipmuri.GitHubRepo
Repo() *multipmuri.GitHubRepo
}
target, ok := input.(multipmuriMinimalInterface)
if !ok {
zap.L().Warn("invalid input", zap.String("input", fmt.Sprintf("%v", input.String())))
return
}
repo := target.RepoEntity()
repo := target.Repo()

// create client
ctx := context.Background()
Expand All @@ -43,7 +43,7 @@ func Pull(input multipmuri.Entity, wg *sync.WaitGroup, token string, db *gorm.DB
}

for {
issues, resp, err := client.Issues.ListByRepo(ctx, repo.Owner(), repo.Repo(), callOpts)
issues, resp, err := client.Issues.ListByRepo(ctx, repo.OwnerID(), repo.RepoID(), callOpts)
if err != nil {
log.Fatal(err)
return
Expand Down
79 changes: 57 additions & 22 deletions github/model.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,15 @@ import (

"github.com/google/go-github/github"
"moul.io/depviz/model"
"moul.io/multipmuri"
)

func FromUser(input *github.User) *model.Account {
entity, err := model.ParseTarget(input.GetHTMLURL())
if err != nil {
panic(err)
}

name := input.GetName()
if name == "" {
name = input.GetLogin()
Expand All @@ -19,7 +25,8 @@ func FromUser(input *github.User) *model.Account {
UpdatedAt: input.GetUpdatedAt().Time,
URL: input.GetURL(),
},
Provider: githubProvider(),
// Type: "user"
Provider: FromServiceURL(multipmuri.ServiceEntity(entity).String()),
Location: input.GetLocation(),
Company: input.GetCompany(),
Blog: input.GetBlog(),
Expand All @@ -31,12 +38,32 @@ func FromUser(input *github.User) *model.Account {
}

func FromRepositoryURL(input string) *model.Repository {
entity, err := model.ParseTarget(input)
if err != nil {
panic(err)
}
owner := multipmuri.OwnerEntity(entity)
return &model.Repository{
Base: model.Base{
ID: input,
URL: input,
},
Provider: githubProvider(),
Provider: FromServiceURL(multipmuri.ServiceEntity(entity).String()),
Owner: FromOwnerURL(owner.String()),
}
}

func FromOwnerURL(input string) *model.Account {
entity, err := model.ParseTarget(input)
if err != nil {
panic(err)
}
return &model.Account{
Base: model.Base{
ID: input,
URL: input,
},
Provider: FromServiceURL(multipmuri.ServiceEntity(entity).String()),
}
}

Expand Down Expand Up @@ -77,30 +104,37 @@ func FromLabel(input *github.Label) *model.Label {
}

func FromIssue(input *github.Issue) *model.Issue {
parts := strings.Split(input.GetHTMLURL(), "/")
url := strings.Replace(input.GetHTMLURL(), "/pull/", "/issues/", -1)
entity, err := model.ParseTarget(input.GetHTMLURL())
if err != nil {
panic(err)
}
repo := multipmuri.RepoEntity(entity)
owner := multipmuri.OwnerEntity(entity)
service := multipmuri.ServiceEntity(entity)

issue := &model.Issue{
Base: model.Base{
ID: url,
URL: url,
ID: entity.String(),
URL: entity.String(),
CreatedAt: input.GetCreatedAt(),
UpdatedAt: input.GetUpdatedAt(),
},
CompletedAt: input.GetClosedAt(),
Repository: FromRepositoryURL(strings.Join(parts[0:len(parts)-2], "/")),
Title: input.GetTitle(),
State: input.GetState(),
Body: input.GetBody(),
IsPR: input.PullRequestLinks != nil,
IsLocked: input.GetLocked(),
NumComments: input.GetComments(),
NumUpvotes: *input.Reactions.PlusOne,
NumDownvotes: *input.Reactions.MinusOne,
Labels: make([]*model.Label, 0),
Assignees: make([]*model.Account, 0),
Author: FromUser(input.User),
Milestone: FromMilestone(input.Milestone),
CompletedAt: input.GetClosedAt(),
Repository: FromRepositoryURL(repo.String()),
RepositoryOwner: FromOwnerURL(owner.String()),
Service: FromServiceURL(service.String()),
Title: input.GetTitle(),
State: input.GetState(),
Body: input.GetBody(),
IsPR: input.PullRequestLinks != nil,
IsLocked: input.GetLocked(),
NumComments: input.GetComments(),
NumUpvotes: *input.Reactions.PlusOne,
NumDownvotes: *input.Reactions.MinusOne,
Labels: make([]*model.Label, 0),
Assignees: make([]*model.Account, 0),
Author: FromUser(input.User),
Milestone: FromMilestone(input.Milestone),
}
for _, label := range input.Labels {
issue.Labels = append(issue.Labels, FromLabel(&label))
Expand All @@ -111,10 +145,11 @@ func FromIssue(input *github.Issue) *model.Issue {
return issue
}

func githubProvider() *model.Provider {
func FromServiceURL(input string) *model.Provider {
return &model.Provider{
Base: model.Base{
ID: "github", // FIXME: support multiple github instances
ID: input,
URL: input,
},
Driver: string(model.GithubDriver),
}
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,6 @@ require (
gopkg.in/yaml.v2 v2.2.2
moul.io/graphman v1.5.0
moul.io/graphman/viz v0.0.0-20190830152634-1bb4245b0456
moul.io/multipmuri v1.6.0
moul.io/multipmuri v1.8.0
moul.io/zapgorm v0.0.0-20190706070406-8138918b527b
)
4 changes: 4 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -369,6 +369,10 @@ moul.io/graphman/viz v0.0.0-20190830152634-1bb4245b0456 h1:hZfuflz4kSOV21yDBTqtQ
moul.io/graphman/viz v0.0.0-20190830152634-1bb4245b0456/go.mod h1:TSPOyvLb1/0QPhaOzlW936GMtFxpgDeLGAXPjkBJNxM=
moul.io/multipmuri v1.6.0 h1:lfRCGN58BMCXDq2lZOwb/FBHhvJdeRaJYBV2C6TY5Kk=
moul.io/multipmuri v1.6.0/go.mod h1:xTE1AUMMTNRFyNRNwnM9E7UgOalny3SNb3B4h5CYv4Q=
moul.io/multipmuri v1.7.0 h1:e3SABTzwiymxtEHlvv+wFIMFftnX7n7p8BbR4dFc5OA=
moul.io/multipmuri v1.7.0/go.mod h1:xTE1AUMMTNRFyNRNwnM9E7UgOalny3SNb3B4h5CYv4Q=
moul.io/multipmuri v1.8.0 h1:KOQ/Nu9pe+5HyjJ8XUO0wxcGvLgrOo4bkxYyb9G0ZnY=
moul.io/multipmuri v1.8.0/go.mod h1:xTE1AUMMTNRFyNRNwnM9E7UgOalny3SNb3B4h5CYv4Q=
moul.io/zapgorm v0.0.0-20190706070406-8138918b527b h1:7A2qUMck+Ikop+5+Ar6wyweFOHTwXEgHbNayY2mEB6Q=
moul.io/zapgorm v0.0.0-20190706070406-8138918b527b/go.mod h1:JDE3xz5BQ1ccnAijE5+T8Qin6T256Bw2Cpdi+qMfWgw=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
6 changes: 1 addition & 5 deletions graph/graph.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,14 +62,10 @@ func Graph(opts *Options) (string, error) {
return "", err
}

issues, err := sql.LoadAllIssues(db)
computed, err := compute.LoadIssuesByTargets(db, opts.Targets)
if err != nil {
return "", err
}

// compute and filter issues
computed := compute.Compute(issues)
computed.FilterByTargets(opts.Targets)
// FIXME: if !opts.ShowOrphans { computed.FilterOrphans() }
// FIXME: if !opts.ShowAllRelated { computed.FilterAllRelated()
// FIXME: if !opts.ShowPRs { computed.FilterPRs()
Expand Down
26 changes: 15 additions & 11 deletions model/models.go
Original file line number Diff line number Diff line change
Expand Up @@ -151,17 +151,21 @@ type Issue struct {
IsHidden bool `json:"is-hidden"`

// relationships
Repository *Repository `json:"repository"`
RepositoryID string `json:"repository_id"`
Milestone *Milestone `json:"milestone"`
MilestoneID string `json:"milestone_id"`
Author *Account `json:"author"`
AuthorID string `json:"author_id"`
Labels []*Label `gorm:"many2many:issue_labels" json:"labels"`
Assignees []*Account `gorm:"many2many:issue_assignees" json:"assignees"`
Parents []*Issue `json:"-" gorm:"many2many:issue_parents;association_jointable_foreignkey:parent_id"`
Children []*Issue `json:"-" gorm:"many2many:issue_children;association_jointable_foreignkey:child_id"`
Related []*Issue `json:"-" gorm:"many2many:issue_related;association_jointable_foreignkey:related_id"`
Repository *Repository `json:"repository"`
RepositoryID string `json:"repository-id"`
RepositoryOwner *Account `json:"repository-owner"`
RepositoryOwnerID string `json:"repository-owner-id"`
Service *Provider `json:"service"`
ServiceID string `json:"service-id"`
Milestone *Milestone `json:"milestone"`
MilestoneID string `json:"milestone-id"`
Author *Account `json:"author"`
AuthorID string `json:"author-id"`
Labels []*Label `gorm:"many2many:issue_labels" json:"labels"`
Assignees []*Account `gorm:"many2many:issue_assignees" json:"assignees"`
Parents []*Issue `json:"-" gorm:"many2many:issue_parents;association_jointable_foreignkey:parent_id"`
Children []*Issue `json:"-" gorm:"many2many:issue_children;association_jointable_foreignkey:child_id"`
Related []*Issue `json:"-" gorm:"many2many:issue_related;association_jointable_foreignkey:related_id"`
}

func (i Issue) String() string {
Expand Down
5 changes: 5 additions & 0 deletions model/targets.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,3 +14,8 @@ func ParseTargets(args []string) ([]multipmuri.Entity, error) {
}
return targets, nil
}

func ParseTarget(arg string) (multipmuri.Entity, error) {
defaultContext := multipmuri.NewGitHubService("")
return defaultContext.RelDecodeString(arg)
}
2 changes: 1 addition & 1 deletion sql/cmd_sql_info.go
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ func runInfo(opts *infoOptions) error {
log.Printf("failed to get count for %q: %v", tableName, err)
continue
}
fmt.Printf("stats: %-20s %3d\n", tableName, count)
fmt.Printf("stats: %-20s %5d\n", tableName, count)
}

return nil
Expand Down
2 changes: 2 additions & 0 deletions sql/helpers.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ package sql

import (
"github.com/jinzhu/gorm"
"go.uber.org/zap"
"moul.io/depviz/model"
)

Expand All @@ -19,5 +20,6 @@ func LoadAllIssues(db *gorm.DB) (model.Issues, error) {
break
}
}
zap.L().Debug("fetched issues", zap.Int("quantity", len(allIssues)))
return allIssues, nil
}

0 comments on commit 951b22e

Please sign in to comment.