1
0
Fork 0
mirror of https://github.com/tmrts/go-patterns.git synced 2025-04-03 13:13:34 +03:00

Corrected mistakes, splited user validation in proxy

This commit is contained in:
Artjom Nemiro 2017-06-21 16:41:29 +03:00
parent 2df165a12f
commit 118698f95a

View file

@ -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
}