
    i                        d Z ddlmZ ddlZddlZddlZddlZddlZddlm	Z	 ddl
mZmZ  ej        d          Z e            Z e            Z e            Ze	 G d d                      Z G d	 d
          ZdS )u~  Gateway streaming consumer — bridges sync agent callbacks to async platform delivery.

The agent fires stream_delta_callback(text) synchronously from its worker thread.
GatewayStreamConsumer:
  1. Receives deltas via on_delta() (thread-safe, sync)
  2. Queues them to an asyncio task via queue.Queue
  3. The async run() task buffers, rate-limits, and progressively edits
     a single message on the target platform

Design: Uses the edit transport (send initial message, then editMessageText).
This is universally supported across Telegram, Discord, and Slack.

Credit: jobless0x (#774, #1312), OutThisLife (#798), clicksingh (#697).
    )annotationsN)	dataclass)AnyOptionalzgateway.stream_consumerc                  Z    e Zd ZU dZdZded<   dZded<   dZd	ed
<   dZded<   dZ	ded<   dS )StreamConsumerConfigz5Runtime config for a single stream consumer instance.g      ?floatedit_interval(   intbuffer_thresholdu    ▉strcursorFboolbuffer_only        fresh_final_after_secondsN)
__name__
__module____qualname____doc__r
   __annotations__r   r   r   r        </home/ubuntu/.hermes/hermes-agent/gateway/stream_consumer.pyr   r   (   sq         ??MFK (+******r   r   c                  Z   e Zd ZdZdZdZdZ	 	 	 d9d:dZed;d            Z	ed;d            Z
d<dZd=dZd<dZddd>dZd=dZd<dZd=d Zd<d!Zd<d"Z ej        d#          Zed?d$            Zd@d'ZdAd(ZdBd*ZedCd.            Zd=d/Zd;d0Zd<d1Zd<d2Z dDd3Z!d;d4Z"dDd5Z#dd6dEd8Z$dS )FGatewayStreamConsumera2  Async consumer that progressively edits a platform message with streamed tokens.

    Usage::

        consumer = GatewayStreamConsumer(adapter, chat_id, config, metadata=metadata)
        # Pass consumer.on_delta as stream_delta_callback to AIAgent
        agent = AIAgent(..., stream_delta_callback=consumer.on_delta)
        # Start the consumer as an asyncio task
        task = asyncio.create_task(consumer.run())
        # ... run agent in thread pool ...
        consumer.finish()  # signal completion
        await task         # wait for final edit
       )z<REASONING_SCRATCHPAD>z<think>z<reasoning>z
<THINKING>z
<thinking>z	<thought>)z</REASONING_SCRATCHPAD>z</think>z</reasoning>z</THINKING>z</thinking>z
</thought>Nadapterr   chat_idr   configOptional[StreamConsumerConfig]metadataOptional[dict]on_new_messageOptional[callable]c                   || _         || _        |pt                      | _        || _        || _        t          j                    | _        d| _	        d | _
        d | _        d| _        d| _        d| _        d| _        d| _        d| _        d| _        | j        j        | _        d| _        t-          |dd          du | _        d| _        d| _        d S )N FTr   r   REQUIRES_EDIT_FINALIZE)r   r    r   cfgr#   _on_new_messagequeueQueue_queue_accumulated_message_id_message_created_ts_already_sent_edit_supported_last_edit_time_last_sent_text_fallback_final_send_fallback_prefix_flood_strikesr
   _current_edit_interval_final_response_sentgetattr_adapter_requires_finalize_in_think_block_think_buffer)selfr   r    r!   r#   r%   s         r   __init__zGatewayStreamConsumer.__init__X   s     3133   .#(;==*. 59 "#"!$)! "&*h&<#$)! G5u==E 	'
  %r   returnr   c                    | j         S )z?True if at least one message was sent or edited during the run.)r2   r?   s    r   already_sentz"GatewayStreamConsumer.already_sent   s     !!r   c                    | j         S )zBTrue when the stream consumer delivered the final assistant reply.)r:   rC   s    r   final_response_sentz)GatewayStreamConsumer.final_response_sent   s     ((r   Nonec                D    | j                             t                     dS )z>Finalize the current stream segment and start a fresh message.N)r.   put_NEW_SEGMENTrC   s    r   on_segment_breakz&GatewayStreamConsumer.on_segment_break   s    %%%%%r   textc                P    |r#| j                             t          |f           dS dS )z7Queue a completed interim assistant commentary message.N)r.   rI   _COMMENTARYr?   rL   s     r   on_commentaryz#GatewayStreamConsumer.on_commentary   s3     	1KOO[$/00000	1 	1r   c                    | j         }|dS 	  |             dS # t          $ r  t                              dd           Y dS w xY w)z8Fire the on_new_message callback, swallowing any errors.Nzon_new_message callback errorT)exc_info)r+   	Exceptionloggerdebug)r?   cbs     r   _notify_new_messagez)GatewayStreamConsumer._notify_new_message   sh    !:F	IBDDDDD 	I 	I 	ILL84LHHHHHH	Is   
 &AAFpreserve_no_editrY   c               x    |r| j         dk    rd S d | _         d | _        d| _        d| _        d| _        d| _        d S )N__no_edit__r(   F)r0   r1   r/   r5   r6   r7   )r?   rY   s     r   _reset_segment_statez*GatewayStreamConsumer._reset_segment_state   sR     	 0M A AF#' !$)! "r   c                r    |r| j                             |           dS ||                                  dS dS )u2  Thread-safe callback — called from the agent's worker thread.

        When *text* is ``None``, signals a tool boundary: the current message
        is finalized and subsequent text will be sent as a new message so it
        appears below any tool-progress messages the gateway sent in between.
        N)r.   rI   rK   rO   s     r   on_deltazGatewayStreamConsumer.on_delta   sJ      	$KOOD!!!!!\!!##### \r   c                D    | j                             t                     dS )z#Signal that the stream is complete.N)r.   rI   _DONErC   s    r   finishzGatewayStreamConsumer.finish   s    r   c                   | j         |z   }d| _         |rm| j        rd}d}| j        D ]:}|                    |          }|dk    r|dk    s||k     r|}t	          |          };|rd| _        |||z   d         }nt          d | j        D                       }t	          |          |k    r|| d         n|| _         dS d}d}| j        D ]}d}	 |                    ||          }|dk    rn|dk    r#| j         p| j                            d          }	n|d|         }
