
    Ki7                       d Z ddlmZ ddlZddlZddlmZ ddlmZ ddl	m
Z
 ddlmZ ddlZddlmZ dd	lmZ dd
lmZ ddlmZmZmZ ddlmZ ddlmZmZmZ ddlm Z  ddl!m"Z"m#Z#m$Z$  ejJ                  e&      Z' G d d      Z(y)z/StreamableHTTP Session Manager for MCP servers.    )annotationsN)AsyncIterator)
HTTPStatus)Any)uuid4)
TaskStatus)Request)Response)ReceiveScopeSend)Server)MCP_SESSION_ID_HEADER
EventStoreStreamableHTTPServerTransport)TransportSecuritySettings)INVALID_REQUEST	ErrorDataJSONRPCErrorc                      e Zd ZdZ	 	 	 	 	 	 d	 	 	 	 	 	 	 	 	 	 	 	 	 d	dZej                  d
d       Z	 	 	 	 	 	 	 	 ddZ	 	 	 	 	 	 	 	 ddZ		 	 	 	 	 	 	 	 ddZ
y)StreamableHTTPSessionManagera  
    Manages StreamableHTTP sessions with optional resumability via event store.

    This class abstracts away the complexity of session management, event storage,
    and request handling for StreamableHTTP transports. It handles:

    1. Session tracking for clients
    2. Resumability via an optional event store
    3. Connection management and lifecycle
    4. Request handling and transport setup
    5. Idle session cleanup via optional timeout

    Important: Only one StreamableHTTPSessionManager instance should be created
    per application. The instance cannot be reused after its run() context has
    completed. If you need to restart the manager, create a new instance.

    Args:
        app: The MCP server instance
        event_store: Optional event store for resumability support. If provided, enables resumable connections
            where clients can reconnect and receive missed events. If None, sessions are still tracked but not
            resumable.
        json_response: Whether to use JSON responses instead of SSE streams
        stateless: If True, creates a completely fresh transport for each request with no session tracking or
            state persistence between requests.
        security_settings: Optional transport security settings.
        retry_interval: Retry interval in milliseconds to suggest to clients in SSE retry field. Used for SSE
            polling behavior.
        session_idle_timeout: Optional idle timeout in seconds for stateful sessions. If set, sessions that
            receive no HTTP requests for this duration will be automatically terminated and removed. When
            retry_interval is also configured, ensure the idle timeout comfortably exceeds the retry interval to
            avoid reaping sessions during normal SSE polling gaps. Default is None (no timeout). A value of 1800
            (30 minutes) is recommended for most deployments.
    Nc                6   ||dk  rt        d      |r|t        d      || _        || _        || _        || _        || _        || _        || _        t        j                         | _        i | _        d | _        t        j                         | _        d| _        y )Nr   z9session_idle_timeout must be a positive number of secondsz7session_idle_timeout is not supported in stateless modeF)
