mirror of
https://github.com/1f349/go-webdav.git
synced 2025-01-21 23:06:23 +00:00
carddav: add filtering and matching helper functions
Updates emersion/hydroxide#159. Signed-off-by: Sebastien Binet <binet@cern.ch> Co-authored-by: Conrad Hoffmann <ch@bitfehler.net>
This commit is contained in:
parent
dc57b81662
commit
6d59672ed4
146
carddav/match.go
Normal file
146
carddav/match.go
Normal file
@ -0,0 +1,146 @@
|
||||
package carddav
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
"github.com/emersion/go-vcard"
|
||||
)
|
||||
|
||||
// Filter returns the filtered list of address objects matching the provided query.
|
||||
// A nil query will return the full list of address objects.
|
||||
func Filter(query *AddressBookQuery, aos []AddressObject) ([]AddressObject, error) {
|
||||
if query == nil {
|
||||
// FIXME: should we always return a copy of the provided slice?
|
||||
return aos, nil
|
||||
}
|
||||
|
||||
n := query.Limit
|
||||
if n <= 0 || n > len(aos) {
|
||||
n = len(aos)
|
||||
}
|
||||
out := make([]AddressObject, 0, n)
|
||||
for _, ao := range aos {
|
||||
ok, err := Match(query, &ao)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if !ok {
|
||||
continue
|
||||
}
|
||||
|
||||
// TODO properties are not currently filtered even if requested
|
||||
|
||||
out = append(out, ao)
|
||||
if len(out) >= n {
|
||||
break
|
||||
}
|
||||
}
|
||||
return out, nil
|
||||
}
|
||||
|
||||
// Match reports whether the provided AddressObject matches the query.
|
||||
func Match(query *AddressBookQuery, ao *AddressObject) (matched bool, err error) {
|
||||
if query == nil {
|
||||
return true, nil
|
||||
}
|
||||
|
||||
switch query.FilterTest {
|
||||
default:
|
||||
return false, fmt.Errorf("unknown query filter test %q", query.FilterTest)
|
||||
|
||||
case FilterAnyOf, "":
|
||||
for _, prop := range query.PropFilters {
|
||||
ok, err := matchPropFilter(prop, ao)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
if ok {
|
||||
return true, nil
|
||||
}
|
||||
}
|
||||
return false, nil
|
||||
|
||||
case FilterAllOf:
|
||||
for _, prop := range query.PropFilters {
|
||||
ok, err := matchPropFilter(prop, ao)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
if !ok {
|
||||
return false, nil
|
||||
}
|
||||
}
|
||||
return true, nil
|
||||
}
|
||||
}
|
||||
|
||||
func matchPropFilter(prop PropFilter, ao *AddressObject) (bool, error) {
|
||||
// TODO: this only matches first field, there could be multiple
|
||||
field := ao.Card.Get(prop.Name)
|
||||
if field == nil {
|
||||
return prop.IsNotDefined, nil
|
||||
} else if prop.IsNotDefined {
|
||||
return false, nil
|
||||
}
|
||||
|
||||
// TODO: handle carddav.PropFilter.Params.
|
||||
if len(prop.TextMatches) == 0 {
|
||||
return true, nil
|
||||
}
|
||||
|
||||
switch prop.Test {
|
||||
default:
|
||||
return false, fmt.Errorf("unknown property filter test %q", prop.Test)
|
||||
|
||||
case FilterAnyOf, "":
|
||||
for _, txt := range prop.TextMatches {
|
||||
ok, err := matchTextMatch(txt, field)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
if ok {
|
||||
return true, nil
|
||||
}
|
||||
}
|
||||
return false, nil
|
||||
|
||||
case FilterAllOf:
|
||||
for _, txt := range prop.TextMatches {
|
||||
ok, err := matchTextMatch(txt, field)
|
||||
if err != nil {
|
||||
return false, err
|
||||
}
|
||||
if !ok {
|
||||
return false, nil
|
||||
}
|
||||
}
|
||||
return true, nil
|
||||
}
|
||||
}
|
||||
|
||||
func matchTextMatch(txt TextMatch, field *vcard.Field) (bool, error) {
|
||||
// TODO: handle text-match collation attribute
|
||||
var ok bool
|
||||
switch txt.MatchType {
|
||||
default:
|
||||
return false, fmt.Errorf("unknown textmatch type %q", txt.MatchType)
|
||||
|
||||
case MatchEquals:
|
||||
ok = txt.Text == field.Value
|
||||
|
||||
case MatchContains, "":
|
||||
ok = strings.Contains(field.Value, txt.Text)
|
||||
|
||||
case MatchStartsWith:
|
||||
ok = strings.HasPrefix(field.Value, txt.Text)
|
||||
|
||||
case MatchEndsWith:
|
||||
ok = strings.HasSuffix(field.Value, txt.Text)
|
||||
}
|
||||
|
||||
if txt.NegateCondition {
|
||||
ok = !ok
|
||||
}
|
||||
return ok, nil
|
||||
}
|
617
carddav/match_test.go
Normal file
617
carddav/match_test.go
Normal file
@ -0,0 +1,617 @@
|
||||
package carddav
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"reflect"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/emersion/go-vcard"
|
||||
)
|
||||
|
||||
func TestFilter(t *testing.T) {
|
||||
newAO := func(str string) AddressObject {
|
||||
card, err := vcard.NewDecoder(strings.NewReader(str)).Decode()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
return AddressObject{
|
||||
Card: card,
|
||||
}
|
||||
}
|
||||
|
||||
alice := newAO(`BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
UID:urn:uuid:4fbe8971-0bc3-424c-9c26-36c3e1eff6b1
|
||||
FN;PID=1.1:Alice Gopher
|
||||
N:Gopher;Alice;;;
|
||||
EMAIL;PID=1.1:alice@example.com
|
||||
CLIENTPIDMAP:1;urn:uuid:53e374d9-337e-4727-8803-a1e9c14e0551
|
||||
END:VCARD`)
|
||||
|
||||
bob := newAO(`BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
UID:urn:uuid:4fbe8971-0bc3-424c-9c26-36c3e1eff6b2
|
||||
FN;PID=1.1:Bob Gopher
|
||||
N:Gopher;Bob;;;
|
||||
EMAIL;PID=1.1:bob@example.com
|
||||
CLIENTPIDMAP:1;urn:uuid:53e374d9-337e-4727-8803-a1e9c14e0552
|
||||
END:VCARD`)
|
||||
|
||||
carla := newAO(`BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
UID:urn:uuid:4fbe8971-0bc3-424c-9c26-36c3e1eff6b3
|
||||
FN;PID=1.1:Carla Gopher
|
||||
N:Gopher;Carla;;;
|
||||
EMAIL;PID=1.1:carla@example.com
|
||||
CLIENTPIDMAP:1;urn:uuid:53e374d9-337e-4727-8803-a1e9c14e0553
|
||||
END:VCARD`)
|
||||
|
||||
for _, tc := range []struct {
|
||||
name string
|
||||
query *AddressBookQuery
|
||||
addrs []AddressObject
|
||||
want []AddressObject
|
||||
err error
|
||||
}{
|
||||
{
|
||||
name: "nil-query",
|
||||
query: nil,
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice, bob, carla},
|
||||
},
|
||||
{
|
||||
name: "no-limit-query",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice, bob, carla},
|
||||
},
|
||||
{
|
||||
name: "limit-1-query",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
Limit: 1,
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice},
|
||||
},
|
||||
{
|
||||
name: "limit-4-query",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
Limit: 4,
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice, bob, carla},
|
||||
},
|
||||
{
|
||||
name: "email-match",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "carla"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{carla},
|
||||
},
|
||||
{
|
||||
name: "email-match-any",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{
|
||||
{Text: "carla@example"},
|
||||
{Text: "alice@example"},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice, carla},
|
||||
},
|
||||
{
|
||||
name: "email-match-all",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{
|
||||
{Text: ""},
|
||||
},
|
||||
}},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{alice, bob, carla},
|
||||
},
|
||||
{
|
||||
name: "email-no-match",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.org"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addrs: []AddressObject{alice, bob, carla},
|
||||
want: []AddressObject{},
|
||||
},
|
||||
} {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
got, err := Filter(tc.query, tc.addrs)
|
||||
switch {
|
||||
case err != nil && tc.err == nil:
|
||||
t.Fatalf("unexpected error: %+v", err)
|
||||
case err != nil && tc.err != nil:
|
||||
if got, want := err.Error(), tc.err.Error(); got != want {
|
||||
t.Fatalf("invalid error:\ngot= %q\nwant=%q", got, want)
|
||||
}
|
||||
case err == nil && tc.err != nil:
|
||||
t.Fatalf("expected an error:\ngot= %+v\nwant=%+v", err, tc.err)
|
||||
case err == nil && tc.err == nil:
|
||||
if got, want := got, tc.want; !reflect.DeepEqual(got, want) {
|
||||
t.Fatalf("invalid filter values:\ngot= %+v\nwant=%+v", got, want)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestMatch(t *testing.T) {
|
||||
newAO := func(str string) AddressObject {
|
||||
card, err := vcard.NewDecoder(strings.NewReader(str)).Decode()
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
return AddressObject{
|
||||
Card: card,
|
||||
}
|
||||
}
|
||||
|
||||
alice := newAO(`BEGIN:VCARD
|
||||
VERSION:4.0
|
||||
UID:urn:uuid:4fbe8971-0bc3-424c-9c26-36c3e1eff6b1
|
||||
FN;PID=1.1:Alice Gopher
|
||||
N:Gopher;Alice;;;
|
||||
EMAIL;PID=1.1:alice@example.com
|
||||
CLIENTPIDMAP:1;urn:uuid:53e374d9-337e-4727-8803-a1e9c14e0556
|
||||
END:VCARD`)
|
||||
|
||||
for _, tc := range []struct {
|
||||
name string
|
||||
query *AddressBookQuery
|
||||
addr AddressObject
|
||||
want bool
|
||||
err error
|
||||
}{
|
||||
{
|
||||
name: "nil-query",
|
||||
query: nil,
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-contains",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-equals-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "alice@example.com",
|
||||
MatchType: MatchEquals,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-equals-not",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "example.com",
|
||||
MatchType: MatchEquals,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "match-email-equals-ok-negate",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "bob@example.com",
|
||||
NegateCondition: true,
|
||||
MatchType: MatchEquals,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-starts-with-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "alice@",
|
||||
MatchType: MatchStartsWith,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-ends-with-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "com",
|
||||
MatchType: MatchEndsWith,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-email-ends-with-not",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: ".org",
|
||||
MatchType: MatchEndsWith,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "match-name-contains-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldName,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "Alice",
|
||||
MatchType: MatchContains,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-name-contains-all-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldName,
|
||||
Test: FilterAllOf,
|
||||
TextMatches: []TextMatch{
|
||||
{
|
||||
Text: "Alice",
|
||||
MatchType: MatchContains,
|
||||
},
|
||||
{
|
||||
Text: "Gopher",
|
||||
MatchType: MatchContains,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-name-contains-all-prop-not",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
FilterTest: FilterAllOf,
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldName,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "Alice",
|
||||
MatchType: MatchContains,
|
||||
}},
|
||||
},
|
||||
{
|
||||
Name: vcard.FieldName,
|
||||
TextMatches: []TextMatch{{
|
||||
Text: "GopherXXX",
|
||||
MatchType: MatchContains,
|
||||
}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "match-name-contains-all-text-match-not",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldName,
|
||||
Test: FilterAllOf,
|
||||
TextMatches: []TextMatch{
|
||||
{
|
||||
Text: "Alice",
|
||||
MatchType: MatchContains,
|
||||
},
|
||||
{
|
||||
Text: "GopherXXX",
|
||||
MatchType: MatchContains,
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: false,
|
||||
},
|
||||
{
|
||||
name: "missing-prop-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
"XXX-not-THERE", // but AllProp is false.
|
||||
},
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "match-all-prop-ok",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
AllProp: true,
|
||||
},
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
want: true,
|
||||
},
|
||||
{
|
||||
name: "invalid-query-filter",
|
||||
query: &AddressBookQuery{
|
||||
DataRequest: AddressDataRequest{
|
||||
Props: []string{
|
||||
vcard.FieldFormattedName,
|
||||
vcard.FieldEmail,
|
||||
vcard.FieldUID,
|
||||
},
|
||||
},
|
||||
FilterTest: "XXX-invalid-filter",
|
||||
PropFilters: []PropFilter{
|
||||
{
|
||||
Name: vcard.FieldEmail,
|
||||
TextMatches: []TextMatch{{Text: "example.com"}},
|
||||
},
|
||||
},
|
||||
},
|
||||
addr: alice,
|
||||
err: fmt.Errorf("unknown query filter test \"XXX-invalid-filter\""),
|
||||
},
|
||||
} {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
got, err := Match(tc.query, &tc.addr)
|
||||
switch {
|
||||
case err != nil && tc.err == nil:
|
||||
t.Fatalf("unexpected error: %+v", err)
|
||||
case err != nil && tc.err != nil:
|
||||
if got, want := err.Error(), tc.err.Error(); got != want {
|
||||
t.Fatalf("invalid error:\ngot= %q\nwant=%q", got, want)
|
||||
}
|
||||
case err == nil && tc.err != nil:
|
||||
t.Fatalf("expected an error:\ngot= %+v\nwant=%+v", err, tc.err)
|
||||
case err == nil && tc.err == nil:
|
||||
if got, want := got, tc.want; got != want {
|
||||
t.Fatalf("invalid match value: got=%v, want=%v", got, want)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user