github.com/keybase/client/go@v0.0.0-20241007131713-f10651d043c8/chat/typingmonitor.go (about) 1 package chat 2 3 import ( 4 "context" 5 "fmt" 6 "sync" 7 "time" 8 9 "strings" 10 11 "github.com/keybase/client/go/chat/globals" 12 "github.com/keybase/client/go/chat/utils" 13 "github.com/keybase/client/go/protocol/chat1" 14 "github.com/keybase/clockwork" 15 ) 16 17 const typingTimeout = 10 * time.Second 18 const maxExtensions = 50 19 20 type typingControlChans struct { 21 typer chat1.TyperInfo 22 23 stopCh chan struct{} 24 extendCh chan struct{} 25 } 26 27 func newTypingControlChans(typer chat1.TyperInfo) *typingControlChans { 28 return &typingControlChans{ 29 typer: typer, 30 // Might not need these buffers, but we really don't want to deadlock 31 stopCh: make(chan struct{}, 5), 32 extendCh: make(chan struct{}, 5), 33 } 34 } 35 36 type TypingMonitor struct { 37 globals.Contextified 38 sync.Mutex 39 utils.DebugLabeler 40 41 timeout time.Duration 42 clock clockwork.Clock 43 typers map[string]*typingControlChans 44 45 // Testing 46 extendCh *chan struct{} 47 } 48 49 func NewTypingMonitor(g *globals.Context) *TypingMonitor { 50 return &TypingMonitor{ 51 Contextified: globals.NewContextified(g), 52 DebugLabeler: utils.NewDebugLabeler(g.ExternalG(), "TypingMonitor", false), 53 typers: make(map[string]*typingControlChans), 54 clock: clockwork.NewRealClock(), 55 timeout: typingTimeout, 56 } 57 } 58 59 func (t *TypingMonitor) SetClock(clock clockwork.Clock) { 60 t.clock = clock 61 } 62 63 func (t *TypingMonitor) SetTimeout(timeout time.Duration) { 64 t.timeout = timeout 65 } 66 67 func (t *TypingMonitor) key(typer chat1.TyperInfo, convID chat1.ConversationID) string { 68 return fmt.Sprintf("%s:%s:%s", typer.Uid, typer.DeviceID, convID) 69 } 70 71 func (t *TypingMonitor) convKey(key string, convID chat1.ConversationID) bool { 72 toks := strings.Split(key, ":") 73 if len(toks) != 3 { 74 return false 75 } 76 return toks[2] == convID.String() 77 } 78 79 func (t *TypingMonitor) notifyConvUpdateLocked(ctx context.Context, convID chat1.ConversationID) { 80 var typers []chat1.TyperInfo 81 for k, v := range t.typers { 82 if t.convKey(k, convID) { 83 typers = append(typers, v.typer) 84 } 85 } 86 87 update := chat1.ConvTypingUpdate{ 88 ConvID: convID, 89 Typers: typers, 90 } 91 t.G().ActivityNotifier.TypingUpdate(ctx, []chat1.ConvTypingUpdate{update}) 92 } 93 94 func (t *TypingMonitor) Update(ctx context.Context, typer chat1.TyperInfo, convID chat1.ConversationID, 95 teamType chat1.TeamType, typing bool) { 96 97 // If this is about ourselves, then don't bother 98 cuid := t.G().Env.GetUID() 99 cdid := t.G().Env.GetDeviceID() 100 if cuid.Equal(typer.Uid) && cdid.Eq(typer.DeviceID) { 101 return 102 } 103 104 // If the update is for a big team we are not currently viewing, don't bother sending it 105 if teamType == chat1.TeamType_COMPLEX && !t.G().Syncer.IsSelectedConversation(convID) { 106 return 107 } 108 109 // Process the update 110 t.Lock() 111 key := t.key(typer, convID) 112 chans, alreadyTyping := t.typers[key] 113 t.Unlock() 114 if typing { 115 if alreadyTyping { 116 // If this key is already typing, let's extend it 117 select { 118 case chans.extendCh <- struct{}{}: 119 default: 120 // This should never happen, but be safe 121 t.Debug(ctx, "Update: overflowed extend channel, dropping update: %s convID: %s", typer, 122 convID) 123 } 124 } else { 125 // Not typing yet, just add it in and spawn waiter 126 chans := newTypingControlChans(typer) 127 t.insertIntoTypers(ctx, key, chans, convID) 128 t.waitOnTyper(ctx, chans, convID) 129 } 130 } else if alreadyTyping { 131 // If they are typing, then stop it 132 select { 133 case chans.stopCh <- struct{}{}: 134 default: 135 // This should never happen, but be safe 136 t.Debug(ctx, "Update: overflowed stop channel, dropping update: %s convID: %s", typer, 137 convID) 138 } 139 } 140 } 141 142 func (t *TypingMonitor) insertIntoTypers(ctx context.Context, key string, chans *typingControlChans, 143 convID chat1.ConversationID) { 144 t.Lock() 145 defer t.Unlock() 146 t.typers[key] = chans 147 t.notifyConvUpdateLocked(ctx, convID) 148 } 149 150 func (t *TypingMonitor) removeFromTypers(ctx context.Context, key string, convID chat1.ConversationID) { 151 t.Lock() 152 defer t.Unlock() 153 delete(t.typers, key) 154 t.notifyConvUpdateLocked(ctx, convID) 155 } 156 157 func (t *TypingMonitor) waitOnTyper(ctx context.Context, chans *typingControlChans, 158 convID chat1.ConversationID) { 159 key := t.key(chans.typer, convID) 160 ctx = globals.BackgroundChatCtx(ctx, t.G()) 161 deadline := t.clock.Now().Add(t.timeout) 162 go func() { 163 extends := 0 164 for { 165 select { 166 case <-t.clock.AfterTime(deadline): 167 // Send notifications and bail 168 t.removeFromTypers(ctx, key, convID) 169 return 170 case <-chans.extendCh: 171 // Loop around to restart timer 172 extends++ 173 if extends > maxExtensions { 174 t.Debug(ctx, "waitOnTyper: max extensions reached: uid: %s convID: %s", chans.typer.Uid, convID) 175 t.removeFromTypers(ctx, key, convID) 176 return 177 } 178 deadline = t.clock.Now().Add(t.timeout) 179 if t.extendCh != nil { 180 // Alerts tests we extended time 181 *t.extendCh <- struct{}{} 182 } 183 continue 184 case <-chans.stopCh: 185 // Stopped typing, just end it and remove entry in typers 186 t.removeFromTypers(ctx, key, convID) 187 return 188 } 189 } 190 }() 191 }