| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145 |
- package main
- import (
- "bytes"
- "encoding/json"
- "fmt"
- "io/ioutil"
- "log"
- "os"
- "os/exec"
- "os/signal"
- "strings"
- "sync"
- "syscall"
- )
- // LeakElem contains the line and commit of a leak
- type LeakElem struct {
- Line string `json:"line"`
- Commit string `json:"commit"`
- }
- // start clones and determines if there are any leaks
- func start(opts *Options) {
- fmt.Printf("\nEvaluating \x1b[37;1m%s\x1b[0m...\n", opts.RepoURL)
- c := make(chan os.Signal, 2)
- signal.Notify(c, os.Interrupt, syscall.SIGTERM)
- err := exec.Command("git", "clone", opts.RepoURL).Run()
- if err != nil {
- log.Printf("failed to clone repo %v", err)
- return
- }
- repoName := getLocalRepoName(opts.RepoURL)
- if err = os.Chdir(repoName); err != nil {
- log.Fatal(err)
- }
- go func() {
- <-c
- cleanup(repoName)
- os.Exit(1)
- }()
- report := getLeaks(repoName, opts.Concurrency)
- if len(report) == 0 {
- fmt.Printf("No Leaks detected for \x1b[35;2m%s\x1b[0m...\n\n", opts.RepoURL)
- }
- cleanup(repoName)
- reportJSON, _ := json.MarshalIndent(report, "", "\t")
- err = ioutil.WriteFile(fmt.Sprintf("%s_leaks.json", repoName), reportJSON, 0644)
- if err != nil {
- log.Fatalf("Can't write to file: %s", err)
- }
- }
- // getLocalRepoName generates the name of the local clone folder based on the given URL
- func getLocalRepoName(url string) string {
- splitSlashes := strings.Split(url, "/")
- name := splitSlashes[len(splitSlashes)-1]
- name = strings.TrimSuffix(name, ".git")
- splitColons := strings.Split(name, ":")
- name = splitColons[len(splitColons)-1]
- return name
- }
- // cleanup deletes the repo
- func cleanup(repoName string) {
- if err := os.Chdir(appRoot); err != nil {
- log.Fatalf("failed cleaning up repo. Does the repo exist? %v", err)
- }
- err := exec.Command("rm", "-rf", repoName).Run()
- if err != nil {
- log.Fatal(err)
- }
- }
- // getLeaks will attempt to find gitleaks
- func getLeaks(repoName string, concurrency int) []LeakElem {
- var (
- out []byte
- err error
- commitWG sync.WaitGroup
- gitLeakReceiverWG sync.WaitGroup
- gitLeaks = make(chan LeakElem)
- report []LeakElem
- )
- if concurrency == 0 {
- concurrency = 100
- }
- semaphoreChan := make(chan struct{}, concurrency)
- go func(commitWG *sync.WaitGroup, gitLeakReceiverWG *sync.WaitGroup) {
- for gitLeak := range gitLeaks {
- fmt.Println(gitLeak)
- report = append(report, gitLeak)
- gitLeakReceiverWG.Done()
- }
- }(&commitWG, &gitLeakReceiverWG)
- out, err = exec.Command("git", "rev-list", "--all", "--remotes", "--topo-order").Output()
- if err != nil {
- log.Fatalf("error retrieving commits%v\n", err)
- }
- commits := bytes.Split(out, []byte("\n"))
- commitWG.Add(len(commits))
- for _, currCommitB := range commits {
- currCommit := string(currCommitB)
- go func(currCommit string, repoName string, commitWG *sync.WaitGroup, gitLeakReceiverWG *sync.WaitGroup) {
- defer commitWG.Done()
- var leakPrs bool
- if err := os.Chdir(fmt.Sprintf("%s/%s", appRoot, repoName)); err != nil {
- log.Fatal(err)
- }
- commitCmp := fmt.Sprintf("%s^!", currCommit)
- semaphoreChan <- struct{}{}
- out, err := exec.Command("git", "diff", commitCmp).Output()
- <-semaphoreChan
- if err != nil {
- return
- }
- lines := checkRegex(string(out))
- if len(lines) == 0 {
- return
- }
- for _, line := range lines {
- leakPrs = checkEntropy(line)
- if leakPrs {
- gitLeakReceiverWG.Add(1)
- gitLeaks <- LeakElem{line, currCommit}
- }
- }
- }(currCommit, repoName, &commitWG, &gitLeakReceiverWG)
- }
- commitWG.Wait()
- gitLeakReceiverWG.Wait()
- return report
- }
|