mirror of
https://github.com/mjl-/mox.git
synced 2025-01-28 15:25:55 +03:00
14aa85482e
in some cases, they were interpreted as meaning "the first sequence/uid", but it should always be "the last sequence/uid", just like patterns of the form "123:*". this wrong interpretation was used in the "fetch" command when combined with "changedsince", and in the search command for some parameters, and during expunge with an explicit uid range. the form "*" and "*:123" aren't very common.
114 lines
3.5 KiB
Go
114 lines
3.5 KiB
Go
package imapserver
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/mjl-/mox/store"
|
|
)
|
|
|
|
func TestNumSetContains(t *testing.T) {
|
|
num := func(v uint32) *setNumber {
|
|
return &setNumber{v, false}
|
|
}
|
|
star := &setNumber{star: true}
|
|
|
|
check := func(v bool) {
|
|
t.Helper()
|
|
if !v {
|
|
t.Fatalf("bad")
|
|
}
|
|
}
|
|
|
|
ss0 := numSet{true, nil} // "$"
|
|
check(ss0.containsSeq(1, []store.UID{2}, []store.UID{2}))
|
|
check(!ss0.containsSeq(1, []store.UID{2}, []store.UID{}))
|
|
|
|
check(ss0.containsUID(1, []store.UID{1}, []store.UID{1}))
|
|
check(ss0.containsUID(2, []store.UID{1, 2, 3}, []store.UID{2}))
|
|
check(!ss0.containsUID(2, []store.UID{1, 2, 3}, []store.UID{}))
|
|
check(!ss0.containsUID(2, []store.UID{}, []store.UID{2}))
|
|
|
|
ss1 := numSet{false, []numRange{{*num(1), nil}}} // Single number 1.
|
|
check(ss1.containsSeq(1, []store.UID{2}, nil))
|
|
check(!ss1.containsSeq(2, []store.UID{1, 2}, nil))
|
|
|
|
check(ss1.containsUID(1, []store.UID{1}, nil))
|
|
check(ss1.containsSeq(1, []store.UID{2}, nil))
|
|
check(!ss1.containsSeq(2, []store.UID{1, 2}, nil))
|
|
|
|
// 2:*
|
|
ss2 := numSet{false, []numRange{{*num(2), star}}}
|
|
check(ss2.containsSeq(1, []store.UID{2}, nil))
|
|
check(!ss2.containsSeq(2, []store.UID{2}, nil))
|
|
check(ss2.containsSeq(2, []store.UID{4, 5}, nil))
|
|
check(ss2.containsSeq(3, []store.UID{4, 5, 6}, nil))
|
|
check(!ss2.containsSeq(4, []store.UID{4, 5, 6}, nil))
|
|
|
|
check(ss2.containsUID(2, []store.UID{2}, nil))
|
|
check(!ss2.containsUID(1, []store.UID{1, 2, 3}, nil))
|
|
check(ss2.containsUID(3, []store.UID{1, 2, 3}, nil))
|
|
check(!ss2.containsUID(2, []store.UID{4, 5}, nil))
|
|
check(!ss2.containsUID(2, []store.UID{1}, nil))
|
|
|
|
check(ss2.containsUID(2, []store.UID{2, 6}, nil))
|
|
check(ss2.containsUID(6, []store.UID{2, 6}, nil))
|
|
|
|
// *:2, same as 2:*
|
|
ss3 := numSet{false, []numRange{{*star, num(2)}}}
|
|
check(ss3.containsSeq(1, []store.UID{2}, nil))
|
|
check(!ss3.containsSeq(2, []store.UID{2}, nil))
|
|
check(ss3.containsSeq(2, []store.UID{4, 5}, nil))
|
|
check(ss3.containsSeq(3, []store.UID{4, 5, 6}, nil))
|
|
check(!ss3.containsSeq(4, []store.UID{4, 5, 6}, nil))
|
|
|
|
check(ss3.containsUID(2, []store.UID{2}, nil))
|
|
check(!ss3.containsUID(1, []store.UID{1, 2, 3}, nil))
|
|
check(ss3.containsUID(3, []store.UID{1, 2, 3}, nil))
|
|
check(!ss3.containsUID(2, []store.UID{4, 5}, nil))
|
|
check(!ss3.containsUID(2, []store.UID{1}, nil))
|
|
|
|
check(ss3.containsUID(2, []store.UID{2, 6}, nil))
|
|
check(ss3.containsUID(6, []store.UID{2, 6}, nil))
|
|
}
|
|
|
|
func TestNumSetInterpret(t *testing.T) {
|
|
parseNumSet := func(s string) numSet {
|
|
p := parser{upper: s}
|
|
return p.xnumSet0(true, false)
|
|
}
|
|
|
|
checkEqual := func(uids []store.UID, a, s string) {
|
|
t.Helper()
|
|
n := parseNumSet(a).interpretStar(uids)
|
|
ns := n.String()
|
|
if ns != s {
|
|
t.Fatalf("%s != %s", ns, s)
|
|
}
|
|
}
|
|
|
|
checkEqual([]store.UID{}, "1:*", "")
|
|
checkEqual([]store.UID{1}, "1:*", "1")
|
|
checkEqual([]store.UID{1, 3}, "1:*", "1:3")
|
|
checkEqual([]store.UID{1, 3}, "4:*", "3")
|
|
checkEqual([]store.UID{1, 3}, "*:4", "3")
|
|
checkEqual([]store.UID{2, 3}, "*:4", "3")
|
|
checkEqual([]store.UID{2, 3}, "*:1", "1:3")
|
|
checkEqual([]store.UID{2, 3}, "1:*", "1:3")
|
|
checkEqual([]store.UID{1, 2, 3}, "1,2,3", "1,2,3")
|
|
checkEqual([]store.UID{}, "1,2,3", "")
|
|
checkEqual([]store.UID{}, "1:3", "")
|
|
checkEqual([]store.UID{}, "3:1", "")
|
|
|
|
iter := parseNumSet("1:3").interpretStar([]store.UID{}).newIter()
|
|
if _, ok := iter.Next(); ok {
|
|
t.Fatalf("expected immediate end for empty iter")
|
|
}
|
|
|
|
iter = parseNumSet("3:1").interpretStar([]store.UID{1, 2}).newIter()
|
|
v0, _ := iter.Next()
|
|
v1, _ := iter.Next()
|
|
_, ok := iter.Next()
|
|
if v0 != 1 || v1 != 2 || ok {
|
|
t.Fatalf("got %v %v %v, expected 1, 2, false", v0, v1, ok)
|
|
}
|
|
}
|