| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 | package httpexpectimport (	"reflect")// Match provides methods to inspect attached regexp match results.type Match struct {	chain      chain	submatches []string	names      map[string]int}// NewMatch returns a new Match object given a reporter used to report// failures and submatches to be inspected.//// reporter should not be nil. submatches and names may be nil.//// Example://   s := "http://example.com/users/john"//   r := regexp.MustCompile(`http://(?P<host>.+)/users/(?P<user>.+)`)//   m := NewMatch(reporter, r.FindStringSubmatch(s), r.SubexpNames())////   m.NotEmpty()//   m.Length().Equal(3)////   m.Index(0).Equal("http://example.com/users/john")//   m.Index(1).Equal("example.com")//   m.Index(2).Equal("john")////   m.Name("host").Equal("example.com")//   m.Name("user").Equal("john")func NewMatch(reporter Reporter, submatches []string, names []string) *Match {	return makeMatch(makeChain(reporter), submatches, names)}func makeMatch(chain chain, submatches []string, names []string) *Match {	if submatches == nil {		submatches = []string{}	}	namemap := map[string]int{}	for n, name := range names {		if name != "" {			namemap[name] = n		}	}	return &Match{chain, submatches, namemap}}// Raw returns underlying submatches attached to Match.// This is the value originally passed to NewMatch.//// Example://  m := NewMatch(t, submatches, names)//  assert.Equal(t, submatches, m.Raw())func (m *Match) Raw() []string {	return m.submatches}// Length returns a new Number object that may be used to inspect// number of submatches.//// Example://  m := NewMatch(t, submatches, names)//  m.Length().Equal(len(submatches))func (m *Match) Length() *Number {	return &Number{m.chain, float64(len(m.submatches))}}// Index returns a new String object that may be used to inspect submatch// with given index.//// Note that submatch with index 0 contains the whole match. If index is out// of bounds, Index reports failure and returns empty (but non-nil) value.//// Example://   s := "http://example.com/users/john"////   r := regexp.MustCompile(`http://(.+)/users/(.+)`)//   m := NewMatch(t, r.FindStringSubmatch(s), nil)////   m.Index(0).Equal("http://example.com/users/john")//   m.Index(1).Equal("example.com")//   m.Index(2).Equal("john")func (m *Match) Index(index int) *String {	if index < 0 || index >= len(m.submatches) {		m.chain.fail(			"\nsubmatch index out of bounds:\n  index %d\n\n  bounds [%d; %d)",			index,			0,			len(m.submatches))		return &String{m.chain, ""}	}	return &String{m.chain, m.submatches[index]}}// Name returns a new String object that may be used to inspect submatch// with given name.//// If there is no submatch with given name, Name reports failure and returns// empty (but non-nil) value.//// Example://   s := "http://example.com/users/john"////   r := regexp.MustCompile(`http://(?P<host>.+)/users/(?P<user>.+)`)//   m := NewMatch(t, r.FindStringSubmatch(s), r.SubexpNames())////   m.Name("host").Equal("example.com")//   m.Name("user").Equal("john")func (m *Match) Name(name string) *String {	index, ok := m.names[name]	if !ok {		m.chain.fail(			"\nsubmatch name not found:\n %q\n\navailable names:\n%s",			name,			dumpValue(m.names))		return &String{m.chain, ""}	}	return m.Index(index)}// Empty succeeds if submatches array is empty.//// Example://  m := NewMatch(t, submatches, names)//  m.Empty()func (m *Match) Empty() *Match {	if len(m.submatches) != 0 {		m.chain.fail("\nexpected zero submatches, but got:\n  %s",			dumpValue(m.submatches))	}	return m}// NotEmpty succeeds if submatches array is non-empty.//// Example://  m := NewMatch(t, submatches, names)//  m.NotEmpty()func (m *Match) NotEmpty() *Match {	if len(m.submatches) == 0 {		m.chain.fail("expected non-zero submatches")	}	return m}// Values succeeds if submatches array, starting from index 1, is equal to// given array.//// Note that submatch with index 0 contains the whole match and is not// included into this check.//// Example://   s := "http://example.com/users/john"//   r := regexp.MustCompile(`http://(.+)/users/(.+)`)//   m := NewMatch(t, r.FindStringSubmatch(s), nil)//   m.Values("example.com", "john")func (m *Match) Values(values ...string) *Match {	if values == nil {		values = []string{}	}	if !reflect.DeepEqual(values, m.getValues()) {		m.chain.fail("\nexpected submatches equal to:\n%s\n\nbut got:\n%s",			dumpValue(values),			dumpValue(m.getValues()))	}	return m}// NotValues succeeds if submatches array, starting from index 1, is not// equal to given array.//// Note that submatch with index 0 contains the whole match and is not// included into this check.//// Example://   s := "http://example.com/users/john"//   r := regexp.MustCompile(`http://(.+)/users/(.+)`)//   m := NewMatch(t, r.FindStringSubmatch(s), nil)//   m.NotValues("example.com", "bob")func (m *Match) NotValues(values ...string) *Match {	if values == nil {		values = []string{}	}	if reflect.DeepEqual(values, m.getValues()) {		m.chain.fail("\nexpected submatches not equal to:\n%s",			dumpValue(values))	}	return m}func (m *Match) getValues() []string {	if len(m.submatches) > 1 {		return m.submatches[1:]	}	return []string{}}
 |