|
	                    d          }|dk    r;| j         s| j                            d          o|

                                dk    }	n#|
|d	z   d         
                                dk    }	|	r|dk    s||k     r|}t	          |          }n|d	z   }|r-| xj        |d|         z  c_        d| _        |||z   d         }nd}| j        D ]G}t          d	t	          |                    D ]'}|                    |d|                   r||k    r|}(H|r*| xj        |d|          z  c_        || d         | _         n| xj        |z  c_        dS |kdS dS )
ab  Add a text delta to the accumulated buffer, suppressing think blocks.

        Uses a state machine that tracks whether we are inside a
        reasoning/thinking block.  Text inside such blocks is silently
        discarded.  Partial tags at buffer boundaries are held back in
        ``_think_buffer`` until enough characters arrive to decide.
        r(   r   FNc              3  4   K   | ]}t          |          V  d S )N)len).0ts     r   	<genexpr>z?GatewayStreamConsumer._filter_and_accumulate.<locals>.<genexpr>   s(      !I!IQ#a&&!I!I!I!I!I!Ir   T
   )r>   r=   _CLOSE_THINK_TAGSfindre   max_OPEN_THINK_TAGSr/   endswithrfindstriprange)r?   rL   bufbest_idxbest_lentagidxmax_tagsearch_startis_boundary	precedinglast_nl	held_backis                 r   _filter_and_accumulatez,GatewayStreamConsumer._filter_and_accumulate   sd     4' K	# J1 , ,C((3--Cbyyh"nnh#&#&s88 	+0D(h1223CC "!I!I$2H!I!I!IIIG;>s88g;M;MgXYYSVD&F 0 / /C#$L/!hhsL99"99!!88$($5 5 !D#'#4#=#=d#C#C (K
 ),DSD	I&/ood&;&;G&"}})-):%: &I(,(9(B(B4(H(H%@(1(9(9R(? !, /8!.E.K.K.M.MQS.S& "HNNcHnn'*H'*3xxH!'*Qw5/8  %%YhY7%%+/D(h1223CC !"I#4 . .!&q#c((!3!3 . .A"||CG44 .Y,-	. ! 1))S9*-==))-0)-=**))S0))FW  K	 K	 K	 K	 K	r   c                b    | j         r%| j        s | xj        | j         z  c_        d| _         dS dS dS )zFlush any held-back partial-tag buffer into accumulated text.

        Called when the stream ends (got_done) so that partial text that
        was held back waiting for a possible opening tag is not lost.
        r(   N)r>   r=   r/   rC   s    r   _flush_think_bufferz)GatewayStreamConsumer._flush_think_buffer"  sR      	$d&: 	$!33!#D	$ 	$ 	$ 	$r   c                
  K   t          | j        dd          }t          d|t          | j        j                  z
  dz
            }	 	 d}d}d}	 	 | j                                        }|t          u rd}nx|t          u rd}nlt          |t                    r+t          |          dk    r|d	         t          u r	|d
         }n,|                     |           n# t          j        $ r Y nw xY w|r|                                  t#          j                    }|| j        z
  }|p|p|du}	| j        j        s6|	p3|| j        k    r| j        p!t          | j                  | j        j        k    }	d}
