microsoft-agents-hosting-core 0.7.0.dev0__tar.gz → 0.7.0.dev4__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (116) hide show
  1. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/PKG-INFO +2 -2
  2. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/__init__.py +24 -0
  3. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/app/streaming/__init__.py +14 -0
  4. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/app/streaming/citation.py +22 -0
  5. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/app/streaming/citation_util.py +85 -0
  6. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/app/streaming/streaming_response.py +411 -0
  7. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/channel_service_adapter.py +27 -11
  8. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/http/__init__.py +17 -0
  9. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/http/_channel_service_routes.py +202 -0
  10. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/http/_http_adapter_base.py +136 -0
  11. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/http/_http_request_protocol.py +36 -0
  12. microsoft_agents_hosting_core-0.7.0.dev4/microsoft_agents/hosting/core/http/_http_response.py +56 -0
  13. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents_hosting_core.egg-info/PKG-INFO +2 -2
  14. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents_hosting_core.egg-info/SOURCES.txt +9 -0
  15. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents_hosting_core.egg-info/requires.txt +1 -1
  16. microsoft_agents_hosting_core-0.7.0.dev4/setup.py +21 -0
  17. microsoft_agents_hosting_core-0.7.0.dev0/setup.py +0 -15
  18. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/LICENSE +0 -0
  19. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/_oauth/__init__.py +0 -0
  20. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/_oauth/_flow_state.py +0 -0
  21. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/_oauth/_flow_storage_client.py +0 -0
  22. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/_oauth/_oauth_flow.py +0 -0
  23. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/activity_handler.py +0 -0
  24. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/agent.py +0 -0
  25. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/__init__.py +0 -0
  26. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/_routes/__init__.py +0 -0
  27. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/_routes/_route.py +0 -0
  28. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/_routes/_route_list.py +0 -0
  29. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/_routes/route_rank.py +0 -0
  30. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/_type_defs.py +0 -0
  31. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/agent_application.py +0 -0
  32. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/app_error.py +0 -0
  33. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/app_options.py +0 -0
  34. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/input_file.py +0 -0
  35. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/__init__.py +0 -0
  36. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_handlers/__init__.py +0 -0
  37. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_handlers/_authorization_handler.py +0 -0
  38. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_handlers/_user_authorization.py +0 -0
  39. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_handlers/agentic_user_authorization.py +0 -0
  40. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_sign_in_response.py +0 -0
  41. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/_sign_in_state.py +0 -0
  42. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/auth_handler.py +0 -0
  43. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/oauth/authorization.py +0 -0
  44. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/query.py +0 -0
  45. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/state/__init__.py +0 -0
  46. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/state/conversation_state.py +0 -0
  47. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/state/state.py +0 -0
  48. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/state/temp_state.py +0 -0
  49. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/state/turn_state.py +0 -0
  50. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/app/typing_indicator.py +0 -0
  51. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/__init__.py +0 -0
  52. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/access_token_provider_base.py +0 -0
  53. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/agent_auth_configuration.py +0 -0
  54. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/anonymous_token_provider.py +0 -0
  55. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/auth_types.py +0 -0
  56. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/authentication_constants.py +0 -0
  57. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/claims_identity.py +0 -0
  58. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/connections.py +0 -0
  59. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/authorization/jwt_token_validator.py +0 -0
  60. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/card_factory.py +0 -0
  61. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/channel_adapter.py +0 -0
  62. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/channel_api_handler_protocol.py +0 -0
  63. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/channel_service_client_factory_base.py +0 -0
  64. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/__init__.py +0 -0
  65. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/agent_conversation_reference.py +0 -0
  66. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/channel_factory_protocol.py +0 -0
  67. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/channel_host_protocol.py +0 -0
  68. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/channel_info_protocol.py +0 -0
  69. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/channel_protocol.py +0 -0
  70. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/channels_configuration.py +0 -0
  71. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/configuration_channel_host.py +0 -0
  72. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/conversation_constants.py +0 -0
  73. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/conversation_id_factory.py +0 -0
  74. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/conversation_id_factory_options.py +0 -0
  75. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/conversation_id_factory_protocol.py +0 -0
  76. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/http_agent_channel.py +0 -0
  77. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/client/http_agent_channel_factory.py +0 -0
  78. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/__init__.py +0 -0
  79. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/agent_sign_in_base.py +0 -0
  80. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/attachments_base.py +0 -0
  81. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/client/__init__.py +0 -0
  82. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/client/connector_client.py +0 -0
  83. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/client/user_token_client.py +0 -0
  84. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/connector_client_base.py +0 -0
  85. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/conversations_base.py +0 -0
  86. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/get_product_info.py +0 -0
  87. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/teams/__init__.py +0 -0
  88. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/teams/teams_connector_client.py +0 -0
  89. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/user_token_base.py +0 -0
  90. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/connector/user_token_client_base.py +0 -0
  91. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/errors/__init__.py +0 -0
  92. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/errors/error_resources.py +0 -0
  93. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/message_factory.py +0 -0
  94. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/middleware_set.py +0 -0
  95. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/rest_channel_service_client_factory.py +0 -0
  96. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/state/__init__.py +0 -0
  97. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/state/agent_state.py +0 -0
  98. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/state/state_property_accessor.py +0 -0
  99. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/state/user_state.py +0 -0
  100. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/__init__.py +0 -0
  101. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/_type_aliases.py +0 -0
  102. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/error_handling.py +0 -0
  103. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/memory_storage.py +0 -0
  104. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/storage.py +0 -0
  105. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/store_item.py +0 -0
  106. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/transcript_file_store.py +0 -0
  107. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/transcript_info.py +0 -0
  108. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/transcript_logger.py +0 -0
  109. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/transcript_memory_store.py +0 -0
  110. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/storage/transcript_store.py +0 -0
  111. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents/hosting/core/turn_context.py +0 -0
  112. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents_hosting_core.egg-info/dependency_links.txt +0 -0
  113. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/microsoft_agents_hosting_core.egg-info/top_level.txt +0 -0
  114. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/pyproject.toml +0 -0
  115. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/readme.md +0 -0
  116. {microsoft_agents_hosting_core-0.7.0.dev0 → microsoft_agents_hosting_core-0.7.0.dev4}/setup.cfg +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: microsoft-agents-hosting-core