ValueErrorRuntimeErrorappevent_storejson_response	statelesssecurity_settingsretry_intervalsession_idle_timeoutanyioLock_session_creation_lock_server_instances_task_group	_run_lock_has_started)selfr   r   r   r   r   r    r!   s           n/home/jay/workspace/scripts/.codegraph-venv/lib/python3.12/site-packages/mcp/server/streamable_http_manager.py__init__z%StreamableHTTPSessionManager.__init__A   s      +0D0IXYY-9XYY&*"!2,$8! ',jjl#KM  !    c                 K   | j                   4 d{    | j                  rt        d      d| _        ddd      d{    t        j                         4 d{   }|| _        t        j                  d       	 d t        j                  d       |j                  j                          d| _        | j                  j                          ddd      d{    y7 7 # 1 d{  7  sw Y   xY w7 # t        j                  d       |j                  j                          d| _        | j                  j                          w xY w7 u# 1 d{  7  sw Y   yxY ww)aw  
        Run the session manager with proper lifecycle management.

        This creates and manages the task group for all session operations.

        Important: This method can only be called once per instance. The same
        StreamableHTTPSessionManager instance cannot be reused after this
        context manager exits. Create a new instance if you need to restart.

        Use this in the lifespan context manager of your Starlette app:

        @contextlib.asynccontextmanager
        async def lifespan(app: Starlette) -> AsyncIterator[None]:
            async with session_manager.run():
                yield
        NzyStreamableHTTPSessionManager .run() can only be called once per instance. Create a new instance if you need to run again.Tz&StreamableHTTP session manager startedz,StreamableHTTP session manager shutting down)r'   r(   r   r"   create_task_groupr&   loggerinfocancel_scopecancelr%   clear)r)   tgs     r*   runz StreamableHTTPSessionManager.runb   s7    & >> 	% 	%  "Y  !%D	% 	% **, 	/ 	/!DKK@A/JK&&(#' &&,,.	/ 	/ 	/	% 	% 	% 	% 	%	/ JK&&(#' &&,,.	/ 	/ 	/ 	/s   E-C&E-C*E-C(E-C?E-"E DAEE- E!E-(E-*C<0C31C<8E-AEEE-E*E!E*&E-c                   K   | j                   t        d      | j                  r| j                  |||       d{    y| j	                  |||       d{    y7 !7 w)a  
        Process ASGI request with proper session handling and transport setup.

        Dispatches to the appropriate handler based on stateless mode.

        Args:
            scope: ASGI scope
            receive: ASGI receive function
            send: ASGI send function
        Nz6Task group is not initialized. Make sure to use run().)r&   r   r   _handle_stateless_request_handle_stateful_request)r)   scopereceivesends       r*   handle_requestz+StreamableHTTPSessionManager.handle_request   sc       #WXX >>00FFF//wEEE GEs!   :A"AA"A A" A"c                   K   t         j                  d       t        d j                  d j                        t
        j                  dd fd} j                  J  j                  j                  |       d{    j                  |||       d{    j                          d{    y7 87 7 	w)z
        Process request in stateless mode - creating a new transport for each request.

        Args:
            scope: ASGI scope
            receive: ASGI receive function
            send: ASGI send function
        z7Stateless mode: Creating new transport for this requestN)mcp_session_idis_json_response_enabledr   r   task_statusc                  K   j                         4 d {   }|\  }}| j                          	 j                  j                  ||j                  j	                         d       d {    d d d       d {    y 7 j7 # t
        $ r t        j                  d       Y 5w xY w7 -# 1 d {  7  sw Y   y xY ww)NTr   zStateless session crashed)connectstartedr   r5   create_initialization_options	Exceptionr/   	exception)rA   streamsread_streamwrite_streamhttp_transportr)   s       r*   run_stateless_serverzTStreamableHTTPSessionManager._handle_stateless_request.<locals>.run_stateless_server   s     %--/ B B7,3)\##%B((,,#$>>@"&	 '   	B B B ! B$$%@ABB B B Bss   CBCB-:B-B.B2C=B+>CBB(%B-'B((B-+C-B?3B64B?;C)rA   TaskStatus[None])r/   debugr   r   r   r"   TASK_STATUS_IGNOREDr&   startr<   	terminate)r)   r9   r:   r;   rM   rL   s   `    @r*   r7   z6StreamableHTTPSessionManager._handle_stateless_request   s      	NO6%)%7%7"44	
 KPJcJc 	B 	B +++$$%9::: ++E7DAAA &&((( 	; 	B 	)s6   A=C B:C B<C 4B>5C <C >C c                   K   t        ||      }|j                  j                  t              }|| j                  v r j                  |   }t
        j                  d       |j                  < j                  0t        j                          j                  z   |j                  _        |j                  |||       d{    y|*t
        j                  d        j                  4 d{    t               j                  }t!        | j"                   j$                   j&                   j(                        j*                  J  j                  j*                  <   t
        j-                  d|        t        j.                  dd fd} j0                  J  j0                  j3                  |       d{    j                  |||       d{    ddd      d{    yt5        dd	t7        t8        d
            }	t;        |	j=                  dd      t>        j@                  d      }
 |
|||       d{    y7 7 e7 7 z7 l# 1 d{  7  sw Y   yxY w7 &w)z
        Process request in stateful mode - maintaining session state between requests.

        Args:
            scope: ASGI scope
            receive: ASGI receive function
            send: ASGI send function
        Nz1Session already exists, handling request directlyzCreating new transport)r>   r?   r   r   r    z'Created new transport with session ID: r@   c                &  K   j                         4 d {   }|\  }}| j                          	 t        j                         }j                  -t        j
                         j                  z   |_        |_        |5  j                  j                  ||j                  j                         d       d {    d d d        |j                  roj                  J t        j                  dj                   d       j                  j!                  j                  d        j#                          d {    j                  r_j                  j                  v rGj(                  s;t        j                  dj                   d       j                  j                  = 	 d d d       d {    y 7 7 # 1 sw Y   xY w7 # t$        $ r& t        j'                  dj                   d       Y w xY w# j                  raj                  j                  v rHj(                  s;t        j                  dj                   d       j                  j                  = w w w w xY w7 # 1 d {  7  sw Y   y xY ww)NFrC   zSession z idle timeoutz crashedzCleaning up crashed session z from active instances.)rD   rE   r"   CancelScoper!   current_timedeadline
idle_scoper   r5   rF   cancelled_caughtr>   r/   r0   r%   poprR   rG   rH   is_terminated)rA   rI   rJ   rK   rX   rL   r)   s        r*   
run_serverzIStreamableHTTPSessionManager._handle_stateful_request.<locals>.run_server  sy    -557 'Z 'Z74;1\#++-$Z
 */):):)<J#88D6;6H6H6JTMfMf6f
 3<F 9!+ "&*hhll$/$0$(HH$J$J$L.3	 '3 '" !" !""  *::'5'D'D'P P'P &h~7T7T6UUb,c d $ 6 6 : :>;X;XZ^ _&4&>&>&@ @ @
 !/ = =$2$A$ATE[E[$[(6(D(D &$B'5'D'D&E F8%8!"
 %)$:$:>;X;X$YO'Z 'Z 'Z!"" " !A( a",,x8U8U7VV^-_`a !/ = =$2$A$ATE[E[$[(6(D(D &$B'5'D'D&E F8%8!"
 %)$:$:>;X;X$Y )E %\ !>='Z 'Z 'Z 'Zs   JGJI<AG;G=G>GA>G GGA+I<0J;I:<JGG		G,HHHHA/I77I<:J<JJJ
Jz2.0zserver-errorzSession not found)codemessage)jsonrpciderrorT)by_aliasexclude_nonezapplication/json)contentstatus_code
media_type)rA   rN   returnNone)!r	   headersgetr   r%   r/   rO   rX   r!   r"   rV   rW   r<   r$   r   hexr   r   r   r   r    r>   r0   rP   r&   rQ   r   r   r   r
   model_dump_jsonr   	NOT_FOUND)r)   r9   r:   r;   requestrequest_mcp_session_id	transportnew_session_idr\   error_responseresponserL   s   `          @r*   r8   z5StreamableHTTPSessionManager._handle_stateful_request   sj     %)!(!4!45J!K "-2HDLbLb2b../EFILLLM##/D4M4M4Y050B0B0DtG`G`0`	$$-**5'4@@@!)LL1222 ?J ?J!&!>#1-1-?-? $ 0 0&*&<&<#'#6#6" &44@@@HV&&~'D'DEEnEUVW INHaHa (Z (ZV ''333&&,,Z888 %33E7DIII?J ?J ?JH *!(/N  &66SW6X&00-H
 5'4000o A?Jx 9 J?J ?J ?J ?Jb 1s   B>I5I-I50I1I54CIIII I$I5/I0AI5I3I5I5III5I0$I'%I0,I5)NFFNNN)r   zMCPServer[Any, Any]r   zEventStore | Noner   boolr   rt   r   z TransportSecuritySettings | Noner    z
int | Noner!   zfloat | None)rg   zAsyncIterator[None])r9   r   r:   r   r;   r   rg   rh   )__name__
__module____qualname____doc__r+   
contextlibasynccontextmanagerr5   r<   r7   r8    r,   r*   r   r      s    J *.#>B%)-1" " '" 	"
 " <" #" +"B ##&/ $&/PFF F 	F
 
F2/)/) /) 	/)
 
/)bo1o1 o1 	o1
 
o1r,   r   ))rx   
__future__r   ry   loggingcollections.abcr   httpr   typingr   uuidr   r"   	anyio.abcr   starlette.requestsr	   starlette.responsesr
   starlette.typesr   r   r   mcp.server.lowlevel.serverr   	MCPServermcp.server.streamable_httpr   r   r   mcp.server.transport_securityr   	mcp.typesr   r   r   	getLoggerru   r/   r   r{   r,   r*   <module>r      sf    5 "   )       & ( 0 0 : 
 D > >			8	$f1 f1r,   