2021-09-16 09:16:07 +02:00
//go:build !windows
2017-08-11 11:04:58 +01:00
// +build !windows
package integration
import (
"bufio"
"net/http"
"os"
2019-11-14 16:40:05 +01:00
"strings"
2017-08-11 11:04:58 +01:00
"syscall"
2024-01-09 17:00:07 +01:00
"testing"
2017-08-11 11:04:58 +01:00
"time"
2024-01-09 17:00:07 +01:00
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/stretchr/testify/suite"
2020-09-16 15:46:04 +02:00
"github.com/traefik/traefik/v2/integration/try"
2021-11-25 10:10:06 +00:00
"github.com/traefik/traefik/v2/pkg/log"
2017-08-11 11:04:58 +01:00
)
2021-11-25 10:10:06 +00:00
const (
traefikTestLogFileRotated = traefikTestLogFile + ".rotated"
traefikTestAccessLogFileRotated = traefikTestAccessLogFile + ".rotated"
)
2020-05-11 12:06:07 +02:00
// Log rotation integration test suite.
2017-08-11 11:04:58 +01:00
type LogRotationSuite struct { BaseSuite }
2024-09-13 05:40:04 -04:00
func TestLogRotationSuite ( t * testing . T ) {
2024-01-09 17:00:07 +01:00
suite . Run ( t , new ( LogRotationSuite ) )
2021-11-25 10:10:06 +00:00
}
2018-01-11 10:04:03 +01:00
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) SetupSuite ( ) {
s . BaseSuite . SetupSuite ( )
os . Remove ( traefikTestAccessLogFile )
os . Remove ( traefikTestLogFile )
os . Remove ( traefikTestAccessLogFileRotated )
s . createComposeProject ( "access_log" )
s . composeUp ( )
}
func ( s * LogRotationSuite ) TearDownSuite ( ) {
s . BaseSuite . TearDownSuite ( )
2021-11-25 10:10:06 +00:00
generatedFiles := [ ] string {
traefikTestLogFile ,
traefikTestLogFileRotated ,
traefikTestAccessLogFile ,
traefikTestAccessLogFileRotated ,
}
for _ , filename := range generatedFiles {
if err := os . Remove ( filename ) ; err != nil {
log . WithoutContext ( ) . Warning ( err )
}
}
2018-01-11 10:04:03 +01:00
}
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) TestAccessLogRotation ( ) {
2017-08-11 11:04:58 +01:00
// Start Traefik
2024-01-09 17:00:07 +01:00
cmd , _ := s . cmdTraefik ( withConfigFile ( "fixtures/access_log_config.toml" ) )
defer s . displayTraefikLogFile ( traefikTestLogFile )
2018-02-05 09:48:03 +01:00
2017-08-11 11:04:58 +01:00
// Verify Traefik started ok
2024-01-09 17:00:07 +01:00
s . verifyEmptyErrorLog ( "traefik.log" )
2017-08-11 11:04:58 +01:00
2024-01-09 17:00:07 +01:00
s . waitForTraefik ( "server1" )
2017-08-11 11:04:58 +01:00
// Make some requests
2018-01-11 10:04:03 +01:00
req , err := http . NewRequest ( http . MethodGet , "http://127.0.0.1:8000/" , nil )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2018-01-11 10:04:03 +01:00
req . Host = "frontend1.docker.local"
err = try . Request ( req , 500 * time . Millisecond , try . StatusCodeIs ( http . StatusOK ) , try . HasBody ( ) )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// Rename access log
2021-11-25 10:10:06 +00:00
err = os . Rename ( traefikTestAccessLogFile , traefikTestAccessLogFileRotated )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// in the midst of the requests, issue SIGUSR1 signal to server process
err = cmd . Process . Signal ( syscall . SIGUSR1 )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// continue issuing requests
2018-01-11 10:04:03 +01:00
err = try . Request ( req , 500 * time . Millisecond , try . StatusCodeIs ( http . StatusOK ) , try . HasBody ( ) )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2018-01-11 10:04:03 +01:00
err = try . Request ( req , 500 * time . Millisecond , try . StatusCodeIs ( http . StatusOK ) , try . HasBody ( ) )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// Verify access.log.rotated output as expected
2024-01-09 17:00:07 +01:00
s . logAccessLogFile ( traefikTestAccessLogFileRotated )
lineCount := s . verifyLogLines ( traefikTestAccessLogFileRotated , 0 , true )
assert . GreaterOrEqual ( s . T ( ) , lineCount , 1 )
2017-08-11 11:04:58 +01:00
2017-10-06 09:20:13 +02:00
// make sure that the access log file is at least created before we do assertions on it
err = try . Do ( 1 * time . Second , func ( ) error {
_ , err := os . Stat ( traefikTestAccessLogFile )
return err
} )
2024-01-09 17:00:07 +01:00
assert . NoError ( s . T ( ) , err , "access log file was not created in time" )
2017-10-06 09:20:13 +02:00
2017-08-11 11:04:58 +01:00
// Verify access.log output as expected
2024-01-09 17:00:07 +01:00
s . logAccessLogFile ( traefikTestAccessLogFile )
lineCount = s . verifyLogLines ( traefikTestAccessLogFile , lineCount , true )
assert . Equal ( s . T ( ) , 3 , lineCount )
2017-08-11 11:04:58 +01:00
2024-01-09 17:00:07 +01:00
s . verifyEmptyErrorLog ( traefikTestLogFile )
2017-08-11 11:04:58 +01:00
}
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) TestTraefikLogRotation ( ) {
2017-08-11 11:04:58 +01:00
// Start Traefik
2024-01-09 17:00:07 +01:00
cmd := s . traefikCmd ( withConfigFile ( "fixtures/traefik_log_config.toml" ) )
2018-02-05 09:48:03 +01:00
2024-01-09 17:00:07 +01:00
s . waitForTraefik ( "server1" )
2017-08-11 11:04:58 +01:00
// Rename traefik log
2024-01-09 17:00:07 +01:00
err := os . Rename ( traefikTestLogFile , traefikTestLogFileRotated )
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// issue SIGUSR1 signal to server process
err = cmd . Process . Signal ( syscall . SIGUSR1 )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
err = cmd . Process . Signal ( syscall . SIGTERM )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// Allow time for switch to be processed
err = try . Do ( 3 * time . Second , func ( ) error {
_ , err = os . Stat ( traefikTestLogFile )
return err
} )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
// we have at least 6 lines in traefik.log.rotated
2024-01-09 17:00:07 +01:00
lineCount := s . verifyLogLines ( traefikTestLogFileRotated , 0 , false )
2017-08-11 11:04:58 +01:00
// GreaterOrEqualThan used to ensure test doesn't break
// If more log entries are output on startup
2024-01-09 17:00:07 +01:00
assert . GreaterOrEqual ( s . T ( ) , lineCount , 5 )
2017-08-11 11:04:58 +01:00
2018-07-03 10:02:03 +02:00
// Verify traefik.log output as expected
2024-01-09 17:00:07 +01:00
lineCount = s . verifyLogLines ( traefikTestLogFile , lineCount , false )
assert . GreaterOrEqual ( s . T ( ) , lineCount , 7 )
2017-08-11 11:04:58 +01:00
}
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) logAccessLogFile ( fileName string ) {
2021-03-04 20:08:03 +01:00
output , err := os . ReadFile ( fileName )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
log . WithoutContext ( ) . Infof ( "Contents of file %s\n%s" , fileName , string ( output ) )
2017-10-06 09:20:13 +02:00
}
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) verifyEmptyErrorLog ( name string ) {
2017-08-11 11:04:58 +01:00
err := try . Do ( 5 * time . Second , func ( ) error {
2021-03-04 20:08:03 +01:00
traefikLog , e2 := os . ReadFile ( name )
2017-08-11 11:04:58 +01:00
if e2 != nil {
return e2
}
2024-01-09 17:00:07 +01:00
assert . Empty ( s . T ( ) , string ( traefikLog ) )
2017-08-11 11:04:58 +01:00
return nil
} )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
}
2024-01-09 17:00:07 +01:00
func ( s * LogRotationSuite ) verifyLogLines ( fileName string , countInit int , accessLog bool ) int {
2017-08-11 11:04:58 +01:00
rotated , err := os . Open ( fileName )
2024-01-09 17:00:07 +01:00
require . NoError ( s . T ( ) , err )
2017-08-11 11:04:58 +01:00
rotatedLog := bufio . NewScanner ( rotated )
count := countInit
for rotatedLog . Scan ( ) {
line := rotatedLog . Text ( )
if accessLog {
if len ( line ) > 0 {
2019-11-14 16:40:05 +01:00
if ! strings . Contains ( line , "/api/rawdata" ) {
2024-01-09 17:00:07 +01:00
s . CheckAccessLogFormat ( line , count )
2019-11-14 16:40:05 +01:00
count ++
}
2017-08-11 11:04:58 +01:00
}
2019-11-14 16:40:05 +01:00
} else {
count ++
2017-08-11 11:04:58 +01:00
}
}
return count
}