Skip to content

Commit d0f4fed

Browse files
committed
Merge remote-tracking branch 'origin/main' into dotnet_unit2
2 parents 3e53a2a + da6f918 commit d0f4fed

File tree

18 files changed

+288
-166
lines changed

18 files changed

+288
-166
lines changed

.github/workflows/dotnet-build.yml

+38-57
Original file line numberDiff line numberDiff line change
@@ -46,8 +46,9 @@ jobs:
4646
- name: workflows has changes
4747
run: echo "workflows has changes"
4848
if: steps.filter.outputs.workflows == 'true'
49+
4950
build:
50-
name: Dotnet Build
51+
name: Dotnet Build & Test
5152
needs: paths-filter
5253
if: needs.paths-filter.outputs.hasChanges == 'true'
5354
defaults:
@@ -64,35 +65,12 @@ jobs:
6465
- uses: actions/checkout@v4
6566
with:
6667
lfs: true
67-
- name: Set up Python ${{ matrix.python-version }}
68-
uses: actions/setup-python@v5
69-
with:
70-
python-version: ${{ matrix.python-version }}
71-
- name: Install jupyter and ipykernel
72-
run: |
73-
python -m pip install --upgrade pip
74-
python -m pip install jupyter
75-
python -m pip install ipykernel
76-
- name: list available kernels
77-
run: |
78-
python -m jupyter kernelspec list
79-
- uses: astral-sh/setup-uv@v5
80-
with:
81-
enable-cache: true
82-
version: "0.5.18"
83-
- run: uv sync --locked --all-extras
84-
working-directory: ./python
85-
- name: Prepare python venv
86-
run: |
87-
source ${{ github.workspace }}/python/.venv/bin/activate
8868
- name: Setup .NET 8.0
8969
uses: actions/setup-dotnet@v4
9070
with:
9171
dotnet-version: '8.0.x'
9272
- name: Restore dependencies
93-
run: |
94-
# dotnet nuget add source --name dotnet-tool https://pkgs.dev.azure.com/dnceng/public/_packaging/dotnet-tools/nuget/v3/index.json --configfile NuGet.config
95-
dotnet restore -bl
73+
run: dotnet restore -bl
9674
- name: Format check
9775
run: |
9876
echo "Format check"
@@ -104,40 +82,43 @@ jobs:
10482
dotnet build --no-restore --configuration Release -bl /p:SignAssembly=true
10583
- name: Unit Test V1
10684
run: dotnet test --no-build -bl --configuration Release --filter "Category=UnitV1"
107-
- name: Unit Test V2
108-
run: dotnet test --no-build -bl --configuration Release --filter "Category=UnitV2"
109-
110-
grpc-unit-tests:
111-
name: Dotnet Grpc unit tests
112-
needs: paths-filter
113-
if: needs.paths-filter.outputs.hasChanges == 'true'
114-
defaults:
115-
run:
116-
working-directory: dotnet
117-
strategy:
118-
fail-fast: false
119-
matrix:
120-
os: [ ubuntu-latest ]
121-
runs-on: ${{ matrix.os }}
122-
timeout-minutes: 30
123-
steps:
124-
- uses: actions/checkout@v4
85+
- name: Unit Test V2 (With Coverage)
86+
run: dotnet test --no-build -bl --configuration Release --filter "Category=UnitV2" --collect:"XPlat Code Coverage"
87+
- name: Install Dev Certs for GRPC
88+
if: matrix.os == 'ubuntu-latest'
89+
run: dotnet dev-certs https --trust
90+
- name: GRPC Tests (With Coverage)
91+
if: matrix.os == 'ubuntu-latest'
92+
run: dotnet test --no-build -bl --configuration Release --filter "Category=GRPC" --collect:"XPlat Code Coverage"
93+
- name: Generate & Merge Coverage Report
94+
if: matrix.os == 'ubuntu-latest'
95+
run: |
96+
# Install reportgenerator
97+
dotnet tool install -g dotnet-reportgenerator-globaltool || dotnet tool update -g dotnet-reportgenerator-globaltool
98+
# Ensure output directory exists
99+
mkdir -p ${{ github.workspace }}/dotnet/coverage-report
100+
# Merge all coverage reports and generate HTML + XML
101+
reportgenerator \
102+
-reports:${{ github.workspace }}/dotnet/**/TestResults/**/coverage.cobertura.xml \
103+
-targetdir:${{ github.workspace }}/dotnet/coverage-report \
104+
-reporttypes:"Cobertura;Html"
105+
ls -R ${{ github.workspace }}/dotnet/coverage-report
106+
- name: Upload Merged Coverage Report
107+
if: matrix.os == 'ubuntu-latest'
108+
uses: actions/upload-artifact@v4
125109
with:
126-
lfs: true
127-
- name: Setup .NET 8.0
128-
uses: actions/setup-dotnet@v4
110+
name: CodeCoverageReport
111+
path: ${{ github.workspace }}/dotnet/coverage-report/
112+
retention-days: 7
113+
- name: Upload Coverage to Codecov
114+
if: matrix.os == 'ubuntu-latest'
115+
uses: codecov/codecov-action@v5
129116
with:
130-
dotnet-version: '8.0.x'
131-
- name: Install dev certs
132-
run: dotnet --version && dotnet dev-certs https --trust
133-
- name: Restore dependencies
134-
run: |
135-
# dotnet nuget add source --name dotnet-tool https://pkgs.dev.azure.com/dnceng/public/_packaging/dotnet-tools/nuget/v3/index.json --configfile NuGet.config
136-
dotnet restore -bl
137-
- name: Build
138-
run: dotnet build --no-restore --configuration Release -bl /p:SignAssembly=true
139-
- name: GRPC tests
140-
run: dotnet test --no-build -bl --configuration Release --filter "Category=GRPC"
117+
files: ${{ github.workspace }}/dotnet/coverage-report/*.xml
118+
flags: unittests
119+
name: dotnet-codecov
120+
fail_ci_if_error: true
121+
token: ${{ secrets.CODECOV_TOKEN }}
141122

142123
integration-test:
143124
strategy:

README.md

+6-1
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,14 @@
77
[![LinkedIn](https://img.shields.io/badge/LinkedIn-Company?style=flat&logo=linkedin&logoColor=white)](https://www.linkedin.com/company/105812540)
88
[![Discord](https://img.shields.io/badge/discord-chat-green?logo=discord)](https://aka.ms/autogen-discord)
99
[![Documentation](https://img.shields.io/badge/Documentation-AutoGen-blue?logo=read-the-docs)](https://microsoft.github.io/autogen/)
10+
[![Blog](https://img.shields.io/badge/Blog-AutoGen-blue?logo=blogger)](https://devblogs.microsoft.com/autogen/)
1011

1112
</div>
1213

14+
<div align="center" style="background-color: rgba(255, 235, 59, 0.5); padding: 10px; border-radius: 5px; margin: 20px 0;">
15+
<strong>Important:</strong> This is the official project. We are not affiliated with any fork or startup. See our <a href="https://x.com/pyautogen/status/1857264760951296210">statement</a>.
16+
</div>
17+
1318
# AutoGen
1419

1520
**AutoGen** is a framework for creating multi-agent AI applications that can act autonomously or work alongside humans.
@@ -130,7 +135,7 @@ With AutoGen you get to join and contribute to a thriving ecosystem. We host wee
130135

131136
Interested in contributing? See [CONTRIBUTING.md](./CONTRIBUTING.md) for guidelines on how to get started. We welcome contributions of all kinds, including bug fixes, new features, and documentation improvements. Join our community and help us make AutoGen better!
132137

133-
Have questions? Check out our [Frequently Asked Questions (FAQ)](./FAQ.md) for answers to common queries. If you don't find what you're looking for, feel free to ask in our [GitHub Discussions](https://github.com/microsoft/autogen/discussions) or join our [Discord server](https://aka.ms/autogen-discord) for real-time support.
138+
Have questions? Check out our [Frequently Asked Questions (FAQ)](./FAQ.md) for answers to common queries. If you don't find what you're looking for, feel free to ask in our [GitHub Discussions](https://github.com/microsoft/autogen/discussions) or join our [Discord server](https://aka.ms/autogen-discord) for real-time support. You can also read our [blog](https://devblogs.microsoft.com/autogen/) for updates.
134139

135140
## Legal Notices
136141

dotnet/test/Microsoft.AutoGen.Core.Grpc.Tests/Microsoft.AutoGen.Core.Grpc.Tests.csproj

+4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,10 @@
88
</PropertyGroup>
99

1010
<ItemGroup>
11+
<PackageReference Include="coverlet.collector">
12+
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
13+
<PrivateAssets>all</PrivateAssets>
14+
</PackageReference>
1115
<ProjectReference Include="..\..\src\Microsoft.AutoGen\Core\Microsoft.AutoGen.Core.csproj" />
1216
<ProjectReference Include="..\..\src\Microsoft.AutoGen\Core.Grpc\Microsoft.AutoGen.Core.Grpc.csproj" />
1317
<PackageReference Include="Microsoft.Extensions.Hosting" />

dotnet/test/Microsoft.AutoGen.Core.Tests/Microsoft.AutoGen.Core.Tests.csproj

+4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,10 @@
88
</PropertyGroup>
99

1010
<ItemGroup>
11+
<PackageReference Include="coverlet.collector">
12+
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
13+
<PrivateAssets>all</PrivateAssets>
14+
</PackageReference>
1115
<ProjectReference Include="..\..\src\Microsoft.AutoGen\Core\Microsoft.AutoGen.Core.csproj" />
1216
<PackageReference Include="Microsoft.Extensions.Hosting" />
1317
</ItemGroup>

dotnet/test/Microsoft.AutoGen.Runtime.Grpc.Tests/GrpcGatewayServiceTests.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212

1313
namespace Microsoft.AutoGen.Runtime.Grpc.Tests;
1414
[Collection(ClusterCollection.Name)]
15-
[Trait("Category", "UnitV2")]
15+
[Trait("Category", "GRPC")]
1616
public class GrpcGatewayServiceTests
1717
{
1818
private readonly ClusterFixture _fixture;

dotnet/test/Microsoft.AutoGen.Runtime.Grpc.Tests/Microsoft.AutoGen.Runtime.Grpc.Tests.csproj

+4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,10 @@
88
</PropertyGroup>
99

1010
<ItemGroup>
11+
<PackageReference Include="coverlet.collector">
12+
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
13+
<PrivateAssets>all</PrivateAssets>
14+
</PackageReference>
1115
<PackageReference Include="Microsoft.AspNetCore.TestHost" />
1216
<PackageReference Include="Microsoft.Orleans.TestingHost" />
1317
</ItemGroup>

python/packages/autogen-agentchat/src/autogen_agentchat/agents/_assistant_agent.py

+3
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@ class AssistantAgentConfig(BaseModel):
6262
tools: List[ComponentModel] | None
6363
handoffs: List[HandoffBase | str] | None = None
6464
model_context: ComponentModel | None = None
65+
memory: List[ComponentModel] | None = None
6566
description: str
6667
system_message: str | None = None
6768
model_client_stream: bool = False
@@ -591,6 +592,7 @@ def _to_config(self) -> AssistantAgentConfig:
591592
tools=[tool.dump_component() for tool in self._tools],
592593
handoffs=list(self._handoffs.values()),
593594
model_context=self._model_context.dump_component(),
595+
memory=[memory.dump_component() for memory in self._memory] if self._memory else None,
594596
description=self.description,
595597
system_message=self._system_messages[0].content
596598
if self._system_messages and isinstance(self._system_messages[0].content, str)
@@ -609,6 +611,7 @@ def _from_config(cls, config: AssistantAgentConfig) -> Self:
609611
tools=[BaseTool.load_component(tool) for tool in config.tools] if config.tools else None,
610612
handoffs=config.handoffs,
611613
model_context=None,
614+
memory=[Memory.load_component(memory) for memory in config.memory] if config.memory else None,
612615
description=config.description,
613616
system_message=config.system_message,
614617
model_client_stream=config.model_client_stream,

python/packages/autogen-agentchat/src/autogen_agentchat/teams/_group_chat/_magentic_one/_magentic_one_orchestrator.py

+7-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import json
22
import logging
3+
import re
34
from typing import Any, Dict, List, Mapping
45

56
from autogen_core import AgentId, CancellationToken, DefaultTopicId, MessageContext, event, rpc
@@ -80,14 +81,12 @@ def __init__(
8081
self._plan = ""
8182
self._n_rounds = 0
8283
self._n_stalls = 0
83-
self._team_description = "\n".join(
84-
[
85-
f"{topic_type}: {description}".strip()
86-
for topic_type, description in zip(
87-
self._participant_topic_types, self._participant_descriptions, strict=True
88-
)
89-
]
90-
)
84+
85+
# Produce a team description. Each agent sould appear on a single line.
86+
self._team_description = ""
87+
for topic_type, description in zip(self._participant_topic_types, self._participant_descriptions, strict=True):
88+
self._team_description += re.sub(r"\s+", " ", f"{topic_type}: {description}").strip() + "\n"
89+
self._team_description = self._team_description.strip()
9190

9291
def _get_task_ledger_facts_prompt(self, task: str) -> str:
9392
return ORCHESTRATOR_TASK_LEDGER_FACTS_PROMPT.format(task=task)

python/packages/autogen-agentchat/tests/test_assistant_agent.py

+9-3
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@
1818
ToolCallRequestEvent,
1919
ToolCallSummaryMessage,
2020
)
21-
from autogen_core import FunctionCall, Image
21+
from autogen_core import ComponentModel, FunctionCall, Image
2222
from autogen_core.memory import ListMemory, Memory, MemoryContent, MemoryMimeType, MemoryQueryResult
2323
from autogen_core.model_context import BufferedChatCompletionContext
2424
from autogen_core.models import (
@@ -754,7 +754,12 @@ async def test_run_with_memory(monkeypatch: pytest.MonkeyPatch) -> None:
754754
"test_agent", model_client=OpenAIChatCompletionClient(model=model, api_key=""), memory=[memory2]
755755
)
756756

757-
result = await agent.run(task="test task")
757+
# Test dump and load component with memory
758+
agent_config: ComponentModel = agent.dump_component()
759+
assert agent_config.provider == "autogen_agentchat.agents.AssistantAgent"
760+
agent2 = AssistantAgent.load_component(agent_config)
761+
762+
result = await agent2.run(task="test task")
758763
assert len(result.messages) > 0
759764
memory_event = next((msg for msg in result.messages if isinstance(msg, MemoryQueryEvent)), None)
760765
assert memory_event is not None
@@ -795,9 +800,10 @@ async def test_assistant_agent_declarative(monkeypatch: pytest.MonkeyPatch) -> N
795800
"test_agent",
796801
model_client=OpenAIChatCompletionClient(model=model, api_key=""),
797802
model_context=model_context,
803+
memory=[ListMemory(name="test_memory")],
798804
)
799805

800-
agent_config = agent.dump_component()
806+
agent_config: ComponentModel = agent.dump_component()
801807
assert agent_config.provider == "autogen_agentchat.agents.AssistantAgent"
802808

803809
agent2 = AssistantAgent.load_component(agent_config)

python/packages/autogen-core/docs/src/user-guide/agentchat-user-guide/tutorial/custom-agents.ipynb

+2-2
Original file line numberDiff line numberDiff line change
@@ -49,7 +49,7 @@
4949
}
5050
],
5151
"source": [
52-
"from typing import AsyncGenerator, List, Sequence, Tuple\n",
52+
"from typing import AsyncGenerator, List, Sequence\n",
5353
"\n",
5454
"from autogen_agentchat.agents import BaseChatAgent\n",
5555
"from autogen_agentchat.base import Response\n",
@@ -310,4 +310,4 @@
310310
},
311311
"nbformat": 4,
312312
"nbformat_minor": 2
313-
}
313+
}

python/packages/autogen-core/src/autogen_core/memory/_base_memory.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
from pydantic import BaseModel, ConfigDict
66

77
from .._cancellation_token import CancellationToken
8+
from .._component_config import ComponentBase
89
from .._image import Image
910
from ..model_context import ChatCompletionContext
1011

@@ -49,7 +50,7 @@ class UpdateContextResult(BaseModel):
4950
memories: MemoryQueryResult
5051

5152

52-
class Memory(ABC):
53+
class Memory(ABC, ComponentBase[BaseModel]):
5354
"""Protocol defining the interface for memory implementations.
5455
5556
A memory is the storage for data that can be used to enrich or modify the model context.
@@ -64,6 +65,8 @@ class Memory(ABC):
6465
See :class:`~autogen_core.memory.ListMemory` for an example implementation.
6566
"""
6667

68+
component_type = "memory"
69+
6770
@abstractmethod
6871
async def update_context(
6972
self,

python/packages/autogen-core/src/autogen_core/memory/_list_memory.py

+27-3
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,25 @@
11
from typing import Any, List
22

3+
from pydantic import BaseModel
4+
from typing_extensions import Self
5+
36
from .._cancellation_token import CancellationToken
7+
from .._component_config import Component
48
from ..model_context import ChatCompletionContext
59
from ..models import SystemMessage
610
from ._base_memory import Memory, MemoryContent, MemoryQueryResult, UpdateContextResult
711

812

9-
class ListMemory(Memory):
13+
class ListMemoryConfig(BaseModel):
14+
"""Configuration for ListMemory component."""
15+
16+
name: str | None = None
17+
"""Optional identifier for this memory instance."""
18+
memory_contents: List[MemoryContent] = []
19+
"""List of memory contents stored in this memory instance."""
20+
21+
22+
class ListMemory(Memory, Component[ListMemoryConfig]):
1023
"""Simple chronological list-based memory implementation.
1124
1225
This memory implementation stores contents in a list and retrieves them in
@@ -53,9 +66,13 @@ async def main() -> None:
5366
5467
"""
5568

56-
def __init__(self, name: str | None = None) -> None:
69+
component_type = "memory"
70+
component_provider_override = "autogen_core.memory.ListMemory"
71+
component_config_schema = ListMemoryConfig
72+
73+
def __init__(self, name: str | None = None, memory_contents: List[MemoryContent] | None = None) -> None:
5774
self._name = name or "default_list_memory"
58-
self._contents: List[MemoryContent] = []
75+
self._contents: List[MemoryContent] = memory_contents if memory_contents is not None else []
5976

6077
@property
6178
def name(self) -> str:
@@ -146,3 +163,10 @@ async def clear(self) -> None:
146163
async def close(self) -> None:
147164
"""Cleanup resources if needed."""
148165
pass
166+
167+
@classmethod
168+
def _from_config(cls, config: ListMemoryConfig) -> Self:
169+
return cls(name=config.name, memory_contents=config.memory_contents)
170+
171+
def _to_config(self) -> ListMemoryConfig:
172+
return ListMemoryConfig(name=self.name, memory_contents=self._contents)

0 commit comments

Comments
 (0)