diff --git a/structural/proxy/main.go b/structural/proxy/main.go index f9242fe..e8a50f8 100644 --- a/structural/proxy/main.go +++ b/structural/proxy/main.go @@ -1,35 +1,38 @@ package main -import "fmt" +import ( + "fmt" +) // For example: // we must a execute some command // so before that we must to create new terminal session // and provide our user name and command func main() { - var excResp string // response of executed command - var excErr error // error of executing command - // Create new instance of Proxy terminal t, err := NewTerminal("gopher") if err != nil { + // panic: User cant be empty + // Or + // panic: You (badUser) are not allowed to use terminal and execute commands panic(err.Error()) } // Execute user command - if excResp, excErr = t.Execute("gopher", "say_hi"); excErr != nil { - fmt.Printf("%s\n", excErr.Error()) // Prints: I know only how to say hello (command: say_hi) + excResp, excErr := t.Execute("say_hi") // Proxy prints to STDOUT -> PROXY: Intercepted execution of user (gopher), asked command (say_hi) + if excErr != nil { + fmt.Printf("ERROR: %s\n", excErr.Error()) // Prints: ERROR: I know only how to execute commands: say_hi, man } // Show execution response - fmt.Println(excResp) // Prints: Hi gopher + fmt.Println(excResp) // Prints: gopher@go_term$: Hi gopher } /* From that it's can be different terminals realizations with different methods, propertys, yda yda... */ -// First we must declare interface with public method its will be implemented in GopherTerminal and Terminal(Proxy) +// ITerminal is interface, it's a public method whose implemented in Terminal(Proxy) and Gopher Terminal type ITerminal interface { Execute(cmd string) (resp string, err error) } @@ -38,19 +41,24 @@ type ITerminal interface { // Its a "huge" structure with different public methods type GopherTerminal struct { // user is a current authorized user - user string + User string } // Execute just runs known commands for current authorized user func (gt *GopherTerminal) Execute(cmd string) (resp string, err error) { + // Set "terminal" prefix for output + prefix := fmt.Sprintf("%s@go_term$:", gt.User) - if cmd != "say_hi" { - err = fmt.Errorf("Unknown command") - return + // Execute some asked commands if we know them + switch cmd { + case "say_hi": + resp = fmt.Sprintf("%s Hi %s", prefix, gt.User) + case "man": + resp = fmt.Sprintf("%s Visit 'https://golang.org/doc/' for Golang documentation", prefix) + default: + err = fmt.Errorf("%s Unknown command", prefix) } - resp = fmt.Sprintf("Hi %s", gt.user) - return } @@ -61,40 +69,44 @@ func (gt *GopherTerminal) Execute(cmd string) (resp string, err error) { // Terminal is a implementation of Proxy, it's validates and sends data to GopherTerminal // As example before send commands, user must be authorized type Terminal struct { - currentUser string + currentUser string gopherTerminal *GopherTerminal } // NewTerminal creates new instance of terminal -func NewTerminal(user string) (t *Terminal, err error) { +func NewTerminal(user string) (t *Terminal, err error) { // Check user if given correctly if user == "" { err = fmt.Errorf("User cant be empty") return } + // Before we execute user commands, we validate current user, if he have rights to do it + if authErr := authorizeUser(user); authErr != nil { + err = fmt.Errorf("You (%s) are not allowed to use terminal and execute commands", user) + return + } + + // Create new instance of terminal and set valid user t = &Terminal{currentUser: user} return } -// ExecuteCommand intercepts execution of command, implements authorizing user, validates it and +// Execute intercepts execution of command, implements authorizing user, validates it and // poxing command to real terminal (gopherTerminal) method -func (t *Terminal) Execute(user, command string) (resp string, err error) { - // Before we execute user commands, we validate current user, if he have rights to do it - if authErr := t.authorizeUser(); authErr != nil { - err = fmt.Errorf("You are not allowed to execute commands") - return - } - +func (t *Terminal) Execute(command string) (resp string, err error) { // If user allowed to execute send commands then, for example we can decide which terminal can be used, remote or local etc.. // but for example we just creating new instance of terminal, // set current user and send user command to execution in terminal - t.gopherTerminal = new(GopherTerminal) - t.gopherTerminal.user = user + t.gopherTerminal = &GopherTerminal{User: t.currentUser} + // For example our proxy can log or output intercepted execution... etc + fmt.Printf("PROXY: Intercepted execution of user (%s), asked command (%s)\n", t.currentUser, command) + + // Transfer data to original object and execute command if resp, err = t.gopherTerminal.Execute(command); err != nil { - err = fmt.Errorf("I know only how to say hello (command: say_hi)") + err = fmt.Errorf("I know only how to execute commands: say_hi, man") return } @@ -102,12 +114,12 @@ func (t *Terminal) Execute(user, command string) (resp string, err error) { } // authorize validates user right to execute commands -func (t *Terminal) authorizeUser() (err error) { +func authorizeUser(user string) (err error) { // As we use terminal like proxy, then // we will intercept user name to validate if it's allowed to execute commands - if t.currentUser != "gopher" { + if user != "gopher" { // Do some logs, notifications etc... - err = fmt.Errorf("User %s in black list", t.currentUser) + err = fmt.Errorf("User %s in black list", user) return }