Skip to content

Commit

Permalink
G107 - SSRF (securego#236)
Browse files Browse the repository at this point in the history
* Initial SSRF Rule

* Added Selector evaluation

* Added source code tests

* Fixed spacing issues

* Fixed Spacingv2

* Removed resty test
  • Loading branch information
cschoenduve-splunk authored and Cosmin Cojocar committed Sep 4, 2018
1 parent 63b25c1 commit 419c929
Show file tree
Hide file tree
Showing 6 changed files with 149 additions and 4 deletions.
7 changes: 4 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,12 +41,13 @@ or to specify a set of rules to explicitly exclude using the '-exclude=' flag.
- G104: Audit errors not checked
- G105: Audit the use of math/big.Int.Exp
- G106: Audit the use of ssh.InsecureIgnoreHostKey
- G107: Url provided to HTTP request as taint input
- G201: SQL query construction using format string
- G202: SQL query construction using string concatenation
- G203: Use of unescaped data in HTML templates
- G204: Audit use of command execution
- G301: Poor file permissions used when creating a directory
- G302: Poor file permisions used with chmod
- G302: Poor file permissions used with chmod
- G303: Creating tempfile using a predictable path
- G304: File path provided as taint input
- G305: File traversal when extracting zip archive
Expand Down Expand Up @@ -79,7 +80,7 @@ that are not considered build artifacts by the compiler (so test files).
As with all automated detection tools there will be cases of false positives. In cases where gosec reports a failure that has been manually verified as being safe it is possible to annotate the code with a '#nosec' comment.

The annotation causes gosec to stop processing any further nodes within the
AST so can apply to a whole block or more granularly to a single expression.
AST so can apply to a whole block or more granularly to a single expression.

```go

Expand Down Expand Up @@ -178,7 +179,7 @@ You can build the docker image as follows:
make image
```

You can run the `gosec` tool in a container against your local Go project. You just have to mount the project in the
You can run the `gosec` tool in a container against your local Go project. You just have to mount the project in the
`GOPATH` of the container:

```
Expand Down
2 changes: 1 addition & 1 deletion cmd/tlsconfig/tlsconfig.go
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ type goTLSConfiguration struct {

// getTLSConfFromURL retrieves the json containing the TLS configurations from the specified URL.
func getTLSConfFromURL(url string) (*ServerSideTLSJson, error) {
r, err := http.Get(url)
r, err := http.Get(url) // #nosec G107
if err != nil {
return nil, err
}
Expand Down
1 change: 1 addition & 0 deletions rules/rulelist.go
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ func Generate(filters ...RuleFilter) RuleList {
{"G104", "Audit errors not checked", NewNoErrorCheck},
{"G105", "Audit the use of big.Exp function", NewUsingBigExp},
{"G106", "Audit the use of ssh.InsecureIgnoreHostKey function", NewSSHHostKey},
{"G107", "Url provided to HTTP request as taint input", NewSSRFCheck},

// injection
{"G201", "SQL query construction using format string", NewSQLStrFormat},
Expand Down
4 changes: 4 additions & 0 deletions rules/rules_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,10 @@ var _ = Describe("gosec rules", func() {
runner("G106", testutils.SampleCodeG106)
})

It("should detect ssrf via http requests with variable url", func() {
runner("G107", testutils.SampleCodeG107)
})

It("should detect sql injection via format strings", func() {
runner("G201", testutils.SampleCodeG201)
})
Expand Down
70 changes: 70 additions & 0 deletions rules/ssrf.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
package rules

import (
"go/ast"
"go/types"

"github.com/securego/gosec"
)

type ssrf struct {
gosec.MetaData
gosec.CallList
}

// ID returns the identifier for this rule
func (r *ssrf) ID() string {
return r.MetaData.ID
}

// ResolveVar tries to resolve the first argument of a call expression
// The first argument is the url
func (r *ssrf) ResolveVar(n *ast.CallExpr, c *gosec.Context) bool {
if len(n.Args) > 0 {
arg := n.Args[0]
if ident, ok := arg.(*ast.Ident); ok {
obj := c.Info.ObjectOf(ident)
if _, ok := obj.(*types.Var); ok && !gosec.TryResolve(ident, c) {
return true
}
}
}
return false
}

// Match inspects AST nodes to determine if certain net/http methods are called with variable input
func (r *ssrf) Match(n ast.Node, c *gosec.Context) (*gosec.Issue, error) {
// Call expression is using http package directly
if node := r.ContainsCallExpr(n, c); node != nil {
if r.ResolveVar(node, c) {
return gosec.NewIssue(c, n, r.ID(), r.What, r.Severity, r.Confidence), nil
}
}
// Look at the last selector identity for methods matching net/http's
if node, ok := n.(*ast.CallExpr); ok {
if selExpr, ok := node.Fun.(*ast.SelectorExpr); ok {
// Pull last selector's identity name and compare to net/http methods
if r.Contains("net/http", selExpr.Sel.Name) {
if r.ResolveVar(node, c) {
return gosec.NewIssue(c, n, r.ID(), r.What, r.Severity, r.Confidence), nil
}
}
}
}
return nil, nil
}

// NewSSRFCheck detects cases where HTTP requests are sent
func NewSSRFCheck(id string, conf gosec.Config) (gosec.Rule, []ast.Node) {
rule := &ssrf{
CallList: gosec.NewCallList(),
MetaData: gosec.MetaData{
ID: id,
What: "Potential HTTP request made with variable url",
Severity: gosec.Medium,
Confidence: gosec.Medium,
},
}
rule.AddAll("net/http", "Do", "Get", "Head", "Post", "PostForm", "RoundTrip")
return rule, []ast.Node{(*ast.CallExpr)(nil)}
}
69 changes: 69 additions & 0 deletions testutils/source.go
Original file line number Diff line number Diff line change
Expand Up @@ -192,6 +192,75 @@ import (
func main() {
_ = ssh.InsecureIgnoreHostKey()
}`, 1}}

// SampleCodeG107 - SSRF via http requests with variable url
SampleCodeG107 = []CodeSample{{`
package main
import (
"net/http"
"io/ioutil"
"fmt"
"os"
)
func main() {
url := os.Getenv("tainted_url")
resp, err := http.Get(url)
if err != nil {
panic(err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
panic(err)
}
fmt.Printf("%s", body)
}`, 1}, {`
package main
import (
"fmt"
"net/http"
)
const url = "http://127.0.0.1"
func main() {
resp, err := http.Get(url)
if err != nil {
fmt.Println(err)
}
fmt.Println(resp.Status)
}`, 0}, {`
package main
import (
"net/http"
"fmt"
"os"
"strconv"
)
type httpWrapper struct {
DesiredCode string
}
func (c *httpWrapper) Get(url string) (*http.Response, error) {
return http.Get(url)
}
func main() {
code := os.Getenv("STATUS_CODE")
var url = os.Getenv("URL")
client := httpWrapper{code}
resp1, err1 := client.Get(url)
if err1 != nil {
fmt.Println(err1)
os.Exit(1)
}
if strconv.Itoa(resp1.StatusCode) == client.DesiredCode {
fmt.Println("True")
} else {
fmt.Println("False")
}
}`, 2}}
// SampleCodeG201 - SQL injection via format string
SampleCodeG201 = []CodeSample{
{`
Expand Down

0 comments on commit 419c929

Please sign in to comment.