Initial Bus implementation

This commit is contained in:
Ian Gulliver
2022-03-15 16:27:52 +00:00
parent eb60a6aacf
commit 88c24f1a40
5 changed files with 120 additions and 69 deletions

50
bus.go Normal file
View 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
View File

@@ -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)
}

View File

@@ -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))
}

View File

@@ -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
}

View File

@@ -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
}