gemcode 0.3.37__tar.gz → 0.3.39__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 (120) hide show
  1. {gemcode-0.3.37/src/gemcode.egg-info → gemcode-0.3.39}/PKG-INFO +1 -1
  2. {gemcode-0.3.37 → gemcode-0.3.39}/pyproject.toml +1 -1
  3. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/agent.py +22 -6
  4. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/cli.py +1 -1
  5. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/invoke.py +2 -2
  6. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/live_audio_engine.py +1 -1
  7. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/notes.py +4 -2
  8. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tui/scrollback.py +8 -1
  9. {gemcode-0.3.37 → gemcode-0.3.39/src/gemcode.egg-info}/PKG-INFO +1 -1
  10. {gemcode-0.3.37 → gemcode-0.3.39}/LICENSE +0 -0
  11. {gemcode-0.3.37 → gemcode-0.3.39}/MANIFEST.in +0 -0
  12. {gemcode-0.3.37 → gemcode-0.3.39}/README.md +0 -0
  13. {gemcode-0.3.37 → gemcode-0.3.39}/setup.cfg +0 -0
  14. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/__init__.py +0 -0
  15. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/__main__.py +0 -0
  16. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/audit.py +0 -0
  17. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/autocompact.py +0 -0
  18. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/callbacks.py +0 -0
  19. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/capability_routing.py +0 -0
  20. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/compaction.py +0 -0
  21. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/computer_use/__init__.py +0 -0
  22. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/computer_use/browser_computer.py +0 -0
  23. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/config.py +0 -0
  24. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/context_budget.py +0 -0
  25. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/context_warning.py +0 -0
  26. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/credentials.py +0 -0
  27. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/hitl_session.py +0 -0
  28. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/hooks.py +0 -0
  29. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/interactions.py +0 -0
  30. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/kairos_daemon.py +0 -0
  31. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/limits.py +0 -0
  32. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/logging_config.py +0 -0
  33. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/mcp_loader.py +0 -0
  34. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/memory/__init__.py +0 -0
  35. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/memory/embedding_memory_service.py +0 -0
  36. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/memory/file_memory_service.py +0 -0
  37. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/modality_tools.py +0 -0
  38. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/model_errors.py +0 -0
  39. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/model_routing.py +0 -0
  40. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/openapi_loader.py +0 -0
  41. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/paths.py +0 -0
  42. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/permissions.py +0 -0
  43. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/plugins/__init__.py +0 -0
  44. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/plugins/terminal_hooks_plugin.py +0 -0
  45. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/plugins/tool_recovery_plugin.py +0 -0
  46. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/pricing.py +0 -0
  47. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/prompt_suggestions.py +0 -0
  48. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/__init__.py +0 -0
  49. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/config.py +0 -0
  50. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/deps.py +0 -0
  51. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/engine.py +0 -0
  52. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/stop_hooks.py +0 -0
  53. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/token_budget.py +0 -0
  54. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/query/transitions.py +0 -0
  55. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/refine.py +0 -0
  56. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/repl_commands.py +0 -0
  57. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/repl_slash.py +0 -0
  58. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/review_agent.py +0 -0
  59. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/session_runtime.py +0 -0
  60. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/session_store.py +0 -0
  61. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/slash_commands.py +0 -0
  62. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/thinking.py +0 -0
  63. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tool_prompt_manifest.py +0 -0
  64. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tool_registry.py +0 -0
  65. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/__init__.py +0 -0
  66. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/bash.py +0 -0
  67. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/browser.py +0 -0
  68. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/edit.py +0 -0
  69. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/filesystem.py +0 -0
  70. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/search.py +0 -0
  71. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/shell.py +0 -0
  72. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/shell_gate.py +0 -0
  73. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/subtask.py +0 -0
  74. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/think.py +0 -0
  75. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/todo.py +0 -0
  76. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools/web.py +0 -0
  77. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tools_inspector.py +0 -0
  78. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/trust.py +0 -0
  79. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tui/input_handler.py +0 -0
  80. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tui/spinner.py +0 -0
  81. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tui/welcome_banner.py +0 -0
  82. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/tui/welcome_rich.py +0 -0
  83. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/version.py +0 -0
  84. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/vertex.py +0 -0
  85. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/web/__init__.py +0 -0
  86. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/web/claude_sse_adapter.py +0 -0
  87. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/web/terminal_repl.py +0 -0
  88. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode/workspace_hints.py +0 -0
  89. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode.egg-info/SOURCES.txt +0 -0
  90. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode.egg-info/dependency_links.txt +0 -0
  91. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode.egg-info/entry_points.txt +0 -0
  92. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode.egg-info/requires.txt +0 -0
  93. {gemcode-0.3.37 → gemcode-0.3.39}/src/gemcode.egg-info/top_level.txt +0 -0
  94. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_agent_instruction.py +0 -0
  95. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_autocompact.py +0 -0
  96. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_capability_routing.py +0 -0
  97. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_claude_web_adapter_sse.py +0 -0
  98. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_cli_init.py +0 -0
  99. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_computer_use_permissions.py +0 -0
  100. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_context_budget.py +0 -0
  101. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_context_warning.py +0 -0
  102. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_credentials.py +0 -0
  103. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_interactive_permission_ask.py +0 -0
  104. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_kairos_scheduler.py +0 -0
  105. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_modality_tools.py +0 -0
  106. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_model_error_retry.py +0 -0
  107. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_model_errors.py +0 -0
  108. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_model_routing.py +0 -0
  109. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_paths.py +0 -0
  110. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_permissions.py +0 -0
  111. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_prompt_suggestions.py +0 -0
  112. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_repl_commands.py +0 -0
  113. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_repl_slash.py +0 -0
  114. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_slash_commands.py +0 -0
  115. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_thinking_config.py +0 -0
  116. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_token_budget.py +0 -0
  117. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_tool_context_circulation.py +0 -0
  118. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_tools.py +0 -0
  119. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_tools_inspector.py +0 -0
  120. {gemcode-0.3.37 → gemcode-0.3.39}/tests/test_workspace_hints.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: gemcode