|	r| j        rt          | j                  |k    r| j        | j                            | j        |          }|D ]#}|                     || j                   d{V  $d| _        d| _        t#          j                    | _        |r| j        | _        dS |rd| _        d| _        d| _        t          | j                  |k    r| j        | j         r| j        !                    dd	|          }||dz  k     r|}| j        d|         }| "                    |           d{V }| j        s|sn[| j        |d         #                    d          | _        d| _        d| _        t          | j                  |k    r| j        | j         | j        }|s|s||| j        j        z  }| "                    ||           d{V }
t#          j                    | _        |r| j        r| j        r!| $                    | j                   d{V  nl|
r| j%        sd| _        n[| j        r(| "                    | j        d           d{V | _        n,| j        s%| "                    | j                   d{V | _        dS |[| &                                 | '                    |           d{V  t#          j                    | _        | &                                 |rK| j        r.|
s,| j        r%| j        dk    r| (                                 d{V  | &                    d           tS          j*        d           d{V  u# tR          j+        $ ri d}| j        rF| j        r?	 tY          | "                    | j                   d{V           }n# tZ          $ r Y nw xY w|r| j        sd| _        Y dS Y dS Y dS tZ          $ r&}t\          /                    d|           Y d}~dS d}~ww xY w)z@Async task that drains the queue and edits the platform message.MAX_MESSAGE_LENGTH     d   TFN   r   rj   r(   ri   finalizer[   rX   g?zStream consumer error: %s)0r;   r   rm   re   r*   r   r.   
get_nowaitr`   rJ   
isinstancetuplerN   r   r,   Emptyr   time	monotonicr4   r   r9   r/   r   r0   truncate_message_send_new_chunkr5   r2   r:   r6   r7   r3   rp   _send_or_editlstrip_send_fallback_finalr<   r\   _send_commentary#_flush_segment_tail_on_edit_failureasynciosleepCancelledErrorr   rS   rT   error)r?   
_raw_limit_safe_limitgot_donegot_segment_breakcommentary_textitemnowelapsedshould_editcurrent_update_visiblechunkschunksplit_atokdisplay_text_best_effort_okes                     r   runzGatewayStreamConsumer.run,  s\      T\+?FF
#zC,@,@@3FGGF	9q* $)!"&#{55775=='+H!<//04-!%dE22 "s4yyA~~$q'U`J`J`.21gO!33D9999 ;   &  /,,... n&& 44 3(3&d2 
 x+ "- # D$?? 2 $ 1Ot011TX5NN   */& B<4#4 B< D-..<< ,4 "&!>!> -{" " &, P PE"&"6"6ud>N"O"OOOOOOOOO,.)/1,/3~/?/?,# #8<8JD5"F, 7/3D,8=D546D1 
 D-..<< ,8 0 9 $(#4#:#:4K#P#P#kQ&666'2H $ 1)8) <#'#5#5e#<#<<<<<<<4 "B " ",0,=hii,H,O,OPT,U,U)+/(/1,% D-..<< ,8 0 9& $(#4L# 8,= 8/BY$7 483E3E$!2 4F 4 4 . . . . . .* ,0>+;+;D( 
 ( d4 d"&";";D<M"N"NNNNNNNNN2d$($Cd 9=D55!- d ?C>P>P $ 1D ?Q ? ? 9 9 9 9 9 9D55 "&!3 d>B>P>PQUQb>c>c8c8c8c8c8c8cD5F".--/////@@@@@@@@@+/>+;+;D(--///  % E )I 6I !,I !,=="FFHHHHHHHHH--t-DDDmD)))))))))cq*f % 	1 	1 	1#O  T%5 &*1C1CDDU1V1V+V+V+V+V+V+V&W&WOO    D  1t'@ 1,0))))1 1 1 1 1 1 	9 	9 	9LL4a888888888	9s   R= $C 4R= 5C  R= ?C  R= C R= C)&R= (C))D'R= G#R= 7CR= =U#-T
U#
TU#TU#4	U#=UU#z[`"']?MEDIA:\s*\S+[`"']?c                    d| vrd| vr| S |                      dd          }t          j                            d|          }t	          j        dd|          }|                                S )u  Strip MEDIA: directives and internal markers from text before display.

        The streaming path delivers raw text chunks that may include
        ``MEDIA:<path>`` tags and ``[[audio_as_voice]]`` directives meant for
        the platform adapter's post-processing.  The actual media files are
        delivered separately via ``_deliver_media_from_response()`` after the
        stream finishes — we just need to hide the raw directives from the
        user.
        zMEDIA:z[[audio_as_voice]]r(   z\n{3,}z

)replacer   	_MEDIA_REsubrerstrip)rL   cleaneds     r   _clean_for_displayz(GatewayStreamConsumer._clean_for_display  sn     4$8$D$DK,,3R88'155b'BB&FG44~~r   reply_to_idOptional[str]c                  K   |                      |          }|                                s|S 	 | j        rt          | j                  ni }| j                            | j        |||           d{V }|j        rV|j        rOt          |j                  | _
        d| _        || _        |                                  t          |j                  S d| _        |S # t          $ r'}t                               d|           |cY d}~S d}~ww xY w)zSend a new message chunk, optionally threaded to a previous message.

        Returns the message_id so callers can thread subsequent chunks.
        )r    contentreply_tor#   NTFzStream send chunk error: %s)r   rq   r#   dictr   sendr    success
