| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of go-ethereum. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with go-ethereum.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-14 13:24:43 +03:00
										 |  |  | // Contains the message filter for fine grained subscriptions. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-12 22:23:42 +01:00
										 |  |  | package whisper | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 18:31:08 +03:00
										 |  |  | import ( | 
					
						
							|  |  |  | 	"crypto/ecdsa" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/event/filter" | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2014-12-12 22:23:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-14 13:24:43 +03:00
										 |  |  | // Filter is used to subscribe to specific types of whisper messages. | 
					
						
							| 
									
										
										
										
											2014-12-12 22:23:42 +01:00
										 |  |  | type Filter struct { | 
					
						
							| 
									
										
										
										
											2015-04-21 18:31:08 +03:00
										 |  |  | 	To     *ecdsa.PublicKey   // Recipient of the message | 
					
						
							|  |  |  | 	From   *ecdsa.PublicKey   // Sender of the message | 
					
						
							|  |  |  | 	Topics [][]Topic          // Topics to filter messages with | 
					
						
							|  |  |  | 	Fn     func(msg *Message) // Handler in case of a match | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-22 12:50:48 +03:00
										 |  |  | // NewFilterTopics creates a 2D topic array used by whisper.Filter from binary | 
					
						
							|  |  |  | // data elements. | 
					
						
							|  |  |  | func NewFilterTopics(data ...[][]byte) [][]Topic { | 
					
						
							|  |  |  | 	filter := make([][]Topic, len(data)) | 
					
						
							|  |  |  | 	for i, condition := range data { | 
					
						
							|  |  |  | 		// Handle the special case of condition == [[]byte{}] | 
					
						
							|  |  |  | 		if len(condition) == 1 && len(condition[0]) == 0 { | 
					
						
							|  |  |  | 			filter[i] = []Topic{} | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Otherwise flatten normally | 
					
						
							|  |  |  | 		filter[i] = NewTopics(condition...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return filter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewFilterTopicsFlat creates a 2D topic array used by whisper.Filter from flat | 
					
						
							|  |  |  | // binary data elements. | 
					
						
							|  |  |  | func NewFilterTopicsFlat(data ...[]byte) [][]Topic { | 
					
						
							|  |  |  | 	filter := make([][]Topic, len(data)) | 
					
						
							|  |  |  | 	for i, element := range data { | 
					
						
							|  |  |  | 		// Only add non-wildcard topics | 
					
						
							|  |  |  | 		filter[i] = make([]Topic, 0, 1) | 
					
						
							|  |  |  | 		if len(element) > 0 { | 
					
						
							|  |  |  | 			filter[i] = append(filter[i], NewTopic(element)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return filter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewFilterTopicsFromStrings creates a 2D topic array used by whisper.Filter | 
					
						
							|  |  |  | // from textual data elements. | 
					
						
							|  |  |  | func NewFilterTopicsFromStrings(data ...[]string) [][]Topic { | 
					
						
							|  |  |  | 	filter := make([][]Topic, len(data)) | 
					
						
							|  |  |  | 	for i, condition := range data { | 
					
						
							|  |  |  | 		// Handle the special case of condition == [""] | 
					
						
							|  |  |  | 		if len(condition) == 1 && condition[0] == "" { | 
					
						
							|  |  |  | 			filter[i] = []Topic{} | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Otherwise flatten normally | 
					
						
							|  |  |  | 		filter[i] = NewTopicsFromStrings(condition...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return filter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewFilterTopicsFromStringsFlat creates a 2D topic array used by whisper.Filter from flat | 
					
						
							|  |  |  | // binary data elements. | 
					
						
							|  |  |  | func NewFilterTopicsFromStringsFlat(data ...string) [][]Topic { | 
					
						
							|  |  |  | 	filter := make([][]Topic, len(data)) | 
					
						
							|  |  |  | 	for i, element := range data { | 
					
						
							|  |  |  | 		// Only add non-wildcard topics | 
					
						
							|  |  |  | 		filter[i] = make([]Topic, 0, 1) | 
					
						
							|  |  |  | 		if element != "" { | 
					
						
							|  |  |  | 			filter[i] = append(filter[i], NewTopicFromString(element)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return filter | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 18:31:08 +03:00
										 |  |  | // filterer is the internal, fully initialized filter ready to match inbound | 
					
						
							|  |  |  | // messages to a variety of criteria. | 
					
						
							|  |  |  | type filterer struct { | 
					
						
							|  |  |  | 	to      string                 // Recipient of the message | 
					
						
							|  |  |  | 	from    string                 // Sender of the message | 
					
						
							|  |  |  | 	matcher *topicMatcher          // Topics to filter messages with | 
					
						
							|  |  |  | 	fn      func(data interface{}) // Handler in case of a match | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Compare checks if the specified filter matches the current one. | 
					
						
							|  |  |  | func (self filterer) Compare(f filter.Filter) bool { | 
					
						
							|  |  |  | 	filter := f.(filterer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Check the message sender and recipient | 
					
						
							|  |  |  | 	if len(self.to) > 0 && self.to != filter.to { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(self.from) > 0 && self.from != filter.from { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Check the topic filtering | 
					
						
							|  |  |  | 	topics := make([]Topic, len(filter.matcher.conditions)) | 
					
						
							|  |  |  | 	for i, group := range filter.matcher.conditions { | 
					
						
							|  |  |  | 		// Message should contain a single topic entry, extract | 
					
						
							|  |  |  | 		for topics[i], _ = range group { | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !self.matcher.Matches(topics) { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return true | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Trigger is called when a filter successfully matches an inbound message. | 
					
						
							|  |  |  | func (self filterer) Trigger(data interface{}) { | 
					
						
							|  |  |  | 	self.fn(data) | 
					
						
							| 
									
										
										
										
											2014-12-12 22:23:42 +01:00
										 |  |  | } |