Added experimental banner grabber implemented in Go.
(Caution: This is my first attempt at writing Go.)
This commit is contained in:
parent
853524c2ae
commit
aa5580ff1a
36
examples/banner-grab-go/README
Normal file
36
examples/banner-grab-go/README
Normal file
@ -0,0 +1,36 @@
|
||||
TCP banner grabber implemented in Go (experimental)
|
||||
======
|
||||
|
||||
This program will make TCP connections to IP addresses provide on
|
||||
stdin, optionally send them a short message, and wait for their
|
||||
responses. Each response is printed to stdout, along with the
|
||||
responding host's IP address. Status messages appear on stderr.
|
||||
|
||||
USING:
|
||||
-----
|
||||
go build banner.go
|
||||
zmap -p 80 -N 1000 -o - | ./banner -port 80 -concurrent 100 -data http-req > banners.out
|
||||
|
||||
|
||||
OPTIONS:
|
||||
-----
|
||||
-concurrent Number of connections that can be going on at once.
|
||||
This, combined with timeouts, will decide the maximum
|
||||
rate at which banners are grabbed. If this value
|
||||
is set higher than 1000, you should use
|
||||
`ulimit -SHn 1000000` and `ulimit -SSn 1000000` to
|
||||
avoid running out of file descriptors (typically capped
|
||||
at 1024). Default: 100.
|
||||
|
||||
-port The port which to connect to hosts on. Default: 80.
|
||||
|
||||
-timeout Connection timeout (seconds). Give up on a host if grabber
|
||||
has not completed by this time. Default: 4 seconds.
|
||||
|
||||
-format Format to output banner responses. One of 'hex', 'ascii',
|
||||
or 'base64'. Default: ascii.
|
||||
|
||||
-d, --data Optional data file. This data will be sent to each host
|
||||
upon successful connection. Occurrences of the
|
||||
string '%s' will be replaced with the current
|
||||
target host's address.
|
160
examples/banner-grab-go/banner.go
Normal file
160
examples/banner-grab-go/banner.go
Normal file
@ -0,0 +1,160 @@
|
||||
/*
|
||||
TCP banner grabber, implemented in go
|
||||
|
||||
This program will make TCP connections to IP addresses provide on
|
||||
stdin, optionally send them a short message, and wait for their
|
||||
responses. Each response is printed to stdout, along with the
|
||||
responding host's IP address. Status messages appear on stderr.
|
||||
*/
|
||||
|
||||
package main
|
||||
|
||||
/*
|
||||
* banner.go Copyright 2013 Regents of the University of Michigan
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
||||
* use this file except in compliance with the License. You may obtain a copy
|
||||
* of the License at http://www.apache.org/licenses/LICENSE-2.0
|
||||
*/
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"encoding/base64"
|
||||
"encoding/hex"
|
||||
"flag"
|
||||
"fmt"
|
||||
"io"
|
||||
"net"
|
||||
"os"
|
||||
"strings"
|
||||
"syscall"
|
||||
"time"
|
||||
)
|
||||
|
||||
var (
|
||||
nConnectFlag = flag.Int("concurrent", 100, "Number of concurrent connections")
|
||||
portFlag = flag.String("port", "80", "Destination port")
|
||||
formatFlag = flag.String("format", "ascii", "Output format for responses ('ascii', 'hex', or 'base64')")
|
||||
timeoutFlag = flag.Int("timeout", 4, "Seconds to wait for each host to respond")
|
||||
dataFileFlag = flag.String("data", "", "File containing message to send to responsive hosts ('%s' will be replaced with host IP)")
|
||||
)
|
||||
|
||||
var messageData = make([]byte, 0) // data read from file specified with dataFile flag
|
||||
|
||||
// Before running main, parse flags and load message data, if applicable
|
||||
func init() {
|
||||
flag.Parse()
|
||||
if *dataFileFlag != "" {
|
||||
fi, err := os.Open(*dataFileFlag)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
buf := make([]byte, 1024)
|
||||
n, err := fi.Read(buf)
|
||||
messageData = buf[0:n]
|
||||
if err != nil && err != io.EOF {
|
||||
panic(err)
|
||||
}
|
||||
fi.Close()
|
||||
}
|
||||
// Increase file descriptor limit
|
||||
rlimit := syscall.Rlimit{Max: uint64(*nConnectFlag + 4), Cur: uint64(*nConnectFlag + 4)}
|
||||
if err := syscall.Setrlimit(syscall.RLIMIT_NOFILE, &rlimit); err != nil {
|
||||
fmt.Fprintf(os.Stderr, "Error setting rlimit: %s", err)
|
||||
}
|
||||
}
|
||||
|
||||
type resultStruct struct {
|
||||
addr string // address of remote host
|
||||
data []byte // data returned from the host, if successful
|
||||
err error // error, if any
|
||||
}
|
||||
|
||||
// Read addresses from addrChan and grab banners from these hosts.
|
||||
// Sends resultStructs to resultChan. Writes to doneChan when complete.
|
||||
func grabber(addrChan chan string, resultChan chan resultStruct, doneChan chan int) {
|
||||
for addr := range addrChan {
|
||||
deadline := time.Now().Add(time.Duration(*timeoutFlag) * time.Second)
|
||||
dialer := net.Dialer{Deadline: deadline}
|
||||
conn, err := dialer.Dial("tcp", net.JoinHostPort(addr, *portFlag))
|
||||
if err != nil {
|
||||
resultChan <- resultStruct{addr, nil, err}
|
||||
continue
|
||||
}
|
||||
conn.SetDeadline(deadline)
|
||||
if len(messageData) > 0 {
|
||||
s := strings.Replace(string(messageData), "%s", addr, -1)
|
||||
if _, err := conn.Write([]byte(s)); err != nil {
|
||||
conn.Close()
|
||||
resultChan <- resultStruct{addr, nil, err}
|
||||
continue
|
||||
}
|
||||
}
|
||||
var buf [1024]byte
|
||||
n, err := conn.Read(buf[:])
|
||||
conn.Close()
|
||||
if err != nil && (err != io.EOF || n == 0) {
|
||||
resultChan <- resultStruct{addr, nil, err}
|
||||
continue
|
||||
}
|
||||
resultChan <- resultStruct{addr, buf[0:n], nil}
|
||||
}
|
||||
doneChan <- 1
|
||||
}
|
||||
|
||||
// Read resultStructs from resultChan, print output, and maintain
|
||||
// status counters. Writes to doneChan when complete.
|
||||
func output(resultChan chan resultStruct, doneChan chan int) {
|
||||
ok, timeout, error := 0, 0, 0
|
||||
for result := range resultChan {
|
||||
if result.err == nil {
|
||||
switch *formatFlag {
|
||||
case "hex":
|
||||
fmt.Printf("%s: %s\n", result.addr,
|
||||
hex.EncodeToString(result.data))
|
||||
case "base64":
|
||||
fmt.Printf("%s: %s\n", result.addr,
|
||||
base64.StdEncoding.EncodeToString(result.data))
|
||||
default:
|
||||
fmt.Printf("%s: %s\n", result.addr,
|
||||
string(result.data))
|
||||
}
|
||||
ok++
|
||||
} else if nerr, ok := result.err.(net.Error); ok && nerr.Timeout() {
|
||||
fmt.Fprintf(os.Stderr, "%s: Timeout\n", result.addr)
|
||||
timeout++
|
||||
} else {
|
||||
fmt.Fprintf(os.Stderr, "%s: Error %s\n", result.addr, result.err)
|
||||
error++
|
||||
}
|
||||
}
|
||||
fmt.Fprintf(os.Stderr, "Complete (OK=%d, timeout=%d, error=%d)\n",
|
||||
ok, timeout, error)
|
||||
doneChan <- 1
|
||||
}
|
||||
|
||||
func main() {
|
||||
addrChan := make(chan string, *nConnectFlag) // pass addresses to grabbers
|
||||
resultChan := make(chan resultStruct, *nConnectFlag) // grabbers send results to output
|
||||
doneChan := make(chan int, *nConnectFlag) // let grabbers signal completion
|
||||
|
||||
// Start grabbers and output thread
|
||||
go output(resultChan, doneChan)
|
||||
for i := 0; i < *nConnectFlag; i++ {
|
||||
go grabber(addrChan, resultChan, doneChan)
|
||||
}
|
||||
|
||||
// Read addresses from stdin and pass to grabbers
|
||||
scanner := bufio.NewScanner(os.Stdin)
|
||||
for scanner.Scan() {
|
||||
addrChan <- scanner.Text()
|
||||
}
|
||||
close(addrChan)
|
||||
|
||||
// Wait for completion
|
||||
for i := 0; i < *nConnectFlag; i++ {
|
||||
<-doneChan
|
||||
}
|
||||
close(resultChan)
|
||||
<-doneChan
|
||||
}
|
3
examples/banner-grab-go/http-req
Normal file
3
examples/banner-grab-go/http-req
Normal file
@ -0,0 +1,3 @@
|
||||
GET / HTTP/1.1
|
||||
Host: %s
|
||||
|
Loading…
Reference in New Issue
Block a user