Subscribe to more than one topic
This commit is contained in:
		
							parent
							
								
									a481f4c448
								
							
						
					
					
						commit
						52136030be
					
				
					 1 changed files with 50 additions and 27 deletions
				
			
		|  | @ -78,9 +78,9 @@ const ( | |||
| 
 | ||||
| var ( | ||||
| 	topicRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}$`) // Regex must match JS & Android app! | ||||
| 	jsonRegex  = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}/json$`) | ||||
| 	sseRegex   = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}/sse$`) | ||||
| 	rawRegex   = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}/raw$`) | ||||
| 	jsonRegex  = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/json$`) | ||||
| 	sseRegex   = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/sse$`) | ||||
| 	rawRegex   = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/raw$`) | ||||
| 
 | ||||
| 	staticRegex = regexp.MustCompile(`^/static/.+`) | ||||
| 
 | ||||
|  | @ -223,7 +223,7 @@ func (s *Server) handleStatic(w http.ResponseWriter, r *http.Request) error { | |||
| } | ||||
| 
 | ||||
| func (s *Server) handlePublish(w http.ResponseWriter, r *http.Request, v *visitor) error { | ||||
| 	t, err := s.topic(r.URL.Path[1:]) | ||||
| 	t, err := s.topicFromID(r.URL.Path[1:]) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | @ -289,7 +289,9 @@ func (s *Server) handleSubscribe(w http.ResponseWriter, r *http.Request, v *visi | |||
| 		return errHTTPTooManyRequests | ||||
| 	} | ||||
| 	defer v.RemoveSubscription() | ||||
| 	t, err := s.topic(strings.TrimSuffix(r.URL.Path[1:], "/"+format)) // Hack | ||||
| 	topicsStr := strings.TrimSuffix(r.URL.Path[1:], "/"+format) // Hack | ||||
| 	topicIDs := strings.Split(topicsStr, ",") | ||||
| 	topics, err := s.topicsFromIDs(topicIDs...) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | @ -314,14 +316,21 @@ func (s *Server) handleSubscribe(w http.ResponseWriter, r *http.Request, v *visi | |||
| 	w.Header().Set("Access-Control-Allow-Origin", "*")            // CORS, allow cross-origin requests | ||||
| 	w.Header().Set("Content-Type", contentType+"; charset=utf-8") // Android/Volley client needs charset! | ||||
| 	if poll { | ||||
| 		return s.sendOldMessages(t, since, sub) | ||||
| 		return s.sendOldMessages(topics, since, sub) | ||||
| 	} | ||||
| 	subscriberID := t.Subscribe(sub) | ||||
| 	defer t.Unsubscribe(subscriberID) | ||||
| 	if err := sub(newOpenMessage(t.id)); err != nil { // Send out open message | ||||
| 	subscriberIDs := make([]int, 0) | ||||
| 	for _, t := range topics { | ||||
| 		subscriberIDs = append(subscriberIDs, t.Subscribe(sub)) | ||||
| 	} | ||||
| 	defer func() { | ||||
| 		for i, subscriberID := range subscriberIDs { | ||||
| 			topics[i].Unsubscribe(subscriberID) // Order! | ||||
| 		} | ||||
| 	}() | ||||
| 	if err := sub(newOpenMessage(topicsStr)); err != nil { // Send out open message | ||||
| 		return err | ||||
| 	} | ||||
| 	if err := s.sendOldMessages(t, since, sub); err != nil { | ||||
| 	if err := s.sendOldMessages(topics, since, sub); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	for { | ||||
|  | @ -330,17 +339,18 @@ func (s *Server) handleSubscribe(w http.ResponseWriter, r *http.Request, v *visi | |||
| 			return nil | ||||
| 		case <-time.After(s.config.KeepaliveInterval): | ||||
| 			v.Keepalive() | ||||
| 			if err := sub(newKeepaliveMessage(t.id)); err != nil { // Send keepalive message | ||||
| 			if err := sub(newKeepaliveMessage(topicsStr)); err != nil { // Send keepalive message | ||||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *Server) sendOldMessages(t *topic, since sinceTime, sub subscriber) error { | ||||
| func (s *Server) sendOldMessages(topics []*topic, since sinceTime, sub subscriber) error { | ||||
| 	if since.IsNone() { | ||||
| 		return nil | ||||
| 	} | ||||
| 	for _, t := range topics { | ||||
| 		messages, err := s.cache.Messages(t.id, since) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
|  | @ -350,6 +360,7 @@ func (s *Server) sendOldMessages(t *topic, since sinceTime, sub subscriber) erro | |||
| 				return err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
|  | @ -382,9 +393,19 @@ func (s *Server) handleOptions(w http.ResponseWriter, r *http.Request) error { | |||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (s *Server) topic(id string) (*topic, error) { | ||||
| func (s *Server) topicFromID(id string) (*topic, error) { | ||||
| 	topics, err := s.topicsFromIDs(id) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return topics[0], nil | ||||
| } | ||||
| 
 | ||||
| func (s *Server) topicsFromIDs(ids... string) ([]*topic, error) { | ||||
| 	s.mu.Lock() | ||||
| 	defer s.mu.Unlock() | ||||
| 	topics := make([]*topic, 0) | ||||
| 	for  _, id := range ids { | ||||
| 		if _, ok := s.topics[id]; !ok { | ||||
| 			if len(s.topics) >= s.config.GlobalTopicLimit { | ||||
| 				return nil, errHTTPTooManyRequests | ||||
|  | @ -394,7 +415,9 @@ func (s *Server) topic(id string) (*topic, error) { | |||
| 				s.topics[id].Subscribe(s.firebase) | ||||
| 			} | ||||
| 		} | ||||
| 	return s.topics[id], nil | ||||
| 		topics = append(topics, s.topics[id]) | ||||
| 	} | ||||
| 	return topics, nil | ||||
| } | ||||
| 
 | ||||
| func (s *Server) updateStatsAndExpire() { | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue