This document shows some example queries to get started.
Show me all CI/CD environment variables my GitHub user has direct or indirect acces to
This query will return paths between your user and potential secrets, via several means:
- CircleCI
- "All members" contexts
- team-restricted contexts
- repository projects
- GitHub Actions
- repository
- environment
- organization
MATCH p=(:User{login:"alice"})-[*..5]->(:EnvironmentVariable) RETURN p
I just compromised a GitHub user called superbot. What new repositories did I gain access to?
MATCH (:User{login:"superbot"})-->(r:Repository) WHERE NOT EXISTS((:User{login:'serain'})-->(r)) RETURN r.name
Show me GitHub Actions secrets without branch protections
To find GitHub Actions environment variables that are not in environments (and therefor accessible to anyone who can open a pull request), we can search for direct relationships between a repository and environment variables:
MATCH p=(:Repository)-->(:EnvironmentVariable) RETURN p
Environments also needn't enforce any branch protections. We can look for environment variables that can be exfiltrated from any environment through a pull request:
MATCH p=(:Repository)-->(e:Environment)-->(:EnvironmentVariable) WHERE e.protectedBranches = false RETURN p
Show me repositories running production terraform plan on pull requests
Production Terraform plans on unreviewed code are [a bad idea](https://alex.kaskaso.li/post/terraform-plan-rce). We attempt to find these by looking at the context values on pull requests' status checks, to get maximum coverage and account for CI/CD systems that may be configured server-side (e.g. AWS CodeBuild).
MATCH (r:Repository)-[:HAS_STATUS_CHECK{pullRequest:TRUE}]->(s:StatusCheck) WHERE s.context =~ "(?=.*(tf|terraform))(?=.*(?<!non)pro?d).*" RETURN r.name
Show me AWS_SECRET_ACCESS_KEY variables my user can access through WRITE access to repositories with CircleCI project environment variables
CircleCI doesn't support branch-level protections for secrets. The implication is that if you can open a PR against a repository, you can exfiltrate secrets from the CI/CD context. These could be production secrets.
Note that our query matches repositories that the user can access both directly and indirectly, through team memberships.
This query is broken down to illustrate the relationships GitOops builds up; the query could be written more succinctly.
MATCH p=(:User{login:"serain"})-[*..2]->(:Repository)-[:HAS_CI]->(:CircleCIProject)-[:EXPOSES_ENVIRONMENT_VARIABLE]->(v:EnvironmentVariable) WHERE v.name =~ ".*AWS.*SECRET.*" RETURN p
Show me repositories that many users have access to with potentially interesting secrets exposed in a pull request's CI/CD context
Here we're using the content of CI/CD configuration files to make educated guesses about interesting pipelines. This is less accurate that using our other relationships, but gives us coverage of unsupported CI/CD systems (as long as we pulled the configuration files).
MATCH (r:Repository)-[HAS_CI_CONFIGURATION]->(f:File{path: ".circleci/config.yml"}) WHERE any(x IN f.env WHERE x =~ ".*(AUTH|SECRET|TOKEN|PASS|PWD|CRED|KEY|PRD|PROD).*") OR any(x IN f.tags WHERE x IN ["aws", "gcp", "terraform"]) WITH r
MATCH (u:User)-[*..2]->(r)-[HAS_STATUS_CHECK{pullRequest:true}]->(s:StatusCheck) WITH r, COUNT(DISTINCT u) AS userCount WHERE userCount > 30 RETURN r
Show me repositories with potential automated deployments that don't have branch protections
MATCH (r:Repository)-[HAS_CI_CONFIGURATION]->(f:File) WHERE any(x IN f.env WHERE x =~ ".*(AUTH|SECRET|TOKEN|PASS|PWD|CRED|KEY|PRD|PROD).*") OR any(x IN f.tags WHERE x IN ["aws", "gcp", "terraform"]) WITH r
MATCH (r) WHERE NOT (r)-[HAS_BRANCH_PROTECTION_RULE]->(:BranchProtectionRule) RETURN r
Do any external contributors have paths to sensitive secrets?
MATCH (u:User) WHERE NOT (u)-[:IS_MEMBER_OF]->(:Organization{login:"fakenews"}) WITH u
MATCH p=(u)-[*..5]->(v:EnvironmentVariable) WHERE v.name =~ ".*(AUTH|SECRET|TOKEN|PASS|PWD|CRED|KEY|PRD|PROD).*" RETURN p
Show me who's still using Jenkins
MATCH (r:Repository{isArchived:FALSE})-->(f:File{path:"Jenkinsfile"}) RETURN r.name
UNION MATCH (r:Repository{isArchived:FALSE})-->(s:StatusCheck) WHERE s.host =~ ".*jenkins.*" RETURN r.name
UNION MATCH (r:Repository{isArchived:FALSE})-->(w:Webhook) WHERE w.host =~ ".*jenkins.*" RETURN r.name