mirror of https://github.com/daffainfo/nuclei.git
Added dns client pool + misc changes to http client pool
parent
c4428824b6
commit
ff4c61a0eb
|
@ -0,0 +1,78 @@
|
||||||
|
package clientpool
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/projectdiscovery/nuclei/v2/pkg/types"
|
||||||
|
"github.com/projectdiscovery/retryabledns"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
poolMutex *sync.RWMutex
|
||||||
|
normalClient *retryabledns.Client
|
||||||
|
clientPool map[string]*retryabledns.Client
|
||||||
|
)
|
||||||
|
|
||||||
|
// defaultResolvers contains the list of resolvers known to be trusted.
|
||||||
|
var defaultResolvers = []string{
|
||||||
|
"1.1.1.1:53", // Cloudflare
|
||||||
|
"1.0.0.1:53", // Cloudflare
|
||||||
|
"8.8.8.8:53", // Google
|
||||||
|
"8.8.4.4:53", // Google
|
||||||
|
}
|
||||||
|
|
||||||
|
// Init initializes the clientpool implementation
|
||||||
|
func Init(options *types.Options) error {
|
||||||
|
// Don't create clients if already created in past.
|
||||||
|
if normalClient != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
poolMutex = &sync.RWMutex{}
|
||||||
|
clientPool = make(map[string]*retryabledns.Client)
|
||||||
|
|
||||||
|
if client, err := Get(options, &Configuration{}); err != nil {
|
||||||
|
return err
|
||||||
|
} else {
|
||||||
|
normalClient = client
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configuration contains the custom configuration options for a client
|
||||||
|
type Configuration struct {
|
||||||
|
// Retries contains the retries for the dns client
|
||||||
|
Retries int
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hash returns the hash of the configuration to allow client pooling
|
||||||
|
func (c *Configuration) Hash() string {
|
||||||
|
builder := &strings.Builder{}
|
||||||
|
builder.Grow(8)
|
||||||
|
builder.WriteString("r")
|
||||||
|
builder.WriteString(strconv.Itoa(c.Retries))
|
||||||
|
hash := builder.String()
|
||||||
|
return hash
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get creates or gets a client for the protocol based on custom configuration
|
||||||
|
func Get(options *types.Options, configuration *Configuration) (*retryabledns.Client, error) {
|
||||||
|
if !(configuration.Retries > 0) {
|
||||||
|
return normalClient, nil
|
||||||
|
}
|
||||||
|
hash := configuration.Hash()
|
||||||
|
poolMutex.RLock()
|
||||||
|
if client, ok := clientPool[hash]; ok {
|
||||||
|
poolMutex.RUnlock()
|
||||||
|
return client, nil
|
||||||
|
}
|
||||||
|
poolMutex.RUnlock()
|
||||||
|
|
||||||
|
client := retryabledns.New(defaultResolvers, configuration.Retries)
|
||||||
|
|
||||||
|
poolMutex.Lock()
|
||||||
|
clientPool[hash] = client
|
||||||
|
poolMutex.Unlock()
|
||||||
|
return client, nil
|
||||||
|
}
|
|
@ -15,19 +15,34 @@ import (
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/projectdiscovery/fastdialer/fastdialer"
|
"github.com/projectdiscovery/fastdialer/fastdialer"
|
||||||
"github.com/projectdiscovery/nuclei/v2/pkg/types"
|
"github.com/projectdiscovery/nuclei/v2/pkg/types"
|
||||||
|
"github.com/projectdiscovery/rawhttp"
|
||||||
"github.com/projectdiscovery/retryablehttp-go"
|
"github.com/projectdiscovery/retryablehttp-go"
|
||||||
"golang.org/x/net/proxy"
|
"golang.org/x/net/proxy"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
dialer *fastdialer.Dialer
|
dialer *fastdialer.Dialer
|
||||||
poolMutex *sync.RWMutex
|
rawhttpClient *rawhttp.Client
|
||||||
clientPool map[string]*retryablehttp.Client
|
poolMutex *sync.RWMutex
|
||||||
|
normalClient *retryablehttp.Client
|
||||||
|
clientPool map[string]*retryablehttp.Client
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
// Init initializes the clientpool implementation
|
||||||
|
func Init(options *types.Options) error {
|
||||||
|
// Don't create clients if already created in past.
|
||||||
|
if normalClient != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
poolMutex = &sync.RWMutex{}
|
poolMutex = &sync.RWMutex{}
|
||||||
clientPool = make(map[string]*retryablehttp.Client)
|
clientPool = make(map[string]*retryablehttp.Client)
|
||||||
|
|
||||||
|
if client, err := Get(options, &Configuration{}); err != nil {
|
||||||
|
return err
|
||||||
|
} else {
|
||||||
|
normalClient = client
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Configuration contains the custom configuration options for a client
|
// Configuration contains the custom configuration options for a client
|
||||||
|
@ -43,6 +58,7 @@ type Configuration struct {
|
||||||
// Hash returns the hash of the configuration to allow client pooling
|
// Hash returns the hash of the configuration to allow client pooling
|
||||||
func (c *Configuration) Hash() string {
|
func (c *Configuration) Hash() string {
|
||||||
builder := &strings.Builder{}
|
builder := &strings.Builder{}
|
||||||
|
builder.Grow(16)
|
||||||
builder.WriteString("t")
|
builder.WriteString("t")
|
||||||
builder.WriteString(strconv.Itoa(c.Threads))
|
builder.WriteString(strconv.Itoa(c.Threads))
|
||||||
builder.WriteString("m")
|
builder.WriteString("m")
|
||||||
|
@ -53,8 +69,19 @@ func (c *Configuration) Hash() string {
|
||||||
return hash
|
return hash
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetRawHTTP returns the rawhttp request client
|
||||||
|
func GetRawHTTP() *rawhttp.Client {
|
||||||
|
if rawhttpClient == nil {
|
||||||
|
rawhttpClient = rawhttp.NewClient(rawhttp.DefaultOptions)
|
||||||
|
}
|
||||||
|
return rawhttpClient
|
||||||
|
}
|
||||||
|
|
||||||
// Get creates or gets a client for the protocol based on custom configuration
|
// Get creates or gets a client for the protocol based on custom configuration
|
||||||
func Get(options *types.Options, configuration *Configuration) (*retryablehttp.Client, error) {
|
func Get(options *types.Options, configuration *Configuration) (*retryablehttp.Client, error) {
|
||||||
|
if !(configuration.Threads > 0 && configuration.MaxRedirects > 0 && configuration.FollowRedirects) {
|
||||||
|
return normalClient, nil
|
||||||
|
}
|
||||||
var proxyURL *url.URL
|
var proxyURL *url.URL
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
|
@ -139,6 +166,7 @@ func Get(options *types.Options, configuration *Configuration) (*retryablehttp.C
|
||||||
Timeout: time.Duration(options.Timeout) * time.Second,
|
Timeout: time.Duration(options.Timeout) * time.Second,
|
||||||
CheckRedirect: makeCheckRedirectFunc(followRedirects, maxRedirects),
|
CheckRedirect: makeCheckRedirectFunc(followRedirects, maxRedirects),
|
||||||
}, retryablehttpOptions)
|
}, retryablehttpOptions)
|
||||||
|
client.CheckRetry = retryablehttp.HostSprayRetryPolicy()
|
||||||
|
|
||||||
poolMutex.Lock()
|
poolMutex.Lock()
|
||||||
clientPool[hash] = client
|
clientPool[hash] = client
|
||||||
|
|
Loading…
Reference in New Issue