mirror of
https://github.com/crazybber/awesome-patterns.git
synced 2024-11-22 12:46:03 +03:00
Add pub/sub messaging pattern
This commit is contained in:
parent
8253fa5702
commit
8b0943a7b2
@ -80,7 +80,7 @@ __Messaging Patterns__:
|
|||||||
| [Fan-In](fan/fan_in.go) | Funnels tasks to a work sink (e.g. server) |
|
| [Fan-In](fan/fan_in.go) | Funnels tasks to a work sink (e.g. server) |
|
||||||
| [Fan-Out](fan/fan_out.go) | Distributes tasks amongs workers |
|
| [Fan-Out](fan/fan_out.go) | Distributes tasks amongs workers |
|
||||||
| [Futures & Promises](futures_promises.go) | Acts as a place-holder of a result that is initally unknown for synchronization purposes |
|
| [Futures & Promises](futures_promises.go) | Acts as a place-holder of a result that is initally unknown for synchronization purposes |
|
||||||
| [Publish/Subscribe](publish_subscribe.go) | Passes information to a collection of recipients who subscribed to a topic |
|
| [Publish/Subscribe](messaging/publish_subscribe.md) | Passes information to a collection of recipients who subscribed to a topic |
|
||||||
| [Push & Pull](push_pull.go) | Distributes messages to multiple workers, arranged in a pipeline |
|
| [Push & Pull](push_pull.go) | Distributes messages to multiple workers, arranged in a pipeline |
|
||||||
|
|
||||||
__Stability Patterns__:
|
__Stability Patterns__:
|
||||||
|
80
messaging/publish_subscribe.md
Normal file
80
messaging/publish_subscribe.md
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
Publish & Subscribe Messaging Pattern
|
||||||
|
============
|
||||||
|
Publish-Subscribe is a messaging pattern used to communicate messages between
|
||||||
|
different components without these components knowing anything about each other's identity.
|
||||||
|
|
||||||
|
It is similar to the Observer behavioral design pattern.
|
||||||
|
The fundamental design principals of both Observer and Publish-Subscribe is the decoupling of
|
||||||
|
those interested in being informed about `Event Messages` from the informer (Observers or Publishers).
|
||||||
|
Meaning that you don't have to program the messages to be sent directly to specific receivers.
|
||||||
|
|
||||||
|
To accomplish this, an intermediary, called a "message broker" or "event bus",
|
||||||
|
receives published messages, and then routes them on to subscribers.
|
||||||
|
|
||||||
|
|
||||||
|
There are three components **messages**, **topics**, **users**.
|
||||||
|
|
||||||
|
```go
|
||||||
|
type Message struct {
|
||||||
|
// Contents
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type Subscription struct {
|
||||||
|
ch chan<- Message
|
||||||
|
|
||||||
|
Inbox chan Message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Subscription) Publish(msg Message) error {
|
||||||
|
if _, ok := ch; !ok {
|
||||||
|
return errors.New("Topic has been closed")
|
||||||
|
}
|
||||||
|
|
||||||
|
ch <- msg
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```go
|
||||||
|
type Topic struct {
|
||||||
|
Subscribers []Session
|
||||||
|
MessageHistory []Message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Topic) Subscribe(uid uint64) (Subscription, error) {
|
||||||
|
// Get session and create one if it's the first
|
||||||
|
|
||||||
|
// Add session to the Topic & MessageHistory
|
||||||
|
|
||||||
|
// Create a subscription
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Topic) Unsubscribe(Subscription) error {
|
||||||
|
// Implementation
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Topic) Delete() error {
|
||||||
|
// Implementation
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```go
|
||||||
|
type User struct {
|
||||||
|
ID uint64
|
||||||
|
Name string
|
||||||
|
}
|
||||||
|
|
||||||
|
type Session struct {
|
||||||
|
User User
|
||||||
|
Timestamp time.Time
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Improvements
|
||||||
|
============
|
||||||
|
Events can be published in a parallel fashion by utilizing stackless goroutines.
|
||||||
|
|
||||||
|
Performance can be improved by dealing with straggler subscribers
|
||||||
|
by using a buffered inbox and you stop sending events once the inbox is full.
|
@ -1,62 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrTopicClosed = errors.New("Topic has been closed")
|
|
||||||
)
|
|
||||||
|
|
||||||
// Message
|
|
||||||
type Message string
|
|
||||||
|
|
||||||
// Topic
|
|
||||||
type Topic struct {
|
|
||||||
Subscribers []Authentication
|
|
||||||
MessageHistory []struct {
|
|
||||||
Author string
|
|
||||||
Message Message
|
|
||||||
Timestamp time.Time
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Subscribe
|
|
||||||
func (t *Topic) Subscribe(Authentication) (Subscription, error) {
|
|
||||||
// Implementation
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unsubscribe
|
|
||||||
func (t *Topic) Unsubscribe(Subscription) error {
|
|
||||||
// Implementation
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete
|
|
||||||
func (t *Topic) Delete() error {
|
|
||||||
// Implementation
|
|
||||||
}
|
|
||||||
|
|
||||||
type Subscription struct {
|
|
||||||
ch chan<- Message
|
|
||||||
|
|
||||||
Inbox chan Message
|
|
||||||
}
|
|
||||||
|
|
||||||
// Publish
|
|
||||||
func (s *Subscription) Publish(msg Message) error {
|
|
||||||
if _, ok := ch; !ok {
|
|
||||||
return ErrTopicClosed
|
|
||||||
}
|
|
||||||
|
|
||||||
ch <- msg
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Authentication
|
|
||||||
type Authentication struct {
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user