restructure

This commit is contained in:
Niccolo Borgioli 2024-05-25 11:51:11 +02:00
parent 8de565cb41
commit 55ae922fa0
No known key found for this signature in database
GPG Key ID: 4897ACD13A65977C
2 changed files with 210 additions and 123 deletions

97
app/http.go Normal file
View File

@ -0,0 +1,97 @@
package main
import (
"fmt"
"net"
"regexp"
"strconv"
)
const (
HTTPDelimiter = "\r\n"
)
type Header struct {
Name string
Value string
}
type Request struct {
Method string
Path string
Version string
Body string
Headers []Header
}
type HttpCode struct {
Code uint
Message string
}
var (
BadRequest = HttpCode{Code: 400, Message: "Bad Response"}
NotFound = HttpCode{Code: 404, Message: "Not Found"}
OK = HttpCode{Code: 200, Message: "OK"}
)
type Response struct {
Code HttpCode
Version string
Body string
BodyRaw []byte
Headers []Header
}
type StringRoute struct {
path string
method string
handler func(req Request) Response
}
type RegexRoute struct {
regex *regexp.Regexp
method string
handler func(req Request, matches []string) Response
}
type Routes struct {
stringRoutes []StringRoute
regexpRoutes []RegexRoute
}
// func createHandler(routes Routes) func(conn net.Conn) {
// return func(conn net.Conn) {
// for _, route := range(routes.stringRoutes){
// }
// }
// }
func Respond(conn net.Conn, response Response) {
fmt.Fprintf(conn, "%s %d %s%s", response.Version, response.Code.Code, response.Code.Message, HTTPDelimiter)
bodySize := 0
if response.Body != "" {
bodySize = len(response.Body)
} else {
bodySize = len(response.BodyRaw)
}
if bodySize > 0 {
response.Headers = append(response.Headers, Header{Name: "Content-Length", Value: strconv.Itoa(bodySize)})
}
for _, header := range response.Headers {
fmt.Fprintf(conn, "%s: %s%s", header.Name, header.Value, HTTPDelimiter)
}
fmt.Fprint(conn, HTTPDelimiter)
if bodySize > 0 {
if response.Body != "" {
fmt.Fprint(conn, response.Body)
} else {
conn.Write(response.BodyRaw)
}
}
// fmt.Fprintf(conn, "HTTP/1.1 %d %s%s%s", response.Code, response.Message, HTTPDelimiter, HTTPDelimiter)
}

View File

