Skip to content

Commit 3ba4119

Browse files
Merge branch 'main' into fix-export-base-client
2 parents 3ea7dde + acdc0de commit 3ba4119

File tree

12 files changed

+134
-32
lines changed

12 files changed

+134
-32
lines changed

.github/workflows/linter.yaml

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ jobs:
1818
with:
1919
python-version-file: .python-version
2020
- name: Install uv
21-
uses: astral-sh/setup-uv@v6
21+
uses: astral-sh/setup-uv@v7
2222
- name: Add uv to PATH
2323
run: |
2424
echo "$HOME/.cargo/bin" >> $GITHUB_PATH

.github/workflows/python-publish.yml

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ jobs:
1515
- uses: actions/checkout@v5
1616

1717
- name: Install uv
18-
uses: astral-sh/setup-uv@v6
18+
uses: astral-sh/setup-uv@v7
1919

2020
- name: "Set up Python"
2121
uses: actions/setup-python@v6
@@ -26,7 +26,7 @@ jobs:
2626
run: uv build
2727

2828
- name: Upload distributions
29-
uses: actions/upload-artifact@v4
29+
uses: actions/upload-artifact@v5
3030
with:
3131
name: release-dists
3232
path: dist/
@@ -40,7 +40,7 @@ jobs:
4040

4141
steps:
4242
- name: Retrieve release distributions
43-
uses: actions/download-artifact@v5
43+
uses: actions/download-artifact@v6
4444
with:
4545
name: release-dists
4646
path: dist/

.github/workflows/unit-tests.yml

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ jobs:
4646
echo "MYSQL_TEST_DSN=mysql+aiomysql://a2a:a2a_password@localhost:3306/a2a_test" >> $GITHUB_ENV
4747
4848
- name: Install uv for Python ${{ matrix.python-version }}
49-
uses: astral-sh/setup-uv@v6
49+
uses: astral-sh/setup-uv@v7
5050
with:
5151
python-version: ${{ matrix.python-version }}
5252
- name: Add uv to PATH

.github/workflows/update-a2a-types.yml

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ jobs:
1818
with:
1919
python-version: '3.10'
2020
- name: Install uv
21-
uses: astral-sh/setup-uv@v6
21+
uses: astral-sh/setup-uv@v7
2222
- name: Configure uv shell
2323
run: echo "$HOME/.cargo/bin" >> $GITHUB_PATH
2424
- name: Install dependencies (datamodel-code-generator)

CHANGELOG.md

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,19 @@
11
# Changelog
22

