[cmd.gwstester] added sample code to test while developing [http.upgrade.request.private|public|types] http upgrade request parser beginning [internal.http.reader] http header parser working for now
This commit is contained in:
parent
d0b83bc272
commit
25879a3310
|
@ -0,0 +1,59 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
"git.xdrm.io/gws/http/upgrade/request"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
|
||||||
|
startTime := time.Now().UnixNano()
|
||||||
|
|
||||||
|
/* (1) Create listening socket
|
||||||
|
---------------------------------------------------------*/
|
||||||
|
/* (1) Create socket */
|
||||||
|
lsock, err := net.Listen("tcp", ":4444")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (2) Accept clients */
|
||||||
|
for {
|
||||||
|
|
||||||
|
sock, err := lsock.Accept()
|
||||||
|
if err != nil {
|
||||||
|
os.Stderr.WriteString(fmt.Sprintf("Connection error: %s\n", err))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Printf("+ new client\n")
|
||||||
|
|
||||||
|
go manageClient(sock)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Printf("Elapsed: %d ns\n", time.Now().UnixNano()-startTime)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func manageClient(sock net.Conn) {
|
||||||
|
|
||||||
|
defer sock.Close()
|
||||||
|
|
||||||
|
for {
|
||||||
|
|
||||||
|
reader := bufio.NewReader(sock)
|
||||||
|
|
||||||
|
_, err := request.Parse(reader)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
sock.Write([]byte("coucou"))
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,78 @@
|
||||||
|
package request
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
func (r *T) parseHeader(b []byte) {
|
||||||
|
|
||||||
|
/* (1) First line -> GET {uri} HTTP/{version}
|
||||||
|
---------------------------------------------------------*/
|
||||||
|
if !r.first {
|
||||||
|
|
||||||
|
/* (1) Split by ' ' */
|
||||||
|
parts := strings.Split(string(b), " ")
|
||||||
|
|
||||||
|
/* (2) Abort on failure */
|
||||||
|
if len(parts) != 3 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (3) Store method */
|
||||||
|
r.httpMethod = parts[0]
|
||||||
|
|
||||||
|
/* (4) Store URI */
|
||||||
|
r.uri = parts[1]
|
||||||
|
|
||||||
|
/* (5) Fail if wrong version format */
|
||||||
|
if len(parts[2]) < len("HTTP/0") {
|
||||||
|
r.Err = fmt.Errorf("Invalid HTTP version [%s] expected [HTTP/] prefix")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (6) Extract version part */
|
||||||
|
versionString := parts[2][len("HTTP/"):]
|
||||||
|
version, err := strconv.ParseFloat(versionString, 32)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
r.Err = fmt.Errorf("Cannot parse HTTP version as float")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (7) Store HTTP version */
|
||||||
|
r.httpVersion = float32(version)
|
||||||
|
|
||||||
|
|
||||||
|
r.first = true
|
||||||
|
return
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* (2) Other lines -> Header-Name: Header-Value
|
||||||
|
---------------------------------------------------------*/
|
||||||
|
|
||||||
|
/* (1) Split by ': ' */
|
||||||
|
parts := strings.Split(string(b), ": ")
|
||||||
|
|
||||||
|
/* (2) Abort on failure */
|
||||||
|
if len(parts) != 2 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (3) Match header */
|
||||||
|
switch strings.ToLower(parts[0]) {
|
||||||
|
case "host": fmt.Printf("[host] '%s'\n", parts[1])
|
||||||
|
case "upgrade": fmt.Printf("[upgrade] '%s'\n", parts[1])
|
||||||
|
case "connection": fmt.Printf("[connection] '%s'\n", parts[1])
|
||||||
|
case "sec-websocket-key": fmt.Printf("[sec-websocket-key] '%s'\n", parts[1])
|
||||||
|
case "origin": fmt.Printf("[origin] '%s'\n", parts[1])
|
||||||
|
case "sec-websocket-protocol": fmt.Printf("[sec-websocket-protocol] '%s'\n", parts[1])
|
||||||
|
case "sec-websocket-version": fmt.Printf("[sec-websocket-version] '%s'\n", parts[1])
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,56 @@
|
||||||
|
package request
|
||||||
|
|
||||||
|
import (
|
||||||
|
"git.xdrm.io/gws/internal/http/reader"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Parse parses a byte array build a request object
|
||||||
|
func Parse(r io.Reader) (request *T, err error) {
|
||||||
|
|
||||||
|
/* (1) Get chunk reader */
|
||||||
|
cr, err := reader.NewReader(r)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Error while creating chunk reader: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (2) Init request */
|
||||||
|
req := new(T)
|
||||||
|
|
||||||
|
/* (3) Parse header line by line */
|
||||||
|
for {
|
||||||
|
|
||||||
|
line, err := cr.Read()
|
||||||
|
if err == io.EOF {
|
||||||
|
fmt.Printf("END OF READING\n");
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Cannot read from reader: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
req.parseHeader(line)
|
||||||
|
|
||||||
|
if req.Err != nil {
|
||||||
|
return nil, fmt.Errorf("Parsing error: %s\n", req.Err);
|
||||||
|
}
|
||||||
|
|
||||||
|
// fmt.Printf("Read line: %s\n", line)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* (2) GET {uri} HTTP/{version}
|
||||||
|
---------------------------------------------------------*/
|
||||||
|
/* (1) Break apart */
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* (1) Check method */
|
||||||
|
return nil, nil
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
package request
|
||||||
|
|
||||||
|
// T represents an HTTP Upgrade request
|
||||||
|
type T struct {
|
||||||
|
httpMethod string
|
||||||
|
uri string
|
||||||
|
httpVersion float32
|
||||||
|
host string
|
||||||
|
origin string
|
||||||
|
key []byte
|
||||||
|
protocols []string
|
||||||
|
version int
|
||||||
|
|
||||||
|
first bool // whether the first line has been read (GET uri HTTP/version)
|
||||||
|
|
||||||
|
Err error
|
||||||
|
}
|
|
@ -0,0 +1,83 @@
|
||||||
|
package reader
|
||||||
|
|
||||||
|
// DISCLAIMER
|
||||||
|
// ----------
|
||||||
|
// Some of the content of this file is inspired or copied from
|
||||||
|
// the golang standard library
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"bufio"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Maximum line length
|
||||||
|
var maxLineLength = 4096
|
||||||
|
|
||||||
|
// Chunk reader
|
||||||
|
type chunkReader struct {
|
||||||
|
reader *bufio.Reader // the reader
|
||||||
|
isEnded bool // If we are done (2 consecutive CRLF)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// New creates a new reader
|
||||||
|
func NewReader(r io.Reader) (reader *chunkReader, err error) {
|
||||||
|
|
||||||
|
br, ok := r.(*bufio.Reader)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("Invalid reader (must implement bufio.Reader")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &chunkReader{reader: br}, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Read reads a chunk, err is io.EOF when done
|
||||||
|
func (r *chunkReader) Read() ([]byte, error){
|
||||||
|
|
||||||
|
/* (1) If already ended */
|
||||||
|
if r.isEnded {
|
||||||
|
return nil, io.EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (2) Read line */
|
||||||
|
var line []byte
|
||||||
|
line, err := r.reader.ReadSlice('\n')
|
||||||
|
|
||||||
|
/* (3) manage errors */
|
||||||
|
if err == io.EOF {
|
||||||
|
err = io.ErrUnexpectedEOF
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (4) Trim */
|
||||||
|
line = removeTrailingSpace(line)
|
||||||
|
|
||||||
|
/* (5) Manage ending line */
|
||||||
|
if len(line) == 0 {
|
||||||
|
r.isEnded = true
|
||||||
|
return line, io.EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
return line, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
func removeTrailingSpace(b []byte) []byte{
|
||||||
|
for len(b) > 0 && isASCIISpace(b[len(b)-1]) {
|
||||||
|
b = b[:len(b)-1]
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
func isASCIISpace(b byte) bool {
|
||||||
|
return b == ' ' || b == '\t' || b == '\r' || b =='\n'
|
||||||
|
}
|
Loading…
Reference in New Issue