code-puppy 0.0.157__py3-none-any.whl → 0.0.159__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.
@@ -1,9 +1,6 @@
1
1
  import os
2
2
 
3
- from code_puppy.command_line.model_picker_completion import (
4
- load_model_names,
5
- update_model_in_input,
6
- )
3
+ from code_puppy.command_line.model_picker_completion import update_model_in_input
7
4
  from code_puppy.command_line.motd import print_motd
8
5
  from code_puppy.command_line.utils import make_directory_table
9
6
  from code_puppy.config import get_config_keys
@@ -60,7 +57,7 @@ def get_commands_help():
60
57
  )
61
58
  help_lines.append(
62
59
  Text("/compact", style="cyan")
63
- + Text(" Summarize and compact current chat history")
60
+ + Text(" Summarize and compact current chat history (uses compaction_strategy config)")
64
61
  )
65
62
  help_lines.append(
66
63
  Text("/dump_context", style="cyan")
@@ -137,18 +134,18 @@ def handle_command(command: str):
137
134
 
138
135
  before_tokens = sum(estimate_tokens_for_message(m) for m in history)
139
136
  compaction_strategy = get_compaction_strategy()
137
+ protected_tokens = get_protected_token_count()
140
138
  emit_info(
141
139
  f"🤔 Compacting {len(history)} messages using {compaction_strategy} strategy... (~{before_tokens} tokens)"
142
140
  )
143
141
 
144
142
  if compaction_strategy == "truncation":
145
- protected_tokens = get_protected_token_count()
146
143
  compacted = truncation(history, protected_tokens)
147
144
  summarized_messages = [] # No summarization in truncation mode
148
145
  else:
149
146
  # Default to summarization
150
147
  compacted, summarized_messages = summarize_messages(
151
- history, with_protection=False
148
+ history, with_protection=True
152
149
  )
153
150
 
154
151
  if not compacted:
@@ -372,6 +369,9 @@ def handle_command(command: str):
372
369
  # Convert /model to /m for internal processing
373
370
  model_command = command.replace("/model", "/m", 1)
374
371
 
372
+ # If no model matched, show available models
373
+ from code_puppy.command_line.model_picker_completion import load_model_names
374
+
375
375
  new_input = update_model_in_input(model_command)
376
376
  if new_input is not None:
377
377
  from code_puppy.agents.runtime_manager import get_runtime_agent_manager
@@ -383,7 +383,6 @@ def handle_command(command: str):
383
383
  manager.reload_agent()
384
384
  emit_success(f"Active model set and loaded: {model}")
385
385
  return True
386
- # If no model matched, show available models
387
386
  model_names = load_model_names()
388
387
  emit_warning("Usage: /model <model-name> or /m <model-name>")
389
388
  emit_warning(f"Available models: {', '.join(model_names)}")
code_puppy/config.py CHANGED
@@ -110,6 +110,8 @@ def get_config_keys():
110
110
  "yolo_mode",
111
111
  "model",
112
112
  "compaction_strategy",
113
+ "protected_token_count",
114
+ "compaction_threshold",
113
115
  "message_limit",
114
116
  "allow_recursion",
115
117
  ]
@@ -162,7 +164,7 @@ _default_model_cache = None
162
164
  def _default_model_from_models_json():
163
165
  """Attempt to load the first model name from models.json.
164
166
 
165
- Falls back to the hard-coded default (``claude-4-0-sonnet``) if the file
167
+ Falls back to the hard-coded default (``gpt-5``) if the file
166
168
  cannot be read for any reason or is empty.
167
169
  """
168
170
  global _default_model_cache
@@ -176,11 +178,17 @@ def _default_model_from_models_json():
176
178
  from code_puppy.model_factory import ModelFactory
177
179
 
178
180
  models_config = ModelFactory.load_config()
179
- first_key = next(iter(models_config)) # Raises StopIteration if empty
180
- _default_model_cache = first_key
181
- return first_key
181
+ if models_config:
182
+ # Get the first key from the models config
183
+ first_key = next(iter(models_config))
184
+ _default_model_cache = first_key
185
+ return first_key
186
+ else:
187
+ # If models_config is empty, fall back to gpt-5
188
+ _default_model_cache = "gpt-5"
189
+ return "gpt-5"
182
190
  except Exception:
183
- # Any problem (network, file missing, empty dict, etc.) => fall back
191
+ # Any problem (network, file missing, empty dict, etc.) => fall back to gpt-5
184
192
  _default_model_cache = "gpt-5"