message_idr   r0   r2   r5   rW   r3   rS   rT   r   )r?   rL   r   metaresultr   s         r   r   z%GatewayStreamConsumer._send_new_chunk  sN     
 &&t,,zz|| 		*.-?4&&&RD<,,$	 -        F ~ 
#&"3 
##&v'8#9#9 %)"'+$ ((***6,---',$"" 	 	 	LL6:::	s$   B"C C 
D%DDDc                    | j         pd}| j        j        rA|                    | j        j                  r"|dt	          | j        j                            }|                     |          S )z>Return the visible text already shown in the streamed message.r(   N)r5   r*   r   ro   re   r   r?   prefixs     r   _visible_prefixz%GatewayStreamConsumer._visible_prefix2  se    %+8? 	4vtx?? 	42c$(/22223F&&v...r   
final_textc                    | j         p|                                 }|r>|                    |          r)|t          |          d                                         S |S )zAReturn only the part of final_text the user has not already seen.N)r7   r   
startswithre   r   )r?   r   r   s      r   _continuation_textz(GatewayStreamConsumer._continuation_text9  s_    &@$*>*>*@*@ 	5j++F33 	5c&kkll+22444r   limitr   	list[str]c                l   t          |           |k    r| gS g }| }t          |          |k    ro|                    dd|          }||dz  k     r|}|                    |d|                    ||d                             d          }t          |          |k    o|r|                    |           |S )z;Split text into reasonably sized chunks for fallback sends.ri   r   r   N)re   rp   appendr   )rL   r   r   	remainingr   s        r   _split_text_chunksz(GatewayStreamConsumer._split_text_chunks@  s     t996M	)nnu$$ tQ66H%1*$$ MM)IXI.///!()),33D99I )nnu$$  	%MM)$$$r   c                v  K   |                      |          }|                     |          }d| _        |                                s|                                r||                                 k    r|}n| j        r| j        r| j        j        r| j        	                    | j        j                  rt| j        dt          | j        j                            }	 | j                            | j        | j        |           d{V }|j        r|| _        n# t          $ r Y nw xY wd| _        d| _        dS t%          | j        dd          }t'          d|dz
            }|                     ||          }d}	d	}
