microsoft-agents-activity 0.4.0.dev18__py3-none-any.whl → 0.5.0__py3-none-any.whl

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.

Potentially problematic release.


This version of microsoft-agents-activity might be problematic. Click here for more details.

Files changed (92) hide show
  1. microsoft_agents/activity/__init__.py +12 -0
  2. microsoft_agents/activity/_channel_id_field_mixin.py +96 -0
  3. microsoft_agents/activity/_load_configuration.py +3 -0
  4. microsoft_agents/activity/_model_utils.py +3 -0
  5. microsoft_agents/activity/_type_aliases.py +3 -0
  6. microsoft_agents/activity/action_types.py +3 -0
  7. microsoft_agents/activity/activity.py +132 -9
  8. microsoft_agents/activity/activity_event_names.py +3 -0
  9. microsoft_agents/activity/activity_importance.py +3 -0
  10. microsoft_agents/activity/activity_types.py +3 -0
  11. microsoft_agents/activity/adaptive_card_invoke_action.py +3 -0
  12. microsoft_agents/activity/adaptive_card_invoke_response.py +3 -0
  13. microsoft_agents/activity/adaptive_card_invoke_value.py +3 -0
  14. microsoft_agents/activity/agents_model.py +3 -0
  15. microsoft_agents/activity/animation_card.py +3 -0
  16. microsoft_agents/activity/attachment.py +3 -0
  17. microsoft_agents/activity/attachment_data.py +3 -0
  18. microsoft_agents/activity/attachment_info.py +3 -0
  19. microsoft_agents/activity/attachment_layout_types.py +3 -0
  20. microsoft_agents/activity/attachment_view.py +3 -0
  21. microsoft_agents/activity/audio_card.py +3 -0
  22. microsoft_agents/activity/basic_card.py +3 -0
  23. microsoft_agents/activity/caller_id_constants.py +3 -0
  24. microsoft_agents/activity/card_action.py +3 -0
  25. microsoft_agents/activity/card_image.py +3 -0
  26. microsoft_agents/activity/channel_account.py +3 -0
  27. microsoft_agents/activity/channel_adapter_protocol.py +3 -0
  28. microsoft_agents/activity/channel_id.py +95 -0
  29. microsoft_agents/activity/channels.py +2 -2
  30. microsoft_agents/activity/contact_relation_update_action_types.py +3 -0
  31. microsoft_agents/activity/conversation_account.py +3 -0
  32. microsoft_agents/activity/conversation_members.py +3 -0
  33. microsoft_agents/activity/conversation_parameters.py +3 -0
  34. microsoft_agents/activity/conversation_reference.py +12 -3
  35. microsoft_agents/activity/conversation_resource_response.py +3 -0
  36. microsoft_agents/activity/conversation_update_types.py +3 -0
  37. microsoft_agents/activity/conversations_result.py +3 -0
  38. microsoft_agents/activity/delivery_modes.py +3 -0
  39. microsoft_agents/activity/end_of_conversation_codes.py +3 -0
  40. microsoft_agents/activity/entity/__init__.py +7 -0
  41. microsoft_agents/activity/entity/entity.py +4 -3
  42. microsoft_agents/activity/entity/entity_types.py +14 -0
  43. microsoft_agents/activity/entity/geo_coordinates.py +9 -3
  44. microsoft_agents/activity/entity/mention.py +5 -2
  45. microsoft_agents/activity/entity/place.py +9 -3
  46. microsoft_agents/activity/entity/product_info.py +20 -0
  47. microsoft_agents/activity/entity/thing.py +9 -3
  48. microsoft_agents/activity/error.py +3 -0
  49. microsoft_agents/activity/error_response.py +3 -0
  50. microsoft_agents/activity/expected_replies.py +3 -0
  51. microsoft_agents/activity/fact.py +3 -0
  52. microsoft_agents/activity/hero_card.py +3 -0
  53. microsoft_agents/activity/inner_http_error.py +3 -0
  54. microsoft_agents/activity/installation_update_action_types.py +3 -0
  55. microsoft_agents/activity/invoke_response.py +3 -0
  56. microsoft_agents/activity/media_card.py +3 -0
  57. microsoft_agents/activity/media_event_value.py +3 -0
  58. microsoft_agents/activity/media_url.py +3 -0
  59. microsoft_agents/activity/message_reaction.py +3 -0
  60. microsoft_agents/activity/message_reaction_types.py +3 -0
  61. microsoft_agents/activity/message_update_types.py +3 -0
  62. microsoft_agents/activity/oauth_card.py +3 -0
  63. microsoft_agents/activity/paged_members_result.py +3 -0
  64. microsoft_agents/activity/receipt_card.py +3 -0
  65. microsoft_agents/activity/receipt_item.py +3 -0
  66. microsoft_agents/activity/resource_response.py +3 -0
  67. microsoft_agents/activity/role_types.py +3 -0
  68. microsoft_agents/activity/semantic_action.py +4 -1
  69. microsoft_agents/activity/semantic_actions_states.py +3 -0
  70. microsoft_agents/activity/sign_in_resource.py +3 -0
  71. microsoft_agents/activity/signin_card.py +3 -0
  72. microsoft_agents/activity/suggested_actions.py +3 -0
  73. microsoft_agents/activity/text_format_types.py +3 -0
  74. microsoft_agents/activity/text_highlight.py +3 -0
  75. microsoft_agents/activity/thumbnail_card.py +3 -0
  76. microsoft_agents/activity/thumbnail_url.py +3 -0
  77. microsoft_agents/activity/token_exchange_invoke_request.py +3 -0
  78. microsoft_agents/activity/token_exchange_invoke_response.py +3 -0
  79. microsoft_agents/activity/token_exchange_resource.py +3 -0
  80. microsoft_agents/activity/token_exchange_state.py +3 -0
  81. microsoft_agents/activity/token_or_sign_in_resource_response.py +21 -0
  82. microsoft_agents/activity/token_post_resource.py +3 -0
  83. microsoft_agents/activity/token_request.py +3 -0
  84. microsoft_agents/activity/transcript.py +3 -0
  85. microsoft_agents/activity/turn_context_protocol.py +3 -0
  86. microsoft_agents/activity/video_card.py +3 -0
  87. microsoft_agents_activity-0.5.0.dist-info/METADATA +173 -0
  88. {microsoft_agents_activity-0.4.0.dev18.dist-info → microsoft_agents_activity-0.5.0.dist-info}/RECORD +91 -85
  89. microsoft_agents_activity-0.5.0.dist-info/licenses/LICENSE +21 -0
  90. microsoft_agents_activity-0.4.0.dev18.dist-info/METADATA +0 -11
  91. {microsoft_agents_activity-0.4.0.dev18.dist-info → microsoft_agents_activity-0.5.0.dist-info}/WHEEL +0 -0
  92. {microsoft_agents_activity-0.4.0.dev18.dist-info → microsoft_agents_activity-0.5.0.dist-info}/top_level.txt +0 -0
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from .action_types import ActionTypes
3
6
  from .activity import Activity
