16
"github.com/containers/podman/v5/pkg/util"
17
"github.com/containers/storage/pkg/lockfile"
18
"github.com/nxadm/tail"
19
"github.com/sirupsen/logrus"
20
"golang.org/x/sys/unix"
25
type EventLogFile struct {
26
options EventerOptions
30
func newLogFileEventer(options EventerOptions) (*EventLogFile, error) {
32
if err := os.MkdirAll(filepath.Dir(options.LogFilePath), 0700); err != nil {
33
return nil, fmt.Errorf("creating events dirs: %w", err)
37
fd, err := os.OpenFile(options.LogFilePath, os.O_RDONLY|os.O_CREATE, 0700)
39
return nil, fmt.Errorf("failed to create event log file: %w", err)
41
return &EventLogFile{options: options}, fd.Close()
45
func (e EventLogFile) Write(ee Event) error {
47
lock, err := lockfile.GetLockFile(e.options.LogFilePath + ".lock")
54
eventJSONString, err := ee.ToJSONString()
59
if _, err := rotateLog(e.options.LogFilePath, eventJSONString, e.options.LogFileMaxSize); err != nil {
63
return e.writeString(eventJSONString)
66
func (e EventLogFile) writeString(s string) error {
67
f, err := os.OpenFile(e.options.LogFilePath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0700)
71
return writeToFile(s, f)
74
func writeToFile(s string, f *os.File) error {
75
_, err := f.WriteString(s + "\n")
79
func (e EventLogFile) getTail(options ReadOptions) (*tail.Tail, error) {
80
seek := tail.SeekInfo{Offset: 0, Whence: io.SeekEnd}
81
if options.FromStart || !options.Stream {
84
stream := options.Stream
85
return tail.TailFile(e.options.LogFilePath, tail.Config{ReOpen: stream, Follow: stream, Location: &seek, Logger: tail.DiscardingLogger, Poll: true})
88
func (e EventLogFile) readRotateEvent(event *Event) (begin bool, end bool, err error) {
89
if event.Status != Rotate {
92
if event.Details.Attributes == nil {
96
switch event.Details.Attributes[rotateEventAttribute] {
97
case rotateEventBegin:
104
err = fmt.Errorf("unknown rotate-event attribute %q", event.Details.Attributes[rotateEventAttribute])
110
func (e EventLogFile) Read(ctx context.Context, options ReadOptions) error {
111
defer close(options.EventChannel)
112
filterMap, err := generateEventFilters(options.Filters, options.Since, options.Until)
114
return fmt.Errorf("failed to parse event filters: %w", err)
116
t, err := e.getTail(options)
120
if len(options.Until) > 0 {
121
untilTime, err := util.ParseInputTime(options.Until, false)
126
time.Sleep(time.Until(untilTime))
127
if err := t.Stop(); err != nil {
128
logrus.Errorf("Stopping logger: %v", err)
132
logrus.Debugf("Reading events from file %q", e.options.LogFilePath)
137
readTime, err := func() (time.Time, error) {
139
lock, err := lockfile.GetLockFile(e.options.LogFilePath + ".lock")
141
return time.Time{}, err
145
return time.Now(), nil
158
t.Kill(errors.New("hangup by client"))
160
case line, ok = <-t.Lines:
168
event, err := newEventFromJSONString(line.Text)
173
case Image, Volume, Pod, Container, Network:
176
begin, end, err := e.readRotateEvent(event)
180
if begin && event.Time.After(readTime) {
186
logrus.Debugf("Skipping already read events after log-file rotation: %v", event)
192
return fmt.Errorf("event type %s is not valid in %s", event.Type.String(), e.options.LogFilePath)
197
if applyFilters(event, filterMap) {
198
options.EventChannel <- event
204
func (e EventLogFile) String() string {
205
return LogFile.String()
209
rotateEventAttribute = "io.podman.event.rotate"
210
rotateEventBegin = "begin"
211
rotateEventEnd = "end"
214
func writeRotateEvent(f *os.File, logFilePath string, begin bool) error {
215
rEvent := NewEvent(Rotate)
217
rEvent.Name = logFilePath
218
rEvent.Attributes = make(map[string]string)
220
rEvent.Attributes[rotateEventAttribute] = rotateEventBegin
222
rEvent.Attributes[rotateEventAttribute] = rotateEventEnd
224
rotateJSONString, err := rEvent.ToJSONString()
228
return writeToFile(rotateJSONString, f)
232
func rotateLog(logfile string, content string, limit uint64) (bool, error) {
233
needsRotation, err := logNeedsRotation(logfile, content, limit)
234
if err != nil || !needsRotation {
237
if err := truncate(logfile); err != nil {
244
func logNeedsRotation(logfile string, content string, limit uint64) (bool, error) {
248
file, err := os.Stat(logfile)
250
if errors.Is(err, os.ErrNotExist) {
256
var filesize = uint64(file.Size())
257
var contentsize = uint64(len([]rune(content)))
258
if filesize+contentsize < limit {
266
func truncate(filePath string) error {
267
orig, err := os.Open(filePath)
273
origFinfo, err := orig.Stat()
278
size := origFinfo.Size()
279
threshold := size / 2
281
tmp, err := os.CreateTemp(path.Dir(filePath), "")
285
tmp, err = os.CreateTemp("", "")
293
if _, err := orig.Seek(threshold, 0); err != nil {
296
reader := bufio.NewReader(orig)
297
if _, err := reader.ReadString('\n'); err != nil {
298
if !errors.Is(err, io.EOF) {
303
if err := writeRotateEvent(tmp, filePath, true); err != nil {
304
return fmt.Errorf("writing rotation event begin marker: %w", err)
306
if _, err := reader.WriteTo(tmp); err != nil {
307
return fmt.Errorf("writing truncated contents: %w", err)
309
if err := writeRotateEvent(tmp, filePath, false); err != nil {
310
return fmt.Errorf("writing rotation event end marker: %w", err)
313
if err := renameLog(tmp.Name(), filePath); err != nil {
314
return fmt.Errorf("writing back %s to %s: %w", tmp.Name(), filePath, err)
321
func renameLog(from, to string) error {
322
err := os.Rename(from, to)
327
if !errors.Is(err, unix.EXDEV) {
333
fFrom, err := os.Open(from)
341
if err := os.Remove(to); err != nil {
342
return fmt.Errorf("recreating file %s: %w", to, err)
345
fTo, err := os.Create(to)
351
if _, err := io.Copy(fTo, fFrom); err != nil {
352
return fmt.Errorf("writing back from temporary file: %w", err)
355
if err := os.Remove(from); err != nil {
356
return fmt.Errorf("removing temporary file: %w", err)