README
¶
Git Repository security checker
This is a fast Go implementation to check Git repositories (local or remote) for some common security issues. It relies heavily on go-git.
Usage
To simply start the scanner as webservice configure a port to use and optionally (highly recommended) a TLS certificate (chain) and private key file:
./gitscanner -port 16092 -ssl-certificate-chain-file ~/.tls/my-cert-chain.crt.pem -ssl-private-key-file ~/.tls/tls.key.pem
If you instead just want to run in single execution mode with CLI don't provide any port and then refer to the possible configuration parameters:
$ ./gitscanner --help
Usage of ./gitscanner:
-branch-pattern string
Optional pattern to match refs against. Only matches will be processed in checks that rely on refs.
-email-pattern string
Pattern to match all commiter and author emails against. This will be used for the commitmeta.CheckCommits check.
-name-pattern string
Pattern to match all commiter and author names against. This will be used for the commitmeta.CheckCommits check.
-password string
An optional password for http basic auth.
-port int
When provided this will startup a webserver including ui that can be used to perform the checks via browser. (default -1)
-repositories string
A comma-separated list of repositories to perform checks against. Can be in these formats:
- http(s)://<remote URL>
- git@<remote URL>
- file://<path>
-ssh-private-key-file string
An optional path to a SSH private key file in PEM format.
-ssh-private-key-password string
An optional password if the given private key file is encrypted.
-ssl-certificate-chain-file string
An optional path to a TLS certificate (chain) in PEM format to enable HTTPS. Only used when port is set.
-ssl-private-key-file string
An optional path to a TLS private key file in PEM format to enable HTTPS. Only used when port is set.
-username string
An optional username for http basic auth.
Performed checks
- binaryfile.SearchBinaries: Searches for any binary files on each branch (local or remote) that matches the
-branch-pattern
. - unicode.SearchUnicode: Searches for specific unicode characters in each file on each branch (local or remote) that matches the
-branch-pattern
. See trojan-source.pdf. - commitmeta.CheckCommits: Checks every commits author and committer name and email for expected match against
-email-pattern
and-name-pattern
.
Build locally
- Install Go.
go build .
Add new tests
Adding tests is very simple:
- Write a go function that has this signature:
package myawesometest
import (
mygit "github.com/MrWong99/gitscanner/git"
"github.com/MrWong99/gitscanner/utils"
)
func MyAwesomeTest(wrapRepo *mygit.ClonedRepo, output chan<- utils.SingleCheck) error {
defer close(output)
// perform checks here and write any found issues into the output channel
}
- Add the function to the list of possible checks in checks/checks.go
REST API
When started in server mode gitscanner will provide the following endpoints:
POST /api/v1/checkRepos - Perform checks for given paths
Status Codes:
200
: checks were performed. Singular could still have failed though.400
: the request body was malformed.
Request Body:
path
: a comma separated list of urls to clone. They can be in these formats:http(s)://<remote URL>
git@<remote URL>
file://<path>
-> will only search on the local filesystem of the server
checks
: a list of check identifiers to determine with checks are to be performed. See GET checks.
Example:
{
"path": "git@github.com:go-git/go-git.git,https://gitlab.com/gitlab-org/gitlab.git",
"checkNames": [
"github.com/MrWong99/gitscanner/checks/commitmeta.CheckCommitAuthor",
"github.com/MrWong99/gitscanner/checks/unicode.SearchUnicode",
"github.com/MrWong99/gitscanner/checks/binaryfile.SearchBinaries"
]
}
Response Body:
- Array of objects with:
date
: ISO encoded timestamp of when the check was startedrepository
: the repository that was checkederror
: if any error occured while opening the repo it will be contained here, else empty stringchecks
: list of checks that contained suspicious results. Each check consists of:origin
: where this issue was found. This can be multiple things, e.g. a path to a file or a commit hash.branch
: the branch (if any) on which the issue was found.checkName
: the name of the check that found this issue.acknowledged
: currently always false, will be used in future updates.additionalInfo
: list of non-specified key-value pairs that differ from check to check.
Example:
[
{
"date": "2021-12-03T00:22:00.6155686+01:00",
"repository": "git@github.com:MrWong99/micasuca.git",
"error": "",
"checks": [
{
"origin": "Commit 65508c0d5f0ea52ce3d93f77f471359f4ec1d1bc",
"branch": "",
"checkName": "github.com/MrWong99/gitscanner/checks/commitmeta.CheckCommitAuthor",
"acknowledged": false,
"additionalInfo": {
"authorEmail": "shady.dude@inter.net",
"authorName": "Jeff",
"commitMessage": "Table view (in progress...)\n",
"commiterEmail": "shady.dude@inter.net",
"commiterName": "jeffHacker",
"numberOfParents": 1
}
},
{
"origin": "app/src/main/res/mipmap-xxhdpi/ic_wishlist_round.png",
"branch": "refs/remotes/origin/master",
"checkName": "github.com/MrWong99/gitscanner/checks/binaryfile.SearchBinaries",
"acknowledged": false,
"additionalInfo": {
"filemode": "0100644",
"filesize": "9.5 kB"
}
},
{
"origin": "gradle/wrapper/gradle-wrapper.jar",
"branch": "refs/remotes/origin/master",
"checkName": "github.com/MrWong99/gitscanner/checks/binaryfile.SearchBinaries",
"acknowledged": false,
"additionalInfo": {
"filemode": "0100644",
"filesize": "54.3 kB"
}
},
{
"origin": "gradlew",
"branch": "refs/remotes/origin/master",
"checkName": "github.com/MrWong99/gitscanner/checks/unicode.SearchUnicode",
"acknowledged": false,
"additionalInfo": {
"character":"'\\u202a'",
"filemode": "0100644",
"filesize": "5.3 kB"
}
}
]
},
{
"date": "2021-12-03T00:22:01.0774318+01:00",
"repository": "https://github.com/Mnaaz/JavaChat",
"error": "",
"checks": []
}
]
GET /api/v1/checks - Retrieve the list of possible checks
Status Codes:
200
: Checks retrieved.
Response Body:
- List of strings for all registered checks
Example:
[
"github.com/MrWong99/gitscanner/checks/binaryfile.SearchBinaries",
"github.com/MrWong99/gitscanner/checks/unicode.SearchUnicode",
"github.com/MrWong99/gitscanner/checks/commitmeta.CheckCommitAuthor"
]
GET /api/v1/config - Retrieve the current application configuration
Status Codes:
200
: configuration was returned.
Response Body:
branchPattern
: pattern to match branches against. The SearchBinaries and SearchUnicode checks use this.namePattern
: pattern to match the commiter and author names against. The CheckCommitAuthor check uses this.emailPattern
: pattern to match the commiter and author emails against. The CheckCommitAuthor check uses this.
Example:
{
"branchPattern": "refs/remotes/origin/main|.*v\\d+",
"namePattern": ".*",
"emailPattern": ".*@shady.com"
}
PUT /api/v1/config - Set the application configuration
Status Codes:
200
: config was updated.400
: the request body was malformed.
Request Body:
branchPattern
: regex pattern to match branches against. The SearchBinaries and SearchUnicode checks use this.namePattern
: regex pattern to match the commiter and author names against. The CheckCommitAuthor check uses this.emailPattern
: regex pattern to match the commiter and author emails against. The CheckCommitAuthor check uses this.
Example:
{
"branchPattern": "refs/remotes/origin/main|.*v\\d+",
"namePattern": ".*",
"emailPattern": ".*@shady.com"
}
PUT /api/v1/config/sshkey - Set the ssh private key to use when using ssh during clone
Status Codes:
200
: config was updated.400
: the request body was malformed.
Request Body:
key
: the ssh private key. Can be additionally encrypted withpassword
: the password that this key was encrypted with if any
Example:
{
"key": "-----BEGIN RSA PRIVATE KEY-----\nyOut41nK1mdUMB?\n-----END RSA PRIVATE KEY-----",
"password": ""
}
PUT /api/v1/config/basicauth - Set the username and password when usic basic authentication during clone
Status Codes:
200
: config was updated.400
: the request body was malformed.
Request Body:
username
: the username to use.password
: the password to use.
Example:
{
"username": "SecureMan",
"password": "1n5EcuR3"
}
Documentation
¶
There is no documentation for this package.