3
- Version: 0.7.0.dev0
3
+ Version: 0.7.0.dev4
4
4
  Summary: Core library for Microsoft Agents
5
5
  Author: Microsoft Corporation
6
6
  License-Expression: MIT
@@ -15,7 +15,7 @@ Classifier: Operating System :: OS Independent
15
15
  Requires-Python: >=3.10
16
16
  Description-Content-Type: text/markdown
17
17
  License-File: LICENSE
18
- Requires-Dist: microsoft-agents-activity==0.7.0.dev0
18
+ Requires-Dist: microsoft-agents-activity==0.7.0.dev4
19
19
  Requires-Dist: pyjwt>=2.10.1
20
20
  Requires-Dist: isodate>=0.6.1
21
21
  Requires-Dist: azure-core>=1.30.0
@@ -10,6 +10,15 @@ from .middleware_set import Middleware
10
10
  from .rest_channel_service_client_factory import RestChannelServiceClientFactory
11
11
  from .turn_context import TurnContext
12
12
 
13
+ # HTTP abstractions
14
+ from .http import (
15
+ HttpRequestProtocol,
16
+ HttpResponse,
17
+ HttpResponseFactory,
18
+ HttpAdapterBase,
19
+ ChannelServiceRoutes,
20
+ )
21
+
13
22
  # Application Style
14
23
  from .app._type_defs import RouteHandler, RouteSelector, StateT
15
24
  from .app.agent_application import AgentApplication
@@ -20,6 +29,13 @@ from .app.query import Query
20
29
  from .app._routes import _Route, _RouteList, RouteRank
21
30
  from .app.typing_indicator import TypingIndicator
22
31
 
32
+ # App Streaming
33
+ from .app.streaming import (
34
+ Citation,
35
+ CitationUtil,
36
+ StreamingResponse,
37
+ )
38
+
23
39
  # App Auth
