log_writer.go 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. package internal
  2. import (
  3. "log"
  4. "os"
  5. "time"
  6. "github.com/v2ray/v2ray-core/common/platform"
  7. "github.com/v2ray/v2ray-core/common/signal"
  8. )
  9. type LogWriter interface {
  10. Log(LogEntry)
  11. Close()
  12. }
  13. type NoOpLogWriter struct {
  14. }
  15. func (this *NoOpLogWriter) Log(entry LogEntry) {
  16. entry.Release()
  17. }
  18. func (this *NoOpLogWriter) Close() {
  19. }
  20. type StdOutLogWriter struct {
  21. logger *log.Logger
  22. cancel *signal.CancelSignal
  23. }
  24. func NewStdOutLogWriter() LogWriter {
  25. return &StdOutLogWriter{
  26. logger: log.New(os.Stdout, "", log.Ldate|log.Ltime),
  27. cancel: signal.NewCloseSignal(),
  28. }
  29. }
  30. func (this *StdOutLogWriter) Log(log LogEntry) {
  31. this.logger.Print(log.String() + platform.LineSeparator())
  32. log.Release()
  33. }
  34. func (this *StdOutLogWriter) Close() {
  35. time.Sleep(500 * time.Millisecond)
  36. }
  37. type FileLogWriter struct {
  38. queue chan string
  39. logger *log.Logger
  40. file *os.File
  41. cancel *signal.CancelSignal
  42. }
  43. func (this *FileLogWriter) Log(log LogEntry) {
  44. select {
  45. case this.queue <- log.String():
  46. default:
  47. // We don't expect this to happen, but don't want to block main thread as well.
  48. }
  49. log.Release()
  50. }
  51. func (this *FileLogWriter) run() {
  52. for {
  53. entry, open := <-this.queue
  54. if !open {
  55. break
  56. }
  57. this.logger.Print(entry + platform.LineSeparator())
  58. }
  59. this.cancel.Done()
  60. }
  61. func (this *FileLogWriter) Close() {
  62. close(this.queue)
  63. <-this.cancel.WaitForDone()
  64. this.file.Close()
  65. }
  66. func NewFileLogWriter(path string) (*FileLogWriter, error) {
  67. file, err := os.OpenFile(path, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
  68. if err != nil {
  69. return nil, err
  70. }
  71. logger := &FileLogWriter{
  72. queue: make(chan string, 16),
  73. logger: log.New(file, "", log.Ldate|log.Ltime),
  74. file: file,
  75. cancel: signal.NewCloseSignal(),
  76. }
  77. go logger.run()
  78. return logger, nil
  79. }