zgrab2/processing.go

183 lines
4.2 KiB
Go

package zgrab2
import (
"bufio"
"encoding/json"
"fmt"
"io"
"net"
"strings"
"sync"
"time"
log "github.com/sirupsen/logrus"
)
// Grab contains all scan responses for a single host
type Grab struct {
IP string `json:"ip,omitempty"`
Domain string `json:"domain,omitempty"`
Data map[string]ScanResponse `json:"data,omitempty"`
}
// ScanTarget is the host that will be scanned
type ScanTarget struct {
IP net.IP
Domain string
}
// scanTargetConnection wraps an existing net.Conn connection, overriding the Read/Write methods to use the configured timeouts
type scanTargetConnection struct {
net.Conn
Timeout time.Duration
}
func (c *scanTargetConnection) Read(b []byte) (n int, err error) {
if c.Timeout > 0 {
if err = c.Conn.SetReadDeadline(time.Now().Add(c.Timeout)); err != nil {
return 0, err
}
}
return c.Conn.Read(b)
}
func (c *scanTargetConnection) Write(b []byte) (n int, err error) {
if c.Timeout > 0 {
if err = c.Conn.SetWriteDeadline(time.Now().Add(c.Timeout)); err != nil {
return 0, err
}
}
return c.Conn.Write(b)
}
// ScanTarget.Open connects to the ScanTarget using the configured flags, and returns a net.Conn that uses the configured timeouts for Read/Write operations.
func (t *ScanTarget) Open(flags *BaseFlags) (net.Conn, error) {
timeout := time.Second * time.Duration(flags.Timeout)
target := fmt.Sprintf("%s:%d", t.IP.String(), flags.Port)
var conn net.Conn
var err error
if timeout > 0 {
conn, err = net.DialTimeout("tcp", target, timeout)
} else {
conn, err = net.Dial("tcp", target)
}
if err != nil {
if conn != nil {
conn.Close()
}
return nil, err
}
return &scanTargetConnection{
Conn: conn,
Timeout: timeout,
}, nil
}
// grabTarget calls handler for each action
func grabTarget(input ScanTarget, m *Monitor) []byte {
moduleResult := make(map[string]ScanResponse)
for _, scannerName := range orderedScanners {
scanner := scanners[scannerName]
name, res := RunScanner(*scanner, m, input)
moduleResult[name] = res
if res.Error != nil && !config.Multiple.ContinueOnError {
break
}
}
var ipstr string
if input.IP == nil {
ipstr = ""
} else {
s := input.IP.String()
ipstr = s
}
a := Grab{IP: ipstr, Domain: input.Domain, Data: moduleResult}
result, err := json.Marshal(a)
if err != nil {
log.Fatalf("unable to marshal data: %s", err)
}
return result
}
// Process sets up an output encoder, input reader, and starts grab workers
func Process(mon *Monitor) {
workers := config.Senders
processQueue := make(chan ScanTarget, workers*4)
outputQueue := make(chan []byte, workers*4)
//Create wait groups
var workerDone sync.WaitGroup
var outputDone sync.WaitGroup
workerDone.Add(int(workers))
outputDone.Add(1)
// Start the output encoder
go func() {
out := bufio.NewWriter(config.outputFile)
defer outputDone.Done()
defer out.Flush()
for result := range outputQueue {
if _, err := out.Write(result); err != nil {
log.Fatal(err)
}
if err := out.WriteByte('\n'); err != nil {
log.Fatal(err)
}
}
}()
//Start all the workers
for i := 0; i < workers; i++ {
go func(i int) {
for _, scannerName := range orderedScanners {
scanner := *scanners[scannerName]
scanner.InitPerSender(i)
}
for obj := range processQueue {
for run := uint(0); run < uint(config.ConnectionsPerHost); run++ {
result := grabTarget(obj, mon)
outputQueue <- result
}
}
workerDone.Done()
}(i)
}
// Read the input, send to workers
input := bufio.NewReader(config.inputFile)
for {
obj, err := input.ReadBytes('\n')
if err == io.EOF {
break
} else if err != nil {
log.Error(err)
}
st := strings.TrimSpace(string(obj))
ipnet, domain, err := ParseTarget(st)
if err != nil {
log.Error(err)
continue
}
var ip net.IP
if ipnet != nil {
if ipnet.Mask != nil {
for ip = ipnet.IP.Mask(ipnet.Mask); ipnet.Contains(ip); incrementIP(ip) {
processQueue <- ScanTarget{IP: duplicateIP(ip), Domain: domain}
}
continue
} else {
ip = ipnet.IP
}
}
processQueue <- ScanTarget{IP: ip, Domain: domain}
}
close(processQueue)
workerDone.Wait()
close(outputQueue)
outputDone.Wait()
}