mirror of https://github.com/daffainfo/nuclei.git
Removing redundant code, cleanup + tests
parent
f2c20dda12
commit
a455b054e5
|
@ -1,8 +1,7 @@
|
||||||
package replacer
|
package replacer
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"github.com/valyala/fasttemplate"
|
||||||
"strings"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Payload marker constants
|
// Payload marker constants
|
||||||
|
@ -12,21 +11,9 @@ const (
|
||||||
MarkerParenthesisClose = "}}"
|
MarkerParenthesisClose = "}}"
|
||||||
)
|
)
|
||||||
|
|
||||||
// New creates a new replacer structure for values replacement on the fly.
|
// Replace replaces placeholders in template with values on the fly.
|
||||||
func New(values map[string]interface{}) *strings.Replacer {
|
func Replace(template string, values map[string]interface{}) string {
|
||||||
replacerItems := make([]string, 0, len(values)*4)
|
new := fasttemplate.ExecuteStringStd(template, MarkerGeneral, MarkerGeneral, values)
|
||||||
|
final := fasttemplate.ExecuteStringStd(new, MarkerParenthesisOpen, MarkerParenthesisClose, values)
|
||||||
for key, val := range values {
|
return final
|
||||||
valueStr := fmt.Sprintf("%s", val)
|
|
||||||
|
|
||||||
replacerItems = append(replacerItems,
|
|
||||||
fmt.Sprintf("%s%s%s", MarkerParenthesisOpen, key, MarkerParenthesisClose),
|
|
||||||
valueStr,
|
|
||||||
)
|
|
||||||
replacerItems = append(replacerItems,
|
|
||||||
fmt.Sprintf("%s%s%s", MarkerGeneral, key, MarkerGeneral),
|
|
||||||
valueStr,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
return strings.NewReplacer(replacerItems...)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -83,9 +83,9 @@ func (r *Request) Make(domain string) (*dns.Msg, error) {
|
||||||
|
|
||||||
var q dns.Question
|
var q dns.Question
|
||||||
|
|
||||||
replacer := replacer.New(map[string]interface{}{"FQDN": domain})
|
final := replacer.Replace(r.Name, map[string]interface{}{"FQDN": domain})
|
||||||
|
|
||||||
q.Name = dns.Fqdn(replacer.Replace(r.Name))
|
q.Name = dns.Fqdn(final)
|
||||||
q.Qclass = r.class
|
q.Qclass = r.class
|
||||||
q.Qtype = r.question
|
q.Qtype = r.question
|
||||||
req.Question = append(req.Question, q)
|
req.Question = append(req.Question, q)
|
||||||
|
|
|
@ -2,7 +2,6 @@ package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"net"
|
"net"
|
||||||
|
@ -41,6 +40,7 @@ type generatedRequest struct {
|
||||||
func (r *requestGenerator) Make(baseURL string, dynamicValues map[string]interface{}) (*generatedRequest, error) {
|
func (r *requestGenerator) Make(baseURL string, dynamicValues map[string]interface{}) (*generatedRequest, error) {
|
||||||
baseURL = strings.TrimSuffix(baseURL, "/")
|
baseURL = strings.TrimSuffix(baseURL, "/")
|
||||||
|
|
||||||
|
// We get the next payload for the request.
|
||||||
data, payloads, ok := r.nextValue()
|
data, payloads, ok := r.nextValue()
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, io.EOF
|
return nil, io.EOF
|
||||||
|
@ -58,7 +58,7 @@ func (r *requestGenerator) Make(baseURL string, dynamicValues map[string]interfa
|
||||||
"Hostname": hostname,
|
"Hostname": hostname,
|
||||||
})
|
})
|
||||||
|
|
||||||
// If data contains \n it's a raw request, process it like that. Else
|
// If data contains \n it's a raw request, process it like raw. Else
|
||||||
// continue with the template based request flow.
|
// continue with the template based request flow.
|
||||||
if strings.Contains(data, "\n") {
|
if strings.Contains(data, "\n") {
|
||||||
return r.makeHTTPRequestFromRaw(ctx, baseURL, data, values, payloads)
|
return r.makeHTTPRequestFromRaw(ctx, baseURL, data, values, payloads)
|
||||||
|
@ -75,7 +75,7 @@ func (r *requestGenerator) Total() int {
|
||||||
}
|
}
|
||||||
|
|
||||||
// baseURLWithTemplatePrefs returns the url for BaseURL keeping
|
// baseURLWithTemplatePrefs returns the url for BaseURL keeping
|
||||||
// the template port and path preference
|
// the template port and path preference over the user provided one.
|
||||||
func baseURLWithTemplatePrefs(data string, parsedURL *url.URL) string {
|
func baseURLWithTemplatePrefs(data string, parsedURL *url.URL) string {
|
||||||
// template port preference over input URL port
|
// template port preference over input URL port
|
||||||
// template has port
|
// template has port
|
||||||
|
@ -91,10 +91,10 @@ func baseURLWithTemplatePrefs(data string, parsedURL *url.URL) string {
|
||||||
|
|
||||||
// MakeHTTPRequestFromModel creates a *http.Request from a request template
|
// MakeHTTPRequestFromModel creates a *http.Request from a request template
|
||||||
func (r *requestGenerator) makeHTTPRequestFromModel(ctx context.Context, data string, values map[string]interface{}) (*generatedRequest, error) {
|
func (r *requestGenerator) makeHTTPRequestFromModel(ctx context.Context, data string, values map[string]interface{}) (*generatedRequest, error) {
|
||||||
URL := replacer.New(values).Replace(data)
|
final := replacer.Replace(data, values)
|
||||||
|
|
||||||
// Build a request on the specified URL
|
// Build a request on the specified URL
|
||||||
req, err := http.NewRequestWithContext(ctx, r.request.Method, URL, nil)
|
req, err := http.NewRequestWithContext(ctx, r.request.Method, final, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -108,32 +108,35 @@ func (r *requestGenerator) makeHTTPRequestFromModel(ctx context.Context, data st
|
||||||
|
|
||||||
// makeHTTPRequestFromRaw creates a *http.Request from a raw request
|
// makeHTTPRequestFromRaw creates a *http.Request from a raw request
|
||||||
func (r *requestGenerator) makeHTTPRequestFromRaw(ctx context.Context, baseURL, data string, values, payloads map[string]interface{}) (*generatedRequest, error) {
|
func (r *requestGenerator) makeHTTPRequestFromRaw(ctx context.Context, baseURL, data string, values, payloads map[string]interface{}) (*generatedRequest, error) {
|
||||||
// Add trailing line
|
// Add trailing line to request body based on content type
|
||||||
|
// handling multipart bodies differently.
|
||||||
|
if !strings.HasSuffix(data, "\r\n") && !strings.HasSuffix(data, "\n") {
|
||||||
|
if !strings.Contains(r.request.Headers["Content-Type"], "multipart") {
|
||||||
data += "\n"
|
data += "\n"
|
||||||
|
} else {
|
||||||
// If we have payloads, handle them by evaluating them at runtime.
|
data += "\r\n"
|
||||||
if len(r.request.Payloads) > 0 {
|
|
||||||
finalPayloads, err := r.getPayloadValues(baseURL, payloads)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
}
|
||||||
return r.handleRawWithPaylods(ctx, data, baseURL, values, finalPayloads)
|
|
||||||
}
|
}
|
||||||
return r.handleRawWithPaylods(ctx, data, baseURL, values, nil)
|
return r.handleRawWithPaylods(ctx, data, baseURL, values, payloads)
|
||||||
}
|
}
|
||||||
|
|
||||||
// handleRawWithPaylods handles raw requests along with paylaods
|
// handleRawWithPaylods handles raw requests along with paylaods
|
||||||
func (r *requestGenerator) handleRawWithPaylods(ctx context.Context, rawRequest, baseURL string, values, generatorValues map[string]interface{}) (*generatedRequest, error) {
|
func (r *requestGenerator) handleRawWithPaylods(ctx context.Context, rawRequest, baseURL string, values, generatorValues map[string]interface{}) (*generatedRequest, error) {
|
||||||
baseValues := generators.CopyMap(values)
|
// Combine the template payloads along with base
|
||||||
finalValues := generators.MergeMaps(baseValues, generatorValues)
|
// request values.
|
||||||
|
finalValues := generators.MergeMaps(generatorValues, values)
|
||||||
// Replace the dynamic variables in the URL if any
|
rawRequest = replacer.Replace(rawRequest, finalValues)
|
||||||
rawRequest = replacer.New(finalValues).Replace(rawRequest)
|
|
||||||
|
|
||||||
|
// Check if the match contains a dynamic variable, for each
|
||||||
|
// found one we will check if it's an expression and can
|
||||||
|
// be compiled, it will be evaluated and the results will be returned.
|
||||||
|
//
|
||||||
|
// The provided keys from finalValues will be used as variable names
|
||||||
|
// for substitution inside the expression.
|
||||||
dynamicValues := make(map[string]interface{})
|
dynamicValues := make(map[string]interface{})
|
||||||
for _, match := range templateExpressionRegex.FindAllString(rawRequest, -1) {
|
for _, match := range templateExpressionRegex.FindAllString(rawRequest, -1) {
|
||||||
// check if the match contains a dynamic variable
|
|
||||||
expr := generators.TrimDelimiters(match)
|
expr := generators.TrimDelimiters(match)
|
||||||
|
|
||||||
compiled, err := govaluate.NewEvaluableExpressionWithFunctions(expr, dsl.HelperFunctions())
|
compiled, err := govaluate.NewEvaluableExpressionWithFunctions(expr, dsl.HelperFunctions())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -142,17 +145,17 @@ func (r *requestGenerator) handleRawWithPaylods(ctx context.Context, rawRequest,
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
dynamicValues[expr] = result
|
dynamicValues[expr] = result // convert base64(<payload_name>) => <base64-representation>
|
||||||
}
|
}
|
||||||
|
|
||||||
// Replacer dynamic values if any in raw request and parse it
|
// Replacer dynamic values if any in raw request and parse it
|
||||||
rawRequest = replacer.New(dynamicValues).Replace(rawRequest)
|
rawRequest = replacer.Replace(rawRequest, dynamicValues)
|
||||||
rawRequestData, err := raw.Parse(rawRequest, baseURL, r.request.Unsafe)
|
rawRequestData, err := raw.Parse(rawRequest, baseURL, r.request.Unsafe)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// rawhttp
|
// Unsafe option uses rawhttp library
|
||||||
if r.request.Unsafe {
|
if r.request.Unsafe {
|
||||||
unsafeReq := &generatedRequest{rawRequest: rawRequestData, meta: generatorValues, original: r.request}
|
unsafeReq := &generatedRequest{rawRequest: rawRequestData, meta: generatorValues, original: r.request}
|
||||||
return unsafeReq, nil
|
return unsafeReq, nil
|
||||||
|
@ -171,12 +174,9 @@ func (r *requestGenerator) handleRawWithPaylods(ctx context.Context, rawRequest,
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// copy headers
|
|
||||||
for key, value := range rawRequestData.Headers {
|
for key, value := range rawRequestData.Headers {
|
||||||
req.Header[key] = []string{value}
|
req.Header[key] = []string{value}
|
||||||
}
|
}
|
||||||
|
|
||||||
request, err := r.fillRequest(req, values)
|
request, err := r.fillRequest(req, values)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -187,14 +187,14 @@ func (r *requestGenerator) handleRawWithPaylods(ctx context.Context, rawRequest,
|
||||||
// fillRequest fills various headers in the request with values
|
// fillRequest fills various headers in the request with values
|
||||||
func (r *requestGenerator) fillRequest(req *http.Request, values map[string]interface{}) (*retryablehttp.Request, error) {
|
func (r *requestGenerator) fillRequest(req *http.Request, values map[string]interface{}) (*retryablehttp.Request, error) {
|
||||||
// Set the header values requested
|
// Set the header values requested
|
||||||
replacer := replacer.New(values)
|
|
||||||
for header, value := range r.request.Headers {
|
for header, value := range r.request.Headers {
|
||||||
req.Header[header] = []string{replacer.Replace(value)}
|
req.Header[header] = []string{replacer.Replace(value, values)}
|
||||||
}
|
}
|
||||||
|
|
||||||
// In case of multiple threads the underlying connection should remain open to allow reuse
|
// In case of multiple threads the underlying connection should remain open to allow reuse
|
||||||
if r.request.Threads <= 0 && req.Header.Get("Connection") == "" {
|
if r.request.Threads <= 0 && req.Header.Get("Connection") == "" {
|
||||||
req.Close = true
|
req.Close = true
|
||||||
|
delete(req.Header, "Connection")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if the user requested a request body
|
// Check if the user requested a request body
|
||||||
|
@ -203,13 +203,11 @@ func (r *requestGenerator) fillRequest(req *http.Request, values map[string]inte
|
||||||
}
|
}
|
||||||
setHeader(req, "User-Agent", "Nuclei - Open-source project (github.com/projectdiscovery/nuclei)")
|
setHeader(req, "User-Agent", "Nuclei - Open-source project (github.com/projectdiscovery/nuclei)")
|
||||||
|
|
||||||
// raw requests are left untouched
|
// Only set these headers on non raw requests
|
||||||
if len(r.request.Raw) > 0 {
|
if len(r.request.Raw) == 0 {
|
||||||
return retryablehttp.FromRequest(req)
|
|
||||||
}
|
|
||||||
setHeader(req, "Accept", "*/*")
|
setHeader(req, "Accept", "*/*")
|
||||||
setHeader(req, "Accept-Language", "en")
|
setHeader(req, "Accept-Language", "en")
|
||||||
|
}
|
||||||
return retryablehttp.FromRequest(req)
|
return retryablehttp.FromRequest(req)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -219,40 +217,3 @@ func setHeader(req *http.Request, name, value string) {
|
||||||
req.Header.Set(name, value)
|
req.Header.Set(name, value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// getPayloadValues returns current payload values for a request
|
|
||||||
func (r *requestGenerator) getPayloadValues(reqURL string, templatePayloads map[string]interface{}) (map[string]interface{}, error) {
|
|
||||||
payloadProcessedValues := make(map[string]interface{})
|
|
||||||
|
|
||||||
for k, v := range templatePayloads {
|
|
||||||
kexp := v.(string)
|
|
||||||
// if it doesn't containing markups, we just continue
|
|
||||||
if !strings.Contains(kexp, replacer.MarkerParenthesisOpen) || strings.Contains(kexp, replacer.MarkerParenthesisClose) || strings.Contains(kexp, replacer.MarkerGeneral) {
|
|
||||||
payloadProcessedValues[k] = v
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// attempts to expand expressions
|
|
||||||
compiled, err := govaluate.NewEvaluableExpressionWithFunctions(kexp, dsl.HelperFunctions())
|
|
||||||
if err != nil {
|
|
||||||
// it is a simple literal payload => proceed with literal value
|
|
||||||
payloadProcessedValues[k] = v
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
// it is an expression - try to solve it
|
|
||||||
expValue, err := compiled.Evaluate(templatePayloads)
|
|
||||||
if err != nil {
|
|
||||||
// an error occurred => proceed with literal value
|
|
||||||
payloadProcessedValues[k] = v
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
payloadProcessedValues[k] = fmt.Sprint(expValue)
|
|
||||||
}
|
|
||||||
var err error
|
|
||||||
if len(payloadProcessedValues) == 0 {
|
|
||||||
err = ErrNoPayload
|
|
||||||
}
|
|
||||||
return payloadProcessedValues, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// ErrNoPayload error to avoid the additional base null request
|
|
||||||
var ErrNoPayload = fmt.Errorf("no payload found")
|
|
||||||
|
|
|
@ -1 +1,68 @@
|
||||||
package http
|
package http
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http/httputil"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/projectdiscovery/nuclei/v2/internal/testutils"
|
||||||
|
"github.com/projectdiscovery/nuclei/v2/pkg/protocols/common/generators"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMakeRequestFromModal(t *testing.T) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMakeRequestFromRaw(t *testing.T) {
|
||||||
|
options := testutils.DefaultOptions
|
||||||
|
|
||||||
|
testutils.Init(options)
|
||||||
|
templateID := "testing-http"
|
||||||
|
request := &Request{
|
||||||
|
ID: templateID,
|
||||||
|
Name: "testing",
|
||||||
|
Payloads: map[string]interface{}{
|
||||||
|
"username": []string{"admin"},
|
||||||
|
"password": []string{"admin", "guest", "password", "test", "12345", "123456"},
|
||||||
|
},
|
||||||
|
AttackType: "clusterbomb",
|
||||||
|
Raw: []string{`GET /manager/html HTTP/1.1
|
||||||
|
Host: {{Hostname}}
|
||||||
|
Authorization: Basic {{base64(username + ':' + password)}}
|
||||||
|
User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/55.0
|
||||||
|
Accept-Language: en-US,en;q=0.9
|
||||||
|
Connection: close`},
|
||||||
|
}
|
||||||
|
executerOpts := testutils.NewMockExecuterOptions(options, &testutils.TemplateInfo{
|
||||||
|
ID: templateID,
|
||||||
|
Info: map[string]interface{}{"severity": "low", "name": "test"},
|
||||||
|
})
|
||||||
|
err := request.Compile(executerOpts)
|
||||||
|
require.Nil(t, err, "could not compile http request")
|
||||||
|
|
||||||
|
generator := request.newGenerator()
|
||||||
|
req, err := generator.Make("https://example.com", map[string]interface{}{})
|
||||||
|
require.Nil(t, err, "could not make http request")
|
||||||
|
|
||||||
|
data, _ := httputil.DumpRequest(req.request.Request, true)
|
||||||
|
fmt.Printf("%s: %+v\n", string(data), req)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetPayloadValues(t *testing.T) {
|
||||||
|
req := &Request{
|
||||||
|
Payloads: map[string]interface{}{
|
||||||
|
"username": []string{"test", "admin", "pass"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
req.generator, err = generators.New(req.Payloads, generators.Sniper, "")
|
||||||
|
require.Nil(t, err, "could not create generators")
|
||||||
|
|
||||||
|
generator := req.newGenerator()
|
||||||
|
_ = generator
|
||||||
|
//values, err := generator.getPayloadValues("https://example.com", map[string]interface{}{
|
||||||
|
// "username": "{{base64('username')}}",
|
||||||
|
//})
|
||||||
|
//fmt.Printf("%+v %+v\n", values, err)
|
||||||
|
}
|
||||||
|
|
|
@ -27,8 +27,7 @@ func (r *Request) ExecuteWithResults(input string, metadata, previous output.Int
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, kv := range r.addresses {
|
for _, kv := range r.addresses {
|
||||||
replacer := replacer.New(map[string]interface{}{"Hostname": address})
|
actualAddress := replacer.Replace(kv.key, map[string]interface{}{"Hostname": address})
|
||||||
actualAddress := replacer.Replace(kv.key)
|
|
||||||
if kv.value != "" {
|
if kv.value != "" {
|
||||||
if strings.Contains(address, ":") {
|
if strings.Contains(address, ":") {
|
||||||
actualAddress, _, _ = net.SplitHostPort(actualAddress)
|
actualAddress, _, _ = net.SplitHostPort(actualAddress)
|
||||||
|
|
Loading…
Reference in New Issue