185
193
  return "gpt-5"
186
194
 
@@ -196,8 +204,7 @@ def _validate_model_exists(model_name: str) -> bool:
196
204
  try:
197
205
  from code_puppy.model_factory import ModelFactory
198
206
 
199
- models_config_path = os.path.join(CONFIG_DIR, "models.json")
200
- models_config = ModelFactory.load_config(models_config_path)
207
+ models_config = ModelFactory.load_config()
201
208
  exists = model_name in models_config
202
209
 
203
210
  # Cache the result
@@ -485,4 +492,4 @@ def save_command_to_history(command: str):
485
492
  error_msg = (
486
493
  f"❌ An unexpected error occurred while saving command history: {str(e)}"
487
494
  )
488
- direct_console.print(f"[bold red]{error_msg}[/bold red]")
495
+ direct_console.print(f"[bold red]{error_msg}[/bold red]")
@@ -58,9 +58,10 @@ class QueueConsole:
58
58
  from rich.console import Console
59
59
 
60
60
  string_io = StringIO()
61
- # Use markup=False to prevent interpretation of square brackets as markup
61
+ # Use markup=True to properly process rich styling
62
+ # Use a reasonable width to prevent wrapping issues
62
63
  temp_console = Console(
63
- file=string_io, width=80, legacy_windows=False, markup=False
64
+ file=string_io, width=80, legacy_windows=False, markup=True
64
65
  )
65
66
  temp_console.print(v)
66
67
  processed_values.append(string_io.getvalue().rstrip("\n"))
@@ -14,6 +14,7 @@ from pydantic_ai.providers.anthropic import AnthropicProvider
14
14
  from pydantic_ai.providers.google_gla import GoogleGLAProvider
15
15
  from pydantic_ai.providers.openai import OpenAIProvider
16
16
  from pydantic_ai.providers.cerebras import CerebrasProvider
17
+ from pydantic_ai.providers.openrouter import OpenRouterProvider
17
18
 
18
19
  from . import callbacks
19
20
  from .config import EXTRA_MODELS_FILE
@@ -248,6 +249,34 @@ class ModelFactory:
248
249
  setattr(model, "provider", provider)
249
250
  return model
250
251
 
252
+ elif model_type == "openrouter":
253
+ # Get API key from config, which can be an environment variable reference or raw value
254
+ api_key_config = model_config.get("api_key")
255
+ api_key = None
256
+
257
+ if api_key_config:
258
+ if api_key_config.startswith("$"):
259
+ # It's an environment variable reference
260
+ env_var_name = api_key_config[1:] # Remove the $ prefix
261
+ api_key = os.environ.get(env_var_name)
262
+ if api_key is None:
263
+ raise ValueError(
264
+ f"OpenRouter API key environment variable '{env_var_name}' not found or is empty. "
265
+ f"Please set the environment variable: export {env_var_name}=your_value"
266
+ )
267
+ else:
268
+ # It's a raw API key value
269
+ api_key = api_key_config
270
+ else:
271
+ # No API key in config, try to get it from the default environment variable
272
+ api_key = os.environ.get("OPENROUTER_API_KEY")
273
+
274
+ provider = OpenRouterProvider(api_key=api_key)
275
+
276
+ model = OpenAIChatModel(model_name=model_config["name"], provider=provider)
277
+ setattr(model, "provider", provider)
278
+ return model
279
+
251
280
  elif model_type == "round_robin":
252
281
  # Get the list of model names to use in the round-robin
253
282
  model_names = model_config.get("models")
code_puppy/models.json CHANGED
@@ -87,12 +87,6 @@
87
87
  },
88
88
  "context_length": 64000
89
89
  },
