Files
tidb/pkg/timer/api/schedule_policy_test.go

188 lines
4.6 KiB
Go

// Copyright 2023 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package api
import (
"fmt"
"testing"
"time"
"github.com/stretchr/testify/require"
)
func TestIntervalPolicy(t *testing.T) {
watermark1 := time.Now()
watermark2, err := time.Parse(time.RFC3339, "2021-11-21T11:21:31Z")
require.NoError(t, err)
cases := []struct {
expr string
err bool
interval time.Duration
}{
{
expr: "6m",
interval: 6 * time.Minute,
},
{
expr: "7h",
interval: 7 * time.Hour,
},
{
expr: "8d",
interval: 8 * 24 * time.Hour,
},
{
expr: "11",
err: true,
},
}
for _, c := range cases {
p, err := CreateSchedEventPolicy(SchedEventInterval, c.expr)
if c.err {
require.Nil(t, p)
require.ErrorContains(t, err, fmt.Sprintf("invalid schedule event expr '%s'", c.expr))
continue
}
require.NotNil(t, p)
require.IsType(t, &SchedIntervalPolicy{}, p)
require.NoError(t, err)
tm, ok := p.NextEventTime(watermark1)
require.True(t, ok)
require.Equal(t, watermark1.Add(c.interval), tm)
tm, ok = p.NextEventTime(watermark2)
require.True(t, ok)
require.Equal(t, watermark2.Add(c.interval), tm)
}
}
func TestCronPolicy(t *testing.T) {
locE2 := time.FixedZone("UTC+1", 2*60*60)
locW2 := time.FixedZone("UTC-1", -2*60*60)
cases := []struct {
expr string
err bool
tm time.Time
next time.Time
}{
{
expr: "",
err: true,
},
{
expr: "aaa",
err: true,
},
{
expr: "61 1 * * *",
err: true,
},
{
expr: "@hourly",
tm: time.Date(2021, 11, 21, 11, 21, 31, 0, time.UTC),
next: time.Date(2021, 11, 21, 12, 0, 0, 0, time.UTC),
},
{
expr: "@hourly",
tm: time.Date(2021, 11, 21, 12, 0, 0, 0, time.Local),
next: time.Date(2021, 11, 21, 13, 0, 0, 0, time.Local),
},
{
expr: "@daily",
tm: time.Date(2021, 11, 21, 11, 21, 31, 0, time.Local),
next: time.Date(2021, 11, 22, 0, 0, 0, 0, time.Local),
},
{
expr: "@weekly",
tm: time.Date(2021, 11, 19, 11, 21, 31, 0, locE2), // Friday
next: time.Date(2021, 11, 21, 0, 0, 0, 0, locE2), // Sunday
},
{
expr: "@monthly",
tm: time.Date(2021, 12, 19, 11, 21, 31, 0, locW2),
next: time.Date(2022, 1, 1, 0, 0, 0, 0, locW2),
},
{
expr: "@yearly",
tm: time.Date(2021, 12, 19, 11, 21, 31, 0, time.UTC),
next: time.Date(2022, 1, 1, 0, 0, 0, 0, time.UTC),
},
{
expr: "12 12 * * *",
tm: time.Date(2021, 12, 19, 11, 21, 31, 0, time.Local),
next: time.Date(2021, 12, 19, 12, 12, 0, 0, time.Local),
},
{
expr: "5 4 21 2 *",
tm: time.Date(2021, 12, 19, 11, 21, 31, 0, locE2),
next: time.Date(2022, 2, 21, 4, 5, 0, 0, locE2),
},
{
expr: "55 16 * 12 0",
tm: time.Date(2021, 12, 21, 11, 21, 31, 0, locW2),
next: time.Date(2021, 12, 26, 16, 55, 0, 0, locW2),
},
{
expr: "12 8,16,19 * * *",
tm: time.Date(2021, 12, 21, 2, 21, 31, 0, locW2),
next: time.Date(2021, 12, 21, 8, 12, 0, 0, locW2),
},
{
expr: "12 8,16,19 * * *",
tm: time.Date(2021, 12, 21, 9, 21, 31, 0, locW2),
next: time.Date(2021, 12, 21, 16, 12, 0, 0, locW2),
},
{
expr: "12 8,16,19 * * *",
tm: time.Date(2021, 12, 21, 19, 21, 31, 0, locW2),
next: time.Date(2021, 12, 22, 8, 12, 0, 0, locW2),
},
{
expr: "12 8,16,19 * * *",
tm: time.Date(2021, 12, 21, 16, 12, 0, 0, time.Local),
next: time.Date(2021, 12, 21, 19, 12, 0, 0, time.Local),
},
{
expr: "* * 29 2 *",
tm: time.Date(2021, 12, 21, 16, 12, 0, 0, time.Local),
next: time.Date(2024, 2, 29, 0, 0, 0, 0, time.Local),
},
{
expr: "* * 30 2 *",
tm: time.Date(2021, 12, 21, 16, 12, 0, 0, time.Local),
next: time.Time{},
},
}
for _, c := range cases {
p, err := CreateSchedEventPolicy(SchedEventCron, c.expr)
if c.err {
require.Nil(t, p)
require.ErrorContains(t, err, fmt.Sprintf("invalid cron expr '%s'", c.expr))
continue
}
require.NoError(t, err)
require.NotNil(t, p)
require.IsType(t, &CronPolicy{}, p)
watermark := c.tm
next, ok := p.NextEventTime(watermark)
require.Equal(t, c.next, next)
require.Equal(t, !next.IsZero(), ok)
}
}