d}|D ]}d}t+          d
          D ]}| j                            | j        || j                   d{V }|j        r nQ|dk    rJ|                     |          r5t2                              d           t7          j        d           d{V   |r|j        sG|r&d| _        d| _        |	| _        |
| _        d	| _         dS d| _        d| _        d	| _        d	| _         dS d}|}
|j        p|	}	|                                  |	| _        d| _        d| _        |d         | _        d	| _        dS )zSend the final continuation after streaming edits stop working.

        Retries each chunk once on flood-control failures with a short delay.
        FNr    r   r   Tr   r   r   r   r(   r   r    r   r#   r   z.Flood control on fallback send, retrying in 3sg      @rc   ) r   r   r6   rq   r   r0   r5   r*   r   ro   re   r   edit_messager    r   rS   r2   r:   r;   rm   r   rr   r   r#   _is_flood_errorrT   rU   r   r   r7   r   rW   )r?   rL   r   continuation
clean_textr   	raw_limit
safe_limitr   last_message_idlast_successful_chunksent_any_chunkr   attempts                 r   r   z*GatewayStreamConsumer._send_fallback_finalQ  s}     
 ,,T22
..z::$)!!!## $	 !! jD4H4H4J4J&J&J) $,  ,55dhoFF	 "&!56LDHO8L8L7L6L!MJ	'+|'@'@$(L'+'7$. (A ( ( " " " " " "
 "> >3=D0$   %)",0)DL*>EE	i#o..
