pygent 0.1.2__py3-none-any.whl → 0.1.4__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.
pygent/__init__.py CHANGED
@@ -7,5 +7,6 @@ except _metadata.PackageNotFoundError: # pragma: no cover - fallback for tests
7
7
  __version__ = "0.0.0"
8
8
 
9
9
  from .agent import Agent, run_interactive # noqa: E402,F401, must come after __version__
10
+ from .models import Model, OpenAIModel # noqa: E402,F401
10
11
 
11
- __all__ = ["Agent", "run_interactive"]
12
+ __all__ = ["Agent", "run_interactive", "Model", "OpenAIModel"]
pygent/agent.py CHANGED
@@ -8,17 +8,14 @@ import time
8
8
  from dataclasses import dataclass, field
9
9
  from typing import Any, Dict, List
10
10
 
11
- try:
12
- import openai # type: ignore
13
- except ModuleNotFoundError: # pragma: no cover - fallback to bundled client
14
- from . import openai_compat as openai
15
11
  from rich.console import Console
16
12
  from rich.panel import Panel
17
13
 
18
14
  from .runtime import Runtime
19
15
  from .tools import TOOL_SCHEMAS, execute_tool
16
+ from .models import Model, OpenAIModel
20
17
 
21
- MODEL = os.getenv("PYGENT_MODEL", "gpt-4o-mini-preview")
18
+ DEFAULT_MODEL = os.getenv("PYGENT_MODEL", "gpt-4.1-mini")
22
19
  SYSTEM_MSG = (
23
20
  "You are Pygent, a sandboxed coding assistant.\n"
24
21
  "Respond with JSON when you need to use a tool."
@@ -27,26 +24,20 @@ SYSTEM_MSG = (
27
24
  console = Console()
28
25
 
29
26
 
30
- def _chat(messages: List[Dict[str, str]]) -> str:
31
- resp = openai.chat.completions.create(
32
- model=MODEL,
33
- messages=messages,
34
- tools=TOOL_SCHEMAS,
35
- tool_choice="auto",
36
- )
37
- return resp.choices[0].message
38
27
 
39
28
 
40
29
  @dataclass
41
30
  class Agent:
42
31
  runtime: Runtime = field(default_factory=Runtime)
32
+ model: Model = field(default_factory=OpenAIModel)
33
+ model_name: str = DEFAULT_MODEL
43
34
  history: List[Dict[str, Any]] = field(default_factory=lambda: [
44
35
  {"role": "system", "content": SYSTEM_MSG}
45
36
  ])
46
37
 
47
38
  def step(self, user_msg: str) -> None:
48
39
  self.history.append({"role": "user", "content": user_msg})
49
- assistant_msg = _chat(self.history)
40
+ assistant_msg = self.model.chat(self.history, self.model_name, TOOL_SCHEMAS)
50
41
  self.history.append(assistant_msg)
51
42
 
52
43
  if assistant_msg.tool_calls:
pygent/models.py ADDED
@@ -0,0 +1,33 @@
1
+ from __future__ import annotations
2
+
3
+ """Model interface and default implementation for OpenAI-compatible APIs."""
4
+
5
+ from typing import Any, Dict, List, Protocol
6
+
7
+ try:
8
+ import openai # type: ignore
9
+ except ModuleNotFoundError: # pragma: no cover - fallback to bundled client
10
+ from . import openai_compat as openai
11
+
12
+ from .openai_compat import Message
13
+
14
+
15
+ class Model(Protocol):
16
+ """Protocol for chat models used by :class:`~pygent.agent.Agent`."""
17
+
18
+ def chat(self, messages: List[Dict[str, Any]], model: str, tools: Any) -> Message:
19
+ """Return the assistant message for the given prompt."""
20
+ ...
21
+
22
+
23
+ class OpenAIModel:
24
+ """Default model using the OpenAI-compatible API."""
25
+
26
+ def chat(self, messages: List[Dict[str, Any]], model: str, tools: Any) -> Message:
27
+ resp = openai.chat.completions.create(
28
+ model=model,
29
+ messages=messages,
30
+ tools=tools,
31
+ tool_choice="auto",
32
+ )
33
+ return resp.choices[0].message
pygent/ui.py ADDED
@@ -0,0 +1,36 @@
1
+ from .agent import Agent
2
+ from .runtime import Runtime
3
+ from .tools import execute_tool, TOOL_SCHEMAS
4
+
5
+
6
+ def run_gui(use_docker: bool | None = None) -> None:
7
+ """Launch a simple Gradio chat interface."""
8
+ try:
9
+ import gradio as gr
10
+ except ModuleNotFoundError as exc: # pragma: no cover - optional
11
+ raise SystemExit(
12
+ "Gradio is required for the GUI. Install with 'pip install pygent[ui]'"
13
+ ) from exc
14
+
15
+ agent = Agent(runtime=Runtime(use_docker=use_docker))
16
+
17
+ def _respond(message: str, history: list[tuple[str, str]] | None) -> str:
18
+ agent.history.append({"role": "user", "content": message})
19
+ assistant_msg = agent.model.chat(agent.history, agent.model_name, TOOL_SCHEMAS)
20
+ agent.history.append(assistant_msg)
21
+ reply = assistant_msg.content or ""
22
+ if assistant_msg.tool_calls:
23
+ for call in assistant_msg.tool_calls:
24
+ output = execute_tool(call, agent.runtime)
25
+ agent.history.append({"role": "tool", "content": output, "tool_call_id": call.id})
26
+ reply += f"\n\n[tool:{call.function.name}]\n{output}"
27
+ return reply
28
+
29
+ try:
30
+ gr.ChatInterface(_respond, title="Pygent").launch()
31
+ finally:
32
+ agent.runtime.cleanup()
33
+
34
+
35
+ def main() -> None: # pragma: no cover
36
+ run_gui()
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pygent
3
- Version: 0.1.2
3
+ Version: 0.1.4
4
4
  Summary: Pygent is a minimalist coding assistant that runs commands in a Docker container when available and falls back to local execution. See https://marianochaves.github.io/pygent for documentation and https://github.com/marianochaves/pygent for the source code.
5
5
  Author-email: Mariano Chaves <mchaves.software@gmail.com>
6
6
  Project-URL: Documentation, https://marianochaves.github.io/pygent
@@ -16,4 +16,6 @@ Provides-Extra: docs
16
16
  Requires-Dist: mkdocs; extra == "docs"
17
17
  Provides-Extra: docker
18
18
  Requires-Dist: docker>=7.0.0; extra == "docker"
19
+ Provides-Extra: ui
20
+ Requires-Dist: gradio; extra == "ui"
19
21
  Dynamic: license-file
@@ -0,0 +1,15 @@
1
+ pygent/__init__.py,sha256=_YO8FYMUMAWlRYCa6OBsfmJ9P4mCiYoOqFBLrQDP8qQ,442
2
+ pygent/agent.py,sha256=aR1lHtI3ouI9rEMZLE5lr8tIdC9-A_ZtvdwsvqwxRMg,2010
3
+ pygent/cli.py,sha256=Hz2FZeNMVhxoT5DjCqphXla3TisGJtPEz921LEcpxrA,527
4
+ pygent/models.py,sha256=_3Y1Z5wL6FUqzC-EOjZe3Vkcq4SzbPdGz6TbshcEB98,992
5
+ pygent/openai_compat.py,sha256=mS6ntl70jpVH3JzfNYEDhg-z7QIQcMqQTuEV5ja7VOo,2173
6
+ pygent/py.typed,sha256=0Wh72UpGSn4lSGW-u3xMV9kxcBHMdwE15IGUqiJTwqo,52
7
+ pygent/runtime.py,sha256=33y4jieNeyZ-9nxtVlOmO236u2fDAAv2GaaEWTQDdm8,3173
8
+ pygent/tools.py,sha256=Ru2_voFgPUVc6YgBTRVByn7vWTxXAXT-loAWFMkXHno,1460
9
+ pygent/ui.py,sha256=xqPAvweghPOBBvoD72HzhN6zlXew_3inb8AN7Ck2zpQ,1328
10
+ pygent-0.1.4.dist-info/licenses/LICENSE,sha256=rIktBU2VR4kHzsWul64cbom2zHIgGqYmABoZwSur6T8,1071
11
+ pygent-0.1.4.dist-info/METADATA,sha256=vzJ-74Xlwf730kRfXEO55ZTSiRQiXQc-9BUKMtcdRyM,913
12
+ pygent-0.1.4.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
13
+ pygent-0.1.4.dist-info/entry_points.txt,sha256=b9j216E5UpuMrQWRZrwyEmacNEAYvw1tCKkZqdIVIOc,70
14
+ pygent-0.1.4.dist-info/top_level.txt,sha256=P26IYsb-ThK5IkGP_bRuGJQ0Q_Y8JCcbYqVpvULdxDw,7
15
+ pygent-0.1.4.dist-info/RECORD,,
@@ -1,2 +1,3 @@
1
1
  [console_scripts]
2
2
  pygent = pygent.cli:main
3
+ pygent-ui = pygent.ui:main
@@ -1,13 +0,0 @@
1
- pygent/__init__.py,sha256=3YOE3tjTGEc987Vz-TqmYwQ4ogLwTmue642Enf4NVBg,360
2
- pygent/agent.py,sha256=s80JFgtDZ4tNEroh0qIJ0A_8QAiYyORD82R6KsM6vEo,2235
3
- pygent/cli.py,sha256=Hz2FZeNMVhxoT5DjCqphXla3TisGJtPEz921LEcpxrA,527
4
- pygent/openai_compat.py,sha256=mS6ntl70jpVH3JzfNYEDhg-z7QIQcMqQTuEV5ja7VOo,2173
5
- pygent/py.typed,sha256=0Wh72UpGSn4lSGW-u3xMV9kxcBHMdwE15IGUqiJTwqo,52
6
- pygent/runtime.py,sha256=33y4jieNeyZ-9nxtVlOmO236u2fDAAv2GaaEWTQDdm8,3173
7
- pygent/tools.py,sha256=Ru2_voFgPUVc6YgBTRVByn7vWTxXAXT-loAWFMkXHno,1460
8
- pygent-0.1.2.dist-info/licenses/LICENSE,sha256=rIktBU2VR4kHzsWul64cbom2zHIgGqYmABoZwSur6T8,1071
9
- pygent-0.1.2.dist-info/METADATA,sha256=utdsiVHvR3fMjqGeNvBNC4oMrD0Ytav6CYyrgAyjnLM,857
10
- pygent-0.1.2.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
11
- pygent-0.1.2.dist-info/entry_points.txt,sha256=ivw-s2f1abmFsbL4173DP1IuMS7sNxQ6gZuDLdu_jKQ,43
12
- pygent-0.1.2.dist-info/top_level.txt,sha256=P26IYsb-ThK5IkGP_bRuGJQ0Q_Y8JCcbYqVpvULdxDw,7
13
- pygent-0.1.2.dist-info/RECORD,,
File without changes