evio/examples/http-server/main.go

158 lines
3.7 KiB
Go

package main
import (
"fmt"
"log"
"strconv"
"strings"
"time"
"github.com/tidwall/evio"
)
type request struct {
proto, method string
path, query string
head, body string
remoteAddr string
}
type conn struct {
addr evio.Addr
is evio.InputStream
}
func main() {
var events evio.Events
var conns = make(map[int]*conn)
events.Serving = func(wakefn func(id int) bool) (action evio.Action) {
log.Print("http server started on port 8080")
return
}
events.Opened = func(id int, addr evio.Addr) (out []byte, opts evio.Options, action evio.Action) {
conns[id] = &conn{addr: addr}
log.Printf("%s: opened", addr.Remote.String())
return
}
events.Closed = func(id int) (action evio.Action) {
c := conns[id]
log.Printf("%s: closed", c.addr.Remote.String())
delete(conns, id)
return
}
events.Data = func(id int, in []byte) (out []byte, action evio.Action) {
c := conns[id]
data := c.is.Begin(in)
// process the pipeline
var req request
for {
leftover, err := parsereq(data, &req)
if err != nil {
// bad thing happened
out = appendresp(out, "500 Error", "", err.Error()+"\n")
action = evio.Close
break
} else if len(leftover) == len(data) {
// request not ready, yet
break
}
// handle the request
req.remoteAddr = c.addr.Remote.String()
out = appendhandle(out, &req)
data = leftover
}
c.is.End(data)
return
}
log.Fatal(evio.Serve(events, "tcp://0.0.0.0:8080"))
}
// appendhandle handles the incoming request and appends the response to
// the provided bytes, which is then returned to the caller.
func appendhandle(b []byte, req *request) []byte {
return appendresp(b, "200 OK", "", "Hello World!\n")
}
// appendresp will append a valid http response to the provide bytes.
// The status param should be the code plus text such as "200 OK".
// The head parameter should be a series of lines ending with "\r\n" or empty.
func appendresp(b []byte, status, head, body string) []byte {
b = append(b, "HTTP/1.1"...)
b = append(b, ' ')
b = append(b, status...)
b = append(b, '\r', '\n')
b = append(b, "Date: "...)
b = time.Now().AppendFormat(b, "Mon, 02 Jan 2006 15:04:05 GMT")
b = append(b, '\r', '\n')
if len(body) > 0 {
b = append(b, "Content-Length: "...)
b = strconv.AppendInt(b, int64(len(body)), 10)
b = append(b, '\r', '\n')
}
b = append(b, head...)
b = append(b, '\r', '\n')
if len(body) > 0 {
b = append(b, body...)
}
return b
}
// parsereq is a very simple http request parser. This operation
// waits for the entire payload to be buffered before returning a
// valid request.
func parsereq(data []byte, req *request) (leftover []byte, err error) {
var s int
var n int
var top string
var clen int
var i int
for ; i < len(data); i++ {
if i > 1 && data[i] == '\n' && data[i-1] == '\r' {
line := string(data[s : i-1])
s = i + 1
if n == 0 {
top = line
parts := strings.Split(top, " ")
if len(parts) != 3 {
return data, fmt.Errorf("malformed request '%s'", top)
}
req.method = parts[0]
req.path = parts[1]
req.proto = parts[2]
for i := 0; i < len(req.path); i++ {
if req.path[i] == '?' {
req.query = req.path[i+1:]
req.path = req.path[:i]
break
}
}
} else if line == "" {
req.head = string(data[len(top)+2 : i+1])
i++
if clen > 0 {
if len(data[i:]) < clen {
break
}
req.body = string(data[i : i+clen])
i += clen
}
return data[i:], nil
} else {
if strings.HasPrefix(line, "Content-Length:") {
n, err := strconv.ParseInt(strings.TrimSpace(line[len("Content-Length:"):]), 10, 64)
if err == nil {
clen = int(n)
}
}
}
n++
}
}
// not enough data
return data, nil
}