24
40
  from .app.oauth import (
25
41
  Authorization,
@@ -99,6 +115,11 @@ __all__ = [
99
115
  "Middleware",
100
116
  "RestChannelServiceClientFactory",
101
117
  "TurnContext",
118
+ "HttpRequestProtocol",
119
+ "HttpResponse",
120
+ "HttpResponseFactory",
121
+ "HttpAdapterBase",
122
+ "ChannelServiceRoutes",
102
123
  "AgentApplication",
103
124
  "ApplicationError",
104
125
  "ApplicationOptions",
@@ -108,6 +129,9 @@ __all__ = [
108
129
  "Route",
109
130
  "RouteHandler",
110
131
  "TypingIndicator",
132
+ "Citation",
133
+ "CitationUtil",
134
+ "StreamingResponse",
111
135
  "ConversationState",
112
136
  "state",
113
137
  "State",
@@ -0,0 +1,14 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ """Streaming response utilities."""
5
+
6
+ from .citation import Citation
7
+ from .citation_util import CitationUtil
8
+ from .streaming_response import StreamingResponse
9
+
10
+ __all__ = [
11
+ "Citation",
12
+ "CitationUtil",
13
+ "StreamingResponse",
14
+ ]
@@ -0,0 +1,22 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ from typing import Optional
5
+ from dataclasses import dataclass
6
+
7
+
8
+ @dataclass
9
+ class Citation:
10
+ """Citations returned by the model."""
11
+
12
+ content: str
13
+ """The content of the citation."""
14
+
15
+ title: Optional[str] = None
16
+ """The title of the citation."""
17
+
18
+ url: Optional[str] = None
19
+ """The URL of the citation."""
20
+
21
+ filepath: Optional[str] = None
22
+ """The filepath of the document."""
@@ -0,0 +1,85 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ import re
5
+ from typing import List, Optional
6
+
7
+ from microsoft_agents.activity import ClientCitation
8
+
9
+
10
+ class CitationUtil:
11
+ """Utility functions for manipulating text and citations."""
12
+
13
+ @staticmethod
14
+ def snippet(text: str, max_length: int) -> str:
15
+ """
16
+ Clips the text to a maximum length in case it exceeds the limit.
17
+
18
+ Args:
19
+ text: The text to clip.
20
+ max_length: The maximum length of the text to return, cutting off the last whole word.
21
+
22
+ Returns:
23
+ The modified text
24
+ """
25
+ if len(text) <= max_length:
26
+ return text
27
+
28
+ snippet = text[:max_length]
29
+ snippet = snippet[: min(len(snippet), snippet.rfind(" "))]
30
+ snippet += "..."
31
+ return snippet
32
+
33
+ @staticmethod
34
+ def format_citations_response(text: str) -> str:
35
+ """
36
+ Convert citation tags `[doc(s)n]` to `[n]` where n is a number.
37
+
38
+ Args:
39
+ text: The text to format.
40
+
41
+ Returns:
42
+ The formatted text.
43
+ """
44
+ return re.sub(r"\[docs?(\d+)\]", r"[\1]", text, flags=re.IGNORECASE)
45
+
46
+ @staticmethod
47
+ def get_used_citations(
48
+ text: str, citations: List[ClientCitation]
49
+ ) -> Optional[List[ClientCitation]]:
50
+ """
51
+ Get the citations used in the text. This will remove any citations that are
52
+ included in the citations array from the response but not referenced in the text.
53
+
54
+ Args:
55
+ text: The text to search for citation references, i.e. [1], [2], etc.
56
+ citations: The list of citations to search for.
57
+
58
+ Returns:
59
+ The list of citations used in the text.
60
+ """
61
+ regex = re.compile(r"\[(\d+)\]", re.IGNORECASE)
62
+ matches = regex.findall(text)
63
+
64
+ if not matches:
65
+ return None
66
+
67
+ # Remove duplicates
68
+ filtered_matches = set(matches)
69
+
70
+ # Add citations
71
+ used_citations = []
72
+ for match in filtered_matches:
73
+ citation_ref = f"[{match}]"
74
+ found = next(
75
+ (
76
+ citation
77
+ for citation in citations
78
+ if f"[{citation.position}]" == citation_ref
79
+ ),
80
+ None,
81
+ )
82
+ if found:
83
+ used_citations.append(found)
84
+
85
+ return used_citations if used_citations else None
@@ -0,0 +1,411 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ import asyncio
5
+ import logging
6
+ from typing import List, Optional, Callable, Literal, TYPE_CHECKING
7
+
8
+ from microsoft_agents.activity import (
9
+ Activity,
10
+ Entity,
11
+ Attachment,
12
+ Channels,
13
+ ClientCitation,
14
+ DeliveryModes,
15
+ SensitivityUsageInfo,
16
+ )
17
+
18
+ if TYPE_CHECKING:
19
+ from microsoft_agents.hosting.core.turn_context import TurnContext
20
+
21
+ from .citation import Citation
22
+ from .citation_util import CitationUtil
23
+
24
+ logger = logging.getLogger(__name__)
25
+
26
+
27
+ class StreamingResponse:
28
+ """
29
+ A helper class for streaming responses to the client.
30
+
31
+ This class is used to send a series of updates to the client in a single response.
32
+ The expected sequence of calls is:
33
+
34
+ `queue_informative_update()`, `queue_text_chunk()`, `queue_text_chunk()`, ..., `end_stream()`.
35
+
36
+ Once `end_stream()` is called, the stream is considered ended and no further updates can be sent.
37
+ """
38
+
39
+ def __init__(self, context: "TurnContext"):
40
+ """
41
+ Creates a new StreamingResponse instance.
42
+
43
+ Args:
44
+ context: Context for the current turn of conversation with the user.
45
+ """
46
+ self._context = context
47
+ self._sequence_number = 1
48
+ self._stream_id: Optional[str] = None
49
+ self._message = ""
50
+ self._attachments: Optional[List[Attachment]] = None
51
+ self._ended = False
52
+ self._cancelled = False
53
+
54
+ # Queue for outgoing activities
55
+ self._queue: List[Callable[[], Activity]] = []
56
+ self._queue_sync: Optional[asyncio.Task] = None
57
+ self._chunk_queued = False
58
+
59
+ # Powered by AI feature flags
60
+ self._enable_feedback_loop = False
61
+ self._feedback_loop_type: Optional[Literal["default", "custom"]] = None
62
+ self._enable_generated_by_ai_label = False
63
+ self._citations: Optional[List[ClientCitation]] = []
64
+ self._sensitivity_label: Optional[SensitivityUsageInfo] = None
65
+
66
+ # Channel information
67
+ self._is_streaming_channel: bool = False
68
+ self._channel_id: Channels = None
69
+ self._interval: float = 0.1 # Default interval for sending updates
70
+ self._set_defaults(context)
71
+
72
+ @property
73
+ def stream_id(self) -> Optional[str]:
74
+ """
75
+ Gets the stream ID of the current response.
76
+ Assigned after the initial update is sent.
77
+ """
78
+ return self._stream_id
79
+
80
+ @property
81
+ def citations(self) -> Optional[List[ClientCitation]]:
82
+ """Gets the citations of the current response."""
83
+ return self._citations
84
+
85
+ @property
86
+ def updates_sent(self) -> int:
87
+ """Gets the number of updates sent for the stream."""
88
+ return self._sequence_number - 1
89
+
90
+ def queue_informative_update(self, text: str) -> None:
91
+ """
92
+ Queues an informative update to be sent to the client.
93
+
94
+ Args:
95
+ text: Text of the update to send.
96
+ """
97
+ if not self._is_streaming_channel:
98
+ return
99
+
100
+ if self._ended:
101
+ raise RuntimeError("The stream has already ended.")
102
+
103
+ # Queue a typing activity
104
+ def create_activity():
105
+ activity = Activity(
106
+ type="typing",
107
+ text=text,
108
+ entities=[
109
+ Entity(
110
+ type="streaminfo",
111
+ stream_type="informative",
112
+ stream_sequence=self._sequence_number,
113
+ )
114
+ ],
115
+ )
116
+ self._sequence_number += 1
117
+ return activity
118
+
119
+ self._queue_activity(create_activity)
120
+
121
+ def queue_text_chunk(
122
+ self, text: str, citations: Optional[List[Citation]] = None
123
+ ) -> None:
124
+ """
125
+ Queues a chunk of partial message text to be sent to the client.
126
+
127
+ The text will be sent as quickly as possible to the client.
128
+ Chunks may be combined before delivery to the client.
129
+
130
+ Args:
131
+ text: Partial text of the message to send.
132
+ citations: Citations to be included in the message.
133
+ """
134
+ if self._cancelled:
135
+ return
136
+ if self._ended:
137
+ raise RuntimeError("The stream has already ended.")
138
+
139
+ # Update full message text
140
+ self._message += text
141
+
142
+ # If there are citations, modify the content so that the sources are numbers instead of [doc1], [doc2], etc.
143
+ self._message = CitationUtil.format_citations_response(self._message)
144
+
145
+ # Queue the next chunk
146
+ self._queue_next_chunk()
147
+
148
+ async def end_stream(self) -> None:
149
+ """
150
+ Ends the stream by sending the final message to the client.
151
+ """
152
+ if self._ended:
153
+ raise RuntimeError("The stream has already ended.")
154
+
155
+ # Queue final message
156
+ self._ended = True
157
+ self._queue_next_chunk()
158
+
159
+ # Wait for the queue to drain
160
+ await self.wait_for_queue()
161
+
162
+ def set_attachments(self, attachments: List[Attachment]) -> None:
163
+ """
164
+ Sets the attachments to attach to the final chunk.
165
+
166
+ Args:
167
+ attachments: List of attachments.
168
+ """
169
+ self._attachments = attachments
170
+
171
+ def set_sensitivity_label(self, sensitivity_label: SensitivityUsageInfo) -> None:
172
+ """
173
+ Sets the sensitivity label to attach to the final chunk.
174
+
175
+ Args:
176
+ sensitivity_label: The sensitivity label.
177
+ """
178
+ self._sensitivity_label = sensitivity_label
179
+
180
+ def set_citations(self, citations: List[Citation]) -> None:
181
+ """
182
+ Sets the citations for the full message.
183
+
184
+ Args:
185
+ citations: Citations to be included in the message.
186
+ """
187
+ if citations:
188
+ if not self._citations:
189
+ self._citations = []
190
+
191
+ curr_pos = len(self._citations)
192
+
193
+ for citation in citations:
194
+ client_citation = ClientCitation(
195
+ type="Claim",
196
+ position=curr_pos + 1,
197
+ appearance={
198
+ "type": "DigitalDocument",
199
+ "name": citation.title or f"Document #{curr_pos + 1}",
200
+ "abstract": CitationUtil.snippet(citation.content, 477),
201
+ },
202
+ )
203
+ curr_pos += 1
204
+ self._citations.append(client_citation)
205
+
206
+ def set_feedback_loop(self, enable_feedback_loop: bool) -> None:
207
+ """
208
+ Sets the Feedback Loop in Teams that allows a user to
209
+ give thumbs up or down to a response.
210
+ Default is False.
211
+
212
+ Args:
213
+ enable_feedback_loop: If true, the feedback loop is enabled.
214
+ """
215
+ self._enable_feedback_loop = enable_feedback_loop
216
+
217
+ def set_feedback_loop_type(
218
+ self, feedback_loop_type: Literal["default", "custom"]
219
+ ) -> None:
220
+ """
221
+ Sets the type of UI to use for the feedback loop.
222
+
223
+ Args:
224
+ feedback_loop_type: The type of the feedback loop.
225
+ """
226
+ self._feedback_loop_type = feedback_loop_type
227
+
228
+ def set_generated_by_ai_label(self, enable_generated_by_ai_label: bool) -> None:
229
+ """
230
+ Sets the Generated by AI label in Teams.
231
+ Default is False.
232
+
233
+ Args:
234
+ enable_generated_by_ai_label: If true, the label is added.
235
+ """
236
+ self._enable_generated_by_ai_label = enable_generated_by_ai_label
237
+
238
+ def get_message(self) -> str:
239
+ """
240
+ Returns the most recently streamed message.
241
+ """
242
+ return self._message
243
+
244
+ async def wait_for_queue(self) -> None:
245
+ """
246
+ Waits for the outgoing activity queue to be empty.
247
+ """
248
+ if self._queue_sync:
249
+ await self._queue_sync
250
+
251
+ def _set_defaults(self, context: "TurnContext"):
252
+ if Channels.ms_teams == context.activity.channel_id.channel:
253
+ self._is_streaming_channel = True
254
+ self._interval = 1.0
255
+ elif Channels.direct_line == context.activity.channel_id.channel:
256
+ self._is_streaming_channel = True
257
+ self._interval = 0.5
258
+ elif context.activity.delivery_mode == DeliveryModes.stream:
259
+ self._is_streaming_channel = True
260
+ self._interval = 0.1
261
+
262
+ self._channel_id = context.activity.channel_id
263
+
264
+ def _queue_next_chunk(self) -> None:
265
+ """
266
+ Queues the next chunk of text to be sent to the client.
267
+ """
268
+ # Are we already waiting to send a chunk?
269
+ if self._chunk_queued:
270
+ return
271
+
272
+ # Queue a chunk of text to be sent
273
+ self._chunk_queued = True
274
+
275
+ def create_activity():
276
+ self._chunk_queued = False
277
+ if self._ended:
278
+ # Send final message
279
+ activity = Activity(
280
+ type="message",
281
+ text=self._message or "end stream response",
282
+ attachments=self._attachments or [],
283
+ entities=[
284
+ Entity(
285
+ type="streaminfo",
286
+ stream_id=self._stream_id,
287
+ stream_type="final",
288
+ stream_sequence=self._sequence_number,
289
+ )
290
+ ],
291
+ )
292
+ elif self._is_streaming_channel:
293
+ # Send typing activity
294
+ activity = Activity(
295
+ type="typing",
296
+ text=self._message,
297
+ entities=[
298
+ Entity(
299
+ type="streaminfo",
300
+ stream_type="streaming",
301
+ stream_sequence=self._sequence_number,
302
+ )
303
+ ],
304
+ )
305
+ else:
306
+ return
307
+ self._sequence_number += 1
308
+ return activity
309
+
310
+ self._queue_activity(create_activity)
311
+
312
+ def _queue_activity(self, factory: Callable[[], Activity]) -> None:
313
+ """
314
+ Queues an activity to be sent to the client.
315
+ """
316
+ self._queue.append(factory)
317
+
318
+ # If there's no sync in progress, start one
319
+ if not self._queue_sync:
320
+ self._queue_sync = asyncio.create_task(self._drain_queue())
321
+
322
+ async def _drain_queue(self) -> None:
323
+ """
324
+ Sends any queued activities to the client until the queue is empty.
325
+ """
326
+ try:
327
+ logger.debug(f"Draining queue with {len(self._queue)} activities.")
328
+ while self._queue:
329
+ factory = self._queue.pop(0)
330
+ activity = factory()
331
+ if activity:
332
+ await self._send_activity(activity)
333
+ except Exception as err:
334
+ if (
335
+ "403" in str(err)
336
+ and self._context.activity.channel_id == Channels.ms_teams
337
+ ):
338
+ logger.warning("Teams channel stopped the stream.")
339
+ self._cancelled = True
340
+ else:
341
+ logger.error(
342
+ f"Error occurred when sending activity while streaming: {err}"
343
+ )
344
+ raise
345
+ finally:
346
+ self._queue_sync = None
347
+
348
+ async def _send_activity(self, activity: Activity) -> None:
349
+ """
350
+ Sends an activity to the client and saves the stream ID returned.
351
+
352
+ Args:
353
+ activity: The activity to send.
354
+ """
355
+
356
+ streaminfo_entity = None
357
+
358
+ if not activity.entities:
359
+ streaminfo_entity = Entity(type="streaminfo")
360
+ activity.entities = [streaminfo_entity]
361
+ else:
362
+ for entity in activity.entities:
363
+ if hasattr(entity, "type") and entity.type == "streaminfo":
364
+ streaminfo_entity = entity
365
+ break
366
+
367
+ if not streaminfo_entity:
368
+ # If no streaminfo entity exists, create one
369
+ streaminfo_entity = Entity(type="streaminfo")
370
+ activity.entities.append(streaminfo_entity)
371
+
372
+ # Set activity ID to the assigned stream ID
373
+ if self._stream_id:
374
+ activity.id = self._stream_id
375
+ streaminfo_entity.stream_id = self._stream_id
376
+
377
+ if self._citations and len(self._citations) > 0 and not self._ended:
378
+ # Filter out the citations unused in content.
379
+ curr_citations = CitationUtil.get_used_citations(
380
+ self._message, self._citations
381
+ )
382
+ if curr_citations:
383
+ activity.entities.append(
384
+ Entity(
385
+ type="https://schema.org/Message",
386
+ schema_type="Message",
387
+ context="https://schema.org",
388
+ id="",
389
+ citation=curr_citations,
390
+ )
391
+ )
392
+
393
+ # Add in Powered by AI feature flags
394
+ if self._ended:
395
+ if self._enable_feedback_loop and self._feedback_loop_type:
396
+ # Add feedback loop to streaminfo entity
397
+ streaminfo_entity.feedback_loop = {"type": self._feedback_loop_type}
398
+ else:
399
+ # Add feedback loop enabled to streaminfo entity
400
+ streaminfo_entity.feedback_loop_enabled = self._enable_feedback_loop
401
+ # Add in Generated by AI
402
+ if self._enable_generated_by_ai_label:
403
+ activity.add_ai_metadata(self._citations, self._sensitivity_label)
404
+
405
+ # Send activity
406
+ response = await self._context.send_activity(activity)
407
+ await asyncio.sleep(self._interval)
408
+
409
+ # Save assigned stream ID
410
+ if not self._stream_id and response:
411
+ self._stream_id = response.id
@@ -341,6 +341,19 @@ class ChannelServiceAdapter(ChannelAdapter, ABC):
341
341
  await connector_client.close()
342
342
  await user_token_client.close()
343
343
 
344
+ def _resolve_if_connector_client_is_needed(self, activity: Activity) -> bool:
345
+ """Determine if a connector client is needed based on the activity's delivery mode and service URL.
346
+
347
+ :param activity: The activity to evaluate.
348
+ :type activity: :class:`microsoft_agents.activity.Activity`
349
+ """
350
+ if activity.delivery_mode in [
351
+ DeliveryModes.expect_replies,
352
+ DeliveryModes.stream,
353
+ ]:
354
+ return False
355
+ return True
356
+
344
357
  async def process_activity(
345
358
  self,
346
359
  claims_identity: ClaimsIdentity,
@@ -403,21 +416,24 @@ class ChannelServiceAdapter(ChannelAdapter, ABC):
403
416
  context.turn_state[self.USER_TOKEN_CLIENT_KEY] = user_token_client
404
417
 
405
418
  # Create the connector client to use for outbound requests.
406
- connector_client: ConnectorClient = (
407
- await self._channel_service_client_factory.create_connector_client(
408
- context,
409
- claims_identity,
410
- activity.service_url,
411
- outgoing_audience,
412
- scopes,
413
- use_anonymous_auth_callback,
419
+ connector_client: Optional[ConnectorClient] = None
420
+ if self._resolve_if_connector_client_is_needed(activity):
421
+ connector_client = (
422
+ await self._channel_service_client_factory.create_connector_client(
423
+ context,
424
+ claims_identity,
425
+ activity.service_url,
426
+ outgoing_audience,
427
+ scopes,
428
+ use_anonymous_auth_callback,
429
+ )
414
430
  )
415
- )
416
- context.turn_state[self._AGENT_CONNECTOR_CLIENT_KEY] = connector_client
431
+ context.turn_state[self._AGENT_CONNECTOR_CLIENT_KEY] = connector_client
417
432
 
418
433
  await self.run_pipeline(context, callback)
419
434
 
420
- await connector_client.close()
435
+ if connector_client:
436
+ await connector_client.close()
421
437
  await user_token_client.close()
422
438
 
423
439
  # If there are any results they will have been left on the TurnContext.
@@ -0,0 +1,17 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ """HTTP abstractions for framework-agnostic adapter implementations."""
5
+
6
+ from ._http_request_protocol import HttpRequestProtocol
7
+ from ._http_response import HttpResponse, HttpResponseFactory
8
+ from ._http_adapter_base import HttpAdapterBase
9
+ from ._channel_service_routes import ChannelServiceRoutes
10
+
11
+ __all__ = [
12
+ "HttpRequestProtocol",
13
+ "HttpResponse",
14
+ "HttpResponseFactory",
15
+ "HttpAdapterBase",
16
+ "ChannelServiceRoutes",
17
+ ]