code-puppy 0.0.169__py3-none-any.whl → 0.0.366__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.
Files changed (243) hide show
  1. code_puppy/__init__.py +7 -1
  2. code_puppy/agents/__init__.py +8 -8
  3. code_puppy/agents/agent_c_reviewer.py +155 -0
  4. code_puppy/agents/agent_code_puppy.py +9 -2
  5. code_puppy/agents/agent_code_reviewer.py +90 -0
  6. code_puppy/agents/agent_cpp_reviewer.py +132 -0
  7. code_puppy/agents/agent_creator_agent.py +48 -9
  8. code_puppy/agents/agent_golang_reviewer.py +151 -0
  9. code_puppy/agents/agent_javascript_reviewer.py +160 -0
  10. code_puppy/agents/agent_manager.py +146 -199
  11. code_puppy/agents/agent_pack_leader.py +383 -0
  12. code_puppy/agents/agent_planning.py +163 -0
  13. code_puppy/agents/agent_python_programmer.py +165 -0
  14. code_puppy/agents/agent_python_reviewer.py +90 -0
  15. code_puppy/agents/agent_qa_expert.py +163 -0
  16. code_puppy/agents/agent_qa_kitten.py +208 -0
  17. code_puppy/agents/agent_security_auditor.py +181 -0
  18. code_puppy/agents/agent_terminal_qa.py +323 -0
  19. code_puppy/agents/agent_typescript_reviewer.py +166 -0
  20. code_puppy/agents/base_agent.py +1713 -1
  21. code_puppy/agents/event_stream_handler.py +350 -0
  22. code_puppy/agents/json_agent.py +12 -1
  23. code_puppy/agents/pack/__init__.py +34 -0
  24. code_puppy/agents/pack/bloodhound.py +304 -0
  25. code_puppy/agents/pack/husky.py +321 -0
  26. code_puppy/agents/pack/retriever.py +393 -0
  27. code_puppy/agents/pack/shepherd.py +348 -0
  28. code_puppy/agents/pack/terrier.py +287 -0
  29. code_puppy/agents/pack/watchdog.py +367 -0
  30. code_puppy/agents/prompt_reviewer.py +145 -0
  31. code_puppy/agents/subagent_stream_handler.py +276 -0
  32. code_puppy/api/__init__.py +13 -0
  33. code_puppy/api/app.py +169 -0
  34. code_puppy/api/main.py +21 -0
  35. code_puppy/api/pty_manager.py +446 -0
  36. code_puppy/api/routers/__init__.py +12 -0
  37. code_puppy/api/routers/agents.py +36 -0
  38. code_puppy/api/routers/commands.py +217 -0
  39. code_puppy/api/routers/config.py +74 -0
  40. code_puppy/api/routers/sessions.py +232 -0
  41. code_puppy/api/templates/terminal.html +361 -0
  42. code_puppy/api/websocket.py +154 -0
  43. code_puppy/callbacks.py +174 -4
  44. code_puppy/chatgpt_codex_client.py +283 -0
  45. code_puppy/claude_cache_client.py +586 -0
  46. code_puppy/cli_runner.py +916 -0
  47. code_puppy/command_line/add_model_menu.py +1079 -0
  48. code_puppy/command_line/agent_menu.py +395 -0
  49. code_puppy/command_line/attachments.py +395 -0
  50. code_puppy/command_line/autosave_menu.py +605 -0
  51. code_puppy/command_line/clipboard.py +527 -0
  52. code_puppy/command_line/colors_menu.py +520 -0
  53. code_puppy/command_line/command_handler.py +233 -627
  54. code_puppy/command_line/command_registry.py +150 -0
  55. code_puppy/command_line/config_commands.py +715 -0
  56. code_puppy/command_line/core_commands.py +792 -0
  57. code_puppy/command_line/diff_menu.py +863 -0
  58. code_puppy/command_line/load_context_completion.py +15 -22
  59. code_puppy/command_line/mcp/base.py +1 -4
  60. code_puppy/command_line/mcp/catalog_server_installer.py +175 -0
  61. code_puppy/command_line/mcp/custom_server_form.py +688 -0
  62. code_puppy/command_line/mcp/custom_server_installer.py +195 -0
  63. code_puppy/command_line/mcp/edit_command.py +148 -0
  64. code_puppy/command_line/mcp/handler.py +9 -4
  65. code_puppy/command_line/mcp/help_command.py +6 -5
  66. code_puppy/command_line/mcp/install_command.py +16 -27
  67. code_puppy/command_line/mcp/install_menu.py +685 -0
  68. code_puppy/command_line/mcp/list_command.py +3 -3
  69. code_puppy/command_line/mcp/logs_command.py +174 -65
  70. code_puppy/command_line/mcp/remove_command.py +2 -2
  71. code_puppy/command_line/mcp/restart_command.py +12 -4
  72. code_puppy/command_line/mcp/search_command.py +17 -11
  73. code_puppy/command_line/mcp/start_all_command.py +22 -13
  74. code_puppy/command_line/mcp/start_command.py +50 -31
  75. code_puppy/command_line/mcp/status_command.py +6 -7
  76. code_puppy/command_line/mcp/stop_all_command.py +11 -8
  77. code_puppy/command_line/mcp/stop_command.py +11 -10
  78. code_puppy/command_line/mcp/test_command.py +2 -2
  79. code_puppy/command_line/mcp/utils.py +1 -1
  80. code_puppy/command_line/mcp/wizard_utils.py +22 -18
  81. code_puppy/command_line/mcp_completion.py +174 -0
  82. code_puppy/command_line/model_picker_completion.py +89 -30
  83. code_puppy/command_line/model_settings_menu.py +884 -0
  84. code_puppy/command_line/motd.py +14 -8
  85. code_puppy/command_line/onboarding_slides.py +179 -0
  86. code_puppy/command_line/onboarding_wizard.py +340 -0
  87. code_puppy/command_line/pin_command_completion.py +329 -0
  88. code_puppy/command_line/prompt_toolkit_completion.py +626 -75
  89. code_puppy/command_line/session_commands.py +296 -0
  90. code_puppy/command_line/utils.py +54 -0
  91. code_puppy/config.py +1181 -51
  92. code_puppy/error_logging.py +118 -0
  93. code_puppy/gemini_code_assist.py +385 -0
  94. code_puppy/gemini_model.py +602 -0
  95. code_puppy/http_utils.py +220 -104
  96. code_puppy/keymap.py +128 -0
  97. code_puppy/main.py +5 -594
  98. code_puppy/{mcp → mcp_}/__init__.py +17 -0
  99. code_puppy/{mcp → mcp_}/async_lifecycle.py +35 -4
  100. code_puppy/{mcp → mcp_}/blocking_startup.py +70 -43
  101. code_puppy/{mcp → mcp_}/captured_stdio_server.py +2 -2
  102. code_puppy/{mcp → mcp_}/config_wizard.py +5 -5
  103. code_puppy/{mcp → mcp_}/dashboard.py +15 -6
  104. code_puppy/{mcp → mcp_}/examples/retry_example.py +4 -1
  105. code_puppy/{mcp → mcp_}/managed_server.py +66 -39
  106. code_puppy/{mcp → mcp_}/manager.py +146 -52
  107. code_puppy/mcp_/mcp_logs.py +224 -0
  108. code_puppy/{mcp → mcp_}/registry.py +6 -6
  109. code_puppy/{mcp → mcp_}/server_registry_catalog.py +25 -8
  110. code_puppy/messaging/__init__.py +199 -2
  111. code_puppy/messaging/bus.py +610 -0
  112. code_puppy/messaging/commands.py +167 -0
  113. code_puppy/messaging/markdown_patches.py +57 -0
  114. code_puppy/messaging/message_queue.py +17 -48
  115. code_puppy/messaging/messages.py +500 -0
  116. code_puppy/messaging/queue_console.py +1 -24
  117. code_puppy/messaging/renderers.py +43 -146
  118. code_puppy/messaging/rich_renderer.py +1027 -0
  119. code_puppy/messaging/spinner/__init__.py +33 -5
  120. code_puppy/messaging/spinner/console_spinner.py +92 -52
  121. code_puppy/messaging/spinner/spinner_base.py +29 -0
  122. code_puppy/messaging/subagent_console.py +461 -0
  123. code_puppy/model_factory.py +686 -80
  124. code_puppy/model_utils.py +167 -0
  125. code_puppy/models.json +86 -104
  126. code_puppy/models_dev_api.json +1 -0
  127. code_puppy/models_dev_parser.py +592 -0
  128. code_puppy/plugins/__init__.py +164 -10
  129. code_puppy/plugins/antigravity_oauth/__init__.py +10 -0
  130. code_puppy/plugins/antigravity_oauth/accounts.py +406 -0
  131. code_puppy/plugins/antigravity_oauth/antigravity_model.py +704 -0
  132. code_puppy/plugins/antigravity_oauth/config.py +42 -0
  133. code_puppy/plugins/antigravity_oauth/constants.py +136 -0
  134. code_puppy/plugins/antigravity_oauth/oauth.py +478 -0
  135. code_puppy/plugins/antigravity_oauth/register_callbacks.py +406 -0
  136. code_puppy/plugins/antigravity_oauth/storage.py +271 -0
  137. code_puppy/plugins/antigravity_oauth/test_plugin.py +319 -0
  138. code_puppy/plugins/antigravity_oauth/token.py +167 -0
  139. code_puppy/plugins/antigravity_oauth/transport.py +767 -0
  140. code_puppy/plugins/antigravity_oauth/utils.py +169 -0
  141. code_puppy/plugins/chatgpt_oauth/__init__.py +8 -0
  142. code_puppy/plugins/chatgpt_oauth/config.py +52 -0
  143. code_puppy/plugins/chatgpt_oauth/oauth_flow.py +328 -0
  144. code_puppy/plugins/chatgpt_oauth/register_callbacks.py +94 -0
  145. code_puppy/plugins/chatgpt_oauth/test_plugin.py +293 -0
  146. code_puppy/plugins/chatgpt_oauth/utils.py +489 -0
  147. code_puppy/plugins/claude_code_oauth/README.md +167 -0
  148. code_puppy/plugins/claude_code_oauth/SETUP.md +93 -0
  149. code_puppy/plugins/claude_code_oauth/__init__.py +6 -0
  150. code_puppy/plugins/claude_code_oauth/config.py +50 -0
  151. code_puppy/plugins/claude_code_oauth/register_callbacks.py +308 -0
  152. code_puppy/plugins/claude_code_oauth/test_plugin.py +283 -0
  153. code_puppy/plugins/claude_code_oauth/utils.py +518 -0
  154. code_puppy/plugins/customizable_commands/__init__.py +0 -0
  155. code_puppy/plugins/customizable_commands/register_callbacks.py +169 -0
  156. code_puppy/plugins/example_custom_command/README.md +280 -0
  157. code_puppy/plugins/example_custom_command/register_callbacks.py +51 -0
  158. code_puppy/plugins/file_permission_handler/__init__.py +4 -0
  159. code_puppy/plugins/file_permission_handler/register_callbacks.py +523 -0
  160. code_puppy/plugins/frontend_emitter/__init__.py +25 -0
  161. code_puppy/plugins/frontend_emitter/emitter.py +121 -0
  162. code_puppy/plugins/frontend_emitter/register_callbacks.py +261 -0
  163. code_puppy/plugins/oauth_puppy_html.py +228 -0
  164. code_puppy/plugins/shell_safety/__init__.py +6 -0
  165. code_puppy/plugins/shell_safety/agent_shell_safety.py +69 -0
  166. code_puppy/plugins/shell_safety/command_cache.py +156 -0
  167. code_puppy/plugins/shell_safety/register_callbacks.py +202 -0
  168. code_puppy/prompts/antigravity_system_prompt.md +1 -0
  169. code_puppy/prompts/codex_system_prompt.md +310 -0
  170. code_puppy/pydantic_patches.py +131 -0
  171. code_puppy/reopenable_async_client.py +8 -8
  172. code_puppy/round_robin_model.py +10 -15
  173. code_puppy/session_storage.py +294 -0
  174. code_puppy/status_display.py +21 -4
  175. code_puppy/summarization_agent.py +52 -14
  176. code_puppy/terminal_utils.py +418 -0
  177. code_puppy/tools/__init__.py +139 -6
  178. code_puppy/tools/agent_tools.py +548 -49
  179. code_puppy/tools/browser/__init__.py +37 -0
  180. code_puppy/tools/browser/browser_control.py +289 -0
  181. code_puppy/tools/browser/browser_interactions.py +545 -0
  182. code_puppy/tools/browser/browser_locators.py +640 -0
  183. code_puppy/tools/browser/browser_manager.py +316 -0
  184. code_puppy/tools/browser/browser_navigation.py +251 -0
  185. code_puppy/tools/browser/browser_screenshot.py +179 -0
  186. code_puppy/tools/browser/browser_scripts.py +462 -0
  187. code_puppy/tools/browser/browser_workflows.py +221 -0
  188. code_puppy/tools/browser/chromium_terminal_manager.py +259 -0
  189. code_puppy/tools/browser/terminal_command_tools.py +521 -0
  190. code_puppy/tools/browser/terminal_screenshot_tools.py +556 -0
  191. code_puppy/tools/browser/terminal_tools.py +525 -0
  192. code_puppy/tools/command_runner.py +941 -153
  193. code_puppy/tools/common.py +1146 -6
  194. code_puppy/tools/display.py +84 -0
  195. code_puppy/tools/file_modifications.py +288 -89
  196. code_puppy/tools/file_operations.py +352 -266
  197. code_puppy/tools/subagent_context.py +158 -0
  198. code_puppy/uvx_detection.py +242 -0
  199. code_puppy/version_checker.py +30 -11
  200. code_puppy-0.0.366.data/data/code_puppy/models.json +110 -0
  201. code_puppy-0.0.366.data/data/code_puppy/models_dev_api.json +1 -0
  202. {code_puppy-0.0.169.dist-info → code_puppy-0.0.366.dist-info}/METADATA +184 -67
  203. code_puppy-0.0.366.dist-info/RECORD +217 -0
  204. {code_puppy-0.0.169.dist-info → code_puppy-0.0.366.dist-info}/WHEEL +1 -1
  205. {code_puppy-0.0.169.dist-info → code_puppy-0.0.366.dist-info}/entry_points.txt +1 -0
  206. code_puppy/agent.py +0 -231
  207. code_puppy/agents/agent_orchestrator.json +0 -26
  208. code_puppy/agents/runtime_manager.py +0 -272
  209. code_puppy/command_line/mcp/add_command.py +0 -183
  210. code_puppy/command_line/meta_command_handler.py +0 -153
  211. code_puppy/message_history_processor.py +0 -490
  212. code_puppy/messaging/spinner/textual_spinner.py +0 -101
  213. code_puppy/state_management.py +0 -200
  214. code_puppy/tui/__init__.py +0 -10
  215. code_puppy/tui/app.py +0 -986
  216. code_puppy/tui/components/__init__.py +0 -21
  217. code_puppy/tui/components/chat_view.py +0 -550
  218. code_puppy/tui/components/command_history_modal.py +0 -218
  219. code_puppy/tui/components/copy_button.py +0 -139
  220. code_puppy/tui/components/custom_widgets.py +0 -63
  221. code_puppy/tui/components/human_input_modal.py +0 -175
  222. code_puppy/tui/components/input_area.py +0 -167
  223. code_puppy/tui/components/sidebar.py +0 -309
  224. code_puppy/tui/components/status_bar.py +0 -182
  225. code_puppy/tui/messages.py +0 -27
  226. code_puppy/tui/models/__init__.py +0 -8
  227. code_puppy/tui/models/chat_message.py +0 -25
  228. code_puppy/tui/models/command_history.py +0 -89
  229. code_puppy/tui/models/enums.py +0 -24
  230. code_puppy/tui/screens/__init__.py +0 -15
  231. code_puppy/tui/screens/help.py +0 -130
  232. code_puppy/tui/screens/mcp_install_wizard.py +0 -803
  233. code_puppy/tui/screens/settings.py +0 -290
  234. code_puppy/tui/screens/tools.py +0 -74
  235. code_puppy-0.0.169.data/data/code_puppy/models.json +0 -128
  236. code_puppy-0.0.169.dist-info/RECORD +0 -112
  237. /code_puppy/{mcp → mcp_}/circuit_breaker.py +0 -0
  238. /code_puppy/{mcp → mcp_}/error_isolation.py +0 -0
  239. /code_puppy/{mcp → mcp_}/health_monitor.py +0 -0
  240. /code_puppy/{mcp → mcp_}/retry_manager.py +0 -0
  241. /code_puppy/{mcp → mcp_}/status_tracker.py +0 -0
  242. /code_puppy/{mcp → mcp_}/system_tools.py +0 -0
  243. {code_puppy-0.0.169.dist-info → code_puppy-0.0.366.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,280 @@
1
+ # Example Custom Command Plugin
2
+
3
+ > **Note**: This example demonstrates **custom commands** via the callback system.
4
+ > For **built-in commands**, see the built-in command files in `code_puppy/command_line/`.
5
+
6
+ ## Overview
7
+
8
+ This plugin demonstrates how to create custom commands using Code Puppy's callback system.
9
+
10
+ **Important**: Custom commands use `register_callback()`, NOT `@register_command`.
11
+
12
+ ## Command Types in Code Puppy
13
+
14
+ ### 1. Built-in Commands (Core Functionality)
15
+ - Use `@register_command` decorator
16
+ - Located in `code_puppy/command_line/core_commands.py`, `session_commands.py`, `config_commands.py`
17
+ - Examples: `/help`, `/cd`, `/set`, `/agent`
18
+ - Check those files for implementation examples
19
+
20
+ ### 2. Custom Commands (Plugins) ← **This Example**
21
+ - Use `register_callback()` function
22
+ - Located in plugin directories like this one
23
+ - Examples: `/woof`, `/echo` (from this plugin)
24
+ - Designed for plugin-specific functionality
25
+
26
+ ## How This Plugin Works
27
+
28
+ ### File Structure
29
+
30
+ ```
31
+ code_puppy/plugins/example_custom_command/
32
+ ├── register_callbacks.py # Plugin implementation
33
+ └── README.md # This file
34
+ ```
35
+
36
+ ### Implementation
37
+
38
+ ```python
39
+ from code_puppy.callbacks import register_callback
40
+ from code_puppy.messaging import emit_info
41
+
42
+ # 1. Define help entries for your commands
43
+ def _custom_help():
44
+ return [
45
+ ("woof", "Emit a playful woof message (no model)"),
46
+ ("echo", "Echo back your text (display only)"),
47
+ ]
48
+
49
+ # 2. Define command handler
50
+ def _handle_custom_command(command: str, name: str):
51
+ """Handle custom commands.
52
+
53
+ Args:
54
+ command: Full command string (e.g., "/woof something")
55
+ name: Command name without slash (e.g., "woof")
56
+
57
+ Returns:
58
+ - None: Command not handled by this plugin
59
+ - True: Command handled successfully
60
+ - str: Text to process as user input to the model
61
+ """
62
+ if name == "woof":
63
+ emit_info("🐶 Woof!")
64
+ return True # Handled, don't invoke model
65
+
66
+ if name == "echo":
67
+ # Extract text after command name
68
+ parts = command.split(maxsplit=1)
69
+ if len(parts) == 2:
70
+ return parts[1] # Return as prompt to model
71
+ return "" # Empty prompt
72
+
73
+ return None # Not our command
74
+
75
+ # 3. Register callbacks
76
+ register_callback("custom_command_help", _custom_help)
77
+ register_callback("custom_command", _handle_custom_command)
78
+ ```
79
+
80
+ ## Commands Provided
81
+
82
+ ### `/woof [text]`
83
+
84
+ **Description**: Playful command that sends a prompt to the model.
85
+
86
+ **Behavior**:
87
+ - Without text: Sends "Tell me a dog fact" to the model
88
+ - With text: Sends your text as the prompt
89
+
90
+ **Examples**:
91
+ ```bash
92
+ /woof
93
+ # → Sends prompt: "Tell me a dog fact"
94
+
95
+ /woof What's the best breed?
96
+ # → Sends prompt: "What's the best breed?"
97
+ ```
98
+
99
+ ### `/echo <text>`
100
+
101
+ **Description**: Display-only command that shows your text.
102
+
103
+ **Behavior**:
104
+ - Shows the text you provide
105
+ - Returns it as input to the model
106
+
107
+ **Examples**:
108
+ ```bash
109
+ /echo Hello world
110
+ # → Displays: "example plugin echo -> Hello world"
111
+ # → Sends to model: "Hello world"
112
+ ```
113
+
114
+ ## Creating Your Own Plugin
115
+
116
+ ### Step 1: Create Plugin Directory
117
+
118
+ ```bash
119
+ mkdir -p code_puppy/plugins/my_plugin
120
+ touch code_puppy/plugins/my_plugin/__init__.py
121
+ touch code_puppy/plugins/my_plugin/register_callbacks.py
122
+ ```
123
+
124
+ ### Step 2: Implement Callbacks
125
+
126
+ ```python
127
+ # code_puppy/plugins/my_plugin/register_callbacks.py
128
+
129
+ from code_puppy.callbacks import register_callback
130
+ from code_puppy.messaging import emit_info, emit_success
131
+
132
+ def _custom_help():
133
+ """Provide help text for /help display."""
134
+ return [
135
+ ("mycommand", "Description of my command"),
136
+ ]
137
+
138
+ def _handle_custom_command(command: str, name: str):
139
+ """Handle your custom commands."""
140
+ if name == "mycommand":
141
+ # Your command logic here
142
+ emit_success("My command executed!")
143
+ return True # Command handled
144
+
145
+ return None # Not our command
146
+
147
+ # Register the callbacks
148
+ register_callback("custom_command_help", _custom_help)
149
+ register_callback("custom_command", _handle_custom_command)
150
+ ```
151
+
152
+ ### Step 3: Test Your Plugin
153
+
154
+ ```bash
155
+ # Restart Code Puppy to load the plugin
156
+ code-puppy
157
+
158
+ # Try your command
159
+ /mycommand
160
+ ```
161
+
162
+ ## Return Value Behaviors
163
+
164
+ Your `_handle_custom_command` function can return:
165
+
166
+ | Return Value | Behavior |
167
+ |-------------|----------|
168
+ | `None` | Command not recognized, try next plugin |
169
+ | `True` | Command handled successfully, no model invocation |
170
+ | `str` | String processed as user input to the model |
171
+ | `MarkdownCommandResult(content)` | Special case for markdown commands |
172
+
173
+ ## Best Practices
174
+
175
+ ### ✅ DO:
176
+
177
+ - **Use for plugin-specific features**: OAuth flows, integrations, utilities
178
+ - **Return `True` for display-only commands**: Avoid unnecessary model calls
179
+ - **Return strings to invoke the model**: Let users interact naturally
180
+ - **Provide clear help text**: Users see this in `/help`
181
+ - **Handle errors gracefully**: Use try/except and emit_error
182
+ - **Keep commands simple**: Complex logic → separate module
183
+
184
+ ### ❌ DON'T:
185
+
186
+ - **Don't use `@register_command`**: That's for built-in commands only
187
+ - **Don't modify global state**: Use Code Puppy's config system
188
+ - **Don't make blocking calls**: Keep commands fast and responsive
189
+ - **Don't invoke the model directly**: Return strings instead
190
+ - **Don't duplicate built-in commands**: Check existing commands first
191
+
192
+ ## Command Execution Order
193
+
194
+ 1. **Built-in commands** checked first (via registry)
195
+ 2. **Legacy fallback** checked (for backward compatibility)
196
+ 3. **Custom commands** checked (via callbacks) ← Your plugin runs here
197
+ 4. If no match, show "Unknown command" warning
198
+
199
+ ## Available Messaging Functions
200
+
201
+ ```python
202
+ from code_puppy.messaging import (
203
+ emit_info, # Blue info message
204
+ emit_success, # Green success message
205
+ emit_warning, # Yellow warning message
206
+ emit_error, # Red error message
207
+ )
208
+
209
+ # Examples
210
+ emit_info("Processing...")
211
+ emit_success("Done!")
212
+ emit_warning("This might take a while")
213
+ emit_error("Something went wrong")
214
+ ```
215
+
216
+ ## Testing Your Plugin
217
+
218
+ ### Manual Testing
219
+
220
+ ```bash
221
+ # Start Code Puppy
222
+ code-puppy
223
+
224
+ # Test your commands
225
+ /mycommand
226
+ /help # Verify your command appears
227
+ ```
228
+
229
+ ### Unit Testing
230
+
231
+ ```python
232
+ # tests/test_my_plugin.py
233
+
234
+ from code_puppy.plugins.my_plugin.register_callbacks import _handle_custom_command
235
+
236
+ def test_my_command():
237
+ result = _handle_custom_command("/mycommand", "mycommand")
238
+ assert result is True
239
+
240
+ def test_unknown_command():
241
+ result = _handle_custom_command("/unknown", "unknown")
242
+ assert result is None
243
+ ```
244
+
245
+ ## Difference from Built-in Commands
246
+
247
+ | Feature | Built-in Commands | Custom Commands (Plugins) |
248
+ |---------|------------------|---------------------------|
249
+ | **Decorator/Function** | `@register_command` | `register_callback()` |
250
+ | **Location** | `core_commands.py`, etc. | Plugin directory |
251
+ | **Purpose** | Core functionality | Plugin features |
252
+ | **Auto-discovery** | Via imports | Via plugin loader |
253
+ | **Priority** | Checked first | Checked last |
254
+ | **Help display** | Automatic | Manual via callback |
255
+
256
+ ## Example Plugins in This Repo
257
+
258
+ - **`example_custom_command/`** (this plugin) - Basic command examples
259
+ - **`customizable_commands/`** - Markdown file commands
260
+ - **`claude_code_oauth/`** - OAuth integration example
261
+ - **`chatgpt_oauth/`** - Another OAuth example
262
+ - **`file_permission_handler/`** - File system integration
263
+
264
+ ## Further Reading
265
+
266
+ - `code_puppy/callbacks.py` - Callback system implementation
267
+ - `code_puppy/command_line/command_handler.py` - Command dispatcher
268
+ - `code_puppy/command_line/core_commands.py` - Example built-in commands
269
+ - `code_puppy/command_line/command_registry.py` - Registry system
270
+
271
+ ## Questions?
272
+
273
+ If you're unsure whether to create a custom command or a built-in command:
274
+
275
+ - **Is it core Code Puppy functionality?** → Use `@register_command` (built-in)
276
+ - Add to appropriate category file: `core_commands.py`, `session_commands.py`, or `config_commands.py`
277
+ - **Is it plugin-specific?** → Use `register_callback()` (custom)
278
+ - Create a plugin directory and use the callback system (like this example)
279
+ - **Is it a prompt template?** → Use markdown file in `.claude/commands/`
280
+ - The `customizable_commands` plugin will auto-load `.md` files
@@ -0,0 +1,51 @@
1
+ from code_puppy.callbacks import register_callback
2
+ from code_puppy.messaging import emit_info
3
+
4
+
5
+ def _custom_help():
6
+ return [
7
+ ("woof", "Emit a playful woof message (no model)"),
8
+ ("echo", "Echo back your text (display only)"),
9
+ ]
10
+
11
+
12
+ def _handle_custom_command(command: str, name: str):
13
+ """Handle a demo custom command.
14
+
15
+ Policy: custom commands must NOT invoke the model. They should emit
16
+ messages or return True to indicate handling. Returning a string is
17
+ treated as a display-only message by the command handler.
18
+
19
+ Supports:
20
+ - /woof → emits a fun message and returns True
21
+ - /echo <text> → emits the text (display-only)
22
+ """
23
+ if not name:
24
+ return None
25
+
26
+ if name == "woof":
27
+ # If extra text is provided, pass it as a prompt; otherwise, send a fun default
28
+ parts = command.split(maxsplit=1)
29
+ if len(parts) == 2:
30
+ text = parts[1]
31
+ emit_info(f"🐶 Woof! sending prompt: {text}")
32
+ return text
33
+ emit_info("🐶 Woof! sending prompt: Tell me a dog fact")
34
+ return "Tell me a dog fact"
35
+
36
+ if name == "echo":
37
+ # Return the rest of the command (after the name) to be treated as input
38
+ # Example: "/echo Hello" → returns "Hello"
39
+ rest = command.split(maxsplit=1)
40
+ if len(rest) == 2:
41
+ text = rest[1]
42
+ emit_info(f"example plugin echo -> {text}")
43
+ return text
44
+ emit_info("example plugin echo (empty)")
45
+ return ""
46
+
47
+ return None
48
+
49
+
50
+ register_callback("custom_command_help", _custom_help)
51
+ register_callback("custom_command", _handle_custom_command)
@@ -0,0 +1,4 @@
1
+ """File Permission Handler Plugin Package."""
2
+
3
+ __version__ = "1.0.0"
4
+ __description__ = "Unified file permission handling system for code-puppy"