mirror of
https://github.com/crazy-max/diun.git
synced 2025-12-27 07:31:37 +01:00
Bumps [github.com/dromara/carbon/v2](https://github.com/dromara/carbon) from 2.6.11 to 2.6.14. - [Release notes](https://github.com/dromara/carbon/releases) - [Commits](https://github.com/dromara/carbon/compare/v2.6.11...v2.6.14) --- updated-dependencies: - dependency-name: github.com/dromara/carbon/v2 dependency-version: 2.6.14 dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com>
189 lines
7.9 KiB
Go
189 lines
7.9 KiB
Go
package carbon
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// format map
|
|
var formatMap = map[byte]string{
|
|
'd': "02", // Day: Day of the month, 2 digits with leading zeros. Eg: 01 to 31.
|
|
'D': "Mon", // Day: A textual representation of a day, three letters. Eg: Mon through Sun.
|
|
'j': "2", // Day: Day of the month without leading zeros. Eg: 1 to 31.
|
|
'l': "Monday", // Day: A full textual representation of the day of the week. Eg: Sunday through Saturday.
|
|
'F': "January", // Month: A full textual representation of a month, such as January or March. Eg: January through December.
|
|
'm': "01", // Month: Numeric representation of a month, with leading zeros. Eg: 01 through 12.
|
|
'M': "Jan", // Month: A short textual representation of a month, three letters. Eg: Jan through Dec.
|
|
'n': "1", // Month: Numeric representation of a month, without leading zeros. Eg: 1 through 12.
|
|
'Y': "2006", // Year: A full numeric representation of a year, 4 digits. Eg: 1999 or 2003.
|
|
'y': "06", // Year: A two digit representation of a year. Eg: 99 or 03.
|
|
'a': "pm", // Time: Lowercase morning or afternoon sign. Eg: am or pm.
|
|
'A': "PM", // Time: Uppercase morning or afternoon sign. Eg: AM or PM.
|
|
'g': "3", // Time: 12-hour format of an hour without leading zeros. Eg: 1 through 12.
|
|
'h': "03", // Time: 12-hour format of an hour with leading zeros. Eg: 01 through 12.
|
|
'H': "15", // Time: 24-hour format of an hour with leading zeros. Eg: 00 through 23.
|
|
'i': "04", // Time: Minutes with leading zeros. Eg: 00 to 59.
|
|
's': "05", // Time: Seconds with leading zeros. Eg: 00 through 59.
|
|
'O': "-0700", // Zone: Difference to Greenwich time (GMT) in hours. Eg: +0200.
|
|
'P': "-07:00", // Zone: Difference to Greenwich time (GMT) with colon between hours and minutes. Eg: +02:00.
|
|
'Q': "Z0700", // Zone: ISO8601 timezone. Eg: Z, +0200.
|
|
'R': "Z07:00", // Zone: ISO8601 colon timezone. Eg: Z, +02:00.
|
|
'Z': "MST", // Zone: Zone name. Eg: UTC, EST, MDT ...
|
|
|
|
'u': "999", // Second: Millisecond. Eg: 999.
|
|
'v': "999999", // Second: Microsecond. Eg: 999999.
|
|
'x': "999999999", // Second: Nanosecond. Eg: 999999999.
|
|
|
|
'S': TimestampLayout, // Timestamp: Timestamp with second precision. Eg: 1699677240.
|
|
'U': TimestampMilliLayout, // Timestamp: Timestamp with millisecond precision. Eg: 1596604455666.
|
|
'V': TimestampMicroLayout, // Timestamp: Timestamp with microsecond precision. Eg: 1596604455666666.
|
|
'X': TimestampNanoLayout, // Timestamp: Timestamp with nanosecond precision. Eg: 1596604455666666666.
|
|
}
|
|
|
|
// default layouts
|
|
var defaultLayouts = []string{
|
|
DateTimeLayout, DateLayout, TimeLayout, DayDateTimeLayout,
|
|
|
|
"2006-01-02 15:04:05 -0700 MST", "2006-01-02T15:04:05Z07:00", "2006-01-02T15:04:05-07:00", "2006-01-02T15:04:05-0700", "2006-01-02T15:04:05",
|
|
|
|
ISO8601Layout, RFC1036Layout, RFC822Layout, RFC822ZLayout, RFC850Layout, RFC1123Layout, RFC1123ZLayout, RFC3339Layout, RFC7231Layout,
|
|
KitchenLayout, CookieLayout, ANSICLayout, UnixDateLayout, RubyDateLayout,
|
|
|
|
ShortDateTimeLayout, ShortDateLayout, ShortTimeLayout,
|
|
|
|
DateTimeMilliLayout, DateTimeMicroLayout, DateTimeNanoLayout,
|
|
DateMilliLayout, DateMicroLayout, DateNanoLayout,
|
|
TimeMilliLayout, TimeMicroLayout, TimeNanoLayout,
|
|
|
|
ShortDateTimeMilliLayout, ShortDateTimeMicroLayout, ShortDateTimeNanoLayout,
|
|
ShortDateMilliLayout, ShortDateMicroLayout, ShortDateNanoLayout,
|
|
ShortTimeMilliLayout, ShortTimeMicroLayout, ShortTimeNanoLayout,
|
|
|
|
ISO8601MilliLayout, ISO8601MicroLayout, ISO8601NanoLayout,
|
|
RFC3339MilliLayout, RFC3339MicroLayout, RFC3339NanoLayout,
|
|
|
|
"15:04:05-07", // postgres time with time zone type
|
|
"2006-01-02 15:04:05-07", // postgres timestamp with time zone type
|
|
"2006-01-02 15:04:05-07:00", // sqlite text type
|
|
"2006-01-02 15:04:05.999999999 -07:00", // sqlserver datetimeoffset type
|
|
"2006",
|
|
"2006-1-2 15:4:5 -0700 MST", "2006-1-2 3:4:5 -0700 MST",
|
|
"2006-1", "2006-1-2", "2006-1-2 15", "2006-1-2 15:4", "2006-1-2 15:4:5", "2006-1-2 15:4:5.999999999",
|
|
"2006.1", "2006.1.2", "2006.1.2 15", "2006.1.2 15:4", "2006.1.2 15:4:5", "2006.1.2 15:4:5.999999999",
|
|
"2006/1", "2006/1/2", "2006/1/2 15", "2006/1/2 15:4", "2006/1/2 15:4:5", "2006/1/2 15:4:5.999999999",
|
|
"2006-01-02 15:04:05PM MST", "2006-01-02 15:04:05.999999999PM MST", "2006-1-2 15:4:5PM MST", "2006-1-2 15:4:5.999999999PM MST",
|
|
"2006-01-02 15:04:05 PM MST", "2006-01-02 15:04:05.999999999 PM MST", "2006-1-2 15:4:5 PM MST", "2006-1-2 15:4:5.999999999 PM MST",
|
|
"1/2/2006", "1/2/2006 15", "1/2/2006 15:4", "1/2/2006 15:4:5", "1/2/2006 15:4:5.999999999",
|
|
"2006-1-2 15:4:5.999999999 -0700 MST", "2006-1-2 15:04:05 -0700 MST", "2006-1-2 15:04:05.999999999 -0700 MST",
|
|
"2006-01-02T15:04:05.999999999", "2006-1-2T3:4:5", "2006-1-2T3:4:5.999999999",
|
|
"2006-01-02T15:04:05.999999999Z07", "2006-1-2T15:4:5Z07", "2006-1-2T15:4:5.999999999Z07",
|
|
"2006-01-02T15:04:05.999999999Z07:00", "2006-1-2T15:4:5Z07:00", "2006-1-2T15:4:5.999999999Z07:00",
|
|
"2006-01-02T15:04:05.999999999-07:00", "2006-1-2T15:4:5-07:00", "2006-1-2T15:4:5.999999999-07:00",
|
|
"2006-01-02T15:04:05.999999999-0700", "2006-1-2T3:4:5-0700", "2006-1-2T3:4:5.999999999-0700",
|
|
"20060102150405-07:00", "20060102150405.999999999-07:00",
|
|
"20060102150405Z07:00", "20060102150405.999999999Z07:00",
|
|
}
|
|
|
|
// layoutCache caches format to layout conversions to avoid repeated parsing
|
|
var layoutCache sync.Map
|
|
|
|
// converts format to layout.
|
|
func format2layout(format string) string {
|
|
// Check cache first
|
|
if cached, exists := layoutCache.Load(format); exists {
|
|
return cached.(string)
|
|
}
|
|
|
|
// Pre-allocate buffer with estimated capacity to reduce allocations
|
|
estimatedSize := len(format) * 2
|
|
buffer := make([]byte, 0, estimatedSize)
|
|
|
|
for i := 0; i < len(format); i++ {
|
|
if layout, ok := formatMap[format[i]]; ok {
|
|
buffer = append(buffer, layout...)
|
|
} else {
|
|
switch format[i] {
|
|
case '\\': // raw output, no parse
|
|
// Ensure we don't go out of bounds
|
|
if i+1 < len(format) {
|
|
buffer = append(buffer, format[i+1])
|
|
i++
|
|
}
|
|
continue
|
|
default:
|
|
buffer = append(buffer, format[i])
|
|
}
|
|
}
|
|
}
|
|
|
|
result := string(buffer)
|
|
|
|
// Cache the result for common formats to improve performance
|
|
// Only cache reasonably short formats to avoid memory bloat
|
|
if len(format) <= 50 {
|
|
layoutCache.Store(format, result)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// timezoneCache caches parsed timezone locations to avoid repeated parsing
|
|
var timezoneCache sync.Map
|
|
|
|
// parses a timezone string as a time.Location instance.
|
|
func parseTimezone(timezone string) (loc *Location, err error) {
|
|
if timezone == "" {
|
|
return nil, ErrEmptyTimezone()
|
|
}
|
|
|
|
// Check cache first
|
|
if cached, exists := timezoneCache.Load(timezone); exists {
|
|
return cached.(*Location), nil
|
|
}
|
|
|
|
if loc, err = time.LoadLocation(timezone); err != nil {
|
|
err = fmt.Errorf("%w: %w", ErrInvalidTimezone(timezone), err)
|
|
return
|
|
}
|
|
|
|
// Cache the successful result
|
|
timezoneCache.Store(timezone, loc)
|
|
return
|
|
}
|
|
|
|
// durationCache caches parsed durations to avoid repeated parsing
|
|
var durationCache sync.Map
|
|
|
|
// parses a duration string as a time.Duration instance.
|
|
func parseDuration(duration string) (dur Duration, err error) {
|
|
if duration == "" {
|
|
return 0, ErrEmptyDuration()
|
|
}
|
|
|
|
// Check cache first for common durations
|
|
if cached, exists := durationCache.Load(duration); exists {
|
|
return cached.(Duration), nil
|
|
}
|
|
|
|
if dur, err = time.ParseDuration(duration); err != nil {
|
|
err = fmt.Errorf("%w: %w", ErrInvalidDuration(duration), err)
|
|
return
|
|
}
|
|
|
|
// Cache the successful result for common durations
|
|
// Only cache reasonably short durations to avoid memory bloat
|
|
if len(duration) <= 20 {
|
|
durationCache.Store(duration, dur)
|
|
}
|
|
return
|
|
}
|
|
|
|
// gets absolute value.
|
|
func getAbsValue(value int64) int64 {
|
|
// Use bit manipulation for better performance
|
|
// For positive numbers: value ^ 0 - 0 = value
|
|
// For negative numbers: value ^ -1 - (-1) = ^value + 1 = -value
|
|
return (value ^ (value >> 63)) - (value >> 63)
|
|
}
|