v0.0.585
	
		
			
	
		
	
	
		
	
		
			All checks were successful
		
		
	
	
		
			
				
	
				Build Docker and Deploy / Run goext test-suite (push) Successful in 2m36s
				
			
		
		
	
	
				
					
				
			
		
			All checks were successful
		
		
	
	Build Docker and Deploy / Run goext test-suite (push) Successful in 2m36s
				
			This commit is contained in:
		
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @@ -32,7 +32,7 @@ require ( | |||||||
| 	github.com/gin-contrib/sse v1.1.0 // indirect | 	github.com/gin-contrib/sse v1.1.0 // indirect | ||||||
| 	github.com/go-playground/locales v0.14.1 // indirect | 	github.com/go-playground/locales v0.14.1 // indirect | ||||||
| 	github.com/go-playground/universal-translator v0.18.1 // indirect | 	github.com/go-playground/universal-translator v0.18.1 // indirect | ||||||
| 	github.com/go-playground/validator/v10 v10.26.0 // indirect | 	github.com/go-playground/validator/v10 v10.27.0 // indirect | ||||||
| 	github.com/goccy/go-json v0.10.5 // indirect | 	github.com/goccy/go-json v0.10.5 // indirect | ||||||
| 	github.com/golang/snappy v1.0.0 // indirect | 	github.com/golang/snappy v1.0.0 // indirect | ||||||
| 	github.com/google/uuid v1.5.0 // indirect | 	github.com/google/uuid v1.5.0 // indirect | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.sum
									
									
									
									
									
								
							| @@ -86,6 +86,8 @@ github.com/go-playground/validator/v10 v10.25.0 h1:5Dh7cjvzR7BRZadnsVOzPhWsrwUr0 | |||||||
| github.com/go-playground/validator/v10 v10.25.0/go.mod h1:GGzBIJMuE98Ic/kJsBXbz1x/7cByt++cQ+YOuDM5wus= | github.com/go-playground/validator/v10 v10.25.0/go.mod h1:GGzBIJMuE98Ic/kJsBXbz1x/7cByt++cQ+YOuDM5wus= | ||||||
| github.com/go-playground/validator/v10 v10.26.0 h1:SP05Nqhjcvz81uJaRfEV0YBSSSGMc/iMaVtFbr3Sw2k= | github.com/go-playground/validator/v10 v10.26.0 h1:SP05Nqhjcvz81uJaRfEV0YBSSSGMc/iMaVtFbr3Sw2k= | ||||||
| github.com/go-playground/validator/v10 v10.26.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo= | github.com/go-playground/validator/v10 v10.26.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo= | ||||||
|  | github.com/go-playground/validator/v10 v10.27.0 h1:w8+XrWVMhGkxOaaowyKH35gFydVHOvC0/uWoy2Fzwn4= | ||||||
|  | github.com/go-playground/validator/v10 v10.27.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo= | ||||||
| github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y= | github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y= | ||||||
| github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg= | github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg= | ||||||
| github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA= | github.com/goccy/go-json v0.10.3 h1:KZ5WoDbxAIgm2HNbYckL0se1fHD6rz5j4ywS6ebzDqA= | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.584" | const GoextVersion = "0.0.585" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2025-06-26T16:48:07+0200" | const GoextVersionTimestamp = "2025-07-04T11:46:00+0200" | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package timeext | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"math" | 	"math" | ||||||
|  | 	"sort" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -142,6 +143,38 @@ func Max(a time.Time, b time.Time) time.Time { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func Avg(v ...time.Time) time.Time { | ||||||
|  | 	if len(v) == 0 { | ||||||
|  | 		return time.Time{} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var sum int64 | ||||||
|  | 	for _, t := range v { | ||||||
|  | 		sum += t.UnixNano() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return time.Unix(0, sum/int64(len(v))) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func Median(v ...time.Time) time.Time { | ||||||
|  | 	if len(v) == 0 { | ||||||
|  | 		return time.Time{} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	sorted := make([]time.Time, len(v)) | ||||||
|  | 	copy(sorted, v) | ||||||
|  | 	sort.Slice(sorted, func(i, j int) bool { | ||||||
|  | 		return sorted[i].UnixNano() < sorted[j].UnixNano() | ||||||
|  | 	}) | ||||||
|  |  | ||||||
|  | 	mid := len(sorted) / 2 | ||||||
|  | 	if len(sorted)%2 == 0 { | ||||||
|  | 		return Avg(sorted[mid-1], sorted[mid]) | ||||||
|  | 	} else { | ||||||
|  | 		return sorted[mid] | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| func UnixFloatSeconds(v float64) time.Time { | func UnixFloatSeconds(v float64) time.Time { | ||||||
| 	sec, dec := math.Modf(v) | 	sec, dec := math.Modf(v) | ||||||
| 	return time.Unix(int64(sec), int64(dec*(1e9))) | 	return time.Unix(int64(sec), int64(dec*(1e9))) | ||||||
|   | |||||||
| @@ -227,3 +227,173 @@ func TestDaysInMonth_FebruaryNonLeapYear(t *testing.T) { | |||||||
| 		t.Errorf("Expected %d but got %d", expected, result) | 		t.Errorf("Expected %d but got %d", expected, result) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestAvg_MultipleValues(t *testing.T) { | ||||||
|  | 	t1 := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t2 := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t3 := time.Date(2022, 1, 5, 0, 0, 0, 0, time.UTC) | ||||||
|  |  | ||||||
|  | 	// Average should be January 3, 2022 (middle date) | ||||||
|  | 	expected := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	result := Avg(t1, t2, t3) | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestAvg_TwoValues(t *testing.T) { | ||||||
|  | 	t1 := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t2 := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  |  | ||||||
|  | 	expected := time.Date(2022, 1, 2, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	result := Avg(t1, t2) | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestAvg_EmptySlice(t *testing.T) { | ||||||
|  | 	result := Avg() | ||||||
|  | 	expected := time.Time{} | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected zero time but got %v", result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestMedian_OddNumberOfValues(t *testing.T) { | ||||||
|  | 	t1 := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t2 := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t3 := time.Date(2022, 1, 5, 0, 0, 0, 0, time.UTC) | ||||||
|  |  | ||||||
|  | 	// Median should be the middle date | ||||||
|  | 	expected := t2 | ||||||
|  | 	result := Median(t1, t2, t3) | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestMedian_EvenNumberOfValues(t *testing.T) { | ||||||
|  | 	t1 := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t2 := time.Date(2022, 1, 2, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t3 := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t4 := time.Date(2022, 1, 4, 0, 0, 0, 0, time.UTC) | ||||||
|  |  | ||||||
|  | 	// Median for even number of values should be average of middle two | ||||||
|  | 	expected := time.Date(2022, 1, 2, 12, 0, 0, 0, time.UTC) | ||||||
|  | 	result := Median(t1, t2, t3, t4) | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestMedian_UnsortedValues(t *testing.T) { | ||||||
|  | 	t1 := time.Date(2022, 1, 5, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t2 := time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC) | ||||||
|  | 	t3 := time.Date(2022, 1, 3, 0, 0, 0, 0, time.UTC) | ||||||
|  |  | ||||||
|  | 	// Median should correctly sort values first | ||||||
|  | 	expected := t3 | ||||||
|  | 	result := Median(t1, t2, t3) | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestMedian_EmptySlice(t *testing.T) { | ||||||
|  | 	result := Median() | ||||||
|  | 	expected := time.Time{} | ||||||
|  |  | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected zero time but got %v", result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToDatePart(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 1, 1, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2022, 1, 1, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToDatePart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToWeekStart(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	// January 5, 2022 was a Wednesday | ||||||
|  | 	tm := time.Date(2022, 1, 5, 13, 14, 15, 0, tz) | ||||||
|  | 	// Should return Monday, January 3, 2022 | ||||||
|  | 	expected := time.Date(2022, 1, 3, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToWeekStart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToWeekStart_WhenMonday(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	// January 3, 2022 was a Monday | ||||||
|  | 	tm := time.Date(2022, 1, 3, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2022, 1, 3, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToWeekStart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToMonthStart(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 1, 15, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2022, 1, 1, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToMonthStart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToMonthEnd(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 1, 15, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2022, 2, 1, 0, 0, 0, 0, tz).Add(-1) | ||||||
|  | 	result := TimeToMonthEnd(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToYearStart(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 5, 15, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2022, 1, 1, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToYearStart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToYearEnd(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 5, 15, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2023, 1, 1, 0, 0, 0, 0, tz).Add(-1) | ||||||
|  | 	result := TimeToYearEnd(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestTimeToNextYearStart(t *testing.T) { | ||||||
|  | 	tz := TimezoneBerlin | ||||||
|  | 	tm := time.Date(2022, 5, 15, 13, 14, 15, 0, tz) | ||||||
|  | 	expected := time.Date(2023, 1, 1, 0, 0, 0, 0, tz) | ||||||
|  | 	result := TimeToNextYearStart(tm, tz) | ||||||
|  | 	if !result.Equal(expected) { | ||||||
|  | 		t.Errorf("Expected %v but got %v", expected, result) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user