package api import ( "fmt" "time" ) type PullRequestsPayload struct { ViewerCreated []PullRequest ReviewRequested []PullRequest CurrentPR *PullRequest } type PullRequest struct { Number int Title string State string URL string HeadRefName string Reviews struct { Nodes []struct { State string Author struct { Login string } } } Commits struct { Nodes []struct { Commit struct { Status struct { Contexts []struct { State string } } CheckSuites struct { Nodes []struct { CheckRuns struct { Nodes []struct { Conclusion string } } } } } } } } type PullRequestReviewStatus struct { ChangesRequested bool Approved bool } func (pr *PullRequest) ReviewStatus() PullRequestReviewStatus { status := PullRequestReviewStatus{} reviewMap := map[string]string{} // Reviews will include every review on record, including consecutive ones // from the same actor. Consolidate them into latest state per reviewer. for _, review := range pr.Reviews.Nodes { reviewMap[review.Author.Login] = review.State } for _, state := range reviewMap { switch state { case "CHANGES_REQUESTED": status.ChangesRequested = true case "APPROVED": status.Approved = true } } return status } type PullRequestChecksStatus struct { Pending int Failing int Passing int Total int } func (pr *PullRequest) ChecksStatus() (summary PullRequestChecksStatus) { if len(pr.Commits.Nodes) == 0 { return } commit := pr.Commits.Nodes[0].Commit for _, status := range commit.Status.Contexts { switch status.State { case "SUCCESS": summary.Passing++ case "EXPECTED", "ERROR", "FAILURE": summary.Failing++ case "PENDING": summary.Pending++ default: panic(fmt.Errorf("unsupported status: %q", status.State)) } summary.Total++ } for _, checkSuite := range commit.CheckSuites.Nodes { for _, checkRun := range checkSuite.CheckRuns.Nodes { switch checkRun.Conclusion { case "SUCCESS", "NEUTRAL": summary.Passing++ case "FAILURE", "CANCELLED", "TIMED_OUT", "ACTION_REQUIRED": summary.Failing++ default: panic(fmt.Errorf("unsupported check conclusion: %q", checkRun.Conclusion)) } summary.Total++ } } return } type Repo interface { RepoName() string RepoOwner() string } type IssuesPayload struct { Assigned []Issue Mentioned []Issue Recent []Issue } type Issue struct { Number int Title string URL string } func Issues(client *Client, ghRepo Repo, currentUsername string) (*IssuesPayload, error) { type issues struct { Issues struct { Edges []struct { Node Issue } } } type response struct { Assigned issues Mentioned issues Recent issues } query := ` fragment issue on Issue { number title } query($owner: String!, $repo: String!, $since: DateTime!, $viewer: String!, $per_page: Int = 10) { assigned: repository(owner: $owner, name: $repo) { issues(filterBy: {assignee: $viewer}, first: $per_page, orderBy: {field: CREATED_AT, direction: DESC}) { edges { node { ...issue } } } } mentioned: repository(owner: $owner, name: $repo) { issues(filterBy: {mentioned: $viewer}, first: $per_page, orderBy: {field: CREATED_AT, direction: DESC}) { edges { node { ...issue } } } } recent: repository(owner: $owner, name: $repo) { issues(filterBy: {since: $since}, first: $per_page, orderBy: {field: CREATED_AT, direction: DESC}) { edges { node { ...issue } } } } } ` owner := ghRepo.RepoOwner() repo := ghRepo.RepoName() since := time.Now().UTC().Add(time.Hour * -24).Format("2006-01-02T15:04:05-0700") variables := map[string]interface{}{ "owner": owner, "repo": repo, "viewer": currentUsername, "since": since, } var resp response err := client.GraphQL(query, variables, &resp) if err != nil { return nil, err } var assigned []Issue for _, edge := range resp.Assigned.Issues.Edges { assigned = append(assigned, edge.Node) } var mentioned []Issue for _, edge := range resp.Mentioned.Issues.Edges { mentioned = append(mentioned, edge.Node) } var recent []Issue for _, edge := range resp.Recent.Issues.Edges { recent = append(recent, edge.Node) } payload := IssuesPayload{ assigned, mentioned, recent, } return &payload, nil } func PullRequests(client *Client, ghRepo Repo, currentBranch, currentUsername string) (*PullRequestsPayload, error) { type edges struct { Edges []struct { Node PullRequest } PageInfo struct { HasNextPage bool EndCursor string } } type response struct { Repository struct { PullRequests edges } ViewerCreated edges ReviewRequested edges } query := ` fragment pr on PullRequest { number title url headRefName commits(last: 1) { nodes { commit { status { contexts { state } } checkSuites(first: 50) { nodes { checkRuns(first: 50) { nodes { conclusion } } } } } } } } fragment prWithReviews on PullRequest { ...pr reviews(last: 20) { nodes { state author { login } } } } query($owner: String!, $repo: String!, $headRefName: String!, $viewerQuery: String!, $reviewerQuery: String!, $per_page: Int = 10) { repository(owner: $owner, name: $repo) { pullRequests(headRefName: $headRefName, states: OPEN, first: 1) { edges { node { ...prWithReviews } } } } viewerCreated: search(query: $viewerQuery, type: ISSUE, first: $per_page) { edges { node { ...prWithReviews } } pageInfo { hasNextPage } } reviewRequested: search(query: $reviewerQuery, type: ISSUE, first: $per_page) { edges { node { ...pr } } pageInfo { hasNextPage } } } ` owner := ghRepo.RepoOwner() repo := ghRepo.RepoName() viewerQuery := fmt.Sprintf("repo:%s/%s state:open is:pr author:%s", owner, repo, currentUsername) reviewerQuery := fmt.Sprintf("repo:%s/%s state:open review-requested:%s", owner, repo, currentUsername) variables := map[string]interface{}{ "viewerQuery": viewerQuery, "reviewerQuery": reviewerQuery, "owner": owner, "repo": repo, "headRefName": currentBranch, } var resp response err := client.GraphQL(query, variables, &resp) if err != nil { return nil, err } var viewerCreated []PullRequest for _, edge := range resp.ViewerCreated.Edges { viewerCreated = append(viewerCreated, edge.Node) } var reviewRequested []PullRequest for _, edge := range resp.ReviewRequested.Edges { reviewRequested = append(reviewRequested, edge.Node) } var currentPR *PullRequest for _, edge := range resp.Repository.PullRequests.Edges { currentPR = &edge.Node } payload := PullRequestsPayload{ viewerCreated, reviewRequested, currentPR, } return &payload, nil } func PullRequestsForBranch(client *Client, ghRepo Repo, branch string) ([]PullRequest, error) { type response struct { Repository struct { PullRequests struct { Edges []struct { Node PullRequest } } } } query := ` query($owner: String!, $repo: String!, $headRefName: String!) { repository(owner: $owner, name: $repo) { pullRequests(headRefName: $headRefName, states: OPEN, first: 1) { edges { node { number title url } } } } }` variables := map[string]interface{}{ "owner": ghRepo.RepoOwner(), "repo": ghRepo.RepoName(), "headRefName": branch, } var resp response err := client.GraphQL(query, variables, &resp) if err != nil { return nil, err } prs := []PullRequest{} for _, edge := range resp.Repository.PullRequests.Edges { prs = append(prs, edge.Node) } return prs, nil } func CreatePullRequest(client *Client, ghRepo Repo, params map[string]interface{}) (*PullRequest, error) { repoID, err := GitHubRepoId(client, ghRepo) if err != nil { return nil, err } query := ` mutation CreatePullRequest($input: CreatePullRequestInput!) { createPullRequest(input: $input) { pullRequest { url } } }` inputParams := map[string]interface{}{ "repositoryId": repoID, } for key, val := range params { inputParams[key] = val } variables := map[string]interface{}{ "input": inputParams, } result := struct { CreatePullRequest struct { PullRequest PullRequest } }{} err = client.GraphQL(query, variables, &result) if err != nil { return nil, err } return &result.CreatePullRequest.PullRequest, nil } func PullRequestList(client *Client, vars map[string]interface{}, limit int) ([]PullRequest, error) { type response struct { Repository struct { PullRequests struct { Edges []struct { Node PullRequest } PageInfo struct { HasNextPage bool EndCursor string } } } } query := ` query( $owner: String!, $repo: String!, $limit: Int!, $endCursor: String, $baseBranch: String, $labels: [String!], $state: [PullRequestState!] = OPEN ) { repository(owner: $owner, name: $repo) { pullRequests( states: $state, baseRefName: $baseBranch, labels: $labels, first: $limit, after: $endCursor, orderBy: {field: CREATED_AT, direction: DESC} ) { edges { node { number title state url headRefName } } pageInfo { hasNextPage endCursor } } } }` prs := []PullRequest{} pageLimit := min(limit, 100) variables := map[string]interface{}{} for name, val := range vars { variables[name] = val } for { variables["limit"] = pageLimit var data response err := client.GraphQL(query, variables, &data) if err != nil { return nil, err } prData := data.Repository.PullRequests for _, edge := range prData.Edges { prs = append(prs, edge.Node) if len(prs) == limit { goto done } } if prData.PageInfo.HasNextPage { variables["endCursor"] = prData.PageInfo.EndCursor pageLimit = min(pageLimit, limit-len(prs)) continue } done: break } return prs, nil } func min(a, b int) int { if a < b { return a } return b }