@@ -17,6 +20,8 @@ from .card_action import CardAction
17
20
  from .card_image import CardImage
18
21
  from .channels import Channels
19
22
  from .channel_account import ChannelAccount
23
+ from ._channel_id_field_mixin import _ChannelIdFieldMixin
24
+ from .channel_id import ChannelId
20
25
  from .conversation_account import ConversationAccount
21
26
  from .conversation_members import ConversationMembers
22
27
  from .conversation_parameters import ConversationParameters
@@ -26,6 +31,7 @@ from .conversations_result import ConversationsResult
26
31
  from .expected_replies import ExpectedReplies
27
32
  from .entity import (
28
33
  Entity,
34
+ EntityTypes,
29
35
  AIEntity,
30
36
  ClientCitation,
31
37
  ClientCitationAppearance,
@@ -36,6 +42,7 @@ from .entity import (
36
42
  SensitivityPattern,
37
43
  GeoCoordinates,
38
44
  Place,
45
+ ProductInfo,
39
46
  Thing,
40
47
  )
41
48
  from .error import Error
@@ -62,6 +69,7 @@ from .thumbnail_url import ThumbnailUrl
62
69
  from .token_exchange_invoke_request import TokenExchangeInvokeRequest
63
70
  from .token_exchange_invoke_response import TokenExchangeInvokeResponse
64
71
  from .token_exchange_state import TokenExchangeState
72
+ from .token_or_sign_in_resource_response import TokenOrSignInResourceResponse
65
73
  from .token_request import TokenRequest
66
74
  from .token_response import TokenResponse
67
75
  from .token_status import TokenStatus
@@ -114,6 +122,8 @@ __all__ = [
114
122
  "CardImage",
115
123
  "Channels",
116
124
  "ChannelAccount",
125
+ "ChannelId",
126
+ "_ChannelIdFieldMixin",
117
127
  "ConversationAccount",
118
128
  "ConversationMembers",
119
129
  "ConversationParameters",
@@ -144,6 +154,7 @@ __all__ = [
144
154
  "OAuthCard",
145
155
  "PagedMembersResult",
146
156
  "Place",
157
+ "ProductInfo",
147
158
  "ReceiptCard",
148
159
  "ReceiptItem",
149
160
  "ResourceResponse",
@@ -185,4 +196,5 @@ __all__ = [
185
196
  "load_configuration_from_env",
186
197
  "ChannelAdapterProtocol",
187
198
  "TurnContextProtocol",
199
+ "TokenOrSignInResourceResponse",
188
200
  ]
@@ -0,0 +1,96 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ from __future__ import annotations
5
+
6
+ import logging
7
+ from typing import Optional, Any
8
+
9
+ from pydantic import (
10
+ ModelWrapValidatorHandler,
11
+ SerializerFunctionWrapHandler,
12
+ computed_field,
13
+ model_validator,
14
+ model_serializer,
15
+ )
16
+
17
+ from .channel_id import ChannelId
18
+
19
+ logger = logging.getLogger(__name__)
20
+
21
+
22
+ # can be generalized in the future, if needed
23
+ class _ChannelIdFieldMixin:
24
+ """A mixin to add a computed field channel_id of type ChannelId to a Pydantic model."""
25
+
26
+ _channel_id: Optional[ChannelId] = None
27
+
28
+ # required to define the setter below
29
+ @computed_field(return_type=Optional[ChannelId], alias="channelId")
30
+ @property
31
+ def channel_id(self) -> Optional[ChannelId]:
32
+ """Gets the _channel_id field"""
33
+ return self._channel_id
34
+
35
+ # necessary for backward compatibility
36
+ # previously, channel_id was directly assigned with strings
37
+ @channel_id.setter
38
+ def channel_id(self, value: Any):
39
+ """Sets the channel_id after validating it as a ChannelId model."""
40
+ if isinstance(value, ChannelId):
41
+ self._channel_id = value
42
+ elif isinstance(value, str):
43
+ self._channel_id = ChannelId(value)
44
+ else:
45
+ raise ValueError(
46
+ f"Invalid type for channel_id: {type(value)}. "
47
+ "Expected ChannelId or str."
48
+ )
49
+
50
+ def _set_validated_channel_id(self, data: Any) -> None:
51
+ """Sets the channel_id after validating it as a ChannelId model."""
52
+ if "channelId" in data:
53
+ self.channel_id = data["channelId"]
54
+ elif "channel_id" in data:
55
+ self.channel_id = data["channel_id"]
56
+
57
+ @model_validator(mode="wrap")
58
+ @classmethod
59
+ def _validate_channel_id(
60
+ cls, data: Any, handler: ModelWrapValidatorHandler
61
+ ) -> _ChannelIdFieldMixin:
62
+ """Validate the _channel_id field after model initialization.
63
+
64
+ :return: The model instance itself.
65
+ """
66
+ try:
67
+ model = handler(data)
68
+ model._set_validated_channel_id(data)
69
+ return model
70
+ except Exception:
71
+ logging.error("Model %s failed to validate with data %s", cls, data)
72
+ raise
73
+
74
+ def _remove_serialized_unset_channel_id(
75
+ self, serialized: dict[str, object]
76
+ ) -> None:
77
+ """Remove the _channel_id field if it is not set."""
78
+ if not self._channel_id:
79
+ if "channelId" in serialized:
80
+ del serialized["channelId"]
81
+ elif "channel_id" in serialized:
82
+ del serialized["channel_id"]
83
+
84
+ @model_serializer(mode="wrap")
85
+ def _serialize_channel_id(
86
+ self, handler: SerializerFunctionWrapHandler
87
+ ) -> dict[str, object]:
88
+ """Serialize the model using Pydantic's standard serialization.
89
+
90
+ :param handler: The serialization handler provided by Pydantic.
91
+ :return: A dictionary representing the serialized model.
92
+ """
93
+ serialized = handler(self)
94
+ if self: # serialization can be called with None
95
+ self._remove_serialized_unset_channel_id(serialized)
96
+ return serialized
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from typing import Any
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from abc import ABC
2
5
  from typing import Any, Callable
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from typing import Annotated
2
5
  from pydantic import StringConstraints
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,7 +1,24 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ from __future__ import annotations
5
+
6
+ import logging
1
7
  from copy import copy
2
8
  from datetime import datetime, timezone
3
- from typing import Optional
4
- from pydantic import Field, SerializeAsAny
9
+ from typing import Optional, Any
10
+
11
+ from pydantic import (
12
+ Field,
13
+ SerializeAsAny,
14
+ model_serializer,
15
+ model_validator,
16
+ SerializerFunctionWrapHandler,
17
+ ModelWrapValidatorHandler,
18
+ computed_field,
19
+ ValidationError,
20
+ )
21
+
5
22
  from .activity_types import ActivityTypes
6
23
  from .channel_account import ChannelAccount
7
24
  from .conversation_account import ConversationAccount
@@ -11,9 +28,11 @@ from .suggested_actions import SuggestedActions
11
28
  from .attachment import Attachment
12
29
  from .entity import (
13
30
  Entity,
31
+ EntityTypes,
14
32
  Mention,
15
33
  AIEntity,
16
34
  ClientCitation,
35
+ ProductInfo,
17
36
  SensitivityUsageInfo,
18
37
  )
19
38
  from .conversation_reference import ConversationReference
@@ -21,12 +40,16 @@ from .text_highlight import TextHighlight
21
40
  from .semantic_action import SemanticAction
22
41
  from .agents_model import AgentsModel
23
42
  from .role_types import RoleTypes
43
+ from ._channel_id_field_mixin import _ChannelIdFieldMixin
44
+ from .channel_id import ChannelId
24
45
  from ._model_utils import pick_model, SkipNone
25
46
  from ._type_aliases import NonEmptyString
26
47
 
48
+ logger = logging.getLogger(__name__)
49
+
27
50
 
28
51
  # TODO: A2A Agent 2 is responding with None as id, had to mark it as optional (investigate)
29
- class Activity(AgentsModel):
52
+ class Activity(AgentsModel, _ChannelIdFieldMixin):
30
53
  """An Activity is the basic communication type for the protocol.
31
54
 
32
55
  :param type: Contains the activity type. Possible values include:
@@ -47,8 +70,8 @@ class Activity(AgentsModel):
47
70
  :type local_timezone: str
48
71
  :param service_url: Contains the URL that specifies the channel's service endpoint. Set by the channel.
49
72
  :type service_url: str
50
- :param channel_id: Contains an ID that uniquely identifies the channel. Set by the channel.
51
- :type channel_id: str
73
+ :param channel_id: Contains an ID that uniquely identifies the channel (and possibly the sub-channel). Set by the channel.
74
+ :type channel_id: ~microsoft_agents.activity.ChannelId
52
75
  :param from_property: Identifies the sender of the message.
53
76
  :type from_property: ~microsoft_agents.activity.ChannelAccount
54
77
  :param conversation: Identifies the conversation to which the activity belongs.
@@ -133,7 +156,6 @@ class Activity(AgentsModel):
133
156
  local_timestamp: datetime = None
134
157
  local_timezone: NonEmptyString = None
135
158
  service_url: NonEmptyString = None
136
- channel_id: NonEmptyString = None
137
159
  from_property: ChannelAccount = Field(None, alias="from")
138
160
  conversation: ConversationAccount = None
139
161
  recipient: ChannelAccount = None
@@ -170,6 +192,99 @@ class Activity(AgentsModel):
170
192
  semantic_action: SemanticAction = None
171
193
  caller_id: NonEmptyString = None
172
194
 
195
+ @model_validator(mode="wrap")
196
+ @classmethod
197
+ def _validate_channel_id(
198
+ cls, data: Any, handler: ModelWrapValidatorHandler[Activity]
199
+ ) -> Activity:
200
+ """Validate the Activity, ensuring consistency between channel_id.sub_channel and productInfo entity.
201
+
202
+ :param data: The input data to validate.
203
+ :param handler: The validation handler provided by Pydantic.
204
+ :return: The validated Activity instance.
205
+ """
206
+ try:
207
+ # run Pydantic's standard validation first
208
+ activity = handler(data)
209
+
210
+ # needed to assign to a computed field
211
+ # needed because we override the mixin validator
212
+ activity._set_validated_channel_id(data)
213
+
214
+ # sync sub_channel with productInfo entity
215
+ product_info = activity.get_product_info_entity()
216
+ if product_info and activity.channel_id:
217
+ if (
218
+ activity.channel_id.sub_channel
219
+ and activity.channel_id.sub_channel != product_info.id
220
+ ):
221
+ raise Exception(
222
+ "Conflict between channel_id.sub_channel and productInfo entity"
223
+ )
224
+ activity.channel_id = ChannelId(
225
+ channel=activity.channel_id.channel,
226
+ sub_channel=product_info.id,
227
+ )
228
+
229
+ return activity
230
+ except ValidationError as exc:
231
+ logger.error("Validation error for Activity: %s", exc, exc_info=True)
232
+ raise
233
+
234
+ @model_serializer(mode="wrap")
235
+ def _serialize_sub_channel_data(
236
+ self, handler: SerializerFunctionWrapHandler
237
+ ) -> dict[str, object]:
238
+ """Serialize the Activity, ensuring consistency between channel_id.sub_channel and productInfo entity.
239
+
240
+ :param handler: The serialization handler provided by Pydantic.
241
+ :return: A dictionary representing the serialized Activity.
242
+ """
243
+
244
+ # run Pydantic's standard serialization first
245
+ serialized = handler(self)
246
+ if not self: # serialization can be called with None
247
+ return serialized
248
+
249
+ # find the ProductInfo entity
250
+ product_info = None
251
+ for i, entity in enumerate(serialized.get("entities") or []):
252
+ if entity.get("type", "") == EntityTypes.PRODUCT_INFO:
253
+ product_info = entity
254
+ break
255
+
256
+ # self.channel_id is the source of truth for serialization
257
+ if self.channel_id and self.channel_id.sub_channel:
258
+ if product_info and product_info.get("id") != self.channel_id.sub_channel:
259
+ raise Exception(
260
+ "Conflict between channel_id.sub_channel and productInfo entity"
261
+ )
262
+ elif not product_info:
263
+ if not serialized.get("entities"):
264
+ serialized["entities"] = []
265
+ serialized["entities"].append(
266
+ {
267
+ "type": EntityTypes.PRODUCT_INFO,
268
+ "id": self.channel_id.sub_channel,
269
+ }
270
+ )
271
+
272
+ # simply serialized channelId value in Activity and relatesTo
273
+ if "channelId" in serialized:
274
+ serialized["channelId"] = self.channel_id.channel
275
+ elif "channel_id" in serialized:
276
+ serialized["channel_id"] = self.channel_id.channel
277
+
278
+ elif product_info: # remove productInfo entity if sub_channel is not set
279
+ del serialized["entities"][i]
280
+ if not serialized["entities"]: # after removal above, list may be empty
281
+ del serialized["entities"]
282
+
283
+ # necessary due to computed_field serialization
284
+ self._remove_serialized_unset_channel_id(serialized)
285
+
286
+ return serialized
287
+
173
288
  def apply_conversation_reference(
174
289
  self, reference: ConversationReference, is_incoming: bool = False
175
290
  ):
@@ -528,6 +643,14 @@ class Activity(AgentsModel):
528
643
  service_url=self.service_url,
529
644
  )
530
645
 
646
+ def get_product_info_entity(self) -> Optional[ProductInfo]:
647
+ if not self.entities:
648
+ return None
649
+ target = EntityTypes.PRODUCT_INFO.lower()
650
+ # validated entities can be Entity, and that prevents us from
651
+ # making assumptions about the casing of the 'type' attribute
652
+ return next(filter(lambda e: e.type.lower() == target, self.entities), None)
653
+
531
654
  def get_mentions(self) -> list[Mention]:
532
655
  """
533
656
  Resolves the mentions from the entities of this activity.
@@ -540,7 +663,7 @@ class Activity(AgentsModel):
540
663
  """
541
664
  if not self.entities:
542
665
  return []
543
- return [x for x in self.entities if x.type.lower() == "mention"]
666
+ return [x for x in self.entities if x.type.lower() == EntityTypes.MENTION]
544
667
 
545
668
  def get_reply_conversation_reference(
546
669
  self, reply: ResourceResponse
@@ -661,7 +784,7 @@ class Activity(AgentsModel):
661
784
  return self.recipient.agentic_app_id
662
785
 
663
786
  def get_agentic_user(self) -> Optional[str]:
664
- """Gets the agentic user (UPN) from the context if it's an agentic request."""
787
+ """Gets the agentic user (agenticUserId) from the context if it's an agentic request."""
665
788
  if not self.is_agentic_request() or not self.recipient:
666
789
  return None
667
- return self.recipient.id
790
+ return self.recipient.agentic_user_id
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from ._type_aliases import NonEmptyString
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from ._type_aliases import NonEmptyString
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .adaptive_card_invoke_action import AdaptiveCardInvokeAction
2
5
  from .token_exchange_invoke_request import TokenExchangeInvokeRequest
3
6
  from .agents_model import AgentsModel
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from __future__ import annotations
2
5
 
3
6
  from pydantic import BaseModel, ConfigDict
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .thumbnail_url import ThumbnailUrl
2
5
  from .media_url import MediaUrl
3
6
  from .card_action import CardAction
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from ._type_aliases import NonEmptyString
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from ._type_aliases import NonEmptyString
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .attachment_view import AttachmentView
2
5
  from .agents_model import AgentsModel
3
6
  from ._type_aliases import NonEmptyString
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from ._type_aliases import NonEmptyString
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from .thumbnail_url import ThumbnailUrl
3
6
  from .media_url import MediaUrl
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .agents_model import AgentsModel
2
5
  from .card_image import CardImage
3
6
  from .card_action import CardAction
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from enum import Enum
2
5
 
3
6
 
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from typing import Optional
2
5
  from .agents_model import AgentsModel
3
6
  from ._type_aliases import NonEmptyString
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from .card_action import CardAction
2
5
  from .agents_model import AgentsModel
3
6
  from ._type_aliases import NonEmptyString
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from typing import Any
2
5
 
3
6
  from pydantic import ConfigDict
@@ -1,3 +1,6 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
1
4
  from abc import abstractmethod
2
5
  from typing import Protocol, List, Callable, Awaitable, Optional
3
6
 
@@ -0,0 +1,95 @@
1
+ # Copyright (c) Microsoft Corporation. All rights reserved.
2
+ # Licensed under the MIT License.
3
+
4
+ from __future__ import annotations
5
+
6
+ from typing import Optional, Any
7
+
8
+ from pydantic_core import CoreSchema, core_schema
9
+ from pydantic import GetCoreSchemaHandler
10
+
11
+
12
+ class ChannelId(str):
13
+ """A ChannelId represents a channel and optional sub-channel in the format 'channel:sub_channel'."""
14
+
15
+ def __init__(
16
+ self,
17
+ value: Optional[str] = None,
18
+ *,
19
+ channel: Optional[str] = None,
20
+ sub_channel: Optional[str] = None,
21
+ ) -> None:
22
+ """Initialize a ChannelId instance.
23
+
24
+ :param value: The full channel ID string in the format 'channel:sub_channel'. Must be provided if channel is not provided.
25
+ :param channel: The main channel string. Must be provided if value is not provided.
26
+ :param sub_channel: The sub-channel string.
27
+ :raises ValueError: If the input parameters are invalid. value and channel cannot both be provided.
28
+ """
29
+ super().__init__()
30
+ if not channel:
31
+ split = self.strip().split(":", 1)
32
+ self._channel = split[0].strip()
33
+ self._sub_channel = split[1].strip() if len(split) == 2 else None
34
+ else:
35
+ self._channel = channel
36
+ self._sub_channel = sub_channel
37
+
38
+ def __new__(
39
+ cls,
40
+ value: Optional[str] = None,
41
+ *,
42
+ channel: Optional[str] = None,
43
+ sub_channel: Optional[str] = None,
44
+ ) -> ChannelId:
45
+ """Create a new ChannelId instance.
46
+
47
+ :param value: The full channel ID string in the format 'channel:sub_channel'. Must be provided if channel is not provided.
48
+ :param channel: The main channel string. Must be provided if value is not provided. Must not contain ':', as it delimits channels and sub channels.
49
+ :param sub_channel: The sub-channel string.
50
+ :return: A new ChannelId instance.
51
+ :raises ValueError: If the input parameters are invalid. value and channel cannot both be provided.
52
+ """
53
+ if isinstance(value, str):
54
+ if channel or sub_channel:
55
+ raise ValueError(
56
+ "If value is provided, channel and sub_channel must be None"
57
+ )
58
+
59
+ value = value.strip()
60
+ if value:
61
+ return str.__new__(cls, value)
62
+ raise TypeError("value must be a non empty string if provided")
63
+ else:
64
+ if (
65
+ not isinstance(channel, str)
66
+ or len(channel.strip()) == 0
67
+ or ":" in channel
68
+ ):
69
+ raise TypeError(
70
+ "channel must be a non empty string, and must not contain the ':' character"
71
+ )
72
+ if sub_channel is not None and (not isinstance(sub_channel, str)):
73
+ raise TypeError("sub_channel must be a string if provided")
74
+ channel = channel.strip()
75
+ sub_channel = sub_channel.strip() if sub_channel else None
76
+ if sub_channel:
77
+ return str.__new__(cls, f"{channel}:{sub_channel}")
78
+ return str.__new__(cls, channel)
79
+
80
+ @property
81
+ def channel(self) -> str:
82
+ """The main channel, e.g. 'email' in 'email:work'."""
83
+ return self._channel # type: ignore[return-value]
84
+
85
+ @property
86
+ def sub_channel(self) -> Optional[str]:
87
+ """The sub-channel, e.g. 'work' in 'email:work'. May be None."""
88
+ return self._sub_channel
89
+
90
+ # https://docs.pydantic.dev/dev/concepts/types/#customizing-validation-with-__get_pydantic_core_schema__
91
+ @classmethod
92
+ def __get_pydantic_core_schema__(
93
+ cls, source_type: Any, handler: GetCoreSchemaHandler
94
+ ) -> CoreSchema:
95
+ return core_schema.no_info_after_validator_function(cls, handler(str))
@@ -74,7 +74,7 @@ class Channels(str, Enum):
74
74
 
75
75
  Args:
76
76
  channel_id (str): The Channel to check the if Suggested Actions are supported in.
77
- button_cnt (int, optional): Defaults to 100. The number of Suggested Actions to check for the Channel.
77
+ button_cnt (int, Optional): Defaults to 100. The number of Suggested Actions to check for the Channel.
78
78
 
79
79
  Returns:
80
80
  bool: True if the Channel supports the button_cnt total Suggested Actions, False if the Channel does not
@@ -107,7 +107,7 @@ class Channels(str, Enum):
107
107
 
108
108
  Args:
109
109
  channel_id (str): The Channel to check if the Card Actions are supported in.
110
- button_cnt (int, optional): Defaults to 100. The number of Card Actions to check for the Channel.
110
+ button_cnt (int, Optional): Defaults to 100. The number of Card Actions to check for the Channel.
111
111
 
112
112
  Returns:
113
113
  bool: True if the Channel supports the button_cnt total Card Actions, False if the Channel does not support