From cb46154ec89fce8ed3b31e6ff800ad687a758e46 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Denis=20Nu=C8=9Biu?= Date: Wed, 8 Nov 2017 15:11:37 +0200 Subject: [PATCH] Refactoring: Instead of using net.Conn use a Client interface instead --- server/main.go | 10 ++++-- server/server/connection.go | 61 +++++++++++++++++++++++++++++++------ server/server/parser.go | 13 ++++---- 3 files changed, 66 insertions(+), 18 deletions(-) diff --git a/server/main.go b/server/main.go index 4a5ca11..9c72920 100644 --- a/server/main.go +++ b/server/main.go @@ -1,9 +1,10 @@ package main import ( - "github.com/metonimie/simpleFTP/server/server" "log" "net" + + "github.com/metonimie/simpleFTP/server/server" ) func main() { @@ -16,12 +17,17 @@ func main() { log.Println("Running on:", "localhost", "port", "8080") for { + conn, err := listener.Accept() if err != nil { log.Print(err) continue } - go server.HandleConnection(conn) + client := server.FTPClient{} + client.SetStack(new(server.StringStack)) + client.SetConnection(conn) + + go server.HandleConnection(&client) } } diff --git a/server/server/connection.go b/server/server/connection.go index 58d637b..0e50948 100644 --- a/server/server/connection.go +++ b/server/server/connection.go @@ -7,24 +7,67 @@ import ( "net" ) -func HandleConnection(c net.Conn) { - defer c.Close() - io.WriteString(c, "Hello and welcome to simple ftp\n") +// ClientStack interface holds the function needed to work with a stack. +type ClientStack interface { + Stack() Stack // Returns the underlying Stack. +} - log.Println(c.RemoteAddr(), "has connected.") +// Client interface provides the blueprints for the Client that is used by the server. +type Client interface { + Connection() net.Conn // Connection returns the connection stream. + SetConnection(conn net.Conn) // SetConnection sets the connection for the client. + Disconnect() // Disconnect closes the Client's connections and clears up resources. +} + +// FTPClient represents a FTPClient connection, it holds a root cage and the underlying connection. +type FTPClient struct { + rootCage *StringStack // rootCage is a StringStack that is used to represent the current directory the client is in. + connection net.Conn +} + +// Stack returns the root cage stack. +func (c *FTPClient) Stack() Stack { + return c.rootCage +} + +// SetStack sets the stack for the FTPClient. +func (c *FTPClient) SetStack(stack *StringStack) { + c.rootCage = stack +} + +// Connection returns the Connection of the client. +func (c *FTPClient) Connection() net.Conn { + return c.connection +} + +// SetConnection sets the given connection to the client. +func (c *FTPClient) SetConnection(conn net.Conn) { + c.connection = conn +} + +// Disconnects the client. +func (c *FTPClient) Disconnect() { + c.connection.Close() +} + +func HandleConnection(client Client) { + defer client.Disconnect() + io.WriteString(client.Connection(), "Hello and welcome to simple ftp\n") + + log.Println(client.Connection().RemoteAddr(), "has connected.") // Process input - input := bufio.NewScanner(c) + input := bufio.NewScanner(client.Connection()) for input.Scan() { - log.Println(c.RemoteAddr(), ":", input.Text()) + log.Println(client.Connection().RemoteAddr(), ":", input.Text()) - err := ProcessInput(c, input.Text()) + err := ProcessInput(client, input.Text()) if err != nil { log.Println(err) - io.WriteString(c, err.Error()+"\n") + io.WriteString(client.Connection(), err.Error()+"\n") } } // Client has left. - log.Println(c.RemoteAddr(), "has disconnected.") + log.Println(client.Connection().RemoteAddr(), "has disconnected.") } diff --git a/server/server/parser.go b/server/server/parser.go index be06729..6cee415 100644 --- a/server/server/parser.go +++ b/server/server/parser.go @@ -1,7 +1,6 @@ package server import ( - "net" "strings" ) @@ -15,7 +14,7 @@ func checkArgumentsLength(length int, expected int) error { return nil } -func ProcessInput(c net.Conn, text string) error { +func ProcessInput(c Client, text string) error { commands := strings.Fields(text) commandsLen := len(commands) @@ -33,7 +32,7 @@ func ProcessInput(c net.Conn, text string) error { } // Get the file - _, err = GetFile(c, commands[1]) + _, err = GetFile(c.Connection(), commands[1]) if err != nil { return &InputError{thisCommand, err} } @@ -43,7 +42,7 @@ func ProcessInput(c net.Conn, text string) error { return &InputError{thisCommand, err} } - err = ListFiles(c) + err = ListFiles(c.Connection()) if err != nil { return &InputError{thisCommand, err} } @@ -57,7 +56,7 @@ func ProcessInput(c net.Conn, text string) error { // clear | hexdump -C var b = []byte{0x1b, 0x5b, 0x48, 0x1b, 0x5b, 0x4a} - c.Write(b) + c.Connection().Write(b) case "help": // Check arguments err := checkArgumentsLength(commandsLen, 1) @@ -65,7 +64,7 @@ func ProcessInput(c net.Conn, text string) error { return &InputError{thisCommand, err} } - err = ShowHelp(c) + err = ShowHelp(c.Connection()) if err != nil { return &InputError{thisCommand, err} } @@ -75,7 +74,7 @@ func ProcessInput(c net.Conn, text string) error { return &InputError{thisCommand, err} } - c.Close() + c.Connection().Close() default: return &InputError{thisCommand, InputInvalidCommand} }