package main import ( "bytes" _ "fmt" "log" "os" "os/exec" "regexp" "strings" ) // go get hunt is a github secret key hunter written in go. target organizations, users, and remote/local repos // gotta be fast type Repo struct { url string name string path string } // memoization for commit1+commit2 hash var cache map[string]bool var appRoot string var regexes map[string]*regexp.Regexp func init() { appRoot, _ = os.Getwd() cache = make(map[string]bool) regexes = map[string]*regexp.Regexp{ "github": regexp.MustCompile(`[g|G][i|I][t|T][h|H][u|U][b|B].+[=|:=|:|<-].*\w+.*`), "aws": regexp.MustCompile(`[a|A][w|W][s|S].+[=|:=|:|<-].*\w+.*`), "heroku": regexp.MustCompile(`[h|H][e|E][r|R][o|O][k|K][u|U].+[=|:=|:|<-].*\w+.*`), "facebook": regexp.MustCompile(`[f|F][a|A][c|C][e|E][b|B][o|O][o|O][k|K].+[=|:=|:|<-].*\w+.*`), "twitter": regexp.MustCompile(`[t|T][w|W][i|I][t|T][t|T][e|E][r|R].+[=|:=|:|<-].*\w+.*`), "reddit": regexp.MustCompile(`[r|R][e|E][d|D][d|D][i|I][t|T].+[=|:=|:|<-].*\w+.*`), "twilio": regexp.MustCompile(`[t|T][w|W][i|I][l|L][i|I][o|O].+[=|:=|:|<-].*\w+.*`), } } func main() { args := os.Args[1:] opts := parseOptions(args) start(opts) } func start(opts *Options) { if opts.Repo != "" { repoStart(opts.Repo) } } func repoStart(repo_url string) { err := exec.Command("git", "clone", repo_url).Run() if err != nil { log.Fatalf("failed to clone repo %v", err) } repo_name := strings.Split(repo_url, "/")[4] if err := os.Chdir(repo_name); err != nil { log.Fatal(err) } repo := Repo{repo_url, repo_name, ""} repo.audit() repo.cleanup() } // cleanup changes to app root and recursive rms target repo func (repo Repo) cleanup() { if err := os.Chdir(appRoot); err != nil { log.Fatalf("failed cleaning up repo %v", err) } err := exec.Command("rm", "-rf", repo.name).Run() if err != nil { log.Fatal(err) } } // (Repo) audit parses git branch --all to audit remote branches func (repo Repo) audit() { var out []byte var err error var branch string var commits [][]byte out, err = exec.Command("git", "branch", "--all").Output() if err != nil { log.Fatalf("error retrieving branches %v\n", err) } // iterate through branches, git rev-list branches := bytes.Split(out, []byte("\n")) for i, branchB := range branches { if i < 2 || i == len(branches)-1 { continue } branch = string(bytes.Trim(branchB, " ")) out, err = exec.Command("git", "rev-list", branch).Output() if err != nil { log.Fatalf("error retrieving commits %v\n", err) } // iterate through commits commits = bytes.Split(out, []byte("\n")) for j, commitB := range commits { if j == len(commits)-2 { break } // TODO need a memoization structure for commitB vs commits[j+1] // memoize the actual diff function diff(string(commitB), string(commits[j+1])) } } } func diff(commit1 string, commit2 string) { _, seen := cache[commit1+commit2] if seen { return } out, err := exec.Command("git", "diff", commit1, commit2).Output() if err != nil { log.Fatalf("error retrieving commits %v\n", err) } cache[commit1+commit2] = true checkRegex(string(out)) }