Initial Bus implementation
This commit is contained in:
50
bus.go
Normal file
50
bus.go
Normal file
@@ -0,0 +1,50 @@
|
||||
package main
|
||||
|
||||
import "sync"
|
||||
|
||||
type Bus struct {
|
||||
mu sync.Mutex
|
||||
chans map[string][]chan Object
|
||||
}
|
||||
|
||||
func NewBus() *Bus {
|
||||
return &Bus{
|
||||
chans: map[string][]chan Object{},
|
||||
}
|
||||
}
|
||||
|
||||
func (b *Bus) Announce(obj Object) {
|
||||
key := ObjectKey(obj)
|
||||
|
||||
b.mu.Lock()
|
||||
defer b.mu.Unlock()
|
||||
|
||||
chans := b.chans[key]
|
||||
newChans := []chan Object{}
|
||||
|
||||
for _, ch := range chans {
|
||||
select {
|
||||
case ch <- obj:
|
||||
newChans = append(newChans, ch)
|
||||
default:
|
||||
close(ch)
|
||||
}
|
||||
}
|
||||
|
||||
if len(chans) != len(newChans) {
|
||||
b.chans[key] = newChans
|
||||
}
|
||||
}
|
||||
|
||||
func (b *Bus) Subscribe(obj Object) chan Object {
|
||||
key := ObjectKey(obj)
|
||||
|
||||
b.mu.Lock()
|
||||
defer b.mu.Unlock()
|
||||
|
||||
ch := make(chan Object, 100)
|
||||
|
||||
b.chans[key] = append(b.chans[key], ch)
|
||||
|
||||
return ch
|
||||
}
|
||||
25
main.go
25
main.go
@@ -5,20 +5,23 @@ import "fmt"
|
||||
import "github.com/google/uuid"
|
||||
|
||||
func main() {
|
||||
store := NewStore("foo")
|
||||
store := NewStore("foo")
|
||||
|
||||
out := &Template{
|
||||
Id: uuid.NewString(),
|
||||
Test: "round trip",
|
||||
}
|
||||
out := &Template{
|
||||
Id: uuid.NewString(),
|
||||
Test: "round trip",
|
||||
}
|
||||
|
||||
store.Write(out)
|
||||
store.Write(out)
|
||||
|
||||
in := &Template{
|
||||
Id: out.Id,
|
||||
}
|
||||
in := &Template{
|
||||
Id: out.Id,
|
||||
}
|
||||
|
||||
store.Read(in)
|
||||
store.Read(in)
|
||||
|
||||
fmt.Printf("%+v\n", in)
|
||||
fmt.Printf("%+v\n", in)
|
||||
|
||||
bus := NewBus()
|
||||
bus.Announce(in)
|
||||
}
|
||||
|
||||
17
object.go
17
object.go
@@ -1,18 +1,17 @@
|
||||
package main
|
||||
|
||||
import "encoding/hex"
|
||||
import "fmt"
|
||||
|
||||
type Object interface {
|
||||
GetType() string
|
||||
GetId() string
|
||||
GetType() string
|
||||
GetId() string
|
||||
}
|
||||
|
||||
func ObjectSafeId(obj Object) string {
|
||||
return hex.EncodeToString([]byte(obj.GetId()))
|
||||
}
|
||||
|
||||
func ObjectKey(obj Object) string {
|
||||
return fmt.Sprintf(
|
||||
"%d:%s:%d:%s",
|
||||
len(obj.GetType()),
|
||||
obj.GetType(),
|
||||
len(obj.GetId()),
|
||||
obj.GetId(),
|
||||
)
|
||||
return fmt.Sprintf("%s:%s", obj.GetType(), ObjectSafeId(obj))
|
||||
}
|
||||
|
||||
89
store.go
89
store.go
@@ -1,74 +1,73 @@
|
||||
package main
|
||||
|
||||
import "encoding/hex"
|
||||
import "encoding/json"
|
||||
import "fmt"
|
||||
import "os"
|
||||
import "path/filepath"
|
||||
|
||||
type Store struct {
|
||||
root string
|
||||
root string
|
||||
}
|
||||
|
||||
func NewStore(root string) *Store {
|
||||
return &Store{
|
||||
root: root,
|
||||
}
|
||||
return &Store{
|
||||
root: root,
|
||||
}
|
||||
}
|
||||
|
||||
func (s *Store) Write(obj Object) error {
|
||||
dir := filepath.Join(s.root, obj.GetType())
|
||||
filename := hex.EncodeToString([]byte(obj.GetId()))
|
||||
dir := filepath.Join(s.root, obj.GetType())
|
||||
filename := ObjectSafeId(obj)
|
||||
|
||||
err := os.MkdirAll(dir, 0700)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err := os.MkdirAll(dir, 0700)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
tmp, err := os.CreateTemp(dir, fmt.Sprintf("%s.*", filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer tmp.Close()
|
||||
tmp, err := os.CreateTemp(dir, fmt.Sprintf("%s.*", filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer tmp.Close()
|
||||
|
||||
enc := json.NewEncoder(tmp)
|
||||
enc.SetEscapeHTML(false)
|
||||
enc := json.NewEncoder(tmp)
|
||||
enc.SetEscapeHTML(false)
|
||||
|
||||
err = enc.Encode(obj)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = enc.Encode(obj)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = tmp.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = tmp.Close()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
err = os.Rename(tmp.Name(), filepath.Join(dir, filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = os.Rename(tmp.Name(), filepath.Join(dir, filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *Store) Read(obj Object) error {
|
||||
dir := filepath.Join(s.root, obj.GetType())
|
||||
filename := hex.EncodeToString([]byte(obj.GetId()))
|
||||
dir := filepath.Join(s.root, obj.GetType())
|
||||
filename := ObjectSafeId(obj)
|
||||
|
||||
fh, err := os.Open(filepath.Join(dir, filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer fh.Close()
|
||||
fh, err := os.Open(filepath.Join(dir, filename))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer fh.Close()
|
||||
|
||||
dec := json.NewDecoder(fh)
|
||||
dec.DisallowUnknownFields()
|
||||
dec := json.NewDecoder(fh)
|
||||
dec.DisallowUnknownFields()
|
||||
|
||||
err = dec.Decode(obj)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
err = dec.Decode(obj)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -1,14 +1,14 @@
|
||||
package main
|
||||
|
||||
type Template struct {
|
||||
Id string
|
||||
Test string
|
||||
Id string
|
||||
Test string
|
||||
}
|
||||
|
||||
func (t *Template) GetType() string {
|
||||
return "template"
|
||||
return "template"
|
||||
}
|
||||
|
||||
func (t *Template) GetId() string {
|
||||
return t.Id
|
||||
return t.Id
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user