Skip to content

Commit 6a091b0

Browse files
committed
fix: remove duplicate items with the cursor paginator in date mode
We get duplicate items with the cursor paginator in date mode if the timestamp of the last item has a non zero fractional part. The solution is to increment the next cursor by one.
1 parent 6b21a40 commit 6a091b0

File tree

2 files changed

+78
-2
lines changed

2 files changed

+78
-2
lines changed

paginator.go

+14-2
Original file line numberDiff line numberDiff line change
@@ -146,9 +146,21 @@ func (p *CursorPaginator) MakeNextURI() null.String {
146146
return null.NewString("", false)
147147
}
148148

149-
// convert to timestamp if cusror mode is Date
149+
// convert to timestamp
150150
if p.Options.CursorOptions.Mode == DateModeCursor {
151-
nextCursor = nextCursor.(time.Time).Unix()
151+
timestamp := nextCursor.(time.Time).Unix()
152+
if !p.Options.CursorOptions.Reverse {
153+
// The next cursor must be the timestamp of the last item incremented by one.
154+
// Otherwise, we would get duplicates as the last item of the current page would be included
155+
// in the next page.
156+
// One problem with this approach is that we may lose items if two items are within the same
157+
// second. If items A and B are within the same second and A is the last item in the page,
158+
// the next cursor will be the timestamp of A incremented by one, and the next page won't
159+
// contain B.
160+
// TODO: The (non-backward-compatible) solution is to increase the precision of timestamps
161+
timestamp++
162+
}
163+
nextCursor = timestamp
152164
}
153165

154166
return null.StringFrom(GenerateCursorURI(p.Limit, nextCursor, p.Options))

paginator_test.go

+64
Original file line numberDiff line numberDiff line change
@@ -25,3 +25,67 @@ func TestNewCursorPaginator(t *testing.T) {
2525
is.True(ok)
2626
is.True(got.After(now), "%v should be after %v", got, now)
2727
}
28+
29+
func TestCursorPaginator_Next_DateMode(t *testing.T) {
30+
is := assert.New(t)
31+
32+
var u *User
33+
is.NoError(db.DropTableIfExists(u).Error)
34+
is.NoError(db.CreateTable(u).Error)
35+
ts := time.Unix(0, 1548252003033986000) // non zero fractional part
36+
is.NoError(db.Create(&User{DateCreation: ts}).Error)
37+
is.NoError(db.Create(&User{DateCreation: ts.Add(time.Second)}).Error)
38+
39+
var users []User
40+
s, err := NewGORMStore(db.Model(u).Order("date_creation"), &users)
41+
is.NoError(err)
42+
43+
v := url.Values{"limit": []string{"1"}, "since": []string{"1"}}
44+
opts := NewOptions()
45+
opts.CursorOptions.Mode = DateModeCursor
46+
opts.CursorOptions.DBName = "date_creation"
47+
opts.CursorOptions.StructName = "DateCreation"
48+
p, err := NewCursorPaginator(s, &http.Request{URL: &url.URL{RawQuery: v.Encode()}}, opts)
49+
is.NoError(err)
50+
is.NoError(p.Page())
51+
is.Len(users, 1)
52+
is.Equal(1, users[0].ID)
53+
54+
next := p.MakeNextURI()
55+
is.True(next.Valid)
56+
// the next uri cursor is the timestamp of the last element incremented by one
57+
is.Contains(next.String, strconv.FormatInt(users[0].DateCreation.Unix()+1, 10))
58+
}
59+
60+
func TestCursorPaginator_Next_DateMode_Reverse(t *testing.T) {
61+
is := assert.New(t)
62+
63+
var u *User
64+
is.NoError(db.DropTableIfExists(u).Error)
65+
is.NoError(db.CreateTable(u).Error)
66+
ts := time.Unix(0, 1548252003033986000) // non zero fractional part
67+
is.NoError(db.Create(&User{DateCreation: ts}).Error)
68+
is.NoError(db.Create(&User{DateCreation: ts.Add(time.Second)}).Error)
69+
70+
var users []User
71+
s, err := NewGORMStore(db.Model(u).Order("date_creation desc"), &users)
72+
is.NoError(err)
73+
74+
since := strconv.FormatInt(time.Now().Unix(), 10)
75+
v := url.Values{"limit": []string{"1"}, "since": []string{since}}
76+
opts := NewOptions()
77+
opts.CursorOptions.Mode = DateModeCursor
78+
opts.CursorOptions.DBName = "date_creation"
79+
opts.CursorOptions.StructName = "DateCreation"
80+
opts.CursorOptions.Reverse = true
81+
p, err := NewCursorPaginator(s, &http.Request{URL: &url.URL{RawQuery: v.Encode()}}, opts)
82+
is.NoError(err)
83+
is.NoError(p.Page())
84+
is.Len(users, 1)
85+
is.Equal(2, users[0].ID)
86+
87+
next := p.MakeNextURI()
88+
is.True(next.Valid)
89+
// the next uri cursor is the timestamp of the last element
90+
is.Contains(next.String, strconv.FormatInt(users[0].DateCreation.Unix(), 10))
91+
}

0 commit comments

Comments
 (0)