90
- "openrouter": {
91
- "type": "openrouter",
92
- "name": "meta-llama/llama-4-maverick:free",
93
- "api_key": "$OPENROUTER_API_KEY",
94
- "context_length": 131072
95
- },
96
90
  "azure-gpt-4.1": {
97
91
  "type": "azure_openai",
98
92
  "name": "gpt-4.1",
@@ -261,8 +261,35 @@ class ChatView(VerticalScroll):
261
261
  else:
262
262
  separator = "\n"
263
263
 
264
- # Update the message content
265
- last_message.content += separator + message.content
264
+ # Handle content concatenation carefully to preserve Rich objects
265
+ if hasattr(last_message.content, "__rich_console__") or hasattr(message.content, "__rich_console__"):
266
+ # If either content is a Rich object, convert both to text and concatenate
267
+ from io import StringIO
268
+ from rich.console import Console
269
+
270
+ # Convert existing content to string
271
+ if hasattr(last_message.content, "__rich_console__"):
272
+ string_io = StringIO()
273
+ temp_console = Console(file=string_io, width=80, legacy_windows=False, markup=False)
274
+ temp_console.print(last_message.content)
275
+ existing_content = string_io.getvalue().rstrip("\n")
276
+ else:
277
+ existing_content = str(last_message.content)
278
+
279
+ # Convert new content to string
280
+ if hasattr(message.content, "__rich_console__"):
281
+ string_io = StringIO()
282
+ temp_console = Console(file=string_io, width=80, legacy_windows=False, markup=False)
283
+ temp_console.print(message.content)
284
+ new_content = string_io.getvalue().rstrip("\n")
285
+ else:
286
+ new_content = str(message.content)
287
+
288
+ # Combine as plain text
289
+ last_message.content = existing_content + separator + new_content
290
+ else:
291
+ # Both are strings, safe to concatenate
292
+ last_message.content += separator + message.content
266
293
 
267
294
  # Update the widget based on message type
268
295
  if last_message.type == MessageType.AGENT_RESPONSE:
@@ -282,25 +309,15 @@ class ChatView(VerticalScroll):
282
309
  copy_button.update_text_to_copy(last_message.content)
283
310
  else:
284
311
  # Handle other message types
285
- content = last_message.content
286
-
287
- # Apply the same rendering logic as in add_message
288
- if (
289
- "[" in content
290
- and "]" in content
291
- and (
292
- content.strip().startswith("$ ")
293
- or content.strip().startswith("git ")
294
- )
295
- ):
296
- # Treat as literal text
297
- last_widget.update(Text(content))
298
- else:
299
- # Try to render markup
300
- try:
301
- last_widget.update(Text.from_markup(content))
302
- except Exception:
303
- last_widget.update(Text(content))
312
+ # After the content concatenation above, content is always a string
313
+ # Try to parse markup when safe to do so
314
+ try:
315
+ # Try to parse as markup first - this handles rich styling correctly
316
+ last_widget.update(Text.from_markup(last_message.content))
317
+ except Exception:
318
+ # If markup parsing fails, fall back to plain text
319
+ # This handles cases where content contains literal square brackets
320
+ last_widget.update(Text(last_message.content))
304
321
 
305
322
  # Add the new message to our tracking lists
306
323
  self.messages.append(message)
@@ -87,12 +87,6 @@
87
87
  },
88
88
  "context_length": 64000
89
89
  },
