-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
369 lines (349 loc) · 9.67 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
// Program s3logger collects json messages over TCP, stores them into
// gzip-compressed files split by time and uploads these files to AWS s3 bucket.
//
// Once started, service accepts TCP connections and expects clients to send
// streams of json objects over such connections. s3logger only closes
// connection if it encounters malformed json or single object size exceeds
// 4 MiB limit. s3logger only reads data from the client.
//
// s3logger accumulates received messages over predefined time window (-t flag)
// to a temporary log file creating new ones as needed; previous files are
// uploaded to s3 bucket in background and removed after successful upload.
// Optionally maximum size of input read can be specified in megabytes (-mb
// flag) to rotate file before reaching predefined time. Program only writes to
// a single temporary log file at a time, so json messages received from
// multiple concurrent connections are interleaved into a single json stream. It
// does its best not to lose messages, but can still drop them if they're coming
// faster than could be saved on disk or there's any disk write error. Stored
// messages are separated by new line (0xa).
//
// s3logger uploads files to a specified bucket using predefined s3 object
// naming scheme:
//
// dt=2018-02-09/20180209T213803_df718a7818e53243.json.gz
//
// It uses dt=YYYY-MM-DD "directories", object name base starting with date
// and time when log file was created (UTC) followed by hex-encoded 64-bit
// random value and .json.gz suffix.
//
// See
// https://godoc.org/github.com/aws/aws-sdk-go/aws/session#hdr-Environment_Variables
// on how to configure s3 bucket access credentials.
//
// s3logger does not use TLS for its listener at the moment as it is expected to
// run on localhost or inside trusted network.
//
// Usage of s3logger:
// -addr string
// address to listen (default "localhost:8080")
// -bucket string
// s3 bucket to upload logs
// -dir string
// directory to keep unsent files (default "/var/spool/s3logger")
// -mb int
// megabytes of input read until file is rotated (0 to disable) (default 512)
// -prefix string
// s3 object name prefix (directory in a bucket)
// -t duration
// time to use single file (min 1m) (default 5m0s)
package main
import (
"bufio"
"compress/gzip"
"context"
"crypto/rand"
"encoding/hex"
"encoding/json"
"errors"
"flag"
"io"
"io/fs"
"log"
"net"
"os"
"os/signal"
"path"
"path/filepath"
"strings"
"sync"
"syscall"
"time"
"github.com/aws/aws-sdk-go-v2/config"
"github.com/aws/aws-sdk-go-v2/feature/s3/manager"
"github.com/aws/aws-sdk-go-v2/service/s3"
"golang.org/x/sync/errgroup"
)
func main() {
log.SetFlags(0)
args := runArgs{
Dir: "/var/spool/s3logger",
D: 5 * time.Minute,
Mb: 512,
Addr: "localhost:8080",
}
flag.StringVar(&args.Addr, "addr", args.Addr, "address to listen")
flag.StringVar(&args.Bucket, "bucket", args.Bucket, "s3 bucket to upload logs")
flag.StringVar(&args.Dir, "dir", args.Dir, "directory to keep unsent files")
flag.StringVar(&args.Prefix, "prefix", args.Prefix, "s3 object name prefix (directory in a bucket)")
flag.DurationVar(&args.D, "t", args.D, "time to use single file (min 1m)")
flag.IntVar(&args.Mb, "mb", args.Mb, "megabytes of input read until file is rotated (0 to disable)")
flag.Parse()
ctx, cancel := signal.NotifyContext(context.Background(), os.Interrupt, syscall.SIGTERM)
defer cancel()
if err := run(ctx, args); err != nil {
log.Fatal(err)
}
}
type runArgs struct {
Addr string
Bucket string
Dir string
Prefix string
D time.Duration
Mb int
}
func run(ctx context.Context, args runArgs) error {
if args.Addr == "" {
return errors.New("empty address")
}
if args.Bucket == "" {
return errors.New("empty bucket name")
}
if args.Dir == "" {
args.Dir = "."
}
if args.D < time.Minute {
args.D = time.Minute
}
if err := os.MkdirAll(args.Dir, 0770); err != nil {
return err
}
cfg, err := config.LoadDefaultConfig(ctx)
if err != nil {
return err
}
upl := manager.NewUploader(s3.NewFromConfig(cfg))
ln, err := net.Listen("tcp", args.Addr)
if err != nil {
return err
}
defer ln.Close()
srv := &server{dir: args.Dir, ch: make(chan json.RawMessage, 1000), wake: make(chan struct{})}
defer func() {
c, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
if err := srv.uploadOnce(c, args.Bucket, args.Prefix, upl); err != nil {
log.Printf("final upload attempt: %v", err)
}
}()
group, ctx := errgroup.WithContext(ctx)
group.Go(func() error { <-ctx.Done(); return ln.Close() })
group.Go(func() error { return srv.ingest(ctx, args.Mb<<20, args.D) })
group.Go(func() error { return srv.upload(ctx, args.D/2, args.Bucket, args.Prefix, upl) })
group.Go(func() error {
for {
conn, err := ln.Accept()
if err != nil {
select {
case <-ctx.Done():
return nil
default:
}
return err
}
go func(conn net.Conn) {
if err := srv.handleConn(ctx, conn); err != nil {
log.Printf("%s: %v", conn.RemoteAddr(), err)
}
}(conn)
}
})
return group.Wait()
}
type server struct {
dir string
ch chan json.RawMessage
wake chan struct{} // to wake uploading goroutine early
mu sync.Mutex
w io.WriteCloser
name string
}
func (srv *server) upload(ctx context.Context, d time.Duration, bucket, prefix string, upl *manager.Uploader) error {
ticker := time.NewTicker(d)
defer ticker.Stop()
for {
select {
case <-ctx.Done():
return nil
case <-ticker.C:
_ = srv.uploadOnce(ctx, bucket, prefix, upl)
case <-srv.wake:
_ = srv.uploadOnce(ctx, bucket, prefix, upl)
}
}
}
func (srv *server) uploadOnce(ctx context.Context, bucket, prefix string, upl *manager.Uploader) error {
walkFn := func(path string, d fs.DirEntry, err error) error {
if err != nil || d.IsDir() || !strings.HasSuffix(path, ".json.gz") {
return nil
}
if d.Name() == filepath.Base(srv.currentName()) {
return nil
}
ctx, cancel := context.WithTimeout(ctx, 10*time.Minute)
defer cancel()
start := time.Now()
switch err := uploadFile(ctx, upl, bucket, prefix, path); err {
case nil:
log.Printf("%q uploaded in %v", path, time.Since(start).Round(100*time.Millisecond))
_ = os.Remove(path)
default:
log.Printf("%q upload: %v", path, err)
}
return nil
}
return filepath.WalkDir(srv.dir, walkFn)
}
func uploadFile(ctx context.Context, upl *manager.Uploader, bucket, prefix, name string) error {
f, err := os.Open(name)
if err != nil {
return err
}
defer f.Close()
key := path.Join(prefix, strings.Replace(filepath.Base(name), ".", "/", 1))
_, err = upl.Upload(ctx, &s3.PutObjectInput{Bucket: &bucket, Key: &key, Body: f})
return err
}
func (srv *server) ingest(ctx context.Context, maxSize int, d time.Duration) error {
var enc *json.Encoder
var size int
timer := time.NewTimer(d)
defer timer.Stop()
for {
select {
case msg := <-srv.ch:
if enc == nil {
switch w, err := srv.create(); err {
case nil:
enc = json.NewEncoder(w)
size = 0
default:
log.Print("file create: ", err)
continue
}
}
if err := enc.Encode(msg); err != nil {
log.Print("message write: ", err)
srv.close()
enc = nil
}
size += len(msg)
if maxSize > 0 && size >= maxSize {
timer.Reset(d)
srv.close()
enc = nil
select {
case srv.wake <- struct{}{}:
default:
}
}
case <-timer.C:
timer.Reset(d)
srv.close()
enc = nil
select {
case srv.wake <- struct{}{}:
default:
}
case <-ctx.Done():
return srv.close()
}
}
}
func (srv *server) create() (io.Writer, error) {
srv.mu.Lock()
defer srv.mu.Unlock()
f, err := os.Create(filepath.Join(srv.dir, randomName()+".json.gz"))
if err != nil {
return nil, err
}
bw := bufio.NewWriterSize(f, 1<<16)
gw, err := gzip.NewWriterLevel(bw, gzip.BestSpeed)
if err != nil {
panic(err)
}
srv.name, srv.w = f.Name(), chain{gw, bw, f}
return srv.w, nil
}
func (srv *server) close() error {
srv.mu.Lock()
defer srv.mu.Unlock()
if srv.w == nil {
return nil
}
w := srv.w
srv.name, srv.w = "", nil
return w.Close()
}
func (srv *server) currentName() string {
srv.mu.Lock()
defer srv.mu.Unlock()
return srv.name
}
func (srv *server) handleConn(ctx context.Context, conn io.ReadCloser) error {
defer conn.Close()
if tc, ok := conn.(*net.TCPConn); ok {
tc.SetKeepAlive(true)
tc.SetKeepAlivePeriod(3 * time.Minute)
}
const maxSize = 4 << 20 // single json object size limit (approximate)
rd := &io.LimitedReader{R: bufio.NewReader(conn), N: maxSize}
dec := json.NewDecoder(rd)
for {
var msg json.RawMessage
switch err := dec.Decode(&msg); err {
case nil:
rd.N = maxSize
case io.EOF:
return nil
default:
return err
}
select {
case srv.ch <- msg:
case <-ctx.Done():
return nil
}
}
}
// chain implements io.WriteCloser that passes writes to the first (0 index)
// Writer and on Close flushes and closes each Writer if they implement relevant
// interfaces.
type chain []io.Writer
func (cw chain) Write(p []byte) (int, error) { return cw[0].Write(p) }
func (cw chain) Close() error {
var errOut error
for _, w := range cw {
if f, ok := w.(interface{ Flush() error }); ok {
if err := f.Flush(); err != nil && errOut == nil {
errOut = err
}
}
if c, ok := w.(io.Closer); ok {
if err := c.Close(); err != nil && errOut == nil {
errOut = err
}
}
}
return errOut
}
// randomName returns base name of the temporary file. It encodes file creation
// date and can be translated to s3 object name in date-sharded "subdirectories"
// by replacing . with /.
func randomName() string {
b := make([]byte, 8)
if _, err := rand.Read(b); err != nil {
panic(err)
}
return time.Now().In(time.UTC).Format("dt=2006-01-02.20060102T150405_") + hex.EncodeToString(b)
}