3+
## [0.3.12](https://github.com/a2aproject/a2a-python/compare/v0.3.11...v0.3.12) (2025-11-12)
4+
5+
6+
### Bug Fixes
7+
8+
* **grpc:** Add `extensions` to `Artifact` converters. ([#523](https://github.com/a2aproject/a2a-python/issues/523)) ([c03129b](https://github.com/a2aproject/a2a-python/commit/c03129b99a663ae1f1ae72f20e4ead7807ede941))
9+
10+
## [0.3.11](https://github.com/a2aproject/a2a-python/compare/v0.3.10...v0.3.11) (2025-11-07)
11+
12+
13+
### Bug Fixes
14+
15+
* add metadata to send message request ([12b4a1d](https://github.com/a2aproject/a2a-python/commit/12b4a1d565a53794f5b55c8bd1728221c906ed41))
16+
317
## [0.3.10](https://github.com/a2aproject/a2a-python/compare/v0.3.9...v0.3.10) (2025-10-21)
418

519

scripts/generate_types.sh

Lines changed: 78 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -4,38 +4,105 @@
44
# Treat unset variables as an error.
55
set -euo pipefail
66

7-
REMOTE_URL="https://raw.githubusercontent.com/a2aproject/A2A/refs/heads/main/specification/json/a2a.json"
7+
# A2A specification version to use
8+
# Can be overridden via environment variable: A2A_SPEC_VERSION=v1.2.0 ./generate_types.sh
9+
# Or via command-line flag: ./generate_types.sh --version v1.2.0 output.py
10+
# Use a specific git tag, branch name, or commit SHA
11+
# Examples: "v1.0.0", "v1.2.0", "main", "abc123def"
12+
A2A_SPEC_VERSION="${A2A_SPEC_VERSION:-v0.3.0}"
13+
14+
# Build URL based on version format
15+
# Tags use /refs/tags/, branches use /refs/heads/, commits use direct ref
16+
build_remote_url() {
17+
local version="$1"
18+
local base_url="https://raw.githubusercontent.com/a2aproject/A2A"
19+
local spec_path="specification/json/a2a.json"
20+
local url_part
21+
22+
if [[ "$version" =~ ^v[0-9]+\.[0-9]+\.[0-9]+$ ]]; then
23+
# Looks like a version tag (v1.0.0, v1.2.3)
24+
url_part="refs/tags/${version}"
25+
elif [[ "$version" =~ ^[0-9a-f]{7,40}$ ]]; then
26+
# Looks like a commit SHA (7+ hex chars)
27+
url_part="${version}"
28+
else
29+
# Assume it's a branch name (main, develop, etc.)
30+
url_part="refs/heads/${version}"
31+
fi
32+
echo "${base_url}/${url_part}/${spec_path}"
33+
}
34+
35+
REMOTE_URL=$(build_remote_url "$A2A_SPEC_VERSION")
836

937
GENERATED_FILE=""
1038
INPUT_FILE=""
1139

1240
# Parse command-line arguments
1341
while [[ $# -gt 0 ]]; do
1442
case "$1" in
15-
--input-file)
16-
INPUT_FILE="$2"
17-
shift 2
18-
;;
19-
*)
20-
GENERATED_FILE="$1"
21-
shift 1
22-
;;
43+
--input-file)
44+
INPUT_FILE="$2"
45+
shift 2
46+
;;
47+
--version)
48+
A2A_SPEC_VERSION="$2"
49+
REMOTE_URL=$(build_remote_url "$A2A_SPEC_VERSION")
50+
shift 2
51+
;;
52+
*)
53+
GENERATED_FILE="$1"
54+
shift 1
55+
;;
2356
esac
2457
done
2558

2659
if [ -z "$GENERATED_FILE" ]; then
27-
echo "Error: Output file path must be provided." >&2
28-
echo "Usage: $0 [--input-file <path>] <output-file-path>"
60+
cat >&2 <<EOF
61+
Error: Output file path must be provided.
62+
Usage: $0 [--input-file <path>] [--version <version>] <output-file-path>
63+
Options:
64+
--input-file <path> Use a local JSON schema file instead of fetching from remote
65+
--version <version> Specify A2A spec version (default: v0.3.0)
66+
Can be a git tag (v1.0.0), branch (main), or commit SHA
67+
Environment variables:
68+
A2A_SPEC_VERSION Override default spec version
69+
Examples:
70+
$0 src/a2a/types.py
71+
$0 --version v1.2.0 src/a2a/types.py
72+
$0 --input-file local/a2a.json src/a2a/types.py
73+
A2A_SPEC_VERSION=main $0 src/a2a/types.py
74+
EOF
2975
exit 1
3076
fi
3177

3278
echo "Running datamodel-codegen..."
3379
declare -a source_args
3480
if [ -n "$INPUT_FILE" ]; then
3581
echo " - Source File: $INPUT_FILE"
82+
if [ ! -f "$INPUT_FILE" ]; then
83+
echo "Error: Input file does not exist: $INPUT_FILE" >&2
84+
exit 1
85+
fi
3686
source_args=("--input" "$INPUT_FILE")
3787
else
88+
echo " - A2A Spec Version: $A2A_SPEC_VERSION"
3889
echo " - Source URL: $REMOTE_URL"
90+
91+
# Validate that the remote URL is accessible
92+
echo " - Validating remote URL..."
93+
if ! curl --fail --silent --head "$REMOTE_URL" >/dev/null 2>&1; then
94+
cat >&2 <<EOF
95+
96+
Error: Unable to access A2A specification at version '$A2A_SPEC_VERSION'
97+
URL: $REMOTE_URL
98+
99+
The version may not exist. Available versions can be found at:
100+
https://github.com/a2aproject/A2A/tags
101+
102+
EOF
103+
exit 1
104+
fi
105+
39106
source_args=("--url" "$REMOTE_URL")
40107
fi
41108
echo " - Output File: $GENERATED_FILE"

src/a2a/client/base_client.py

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
from collections.abc import AsyncIterator
2+
from typing import Any
23

34
from a2a.client.client import (
45
Client,
@@ -47,6 +48,7 @@ async def send_message(
4748
request: Message,
4849
*,
4950
context: ClientCallContext | None = None,
51+
request_metadata: dict[str, Any] | None = None,
5052
) -> AsyncIterator[ClientEvent | Message]:
5153
"""Sends a message to the agent.
5254
@@ -57,6 +59,7 @@ async def send_message(
5759
Args:
5860
request: The message to send to the agent.
5961
context: The client call context.
62+
request_metadata: Extensions Metadata attached to the request.
6063
6164
Yields:
6265
An async iterator of `ClientEvent` or a final `Message` response.
@@ -70,7 +73,9 @@ async def send_message(
7073
else None
7174
),
7275
)
73-
params = MessageSendParams(message=request, configuration=config)
76+
params = MessageSendParams(
77+
message=request, configuration=config, metadata=request_metadata
78+
)
7479

7580
if not self._config.streaming or not self._card.capabilities.streaming:
7681
response = await self._transport.send_message(

src/a2a/client/client.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ async def send_message(
110110
request: Message,
111111
*,
112112
context: ClientCallContext | None = None,
113+
request_metadata: dict[str, Any] | None = None,
113114
) -> AsyncIterator[ClientEvent | Message]:
114115
"""Sends a message to the server.
115116

src/a2a/utils/proto_utils.py

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ def make_dict_serializable(value: Any) -> Any:
5757
Returns:
5858
A serializable value.
5959
"""
60-
if isinstance(value, (str, int, float, bool)) or value is None:
60+
if isinstance(value, str | int | float | bool) or value is None:
6161
return value
6262
if isinstance(value, dict):
6363
return {k: make_dict_serializable(v) for k, v in value.items()}
@@ -140,6 +140,7 @@ def message(cls, message: types.Message | None) -> a2a_pb2.Message | None:
140140
task_id=message.task_id or '',
141141
role=cls.role(message.role),
142142
metadata=cls.metadata(message.metadata),
143+
extensions=message.extensions or [],
143144
)
144145

145146
@classmethod
@@ -239,6 +240,7 @@ def artifact(cls, artifact: types.Artifact) -> a2a_pb2.Artifact:
239240
metadata=cls.metadata(artifact.metadata),
240241
name=artifact.name,
241242
parts=[cls.part(p) for p in artifact.parts],
243+
extensions=artifact.extensions or [],
242244
)
243245

244246
@classmethod
@@ -695,6 +697,7 @@ def artifact(cls, artifact: a2a_pb2.Artifact) -> types.Artifact:
695697
metadata=cls.metadata(artifact.metadata),
696698
name=artifact.name,
697699
parts=[cls.part(p) for p in artifact.parts],
700+
extensions=artifact.extensions or None,
698701
)
699702

700703
@classmethod

src/a2a/utils/task.py

Lines changed: 2 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -83,11 +83,9 @@ def apply_history_length(task: Task, history_length: int | None) -> Task:
8383
A new task object with limited history
8484
"""
8585
# Apply historyLength parameter if specified
86-
if history_length is not None and task.history:
86+
if history_length is not None and history_length > 0 and task.history:
8787
# Limit history to the most recent N messages
88-
limited_history = (
89-
task.history[-history_length:] if history_length > 0 else []
90-
)
88+
limited_history = task.history[-history_length:]
9189
# Create a new task instance with limited history
9290
return task.model_copy(update={'history': limited_history})
9391

0 commit comments

Comments
 (0)