3
- Version: 0.3.37
3
+ Version: 0.3.39
4
4
  Summary: Local-first coding agent on Google Gemini + ADK
5
5
  Author: GemCode Contributors
6
6
  License: Apache License
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "gemcode"
7
- version = "0.3.37"
7
+ version = "0.3.39"
8
8
  description = "Local-first coding agent on Google Gemini + ADK"
9
9
  readme = "README.md"
10
10
  requires-python = ">=3.11"
@@ -397,8 +397,24 @@ You run locally via the GemCode CLI. You are the same agent the user launched
397
397
  {_build_runtime_facts(cfg)}
398
398
 
399
399
  ## Core identity and approach
400
- You are a senior engineer who *acts*, not just advises. When given a task:
401
- 1. **Orient** use `list_directory`, `glob_files`, `grep_content`, `read_file` to understand structure. These tools need **no permission** and are instant. Start here.
400
+
401
+ ### Classify before acting choose the right mode for each message
402
+
403
+ Not every message is a coding task. Read the intent first and pick the right mode:
404
+
405
+ | Message type | Examples | Right action |
406
+ |---|---|---|
407
+ | **Greeting / chitchat** | "hi", "hii", "hello", "how are you", "thanks" | Reply directly. No tools. No orientation. |
408
+ | **Pure question / concept** | "what is a closure?", "explain OAuth", "which is better, Redis or Postgres?" | Answer from knowledge. No tools unless the answer requires inspecting *this* repo. |
409
+ | **Vague project question** | "how does auth work here?", "what's the folder structure?" | Call 1–2 read-only tools to find the answer, then reply. Keep it focused. |
410
+ | **Engineering task** | "add pagination", "fix the bug in auth.ts", "refactor the DB layer" | Full workflow: orient → plan → execute → verify. |
411
+ | **Analysis / research** | "analyse the whole backend", "summarise all API endpoints" | Systematic tool use: list → read → grep → synthesise. |
412
+
413
+ **NEVER call `list_directory`, `read_project_notes`, or any tool in response to a greeting or a general conversational message.** If you wouldn't call a tool to answer "hi" in a conversation, don't call it here either.
414
+
415
+ ### Engineering task workflow
416
+ When the message is a real engineering task:
417
+ 1. **Orient** — use `list_directory`, `glob_files`, `grep_content`, `read_file` to understand structure. These tools need **no permission** and are instant.
402
418
  2. **Plan** — for complex tasks, call `todo_write` upfront to map out the work.
403
419
  3. **Execute** — make the changes, run the checks, iterate.
404
420
  4. **Verify** — confirm the result is correct before reporting done.
@@ -426,8 +442,9 @@ You have native deep thinking capability — use it actively:
426
442
  - **For trade-off decisions** (which library, which pattern, which approach): reason through the pros/cons given this specific codebase.
427
443
 
428
444
  ## Interpreting requests
429
- - Treat every message as a software engineering task start working immediately. Do NOT respond with just a greeting or introduction.
430
- - If vague ("fix it", "the config", "rename that"), **infer from the repo**: search, read, then act. Do not give abstract advice when concrete files exist.
445
+ - **Greetings / conversational messages** ("hi", "hii", "hey", "thanks", "cool"): reply naturally with one sentence. Do NOT call any tools. Do NOT apologise for not running code.
446
+ - **General questions** that don't mention files or code ("what is X?", "explain Y"): answer from your knowledge directly. Only use tools if verifying something in *this specific repo* would materially improve the answer.
447
+ - **Engineering tasks** ("fix", "add", "refactor", "analyse", "debug"): infer from the repo — search, read, then act. Do not give abstract advice when concrete files exist.
431
448
  - If the user refers to symbols or behaviors, **find them** with `glob_files`/`grep_content`/`list_directory` — never ask them to paste paths you can discover yourself.
