access.go 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. package log
  2. import (
  3. "log"
  4. "os"
  5. )
  6. // AccessStatus is the status of an access request from clients.
  7. type AccessStatus string
  8. const (
  9. AccessAccepted = AccessStatus("accepted")
  10. AccessRejected = AccessStatus("rejected")
  11. )
  12. type accessLogger interface {
  13. Log(from, to string, status AccessStatus, reason string)
  14. }
  15. type noOpAccessLogger struct {
  16. }
  17. func (logger *noOpAccessLogger) Log(from, to string, status AccessStatus, reason string) {
  18. // Swallow
  19. }
  20. type accessLog struct {
  21. From string
  22. To string
  23. Status AccessStatus
  24. Reason string
  25. }
  26. type fileAccessLogger struct {
  27. queue chan *accessLog
  28. logger *log.Logger
  29. file *os.File
  30. }
  31. func (logger *fileAccessLogger) close() {
  32. logger.file.Close()
  33. }
  34. func (logger *fileAccessLogger) Log(from, to string, status AccessStatus, reason string) {
  35. logger.queue <- &accessLog{
  36. From: from,
  37. To: to,
  38. Status: status,
  39. Reason: reason,
  40. }
  41. }
  42. func (logger *fileAccessLogger) Run() {
  43. for entry := range logger.queue {
  44. logger.logger.Println(entry.From + " " + string(entry.Status) + " " + entry.To + " " + entry.Reason)
  45. }
  46. }
  47. func newFileAccessLogger(path string) accessLogger {
  48. file, err := os.OpenFile(path, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
  49. if err != nil {
  50. log.Printf("Unable to create or open file (%s): %v\n", path, err)
  51. return nil
  52. }
  53. return &fileAccessLogger{
  54. queue: make(chan *accessLog, 16),
  55. logger: log.New(file, "", log.Ldate|log.Ltime),
  56. file: file,
  57. }
  58. }
  59. var accessLoggerInstance accessLogger = &noOpAccessLogger{}
  60. // InitAccessLogger initializes the access logger to write into the give file.
  61. func InitAccessLogger(file string) {
  62. logger := newFileAccessLogger(file)
  63. if logger != nil {
  64. go logger.(*fileAccessLogger).Run()
  65. accessLoggerInstance = logger
  66. }
  67. }
  68. // Access writes an access log.
  69. func Access(from, to string, status AccessStatus, reason string) {
  70. accessLoggerInstance.Log(from, to, status, reason)
  71. }