| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- 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 <branch>
- 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))
- }
|