((zBB)- " *	' *	'EF 88  #|00 L!!]  1          
 > Ea<<D$8$8$@$@<LLH   "-,,,,,,,,,,  ! 	 *.D&04D-'6D$+@D(,.D)FF &+"#' ')$(*%!N$)!$/B?O $$&&&&*!$(!%bz "s   ;D 
D&%D&c                h    t          |dd          pd}|                                }d|v pd|v pd|v S )zFCheck if a SendResult failure is due to flood control / rate limiting.r   r(   floodzretry afterrate)r;   lower)r?   r   err	err_lowers       r   r   z%GatewayStreamConsumer._is_flood_error  sD    fgr**0bIIKK	)#X}	'AXVyEXXr   c                H  K   | j         s|                                  d{V  | j        p|                                 }| j        }|r>|                    |          r)|t          |          d                                         }|                     |          }|	                                sdS 	 | j
                            | j        || j                   d{V }|j        r	d| _        dS dS # t           $ r&}t"                              d|           Y d}~dS d}~ww xY w)aK  Deliver un-sent tail content before a segment-break reset.

        When an edit fails (flood control, transport error) and a tool
        boundary arrives before the next retry, ``_accumulated`` holds text
        that was generated but never shown to the user. Without this flush,
        the segment reset would discard that tail and leave a frozen cursor
        in the partial message.

        Sends the tail that sits after the last successfully-delivered
        prefix as a new message, and best-effort strips the stuck cursor
        from the previous partial message.
        Nr   Tz"Segment-break tail flush error: %s)r6   _try_strip_cursorr7   r   r/   r   re   r   r   rq   r   r   r    r#   r   r2   rS   rT   r   )r?   visibletailr   r   s        r   r   z9GatewayStreamConsumer._flush_segment_tail_on_edit_failure  sv      ( 	+((*********'A4+?+?+A+A  	0tw// 	0G&--//D&&t,,zz|| 	F		B<,, -        F
 ~ *%)"""* * 	B 	B 	BLL=qAAAAAAAAA	Bs   2;C1 1
D!;DD!c                  K   | j         r| j         dk    rdS |                                 }|r|                                sdS 	 | j                            | j        | j         |           d{V  || _        dS # t          $ r Y dS w xY w)u   Best-effort edit to remove the cursor from the last visible message.

        Called when entering fallback mode so the user doesn't see a stuck
        cursor (▉) in the partial message.
        r[   Nr   )r0   r   rq   r   r   r    r5   rS   r   s     r   r   z'GatewayStreamConsumer._try_strip_cursor  s        	4#3}#D#DF%%'' 	V\\^^ 	F	,+++ ,         
 $*D    	 	 	DD	s   4A: :
BBc                b  K   |                      |          }|                                sdS 	 | j                            | j        || j                   d{V }|j        r|                                  |j        S # t          $ r&}t          
                    d|           Y d}~dS d}~ww xY w)z6Send a completed interim assistant commentary message.Fr   NzCommentary send error: %s)r   rq   r   r   r    r#   r   rW   rS   rT   r   )r?   rL   r   r   s       r   r   z&GatewayStreamConsumer._send_commentary  s      &&t,,zz|| 	5	<,, -        F ~ + ((***>! 	 	 	LL4a88855555	s   AA> >
B.B))B.c                    t          | j        dd          pd}|dk    rdS | j        r| j        dk    rdS | j        dS t	          j                    | j        z
  }||k    S )a  Return True when a long-lived preview should be replaced with a
        fresh final message instead of an edit.

        Conditions:
        - Fresh-final is enabled (``fresh_final_after_seconds > 0``).
        - We have a real preview message id (not the ``__no_edit__`` sentinel
          and not ``None``).
        - The preview has been visible for at least the configured threshold.

        Ported from openclaw/openclaw#72038.
        r   r   r   Fr[   )r;   r*   r0   r1   r   r   )r?   	thresholdages      r   _should_send_fresh_finalz.GatewayStreamConsumer._should_send_fresh_final  sv     DH&A3GGN3	>>5 	4#3}#D#D5#+5n!99ir   c                  K   | j         }	 | j                            | j        || j                   d{V }n3# t
          $ r&}t                              d|           Y d}~dS d}~ww xY wt          |dd          sdS |rj|dk    rdt          | j        dd          }|L	  || j        |           d{V  n3# t
          $ r&}t                              d||           Y d}~nd}~ww xY wt          |d	d          }|r || _         t          j
                    | _        nd| _         d| _        d
| _        || _        d
| _        d
S )aC  Send ``text`` as a brand-new message (best-effort delete the old
        preview) so the platform's visible timestamp reflects completion
        time.  Returns True on successful delivery, False on any failure so
        the caller falls back to the normal edit path.

        Ported from openclaw/openclaw#72038.
        r   Nz1Fresh-final send failed, falling back to edit: %sFr   r[   delete_messagez+Fresh-final preview cleanup failed (%s): %sr   T)r0   r   r   r    r#   rS   rT   rU   r;   r   r   r1   r2   r5   r:   )r?   rL   old_message_idr   r   	delete_fnnew_message_ids          r   _try_fresh_finalz&GatewayStreamConsumer._try_fresh_final$  s      )	<,, -        FF
  	 	 	LLLaPPP55555	 vy%00 	5  		n==.>EEI$#)DL.AAAAAAAAAA    LLE&        !t<< 	,-D'+~'7'7D$$
  -D'+D$!#$(!ts-   -9 
A)A$$A) B8 8
C(C##C(r   r   c                 K   |                      |          }|}| j        j        r |                    | j        j        d          }|                                }|sdS |                                sdS d}| j        /| j        j        r#| j        j        |v rt          |          |k     rdS 	 | j        | j        r|| j        k    r|r| j	        sdS |r1| 
                                r|                     |           d{V rdS | j                            | j        | j        ||           d{V }|j        rd| _        || _        d| _        dS |                     |          r| xj        dz  c_        t'          | j        dz  d	          | _        t*                              d
| j        | j        | j                   | j        | j        k     rt1          j                    | _        dS t*                              d| j                   |                                 | _        d| _        d| _        d| _        |                                  d{V  dS dS | j                            | j        || j                    d{V }|j        r|j!        r%|j!        | _        t1          j                    | _"        nd| _        d| _        || _        |j!        s'|                                 | _        d| _        d| _        | #                                 dS d| _        dS # tH          $ r&}t*          %                    d|           Y d}~dS d}~ww xY w)a\  Send or edit the streaming message.

        Returns True if the text was successfully delivered (sent or edited),
        False otherwise.  Callers like the overflow split loop use this to
        decide whether to advance past the delivered chunk.

        ``finalize`` is True when this is the last edit in a streaming
        sequence.
        r(   T   N)r    r   r   r   r   rj   r   g      $@u@   Flood control on edit (strike %d/%d), backoff interval → %.1fsFz0Edit failed (strikes=%d), entering fallback moder   r[   zStream send/edit error: %s)&r   r*   r   r   rq   r0   re   r3   r5   r<   r   r   r   r   r    r   r2   r8   r   minr9   rT   rU   _MAX_FLOOD_STRIKESr   r   r4   r   r7   r6   r   r   r#   r   r1   rW   rS   r   )r?   rL   r   visible_without_cursor_visible_stripped_MIN_NEW_MSG_CHARSr   r   s           r   r   z#GatewayStreamConsumer._send_or_editY  s$      &&t,, "&8? 	Y%;%C%CDHOUW%X%X"288::  	4zz|| 	4 $HO %HOt++)**-???4z	+' R! t333  4%)%D 4  $t !$ 99;;$ #'"7"7"="=======$
  $t#'<#<#< $#'#3 $!)	 $= $ $      F ~ ,%-1*/3,./+#t  //77 - //14//:= $ ;a ?; ;D7 #LL!= $ 3 $ 7 $ ;    $2T5LLL 8<~7G7G 4',u
 N /   150D0D0F0F-481/4,-1* #44666666666$u !5  $|00 L !]  1          
 > !( 5+1+<( 48>3C3C00/4,)-D&+/D(!, 9040D0D0F0F-481 ,9(
 ,,...4 ,1D( 5 	 	 	LL5q99955555	s@   -$L 1L A
L BL *A(L B7L L 
M"MM)NNN)
r   r   r    r   r!   r"   r#   r$   r%   r&   )rA   r   )rA   rG   )rL   r   rA   rG   )rY   r   rA   rG   )rL   r   rA   r   )rL   r   r   r   rA   r   )rA   r   )r   r   rA   r   )rL   r   r   r   rA   r   )rL   r   rA   r   )rL   r   r   r   rA   r   )%r   r   r   r   r   rn   rk   r@   propertyrD   rF   rK   rP   rW   r\   r^   ra   r   r   r   r   compiler   staticmethodr   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r   9   s          
 26#'-11  1  1  1  1 f " " " X" ) ) ) X)& & & &1 1 1 1
I I I I @E # # # # # #
$ 
$ 
$ 
$   V V V Vp$ $ $ $L9 L9 L9 L9b 
:;;I      \ &   >/ / / /       \ e# e# e# e#NY Y Y YB B B BB   *   4       ,3 3 3 3j BG a a a a a a a ar   r   )r   
__future__r   r   loggingr,   r   r   dataclassesr   typingr   r   	getLoggerrT   objectr`   rJ   rN   r   r   r   r   r   <module>r     s$    # " " " " "    				  ! ! ! ! ! !                		4	5	5 	 vxx fhh + + + + + + + + A A A A A A A A A Ar   