@ -3,79 +3,25 @@ package main
import ( import (
"fmt" "fmt"
"log" "log"
"path/filepath"
"regexp"
"strconv"
"strings"
// Uncomment this block to pass the first stage
"net" "net"
"os" "os"
"path/filepath"
"regexp"
"strings"
) )
const ( func getFilepath(filename string) string {
HTTPDelimiter = "\r\n" if len(os.Args) != 3 {
) log.Fatal("Not enough args")
type Header struct {
Name string
Value string
}
type Request struct {
Method string
Path string
Version string
Body string
Headers []Header
}
type HttpCode struct {
Code uint
Message string
}
var (
BadRequest = HttpCode{Code: 400, Message: "Bad Response"}
NotFound = HttpCode{Code: 404, Message: "Not Found"}
OK = HttpCode{Code: 200, Message: "OK"}
)
type Response struct {
Code HttpCode
Version string
Body string
BodyRaw []byte
Headers []Header
}
func Respond(conn net.Conn, response Response) {
fmt.Fprintf(conn, "%s %d %s%s", response.Version, response.Code.Code, response.Code.Message, HTTPDelimiter)
bodySize := 0
if response.Body != "" {
bodySize = len(response.Body)
} else {
bodySize = len(response.BodyRaw)
} }
if bodySize > 0 { dir, err := filepath.Abs(os.Args[2])
response.Headers = append(response.Headers, Header{Name: "Content-Length", Value: strconv.Itoa(bodySize)}) if err != nil {
log.Fatal(err)
} }
for _, header := range response.Headers { return filepath.Join(dir, filename)
fmt.Fprintf(conn, "%s: %s%s", header.Name, header.Value, HTTPDelimiter)
}
fmt.Fprint(conn, HTTPDelimiter)
if bodySize > 0 {
if response.Body != "" {
fmt.Fprint(conn, response.Body)
} else {
conn.Write(response.BodyRaw)
}
}
// fmt.Fprintf(conn, "HTTP/1.1 %d %s%s%s", response.Code, response.Message, HTTPDelimiter, HTTPDelimiter)
} }
func handleConnection(conn net.Conn) { func handleConnection(conn net.Conn, routes Routes) {
defer conn.Close() defer conn.Close()
buffer := make([]byte, 1024) buffer := make([]byte, 1024)
n, err := conn.Read(buffer) n, err := conn.Read(buffer)
@ -116,92 +62,136 @@ func handleConnection(conn net.Conn) {
} }
fmt.Println(request) fmt.Println(request)
if request.Path == "/" { for _, route := range routes.stringRoutes {
Respond(conn, Response{Version: request.Version, Code: OK}) if request.Path == route.path && request.Method == route.method {
return Respond(conn, route.handler(request))
}
if request.Path == "/user-agent" {
for _, header := range request.Headers {
if header.Name != "User-Agent" {
continue
}
Respond(conn, Response{
Version: request.Version,
Code: OK,
Body: header.Value,
Headers: []Header{{Name: "Content-Type", Value: "text/plain"}},
})
return return
} }
Respond(conn, Response{Version: request.Version, Code: BadRequest})
} }
reFiles := regexp.MustCompile(`^/files/([A-Za-z0-9_\-.]+)$`) for _, route := range routes.regexpRoutes {
if matches := reFiles.FindStringSubmatch(request.Path); len(matches) > 0 { if request.Method != route.method {
continue
if len(os.Args) != 3 {
log.Fatal("Not enough args")
} }
dir, err := filepath.Abs(os.Args[2]) if matches := route.regex.FindStringSubmatch(request.Path); len(matches) > 0 {
if err != nil { Respond(conn, route.handler(request, matches))
log.Fatal(err)
}
filename := filepath.Join(dir, matches[1])
// _, err:=os.Stat(filename)
// fmt.Println(file)
file, err := os.ReadFile(filename)
if os.IsNotExist(err) {
Respond(conn, Response{Version: request.Version, Code: NotFound})
return return
} }
if err != nil {
log.Fatal(err)
}
Respond(conn, Response{
Version: request.Version,
Code: OK,
BodyRaw: file,
Headers: []Header{{Name: "Content-Type", Value: "application/octet-stream"}},
})
return
} }
reEcho := regexp.MustCompile(`^/echo/([A-Za-z]+)$`) // if request.Path == "/" {
if matches := reEcho.FindStringSubmatch(request.Path); len(matches) > 0 { // Respond(conn, Response{Version: request.Version, Code: OK})
Respond(conn, Response{ // return
Version: request.Version, // }
Code: OK,
Body: matches[1], // if request.Path == "/user-agent" {
Headers: []Header{{Name: "Content-Type", Value: "text/plain"}}, // for _, header := range request.Headers {
}) // if header.Name != "User-Agent" {
return // continue
} // }
// Respond(conn, Response{
// Version: request.Version,
// Code: OK,
// Body: header.Value,
// Headers: []Header{{Name: "Content-Type", Value: "text/plain"}},
// })
// return
// }
// Respond(conn, Response{Version: request.Version, Code: BadRequest})
// }
// reFiles := regexp.MustCompile(`^/files/([A-Za-z0-9_\-.]+)$`)
// if matches := reFiles.FindStringSubmatch(request.Path); len(matches) > 0 {
// filename := getFilepath(matches[1])
// file, err := os.ReadFile(filename)
// if os.IsNotExist(err) {
// Respond(conn, Response{Version: request.Version, Code: NotFound})
// return
// }
// if err != nil {
// log.Fatal(err)
// }
// Respond(conn, Response{
// Version: request.Version,
// Code: OK,
// BodyRaw: file,
// Headers: []Header{{Name: "Content-Type", Value: "application/octet-stream"}},
// })
// return
// }
// reEcho := regexp.MustCompile(`^/echo/([A-Za-z]+)$`)
// if matches := reEcho.FindStringSubmatch(request.Path); len(matches) > 0 {
// Respond(conn, Response{
// Version: request.Version,
// Code: OK,
// Body: matches[1],
// Headers: []Header{{Name: "Content-Type", Value: "text/plain"}},
// })
// return
// }
Respond(conn, Response{Version: request.Version, Code: NotFound}) Respond(conn, Response{Version: request.Version, Code: NotFound})
} }
func main() { func main() {
// You can use print statements as follows for debugging, they'll be visible when running tests.
fmt.Println("Logs from your program will appear here!") fmt.Println("Logs from your program will appear here!")
// Uncomment this block to pass the first stage
//
l, err := net.Listen("tcp", "0.0.0.0:4221") l, err := net.Listen("tcp", "0.0.0.0:4221")
if err != nil { if err != nil {
fmt.Println("Failed to bind to port 4221") fmt.Println("Failed to bind to port 4221")
os.Exit(1) os.Exit(1)
} }
routes := Routes{
stringRoutes: []StringRoute{
// ROOT
{path: "/", method: "GET", handler: func(req Request) Response {
return Response{Version: req.Version, Code: OK}
}},
// USER AGENT
{path: "/user-agent", method: "GET", handler: func(req Request) Response {
for _, header := range req.Headers {
if header.Name != "User-Agent" {
continue
}
return Response{
Version: req.Version,
Code: OK,
Body: header.Value,
Headers: []Header{{Name: "Content-Type", Value: "text/plain"}},
}
}
return Response{Version: req.Version, Code: BadRequest}
}},
},
regexpRoutes: []RegexRoute{
// PATH PARAMETER
{
regex: regexp.MustCompile(`^/echo/([A-Za-z]+)$`),
method: "GET",
handler: func(req Request, matches []string) Response {
return Response{
Version: req.Version,
Code: OK,
Body: matches[1],
Headers: []Header{{Name: "Content-Type", Value: "text/plain"}},
}
}},
},
}
for { for {
conn, err := l.Accept() conn, err := l.Accept()
if err != nil { if err != nil {
fmt.Println("Error accepting connection: ", err.Error()) fmt.Println("Error accepting connection: ", err.Error())
os.Exit(1) os.Exit(1)
} }
go handleConnection(conn) go handleConnection(conn, routes)
} }
} }