code-puppy 0.0.377__py3-none-any.whl → 0.0.379__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.
@@ -893,6 +893,7 @@ class AddModelMenu:
893
893
  kb = KeyBindings()
894
894
 
895
895
  @kb.add("up")
896
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
896
897
  def _(event):
897
898
  if self.view_mode == "providers":
898
899
  if self.selected_provider_idx > 0:
@@ -905,6 +906,7 @@ class AddModelMenu:
905
906
  self.update_display()
906
907
 
907
908
  @kb.add("down")
909
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
908
910
  def _(event):
909
911
  if self.view_mode == "providers":
910
912
  if self.selected_provider_idx < len(self.providers) - 1:
@@ -470,6 +470,7 @@ async def interactive_autosave_picker() -> Optional[str]:
470
470
  kb = KeyBindings()
471
471
 
472
472
  @kb.add("up")
473
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
473
474
  def _(event):
474
475
  if browse_mode[0]:
475
476
  # In browse mode: go to older message
@@ -485,6 +486,7 @@ async def interactive_autosave_picker() -> Optional[str]:
485
486
  update_display()
486
487
 
487
488
  @kb.add("down")
489
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
488
490
  def _(event):
489
491
  if browse_mode[0]:
490
492
  # In browse mode: go to newer message
@@ -299,6 +299,7 @@ async def _split_panel_selector(
299
299
  kb = KeyBindings()
300
300
 
301
301
  @kb.add("up")
302
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
302
303
  def move_up(event):
303
304
  if choices:
304
305
  # Skip separator lines
@@ -310,6 +311,7 @@ async def _split_panel_selector(
310
311
  event.app.invalidate()
311
312
 
312
313
  @kb.add("down")
314
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
313
315
  def move_down(event):
314
316
  if choices:
315
317
  # Skip separator lines
@@ -533,6 +533,7 @@ async def _split_panel_selector(
533
533
  kb = KeyBindings()
534
534
 
535
535
  @kb.add("up")
536
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
536
537
  def move_up(event):
537
538
  if choices:
538
539
  selected_index[0] = (selected_index[0] - 1) % len(choices)
@@ -540,6 +541,7 @@ async def _split_panel_selector(
540
541
  event.app.invalidate()
541
542
 
542
543
  @kb.add("down")
544
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
543
545
  def move_down(event):
544
546
  if choices:
545
547
  selected_index[0] = (selected_index[0] + 1) % len(choices)
@@ -743,6 +743,7 @@ class ModelSettingsMenu:
743
743
  kb = KeyBindings()
744
744
 
745
745
  @kb.add("up")
746
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
746
747
  def _(event):
747
748
  if self.view_mode == "models":
748
749
  if self.model_index > 0:
@@ -755,6 +756,7 @@ class ModelSettingsMenu:
755
756
  self.update_display()
756
757
 
757
758
  @kb.add("down")
759
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
758
760
  def _(event):
759
761
  if self.view_mode == "models":
760
762
  if self.model_index < len(self.all_models) - 1:
@@ -249,12 +249,14 @@ async def run_onboarding_wizard() -> Optional[str]:
249
249
 
250
250
  @kb.add("down")
251
251
  @kb.add("j")
252
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
252
253
  def next_option(event):
253
254
  wizard.next_option()
254
255
  event.app.invalidate()
255
256
 
256
257
  @kb.add("up")
257
258
  @kb.add("k")
259
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
258
260
  def prev_option(event):
259
261
  wizard.prev_option()
260
262
  event.app.invalidate()
code_puppy/http_utils.py CHANGED
@@ -102,17 +102,24 @@ class RetryingAsyncClient(httpx.AsyncClient):
102
102
 
103
103
  This replaces the Tenacity transport with a more direct subclass implementation,
104
104
  which plays nicer with proxies and custom transports (like Antigravity).
105
+
106
+ Special handling for Cerebras: Their Retry-After headers are absurdly aggressive
107
+ (often 60s), so we ignore them and use a 3s base backoff instead.
105
108
  """
106
109
 
107
110
  def __init__(
108
111
  self,
109
112
  retry_status_codes: tuple = (429, 502, 503, 504),
110
113
  max_retries: int = 5,
114
+ model_name: str = "",
111
115
  **kwargs,
112
116
  ):
113
117
  super().__init__(**kwargs)
114
118
  self.retry_status_codes = retry_status_codes
115
119
  self.max_retries = max_retries
120
+ self.model_name = model_name.lower() if model_name else ""
121
+ # Cerebras sends crazy aggressive Retry-After headers (60s), ignore them
122
+ self._ignore_retry_headers = "cerebras" in self.model_name
116
123
 
117
124
  async def send(self, request: httpx.Request, **kwargs: Any) -> httpx.Response:
118
125
  """Send request with automatic retries for rate limits and server errors."""
@@ -131,32 +138,39 @@ class RetryingAsyncClient(httpx.AsyncClient):
131
138
  # Close response if we're going to retry
132
139
  await response.aclose()
133
140
 
134
- # Determine wait time
135
- wait_time = 1.0 * (
136
- 2**attempt
137
- ) # Default exponential backoff: 1s, 2s, 4s...
138
-
139
- # Check Retry-After header
140
- retry_after = response.headers.get("Retry-After")
141
- if retry_after:
142
- try:
143
- wait_time = float(retry_after)
144
- except ValueError:
145
- # Try parsing http-date
146
- from email.utils import parsedate_to_datetime
141
+ # Determine wait time - Cerebras gets special treatment
142
+ if self._ignore_retry_headers:
143
+ # Cerebras: 3s base with exponential backoff (3s, 6s, 12s...)
144
+ wait_time = 3.0 * (2**attempt)
145
+ else:
146
+ # Default exponential backoff: 1s, 2s, 4s...
147
+ wait_time = 1.0 * (2**attempt)
147
148
 
149
+ # Check Retry-After header (only for non-Cerebras)
150
+ retry_after = response.headers.get("Retry-After")
151
+ if retry_after:
148
152
  try:
149
- date = parsedate_to_datetime(retry_after)
150
- wait_time = date.timestamp() - time.time()
151
- except Exception:
152
- pass
153
+ wait_time = float(retry_after)
154
+ except ValueError:
155
+ # Try parsing http-date
156
+ from email.utils import parsedate_to_datetime
157
+
158
+ try:
159
+ date = parsedate_to_datetime(retry_after)
160
+ wait_time = date.timestamp() - time.time()
161
+ except Exception:
162
+ pass
153
163
 
154
164
  # Cap wait time
155
165
  wait_time = max(0.5, min(wait_time, 60.0))
156
166
 
157
167
  if attempt < self.max_retries:
168
+ provider_note = (
169
+ " (ignoring header)" if self._ignore_retry_headers else ""
170
+ )
158
171
  emit_info(
159
- f"HTTP retry: {response.status_code} received. Waiting {wait_time:.1f}s (attempt {attempt + 1}/{self.max_retries})"
172
+ f"HTTP retry: {response.status_code} received{provider_note}. "
173
+ f"Waiting {wait_time:.1f}s (attempt {attempt + 1}/{self.max_retries})"
160
174
  )
161
175
  await asyncio.sleep(wait_time)
162
176
 
@@ -219,12 +233,14 @@ def create_async_client(
219
233
  verify: Union[bool, str] = None,
220
234
  headers: Optional[Dict[str, str]] = None,
221
235
  retry_status_codes: tuple = (429, 502, 503, 504),
236
+ model_name: str = "",
222
237
  ) -> httpx.AsyncClient:
223
238
  config = _resolve_proxy_config(verify)
224
239
 
225
240
  if not config.disable_retry:
226
241
  return RetryingAsyncClient(
227
242
  retry_status_codes=retry_status_codes,
243
+ model_name=model_name,
228
244
  proxy=config.proxy_url,
229
245
  verify=config.verify,
230
246
  headers=headers or {},
@@ -290,6 +306,7 @@ def create_reopenable_async_client(
290
306
  verify: Union[bool, str] = None,
291
307
  headers: Optional[Dict[str, str]] = None,
292
308
  retry_status_codes: tuple = (429, 502, 503, 504),
309
+ model_name: str = "",
293
310
  ) -> Union[ReopenableAsyncClient, httpx.AsyncClient]:
294
311
  config = _resolve_proxy_config(verify)
295
312
 
@@ -309,12 +326,15 @@ def create_reopenable_async_client(
309
326
  kwargs = {**base_kwargs, "client_class": client_class}
310
327
  if not config.disable_retry:
311
328
  kwargs["retry_status_codes"] = retry_status_codes
329
+ kwargs["model_name"] = model_name
312
330
  return ReopenableAsyncClient(**kwargs)
313
331
  else:
314
332
  # Fallback to RetryingAsyncClient or plain AsyncClient
315
333
  if not config.disable_retry:
316
334
  return RetryingAsyncClient(
317
- retry_status_codes=retry_status_codes, **base_kwargs
335
+ retry_status_codes=retry_status_codes,
336
+ model_name=model_name,
337
+ **base_kwargs,
318
338
  )
319
339
  else:
320
340
  return httpx.AsyncClient(**base_kwargs)
@@ -585,7 +585,12 @@ class ModelFactory:
585
585
  return None
586
586
  # Add Cerebras 3rd party integration header
587
587
  headers["X-Cerebras-3rd-Party-Integration"] = "code-puppy"
588
- client = create_async_client(headers=headers, verify=verify)
588
+ # Pass "cerebras" so RetryingAsyncClient knows to ignore Cerebras's
589
+ # absurdly aggressive Retry-After headers (they send 60s!)
590
+ # Note: model_config["name"] is "zai-glm-4.7", not "cerebras"
591
+ client = create_async_client(
592
+ headers=headers, verify=verify, model_name="cerebras"
593
+ )
589
594
  provider_args = dict(
590
595
  api_key=api_key,
591
596
  http_client=client,
@@ -880,18 +880,22 @@ async def arrow_select_async(
880
880
  lines.append(border_bottom)
881
881
  lines.append("")
882
882
 
883
- lines.append("<ansicyan>(Use ↑↓ arrows to select, Enter to confirm)</ansicyan>")
883
+ lines.append(
884
+ "<ansicyan>(Use ↑↓ or Ctrl+P/N to select, Enter to confirm)</ansicyan>"
885
+ )
884
886
  return HTML("\n".join(lines))
885
887
 
886
888
  # Key bindings
887
889
  kb = KeyBindings()
888
890
 
889
891
  @kb.add("up")
892
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
890
893
  def move_up(event):
891
894
  selected_index[0] = (selected_index[0] - 1) % len(choices)
892
895
  event.app.invalidate() # Force redraw to update preview
893
896
 
894
897
  @kb.add("down")
898
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
895
899
  def move_down(event):
896
900
  selected_index[0] = (selected_index[0] + 1) % len(choices)
897
901
  event.app.invalidate() # Force redraw to update preview
@@ -957,18 +961,22 @@ def arrow_select(message: str, choices: list[str]) -> str:
957
961
  else:
958
962
  lines.append(f" {choice}")
959
963
  lines.append("")
960
- lines.append("<ansicyan>(Use ↑↓ arrows to select, Enter to confirm)</ansicyan>")
964
+ lines.append(
965
+ "<ansicyan>(Use ↑↓ or Ctrl+P/N to select, Enter to confirm)</ansicyan>"
966
+ )
961
967
  return HTML("\n".join(lines))
962
968
 
963
969
  # Key bindings
964
970
  kb = KeyBindings()
965
971
 
966
972
  @kb.add("up")
973
+ @kb.add("c-p") # Ctrl+P = previous (Emacs-style)
967
974
  def move_up(event):
968
975
  selected_index[0] = (selected_index[0] - 1) % len(choices)
969
976
  event.app.invalidate() # Force redraw to update preview
970
977
 
971
978
  @kb.add("down")
979
+ @kb.add("c-n") # Ctrl+N = next (Emacs-style)
972
980
  def move_down(event):
973
981
  selected_index[0] = (selected_index[0] + 1) % len(choices)
974
982
  event.app.invalidate() # Force redraw to update preview
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: code-puppy
3
- Version: 0.0.377
3
+ Version: 0.0.379
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
@@ -8,10 +8,10 @@ code_puppy/config.py,sha256=eAyVqiu8SwzJQpaJu80rlJvd8XXY51DqafMsP5lBRI4,55827
8
8
  code_puppy/error_logging.py,sha256=a80OILCUtJhexI6a9GM-r5LqIdjvSRzggfgPp2jv1X0,3297
9
9
  code_puppy/gemini_code_assist.py,sha256=KGS7sO5OLc83nDF3xxS-QiU6vxW9vcm6hmzilu79Ef8,13867
10
10
  code_puppy/gemini_model.py,sha256=UHb5vFC9zrEdFJ-yCN3vNCdp0UxP156BL_fwbnEhaw8,27988
11
- code_puppy/http_utils.py,sha256=SAH6EOdbR6Cbfmi-4EtHDqRDBUV5bWtGc-5nr44F0Is,10418
11
+ code_puppy/http_utils.py,sha256=Xel6UyQGO6X_5BlrtPYO1hVXN24aqP72tn5g1uUkluU,11491
12
12
  code_puppy/keymap.py,sha256=IvMkTlB_bIqOWpbTpmftkdyjhtD5todXuEIw1zCZ4u0,3584
13
13
  code_puppy/main.py,sha256=82r3vZy_XcyEsenLn82BnUusaoyL3Bpm_Th_jKgqecE,273
14
- code_puppy/model_factory.py,sha256=VFMUY0u8oqHn9f1QqpWsxvXujlK0Ku5NEwfuaTcv2rs,32060
14
+ code_puppy/model_factory.py,sha256=5ndms7O0_bJftizpJMbsdYs3geixn3vZKL2105zEgmM,32338
15
15
  code_puppy/model_switching.py,sha256=3IsnSWKHLWzI5d2WDYNg0Xr78BeYNN1WrZuzas-lYJ4,2064
16
16
  code_puppy/model_utils.py,sha256=cG1V4fsIOEQIb0W88FyGcsMWNv8SpmRAXN3A7LBEoyE,5116
17
17
  code_puppy/models.json,sha256=SC7N2lV1Q8ikXlalRNqABkNvuuL_8fIIk638739-gGY,3319
@@ -69,25 +69,25 @@ code_puppy/api/routers/config.py,sha256=uDUFYZqki0fQd0U5EHpfTgqlZaRFfmhPyWrIHXNB
69
69
  code_puppy/api/routers/sessions.py,sha256=GqYRT7IJYPpEdTseLF3FIpbvvD86lIqwwPswL31D9Wc,6786
70
70
  code_puppy/api/templates/terminal.html,sha256=9alh6tTbLyXPDjBvkXw8nEWPXB-m_LIceGGRYpSLuyo,13125
71
71
  code_puppy/command_line/__init__.py,sha256=y7WeRemfYppk8KVbCGeAIiTuiOszIURCDjOMZv_YRmU,45
72
- code_puppy/command_line/add_model_menu.py,sha256=nKE5KgecxFvR_9EDy1W_swVvMs97Qg8f1gZETpBjMUc,43479
72
+ code_puppy/command_line/add_model_menu.py,sha256=xsPhJYpqHv6mbNc5NVz46N5XzdJz1wbtDg3LaP2KRK0,43591
73
73
  code_puppy/command_line/agent_menu.py,sha256=4SVPS0eA7YfpxacNk0Kel16bzqQ3bBGe8dqCCOI2A8s,20915
74
74
  code_puppy/command_line/attachments.py,sha256=4Q5I2Es4j0ltnz5wjw2z0QXMsiMJvEfWRkPf_lJeITM,13093
75
- code_puppy/command_line/autosave_menu.py,sha256=de7nOmFmEH6x5T7C95U8N8xgxxeF-l5lgaJzGJsF3ZY,19824
75
+ code_puppy/command_line/autosave_menu.py,sha256=41YBToiWGglTKEdEBeNTqI2D3wO4BCBfv0WXZILOhnw,19928
76
76
  code_puppy/command_line/clipboard.py,sha256=oe9bfAX5RnT81FiYrDmhvHaePS1tAT-NFG1fSXubSD4,16869
77
- code_puppy/command_line/colors_menu.py,sha256=tozEzjxqxps8JnEaIADEveOO7eihjA5gUYnicHntTPA,17431
77
+ code_puppy/command_line/colors_menu.py,sha256=AI36z4vicbhwDQkziQZXCfy_F59bJLYs-1p9-N7Enn0,17535
78
78
  code_puppy/command_line/command_handler.py,sha256=yDR_NAvgwjFKxUuQMY5yzkkihFovDiT24wyKALmJSRU,10969
79
79
  code_puppy/command_line/command_registry.py,sha256=qFySsw1g8dol3kgi0p6cXrIDlP11_OhOoaQ5nAadWXg,4416
80
80
  code_puppy/command_line/config_commands.py,sha256=uwS7ln0I0w0A3tSSWZEtq7Xgj-VOaGEYLE1CihdQ8ic,25759
81
81
  code_puppy/command_line/core_commands.py,sha256=QTQt2CS9_6ExcgS6BLgRZWkXDaSb-KC_tWplUkOGaMA,27133
82
- code_puppy/command_line/diff_menu.py,sha256=_Gr9SP9fbItk-08dya9WTAR53s_PlyAvEnbt-8VWKPk,24141
82
+ code_puppy/command_line/diff_menu.py,sha256=tokSiQXv9cWDd7HTBEkKKnxZjwSNyVjpIReXe-K2vL4,24245
83
83
  code_puppy/command_line/file_path_completion.py,sha256=gw8NpIxa6GOpczUJRyh7VNZwoXKKn-yvCqit7h2y6Gg,2931
84
84
  code_puppy/command_line/load_context_completion.py,sha256=a3JvLDeLLSYxVgTjAdqWzS4spjv6ccCrK2LKZgVJ1IM,2202
85
85
  code_puppy/command_line/mcp_completion.py,sha256=eKzW2O7gun7HoHekOW0XVXhNS5J2xCtK7aaWyA8bkZk,6952
86
86
  code_puppy/command_line/model_picker_completion.py,sha256=YRudzwGVtIjr02MyeIdmbkDhS00ENjCt9k3nATT3KdM,6143
87
- code_puppy/command_line/model_settings_menu.py,sha256=RFdYe0CGApu-BE5lubRfZ5lBaAjt_MeX5AUmOdaAt-s,33596
87
+ code_puppy/command_line/model_settings_menu.py,sha256=1SrhwmiL8fTqaM-frV0mSnssiVvjKfjMNoZAm1ofiYo,33708
88
88
  code_puppy/command_line/motd.py,sha256=XuIk3UTLawwVFM-NfoaJGU5F2hPLASTFXq84UdDMT0Q,2408
89
89
  code_puppy/command_line/onboarding_slides.py,sha256=itqAsuHzjHpD_XNz6FniBIYr6dNyP1AW_XQZQ6SbVek,7125
90
- code_puppy/command_line/onboarding_wizard.py,sha256=U5lV_1P3IwDYZUHar0zKgdp121zzkvOwwORvdCZwFcw,10241
90
+ code_puppy/command_line/onboarding_wizard.py,sha256=mAFmQsPU8N48MGxHXOPBFRz7zeC4otdlqM6MpYVmrrc,10353
91
91
  code_puppy/command_line/pin_command_completion.py,sha256=juSvdqRpk7AdfkPy1DJx5NzfEUU5KYGlChvP0hisM18,11667
92
92
  code_puppy/command_line/prompt_toolkit_completion.py,sha256=49GM3jVE89G1M3XroMZk2LhGgXpOO8XZ0Sg8h4a6LLw,32806
93
93
  code_puppy/command_line/session_commands.py,sha256=Jh8GGfhlfBAEVfucKLbcZjNaXYd0twImiOwq2ZnGdQQ,9902
@@ -205,7 +205,7 @@ code_puppy/prompts/antigravity_system_prompt.md,sha256=ZaTfRyY57ttROyZMmOBtqZQu1
205
205
  code_puppy/tools/__init__.py,sha256=VsXs27kCtgq63LnkHMrtH39q13zjezvayjdiyges_Io,13919
206
206
  code_puppy/tools/agent_tools.py,sha256=xDO0LPc0YUQHSqA29SEQH4t9hcbyApaiUWWgSdx6wE4,25511
207
207
  code_puppy/tools/command_runner.py,sha256=C3wGcxt6slC1d5X4twKSoUjGdGYi02k2sAQSygoOt54,51518
208
- code_puppy/tools/common.py,sha256=lVtF94cn6jtC5YKfitV7L3rk37Ts2gMoHLQrqDFD2E4,46411
208
+ code_puppy/tools/common.py,sha256=wNogZr0glUxKhJoF1S2iVsT8SAXjn_rXYIyY8__l8YI,46673
209
209
  code_puppy/tools/display.py,sha256=-ulDyq55178f8O_TAEmnxGoy_ZdFkbHBw-W4ul851GM,2675
210
210
  code_puppy/tools/file_modifications.py,sha256=vz9n7R0AGDSdLUArZr_55yJLkyI30M8zreAppxIx02M,29380
211
211
  code_puppy/tools/file_operations.py,sha256=CqhpuBnOFOcQCIYXOujskxq2VMLWYJhibYrH0YcPSfA,35692
@@ -225,10 +225,10 @@ code_puppy/tools/browser/chromium_terminal_manager.py,sha256=w1thQ_ACb6oV45L93TS
225
225
  code_puppy/tools/browser/terminal_command_tools.py,sha256=9byOZku-dwvTtCl532xt7Lumed_jTn0sLvUe_X75XCQ,19068
226
226
  code_puppy/tools/browser/terminal_screenshot_tools.py,sha256=J_21YO_495NvYgNFu9KQP6VYg2K_f8CtSdZuF94Yhnw,18448
227
227
  code_puppy/tools/browser/terminal_tools.py,sha256=F5LjVH3udSCFHmqC3O1UJLoLozZFZsEdX42jOmkqkW0,17853
228
- code_puppy-0.0.377.data/data/code_puppy/models.json,sha256=SC7N2lV1Q8ikXlalRNqABkNvuuL_8fIIk638739-gGY,3319
229
- code_puppy-0.0.377.data/data/code_puppy/models_dev_api.json,sha256=wHjkj-IM_fx1oHki6-GqtOoCrRMR0ScK0f-Iz0UEcy8,548187
230
- code_puppy-0.0.377.dist-info/METADATA,sha256=h4-FOgFscu0NlRuJK2QQS5-zzmCVUzx8wod813P7sPU,27604
231
- code_puppy-0.0.377.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
232
- code_puppy-0.0.377.dist-info/entry_points.txt,sha256=Tp4eQC99WY3HOKd3sdvb22vZODRq0XkZVNpXOag_KdI,91
233
- code_puppy-0.0.377.dist-info/licenses/LICENSE,sha256=31u8x0SPgdOq3izJX41kgFazWsM43zPEF9eskzqbJMY,1075
234
- code_puppy-0.0.377.dist-info/RECORD,,
228
+ code_puppy-0.0.379.data/data/code_puppy/models.json,sha256=SC7N2lV1Q8ikXlalRNqABkNvuuL_8fIIk638739-gGY,3319
229
+ code_puppy-0.0.379.data/data/code_puppy/models_dev_api.json,sha256=wHjkj-IM_fx1oHki6-GqtOoCrRMR0ScK0f-Iz0UEcy8,548187
230
+ code_puppy-0.0.379.dist-info/METADATA,sha256=ldoaFKnfpIocHpmV_ZBTVEjfW0l7YWxIX7qRXJt1rQM,27604
231
+ code_puppy-0.0.379.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
232
+ code_puppy-0.0.379.dist-info/entry_points.txt,sha256=Tp4eQC99WY3HOKd3sdvb22vZODRq0XkZVNpXOag_KdI,91
233
+ code_puppy-0.0.379.dist-info/licenses/LICENSE,sha256=31u8x0SPgdOq3izJX41kgFazWsM43zPEF9eskzqbJMY,1075
234
+ code_puppy-0.0.379.dist-info/RECORD,,