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 string 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 string) (out []byte, opts evio.Options, action evio.Action) { conns[id] = &conn{addr: addr} log.Printf("%s: opened", addr) return } events.Closed = func(id int) (action evio.Action) { c := conns[id] log.Printf("%s: closed", c.addr) 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 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 }