ffuf/pkg/input/wordlist.go
Ice3man 3d8e233097 Added ignore comment option (#138)
* Added ignore comment option

* Ignore blank lines & added changelog entry
2020-01-15 11:19:18 +02:00

168 lines
3.6 KiB
Go

package input
import (
"bufio"
"os"
"regexp"
"strings"
"github.com/ffuf/ffuf/pkg/ffuf"
)
type WordlistInput struct {
config *ffuf.Config
data [][]byte
position int
keyword string
}
func NewWordlistInput(keyword string, value string, conf *ffuf.Config) (*WordlistInput, error) {
var wl WordlistInput
wl.keyword = keyword
wl.config = conf
wl.position = 0
var valid bool
var err error
// stdin?
if value == "-" {
// yes
valid = true
} else {
// no
valid, err = wl.validFile(value)
}
if err != nil {
return &wl, err
}
if valid {
err = wl.readFile(value)
}
return &wl, err
}
//Position will return the current position in the input list
func (w *WordlistInput) Position() int {
return w.position
}
//ResetPosition resets the position back to beginning of the wordlist.
func (w *WordlistInput) ResetPosition() {
w.position = 0
}
//Keyword returns the keyword assigned to this InternalInputProvider
func (w *WordlistInput) Keyword() string {
return w.keyword
}
//Next will increment the cursor position, and return a boolean telling if there's words left in the list
func (w *WordlistInput) Next() bool {
if w.position >= len(w.data) {
return false
}
return true
}
//IncrementPosition will increment the current position in the inputprovider data slice
func (w *WordlistInput) IncrementPosition() {
w.position += 1
}
//Value returns the value from wordlist at current cursor position
func (w *WordlistInput) Value() []byte {
return w.data[w.position]
}
//Total returns the size of wordlist
func (w *WordlistInput) Total() int {
return len(w.data)
}
//validFile checks that the wordlist file exists and can be read
func (w *WordlistInput) validFile(path string) (bool, error) {
_, err := os.Stat(path)
if err != nil {
return false, err
}
f, err := os.Open(path)
if err != nil {
return false, err
}
f.Close()
return true, nil
}
//readFile reads the file line by line to a byte slice
func (w *WordlistInput) readFile(path string) error {
var file *os.File
var err error
if path == "-" {
file = os.Stdin
} else {
file, err = os.Open(path)
if err != nil {
return err
}
}
defer file.Close()
var data [][]byte
var ok bool
reader := bufio.NewScanner(file)
re := regexp.MustCompile(`(?i)%ext%`)
for reader.Scan() {
if w.config.DirSearchCompat && len(w.config.Extensions) > 0 {
text := []byte(reader.Text())
if re.Match(text) {
for _, ext := range w.config.Extensions {
contnt := re.ReplaceAll(text, []byte(ext))
data = append(data, []byte(contnt))
}
} else {
text := reader.Text()
if w.config.IgnoreWordlistComments {
text, ok = stripComments(text)
if !ok {
continue
}
}
data = append(data, []byte(text))
}
} else {
text := reader.Text()
if w.config.IgnoreWordlistComments {
text, ok = stripComments(text)
if !ok {
continue
}
}
data = append(data, []byte(text))
if w.keyword == "FUZZ" && len(w.config.Extensions) > 0 {
for _, ext := range w.config.Extensions {
data = append(data, []byte(text+ext))
}
}
}
}
w.data = data
return reader.Err()
}
// stripComments removes all kind of comments from the word
func stripComments(text string) (string, bool) {
// If the line starts with a # ignoring any space on the left,
// return blank.
if strings.HasPrefix(strings.TrimLeft(text, " "), "#") {
return "", false
}
// If the line has # later after a space, that's a comment.
// Only send the word upto space to the routine.
index := strings.Index(text, " #")
if index == -1 {
return text, true
}
return text[:index], true
}