432
449
  - **Never propose edits to files you haven't read.** Read first, then edit.
433
450
  - When something fails, diagnose (re-read the error, check assumptions) before switching strategy. Do not repeat the same failed call.
@@ -615,8 +632,7 @@ You have two tools to persist project insights across sessions, like Claude Code
615
632
  Call this **immediately** when you discover something useful — not just at the end of tasks.
616
633
  Notes are loaded at session start so future sessions inherit this knowledge.
617
634
 
618
- - **`read_project_notes()`** — read current notes before starting a new project task.
619
- If notes exist and you haven't read them yet, read them first to avoid re-discovering known information."""
635
+ - **`read_project_notes()`** — read current notes **only when starting a real engineering task** (editing, debugging, building). Do NOT call this for greetings or general questions. If notes exist and you're about to work on a task, read them once to avoid re-discovering known information."""
620
636
 
621
637
  # Inject capability-specific strategy sections only when those caps are on.
622
638
  if getattr(cfg, "enable_computer_use", False):
@@ -27,7 +27,7 @@ def _events_to_text(events) -> str:
27
27
  if not event.content or not event.content.parts:
28
28
  continue
29
29
  for part in event.content.parts:
30
- if part.text and event.author and event.author != "user":
30
+ if part.text and getattr(event, "author", None) != "user":
31
31
  parts.append(part.text)
32
32
  return "".join(parts)
33
33
 
@@ -27,8 +27,8 @@ def _events_to_text(events: list[Any]) -> str:
27
27
  content = getattr(event, "content", None)
28
28
  if not content or not getattr(content, "parts", None):
29
29
  continue
30
- author = getattr(event, "author", None)
31
- if not author or author == "user":
30
+ # Omit only user-authored events; model events may have author=None.
31
+ if getattr(event, "author", None) == "user":
32
32
  continue
33
33
  for part in getattr(content, "parts", []) or []:
34
34
  t = getattr(part, "text", None)
@@ -89,7 +89,7 @@ async def run_live_audio(
89
89
  for part in event.content.parts:
90
90
  part_text = getattr(part, "text", None)
91
91
  # We only print model-authored text to avoid echoing user input.
92
- if part_text and getattr(event, "author", None) and event.author != "user":
92
+ if part_text and getattr(event, "author", None) != "user":
93
93
  sys.stdout.write(part_text)
94
94
  sys.stdout.flush()
95
95
  printed_any = True
@@ -86,8 +86,10 @@ def build_notes_tools(project_root: Path) -> list:
86
86
  """
87
87
  Read the current contents of .gemcode/notes.md.
88
88
 
89
- Use this to check what has been previously noted about this project before
90
- starting a task avoids re-discovering things already documented.
89
+ ONLY call this when you are about to work on an actual engineering task
90
+ (editing files, debugging, building something) and want to avoid re-discovering
91
+ already-known project facts. Do NOT call this for greetings, chitchat, or
92
+ general questions that don't require project context.
91
93
 
92
94
  Returns the full notes content as a string, or an empty string if no notes exist.
93
95
  """
@@ -577,7 +577,9 @@ async def run_gemcode_scrollback_tui(
577
577
  try:
578
578
  if not ev.content or not ev.content.parts:
579
579
  continue
580
- if not getattr(ev, "author", None) or ev.author == "user":
580
+ # Only skip user turns. ADK often omits `author` on model events — do NOT
581
+ # skip those or the assistant text never renders (blank reply, ↓0 tokens).
582
+ if getattr(ev, "author", None) == "user":
581
583
  continue
582
584
  for part in ev.content.parts:
583
585
  delta = getattr(part, "text", None)
@@ -616,6 +618,11 @@ async def run_gemcode_scrollback_tui(
616
618
  if not confirmation_fcs:
617
619
  thought_text = "".join(buffered_thought).strip()
618
620
  final_text = "".join(buffered_final).strip()
621
+ # Gemini with thinking enabled may emit visible text only in "thought" parts;
622
+ # usage can show thoughts_token_count > 0 and candidates_token_count == 0.
623
+ if not final_text and thought_text:
624
+ final_text = thought_text
625
+ thought_text = ""
619
626
 
620
627
  # ── Thinking display (collapsed by default, verbose with /thinking verbose)
621
628
  if thought_text and not (
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: gemcode
3
- Version: 0.3.37
3
+ Version: 0.3.39
4
4
  Summary: Local-first coding agent on Google Gemini + ADK
5
5
  Author: GemCode Contributors
6
6
  License: Apache License
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes