Fixed a bunch of bugs found with go-fuzz
This commit is contained in:
parent
97f53e068d
commit
01f751805e
@ -8,34 +8,28 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func index(w gopher.ResponseWriter, r *gopher.Request) {
|
func index(w gopher.ResponseWriter, r *gopher.Request) {
|
||||||
w.WriteItem(
|
w.WriteItem(&gopher.Item{
|
||||||
gopher.Item{
|
Type: gopher.DIRECTORY,
|
||||||
Type: gopher.DIRECTORY,
|
Selector: "/hello",
|
||||||
Selector: "/hello",
|
Description: "hello",
|
||||||
Description: "hello",
|
|
||||||
|
|
||||||
// TLS Resource
|
// TLS Resource
|
||||||
Host: "localhost",
|
Host: "localhost",
|
||||||
Port: 73,
|
Port: 73,
|
||||||
Extras: []string{"TLS"},
|
Extras: []string{"TLS"},
|
||||||
},
|
})
|
||||||
)
|
w.WriteItem(&gopher.Item{
|
||||||
w.WriteItem(
|
Type: gopher.FILE,
|
||||||
gopher.Item{
|
Selector: "/foo",
|
||||||
Type: gopher.FILE,
|
Description: "foo",
|
||||||
Selector: "/foo",
|
})
|
||||||
Description: "foo",
|
w.WriteItem(&gopher.Item{
|
||||||
},
|
Type: gopher.DIRECTORY,
|
||||||
)
|
Selector: "/",
|
||||||
w.WriteItem(
|
Description: "Floodgap",
|
||||||
gopher.Item{
|
Host: "gopher.floodgap.com",
|
||||||
Type: gopher.DIRECTORY,
|
Port: 70,
|
||||||
Selector: "/",
|
})
|
||||||
Description: "Floodgap",
|
|
||||||
Host: "gopher.floodgap.com",
|
|
||||||
Port: 70,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func hello(w gopher.ResponseWriter, r *gopher.Request) {
|
func hello(w gopher.ResponseWriter, r *gopher.Request) {
|
||||||
|
149
gopher.go
149
gopher.go
@ -157,8 +157,61 @@ type Item struct {
|
|||||||
Extras []string `json:"extras"`
|
Extras []string `json:"extras"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ParseItem parses a line of text into an item
|
||||||
|
func ParseItem(line string) (item *Item, err error) {
|
||||||
|
parts := strings.Split(strings.Trim(line, "\r\n"), "\t")
|
||||||
|
|
||||||
|
if len(parts[0]) < 1 {
|
||||||
|
return nil, errors.New("no item type: " + string(line))
|
||||||
|
}
|
||||||
|
|
||||||
|
item = &Item{
|
||||||
|
Type: ItemType(parts[0][0]),
|
||||||
|
Description: string(parts[0][1:]),
|
||||||
|
Extras: make([]string, 0),
|
||||||
|
}
|
||||||
|
|
||||||
|
// Selector
|
||||||
|
if len(parts) > 1 {
|
||||||
|
item.Selector = string(parts[1])
|
||||||
|
} else {
|
||||||
|
item.Selector = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Host
|
||||||
|
if len(parts) > 2 {
|
||||||
|
item.Host = string(parts[2])
|
||||||
|
} else {
|
||||||
|
item.Host = "null.host"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Port
|
||||||
|
if len(parts) > 3 {
|
||||||
|
port, err := strconv.Atoi(string(parts[3]))
|
||||||
|
if err != nil {
|
||||||
|
// Ignore parsing errors for bad servers for INFO types
|
||||||
|
if item.Type != INFO {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
item.Port = 0
|
||||||
|
}
|
||||||
|
item.Port = port
|
||||||
|
} else {
|
||||||
|
item.Port = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extras
|
||||||
|
if len(parts) >= 4 {
|
||||||
|
for _, v := range parts[4:] {
|
||||||
|
item.Extras = append(item.Extras, string(v))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// MarshalJSON serializes an Item into a JSON structure
|
// MarshalJSON serializes an Item into a JSON structure
|
||||||
func (i Item) MarshalJSON() ([]byte, error) {
|
func (i *Item) MarshalJSON() ([]byte, error) {
|
||||||
return json.Marshal(struct {
|
return json.Marshal(struct {
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
Description string `json:"description"`
|
Description string `json:"description"`
|
||||||
@ -177,7 +230,7 @@ func (i Item) MarshalJSON() ([]byte, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// MarshalText serializes an Item into an array of bytes
|
// MarshalText serializes an Item into an array of bytes
|
||||||
func (i Item) MarshalText() ([]byte, error) {
|
func (i *Item) MarshalText() ([]byte, error) {
|
||||||
b := []byte{}
|
b := []byte{}
|
||||||
b = append(b, byte(i.Type))
|
b = append(b, byte(i.Type))
|
||||||
b = append(b, []byte(i.Description)...)
|
b = append(b, []byte(i.Description)...)
|
||||||
@ -198,51 +251,6 @@ func (i Item) MarshalText() ([]byte, error) {
|
|||||||
return b, nil
|
return b, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Item) parse(line string) error {
|
|
||||||
parts := strings.Split(line, "\t")
|
|
||||||
|
|
||||||
if len(parts[0]) < 1 {
|
|
||||||
return errors.New("no item type: " + string(line))
|
|
||||||
}
|
|
||||||
|
|
||||||
i.Type = ItemType(parts[0][0])
|
|
||||||
i.Description = string(parts[0][1:])
|
|
||||||
|
|
||||||
if len(parts) > 1 {
|
|
||||||
i.Selector = string(parts[1])
|
|
||||||
} else {
|
|
||||||
i.Selector = ""
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(parts) > 2 {
|
|
||||||
i.Host = string(parts[2])
|
|
||||||
} else {
|
|
||||||
i.Host = "null.host"
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(parts) > 3 {
|
|
||||||
port, err := strconv.Atoi(string(parts[3]))
|
|
||||||
if err != nil {
|
|
||||||
// Ignore parsing errors for bad servers for INFO types
|
|
||||||
if i.Type != INFO {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
i.Port = 0
|
|
||||||
}
|
|
||||||
i.Port = port
|
|
||||||
} else {
|
|
||||||
i.Port = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(parts) >= 4 {
|
|
||||||
for _, v := range parts[4:] {
|
|
||||||
i.Extras = append(i.Extras, string(v))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (i *Item) isDirectoryLike() bool {
|
func (i *Item) isDirectoryLike() bool {
|
||||||
switch i.Type {
|
switch i.Type {
|
||||||
case DIRECTORY:
|
case DIRECTORY:
|
||||||
@ -256,7 +264,7 @@ func (i *Item) isDirectoryLike() bool {
|
|||||||
|
|
||||||
// Directory representes a Gopher Menu of Items
|
// Directory representes a Gopher Menu of Items
|
||||||
type Directory struct {
|
type Directory struct {
|
||||||
Items []Item `json:"items"`
|
Items []*Item `json:"items"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ToJSON returns the Directory as JSON bytes
|
// ToJSON returns the Directory as JSON bytes
|
||||||
@ -401,7 +409,7 @@ func (i *Item) FetchDirectory() (Directory, error) {
|
|||||||
scanner := bufio.NewScanner(reader)
|
scanner := bufio.NewScanner(reader)
|
||||||
scanner.Split(bufio.ScanLines)
|
scanner.Split(bufio.ScanLines)
|
||||||
|
|
||||||
var items []Item
|
var items []*Item
|
||||||
|
|
||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
line := strings.Trim(scanner.Text(), "\r\n")
|
line := strings.Trim(scanner.Text(), "\r\n")
|
||||||
@ -414,8 +422,7 @@ func (i *Item) FetchDirectory() (Directory, error) {
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
item := Item{}
|
item, err := ParseItem(line)
|
||||||
err := item.parse(line)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Error parsing %q: %q", line, err)
|
log.Printf("Error parsing %q: %q", line, err)
|
||||||
continue
|
continue
|
||||||
@ -1032,7 +1039,7 @@ type ResponseWriter interface {
|
|||||||
WriteInfo(msg string) error
|
WriteInfo(msg string) error
|
||||||
|
|
||||||
// WriteItem writes an item
|
// WriteItem writes an item
|
||||||
WriteItem(i Item) error
|
WriteItem(i *Item) error
|
||||||
}
|
}
|
||||||
|
|
||||||
// A response represents the server side of a Gopher response.
|
// A response represents the server side of a Gopher response.
|
||||||
@ -1072,7 +1079,7 @@ func (w *response) WriteError(err string) error {
|
|||||||
return e
|
return e
|
||||||
}
|
}
|
||||||
|
|
||||||
i := Item{
|
i := &Item{
|
||||||
Type: ERROR,
|
Type: ERROR,
|
||||||
Description: err,
|
Description: err,
|
||||||
Host: "error.host",
|
Host: "error.host",
|
||||||
@ -1092,7 +1099,7 @@ func (w *response) WriteInfo(msg string) error {
|
|||||||
return e
|
return e
|
||||||
}
|
}
|
||||||
|
|
||||||
i := Item{
|
i := &Item{
|
||||||
Type: INFO,
|
Type: INFO,
|
||||||
Description: msg,
|
Description: msg,
|
||||||
Host: "error.host",
|
Host: "error.host",
|
||||||
@ -1102,7 +1109,7 @@ func (w *response) WriteInfo(msg string) error {
|
|||||||
return w.WriteItem(i)
|
return w.WriteItem(i)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *response) WriteItem(i Item) error {
|
func (w *response) WriteItem(i *Item) error {
|
||||||
if w.rt == 0 {
|
if w.rt == 0 {
|
||||||
w.rt = 2
|
w.rt = 2
|
||||||
}
|
}
|
||||||
@ -1275,15 +1282,13 @@ func dirList(w ResponseWriter, r *Request, f File, fs FileSystem) {
|
|||||||
Error(w, "Error reading directory")
|
Error(w, "Error reading directory")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
w.WriteItem(
|
w.WriteItem(&Item{
|
||||||
Item{
|
Type: DIRECTORY,
|
||||||
Type: DIRECTORY,
|
Description: file.Name(),
|
||||||
Description: file.Name(),
|
Selector: pathname,
|
||||||
Selector: pathname,
|
Host: r.LocalHost,
|
||||||
Host: r.LocalHost,
|
Port: r.LocalPort,
|
||||||
Port: r.LocalPort,
|
})
|
||||||
},
|
|
||||||
)
|
|
||||||
} else if file.Mode()&os.ModeType == 0 {
|
} else if file.Mode()&os.ModeType == 0 {
|
||||||
pathname, err := filepath.Rel(
|
pathname, err := filepath.Rel(
|
||||||
root,
|
root,
|
||||||
@ -1296,15 +1301,13 @@ func dirList(w ResponseWriter, r *Request, f File, fs FileSystem) {
|
|||||||
|
|
||||||
itemtype := GetItemType(path.Join(fullpath, file.Name()))
|
itemtype := GetItemType(path.Join(fullpath, file.Name()))
|
||||||
|
|
||||||
w.WriteItem(
|
w.WriteItem(&Item{
|
||||||
Item{
|
Type: itemtype,
|
||||||
Type: itemtype,
|
Description: file.Name(),
|
||||||
Description: file.Name(),
|
Selector: pathname,
|
||||||
Selector: pathname,
|
Host: r.LocalHost,
|
||||||
Host: r.LocalHost,
|
Port: r.LocalPort,
|
||||||
Port: r.LocalPort,
|
})
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -36,30 +36,89 @@ func TestGet(t *testing.T) {
|
|||||||
assert := assert.New(t)
|
assert := assert.New(t)
|
||||||
|
|
||||||
res, err := gopher.Get("gopher://localhost:7000/1hello")
|
res, err := gopher.Get("gopher://localhost:7000/1hello")
|
||||||
assert.Nil(err)
|
assert.NoError(err)
|
||||||
|
|
||||||
b, err := res.Dir.ToText()
|
|
||||||
assert.Nil(err)
|
|
||||||
|
|
||||||
t.Logf("res: %s", string(b))
|
|
||||||
|
|
||||||
assert.Len(res.Dir.Items, 1)
|
assert.Len(res.Dir.Items, 1)
|
||||||
|
|
||||||
assert.Equal(res.Dir.Items[0].Type, gopher.INFO)
|
assert.Equal(res.Dir.Items[0].Type, gopher.INFO)
|
||||||
assert.Equal(res.Dir.Items[0].Description, "Hello World!")
|
assert.Equal(res.Dir.Items[0].Description, "Hello World!")
|
||||||
|
|
||||||
|
out, err := res.Dir.ToText()
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.Equal(string(out), "iHello World!\t\terror.host\t1\r\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFileServer(t *testing.T) {
|
func TestFileServer(t *testing.T) {
|
||||||
assert := assert.New(t)
|
assert := assert.New(t)
|
||||||
|
|
||||||
res, err := gopher.Get("gopher://localhost:7000/")
|
res, err := gopher.Get("gopher://localhost:7000/")
|
||||||
assert.Nil(err)
|
assert.NoError(err)
|
||||||
assert.Len(res.Dir.Items, 5)
|
assert.Len(res.Dir.Items, 5)
|
||||||
|
|
||||||
json, err := res.Dir.ToJSON()
|
json, err := res.Dir.ToJSON()
|
||||||
assert.Nil(err)
|
assert.Nil(err)
|
||||||
|
|
||||||
assert.JSONEq(string(json), `{"items":[{"type":"0","description":"LICENSE","selector":"LICENSE","host":"127.0.0.1","port":7000,"extras":null},{"type":"0","description":"README.md","selector":"README.md","host":"127.0.0.1","port":7000,"extras":null},{"type":"1","description":"examples","selector":"examples","host":"127.0.0.1","port":7000,"extras":null},{"type":"0","description":"gopher.go","selector":"gopher.go","host":"127.0.0.1","port":7000,"extras":null},{"type":"0","description":"gopher_test.go","selector":"gopher_test.go","host":"127.0.0.1","port":7000,"extras":null}]}`)
|
assert.JSONEq(string(json), `{"items":[{"type":"0","description":"LICENSE","selector":"LICENSE","host":"127.0.0.1","port":7000,"extras":[]},{"type":"0","description":"README.md","selector":"README.md","host":"127.0.0.1","port":7000,"extras":[]},{"type":"1","description":"examples","selector":"examples","host":"127.0.0.1","port":7000,"extras":[]},{"type":"0","description":"gopher.go","selector":"gopher.go","host":"127.0.0.1","port":7000,"extras":[]},{"type":"0","description":"gopher_test.go","selector":"gopher_test.go","host":"127.0.0.1","port":7000,"extras":[]}]}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseItemNull(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
item, err := gopher.ParseItem("")
|
||||||
|
assert.Nil(item)
|
||||||
|
assert.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseItem(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
item, err := gopher.ParseItem("0foo\t/foo\tlocalhost\t70\r\n")
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.NotNil(item)
|
||||||
|
assert.Equal(item, &gopher.Item{
|
||||||
|
Type: gopher.FILE,
|
||||||
|
Description: "foo",
|
||||||
|
Selector: "/foo",
|
||||||
|
Host: "localhost",
|
||||||
|
Port: 70,
|
||||||
|
Extras: []string{},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseItemMarshal(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
data := "0foo\t/foo\tlocalhost\t70\r\n"
|
||||||
|
item, err := gopher.ParseItem(data)
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.NotNil(item)
|
||||||
|
assert.Equal(item, &gopher.Item{
|
||||||
|
Type: gopher.FILE,
|
||||||
|
Description: "foo",
|
||||||
|
Selector: "/foo",
|
||||||
|
Host: "localhost",
|
||||||
|
Port: 70,
|
||||||
|
Extras: []string{},
|
||||||
|
})
|
||||||
|
|
||||||
|
data1, err := item.MarshalText()
|
||||||
|
assert.Nil(err)
|
||||||
|
assert.Equal(data, string(data1))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseItemMarshalIdempotency(t *testing.T) {
|
||||||
|
assert := assert.New(t)
|
||||||
|
|
||||||
|
data := "0"
|
||||||
|
item, err := gopher.ParseItem(data)
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.NotNil(item)
|
||||||
|
|
||||||
|
data1, err := item.MarshalText()
|
||||||
|
assert.Nil(err)
|
||||||
|
|
||||||
|
item1, err := gopher.ParseItem(string(data1))
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.NotNil(item1)
|
||||||
|
assert.Equal(item, item1)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestMain(m *testing.M) {
|
func TestMain(m *testing.M) {
|
||||||
|
Loading…
Reference in New Issue
Block a user