Highly-opinionated (ex-bullshit-free) MTPROTO proxy for Telegram. If you use v1.0 or upgrade broke you proxy, please read the chapter Version 2
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

event_stream.go 1.3KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364
  1. package events
  2. import (
  3. "context"
  4. "runtime"
  5. "github.com/9seconds/mtg/v2/mtglib"
  6. "github.com/OneOfOne/xxhash"
  7. )
  8. type eventStream struct {
  9. ctx context.Context
  10. ctxCancel context.CancelFunc
  11. chans []chan mtglib.Event
  12. }
  13. func (e eventStream) Send(ctx context.Context, evt mtglib.Event) {
  14. chanNo := int(xxhash.ChecksumString32(evt.ConnectionID())) % len(e.chans)
  15. select {
  16. case <-ctx.Done():
  17. case <-e.ctx.Done():
  18. case e.chans[chanNo] <- evt:
  19. }
  20. }
  21. func (e eventStream) Shutdown() {
  22. e.ctxCancel()
  23. }
  24. func NewEventStream(observerFactories []ObserverFactory) mtglib.EventStream {
  25. ctx, cancel := context.WithCancel(context.Background())
  26. rv := eventStream{
  27. ctx: ctx,
  28. ctxCancel: cancel,
  29. chans: make([]chan mtglib.Event, runtime.NumCPU()),
  30. }
  31. for i := 0; i < runtime.NumCPU(); i++ {
  32. rv.chans[i] = make(chan mtglib.Event, 1)
  33. go eventStreamProcessor(ctx, rv.chans[i], newMultiObserver(observerFactories))
  34. }
  35. return rv
  36. }
  37. func eventStreamProcessor(ctx context.Context, eventChan <-chan mtglib.Event, observer Observer) {
  38. defer observer.Shutdown()
  39. for {
  40. select {
  41. case <-ctx.Done():
  42. return
  43. case evt := <-eventChan:
  44. switch typedEvt := evt.(type) {
  45. case mtglib.EventStart:
  46. observer.EventStart(typedEvt)
  47. case mtglib.EventFinish:
  48. observer.EventFinish(typedEvt)
  49. }
  50. }
  51. }
  52. }