2022-12-03 15:08:43 +00:00
|
|
|
package logging
|
2022-11-19 18:19:25 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"sync"
|
|
|
|
"time"
|
2022-12-03 15:08:43 +00:00
|
|
|
|
|
|
|
"github.com/chubin/wttr.in/internal/util"
|
2022-11-19 18:19:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Logging request.
|
|
|
|
//
|
|
|
|
|
|
|
|
// RequestLogger logs all incoming HTTP requests.
|
|
|
|
type RequestLogger struct {
|
|
|
|
buf map[logEntry]int
|
|
|
|
filename string
|
|
|
|
m sync.Mutex
|
|
|
|
|
|
|
|
period time.Duration
|
|
|
|
lastFlush time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
type logEntry struct {
|
|
|
|
Proto string
|
|
|
|
IP string
|
|
|
|
URI string
|
|
|
|
UserAgent string
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewRequestLogger returns a new RequestLogger for the specified log file.
|
|
|
|
// Flush logging entries after period of time.
|
2022-12-04 09:35:58 +00:00
|
|
|
//
|
|
|
|
// If filename is empty, no log will be written, and all logging entries
|
|
|
|
// will be silently dropped.
|
2022-11-19 18:19:25 +00:00
|
|
|
func NewRequestLogger(filename string, period time.Duration) *RequestLogger {
|
|
|
|
return &RequestLogger{
|
|
|
|
buf: map[logEntry]int{},
|
|
|
|
filename: filename,
|
|
|
|
m: sync.Mutex{},
|
|
|
|
period: period,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log logs information about a HTTP request.
|
|
|
|
func (rl *RequestLogger) Log(r *http.Request) error {
|
|
|
|
le := logEntry{
|
|
|
|
Proto: "http",
|
2022-12-03 15:08:43 +00:00
|
|
|
IP: util.ReadUserIP(r),
|
2022-11-19 18:19:25 +00:00
|
|
|
URI: r.RequestURI,
|
|
|
|
UserAgent: r.Header.Get("User-Agent"),
|
|
|
|
}
|
2022-11-20 09:03:31 +00:00
|
|
|
if r.TLS != nil {
|
|
|
|
le.Proto = "https"
|
|
|
|
}
|
2022-11-19 18:19:25 +00:00
|
|
|
|
|
|
|
rl.m.Lock()
|
|
|
|
rl.buf[le]++
|
|
|
|
rl.m.Unlock()
|
|
|
|
|
|
|
|
if time.Since(rl.lastFlush) > rl.period {
|
|
|
|
return rl.flush()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// flush stores log data to disk, and flushes the buffer.
|
|
|
|
func (rl *RequestLogger) flush() error {
|
|
|
|
rl.m.Lock()
|
|
|
|
defer rl.m.Unlock()
|
|
|
|
|
|
|
|
// It is possible, that while waiting the mutex,
|
|
|
|
// the buffer was already flushed.
|
|
|
|
if time.Since(rl.lastFlush) <= rl.period {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-12-04 09:35:58 +00:00
|
|
|
if rl.filename != "" {
|
|
|
|
// Generate log output.
|
|
|
|
output := ""
|
|
|
|
for k, hitsNumber := range rl.buf {
|
|
|
|
output += fmt.Sprintf("%s %3d %s\n", time.Now().Format(time.RFC3339), hitsNumber, k.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open log file.
|
|
|
|
f, err := os.OpenFile(rl.filename, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
// Save output to log file.
|
|
|
|
_, err = f.Write([]byte(output))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-11-19 18:19:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Flush buffer.
|
|
|
|
rl.buf = map[logEntry]int{}
|
|
|
|
rl.lastFlush = time.Now()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-11-19 18:34:11 +00:00
|
|
|
// String returns string representation of logEntry.
|
2022-11-19 18:19:25 +00:00
|
|
|
func (e *logEntry) String() string {
|
|
|
|
return fmt.Sprintf(
|
|
|
|
"%s %s %s %s",
|
|
|
|
e.Proto,
|
|
|
|
e.IP,
|
|
|
|
e.URI,
|
|
|
|
e.UserAgent,
|
|
|
|
)
|
|
|
|
}
|