Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

✨ Add support for github GHES #2999

Merged
merged 12 commits into from
May 18, 2023
14 changes: 14 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -404,6 +404,20 @@ RESULTS
|---------|------------------------|--------------------------------|--------------------------------|---------------------------------------------------------------------------|
```

##### Using GitHub Enterprise Server (GHES) based Repository

To use a GitHub Enterprise host `github.corp.com`, use the `GH_HOST` environment variable.

```shell
# Set the GitHub Enterprise host without https prefix or slash with relevant authentication token
export GH_HOST=github.corp.com
export GITHUB_AUTH_TOKEN=token

scorecard --repo=github.corp.com/org/repo
# OR without github host url
scorecard --repo=org/repo
```

##### Using a Package manager

For projects in the `--npm`, `--pypi`, or `--rubygems` ecosystems, you have the
Expand Down
20 changes: 20 additions & 0 deletions checker/client_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ func TestGetClients(t *testing.T) { //nolint:gocognit
shouldCIIBeNil bool
wantErr bool
experimental bool
isGhHost bool
}{
{
name: "localURI is not empty",
Expand Down Expand Up @@ -94,6 +95,21 @@ func TestGetClients(t *testing.T) { //nolint:gocognit
wantErr: false,
experimental: true,
},
{
name: "repoURI is corp github host",
args: args{
ctx: context.Background(),
repoURI: "https://github.corp.com/ossf/scorecard",
localURI: "",
},
shouldOSSFuzzBeNil: false,
shouldRepoClientBeNil: false,
shouldVulnClientBeNil: false,
shouldRepoBeNil: false,
shouldCIIBeNil: false,
wantErr: false,
isGhHost: true,
},
}

for _, tt := range tests {
Expand All @@ -102,6 +118,10 @@ func TestGetClients(t *testing.T) { //nolint:gocognit
if tt.experimental {
t.Setenv("SCORECARD_EXPERIMENTAL", "true")
}
if tt.isGhHost {
t.Setenv("GH_HOST", "github.corp.com")
t.Setenv("GH_TOKEN", "PAT")
}
got, repoClient, ossFuzzClient, ciiClient, vulnsClient, err := GetClients(tt.args.ctx, tt.args.repoURI, tt.args.localURI, tt.args.logger) //nolint:lll
if (err != nil) != tt.wantErr {
t.Fatalf("GetClients() error = %v, wantErr %v", err, tt.wantErr)
Expand Down
32 changes: 29 additions & 3 deletions clients/githubrepo/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,8 @@ import (
"errors"
"fmt"
"net/http"
"os"
"strings"
"time"

"github.com/google/go-github/v38/github"
Expand Down Expand Up @@ -59,6 +61,8 @@ type Client struct {
commitDepth int
}

const defaultGhHost = "github.com"

// InitRepo sets up the GitHub repo in local storage for improving performance and GitHub token usage efficiency.
func (client *Client) InitRepo(inputRepo clients.Repo, commitSHA string, commitDepth int) error {
ghRepo, ok := inputRepo.(*repoURL)
Expand Down Expand Up @@ -126,7 +130,11 @@ func (client *Client) InitRepo(inputRepo clients.Repo, commitSHA string, commitD

// URI implements RepoClient.URI.
func (client *Client) URI() string {
return fmt.Sprintf("github.com/%s/%s", client.repourl.owner, client.repourl.repo)
host, isHost := os.LookupEnv("GH_HOST")
if !isHost {
host = defaultGhHost
}
return fmt.Sprintf("%s/%s/%s", host, client.repourl.owner, client.repourl.repo)
}

// LocalPath implements RepoClient.LocalPath.
Expand Down Expand Up @@ -259,8 +267,26 @@ func CreateGithubRepoClientWithTransport(ctx context.Context, rt http.RoundTripp
httpClient := &http.Client{
Transport: rt,
}
client := github.NewClient(httpClient)
graphClient := githubv4.NewClient(httpClient)

var client *github.Client
var graphClient *githubv4.Client
githubHost, isGhHost := os.LookupEnv("GH_HOST")

if isGhHost && githubHost != defaultGhHost {
githubRestURL := fmt.Sprintf("https://%s/api/v3", strings.TrimSpace(githubHost))
githubGraphqlURL := fmt.Sprintf("https://%s/api/graphql", strings.TrimSpace(githubHost))

var err error
client, err = github.NewEnterpriseClient(githubRestURL, githubRestURL, httpClient)
if err != nil {
panic(fmt.Errorf("error during CreateGithubRepoClientWithTransport:EnterpriseClient: %w", err))
}

graphClient = githubv4.NewEnterpriseClient(githubGraphqlURL, httpClient)
} else {
client = github.NewClient(httpClient)
graphClient = githubv4.NewClient(httpClient)
}

return &Client{
ctx: ctx,
Expand Down
9 changes: 8 additions & 1 deletion clients/githubrepo/repo.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ package githubrepo
import (
"fmt"
"net/url"
"os"
"strings"

"github.com/ossf/scorecard/v4/clients"
Expand All @@ -42,7 +43,11 @@ func (r *repoURL) parse(input string) error {
// This will takes care for repo/owner format.
// By default it will use github.com
case l == two:
t = "github.com/" + c[0] + "/" + c[1]
githubHost, isGhHost := os.LookupEnv("GH_HOST")
if !isGhHost {
githubHost = "github.com"
}
t = githubHost + "/" + c[0] + "/" + c[1]
case l >= three:
t = input
}
Expand Down Expand Up @@ -83,8 +88,10 @@ func (r *repoURL) String() string {

// IsValid implements Repo.IsValid.
func (r *repoURL) IsValid() error {
githubHost := os.Getenv("GH_HOST")
switch r.host {
case "github.com":
case githubHost:
default:
return sce.WithMessage(sce.ErrorUnsupportedHost, r.host)
}
Expand Down
36 changes: 31 additions & 5 deletions clients/githubrepo/repo_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,15 @@ import (
"github.com/google/go-cmp/cmp"
)

// nolint:paralleltest
// because we are using t.Setenv.
func TestRepoURL_IsValid(t *testing.T) {
t.Parallel()
tests := []struct {
name string
inputURL string
expected repoURL
wantErr bool
ghHost bool
}{
{
name: "Valid http address",
Expand Down Expand Up @@ -59,7 +61,7 @@ func TestRepoURL_IsValid(t *testing.T) {
wantErr: true,
},
{
name: "github repository",
name: "Github repository",
expected: repoURL{
host: "github.com",
owner: "foo",
Expand All @@ -69,7 +71,7 @@ func TestRepoURL_IsValid(t *testing.T) {
wantErr: false,
},
{
name: "github repository",
name: "Github repository with host",
expected: repoURL{
host: "github.com",
owner: "foo",
Expand All @@ -78,11 +80,36 @@ func TestRepoURL_IsValid(t *testing.T) {
inputURL: "https://github.com/foo/kubeflow",
wantErr: false,
},
{
name: "Enterprise github repository with host",
expected: repoURL{
host: "github.corp.com",
owner: "corpfoo",
repo: "kubeflow",
},
inputURL: "https://github.corp.com/corpfoo/kubeflow",
wantErr: false,
ghHost: true,
},
{
name: "Enterprise github repository",
expected: repoURL{
host: "github.corp.com",
owner: "corpfoo",
repo: "kubeflow",
},
inputURL: "corpfoo/kubeflow",
wantErr: false,
ghHost: true,
},
}
for _, tt := range tests {
tt := tt // Re-initializing variable so it is not changed while executing the closure below
t.Run(tt.name, func(t *testing.T) {
t.Parallel()
if tt.ghHost {
t.Setenv("GH_HOST", "github.corp.com")
}

r := repoURL{
host: tt.expected.host,
owner: tt.expected.owner,
Expand All @@ -97,7 +124,6 @@ func TestRepoURL_IsValid(t *testing.T) {
if !tt.wantErr && !cmp.Equal(tt.expected, r, cmp.AllowUnexported(repoURL{})) {
t.Errorf("Got diff: %s", cmp.Diff(tt.expected, r))
}

if !cmp.Equal(r.Host(), tt.expected.host) {
t.Errorf("%s expected host: %s got host %s", tt.inputURL, tt.expected.host, r.Host())
}
Expand Down