log.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. package log
  2. //go:generate go run github.com/v2fly/v2ray-core/v4/common/errors/errorgen
  3. import (
  4. "context"
  5. "sync"
  6. "github.com/v2fly/v2ray-core/v4/common"
  7. "github.com/v2fly/v2ray-core/v4/common/log"
  8. )
  9. // Instance is a log.Handler that handles logs.
  10. type Instance struct {
  11. sync.RWMutex
  12. config *Config
  13. accessLogger log.Handler
  14. errorLogger log.Handler
  15. active bool
  16. }
  17. // New creates a new log.Instance based on the given config.
  18. func New(ctx context.Context, config *Config) (*Instance, error) {
  19. g := &Instance{
  20. config: config,
  21. active: false,
  22. }
  23. log.RegisterHandler(g)
  24. // start logger instantly on inited
  25. // other modules would log during init
  26. if err := g.startInternal(); err != nil {
  27. return nil, err
  28. }
  29. newError("Logger started").AtDebug().WriteToLog()
  30. return g, nil
  31. }
  32. func (g *Instance) initAccessLogger() error {
  33. handler, err := createHandler(g.config.AccessLogType, HandlerCreatorOptions{
  34. Path: g.config.AccessLogPath,
  35. })
  36. if err != nil {
  37. return err
  38. }
  39. g.accessLogger = handler
  40. return nil
  41. }
  42. func (g *Instance) initErrorLogger() error {
  43. handler, err := createHandler(g.config.ErrorLogType, HandlerCreatorOptions{
  44. Path: g.config.ErrorLogPath,
  45. })
  46. if err != nil {
  47. return err
  48. }
  49. g.errorLogger = handler
  50. return nil
  51. }
  52. // Type implements common.HasType.
  53. func (*Instance) Type() interface{} {
  54. return (*Instance)(nil)
  55. }
  56. func (g *Instance) startInternal() error {
  57. g.Lock()
  58. defer g.Unlock()
  59. if g.active {
  60. return nil
  61. }
  62. g.active = true
  63. if err := g.initAccessLogger(); err != nil {
  64. return newError("failed to initialize access logger").Base(err).AtWarning()
  65. }
  66. if err := g.initErrorLogger(); err != nil {
  67. return newError("failed to initialize error logger").Base(err).AtWarning()
  68. }
  69. return nil
  70. }
  71. // Start implements common.Runnable.Start().
  72. func (g *Instance) Start() error {
  73. return g.startInternal()
  74. }
  75. // Handle implements log.Handler.
  76. func (g *Instance) Handle(msg log.Message) {
  77. g.RLock()
  78. defer g.RUnlock()
  79. if !g.active {
  80. return
  81. }
  82. switch msg := msg.(type) {
  83. case *log.AccessMessage:
  84. if g.accessLogger != nil {
  85. g.accessLogger.Handle(msg)
  86. }
  87. case *log.GeneralMessage:
  88. if g.errorLogger != nil && msg.Severity <= g.config.ErrorLogLevel {
  89. g.errorLogger.Handle(msg)
  90. }
  91. default:
  92. // Swallow
  93. }
  94. }
  95. // Close implements common.Closable.Close().
  96. func (g *Instance) Close() error {
  97. newError("Logger closing").AtDebug().WriteToLog()
  98. g.Lock()
  99. defer g.Unlock()
  100. if !g.active {
  101. return nil
  102. }
  103. g.active = false
  104. common.Close(g.accessLogger)
  105. g.accessLogger = nil
  106. common.Close(g.errorLogger)
  107. g.errorLogger = nil
  108. return nil
  109. }
  110. func init() {
  111. common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, config interface{}) (interface{}, error) {
  112. return New(ctx, config.(*Config))
  113. }))
  114. }