90
- "openrouter": {
91
- "type": "openrouter",
92
- "name": "meta-llama/llama-4-maverick:free",
93
- "api_key": "$OPENROUTER_API_KEY",
94
- "context_length": 131072
95
- },
96
90
  "azure-gpt-4.1": {
97
91
  "type": "azure_openai",
98
92
  "name": "gpt-4.1",
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: code-puppy
3
- Version: 0.0.157
3
+ Version: 0.0.159
4
4
  Summary: Code generation agent
5
5
  Project-URL: repository, https://github.com/mpfaffenberger/code_puppy
6
6
  Project-URL: HomePage, https://github.com/mpfaffenberger/code_puppy
@@ -2,12 +2,12 @@ code_puppy/__init__.py,sha256=ehbM1-wMjNmOXk_DBhhJECFyBv2dRHwwo7ucjHeM68E,107
2
2
  code_puppy/__main__.py,sha256=pDVssJOWP8A83iFkxMLY9YteHYat0EyWDQqMkKHpWp4,203
3
3
  code_puppy/agent.py,sha256=hZOrJJJDwtLtb82b6Zyt5CZb8RbzDtvHMIBUPVYsgOM,8133
4
4
  code_puppy/callbacks.py,sha256=6wYB6K_fGSCkKKEFaYOYkJT45WaV5W_NhUIzcvVH_nU,5060
5
- code_puppy/config.py,sha256=nD-1Ftwzd1LgZIi5Pi0P4_8DvPkZiOwFXQ5UuYFMO6E,16207
5
+ code_puppy/config.py,sha256=Kp21bilRjqtgC8mYrLTIgR7CZ6_wbvRbsS_Rw7Cx6gs,16385
6
6
  code_puppy/http_utils.py,sha256=BAvt4hed7fVMXglA7eS9gOb08h2YTuOyai6VmQq09fg,3432
7
7
  code_puppy/main.py,sha256=tYLfhUjPTJ-4S1r-pr-jSbn6kIU1iYvt2Z8lxI7zDFY,22220
8
8
  code_puppy/message_history_processor.py,sha256=aV-vcRcOQJPZPlrokB4CaLMxEU3Y4nDiabb9Ov_sJeU,15933
9
- code_puppy/model_factory.py,sha256=0tJWwmKHSqNGEiakk1aGdUrlZyQy0QM4yShxHqxmXpA,11758
10
- code_puppy/models.json,sha256=dAfpMMI2EEeOMv0ynHSmMuJAYDLcZrs5gCLX3voC4-A,3252
9
+ code_puppy/model_factory.py,sha256=_drvOUKx2ivLY_TfE-D6gdZDJzIMyPqf2IWU83KEIE4,13158
10
+ code_puppy/models.json,sha256=iXmLZGflnQcu2DRh4WUlgAhoXdvoxUc7KBhB8YxawXM,3088
11
11
  code_puppy/reopenable_async_client.py,sha256=4UJRaMp5np8cbef9F0zKQ7TPKOfyf5U-Kv-0zYUWDho,8274
12
12
  code_puppy/round_robin_model.py,sha256=SEN3VSwTgC5wHjx2sZsHQLPWOycf4jGwzB-EydgqkdY,5643
13
13
  code_puppy/state_management.py,sha256=o4mNBCPblRyVrNBH-992-1YqffgH6AKHU7iZRqgP1LI,5925
@@ -23,7 +23,7 @@ code_puppy/agents/base_agent.py,sha256=XYSff6IQX9Z6C7hPVbN_YXC2xfjwd268e2jtG3ZGn
23
23
  code_puppy/agents/json_agent.py,sha256=y6AYE3Fx9LhmemcPzt46d7359MNnkGIjU83YBGNer2g,4533
24
24
  code_puppy/agents/runtime_manager.py,sha256=fUOBpmETo3wTyLc5wWBfGKSX1HFRQWSpuwxYAOyA-_8,10059
25
25
  code_puppy/command_line/__init__.py,sha256=y7WeRemfYppk8KVbCGeAIiTuiOszIURCDjOMZv_YRmU,45
26
- code_puppy/command_line/command_handler.py,sha256=KVZySqUMJwrRadKSUCcWa62FZ_lge5XjxD0Hebzr2EE,24520
26
+ code_puppy/command_line/command_handler.py,sha256=w0sE_oGRe6oV05a8-9vJmz0gZWRq_whE8a2mk8Tx7M8,24612
27
27
  code_puppy/command_line/file_path_completion.py,sha256=gw8NpIxa6GOpczUJRyh7VNZwoXKKn-yvCqit7h2y6Gg,2931
28
28
  code_puppy/command_line/load_context_completion.py,sha256=6eZxV6Bs-EFwZjN93V8ZDZUC-6RaWxvtZk-04Wtikyw,2240
29
29
  code_puppy/command_line/meta_command_handler.py,sha256=80aK5JQOaqjt149qBmSsM02uy2Cikkee8zaQnu5u2KQ,5712
@@ -69,7 +69,7 @@ code_puppy/mcp/system_tools.py,sha256=7_oR8k0c8YjtCcYF9g7A946oAGuKOf_i-92aJH7Vml
69
69
  code_puppy/mcp/examples/retry_example.py,sha256=En3LiqECYmG00gWaa7K9L9vvGk1VJyYtKWcv3p3gzCc,7220
70
70
  code_puppy/messaging/__init__.py,sha256=h2eZ7nJblKF71_dNUIBj3vL5RDw7WGy8nh6T_EYVrcA,1176
71
71
  code_puppy/messaging/message_queue.py,sha256=CDVpstLee_RbCBeJWv2eON3c3qhlEISf2i9aSXJTLU4,12600
72
- code_puppy/messaging/queue_console.py,sha256=L4QFUsR_emAFsIRan3-rnL-F1LODF4DG6je-IlUTxNc,10864
72
+ code_puppy/messaging/queue_console.py,sha256=hf32bKfAOdAaxYuARnmDuWhq4ET77xMWDvu5_T2JggY,10912
73
73
  code_puppy/messaging/renderers.py,sha256=9VOpVmu7emyyg1CXgm17u4IzMNcLHvueBl7G14pLQho,16123
74
74
  code_puppy/messaging/spinner/__init__.py,sha256=9mkXPYojafydBOAMh9ZUrB4X6uH5Iqz_-E-Obpd72ko,1365
75
75
  code_puppy/messaging/spinner/console_spinner.py,sha256=DhEXjD37_FuJwcNNfIP2D0y1ruGlFmuCCuaUW4DFrNg,6932
@@ -87,7 +87,7 @@ code_puppy/tui/__init__.py,sha256=XesAxIn32zLPOmvpR2wIDxDAnnJr81a5pBJB4cZp1Xs,32
87
87
  code_puppy/tui/app.py,sha256=nPOzwlusjdWzBfu__EbC3Q0etkPrqRq-2g-mk4IcfG4,39378
88
88
  code_puppy/tui/messages.py,sha256=zQoToWI0eWdT36NEsY6RdCFzcDfAmfvoPlHv8jiCbgo,720
89
89
  code_puppy/tui/components/__init__.py,sha256=uj5pnk3s6SEN3SbFI0ZnzaA2KK1NNg8TfUj6U-Z732U,455
90
- code_puppy/tui/components/chat_view.py,sha256=wKdrCWgoo8p_f9rL9lB6PDuW4eVdMSZe88TVo--sGF0,18547
90
+ code_puppy/tui/components/chat_view.py,sha256=UsFdypI6RGcj0ieCWhbr0yba5Y84JVDQ-KKgbrktPLM,19851
91
91
  code_puppy/tui/components/command_history_modal.py,sha256=pUPEQvoCWa2iUnuMgNwO22y8eUbyw0HpcPH3wAosHvU,7097
92
92
  code_puppy/tui/components/copy_button.py,sha256=E4-OJYk5YNzDf-E81NyiVGKsTRPrUX-RnQ8qFuVnabw,4375
93
93
  code_puppy/tui/components/custom_widgets.py,sha256=pnjkB3ZNa5lwSrAXUFlhN9AHNh4uMTpSap8AdbpecKw,1986
@@ -104,9 +104,9 @@ code_puppy/tui/screens/help.py,sha256=eJuPaOOCp7ZSUlecearqsuX6caxWv7NQszUh0tZJjB
104
104
  code_puppy/tui/screens/mcp_install_wizard.py,sha256=xqwN5omltMkfxWZwXj3D2PbXbtrxUi1dT0XT77oxOKk,27685
105
105
  code_puppy/tui/screens/settings.py,sha256=GMpv-qa08rorAE9mj3AjmqjZFPhmeJ_GWd-DBHG6iAA,10671
106
106
  code_puppy/tui/screens/tools.py,sha256=3pr2Xkpa9Js6Yhf1A3_wQVRzFOui-KDB82LwrsdBtyk,1715
107
- code_puppy-0.0.157.data/data/code_puppy/models.json,sha256=dAfpMMI2EEeOMv0ynHSmMuJAYDLcZrs5gCLX3voC4-A,3252
108
- code_puppy-0.0.157.dist-info/METADATA,sha256=NZLRr--sgoI8Mi0P8lHEzIMO4oLTP9vzTMFVOfYYYmg,19567
109
- code_puppy-0.0.157.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
110
- code_puppy-0.0.157.dist-info/entry_points.txt,sha256=d8YkBvIUxF-dHNJAj-x4fPEqizbY5d_TwvYpc01U5kw,58
111
- code_puppy-0.0.157.dist-info/licenses/LICENSE,sha256=31u8x0SPgdOq3izJX41kgFazWsM43zPEF9eskzqbJMY,1075
112
- code_puppy-0.0.157.dist-info/RECORD,,
107
+ code_puppy-0.0.159.data/data/code_puppy/models.json,sha256=iXmLZGflnQcu2DRh4WUlgAhoXdvoxUc7KBhB8YxawXM,3088
108
+ code_puppy-0.0.159.dist-info/METADATA,sha256=jRdT4LBZq38nevqbxsJkIscP7I4KHAwvf02FdfCmlus,19567
109
+ code_puppy-0.0.159.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
110
+ code_puppy-0.0.159.dist-info/entry_points.txt,sha256=d8YkBvIUxF-dHNJAj-x4fPEqizbY5d_TwvYpc01U5kw,58
111
+ code_puppy-0.0.159.dist-info/licenses/LICENSE,sha256=31u8x0SPgdOq3izJX41kgFazWsM43zPEF9eskzqbJMY,1075
112
+ code_puppy-0.0.159.dist-info/RECORD,,