robot_lab 0.1.0 → 0.2.1

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 (242) hide show
  1. checksums.yaml +4 -4
  2. data/.architecture/AGENTS.md +32 -0
  3. data/.architecture/config.yml +8 -0
  4. data/.architecture/members.yml +60 -0
  5. data/.architecture/reviews/feature-free-will.md +490 -0
  6. data/.architecture/reviews/overall-codebase.md +427 -0
  7. data/.claude/settings.local.json +57 -0
  8. data/.codex/config.toml +2 -0
  9. data/.irbrc +2 -2
  10. data/.rubocop.yml +172 -0
  11. data/CHANGELOG.md +72 -0
  12. data/CLAUDE.md +139 -0
  13. data/README.md +91 -95
  14. data/Rakefile +109 -3
  15. data/agent2agent_review.md +192 -0
  16. data/agentf_improvements.md +253 -0
  17. data/agents.md +14 -0
  18. data/docs/examples/index.md +37 -2
  19. data/docs/getting-started/configuration.md +20 -7
  20. data/docs/guides/index.md +16 -16
  21. data/docs/guides/knowledge.md +7 -1
  22. data/docs/guides/observability.md +132 -0
  23. data/docs/index.md +30 -3
  24. data/docs/superpowers/plans/2026-05-06-agentskills.md +1303 -0
  25. data/docs/superpowers/specs/2026-05-06-agentskills-design.md +247 -0
  26. data/examples/.envrc +1 -0
  27. data/examples/01_simple_robot.rb +5 -9
  28. data/examples/02_tools.rb +5 -9
  29. data/examples/03_network.rb +8 -9
  30. data/examples/04_mcp.rb +21 -29
  31. data/examples/05_streaming.rb +12 -18
  32. data/examples/06_prompt_templates.rb +11 -19
  33. data/examples/07_network_memory.rb +16 -31
  34. data/examples/08_llm_config.rb +10 -22
  35. data/examples/09_chaining.rb +16 -27
  36. data/examples/10_memory.rb +12 -28
  37. data/examples/11_network_introspection.rb +15 -29
  38. data/examples/12_message_bus.rb +5 -12
  39. data/examples/13_spawn.rb +5 -10
  40. data/examples/14_rusty_circuit/.envrc +1 -0
  41. data/examples/14_rusty_circuit/comic.rb +2 -0
  42. data/examples/14_rusty_circuit/heckler.rb +1 -1
  43. data/examples/14_rusty_circuit/open_mic.rb +1 -3
  44. data/examples/14_rusty_circuit/scout.rb +2 -0
  45. data/examples/15_memory_network_and_bus/.envrc +1 -0
  46. data/examples/15_memory_network_and_bus/editorial_pipeline.rb +6 -3
  47. data/examples/15_memory_network_and_bus/linux_writer.rb +1 -1
  48. data/examples/15_memory_network_and_bus/output/combined_article.md +6 -6
  49. data/examples/15_memory_network_and_bus/output/final_article.md +6 -8
  50. data/examples/15_memory_network_and_bus/output/linux_draft.md +4 -2
  51. data/examples/15_memory_network_and_bus/output/mac_draft.md +3 -3
  52. data/examples/15_memory_network_and_bus/output/memory.json +6 -6
  53. data/examples/15_memory_network_and_bus/output/revision_1.md +10 -11
  54. data/examples/15_memory_network_and_bus/output/revision_2.md +6 -8
  55. data/examples/15_memory_network_and_bus/output/windows_draft.md +3 -3
  56. data/examples/16_writers_room/.envrc +1 -0
  57. data/examples/16_writers_room/writers_room.rb +2 -4
  58. data/examples/17_skills.rb +8 -17
  59. data/examples/18_rails/Gemfile +1 -0
  60. data/examples/19_token_tracking.rb +9 -15
  61. data/examples/20_circuit_breaker.rb +10 -19
  62. data/examples/21_learning_loop.rb +11 -20
  63. data/examples/22_context_compression.rb +6 -13
  64. data/examples/23_convergence.rb +6 -17
  65. data/examples/24_structured_delegation.rb +11 -15
  66. data/examples/25_history_search.rb +5 -12
  67. data/examples/26_document_store.rb +6 -13
  68. data/examples/27_incident_response/incident_response.rb +4 -5
  69. data/examples/28_mcp_discovery.rb +8 -11
  70. data/examples/29_ractor_tools.rb +4 -9
  71. data/examples/30_ractor_network.rb +10 -19
  72. data/examples/31_launch_assessment.rb +10 -23
  73. data/examples/32_newsletter_reader.rb +188 -0
  74. data/examples/33_stock_generator.rb +80 -0
  75. data/examples/33_stock_predictor.rb +306 -0
  76. data/examples/34_agentskills.rb +72 -0
  77. data/examples/README.md +1 -1
  78. data/examples/common.rb +76 -0
  79. data/examples/ruboruby.md +423 -0
  80. data/examples/temp.md +51 -0
  81. data/lib/robot_lab/agent_skill.rb +63 -0
  82. data/lib/robot_lab/agent_skill_catalog.rb +74 -0
  83. data/lib/robot_lab/ask_user.rb +2 -2
  84. data/lib/robot_lab/bus_poller.rb +12 -5
  85. data/lib/robot_lab/config.rb +1 -12
  86. data/lib/robot_lab/delegation_future.rb +1 -1
  87. data/lib/robot_lab/doom_loop_detector.rb +98 -0
  88. data/lib/robot_lab/history_compressor.rb +4 -10
  89. data/lib/robot_lab/mcp/client.rb +1 -2
  90. data/lib/robot_lab/mcp/connection_poller.rb +3 -3
  91. data/lib/robot_lab/mcp/server.rb +1 -1
  92. data/lib/robot_lab/mcp/server_discovery.rb +0 -2
  93. data/lib/robot_lab/memory.rb +32 -27
  94. data/lib/robot_lab/memory_change.rb +2 -2
  95. data/lib/robot_lab/message.rb +4 -4
  96. data/lib/robot_lab/network.rb +11 -6
  97. data/lib/robot_lab/robot/agent_skill_matching.rb +99 -0
  98. data/lib/robot_lab/robot/bus_messaging.rb +9 -27
  99. data/lib/robot_lab/robot/history_search.rb +4 -1
  100. data/lib/robot_lab/robot/mcp_management.rb +5 -11
  101. data/lib/robot_lab/robot/template_rendering.rb +60 -40
  102. data/lib/robot_lab/robot.rb +323 -206
  103. data/lib/robot_lab/robot_result.rb +6 -5
  104. data/lib/robot_lab/run_config.rb +5 -11
  105. data/lib/robot_lab/script_tool.rb +76 -0
  106. data/lib/robot_lab/state_proxy.rb +7 -5
  107. data/lib/robot_lab/tool.rb +3 -3
  108. data/lib/robot_lab/tool_config.rb +1 -1
  109. data/lib/robot_lab/tool_manifest.rb +5 -7
  110. data/lib/robot_lab/user_message.rb +2 -2
  111. data/lib/robot_lab/version.rb +1 -1
  112. data/lib/robot_lab/waiter.rb +1 -1
  113. data/lib/robot_lab.rb +41 -52
  114. data/logfile +8 -0
  115. data/mkdocs.yml +2 -3
  116. data/robot_concurrency.md +38 -0
  117. data/simple_acp_review.md +298 -0
  118. data/site/404.html +2300 -0
  119. data/site/api/core/index.html +2706 -0
  120. data/site/api/core/memory/index.html +3793 -0
  121. data/site/api/core/network/index.html +3500 -0
  122. data/site/api/core/robot/index.html +4566 -0
  123. data/site/api/core/state/index.html +3390 -0
  124. data/site/api/core/tool/index.html +3843 -0
  125. data/site/api/index.html +2635 -0
  126. data/site/api/mcp/client/index.html +3435 -0
  127. data/site/api/mcp/index.html +2783 -0
  128. data/site/api/mcp/server/index.html +3252 -0
  129. data/site/api/mcp/transports/index.html +3352 -0
  130. data/site/api/messages/index.html +2641 -0
  131. data/site/api/messages/text-message/index.html +3087 -0
  132. data/site/api/messages/tool-call-message/index.html +3159 -0
  133. data/site/api/messages/tool-result-message/index.html +3252 -0
  134. data/site/api/messages/user-message/index.html +3212 -0
  135. data/site/api/streaming/context/index.html +3282 -0
  136. data/site/api/streaming/events/index.html +3347 -0
  137. data/site/api/streaming/index.html +2738 -0
  138. data/site/architecture/core-concepts/index.html +3757 -0
  139. data/site/architecture/index.html +2797 -0
  140. data/site/architecture/message-flow/index.html +3238 -0
  141. data/site/architecture/network-orchestration/index.html +3433 -0
  142. data/site/architecture/robot-execution/index.html +3140 -0
  143. data/site/architecture/state-management/index.html +3498 -0
  144. data/site/assets/css/custom.css +56 -0
  145. data/site/assets/images/favicon.png +0 -0
  146. data/site/assets/images/robot_lab.jpg +0 -0
  147. data/site/assets/javascripts/bundle.79ae519e.min.js +16 -0
  148. data/site/assets/javascripts/bundle.79ae519e.min.js.map +7 -0
  149. data/site/assets/javascripts/lunr/min/lunr.ar.min.js +1 -0
  150. data/site/assets/javascripts/lunr/min/lunr.da.min.js +18 -0
  151. data/site/assets/javascripts/lunr/min/lunr.de.min.js +18 -0
  152. data/site/assets/javascripts/lunr/min/lunr.du.min.js +18 -0
  153. data/site/assets/javascripts/lunr/min/lunr.el.min.js +1 -0
  154. data/site/assets/javascripts/lunr/min/lunr.es.min.js +18 -0
  155. data/site/assets/javascripts/lunr/min/lunr.fi.min.js +18 -0
  156. data/site/assets/javascripts/lunr/min/lunr.fr.min.js +18 -0
  157. data/site/assets/javascripts/lunr/min/lunr.he.min.js +1 -0
  158. data/site/assets/javascripts/lunr/min/lunr.hi.min.js +1 -0
  159. data/site/assets/javascripts/lunr/min/lunr.hu.min.js +18 -0
  160. data/site/assets/javascripts/lunr/min/lunr.hy.min.js +1 -0
  161. data/site/assets/javascripts/lunr/min/lunr.it.min.js +18 -0
  162. data/site/assets/javascripts/lunr/min/lunr.ja.min.js +1 -0
  163. data/site/assets/javascripts/lunr/min/lunr.jp.min.js +1 -0
  164. data/site/assets/javascripts/lunr/min/lunr.kn.min.js +1 -0
  165. data/site/assets/javascripts/lunr/min/lunr.ko.min.js +1 -0
  166. data/site/assets/javascripts/lunr/min/lunr.multi.min.js +1 -0
  167. data/site/assets/javascripts/lunr/min/lunr.nl.min.js +18 -0
  168. data/site/assets/javascripts/lunr/min/lunr.no.min.js +18 -0
  169. data/site/assets/javascripts/lunr/min/lunr.pt.min.js +18 -0
  170. data/site/assets/javascripts/lunr/min/lunr.ro.min.js +18 -0
  171. data/site/assets/javascripts/lunr/min/lunr.ru.min.js +18 -0
  172. data/site/assets/javascripts/lunr/min/lunr.sa.min.js +1 -0
  173. data/site/assets/javascripts/lunr/min/lunr.stemmer.support.min.js +1 -0
  174. data/site/assets/javascripts/lunr/min/lunr.sv.min.js +18 -0
  175. data/site/assets/javascripts/lunr/min/lunr.ta.min.js +1 -0
  176. data/site/assets/javascripts/lunr/min/lunr.te.min.js +1 -0
  177. data/site/assets/javascripts/lunr/min/lunr.th.min.js +1 -0
  178. data/site/assets/javascripts/lunr/min/lunr.tr.min.js +18 -0
  179. data/site/assets/javascripts/lunr/min/lunr.vi.min.js +1 -0
  180. data/site/assets/javascripts/lunr/min/lunr.zh.min.js +1 -0
  181. data/site/assets/javascripts/lunr/tinyseg.js +206 -0
  182. data/site/assets/javascripts/lunr/wordcut.js +6708 -0
  183. data/site/assets/javascripts/workers/search.2c215733.min.js +42 -0
  184. data/site/assets/javascripts/workers/search.2c215733.min.js.map +7 -0
  185. data/site/assets/stylesheets/main.484c7ddc.min.css +1 -0
  186. data/site/assets/stylesheets/main.484c7ddc.min.css.map +1 -0
  187. data/site/assets/stylesheets/palette.ab4e12ef.min.css +1 -0
  188. data/site/assets/stylesheets/palette.ab4e12ef.min.css.map +1 -0
  189. data/site/concepts/index.html +3455 -0
  190. data/site/examples/basic-chat/index.html +2880 -0
  191. data/site/examples/index.html +2907 -0
  192. data/site/examples/mcp-server/index.html +3018 -0
  193. data/site/examples/multi-robot-network/index.html +3131 -0
  194. data/site/examples/rails-application/index.html +3329 -0
  195. data/site/examples/tool-usage/index.html +3085 -0
  196. data/site/getting-started/configuration/index.html +3745 -0
  197. data/site/getting-started/index.html +2572 -0
  198. data/site/getting-started/installation/index.html +2981 -0
  199. data/site/getting-started/quick-start/index.html +2942 -0
  200. data/site/guides/building-robots/index.html +4290 -0
  201. data/site/guides/creating-networks/index.html +3858 -0
  202. data/site/guides/index.html +2586 -0
  203. data/site/guides/mcp-integration/index.html +3581 -0
  204. data/site/guides/memory/index.html +3586 -0
  205. data/site/guides/rails-integration/index.html +4019 -0
  206. data/site/guides/streaming/index.html +3157 -0
  207. data/site/guides/using-tools/index.html +3802 -0
  208. data/site/index.html +2671 -0
  209. data/site/search/search_index.json +1 -0
  210. data/site/sitemap.xml +183 -0
  211. data/site/sitemap.xml.gz +0 -0
  212. data/site/tags.json +1 -0
  213. data/temp.md +6 -0
  214. data/tool_manifest_plan.md +155 -0
  215. metadata +154 -92
  216. data/docs/examples/rails-application.md +0 -419
  217. data/docs/guides/ractor-parallelism.md +0 -364
  218. data/docs/guides/rails-integration.md +0 -681
  219. data/docs/superpowers/plans/2026-04-14-ractor-integration.md +0 -1538
  220. data/docs/superpowers/specs/2026-04-14-ractor-integration-design.md +0 -258
  221. data/lib/generators/robot_lab/install_generator.rb +0 -90
  222. data/lib/generators/robot_lab/job_generator.rb +0 -40
  223. data/lib/generators/robot_lab/robot_generator.rb +0 -55
  224. data/lib/generators/robot_lab/templates/initializer.rb.tt +0 -42
  225. data/lib/generators/robot_lab/templates/job.rb.tt +0 -21
  226. data/lib/generators/robot_lab/templates/migration.rb.tt +0 -32
  227. data/lib/generators/robot_lab/templates/result_model.rb.tt +0 -52
  228. data/lib/generators/robot_lab/templates/robot.rb.tt +0 -31
  229. data/lib/generators/robot_lab/templates/robot_job.rb.tt +0 -18
  230. data/lib/generators/robot_lab/templates/robot_test.rb.tt +0 -34
  231. data/lib/generators/robot_lab/templates/routing_robot.rb.tt +0 -59
  232. data/lib/generators/robot_lab/templates/thread_model.rb.tt +0 -40
  233. data/lib/robot_lab/document_store.rb +0 -155
  234. data/lib/robot_lab/ractor_boundary.rb +0 -42
  235. data/lib/robot_lab/ractor_job.rb +0 -37
  236. data/lib/robot_lab/ractor_memory_proxy.rb +0 -85
  237. data/lib/robot_lab/ractor_network_scheduler.rb +0 -154
  238. data/lib/robot_lab/ractor_worker_pool.rb +0 -117
  239. data/lib/robot_lab/rails_integration/engine.rb +0 -29
  240. data/lib/robot_lab/rails_integration/job.rb +0 -158
  241. data/lib/robot_lab/rails_integration/railtie.rb +0 -51
  242. data/lib/robot_lab/rails_integration/turbo_stream_callbacks.rb +0 -72
@@ -0,0 +1,4019 @@
1
+
2
+ <!doctype html>
3
+ <html lang="en" class="no-js">
4
+ <head>
5
+
6
+ <meta charset="utf-8">
7
+ <meta name="viewport" content="width=device-width,initial-scale=1">
8
+
9
+ <meta name="description" content="Multi-robot LLM workflow orchestration for Ruby">
10
+
11
+
12
+ <meta name="author" content="Dewayne VanHoozer">
13
+
14
+
15
+ <link rel="canonical" href="https://madbomber.github.io/robot_lab/guides/rails-integration/">
16
+
17
+
18
+ <link rel="prev" href="../memory/">
19
+
20
+
21
+ <link rel="next" href="../../api/">
22
+
23
+
24
+
25
+
26
+
27
+ <link rel="icon" href="../../assets/images/favicon.png">
28
+ <meta name="generator" content="mkdocs-1.6.1, mkdocs-material-9.7.1">
29
+
30
+
31
+
32
+ <title>Rails Integration - RobotLab</title>
33
+
34
+
35
+
36
+ <link rel="stylesheet" href="../../assets/stylesheets/main.484c7ddc.min.css">
37
+
38
+
39
+ <link rel="stylesheet" href="../../assets/stylesheets/palette.ab4e12ef.min.css">
40
+
41
+
42
+
43
+
44
+
45
+
46
+
47
+
48
+
49
+
50
+
51
+
52
+ <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
53
+ <link rel="stylesheet" href="https://fonts.googleapis.com/css?family=Roboto:300,300i,400,400i,700,700i%7CRoboto+Mono:400,400i,700,700i&display=fallback">
54
+ <style>:root{--md-text-font:"Roboto";--md-code-font:"Roboto Mono"}</style>
55
+
56
+
57
+
58
+ <link rel="stylesheet" href="../../assets/css/custom.css">
59
+
60
+ <script>__md_scope=new URL("../..",location),__md_hash=e=>[...e].reduce(((e,_)=>(e<<5)-e+_.charCodeAt(0)),0),__md_get=(e,_=localStorage,t=__md_scope)=>JSON.parse(_.getItem(t.pathname+"."+e)),__md_set=(e,_,t=localStorage,a=__md_scope)=>{try{t.setItem(a.pathname+"."+e,JSON.stringify(_))}catch(e){}}</script>
61
+
62
+
63
+
64
+
65
+
66
+
67
+
68
+ </head>
69
+
70
+
71
+
72
+
73
+
74
+
75
+
76
+
77
+
78
+ <body dir="ltr" data-md-color-scheme="default" data-md-color-primary="deep-purple" data-md-color-accent="amber">
79
+
80
+
81
+ <input class="md-toggle" data-md-toggle="drawer" type="checkbox" id="__drawer" autocomplete="off">
82
+ <input class="md-toggle" data-md-toggle="search" type="checkbox" id="__search" autocomplete="off">
83
+ <label class="md-overlay" for="__drawer"></label>
84
+ <div data-md-component="skip">
85
+
86
+
87
+ <a href="#rails-integration" class="md-skip">
88
+ Skip to content
89
+ </a>
90
+
91
+ </div>
92
+ <div data-md-component="announce">
93
+
94
+ </div>
95
+
96
+
97
+
98
+
99
+
100
+
101
+ <header class="md-header md-header--shadow md-header--lifted" data-md-component="header">
102
+ <nav class="md-header__inner md-grid" aria-label="Header">
103
+ <a href="../.." title="RobotLab" class="md-header__button md-logo" aria-label="RobotLab" data-md-component="logo">
104
+
105
+
106
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12 2a2 2 0 0 1 2 2c0 .74-.4 1.39-1 1.73V7h1a7 7 0 0 1 7 7h1a1 1 0 0 1 1 1v3a1 1 0 0 1-1 1h-1v1a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-1H2a1 1 0 0 1-1-1v-3a1 1 0 0 1 1-1h1a7 7 0 0 1 7-7h1V5.73c-.6-.34-1-.99-1-1.73a2 2 0 0 1 2-2M7.5 13A2.5 2.5 0 0 0 5 15.5 2.5 2.5 0 0 0 7.5 18a2.5 2.5 0 0 0 2.5-2.5A2.5 2.5 0 0 0 7.5 13m9 0a2.5 2.5 0 0 0-2.5 2.5 2.5 2.5 0 0 0 2.5 2.5 2.5 2.5 0 0 0 2.5-2.5 2.5 2.5 0 0 0-2.5-2.5"/></svg>
107
+
108
+ </a>
109
+ <label class="md-header__button md-icon" for="__drawer">
110
+
111
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M3 6h18v2H3zm0 5h18v2H3zm0 5h18v2H3z"/></svg>
112
+ </label>
113
+ <div class="md-header__title" data-md-component="header-title">
114
+ <div class="md-header__ellipsis">
115
+ <div class="md-header__topic">
116
+ <span class="md-ellipsis">
117
+ RobotLab
118
+ </span>
119
+ </div>
120
+ <div class="md-header__topic" data-md-component="header-topic">
121
+ <span class="md-ellipsis">
122
+
123
+ Rails Integration
124
+
125
+ </span>
126
+ </div>
127
+ </div>
128
+ </div>
129
+
130
+
131
+ <form class="md-header__option" data-md-component="palette">
132
+
133
+
134
+
135
+
136
+ <input class="md-option" data-md-color-media="" data-md-color-scheme="default" data-md-color-primary="deep-purple" data-md-color-accent="amber" aria-label="Switch to dark mode" type="radio" name="__palette" id="__palette_0">
137
+
138
+ <label class="md-header__button md-icon" title="Switch to dark mode" for="__palette_1" hidden>
139
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12 8a4 4 0 0 0-4 4 4 4 0 0 0 4 4 4 4 0 0 0 4-4 4 4 0 0 0-4-4m0 10a6 6 0 0 1-6-6 6 6 0 0 1 6-6 6 6 0 0 1 6 6 6 6 0 0 1-6 6m8-9.31V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69L23.31 12z"/></svg>
140
+ </label>
141
+
142
+
143
+
144
+
145
+
146
+ <input class="md-option" data-md-color-media="" data-md-color-scheme="slate" data-md-color-primary="deep-purple" data-md-color-accent="amber" aria-label="Switch to light mode" type="radio" name="__palette" id="__palette_1">
147
+
148
+ <label class="md-header__button md-icon" title="Switch to light mode" for="__palette_0" hidden>
149
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12 18c-.89 0-1.74-.2-2.5-.55C11.56 16.5 13 14.42 13 12s-1.44-4.5-3.5-5.45C10.26 6.2 11.11 6 12 6a6 6 0 0 1 6 6 6 6 0 0 1-6 6m8-9.31V4h-4.69L12 .69 8.69 4H4v4.69L.69 12 4 15.31V20h4.69L12 23.31 15.31 20H20v-4.69L23.31 12z"/></svg>
150
+ </label>
151
+
152
+
153
+ </form>
154
+
155
+
156
+
157
+ <script>var palette=__md_get("__palette");if(palette&&palette.color){if("(prefers-color-scheme)"===palette.color.media){var media=matchMedia("(prefers-color-scheme: light)"),input=document.querySelector(media.matches?"[data-md-color-media='(prefers-color-scheme: light)']":"[data-md-color-media='(prefers-color-scheme: dark)']");palette.color.media=input.getAttribute("data-md-color-media"),palette.color.scheme=input.getAttribute("data-md-color-scheme"),palette.color.primary=input.getAttribute("data-md-color-primary"),palette.color.accent=input.getAttribute("data-md-color-accent")}for(var[key,value]of Object.entries(palette.color))document.body.setAttribute("data-md-color-"+key,value)}</script>
158
+
159
+
160
+
161
+
162
+
163
+ <label class="md-header__button md-icon" for="__search">
164
+
165
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M9.5 3A6.5 6.5 0 0 1 16 9.5c0 1.61-.59 3.09-1.56 4.23l.27.27h.79l5 5-1.5 1.5-5-5v-.79l-.27-.27A6.52 6.52 0 0 1 9.5 16 6.5 6.5 0 0 1 3 9.5 6.5 6.5 0 0 1 9.5 3m0 2C7 5 5 7 5 9.5S7 14 9.5 14 14 12 14 9.5 12 5 9.5 5"/></svg>
166
+ </label>
167
+ <div class="md-search" data-md-component="search" role="dialog">
168
+ <label class="md-search__overlay" for="__search"></label>
169
+ <div class="md-search__inner" role="search">
170
+ <form class="md-search__form" name="search">
171
+ <input type="text" class="md-search__input" name="query" aria-label="Search" placeholder="Search" autocapitalize="off" autocorrect="off" autocomplete="off" spellcheck="false" data-md-component="search-query" required>
172
+ <label class="md-search__icon md-icon" for="__search">
173
+
174
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M9.5 3A6.5 6.5 0 0 1 16 9.5c0 1.61-.59 3.09-1.56 4.23l.27.27h.79l5 5-1.5 1.5-5-5v-.79l-.27-.27A6.52 6.52 0 0 1 9.5 16 6.5 6.5 0 0 1 3 9.5 6.5 6.5 0 0 1 9.5 3m0 2C7 5 5 7 5 9.5S7 14 9.5 14 14 12 14 9.5 12 5 9.5 5"/></svg>
175
+
176
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M20 11v2H8l5.5 5.5-1.42 1.42L4.16 12l7.92-7.92L13.5 5.5 8 11z"/></svg>
177
+ </label>
178
+ <nav class="md-search__options" aria-label="Search">
179
+
180
+ <a href="javascript:void(0)" class="md-search__icon md-icon" title="Share" aria-label="Share" data-clipboard data-clipboard-text="" data-md-component="search-share" tabindex="-1">
181
+
182
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M18 16.08c-.76 0-1.44.3-1.96.77L8.91 12.7c.05-.23.09-.46.09-.7s-.04-.47-.09-.7l7.05-4.11c.54.5 1.25.81 2.04.81a3 3 0 0 0 3-3 3 3 0 0 0-3-3 3 3 0 0 0-3 3c0 .24.04.47.09.7L8.04 9.81C7.5 9.31 6.79 9 6 9a3 3 0 0 0-3 3 3 3 0 0 0 3 3c.79 0 1.5-.31 2.04-.81l7.12 4.15c-.05.21-.08.43-.08.66 0 1.61 1.31 2.91 2.92 2.91s2.92-1.3 2.92-2.91A2.92 2.92 0 0 0 18 16.08"/></svg>
183
+ </a>
184
+
185
+ <button type="reset" class="md-search__icon md-icon" title="Clear" aria-label="Clear" tabindex="-1">
186
+
187
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M19 6.41 17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/></svg>
188
+ </button>
189
+ </nav>
190
+
191
+ <div class="md-search__suggest" data-md-component="search-suggest"></div>
192
+
193
+ </form>
194
+ <div class="md-search__output">
195
+ <div class="md-search__scrollwrap" tabindex="0" data-md-scrollfix>
196
+ <div class="md-search-result" data-md-component="search-result">
197
+ <div class="md-search-result__meta">
198
+ Initializing search
199
+ </div>
200
+ <ol class="md-search-result__list" role="presentation"></ol>
201
+ </div>
202
+ </div>
203
+ </div>
204
+ </div>
205
+ </div>
206
+
207
+
208
+
209
+ <div class="md-header__source">
210
+ <a href="https://github.com/madbomber/robot_lab" title="Go to repository" class="md-source" data-md-component="source">
211
+ <div class="md-source__icon md-icon">
212
+
213
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><!--! Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2025 Fonticons, Inc.--><path d="M173.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3.3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6m-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5.3-6.2 2.3m44.2-1.7c-2.9.7-4.9 2.6-4.6 4.9.3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9M252.8 8C114.1 8 8 113.3 8 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C436.2 457.8 504 362.9 504 252 504 113.3 391.5 8 252.8 8M105.2 352.9c-1.3 1-1 3.3.7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1m-10.8-8.1c-.7 1.3.3 2.9 2.3 3.9 1.6 1 3.6.7 4.3-.7.7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3.7m32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3.7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1m-11.4-14.7c-1.6 1-1.6 3.6 0 5.9s4.3 3.3 5.6 2.3c1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2"/></svg>
214
+ </div>
215
+ <div class="md-source__repository">
216
+ madbomber/robot_lab
217
+ </div>
218
+ </a>
219
+ </div>
220
+
221
+ </nav>
222
+
223
+
224
+
225
+ <nav class="md-tabs" aria-label="Tabs" data-md-component="tabs">
226
+ <div class="md-grid">
227
+ <ul class="md-tabs__list">
228
+
229
+
230
+
231
+
232
+
233
+
234
+
235
+
236
+ <li class="md-tabs__item">
237
+ <a href="../.." class="md-tabs__link">
238
+
239
+
240
+
241
+
242
+
243
+ Home
244
+
245
+ </a>
246
+ </li>
247
+
248
+
249
+
250
+
251
+
252
+
253
+
254
+
255
+
256
+
257
+
258
+ <li class="md-tabs__item">
259
+ <a href="../../getting-started/" class="md-tabs__link">
260
+
261
+
262
+
263
+
264
+
265
+ Getting Started
266
+
267
+ </a>
268
+ </li>
269
+
270
+
271
+
272
+
273
+
274
+
275
+
276
+
277
+
278
+
279
+
280
+ <li class="md-tabs__item">
281
+ <a href="../../architecture/" class="md-tabs__link">
282
+
283
+
284
+
285
+
286
+
287
+ Architecture
288
+
289
+ </a>
290
+ </li>
291
+
292
+
293
+
294
+
295
+
296
+
297
+
298
+
299
+
300
+
301
+
302
+
303
+
304
+ <li class="md-tabs__item md-tabs__item--active">
305
+ <a href="../" class="md-tabs__link">
306
+
307
+
308
+
309
+
310
+
311
+ Guides
312
+
313
+ </a>
314
+ </li>
315
+
316
+
317
+
318
+
319
+
320
+
321
+
322
+
323
+
324
+
325
+
326
+ <li class="md-tabs__item">
327
+ <a href="../../api/" class="md-tabs__link">
328
+
329
+
330
+
331
+
332
+
333
+ API Reference
334
+
335
+ </a>
336
+ </li>
337
+
338
+
339
+
340
+
341
+
342
+
343
+
344
+
345
+
346
+
347
+
348
+ <li class="md-tabs__item">
349
+ <a href="../../examples/" class="md-tabs__link">
350
+
351
+
352
+
353
+
354
+
355
+ Examples
356
+
357
+ </a>
358
+ </li>
359
+
360
+
361
+
362
+
363
+ </ul>
364
+ </div>
365
+ </nav>
366
+
367
+
368
+ </header>
369
+
370
+ <div class="md-container" data-md-component="container">
371
+
372
+
373
+
374
+
375
+ <main class="md-main" data-md-component="main">
376
+ <div class="md-main__inner md-grid">
377
+
378
+
379
+
380
+ <div class="md-sidebar md-sidebar--primary" data-md-component="sidebar" data-md-type="navigation" >
381
+ <div class="md-sidebar__scrollwrap">
382
+ <div class="md-sidebar__inner">
383
+
384
+
385
+
386
+
387
+
388
+
389
+ <nav class="md-nav md-nav--primary md-nav--lifted" aria-label="Navigation" data-md-level="0">
390
+ <label class="md-nav__title" for="__drawer">
391
+ <a href="../.." title="RobotLab" class="md-nav__button md-logo" aria-label="RobotLab" data-md-component="logo">
392
+
393
+
394
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M12 2a2 2 0 0 1 2 2c0 .74-.4 1.39-1 1.73V7h1a7 7 0 0 1 7 7h1a1 1 0 0 1 1 1v3a1 1 0 0 1-1 1h-1v1a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-1H2a1 1 0 0 1-1-1v-3a1 1 0 0 1 1-1h1a7 7 0 0 1 7-7h1V5.73c-.6-.34-1-.99-1-1.73a2 2 0 0 1 2-2M7.5 13A2.5 2.5 0 0 0 5 15.5 2.5 2.5 0 0 0 7.5 18a2.5 2.5 0 0 0 2.5-2.5A2.5 2.5 0 0 0 7.5 13m9 0a2.5 2.5 0 0 0-2.5 2.5 2.5 2.5 0 0 0 2.5 2.5 2.5 2.5 0 0 0 2.5-2.5 2.5 2.5 0 0 0-2.5-2.5"/></svg>
395
+
396
+ </a>
397
+ RobotLab
398
+ </label>
399
+
400
+ <div class="md-nav__source">
401
+ <a href="https://github.com/madbomber/robot_lab" title="Go to repository" class="md-source" data-md-component="source">
402
+ <div class="md-source__icon md-icon">
403
+
404
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><!--! Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2025 Fonticons, Inc.--><path d="M173.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3.3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6m-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5.3-6.2 2.3m44.2-1.7c-2.9.7-4.9 2.6-4.6 4.9.3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9M252.8 8C114.1 8 8 113.3 8 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C436.2 457.8 504 362.9 504 252 504 113.3 391.5 8 252.8 8M105.2 352.9c-1.3 1-1 3.3.7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1m-10.8-8.1c-.7 1.3.3 2.9 2.3 3.9 1.6 1 3.6.7 4.3-.7.7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3.7m32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3.7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1m-11.4-14.7c-1.6 1-1.6 3.6 0 5.9s4.3 3.3 5.6 2.3c1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2"/></svg>
405
+ </div>
406
+ <div class="md-source__repository">
407
+ madbomber/robot_lab
408
+ </div>
409
+ </a>
410
+ </div>
411
+
412
+ <ul class="md-nav__list" data-md-scrollfix>
413
+
414
+
415
+
416
+
417
+
418
+
419
+
420
+
421
+
422
+
423
+
424
+
425
+
426
+
427
+
428
+
429
+
430
+
431
+
432
+
433
+
434
+
435
+
436
+ <li class="md-nav__item md-nav__item--nested">
437
+
438
+
439
+
440
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_1" >
441
+
442
+
443
+ <div class="md-nav__link md-nav__container">
444
+ <a href="../.." class="md-nav__link ">
445
+
446
+
447
+
448
+ <span class="md-ellipsis">
449
+
450
+
451
+ Home
452
+
453
+
454
+
455
+ </span>
456
+
457
+
458
+
459
+ </a>
460
+
461
+
462
+ <label class="md-nav__link " for="__nav_1" id="__nav_1_label" tabindex="0">
463
+ <span class="md-nav__icon md-icon"></span>
464
+ </label>
465
+
466
+ </div>
467
+
468
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_1_label" aria-expanded="false">
469
+ <label class="md-nav__title" for="__nav_1">
470
+ <span class="md-nav__icon md-icon"></span>
471
+
472
+
473
+ Home
474
+
475
+
476
+ </label>
477
+ <ul class="md-nav__list" data-md-scrollfix>
478
+
479
+
480
+
481
+
482
+
483
+
484
+
485
+
486
+
487
+ <li class="md-nav__item">
488
+ <a href="../../concepts/" class="md-nav__link">
489
+
490
+
491
+
492
+ <span class="md-ellipsis">
493
+
494
+
495
+ Concepts
496
+
497
+
498
+
499
+ </span>
500
+
501
+
502
+
503
+ </a>
504
+ </li>
505
+
506
+
507
+
508
+
509
+ </ul>
510
+ </nav>
511
+
512
+ </li>
513
+
514
+
515
+
516
+
517
+
518
+
519
+
520
+
521
+
522
+
523
+
524
+
525
+
526
+
527
+
528
+
529
+
530
+
531
+
532
+
533
+
534
+
535
+
536
+
537
+
538
+
539
+
540
+
541
+
542
+ <li class="md-nav__item md-nav__item--nested">
543
+
544
+
545
+
546
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_2" >
547
+
548
+
549
+ <div class="md-nav__link md-nav__container">
550
+ <a href="../../getting-started/" class="md-nav__link ">
551
+
552
+
553
+
554
+ <span class="md-ellipsis">
555
+
556
+
557
+ Getting Started
558
+
559
+
560
+
561
+ </span>
562
+
563
+
564
+
565
+ </a>
566
+
567
+
568
+ <label class="md-nav__link " for="__nav_2" id="__nav_2_label" tabindex="0">
569
+ <span class="md-nav__icon md-icon"></span>
570
+ </label>
571
+
572
+ </div>
573
+
574
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_2_label" aria-expanded="false">
575
+ <label class="md-nav__title" for="__nav_2">
576
+ <span class="md-nav__icon md-icon"></span>
577
+
578
+
579
+ Getting Started
580
+
581
+
582
+ </label>
583
+ <ul class="md-nav__list" data-md-scrollfix>
584
+
585
+
586
+
587
+
588
+
589
+
590
+
591
+
592
+
593
+ <li class="md-nav__item">
594
+ <a href="../../getting-started/installation/" class="md-nav__link">
595
+
596
+
597
+
598
+ <span class="md-ellipsis">
599
+
600
+
601
+ Installation
602
+
603
+
604
+
605
+ </span>
606
+
607
+
608
+
609
+ </a>
610
+ </li>
611
+
612
+
613
+
614
+
615
+
616
+
617
+
618
+
619
+
620
+
621
+ <li class="md-nav__item">
622
+ <a href="../../getting-started/quick-start/" class="md-nav__link">
623
+
624
+
625
+
626
+ <span class="md-ellipsis">
627
+
628
+
629
+ Quick Start
630
+
631
+
632
+
633
+ </span>
634
+
635
+
636
+
637
+ </a>
638
+ </li>
639
+
640
+
641
+
642
+
643
+
644
+
645
+
646
+
647
+
648
+
649
+ <li class="md-nav__item">
650
+ <a href="../../getting-started/configuration/" class="md-nav__link">
651
+
652
+
653
+
654
+ <span class="md-ellipsis">
655
+
656
+
657
+ Configuration
658
+
659
+
660
+
661
+ </span>
662
+
663
+
664
+
665
+ </a>
666
+ </li>
667
+
668
+
669
+
670
+
671
+ </ul>
672
+ </nav>
673
+
674
+ </li>
675
+
676
+
677
+
678
+
679
+
680
+
681
+
682
+
683
+
684
+
685
+
686
+
687
+
688
+
689
+
690
+
691
+
692
+
693
+
694
+
695
+
696
+
697
+
698
+
699
+
700
+
701
+
702
+
703
+
704
+
705
+
706
+
707
+
708
+ <li class="md-nav__item md-nav__item--nested">
709
+
710
+
711
+
712
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_3" >
713
+
714
+
715
+ <div class="md-nav__link md-nav__container">
716
+ <a href="../../architecture/" class="md-nav__link ">
717
+
718
+
719
+
720
+ <span class="md-ellipsis">
721
+
722
+
723
+ Architecture
724
+
725
+
726
+
727
+ </span>
728
+
729
+
730
+
731
+ </a>
732
+
733
+
734
+ <label class="md-nav__link " for="__nav_3" id="__nav_3_label" tabindex="0">
735
+ <span class="md-nav__icon md-icon"></span>
736
+ </label>
737
+
738
+ </div>
739
+
740
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_3_label" aria-expanded="false">
741
+ <label class="md-nav__title" for="__nav_3">
742
+ <span class="md-nav__icon md-icon"></span>
743
+
744
+
745
+ Architecture
746
+
747
+
748
+ </label>
749
+ <ul class="md-nav__list" data-md-scrollfix>
750
+
751
+
752
+
753
+
754
+
755
+
756
+
757
+
758
+
759
+ <li class="md-nav__item">
760
+ <a href="../../architecture/core-concepts/" class="md-nav__link">
761
+
762
+
763
+
764
+ <span class="md-ellipsis">
765
+
766
+
767
+ Core Concepts
768
+
769
+
770
+
771
+ </span>
772
+
773
+
774
+
775
+ </a>
776
+ </li>
777
+
778
+
779
+
780
+
781
+
782
+
783
+
784
+
785
+
786
+
787
+ <li class="md-nav__item">
788
+ <a href="../../architecture/robot-execution/" class="md-nav__link">
789
+
790
+
791
+
792
+ <span class="md-ellipsis">
793
+
794
+
795
+ Robot Execution
796
+
797
+
798
+
799
+ </span>
800
+
801
+
802
+
803
+ </a>
804
+ </li>
805
+
806
+
807
+
808
+
809
+
810
+
811
+
812
+
813
+
814
+
815
+ <li class="md-nav__item">
816
+ <a href="../../architecture/network-orchestration/" class="md-nav__link">
817
+
818
+
819
+
820
+ <span class="md-ellipsis">
821
+
822
+
823
+ Network Orchestration
824
+
825
+
826
+
827
+ </span>
828
+
829
+
830
+
831
+ </a>
832
+ </li>
833
+
834
+
835
+
836
+
837
+
838
+
839
+
840
+
841
+
842
+
843
+ <li class="md-nav__item">
844
+ <a href="../../architecture/state-management/" class="md-nav__link">
845
+
846
+
847
+
848
+ <span class="md-ellipsis">
849
+
850
+
851
+ State Management
852
+
853
+
854
+
855
+ </span>
856
+
857
+
858
+
859
+ </a>
860
+ </li>
861
+
862
+
863
+
864
+
865
+
866
+
867
+
868
+
869
+
870
+
871
+ <li class="md-nav__item">
872
+ <a href="../../architecture/message-flow/" class="md-nav__link">
873
+
874
+
875
+
876
+ <span class="md-ellipsis">
877
+
878
+
879
+ Message Flow
880
+
881
+
882
+
883
+ </span>
884
+
885
+
886
+
887
+ </a>
888
+ </li>
889
+
890
+
891
+
892
+
893
+ </ul>
894
+ </nav>
895
+
896
+ </li>
897
+
898
+
899
+
900
+
901
+
902
+
903
+
904
+
905
+
906
+
907
+
908
+
909
+
910
+
911
+
912
+
913
+
914
+
915
+
916
+
917
+
918
+
919
+
920
+
921
+
922
+
923
+
924
+
925
+
926
+
927
+
928
+
929
+
930
+
931
+
932
+
933
+
934
+
935
+
936
+
937
+
938
+
939
+ <li class="md-nav__item md-nav__item--active md-nav__item--section md-nav__item--nested">
940
+
941
+
942
+
943
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_4" checked>
944
+
945
+
946
+ <div class="md-nav__link md-nav__container">
947
+ <a href="../" class="md-nav__link ">
948
+
949
+
950
+
951
+ <span class="md-ellipsis">
952
+
953
+
954
+ Guides
955
+
956
+
957
+
958
+ </span>
959
+
960
+
961
+
962
+ </a>
963
+
964
+
965
+ <label class="md-nav__link " for="__nav_4" id="__nav_4_label" tabindex="">
966
+ <span class="md-nav__icon md-icon"></span>
967
+ </label>
968
+
969
+ </div>
970
+
971
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_4_label" aria-expanded="true">
972
+ <label class="md-nav__title" for="__nav_4">
973
+ <span class="md-nav__icon md-icon"></span>
974
+
975
+
976
+ Guides
977
+
978
+
979
+ </label>
980
+ <ul class="md-nav__list" data-md-scrollfix>
981
+
982
+
983
+
984
+
985
+
986
+
987
+
988
+
989
+
990
+ <li class="md-nav__item">
991
+ <a href="../building-robots/" class="md-nav__link">
992
+
993
+
994
+
995
+ <span class="md-ellipsis">
996
+
997
+
998
+ Building Robots
999
+
1000
+
1001
+
1002
+ </span>
1003
+
1004
+
1005
+
1006
+ </a>
1007
+ </li>
1008
+
1009
+
1010
+
1011
+
1012
+
1013
+
1014
+
1015
+
1016
+
1017
+
1018
+ <li class="md-nav__item">
1019
+ <a href="../creating-networks/" class="md-nav__link">
1020
+
1021
+
1022
+
1023
+ <span class="md-ellipsis">
1024
+
1025
+
1026
+ Creating Networks
1027
+
1028
+
1029
+
1030
+ </span>
1031
+
1032
+
1033
+
1034
+ </a>
1035
+ </li>
1036
+
1037
+
1038
+
1039
+
1040
+
1041
+
1042
+
1043
+
1044
+
1045
+
1046
+ <li class="md-nav__item">
1047
+ <a href="../using-tools/" class="md-nav__link">
1048
+
1049
+
1050
+
1051
+ <span class="md-ellipsis">
1052
+
1053
+
1054
+ Using Tools
1055
+
1056
+
1057
+
1058
+ </span>
1059
+
1060
+
1061
+
1062
+ </a>
1063
+ </li>
1064
+
1065
+
1066
+
1067
+
1068
+
1069
+
1070
+
1071
+
1072
+
1073
+
1074
+ <li class="md-nav__item">
1075
+ <a href="../mcp-integration/" class="md-nav__link">
1076
+
1077
+
1078
+
1079
+ <span class="md-ellipsis">
1080
+
1081
+
1082
+ MCP Integration
1083
+
1084
+
1085
+
1086
+ </span>
1087
+
1088
+
1089
+
1090
+ </a>
1091
+ </li>
1092
+
1093
+
1094
+
1095
+
1096
+
1097
+
1098
+
1099
+
1100
+
1101
+
1102
+ <li class="md-nav__item">
1103
+ <a href="../streaming/" class="md-nav__link">
1104
+
1105
+
1106
+
1107
+ <span class="md-ellipsis">
1108
+
1109
+
1110
+ Streaming Responses
1111
+
1112
+
1113
+
1114
+ </span>
1115
+
1116
+
1117
+
1118
+ </a>
1119
+ </li>
1120
+
1121
+
1122
+
1123
+
1124
+
1125
+
1126
+
1127
+
1128
+
1129
+
1130
+ <li class="md-nav__item">
1131
+ <a href="../memory/" class="md-nav__link">
1132
+
1133
+
1134
+
1135
+ <span class="md-ellipsis">
1136
+
1137
+
1138
+ Memory System
1139
+
1140
+
1141
+
1142
+ </span>
1143
+
1144
+
1145
+
1146
+ </a>
1147
+ </li>
1148
+
1149
+
1150
+
1151
+
1152
+
1153
+
1154
+
1155
+
1156
+
1157
+
1158
+
1159
+
1160
+ <li class="md-nav__item md-nav__item--active">
1161
+
1162
+ <input class="md-nav__toggle md-toggle" type="checkbox" id="__toc">
1163
+
1164
+
1165
+
1166
+
1167
+
1168
+ <label class="md-nav__link md-nav__link--active" for="__toc">
1169
+
1170
+
1171
+
1172
+ <span class="md-ellipsis">
1173
+
1174
+
1175
+ Rails Integration
1176
+
1177
+
1178
+
1179
+ </span>
1180
+
1181
+
1182
+
1183
+ <span class="md-nav__icon md-icon"></span>
1184
+ </label>
1185
+
1186
+ <a href="./" class="md-nav__link md-nav__link--active">
1187
+
1188
+
1189
+
1190
+ <span class="md-ellipsis">
1191
+
1192
+
1193
+ Rails Integration
1194
+
1195
+
1196
+
1197
+ </span>
1198
+
1199
+
1200
+
1201
+ </a>
1202
+
1203
+
1204
+
1205
+
1206
+
1207
+ <nav class="md-nav md-nav--secondary" aria-label="On this page">
1208
+
1209
+
1210
+
1211
+
1212
+
1213
+
1214
+ <label class="md-nav__title" for="__toc">
1215
+ <span class="md-nav__icon md-icon"></span>
1216
+ On this page
1217
+ </label>
1218
+ <ul class="md-nav__list" data-md-component="toc" data-md-scrollfix>
1219
+
1220
+ <li class="md-nav__item">
1221
+ <a href="#installation" class="md-nav__link">
1222
+ <span class="md-ellipsis">
1223
+
1224
+ Installation
1225
+
1226
+ </span>
1227
+ </a>
1228
+
1229
+ <nav class="md-nav" aria-label="Installation">
1230
+ <ul class="md-nav__list">
1231
+
1232
+ <li class="md-nav__item">
1233
+ <a href="#generate-files" class="md-nav__link">
1234
+ <span class="md-ellipsis">
1235
+
1236
+ Generate Files
1237
+
1238
+ </span>
1239
+ </a>
1240
+
1241
+ </li>
1242
+
1243
+ <li class="md-nav__item">
1244
+ <a href="#run-migrations" class="md-nav__link">
1245
+ <span class="md-ellipsis">
1246
+
1247
+ Run Migrations
1248
+
1249
+ </span>
1250
+ </a>
1251
+
1252
+ </li>
1253
+
1254
+ </ul>
1255
+ </nav>
1256
+
1257
+ </li>
1258
+
1259
+ <li class="md-nav__item">
1260
+ <a href="#configuration" class="md-nav__link">
1261
+ <span class="md-ellipsis">
1262
+
1263
+ Configuration
1264
+
1265
+ </span>
1266
+ </a>
1267
+
1268
+ <nav class="md-nav" aria-label="Configuration">
1269
+ <ul class="md-nav__list">
1270
+
1271
+ <li class="md-nav__item">
1272
+ <a href="#project-config-file" class="md-nav__link">
1273
+ <span class="md-ellipsis">
1274
+
1275
+ Project Config File
1276
+
1277
+ </span>
1278
+ </a>
1279
+
1280
+ </li>
1281
+
1282
+ <li class="md-nav__item">
1283
+ <a href="#environment-variables" class="md-nav__link">
1284
+ <span class="md-ellipsis">
1285
+
1286
+ Environment Variables
1287
+
1288
+ </span>
1289
+ </a>
1290
+
1291
+ </li>
1292
+
1293
+ <li class="md-nav__item">
1294
+ <a href="#initializer-logger-only" class="md-nav__link">
1295
+ <span class="md-ellipsis">
1296
+
1297
+ Initializer (Logger Only)
1298
+
1299
+ </span>
1300
+ </a>
1301
+
1302
+ </li>
1303
+
1304
+ <li class="md-nav__item">
1305
+ <a href="#accessing-configuration" class="md-nav__link">
1306
+ <span class="md-ellipsis">
1307
+
1308
+ Accessing Configuration
1309
+
1310
+ </span>
1311
+ </a>
1312
+
1313
+ </li>
1314
+
1315
+ </ul>
1316
+ </nav>
1317
+
1318
+ </li>
1319
+
1320
+ <li class="md-nav__item">
1321
+ <a href="#creating-robots" class="md-nav__link">
1322
+ <span class="md-ellipsis">
1323
+
1324
+ Creating Robots
1325
+
1326
+ </span>
1327
+ </a>
1328
+
1329
+ <nav class="md-nav" aria-label="Creating Robots">
1330
+ <ul class="md-nav__list">
1331
+
1332
+ <li class="md-nav__item">
1333
+ <a href="#robot-generator" class="md-nav__link">
1334
+ <span class="md-ellipsis">
1335
+
1336
+ Robot Generator
1337
+
1338
+ </span>
1339
+ </a>
1340
+
1341
+ </li>
1342
+
1343
+ <li class="md-nav__item">
1344
+ <a href="#robot-class" class="md-nav__link">
1345
+ <span class="md-ellipsis">
1346
+
1347
+ Robot Class
1348
+
1349
+ </span>
1350
+ </a>
1351
+
1352
+ </li>
1353
+
1354
+ <li class="md-nav__item">
1355
+ <a href="#routing-robot-class" class="md-nav__link">
1356
+ <span class="md-ellipsis">
1357
+
1358
+ Routing Robot Class
1359
+
1360
+ </span>
1361
+ </a>
1362
+
1363
+ </li>
1364
+
1365
+ <li class="md-nav__item">
1366
+ <a href="#custom-tool" class="md-nav__link">
1367
+ <span class="md-ellipsis">
1368
+
1369
+ Custom Tool
1370
+
1371
+ </span>
1372
+ </a>
1373
+
1374
+ </li>
1375
+
1376
+ <li class="md-nav__item">
1377
+ <a href="#using-in-controllers" class="md-nav__link">
1378
+ <span class="md-ellipsis">
1379
+
1380
+ Using in Controllers
1381
+
1382
+ </span>
1383
+ </a>
1384
+
1385
+ </li>
1386
+
1387
+ <li class="md-nav__item">
1388
+ <a href="#using-a-network-in-controllers" class="md-nav__link">
1389
+ <span class="md-ellipsis">
1390
+
1391
+ Using a Network in Controllers
1392
+
1393
+ </span>
1394
+ </a>
1395
+
1396
+ </li>
1397
+
1398
+ </ul>
1399
+ </nav>
1400
+
1401
+ </li>
1402
+
1403
+ <li class="md-nav__item">
1404
+ <a href="#prompt-templates" class="md-nav__link">
1405
+ <span class="md-ellipsis">
1406
+
1407
+ Prompt Templates
1408
+
1409
+ </span>
1410
+ </a>
1411
+
1412
+ <nav class="md-nav" aria-label="Prompt Templates">
1413
+ <ul class="md-nav__list">
1414
+
1415
+ <li class="md-nav__item">
1416
+ <a href="#template-location" class="md-nav__link">
1417
+ <span class="md-ellipsis">
1418
+
1419
+ Template Location
1420
+
1421
+ </span>
1422
+ </a>
1423
+
1424
+ </li>
1425
+
1426
+ <li class="md-nav__item">
1427
+ <a href="#template-format" class="md-nav__link">
1428
+ <span class="md-ellipsis">
1429
+
1430
+ Template Format
1431
+
1432
+ </span>
1433
+ </a>
1434
+
1435
+ </li>
1436
+
1437
+ <li class="md-nav__item">
1438
+ <a href="#template-usage" class="md-nav__link">
1439
+ <span class="md-ellipsis">
1440
+
1441
+ Template Usage
1442
+
1443
+ </span>
1444
+ </a>
1445
+
1446
+ </li>
1447
+
1448
+ </ul>
1449
+ </nav>
1450
+
1451
+ </li>
1452
+
1453
+ <li class="md-nav__item">
1454
+ <a href="#action-cable-integration" class="md-nav__link">
1455
+ <span class="md-ellipsis">
1456
+
1457
+ Action Cable Integration
1458
+
1459
+ </span>
1460
+ </a>
1461
+
1462
+ <nav class="md-nav" aria-label="Action Cable Integration">
1463
+ <ul class="md-nav__list">
1464
+
1465
+ <li class="md-nav__item">
1466
+ <a href="#channel" class="md-nav__link">
1467
+ <span class="md-ellipsis">
1468
+
1469
+ Channel
1470
+
1471
+ </span>
1472
+ </a>
1473
+
1474
+ </li>
1475
+
1476
+ <li class="md-nav__item">
1477
+ <a href="#javascript-client" class="md-nav__link">
1478
+ <span class="md-ellipsis">
1479
+
1480
+ JavaScript Client
1481
+
1482
+ </span>
1483
+ </a>
1484
+
1485
+ </li>
1486
+
1487
+ </ul>
1488
+ </nav>
1489
+
1490
+ </li>
1491
+
1492
+ <li class="md-nav__item">
1493
+ <a href="#background-jobs" class="md-nav__link">
1494
+ <span class="md-ellipsis">
1495
+
1496
+ Background Jobs
1497
+
1498
+ </span>
1499
+ </a>
1500
+
1501
+ <nav class="md-nav" aria-label="Background Jobs">
1502
+ <ul class="md-nav__list">
1503
+
1504
+ <li class="md-nav__item">
1505
+ <a href="#robotrunjob-generated" class="md-nav__link">
1506
+ <span class="md-ellipsis">
1507
+
1508
+ RobotRunJob (Generated)
1509
+
1510
+ </span>
1511
+ </a>
1512
+
1513
+ </li>
1514
+
1515
+ <li class="md-nav__item">
1516
+ <a href="#turbo-stream-token-streaming" class="md-nav__link">
1517
+ <span class="md-ellipsis">
1518
+
1519
+ Turbo Stream Token Streaming
1520
+
1521
+ </span>
1522
+ </a>
1523
+
1524
+ <nav class="md-nav" aria-label="Turbo Stream Token Streaming">
1525
+ <ul class="md-nav__list">
1526
+
1527
+ <li class="md-nav__item">
1528
+ <a href="#view-setup" class="md-nav__link">
1529
+ <span class="md-ellipsis">
1530
+
1531
+ View Setup
1532
+
1533
+ </span>
1534
+ </a>
1535
+
1536
+ </li>
1537
+
1538
+ <li class="md-nav__item">
1539
+ <a href="#turbostreamcallbacks-api" class="md-nav__link">
1540
+ <span class="md-ellipsis">
1541
+
1542
+ TurboStreamCallbacks API
1543
+
1544
+ </span>
1545
+ </a>
1546
+
1547
+ </li>
1548
+
1549
+ </ul>
1550
+ </nav>
1551
+
1552
+ </li>
1553
+
1554
+ <li class="md-nav__item">
1555
+ <a href="#custom-background-job" class="md-nav__link">
1556
+ <span class="md-ellipsis">
1557
+
1558
+ Custom Background Job
1559
+
1560
+ </span>
1561
+ </a>
1562
+
1563
+ </li>
1564
+
1565
+ </ul>
1566
+ </nav>
1567
+
1568
+ </li>
1569
+
1570
+ <li class="md-nav__item">
1571
+ <a href="#testing" class="md-nav__link">
1572
+ <span class="md-ellipsis">
1573
+
1574
+ Testing
1575
+
1576
+ </span>
1577
+ </a>
1578
+
1579
+ <nav class="md-nav" aria-label="Testing">
1580
+ <ul class="md-nav__list">
1581
+
1582
+ <li class="md-nav__item">
1583
+ <a href="#test-configuration" class="md-nav__link">
1584
+ <span class="md-ellipsis">
1585
+
1586
+ Test Configuration
1587
+
1588
+ </span>
1589
+ </a>
1590
+
1591
+ </li>
1592
+
1593
+ <li class="md-nav__item">
1594
+ <a href="#robot-tests" class="md-nav__link">
1595
+ <span class="md-ellipsis">
1596
+
1597
+ Robot Tests
1598
+
1599
+ </span>
1600
+ </a>
1601
+
1602
+ </li>
1603
+
1604
+ <li class="md-nav__item">
1605
+ <a href="#integration-tests" class="md-nav__link">
1606
+ <span class="md-ellipsis">
1607
+
1608
+ Integration Tests
1609
+
1610
+ </span>
1611
+ </a>
1612
+
1613
+ </li>
1614
+
1615
+ </ul>
1616
+ </nav>
1617
+
1618
+ </li>
1619
+
1620
+ <li class="md-nav__item">
1621
+ <a href="#models" class="md-nav__link">
1622
+ <span class="md-ellipsis">
1623
+
1624
+ Models
1625
+
1626
+ </span>
1627
+ </a>
1628
+
1629
+ <nav class="md-nav" aria-label="Models">
1630
+ <ul class="md-nav__list">
1631
+
1632
+ <li class="md-nav__item">
1633
+ <a href="#thread-model" class="md-nav__link">
1634
+ <span class="md-ellipsis">
1635
+
1636
+ Thread Model
1637
+
1638
+ </span>
1639
+ </a>
1640
+
1641
+ </li>
1642
+
1643
+ <li class="md-nav__item">
1644
+ <a href="#result-model" class="md-nav__link">
1645
+ <span class="md-ellipsis">
1646
+
1647
+ Result Model
1648
+
1649
+ </span>
1650
+ </a>
1651
+
1652
+ </li>
1653
+
1654
+ </ul>
1655
+ </nav>
1656
+
1657
+ </li>
1658
+
1659
+ <li class="md-nav__item">
1660
+ <a href="#best-practices" class="md-nav__link">
1661
+ <span class="md-ellipsis">
1662
+
1663
+ Best Practices
1664
+
1665
+ </span>
1666
+ </a>
1667
+
1668
+ <nav class="md-nav" aria-label="Best Practices">
1669
+ <ul class="md-nav__list">
1670
+
1671
+ <li class="md-nav__item">
1672
+ <a href="#1-use-service-objects" class="md-nav__link">
1673
+ <span class="md-ellipsis">
1674
+
1675
+ 1. Use Service Objects
1676
+
1677
+ </span>
1678
+ </a>
1679
+
1680
+ </li>
1681
+
1682
+ <li class="md-nav__item">
1683
+ <a href="#2-handle-errors" class="md-nav__link">
1684
+ <span class="md-ellipsis">
1685
+
1686
+ 2. Handle Errors
1687
+
1688
+ </span>
1689
+ </a>
1690
+
1691
+ </li>
1692
+
1693
+ <li class="md-nav__item">
1694
+ <a href="#3-rate-limiting" class="md-nav__link">
1695
+ <span class="md-ellipsis">
1696
+
1697
+ 3. Rate Limiting
1698
+
1699
+ </span>
1700
+ </a>
1701
+
1702
+ </li>
1703
+
1704
+ </ul>
1705
+ </nav>
1706
+
1707
+ </li>
1708
+
1709
+ <li class="md-nav__item">
1710
+ <a href="#next-steps" class="md-nav__link">
1711
+ <span class="md-ellipsis">
1712
+
1713
+ Next Steps
1714
+
1715
+ </span>
1716
+ </a>
1717
+
1718
+ </li>
1719
+
1720
+ </ul>
1721
+
1722
+ </nav>
1723
+
1724
+ </li>
1725
+
1726
+
1727
+
1728
+
1729
+ </ul>
1730
+ </nav>
1731
+
1732
+ </li>
1733
+
1734
+
1735
+
1736
+
1737
+
1738
+
1739
+
1740
+
1741
+
1742
+
1743
+
1744
+
1745
+
1746
+
1747
+
1748
+
1749
+
1750
+
1751
+
1752
+
1753
+
1754
+
1755
+
1756
+
1757
+
1758
+
1759
+
1760
+
1761
+
1762
+
1763
+
1764
+ <li class="md-nav__item md-nav__item--nested">
1765
+
1766
+
1767
+
1768
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_5" >
1769
+
1770
+
1771
+ <div class="md-nav__link md-nav__container">
1772
+ <a href="../../api/" class="md-nav__link ">
1773
+
1774
+
1775
+
1776
+ <span class="md-ellipsis">
1777
+
1778
+
1779
+ API Reference
1780
+
1781
+
1782
+
1783
+ </span>
1784
+
1785
+
1786
+
1787
+ </a>
1788
+
1789
+
1790
+ <label class="md-nav__link " for="__nav_5" id="__nav_5_label" tabindex="0">
1791
+ <span class="md-nav__icon md-icon"></span>
1792
+ </label>
1793
+
1794
+ </div>
1795
+
1796
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_5_label" aria-expanded="false">
1797
+ <label class="md-nav__title" for="__nav_5">
1798
+ <span class="md-nav__icon md-icon"></span>
1799
+
1800
+
1801
+ API Reference
1802
+
1803
+
1804
+ </label>
1805
+ <ul class="md-nav__list" data-md-scrollfix>
1806
+
1807
+
1808
+
1809
+
1810
+
1811
+
1812
+
1813
+
1814
+
1815
+
1816
+
1817
+
1818
+
1819
+
1820
+
1821
+
1822
+
1823
+
1824
+
1825
+
1826
+
1827
+
1828
+
1829
+
1830
+
1831
+
1832
+
1833
+
1834
+
1835
+
1836
+
1837
+
1838
+
1839
+ <li class="md-nav__item md-nav__item--nested">
1840
+
1841
+
1842
+
1843
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_5_2" >
1844
+
1845
+
1846
+ <div class="md-nav__link md-nav__container">
1847
+ <a href="../../api/core/" class="md-nav__link ">
1848
+
1849
+
1850
+
1851
+ <span class="md-ellipsis">
1852
+
1853
+
1854
+ Core Classes
1855
+
1856
+
1857
+
1858
+ </span>
1859
+
1860
+
1861
+
1862
+ </a>
1863
+
1864
+
1865
+ <label class="md-nav__link " for="__nav_5_2" id="__nav_5_2_label" tabindex="0">
1866
+ <span class="md-nav__icon md-icon"></span>
1867
+ </label>
1868
+
1869
+ </div>
1870
+
1871
+ <nav class="md-nav" data-md-level="2" aria-labelledby="__nav_5_2_label" aria-expanded="false">
1872
+ <label class="md-nav__title" for="__nav_5_2">
1873
+ <span class="md-nav__icon md-icon"></span>
1874
+
1875
+
1876
+ Core Classes
1877
+
1878
+
1879
+ </label>
1880
+ <ul class="md-nav__list" data-md-scrollfix>
1881
+
1882
+
1883
+
1884
+
1885
+
1886
+
1887
+
1888
+
1889
+
1890
+ <li class="md-nav__item">
1891
+ <a href="../../api/core/robot/" class="md-nav__link">
1892
+
1893
+
1894
+
1895
+ <span class="md-ellipsis">
1896
+
1897
+
1898
+ Robot
1899
+
1900
+
1901
+
1902
+ </span>
1903
+
1904
+
1905
+
1906
+ </a>
1907
+ </li>
1908
+
1909
+
1910
+
1911
+
1912
+
1913
+
1914
+
1915
+
1916
+
1917
+
1918
+ <li class="md-nav__item">
1919
+ <a href="../../api/core/network/" class="md-nav__link">
1920
+
1921
+
1922
+
1923
+ <span class="md-ellipsis">
1924
+
1925
+
1926
+ Network
1927
+
1928
+
1929
+
1930
+ </span>
1931
+
1932
+
1933
+
1934
+ </a>
1935
+ </li>
1936
+
1937
+
1938
+
1939
+
1940
+
1941
+
1942
+
1943
+
1944
+
1945
+
1946
+ <li class="md-nav__item">
1947
+ <a href="../../api/core/state/" class="md-nav__link">
1948
+
1949
+
1950
+
1951
+ <span class="md-ellipsis">
1952
+
1953
+
1954
+ State
1955
+
1956
+
1957
+
1958
+ </span>
1959
+
1960
+
1961
+
1962
+ </a>
1963
+ </li>
1964
+
1965
+
1966
+
1967
+
1968
+
1969
+
1970
+
1971
+
1972
+
1973
+
1974
+ <li class="md-nav__item">
1975
+ <a href="../../api/core/tool/" class="md-nav__link">
1976
+
1977
+
1978
+
1979
+ <span class="md-ellipsis">
1980
+
1981
+
1982
+ Tool
1983
+
1984
+
1985
+
1986
+ </span>
1987
+
1988
+
1989
+
1990
+ </a>
1991
+ </li>
1992
+
1993
+
1994
+
1995
+
1996
+
1997
+
1998
+
1999
+
2000
+
2001
+
2002
+ <li class="md-nav__item">
2003
+ <a href="../../api/core/memory/" class="md-nav__link">
2004
+
2005
+
2006
+
2007
+ <span class="md-ellipsis">
2008
+
2009
+
2010
+ Memory
2011
+
2012
+
2013
+
2014
+ </span>
2015
+
2016
+
2017
+
2018
+ </a>
2019
+ </li>
2020
+
2021
+
2022
+
2023
+
2024
+ </ul>
2025
+ </nav>
2026
+
2027
+ </li>
2028
+
2029
+
2030
+
2031
+
2032
+
2033
+
2034
+
2035
+
2036
+
2037
+
2038
+
2039
+
2040
+
2041
+
2042
+
2043
+
2044
+
2045
+
2046
+
2047
+
2048
+
2049
+
2050
+
2051
+
2052
+
2053
+
2054
+
2055
+
2056
+
2057
+
2058
+
2059
+
2060
+ <li class="md-nav__item md-nav__item--nested">
2061
+
2062
+
2063
+
2064
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_5_3" >
2065
+
2066
+
2067
+ <div class="md-nav__link md-nav__container">
2068
+ <a href="../../api/messages/" class="md-nav__link ">
2069
+
2070
+
2071
+
2072
+ <span class="md-ellipsis">
2073
+
2074
+
2075
+ Messages
2076
+
2077
+
2078
+
2079
+ </span>
2080
+
2081
+
2082
+
2083
+ </a>
2084
+
2085
+
2086
+ <label class="md-nav__link " for="__nav_5_3" id="__nav_5_3_label" tabindex="0">
2087
+ <span class="md-nav__icon md-icon"></span>
2088
+ </label>
2089
+
2090
+ </div>
2091
+
2092
+ <nav class="md-nav" data-md-level="2" aria-labelledby="__nav_5_3_label" aria-expanded="false">
2093
+ <label class="md-nav__title" for="__nav_5_3">
2094
+ <span class="md-nav__icon md-icon"></span>
2095
+
2096
+
2097
+ Messages
2098
+
2099
+
2100
+ </label>
2101
+ <ul class="md-nav__list" data-md-scrollfix>
2102
+
2103
+
2104
+
2105
+
2106
+
2107
+
2108
+
2109
+
2110
+
2111
+ <li class="md-nav__item">
2112
+ <a href="../../api/messages/user-message/" class="md-nav__link">
2113
+
2114
+
2115
+
2116
+ <span class="md-ellipsis">
2117
+
2118
+
2119
+ UserMessage
2120
+
2121
+
2122
+
2123
+ </span>
2124
+
2125
+
2126
+
2127
+ </a>
2128
+ </li>
2129
+
2130
+
2131
+
2132
+
2133
+
2134
+
2135
+
2136
+
2137
+
2138
+
2139
+ <li class="md-nav__item">
2140
+ <a href="../../api/messages/text-message/" class="md-nav__link">
2141
+
2142
+
2143
+
2144
+ <span class="md-ellipsis">
2145
+
2146
+
2147
+ TextMessage
2148
+
2149
+
2150
+
2151
+ </span>
2152
+
2153
+
2154
+
2155
+ </a>
2156
+ </li>
2157
+
2158
+
2159
+
2160
+
2161
+
2162
+
2163
+
2164
+
2165
+
2166
+
2167
+ <li class="md-nav__item">
2168
+ <a href="../../api/messages/tool-call-message/" class="md-nav__link">
2169
+
2170
+
2171
+
2172
+ <span class="md-ellipsis">
2173
+
2174
+
2175
+ ToolCallMessage
2176
+
2177
+
2178
+
2179
+ </span>
2180
+
2181
+
2182
+
2183
+ </a>
2184
+ </li>
2185
+
2186
+
2187
+
2188
+
2189
+
2190
+
2191
+
2192
+
2193
+
2194
+
2195
+ <li class="md-nav__item">
2196
+ <a href="../../api/messages/tool-result-message/" class="md-nav__link">
2197
+
2198
+
2199
+
2200
+ <span class="md-ellipsis">
2201
+
2202
+
2203
+ ToolResultMessage
2204
+
2205
+
2206
+
2207
+ </span>
2208
+
2209
+
2210
+
2211
+ </a>
2212
+ </li>
2213
+
2214
+
2215
+
2216
+
2217
+ </ul>
2218
+ </nav>
2219
+
2220
+ </li>
2221
+
2222
+
2223
+
2224
+
2225
+
2226
+
2227
+
2228
+
2229
+
2230
+
2231
+
2232
+
2233
+
2234
+
2235
+
2236
+
2237
+
2238
+
2239
+
2240
+
2241
+
2242
+
2243
+
2244
+
2245
+
2246
+
2247
+
2248
+
2249
+
2250
+
2251
+ <li class="md-nav__item md-nav__item--nested">
2252
+
2253
+
2254
+
2255
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_5_4" >
2256
+
2257
+
2258
+ <div class="md-nav__link md-nav__container">
2259
+ <a href="../../api/mcp/" class="md-nav__link ">
2260
+
2261
+
2262
+
2263
+ <span class="md-ellipsis">
2264
+
2265
+
2266
+ MCP
2267
+
2268
+
2269
+
2270
+ </span>
2271
+
2272
+
2273
+
2274
+ </a>
2275
+
2276
+
2277
+ <label class="md-nav__link " for="__nav_5_4" id="__nav_5_4_label" tabindex="0">
2278
+ <span class="md-nav__icon md-icon"></span>
2279
+ </label>
2280
+
2281
+ </div>
2282
+
2283
+ <nav class="md-nav" data-md-level="2" aria-labelledby="__nav_5_4_label" aria-expanded="false">
2284
+ <label class="md-nav__title" for="__nav_5_4">
2285
+ <span class="md-nav__icon md-icon"></span>
2286
+
2287
+
2288
+ MCP
2289
+
2290
+
2291
+ </label>
2292
+ <ul class="md-nav__list" data-md-scrollfix>
2293
+
2294
+
2295
+
2296
+
2297
+
2298
+
2299
+
2300
+
2301
+
2302
+ <li class="md-nav__item">
2303
+ <a href="../../api/mcp/client/" class="md-nav__link">
2304
+
2305
+
2306
+
2307
+ <span class="md-ellipsis">
2308
+
2309
+
2310
+ Client
2311
+
2312
+
2313
+
2314
+ </span>
2315
+
2316
+
2317
+
2318
+ </a>
2319
+ </li>
2320
+
2321
+
2322
+
2323
+
2324
+
2325
+
2326
+
2327
+
2328
+
2329
+
2330
+ <li class="md-nav__item">
2331
+ <a href="../../api/mcp/server/" class="md-nav__link">
2332
+
2333
+
2334
+
2335
+ <span class="md-ellipsis">
2336
+
2337
+
2338
+ Server
2339
+
2340
+
2341
+
2342
+ </span>
2343
+
2344
+
2345
+
2346
+ </a>
2347
+ </li>
2348
+
2349
+
2350
+
2351
+
2352
+
2353
+
2354
+
2355
+
2356
+
2357
+
2358
+ <li class="md-nav__item">
2359
+ <a href="../../api/mcp/transports/" class="md-nav__link">
2360
+
2361
+
2362
+
2363
+ <span class="md-ellipsis">
2364
+
2365
+
2366
+ Transports
2367
+
2368
+
2369
+
2370
+ </span>
2371
+
2372
+
2373
+
2374
+ </a>
2375
+ </li>
2376
+
2377
+
2378
+
2379
+
2380
+ </ul>
2381
+ </nav>
2382
+
2383
+ </li>
2384
+
2385
+
2386
+
2387
+
2388
+
2389
+
2390
+
2391
+
2392
+
2393
+
2394
+
2395
+
2396
+
2397
+
2398
+
2399
+
2400
+
2401
+
2402
+
2403
+
2404
+
2405
+
2406
+
2407
+
2408
+
2409
+
2410
+
2411
+
2412
+ <li class="md-nav__item md-nav__item--nested">
2413
+
2414
+
2415
+
2416
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_5_5" >
2417
+
2418
+
2419
+ <div class="md-nav__link md-nav__container">
2420
+ <a href="../../api/streaming/" class="md-nav__link ">
2421
+
2422
+
2423
+
2424
+ <span class="md-ellipsis">
2425
+
2426
+
2427
+ Streaming
2428
+
2429
+
2430
+
2431
+ </span>
2432
+
2433
+
2434
+
2435
+ </a>
2436
+
2437
+
2438
+ <label class="md-nav__link " for="__nav_5_5" id="__nav_5_5_label" tabindex="0">
2439
+ <span class="md-nav__icon md-icon"></span>
2440
+ </label>
2441
+
2442
+ </div>
2443
+
2444
+ <nav class="md-nav" data-md-level="2" aria-labelledby="__nav_5_5_label" aria-expanded="false">
2445
+ <label class="md-nav__title" for="__nav_5_5">
2446
+ <span class="md-nav__icon md-icon"></span>
2447
+
2448
+
2449
+ Streaming
2450
+
2451
+
2452
+ </label>
2453
+ <ul class="md-nav__list" data-md-scrollfix>
2454
+
2455
+
2456
+
2457
+
2458
+
2459
+
2460
+
2461
+
2462
+
2463
+ <li class="md-nav__item">
2464
+ <a href="../../api/streaming/context/" class="md-nav__link">
2465
+
2466
+
2467
+
2468
+ <span class="md-ellipsis">
2469
+
2470
+
2471
+ Context
2472
+
2473
+
2474
+
2475
+ </span>
2476
+
2477
+
2478
+
2479
+ </a>
2480
+ </li>
2481
+
2482
+
2483
+
2484
+
2485
+
2486
+
2487
+
2488
+
2489
+
2490
+
2491
+ <li class="md-nav__item">
2492
+ <a href="../../api/streaming/events/" class="md-nav__link">
2493
+
2494
+
2495
+
2496
+ <span class="md-ellipsis">
2497
+
2498
+
2499
+ Events
2500
+
2501
+
2502
+
2503
+ </span>
2504
+
2505
+
2506
+
2507
+ </a>
2508
+ </li>
2509
+
2510
+
2511
+
2512
+
2513
+ </ul>
2514
+ </nav>
2515
+
2516
+ </li>
2517
+
2518
+
2519
+
2520
+
2521
+ </ul>
2522
+ </nav>
2523
+
2524
+ </li>
2525
+
2526
+
2527
+
2528
+
2529
+
2530
+
2531
+
2532
+
2533
+
2534
+
2535
+
2536
+
2537
+
2538
+
2539
+
2540
+
2541
+
2542
+
2543
+
2544
+
2545
+
2546
+
2547
+
2548
+
2549
+
2550
+
2551
+
2552
+
2553
+
2554
+
2555
+
2556
+
2557
+
2558
+ <li class="md-nav__item md-nav__item--nested">
2559
+
2560
+
2561
+
2562
+ <input class="md-nav__toggle md-toggle " type="checkbox" id="__nav_6" >
2563
+
2564
+
2565
+ <div class="md-nav__link md-nav__container">
2566
+ <a href="../../examples/" class="md-nav__link ">
2567
+
2568
+
2569
+
2570
+ <span class="md-ellipsis">
2571
+
2572
+
2573
+ Examples
2574
+
2575
+
2576
+
2577
+ </span>
2578
+
2579
+
2580
+
2581
+ </a>
2582
+
2583
+
2584
+ <label class="md-nav__link " for="__nav_6" id="__nav_6_label" tabindex="0">
2585
+ <span class="md-nav__icon md-icon"></span>
2586
+ </label>
2587
+
2588
+ </div>
2589
+
2590
+ <nav class="md-nav" data-md-level="1" aria-labelledby="__nav_6_label" aria-expanded="false">
2591
+ <label class="md-nav__title" for="__nav_6">
2592
+ <span class="md-nav__icon md-icon"></span>
2593
+
2594
+
2595
+ Examples
2596
+
2597
+
2598
+ </label>
2599
+ <ul class="md-nav__list" data-md-scrollfix>
2600
+
2601
+
2602
+
2603
+
2604
+
2605
+
2606
+
2607
+
2608
+
2609
+ <li class="md-nav__item">
2610
+ <a href="../../examples/basic-chat/" class="md-nav__link">
2611
+
2612
+
2613
+
2614
+ <span class="md-ellipsis">
2615
+
2616
+
2617
+ Basic Chat
2618
+
2619
+
2620
+
2621
+ </span>
2622
+
2623
+
2624
+
2625
+ </a>
2626
+ </li>
2627
+
2628
+
2629
+
2630
+
2631
+
2632
+
2633
+
2634
+
2635
+
2636
+
2637
+ <li class="md-nav__item">
2638
+ <a href="../../examples/multi-robot-network/" class="md-nav__link">
2639
+
2640
+
2641
+
2642
+ <span class="md-ellipsis">
2643
+
2644
+
2645
+ Multi-Robot Network
2646
+
2647
+
2648
+
2649
+ </span>
2650
+
2651
+
2652
+
2653
+ </a>
2654
+ </li>
2655
+
2656
+
2657
+
2658
+
2659
+
2660
+
2661
+
2662
+
2663
+
2664
+
2665
+ <li class="md-nav__item">
2666
+ <a href="../../examples/tool-usage/" class="md-nav__link">
2667
+
2668
+
2669
+
2670
+ <span class="md-ellipsis">
2671
+
2672
+
2673
+ Tool Usage
2674
+
2675
+
2676
+
2677
+ </span>
2678
+
2679
+
2680
+
2681
+ </a>
2682
+ </li>
2683
+
2684
+
2685
+
2686
+
2687
+
2688
+
2689
+
2690
+
2691
+
2692
+
2693
+ <li class="md-nav__item">
2694
+ <a href="../../examples/mcp-server/" class="md-nav__link">
2695
+
2696
+
2697
+
2698
+ <span class="md-ellipsis">
2699
+
2700
+
2701
+ MCP Server
2702
+
2703
+
2704
+
2705
+ </span>
2706
+
2707
+
2708
+
2709
+ </a>
2710
+ </li>
2711
+
2712
+
2713
+
2714
+
2715
+
2716
+
2717
+
2718
+
2719
+
2720
+
2721
+ <li class="md-nav__item">
2722
+ <a href="../../examples/rails-application/" class="md-nav__link">
2723
+
2724
+
2725
+
2726
+ <span class="md-ellipsis">
2727
+
2728
+
2729
+ Rails Application
2730
+
2731
+
2732
+
2733
+ </span>
2734
+
2735
+
2736
+
2737
+ </a>
2738
+ </li>
2739
+
2740
+
2741
+
2742
+
2743
+ </ul>
2744
+ </nav>
2745
+
2746
+ </li>
2747
+
2748
+
2749
+
2750
+ </ul>
2751
+ </nav>
2752
+ </div>
2753
+ </div>
2754
+ </div>
2755
+
2756
+
2757
+
2758
+ <div class="md-sidebar md-sidebar--secondary" data-md-component="sidebar" data-md-type="toc" >
2759
+ <div class="md-sidebar__scrollwrap">
2760
+ <div class="md-sidebar__inner">
2761
+
2762
+
2763
+
2764
+
2765
+ <nav class="md-nav md-nav--secondary" aria-label="On this page">
2766
+
2767
+
2768
+
2769
+
2770
+
2771
+
2772
+ <label class="md-nav__title" for="__toc">
2773
+ <span class="md-nav__icon md-icon"></span>
2774
+ On this page
2775
+ </label>
2776
+ <ul class="md-nav__list" data-md-component="toc" data-md-scrollfix>
2777
+
2778
+ <li class="md-nav__item">
2779
+ <a href="#installation" class="md-nav__link">
2780
+ <span class="md-ellipsis">
2781
+
2782
+ Installation
2783
+
2784
+ </span>
2785
+ </a>
2786
+
2787
+ <nav class="md-nav" aria-label="Installation">
2788
+ <ul class="md-nav__list">
2789
+
2790
+ <li class="md-nav__item">
2791
+ <a href="#generate-files" class="md-nav__link">
2792
+ <span class="md-ellipsis">
2793
+
2794
+ Generate Files
2795
+
2796
+ </span>
2797
+ </a>
2798
+
2799
+ </li>
2800
+
2801
+ <li class="md-nav__item">
2802
+ <a href="#run-migrations" class="md-nav__link">
2803
+ <span class="md-ellipsis">
2804
+
2805
+ Run Migrations
2806
+
2807
+ </span>
2808
+ </a>
2809
+
2810
+ </li>
2811
+
2812
+ </ul>
2813
+ </nav>
2814
+
2815
+ </li>
2816
+
2817
+ <li class="md-nav__item">
2818
+ <a href="#configuration" class="md-nav__link">
2819
+ <span class="md-ellipsis">
2820
+
2821
+ Configuration
2822
+
2823
+ </span>
2824
+ </a>
2825
+
2826
+ <nav class="md-nav" aria-label="Configuration">
2827
+ <ul class="md-nav__list">
2828
+
2829
+ <li class="md-nav__item">
2830
+ <a href="#project-config-file" class="md-nav__link">
2831
+ <span class="md-ellipsis">
2832
+
2833
+ Project Config File
2834
+
2835
+ </span>
2836
+ </a>
2837
+
2838
+ </li>
2839
+
2840
+ <li class="md-nav__item">
2841
+ <a href="#environment-variables" class="md-nav__link">
2842
+ <span class="md-ellipsis">
2843
+
2844
+ Environment Variables
2845
+
2846
+ </span>
2847
+ </a>
2848
+
2849
+ </li>
2850
+
2851
+ <li class="md-nav__item">
2852
+ <a href="#initializer-logger-only" class="md-nav__link">
2853
+ <span class="md-ellipsis">
2854
+
2855
+ Initializer (Logger Only)
2856
+
2857
+ </span>
2858
+ </a>
2859
+
2860
+ </li>
2861
+
2862
+ <li class="md-nav__item">
2863
+ <a href="#accessing-configuration" class="md-nav__link">
2864
+ <span class="md-ellipsis">
2865
+
2866
+ Accessing Configuration
2867
+
2868
+ </span>
2869
+ </a>
2870
+
2871
+ </li>
2872
+
2873
+ </ul>
2874
+ </nav>
2875
+
2876
+ </li>
2877
+
2878
+ <li class="md-nav__item">
2879
+ <a href="#creating-robots" class="md-nav__link">
2880
+ <span class="md-ellipsis">
2881
+
2882
+ Creating Robots
2883
+
2884
+ </span>
2885
+ </a>
2886
+
2887
+ <nav class="md-nav" aria-label="Creating Robots">
2888
+ <ul class="md-nav__list">
2889
+
2890
+ <li class="md-nav__item">
2891
+ <a href="#robot-generator" class="md-nav__link">
2892
+ <span class="md-ellipsis">
2893
+
2894
+ Robot Generator
2895
+
2896
+ </span>
2897
+ </a>
2898
+
2899
+ </li>
2900
+
2901
+ <li class="md-nav__item">
2902
+ <a href="#robot-class" class="md-nav__link">
2903
+ <span class="md-ellipsis">
2904
+
2905
+ Robot Class
2906
+
2907
+ </span>
2908
+ </a>
2909
+
2910
+ </li>
2911
+
2912
+ <li class="md-nav__item">
2913
+ <a href="#routing-robot-class" class="md-nav__link">
2914
+ <span class="md-ellipsis">
2915
+
2916
+ Routing Robot Class
2917
+
2918
+ </span>
2919
+ </a>
2920
+
2921
+ </li>
2922
+
2923
+ <li class="md-nav__item">
2924
+ <a href="#custom-tool" class="md-nav__link">
2925
+ <span class="md-ellipsis">
2926
+
2927
+ Custom Tool
2928
+
2929
+ </span>
2930
+ </a>
2931
+
2932
+ </li>
2933
+
2934
+ <li class="md-nav__item">
2935
+ <a href="#using-in-controllers" class="md-nav__link">
2936
+ <span class="md-ellipsis">
2937
+
2938
+ Using in Controllers
2939
+
2940
+ </span>
2941
+ </a>
2942
+
2943
+ </li>
2944
+
2945
+ <li class="md-nav__item">
2946
+ <a href="#using-a-network-in-controllers" class="md-nav__link">
2947
+ <span class="md-ellipsis">
2948
+
2949
+ Using a Network in Controllers
2950
+
2951
+ </span>
2952
+ </a>
2953
+
2954
+ </li>
2955
+
2956
+ </ul>
2957
+ </nav>
2958
+
2959
+ </li>
2960
+
2961
+ <li class="md-nav__item">
2962
+ <a href="#prompt-templates" class="md-nav__link">
2963
+ <span class="md-ellipsis">
2964
+
2965
+ Prompt Templates
2966
+
2967
+ </span>
2968
+ </a>
2969
+
2970
+ <nav class="md-nav" aria-label="Prompt Templates">
2971
+ <ul class="md-nav__list">
2972
+
2973
+ <li class="md-nav__item">
2974
+ <a href="#template-location" class="md-nav__link">
2975
+ <span class="md-ellipsis">
2976
+
2977
+ Template Location
2978
+
2979
+ </span>
2980
+ </a>
2981
+
2982
+ </li>
2983
+
2984
+ <li class="md-nav__item">
2985
+ <a href="#template-format" class="md-nav__link">
2986
+ <span class="md-ellipsis">
2987
+
2988
+ Template Format
2989
+
2990
+ </span>
2991
+ </a>
2992
+
2993
+ </li>
2994
+
2995
+ <li class="md-nav__item">
2996
+ <a href="#template-usage" class="md-nav__link">
2997
+ <span class="md-ellipsis">
2998
+
2999
+ Template Usage
3000
+
3001
+ </span>
3002
+ </a>
3003
+
3004
+ </li>
3005
+
3006
+ </ul>
3007
+ </nav>
3008
+
3009
+ </li>
3010
+
3011
+ <li class="md-nav__item">
3012
+ <a href="#action-cable-integration" class="md-nav__link">
3013
+ <span class="md-ellipsis">
3014
+
3015
+ Action Cable Integration
3016
+
3017
+ </span>
3018
+ </a>
3019
+
3020
+ <nav class="md-nav" aria-label="Action Cable Integration">
3021
+ <ul class="md-nav__list">
3022
+
3023
+ <li class="md-nav__item">
3024
+ <a href="#channel" class="md-nav__link">
3025
+ <span class="md-ellipsis">
3026
+
3027
+ Channel
3028
+
3029
+ </span>
3030
+ </a>
3031
+
3032
+ </li>
3033
+
3034
+ <li class="md-nav__item">
3035
+ <a href="#javascript-client" class="md-nav__link">
3036
+ <span class="md-ellipsis">
3037
+
3038
+ JavaScript Client
3039
+
3040
+ </span>
3041
+ </a>
3042
+
3043
+ </li>
3044
+
3045
+ </ul>
3046
+ </nav>
3047
+
3048
+ </li>
3049
+
3050
+ <li class="md-nav__item">
3051
+ <a href="#background-jobs" class="md-nav__link">
3052
+ <span class="md-ellipsis">
3053
+
3054
+ Background Jobs
3055
+
3056
+ </span>
3057
+ </a>
3058
+
3059
+ <nav class="md-nav" aria-label="Background Jobs">
3060
+ <ul class="md-nav__list">
3061
+
3062
+ <li class="md-nav__item">
3063
+ <a href="#robotrunjob-generated" class="md-nav__link">
3064
+ <span class="md-ellipsis">
3065
+
3066
+ RobotRunJob (Generated)
3067
+
3068
+ </span>
3069
+ </a>
3070
+
3071
+ </li>
3072
+
3073
+ <li class="md-nav__item">
3074
+ <a href="#turbo-stream-token-streaming" class="md-nav__link">
3075
+ <span class="md-ellipsis">
3076
+
3077
+ Turbo Stream Token Streaming
3078
+
3079
+ </span>
3080
+ </a>
3081
+
3082
+ <nav class="md-nav" aria-label="Turbo Stream Token Streaming">
3083
+ <ul class="md-nav__list">
3084
+
3085
+ <li class="md-nav__item">
3086
+ <a href="#view-setup" class="md-nav__link">
3087
+ <span class="md-ellipsis">
3088
+
3089
+ View Setup
3090
+
3091
+ </span>
3092
+ </a>
3093
+
3094
+ </li>
3095
+
3096
+ <li class="md-nav__item">
3097
+ <a href="#turbostreamcallbacks-api" class="md-nav__link">
3098
+ <span class="md-ellipsis">
3099
+
3100
+ TurboStreamCallbacks API
3101
+
3102
+ </span>
3103
+ </a>
3104
+
3105
+ </li>
3106
+
3107
+ </ul>
3108
+ </nav>
3109
+
3110
+ </li>
3111
+
3112
+ <li class="md-nav__item">
3113
+ <a href="#custom-background-job" class="md-nav__link">
3114
+ <span class="md-ellipsis">
3115
+
3116
+ Custom Background Job
3117
+
3118
+ </span>
3119
+ </a>
3120
+
3121
+ </li>
3122
+
3123
+ </ul>
3124
+ </nav>
3125
+
3126
+ </li>
3127
+
3128
+ <li class="md-nav__item">
3129
+ <a href="#testing" class="md-nav__link">
3130
+ <span class="md-ellipsis">
3131
+
3132
+ Testing
3133
+
3134
+ </span>
3135
+ </a>
3136
+
3137
+ <nav class="md-nav" aria-label="Testing">
3138
+ <ul class="md-nav__list">
3139
+
3140
+ <li class="md-nav__item">
3141
+ <a href="#test-configuration" class="md-nav__link">
3142
+ <span class="md-ellipsis">
3143
+
3144
+ Test Configuration
3145
+
3146
+ </span>
3147
+ </a>
3148
+
3149
+ </li>
3150
+
3151
+ <li class="md-nav__item">
3152
+ <a href="#robot-tests" class="md-nav__link">
3153
+ <span class="md-ellipsis">
3154
+
3155
+ Robot Tests
3156
+
3157
+ </span>
3158
+ </a>
3159
+
3160
+ </li>
3161
+
3162
+ <li class="md-nav__item">
3163
+ <a href="#integration-tests" class="md-nav__link">
3164
+ <span class="md-ellipsis">
3165
+
3166
+ Integration Tests
3167
+
3168
+ </span>
3169
+ </a>
3170
+
3171
+ </li>
3172
+
3173
+ </ul>
3174
+ </nav>
3175
+
3176
+ </li>
3177
+
3178
+ <li class="md-nav__item">
3179
+ <a href="#models" class="md-nav__link">
3180
+ <span class="md-ellipsis">
3181
+
3182
+ Models
3183
+
3184
+ </span>
3185
+ </a>
3186
+
3187
+ <nav class="md-nav" aria-label="Models">
3188
+ <ul class="md-nav__list">
3189
+
3190
+ <li class="md-nav__item">
3191
+ <a href="#thread-model" class="md-nav__link">
3192
+ <span class="md-ellipsis">
3193
+
3194
+ Thread Model
3195
+
3196
+ </span>
3197
+ </a>
3198
+
3199
+ </li>
3200
+
3201
+ <li class="md-nav__item">
3202
+ <a href="#result-model" class="md-nav__link">
3203
+ <span class="md-ellipsis">
3204
+
3205
+ Result Model
3206
+
3207
+ </span>
3208
+ </a>
3209
+
3210
+ </li>
3211
+
3212
+ </ul>
3213
+ </nav>
3214
+
3215
+ </li>
3216
+
3217
+ <li class="md-nav__item">
3218
+ <a href="#best-practices" class="md-nav__link">
3219
+ <span class="md-ellipsis">
3220
+
3221
+ Best Practices
3222
+
3223
+ </span>
3224
+ </a>
3225
+
3226
+ <nav class="md-nav" aria-label="Best Practices">
3227
+ <ul class="md-nav__list">
3228
+
3229
+ <li class="md-nav__item">
3230
+ <a href="#1-use-service-objects" class="md-nav__link">
3231
+ <span class="md-ellipsis">
3232
+
3233
+ 1. Use Service Objects
3234
+
3235
+ </span>
3236
+ </a>
3237
+
3238
+ </li>
3239
+
3240
+ <li class="md-nav__item">
3241
+ <a href="#2-handle-errors" class="md-nav__link">
3242
+ <span class="md-ellipsis">
3243
+
3244
+ 2. Handle Errors
3245
+
3246
+ </span>
3247
+ </a>
3248
+
3249
+ </li>
3250
+
3251
+ <li class="md-nav__item">
3252
+ <a href="#3-rate-limiting" class="md-nav__link">
3253
+ <span class="md-ellipsis">
3254
+
3255
+ 3. Rate Limiting
3256
+
3257
+ </span>
3258
+ </a>
3259
+
3260
+ </li>
3261
+
3262
+ </ul>
3263
+ </nav>
3264
+
3265
+ </li>
3266
+
3267
+ <li class="md-nav__item">
3268
+ <a href="#next-steps" class="md-nav__link">
3269
+ <span class="md-ellipsis">
3270
+
3271
+ Next Steps
3272
+
3273
+ </span>
3274
+ </a>
3275
+
3276
+ </li>
3277
+
3278
+ </ul>
3279
+
3280
+ </nav>
3281
+ </div>
3282
+ </div>
3283
+ </div>
3284
+
3285
+
3286
+
3287
+ <div class="md-content" data-md-component="content">
3288
+
3289
+
3290
+
3291
+
3292
+
3293
+
3294
+
3295
+ <article class="md-content__inner md-typeset">
3296
+
3297
+
3298
+
3299
+
3300
+
3301
+ <a href="https://github.com/madbomber/robot_lab/edit/main/docs/guides/rails-integration.md" title="Edit this page" class="md-content__button md-icon" rel="edit">
3302
+
3303
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M10 20H6V4h7v5h5v3.1l2-2V8l-6-6H6c-1.1 0-2 .9-2 2v16c0 1.1.9 2 2 2h4zm10.2-7c.1 0 .3.1.4.2l1.3 1.3c.2.2.2.6 0 .8l-1 1-2.1-2.1 1-1c.1-.1.2-.2.4-.2m0 3.9L14.1 23H12v-2.1l6.1-6.1z"/></svg>
3304
+ </a>
3305
+
3306
+
3307
+
3308
+
3309
+
3310
+ <a href="https://github.com/madbomber/robot_lab/raw/main/docs/guides/rails-integration.md" title="View source of this page" class="md-content__button md-icon">
3311
+
3312
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M17 18c.56 0 1 .44 1 1s-.44 1-1 1-1-.44-1-1 .44-1 1-1m0-3c-2.73 0-5.06 1.66-6 4 .94 2.34 3.27 4 6 4s5.06-1.66 6-4c-.94-2.34-3.27-4-6-4m0 6.5a2.5 2.5 0 0 1-2.5-2.5 2.5 2.5 0 0 1 2.5-2.5 2.5 2.5 0 0 1 2.5 2.5 2.5 2.5 0 0 1-2.5 2.5M9.27 20H6V4h7v5h5v4.07c.7.08 1.36.25 2 .49V8l-6-6H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h4.5a8.2 8.2 0 0 1-1.23-2"/></svg>
3313
+ </a>
3314
+
3315
+
3316
+
3317
+ <h1 id="rails-integration">Rails Integration<a class="headerlink" href="#rails-integration" title="Permanent link">&para;</a></h1>
3318
+ <p>RobotLab integrates seamlessly with Ruby on Rails applications.</p>
3319
+ <h2 id="installation">Installation<a class="headerlink" href="#installation" title="Permanent link">&para;</a></h2>
3320
+ <h3 id="generate-files">Generate Files<a class="headerlink" href="#generate-files" title="Permanent link">&para;</a></h3>
3321
+ <div class="language-bash highlight"><pre><span></span><code><span id="__span-0-1"><a id="__codelineno-0-1" name="__codelineno-0-1" href="#__codelineno-0-1"></a>rails<span class="w"> </span>generate<span class="w"> </span>robot_lab:install
3322
+ </span></code></pre></div>
3323
+ <p>This creates:</p>
3324
+ <div class="language-text highlight"><pre><span></span><code><span id="__span-1-1"><a id="__codelineno-1-1" name="__codelineno-1-1" href="#__codelineno-1-1"></a>config/initializers/robot_lab.rb # Logger setup
3325
+ </span><span id="__span-1-2"><a id="__codelineno-1-2" name="__codelineno-1-2" href="#__codelineno-1-2"></a>db/migrate/*_create_robot_lab_tables.rb # Database tables
3326
+ </span><span id="__span-1-3"><a id="__codelineno-1-3" name="__codelineno-1-3" href="#__codelineno-1-3"></a>app/models/robot_lab_thread.rb # Thread model
3327
+ </span><span id="__span-1-4"><a id="__codelineno-1-4" name="__codelineno-1-4" href="#__codelineno-1-4"></a>app/models/robot_lab_result.rb # Result model
3328
+ </span><span id="__span-1-5"><a id="__codelineno-1-5" name="__codelineno-1-5" href="#__codelineno-1-5"></a>app/jobs/robot_run_job.rb # Background job for robot execution
3329
+ </span><span id="__span-1-6"><a id="__codelineno-1-6" name="__codelineno-1-6" href="#__codelineno-1-6"></a>app/robots/ # Directory for robots
3330
+ </span><span id="__span-1-7"><a id="__codelineno-1-7" name="__codelineno-1-7" href="#__codelineno-1-7"></a>app/tools/ # Directory for tools
3331
+ </span></code></pre></div>
3332
+ <p>Options:</p>
3333
+ <ul>
3334
+ <li><code>--skip-migration</code> — Skip database migration generation</li>
3335
+ <li><code>--skip-job</code> — Skip background job generation</li>
3336
+ </ul>
3337
+ <h3 id="run-migrations">Run Migrations<a class="headerlink" href="#run-migrations" title="Permanent link">&para;</a></h3>
3338
+ <div class="language-bash highlight"><pre><span></span><code><span id="__span-2-1"><a id="__codelineno-2-1" name="__codelineno-2-1" href="#__codelineno-2-1"></a>rails<span class="w"> </span>db:migrate
3339
+ </span></code></pre></div>
3340
+ <h2 id="configuration">Configuration<a class="headerlink" href="#configuration" title="Permanent link">&para;</a></h2>
3341
+ <p>RobotLab uses <a href="https://github.com/madbomber/myway_config">MywayConfig</a> for configuration. There is no <code>RobotLab.configure</code> block. Instead, settings are loaded from YAML files and environment variables in the following priority order:</p>
3342
+ <ol>
3343
+ <li><strong>Bundled defaults</strong> (<code>lib/robot_lab/config/defaults.yml</code>)</li>
3344
+ <li><strong>Environment-specific overrides</strong> (development, test, production sections)</li>
3345
+ <li><strong>XDG user config</strong> (<code>~/.config/robot_lab/config.yml</code>)</li>
3346
+ <li><strong>Project config</strong> (<code>./config/robot_lab.yml</code>)</li>
3347
+ <li><strong>Environment variables</strong> (<code>ROBOT_LAB_*</code> prefix)</li>
3348
+ </ol>
3349
+ <h3 id="project-config-file">Project Config File<a class="headerlink" href="#project-config-file" title="Permanent link">&para;</a></h3>
3350
+ <div class="language-yaml highlight"><span class="filename">config/robot_lab.yml</span><pre><span></span><code><span id="__span-3-1"><a id="__codelineno-3-1" name="__codelineno-3-1" href="#__codelineno-3-1"></a><span class="nt">defaults</span><span class="p">:</span>
3351
+ </span><span id="__span-3-2"><a id="__codelineno-3-2" name="__codelineno-3-2" href="#__codelineno-3-2"></a><span class="w"> </span><span class="nt">ruby_llm</span><span class="p">:</span>
3352
+ </span><span id="__span-3-3"><a id="__codelineno-3-3" name="__codelineno-3-3" href="#__codelineno-3-3"></a><span class="w"> </span><span class="nt">anthropic_api_key</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">&lt;%= ENV[&#39;ANTHROPIC_API_KEY&#39;] %&gt;</span>
3353
+ </span><span id="__span-3-4"><a id="__codelineno-3-4" name="__codelineno-3-4" href="#__codelineno-3-4"></a><span class="w"> </span><span class="nt">openai_api_key</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">&lt;%= ENV[&#39;OPENAI_API_KEY&#39;] %&gt;</span>
3354
+ </span><span id="__span-3-5"><a id="__codelineno-3-5" name="__codelineno-3-5" href="#__codelineno-3-5"></a><span class="w"> </span><span class="nt">model</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">claude-sonnet-4</span>
3355
+ </span><span id="__span-3-6"><a id="__codelineno-3-6" name="__codelineno-3-6" href="#__codelineno-3-6"></a><span class="w"> </span><span class="nt">request_timeout</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">180</span>
3356
+ </span><span id="__span-3-7"><a id="__codelineno-3-7" name="__codelineno-3-7" href="#__codelineno-3-7"></a>
3357
+ </span><span id="__span-3-8"><a id="__codelineno-3-8" name="__codelineno-3-8" href="#__codelineno-3-8"></a><span class="w"> </span><span class="c1"># Template path auto-detected as app/prompts in Rails</span>
3358
+ </span><span id="__span-3-9"><a id="__codelineno-3-9" name="__codelineno-3-9" href="#__codelineno-3-9"></a><span class="w"> </span><span class="c1"># template_path: app/prompts</span>
3359
+ </span><span id="__span-3-10"><a id="__codelineno-3-10" name="__codelineno-3-10" href="#__codelineno-3-10"></a>
3360
+ </span><span id="__span-3-11"><a id="__codelineno-3-11" name="__codelineno-3-11" href="#__codelineno-3-11"></a><span class="nt">development</span><span class="p">:</span>
3361
+ </span><span id="__span-3-12"><a id="__codelineno-3-12" name="__codelineno-3-12" href="#__codelineno-3-12"></a><span class="w"> </span><span class="nt">ruby_llm</span><span class="p">:</span>
3362
+ </span><span id="__span-3-13"><a id="__codelineno-3-13" name="__codelineno-3-13" href="#__codelineno-3-13"></a><span class="w"> </span><span class="nt">model</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">claude-haiku-3</span>
3363
+ </span><span id="__span-3-14"><a id="__codelineno-3-14" name="__codelineno-3-14" href="#__codelineno-3-14"></a><span class="w"> </span><span class="nt">log_level</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">:debug</span>
3364
+ </span><span id="__span-3-15"><a id="__codelineno-3-15" name="__codelineno-3-15" href="#__codelineno-3-15"></a>
3365
+ </span><span id="__span-3-16"><a id="__codelineno-3-16" name="__codelineno-3-16" href="#__codelineno-3-16"></a><span class="nt">test</span><span class="p">:</span>
3366
+ </span><span id="__span-3-17"><a id="__codelineno-3-17" name="__codelineno-3-17" href="#__codelineno-3-17"></a><span class="w"> </span><span class="nt">streaming_enabled</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">false</span>
3367
+ </span><span id="__span-3-18"><a id="__codelineno-3-18" name="__codelineno-3-18" href="#__codelineno-3-18"></a><span class="w"> </span><span class="nt">ruby_llm</span><span class="p">:</span>
3368
+ </span><span id="__span-3-19"><a id="__codelineno-3-19" name="__codelineno-3-19" href="#__codelineno-3-19"></a><span class="w"> </span><span class="nt">model</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">claude-3-haiku-20240307</span>
3369
+ </span><span id="__span-3-20"><a id="__codelineno-3-20" name="__codelineno-3-20" href="#__codelineno-3-20"></a><span class="w"> </span><span class="nt">request_timeout</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">30</span>
3370
+ </span><span id="__span-3-21"><a id="__codelineno-3-21" name="__codelineno-3-21" href="#__codelineno-3-21"></a>
3371
+ </span><span id="__span-3-22"><a id="__codelineno-3-22" name="__codelineno-3-22" href="#__codelineno-3-22"></a><span class="nt">production</span><span class="p">:</span>
3372
+ </span><span id="__span-3-23"><a id="__codelineno-3-23" name="__codelineno-3-23" href="#__codelineno-3-23"></a><span class="w"> </span><span class="nt">ruby_llm</span><span class="p">:</span>
3373
+ </span><span id="__span-3-24"><a id="__codelineno-3-24" name="__codelineno-3-24" href="#__codelineno-3-24"></a><span class="w"> </span><span class="nt">request_timeout</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">180</span>
3374
+ </span><span id="__span-3-25"><a id="__codelineno-3-25" name="__codelineno-3-25" href="#__codelineno-3-25"></a><span class="w"> </span><span class="nt">max_retries</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">5</span>
3375
+ </span></code></pre></div>
3376
+ <h3 id="environment-variables">Environment Variables<a class="headerlink" href="#environment-variables" title="Permanent link">&para;</a></h3>
3377
+ <p>Environment variables use the <code>ROBOT_LAB_</code> prefix with double underscores for nested keys:</p>
3378
+ <div class="language-bash highlight"><pre><span></span><code><span id="__span-4-1"><a id="__codelineno-4-1" name="__codelineno-4-1" href="#__codelineno-4-1"></a><span class="nv">ROBOT_LAB_RUBY_LLM__ANTHROPIC_API_KEY</span><span class="o">=</span>sk-ant-...
3379
+ </span><span id="__span-4-2"><a id="__codelineno-4-2" name="__codelineno-4-2" href="#__codelineno-4-2"></a><span class="nv">ROBOT_LAB_RUBY_LLM__MODEL</span><span class="o">=</span>claude-sonnet-4
3380
+ </span><span id="__span-4-3"><a id="__codelineno-4-3" name="__codelineno-4-3" href="#__codelineno-4-3"></a><span class="nv">ROBOT_LAB_RUBY_LLM__REQUEST_TIMEOUT</span><span class="o">=</span><span class="m">180</span>
3381
+ </span></code></pre></div>
3382
+ <p>RobotLab also falls back to standard provider environment variables (e.g. <code>ANTHROPIC_API_KEY</code>, <code>OPENAI_API_KEY</code>) when the prefixed versions are not set.</p>
3383
+ <h3 id="initializer-logger-only">Initializer (Logger Only)<a class="headerlink" href="#initializer-logger-only" title="Permanent link">&para;</a></h3>
3384
+ <p>The only runtime-writable config attribute is the logger. The generated initializer sets it to the Rails logger:</p>
3385
+ <div class="language-ruby highlight"><span class="filename">config/initializers/robot_lab.rb</span><pre><span></span><code><span id="__span-5-1"><a id="__codelineno-5-1" name="__codelineno-5-1" href="#__codelineno-5-1"></a><span class="c1"># frozen_string_literal: true</span>
3386
+ </span><span id="__span-5-2"><a id="__codelineno-5-2" name="__codelineno-5-2" href="#__codelineno-5-2"></a>
3387
+ </span><span id="__span-5-3"><a id="__codelineno-5-3" name="__codelineno-5-3" href="#__codelineno-5-3"></a><span class="c1"># Set the RobotLab logger to use Rails.logger</span>
3388
+ </span><span id="__span-5-4"><a id="__codelineno-5-4" name="__codelineno-5-4" href="#__codelineno-5-4"></a><span class="no">RobotLab</span><span class="o">.</span><span class="n">config</span><span class="o">.</span><span class="n">logger</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">Rails</span><span class="o">.</span><span class="n">logger</span>
3389
+ </span></code></pre></div>
3390
+ <h3 id="accessing-configuration">Accessing Configuration<a class="headerlink" href="#accessing-configuration" title="Permanent link">&para;</a></h3>
3391
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-6-1"><a id="__codelineno-6-1" name="__codelineno-6-1" href="#__codelineno-6-1"></a><span class="c1"># Read configuration values</span>
3392
+ </span><span id="__span-6-2"><a id="__codelineno-6-2" name="__codelineno-6-2" href="#__codelineno-6-2"></a><span class="no">RobotLab</span><span class="o">.</span><span class="n">config</span><span class="o">.</span><span class="n">ruby_llm</span><span class="o">.</span><span class="n">model</span><span class="w"> </span><span class="c1">#=&gt; &quot;claude-sonnet-4&quot;</span>
3393
+ </span><span id="__span-6-3"><a id="__codelineno-6-3" name="__codelineno-6-3" href="#__codelineno-6-3"></a><span class="no">RobotLab</span><span class="o">.</span><span class="n">config</span><span class="o">.</span><span class="n">ruby_llm</span><span class="o">.</span><span class="n">anthropic_api_key</span><span class="w"> </span><span class="c1">#=&gt; &quot;sk-ant-...&quot;</span>
3394
+ </span><span id="__span-6-4"><a id="__codelineno-6-4" name="__codelineno-6-4" href="#__codelineno-6-4"></a><span class="no">RobotLab</span><span class="o">.</span><span class="n">config</span><span class="o">.</span><span class="n">ruby_llm</span><span class="o">.</span><span class="n">request_timeout</span><span class="w"> </span><span class="c1">#=&gt; 120</span>
3395
+ </span><span id="__span-6-5"><a id="__codelineno-6-5" name="__codelineno-6-5" href="#__codelineno-6-5"></a><span class="no">RobotLab</span><span class="o">.</span><span class="n">config</span><span class="o">.</span><span class="n">streaming_enabled</span><span class="w"> </span><span class="c1">#=&gt; true</span>
3396
+ </span></code></pre></div>
3397
+ <h2 id="creating-robots">Creating Robots<a class="headerlink" href="#creating-robots" title="Permanent link">&para;</a></h2>
3398
+ <h3 id="robot-generator">Robot Generator<a class="headerlink" href="#robot-generator" title="Permanent link">&para;</a></h3>
3399
+ <div class="language-bash highlight"><pre><span></span><code><span id="__span-7-1"><a id="__codelineno-7-1" name="__codelineno-7-1" href="#__codelineno-7-1"></a>rails<span class="w"> </span>generate<span class="w"> </span>robot_lab:robot<span class="w"> </span>Support
3400
+ </span><span id="__span-7-2"><a id="__codelineno-7-2" name="__codelineno-7-2" href="#__codelineno-7-2"></a>rails<span class="w"> </span>generate<span class="w"> </span>robot_lab:robot<span class="w"> </span>Billing<span class="w"> </span>--description<span class="o">=</span><span class="s2">&quot;Handles billing inquiries&quot;</span>
3401
+ </span><span id="__span-7-3"><a id="__codelineno-7-3" name="__codelineno-7-3" href="#__codelineno-7-3"></a>rails<span class="w"> </span>generate<span class="w"> </span>robot_lab:robot<span class="w"> </span>Router<span class="w"> </span>--routing
3402
+ </span></code></pre></div>
3403
+ <h3 id="robot-class">Robot Class<a class="headerlink" href="#robot-class" title="Permanent link">&para;</a></h3>
3404
+ <p>Robots are plain Ruby classes with a <code>.build</code> factory method that calls <code>RobotLab.build</code> with keyword arguments:</p>
3405
+ <div class="language-ruby highlight"><span class="filename">app/robots/support_robot.rb</span><pre><span></span><code><span id="__span-8-1"><a id="__codelineno-8-1" name="__codelineno-8-1" href="#__codelineno-8-1"></a><span class="c1"># frozen_string_literal: true</span>
3406
+ </span><span id="__span-8-2"><a id="__codelineno-8-2" name="__codelineno-8-2" href="#__codelineno-8-2"></a>
3407
+ </span><span id="__span-8-3"><a id="__codelineno-8-3" name="__codelineno-8-3" href="#__codelineno-8-3"></a><span class="k">class</span><span class="w"> </span><span class="nc">SupportRobot</span>
3408
+ </span><span id="__span-8-4"><a id="__codelineno-8-4" name="__codelineno-8-4" href="#__codelineno-8-4"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nc">self</span><span class="o">.</span><span class="nf">build</span><span class="p">(</span><span class="o">**</span><span class="n">options</span><span class="p">)</span>
3409
+ </span><span id="__span-8-5"><a id="__codelineno-8-5" name="__codelineno-8-5" href="#__codelineno-8-5"></a><span class="w"> </span><span class="no">RobotLab</span><span class="o">.</span><span class="n">build</span><span class="p">(</span>
3410
+ </span><span id="__span-8-6"><a id="__codelineno-8-6" name="__codelineno-8-6" href="#__codelineno-8-6"></a><span class="w"> </span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;support&quot;</span><span class="p">,</span>
3411
+ </span><span id="__span-8-7"><a id="__codelineno-8-7" name="__codelineno-8-7" href="#__codelineno-8-7"></a><span class="w"> </span><span class="ss">description</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;Handles customer support inquiries&quot;</span><span class="p">,</span>
3412
+ </span><span id="__span-8-8"><a id="__codelineno-8-8" name="__codelineno-8-8" href="#__codelineno-8-8"></a><span class="w"> </span><span class="ss">system_prompt</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;You are a helpful support assistant.&quot;</span><span class="p">,</span>
3413
+ </span><span id="__span-8-9"><a id="__codelineno-8-9" name="__codelineno-8-9" href="#__codelineno-8-9"></a><span class="w"> </span><span class="ss">model</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;claude-sonnet-4&quot;</span><span class="p">,</span>
3414
+ </span><span id="__span-8-10"><a id="__codelineno-8-10" name="__codelineno-8-10" href="#__codelineno-8-10"></a><span class="w"> </span><span class="ss">local_tools</span><span class="p">:</span><span class="w"> </span><span class="o">[</span><span class="no">OrderLookup</span><span class="o">]</span><span class="p">,</span>
3415
+ </span><span id="__span-8-11"><a id="__codelineno-8-11" name="__codelineno-8-11" href="#__codelineno-8-11"></a><span class="w"> </span><span class="o">**</span><span class="n">options</span>
3416
+ </span><span id="__span-8-12"><a id="__codelineno-8-12" name="__codelineno-8-12" href="#__codelineno-8-12"></a><span class="w"> </span><span class="p">)</span>
3417
+ </span><span id="__span-8-13"><a id="__codelineno-8-13" name="__codelineno-8-13" href="#__codelineno-8-13"></a><span class="w"> </span><span class="k">end</span>
3418
+ </span><span id="__span-8-14"><a id="__codelineno-8-14" name="__codelineno-8-14" href="#__codelineno-8-14"></a><span class="k">end</span>
3419
+ </span></code></pre></div>
3420
+ <h3 id="routing-robot-class">Routing Robot Class<a class="headerlink" href="#routing-robot-class" title="Permanent link">&para;</a></h3>
3421
+ <p>A routing robot classifies requests and activates optional tasks in a Network. It subclasses <code>RobotLab::Robot</code> and overrides <code>call(result)</code>:</p>
3422
+ <div class="language-ruby highlight"><span class="filename">app/robots/classifier_robot.rb</span><pre><span></span><code><span id="__span-9-1"><a id="__codelineno-9-1" name="__codelineno-9-1" href="#__codelineno-9-1"></a><span class="c1"># frozen_string_literal: true</span>
3423
+ </span><span id="__span-9-2"><a id="__codelineno-9-2" name="__codelineno-9-2" href="#__codelineno-9-2"></a>
3424
+ </span><span id="__span-9-3"><a id="__codelineno-9-3" name="__codelineno-9-3" href="#__codelineno-9-3"></a><span class="k">class</span><span class="w"> </span><span class="nc">ClassifierRobot</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Robot</span>
3425
+ </span><span id="__span-9-4"><a id="__codelineno-9-4" name="__codelineno-9-4" href="#__codelineno-9-4"></a><span class="w"> </span><span class="no">SYSTEM_PROMPT</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="o">&lt;&lt;~</span><span class="dl">PROMPT</span>
3426
+ </span><span id="__span-9-5"><a id="__codelineno-9-5" name="__codelineno-9-5" href="#__codelineno-9-5"></a><span class="sh"> You are a routing robot that classifies user requests.</span>
3427
+ </span><span id="__span-9-6"><a id="__codelineno-9-6" name="__codelineno-9-6" href="#__codelineno-9-6"></a>
3428
+ </span><span id="__span-9-7"><a id="__codelineno-9-7" name="__codelineno-9-7" href="#__codelineno-9-7"></a><span class="sh"> Analyze the user&#39;s request and respond with ONLY the category name.</span>
3429
+ </span><span id="__span-9-8"><a id="__codelineno-9-8" name="__codelineno-9-8" href="#__codelineno-9-8"></a><span class="sh"> Valid categories: billing, technical, general</span>
3430
+ </span><span id="__span-9-9"><a id="__codelineno-9-9" name="__codelineno-9-9" href="#__codelineno-9-9"></a><span class="dl"> PROMPT</span>
3431
+ </span><span id="__span-9-10"><a id="__codelineno-9-10" name="__codelineno-9-10" href="#__codelineno-9-10"></a>
3432
+ </span><span id="__span-9-11"><a id="__codelineno-9-11" name="__codelineno-9-11" href="#__codelineno-9-11"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nc">self</span><span class="o">.</span><span class="nf">build</span><span class="p">(</span><span class="o">**</span><span class="n">options</span><span class="p">)</span>
3433
+ </span><span id="__span-9-12"><a id="__codelineno-9-12" name="__codelineno-9-12" href="#__codelineno-9-12"></a><span class="w"> </span><span class="kp">new</span><span class="p">(</span>
3434
+ </span><span id="__span-9-13"><a id="__codelineno-9-13" name="__codelineno-9-13" href="#__codelineno-9-13"></a><span class="w"> </span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;classifier&quot;</span><span class="p">,</span>
3435
+ </span><span id="__span-9-14"><a id="__codelineno-9-14" name="__codelineno-9-14" href="#__codelineno-9-14"></a><span class="w"> </span><span class="ss">description</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;Classifies support requests&quot;</span><span class="p">,</span>
3436
+ </span><span id="__span-9-15"><a id="__codelineno-9-15" name="__codelineno-9-15" href="#__codelineno-9-15"></a><span class="w"> </span><span class="ss">system_prompt</span><span class="p">:</span><span class="w"> </span><span class="no">SYSTEM_PROMPT</span><span class="p">,</span>
3437
+ </span><span id="__span-9-16"><a id="__codelineno-9-16" name="__codelineno-9-16" href="#__codelineno-9-16"></a><span class="w"> </span><span class="o">**</span><span class="n">options</span>
3438
+ </span><span id="__span-9-17"><a id="__codelineno-9-17" name="__codelineno-9-17" href="#__codelineno-9-17"></a><span class="w"> </span><span class="p">)</span>
3439
+ </span><span id="__span-9-18"><a id="__codelineno-9-18" name="__codelineno-9-18" href="#__codelineno-9-18"></a><span class="w"> </span><span class="k">end</span>
3440
+ </span><span id="__span-9-19"><a id="__codelineno-9-19" name="__codelineno-9-19" href="#__codelineno-9-19"></a>
3441
+ </span><span id="__span-9-20"><a id="__codelineno-9-20" name="__codelineno-9-20" href="#__codelineno-9-20"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">call</span><span class="p">(</span><span class="n">result</span><span class="p">)</span>
3442
+ </span><span id="__span-9-21"><a id="__codelineno-9-21" name="__codelineno-9-21" href="#__codelineno-9-21"></a><span class="w"> </span><span class="n">context</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">extract_run_context</span><span class="p">(</span><span class="n">result</span><span class="p">)</span>
3443
+ </span><span id="__span-9-22"><a id="__codelineno-9-22" name="__codelineno-9-22" href="#__codelineno-9-22"></a><span class="w"> </span><span class="n">message</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">context</span><span class="o">.</span><span class="n">delete</span><span class="p">(</span><span class="ss">:message</span><span class="p">)</span>
3444
+ </span><span id="__span-9-23"><a id="__codelineno-9-23" name="__codelineno-9-23" href="#__codelineno-9-23"></a>
3445
+ </span><span id="__span-9-24"><a id="__codelineno-9-24" name="__codelineno-9-24" href="#__codelineno-9-24"></a><span class="w"> </span><span class="n">robot_result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">run</span><span class="p">(</span><span class="n">message</span><span class="p">,</span><span class="w"> </span><span class="o">**</span><span class="n">context</span><span class="p">)</span>
3446
+ </span><span id="__span-9-25"><a id="__codelineno-9-25" name="__codelineno-9-25" href="#__codelineno-9-25"></a>
3447
+ </span><span id="__span-9-26"><a id="__codelineno-9-26" name="__codelineno-9-26" href="#__codelineno-9-26"></a><span class="w"> </span><span class="n">new_result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">result</span>
3448
+ </span><span id="__span-9-27"><a id="__codelineno-9-27" name="__codelineno-9-27" href="#__codelineno-9-27"></a><span class="w"> </span><span class="o">.</span><span class="n">with_context</span><span class="p">(</span><span class="vi">@name</span><span class="o">.</span><span class="n">to_sym</span><span class="p">,</span><span class="w"> </span><span class="n">robot_result</span><span class="p">)</span>
3449
+ </span><span id="__span-9-28"><a id="__codelineno-9-28" name="__codelineno-9-28" href="#__codelineno-9-28"></a><span class="w"> </span><span class="o">.</span><span class="n">continue</span><span class="p">(</span><span class="n">robot_result</span><span class="p">)</span>
3450
+ </span><span id="__span-9-29"><a id="__codelineno-9-29" name="__codelineno-9-29" href="#__codelineno-9-29"></a>
3451
+ </span><span id="__span-9-30"><a id="__codelineno-9-30" name="__codelineno-9-30" href="#__codelineno-9-30"></a><span class="w"> </span><span class="n">category</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot_result</span><span class="o">.</span><span class="n">last_text_content</span><span class="o">.</span><span class="n">to_s</span><span class="o">.</span><span class="n">strip</span><span class="o">.</span><span class="n">downcase</span>
3452
+ </span><span id="__span-9-31"><a id="__codelineno-9-31" name="__codelineno-9-31" href="#__codelineno-9-31"></a>
3453
+ </span><span id="__span-9-32"><a id="__codelineno-9-32" name="__codelineno-9-32" href="#__codelineno-9-32"></a><span class="w"> </span><span class="k">case</span><span class="w"> </span><span class="n">category</span>
3454
+ </span><span id="__span-9-33"><a id="__codelineno-9-33" name="__codelineno-9-33" href="#__codelineno-9-33"></a><span class="w"> </span><span class="k">when</span><span class="w"> </span><span class="sr">/billing/</span><span class="w"> </span><span class="k">then</span><span class="w"> </span><span class="n">new_result</span><span class="o">.</span><span class="n">activate</span><span class="p">(</span><span class="ss">:billing</span><span class="p">)</span>
3455
+ </span><span id="__span-9-34"><a id="__codelineno-9-34" name="__codelineno-9-34" href="#__codelineno-9-34"></a><span class="w"> </span><span class="k">when</span><span class="w"> </span><span class="sr">/technical/</span><span class="w"> </span><span class="k">then</span><span class="w"> </span><span class="n">new_result</span><span class="o">.</span><span class="n">activate</span><span class="p">(</span><span class="ss">:technical</span><span class="p">)</span>
3456
+ </span><span id="__span-9-35"><a id="__codelineno-9-35" name="__codelineno-9-35" href="#__codelineno-9-35"></a><span class="w"> </span><span class="k">else</span><span class="w"> </span><span class="n">new_result</span><span class="o">.</span><span class="n">activate</span><span class="p">(</span><span class="ss">:general</span><span class="p">)</span>
3457
+ </span><span id="__span-9-36"><a id="__codelineno-9-36" name="__codelineno-9-36" href="#__codelineno-9-36"></a><span class="w"> </span><span class="k">end</span>
3458
+ </span><span id="__span-9-37"><a id="__codelineno-9-37" name="__codelineno-9-37" href="#__codelineno-9-37"></a><span class="w"> </span><span class="k">end</span>
3459
+ </span><span id="__span-9-38"><a id="__codelineno-9-38" name="__codelineno-9-38" href="#__codelineno-9-38"></a><span class="k">end</span>
3460
+ </span></code></pre></div>
3461
+ <p>Use the routing robot as the first task in a network:</p>
3462
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-10-1"><a id="__codelineno-10-1" name="__codelineno-10-1" href="#__codelineno-10-1"></a><span class="n">classifier</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">ClassifierRobot</span><span class="o">.</span><span class="n">build</span>
3463
+ </span><span id="__span-10-2"><a id="__codelineno-10-2" name="__codelineno-10-2" href="#__codelineno-10-2"></a><span class="n">billing</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">BillingRobot</span><span class="o">.</span><span class="n">build</span>
3464
+ </span><span id="__span-10-3"><a id="__codelineno-10-3" name="__codelineno-10-3" href="#__codelineno-10-3"></a><span class="n">technical</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">TechnicalRobot</span><span class="o">.</span><span class="n">build</span>
3465
+ </span><span id="__span-10-4"><a id="__codelineno-10-4" name="__codelineno-10-4" href="#__codelineno-10-4"></a>
3466
+ </span><span id="__span-10-5"><a id="__codelineno-10-5" name="__codelineno-10-5" href="#__codelineno-10-5"></a><span class="n">network</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">.</span><span class="n">create_network</span><span class="p">(</span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;support&quot;</span><span class="p">)</span><span class="w"> </span><span class="k">do</span>
3467
+ </span><span id="__span-10-6"><a id="__codelineno-10-6" name="__codelineno-10-6" href="#__codelineno-10-6"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:classifier</span><span class="p">,</span><span class="w"> </span><span class="n">classifier</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:none</span>
3468
+ </span><span id="__span-10-7"><a id="__codelineno-10-7" name="__codelineno-10-7" href="#__codelineno-10-7"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:billing</span><span class="p">,</span><span class="w"> </span><span class="n">billing</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:optional</span>
3469
+ </span><span id="__span-10-8"><a id="__codelineno-10-8" name="__codelineno-10-8" href="#__codelineno-10-8"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:technical</span><span class="p">,</span><span class="w"> </span><span class="n">technical</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:optional</span>
3470
+ </span><span id="__span-10-9"><a id="__codelineno-10-9" name="__codelineno-10-9" href="#__codelineno-10-9"></a><span class="k">end</span>
3471
+ </span><span id="__span-10-10"><a id="__codelineno-10-10" name="__codelineno-10-10" href="#__codelineno-10-10"></a>
3472
+ </span><span id="__span-10-11"><a id="__codelineno-10-11" name="__codelineno-10-11" href="#__codelineno-10-11"></a><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">network</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="ss">message</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;I was charged twice&quot;</span><span class="p">)</span>
3473
+ </span></code></pre></div>
3474
+ <h3 id="custom-tool">Custom Tool<a class="headerlink" href="#custom-tool" title="Permanent link">&para;</a></h3>
3475
+ <p>Tools subclass <code>RobotLab::Tool</code> (which extends <code>RubyLLM::Tool</code>):</p>
3476
+ <div class="language-ruby highlight"><span class="filename">app/tools/order_lookup.rb</span><pre><span></span><code><span id="__span-11-1"><a id="__codelineno-11-1" name="__codelineno-11-1" href="#__codelineno-11-1"></a><span class="c1"># frozen_string_literal: true</span>
3477
+ </span><span id="__span-11-2"><a id="__codelineno-11-2" name="__codelineno-11-2" href="#__codelineno-11-2"></a>
3478
+ </span><span id="__span-11-3"><a id="__codelineno-11-3" name="__codelineno-11-3" href="#__codelineno-11-3"></a><span class="k">class</span><span class="w"> </span><span class="nc">OrderLookup</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Tool</span>
3479
+ </span><span id="__span-11-4"><a id="__codelineno-11-4" name="__codelineno-11-4" href="#__codelineno-11-4"></a><span class="w"> </span><span class="n">description</span><span class="w"> </span><span class="s2">&quot;Look up an order by ID&quot;</span>
3480
+ </span><span id="__span-11-5"><a id="__codelineno-11-5" name="__codelineno-11-5" href="#__codelineno-11-5"></a><span class="w"> </span><span class="n">param</span><span class="w"> </span><span class="ss">:order_id</span><span class="p">,</span><span class="w"> </span><span class="ss">type</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;string&quot;</span><span class="p">,</span><span class="w"> </span><span class="ss">desc</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;The order ID to look up&quot;</span>
3481
+ </span><span id="__span-11-6"><a id="__codelineno-11-6" name="__codelineno-11-6" href="#__codelineno-11-6"></a>
3482
+ </span><span id="__span-11-7"><a id="__codelineno-11-7" name="__codelineno-11-7" href="#__codelineno-11-7"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">execute</span><span class="p">(</span><span class="ss">order_id</span><span class="p">:)</span>
3483
+ </span><span id="__span-11-8"><a id="__codelineno-11-8" name="__codelineno-11-8" href="#__codelineno-11-8"></a><span class="w"> </span><span class="n">order</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">Order</span><span class="o">.</span><span class="n">find_by</span><span class="p">(</span><span class="nb">id</span><span class="p">:</span><span class="w"> </span><span class="n">order_id</span><span class="p">)</span>
3484
+ </span><span id="__span-11-9"><a id="__codelineno-11-9" name="__codelineno-11-9" href="#__codelineno-11-9"></a><span class="w"> </span><span class="k">return</span><span class="w"> </span><span class="s2">&quot;Order not found&quot;</span><span class="w"> </span><span class="k">unless</span><span class="w"> </span><span class="n">order</span>
3485
+ </span><span id="__span-11-10"><a id="__codelineno-11-10" name="__codelineno-11-10" href="#__codelineno-11-10"></a>
3486
+ </span><span id="__span-11-11"><a id="__codelineno-11-11" name="__codelineno-11-11" href="#__codelineno-11-11"></a><span class="w"> </span><span class="p">{</span>
3487
+ </span><span id="__span-11-12"><a id="__codelineno-11-12" name="__codelineno-11-12" href="#__codelineno-11-12"></a><span class="w"> </span><span class="nb">id</span><span class="p">:</span><span class="w"> </span><span class="n">order</span><span class="o">.</span><span class="n">id</span><span class="p">,</span>
3488
+ </span><span id="__span-11-13"><a id="__codelineno-11-13" name="__codelineno-11-13" href="#__codelineno-11-13"></a><span class="w"> </span><span class="ss">status</span><span class="p">:</span><span class="w"> </span><span class="n">order</span><span class="o">.</span><span class="n">status</span><span class="p">,</span>
3489
+ </span><span id="__span-11-14"><a id="__codelineno-11-14" name="__codelineno-11-14" href="#__codelineno-11-14"></a><span class="w"> </span><span class="ss">total</span><span class="p">:</span><span class="w"> </span><span class="n">order</span><span class="o">.</span><span class="n">total</span><span class="o">.</span><span class="n">to_s</span><span class="p">,</span>
3490
+ </span><span id="__span-11-15"><a id="__codelineno-11-15" name="__codelineno-11-15" href="#__codelineno-11-15"></a><span class="w"> </span><span class="ss">created_at</span><span class="p">:</span><span class="w"> </span><span class="n">order</span><span class="o">.</span><span class="n">created_at</span><span class="o">.</span><span class="n">iso8601</span>
3491
+ </span><span id="__span-11-16"><a id="__codelineno-11-16" name="__codelineno-11-16" href="#__codelineno-11-16"></a><span class="w"> </span><span class="p">}</span><span class="o">.</span><span class="n">to_json</span>
3492
+ </span><span id="__span-11-17"><a id="__codelineno-11-17" name="__codelineno-11-17" href="#__codelineno-11-17"></a><span class="w"> </span><span class="k">end</span>
3493
+ </span><span id="__span-11-18"><a id="__codelineno-11-18" name="__codelineno-11-18" href="#__codelineno-11-18"></a><span class="k">end</span>
3494
+ </span></code></pre></div>
3495
+ <h3 id="using-in-controllers">Using in Controllers<a class="headerlink" href="#using-in-controllers" title="Permanent link">&para;</a></h3>
3496
+ <div class="language-ruby highlight"><span class="filename">app/controllers/chat_controller.rb</span><pre><span></span><code><span id="__span-12-1"><a id="__codelineno-12-1" name="__codelineno-12-1" href="#__codelineno-12-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatController</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationController</span>
3497
+ </span><span id="__span-12-2"><a id="__codelineno-12-2" name="__codelineno-12-2" href="#__codelineno-12-2"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">create</span>
3498
+ </span><span id="__span-12-3"><a id="__codelineno-12-3" name="__codelineno-12-3" href="#__codelineno-12-3"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3499
+ </span><span id="__span-12-4"><a id="__codelineno-12-4" name="__codelineno-12-4" href="#__codelineno-12-4"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="n">params</span><span class="o">[</span><span class="ss">:message</span><span class="o">]</span><span class="p">)</span>
3500
+ </span><span id="__span-12-5"><a id="__codelineno-12-5" name="__codelineno-12-5" href="#__codelineno-12-5"></a>
3501
+ </span><span id="__span-12-6"><a id="__codelineno-12-6" name="__codelineno-12-6" href="#__codelineno-12-6"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span>
3502
+ </span><span id="__span-12-7"><a id="__codelineno-12-7" name="__codelineno-12-7" href="#__codelineno-12-7"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3503
+ </span><span id="__span-12-8"><a id="__codelineno-12-8" name="__codelineno-12-8" href="#__codelineno-12-8"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">robot_name</span>
3504
+ </span><span id="__span-12-9"><a id="__codelineno-12-9" name="__codelineno-12-9" href="#__codelineno-12-9"></a><span class="w"> </span><span class="p">}</span>
3505
+ </span><span id="__span-12-10"><a id="__codelineno-12-10" name="__codelineno-12-10" href="#__codelineno-12-10"></a><span class="w"> </span><span class="k">end</span>
3506
+ </span><span id="__span-12-11"><a id="__codelineno-12-11" name="__codelineno-12-11" href="#__codelineno-12-11"></a><span class="k">end</span>
3507
+ </span></code></pre></div>
3508
+ <h3 id="using-a-network-in-controllers">Using a Network in Controllers<a class="headerlink" href="#using-a-network-in-controllers" title="Permanent link">&para;</a></h3>
3509
+ <p>Networks use <code>RobotLab.create_network</code> with a block DSL that defines tasks. Each task wraps a robot with dependency declarations:</p>
3510
+ <div class="language-ruby highlight"><span class="filename">app/controllers/chat_controller.rb</span><pre><span></span><code><span id="__span-13-1"><a id="__codelineno-13-1" name="__codelineno-13-1" href="#__codelineno-13-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatController</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationController</span>
3511
+ </span><span id="__span-13-2"><a id="__codelineno-13-2" name="__codelineno-13-2" href="#__codelineno-13-2"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">create</span>
3512
+ </span><span id="__span-13-3"><a id="__codelineno-13-3" name="__codelineno-13-3" href="#__codelineno-13-3"></a><span class="w"> </span><span class="n">support_robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3513
+ </span><span id="__span-13-4"><a id="__codelineno-13-4" name="__codelineno-13-4" href="#__codelineno-13-4"></a><span class="w"> </span><span class="n">billing_robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">BillingRobot</span><span class="o">.</span><span class="n">build</span>
3514
+ </span><span id="__span-13-5"><a id="__codelineno-13-5" name="__codelineno-13-5" href="#__codelineno-13-5"></a>
3515
+ </span><span id="__span-13-6"><a id="__codelineno-13-6" name="__codelineno-13-6" href="#__codelineno-13-6"></a><span class="w"> </span><span class="n">network</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">.</span><span class="n">create_network</span><span class="p">(</span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;customer_service&quot;</span><span class="p">)</span><span class="w"> </span><span class="k">do</span>
3516
+ </span><span id="__span-13-7"><a id="__codelineno-13-7" name="__codelineno-13-7" href="#__codelineno-13-7"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:support</span><span class="p">,</span><span class="w"> </span><span class="n">support_robot</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:none</span>
3517
+ </span><span id="__span-13-8"><a id="__codelineno-13-8" name="__codelineno-13-8" href="#__codelineno-13-8"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:billing</span><span class="p">,</span><span class="w"> </span><span class="n">billing_robot</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:optional</span>
3518
+ </span><span id="__span-13-9"><a id="__codelineno-13-9" name="__codelineno-13-9" href="#__codelineno-13-9"></a><span class="w"> </span><span class="k">end</span>
3519
+ </span><span id="__span-13-10"><a id="__codelineno-13-10" name="__codelineno-13-10" href="#__codelineno-13-10"></a>
3520
+ </span><span id="__span-13-11"><a id="__codelineno-13-11" name="__codelineno-13-11" href="#__codelineno-13-11"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">network</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="ss">message</span><span class="p">:</span><span class="w"> </span><span class="n">params</span><span class="o">[</span><span class="ss">:message</span><span class="o">]</span><span class="p">,</span><span class="w"> </span><span class="ss">user_id</span><span class="p">:</span><span class="w"> </span><span class="n">current_user</span><span class="o">.</span><span class="n">id</span><span class="p">)</span>
3521
+ </span><span id="__span-13-12"><a id="__codelineno-13-12" name="__codelineno-13-12" href="#__codelineno-13-12"></a>
3522
+ </span><span id="__span-13-13"><a id="__codelineno-13-13" name="__codelineno-13-13" href="#__codelineno-13-13"></a><span class="w"> </span><span class="c1"># result is a SimpleFlow::Result</span>
3523
+ </span><span id="__span-13-14"><a id="__codelineno-13-14" name="__codelineno-13-14" href="#__codelineno-13-14"></a><span class="w"> </span><span class="c1"># result.value is a RobotResult from the last robot</span>
3524
+ </span><span id="__span-13-15"><a id="__codelineno-13-15" name="__codelineno-13-15" href="#__codelineno-13-15"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span>
3525
+ </span><span id="__span-13-16"><a id="__codelineno-13-16" name="__codelineno-13-16" href="#__codelineno-13-16"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">value</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3526
+ </span><span id="__span-13-17"><a id="__codelineno-13-17" name="__codelineno-13-17" href="#__codelineno-13-17"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">value</span><span class="o">.</span><span class="n">robot_name</span>
3527
+ </span><span id="__span-13-18"><a id="__codelineno-13-18" name="__codelineno-13-18" href="#__codelineno-13-18"></a><span class="w"> </span><span class="p">}</span>
3528
+ </span><span id="__span-13-19"><a id="__codelineno-13-19" name="__codelineno-13-19" href="#__codelineno-13-19"></a><span class="w"> </span><span class="k">end</span>
3529
+ </span><span id="__span-13-20"><a id="__codelineno-13-20" name="__codelineno-13-20" href="#__codelineno-13-20"></a><span class="k">end</span>
3530
+ </span></code></pre></div>
3531
+ <h2 id="prompt-templates">Prompt Templates<a class="headerlink" href="#prompt-templates" title="Permanent link">&para;</a></h2>
3532
+ <h3 id="template-location">Template Location<a class="headerlink" href="#template-location" title="Permanent link">&para;</a></h3>
3533
+ <p>Templates are <code>.md</code> files with YAML front matter, stored in <code>app/prompts/</code> (auto-configured for Rails):</p>
3534
+ <div class="language-text highlight"><pre><span></span><code><span id="__span-14-1"><a id="__codelineno-14-1" name="__codelineno-14-1" href="#__codelineno-14-1"></a>app/prompts/
3535
+ </span><span id="__span-14-2"><a id="__codelineno-14-2" name="__codelineno-14-2" href="#__codelineno-14-2"></a>├── support.md
3536
+ </span><span id="__span-14-3"><a id="__codelineno-14-3" name="__codelineno-14-3" href="#__codelineno-14-3"></a>├── billing.md
3537
+ </span><span id="__span-14-4"><a id="__codelineno-14-4" name="__codelineno-14-4" href="#__codelineno-14-4"></a>└── router.md
3538
+ </span></code></pre></div>
3539
+ <h3 id="template-format">Template Format<a class="headerlink" href="#template-format" title="Permanent link">&para;</a></h3>
3540
+ <div class="language-markdown highlight"><span class="filename">app/prompts/support.md</span><pre><span></span><code><span id="__span-15-1"><a id="__codelineno-15-1" name="__codelineno-15-1" href="#__codelineno-15-1"></a>---
3541
+ </span><span id="__span-15-2"><a id="__codelineno-15-2" name="__codelineno-15-2" href="#__codelineno-15-2"></a>description: Customer support assistant
3542
+ </span><span id="__span-15-3"><a id="__codelineno-15-3" name="__codelineno-15-3" href="#__codelineno-15-3"></a>parameters:
3543
+ </span><span id="__span-15-4"><a id="__codelineno-15-4" name="__codelineno-15-4" href="#__codelineno-15-4"></a> company_name: null
3544
+ </span><span id="__span-15-5"><a id="__codelineno-15-5" name="__codelineno-15-5" href="#__codelineno-15-5"></a> tone: friendly
3545
+ </span><span id="__span-15-6"><a id="__codelineno-15-6" name="__codelineno-15-6" href="#__codelineno-15-6"></a>model: claude-sonnet-4
3546
+ </span><span id="__span-15-7"><a id="__codelineno-15-7" name="__codelineno-15-7" href="#__codelineno-15-7"></a><span class="gu">temperature: 0.7</span>
3547
+ </span><span id="__span-15-8"><a id="__codelineno-15-8" name="__codelineno-15-8" href="#__codelineno-15-8"></a><span class="gu">---</span>
3548
+ </span><span id="__span-15-9"><a id="__codelineno-15-9" name="__codelineno-15-9" href="#__codelineno-15-9"></a>You are a support agent for &lt;%= company_name %&gt;.
3549
+ </span><span id="__span-15-10"><a id="__codelineno-15-10" name="__codelineno-15-10" href="#__codelineno-15-10"></a>Respond in a &lt;%= tone %&gt; manner.
3550
+ </span><span id="__span-15-11"><a id="__codelineno-15-11" name="__codelineno-15-11" href="#__codelineno-15-11"></a>
3551
+ </span><span id="__span-15-12"><a id="__codelineno-15-12" name="__codelineno-15-12" href="#__codelineno-15-12"></a>Your responsibilities:
3552
+ </span><span id="__span-15-13"><a id="__codelineno-15-13" name="__codelineno-15-13" href="#__codelineno-15-13"></a><span class="k">-</span><span class="w"> </span>Answer product questions
3553
+ </span><span id="__span-15-14"><a id="__codelineno-15-14" name="__codelineno-15-14" href="#__codelineno-15-14"></a><span class="k">-</span><span class="w"> </span>Help with order issues
3554
+ </span><span id="__span-15-15"><a id="__codelineno-15-15" name="__codelineno-15-15" href="#__codelineno-15-15"></a><span class="k">-</span><span class="w"> </span>Provide friendly assistance
3555
+ </span></code></pre></div>
3556
+ <h3 id="template-usage">Template Usage<a class="headerlink" href="#template-usage" title="Permanent link">&para;</a></h3>
3557
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-16-1"><a id="__codelineno-16-1" name="__codelineno-16-1" href="#__codelineno-16-1"></a><span class="c1"># Pass context to fill template parameters</span>
3558
+ </span><span id="__span-16-2"><a id="__codelineno-16-2" name="__codelineno-16-2" href="#__codelineno-16-2"></a><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">.</span><span class="n">build</span><span class="p">(</span>
3559
+ </span><span id="__span-16-3"><a id="__codelineno-16-3" name="__codelineno-16-3" href="#__codelineno-16-3"></a><span class="w"> </span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;support&quot;</span><span class="p">,</span>
3560
+ </span><span id="__span-16-4"><a id="__codelineno-16-4" name="__codelineno-16-4" href="#__codelineno-16-4"></a><span class="w"> </span><span class="ss">template</span><span class="p">:</span><span class="w"> </span><span class="ss">:support</span><span class="p">,</span>
3561
+ </span><span id="__span-16-5"><a id="__codelineno-16-5" name="__codelineno-16-5" href="#__codelineno-16-5"></a><span class="w"> </span><span class="ss">context</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">company_name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;Acme Corp&quot;</span><span class="w"> </span><span class="p">}</span>
3562
+ </span><span id="__span-16-6"><a id="__codelineno-16-6" name="__codelineno-16-6" href="#__codelineno-16-6"></a><span class="p">)</span>
3563
+ </span><span id="__span-16-7"><a id="__codelineno-16-7" name="__codelineno-16-7" href="#__codelineno-16-7"></a>
3564
+ </span><span id="__span-16-8"><a id="__codelineno-16-8" name="__codelineno-16-8" href="#__codelineno-16-8"></a><span class="c1"># Parameters with defaults (like `tone: friendly`) are optional.</span>
3565
+ </span><span id="__span-16-9"><a id="__codelineno-16-9" name="__codelineno-16-9" href="#__codelineno-16-9"></a><span class="c1"># Parameters set to null are required and must be provided via context.</span>
3566
+ </span><span id="__span-16-10"><a id="__codelineno-16-10" name="__codelineno-16-10" href="#__codelineno-16-10"></a><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="s2">&quot;I need help with my order&quot;</span><span class="p">)</span>
3567
+ </span></code></pre></div>
3568
+ <h2 id="action-cable-integration">Action Cable Integration<a class="headerlink" href="#action-cable-integration" title="Permanent link">&para;</a></h2>
3569
+ <h3 id="channel">Channel<a class="headerlink" href="#channel" title="Permanent link">&para;</a></h3>
3570
+ <div class="language-ruby highlight"><span class="filename">app/channels/chat_channel.rb</span><pre><span></span><code><span id="__span-17-1"><a id="__codelineno-17-1" name="__codelineno-17-1" href="#__codelineno-17-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatChannel</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationCable</span><span class="o">::</span><span class="no">Channel</span>
3571
+ </span><span id="__span-17-2"><a id="__codelineno-17-2" name="__codelineno-17-2" href="#__codelineno-17-2"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">subscribed</span>
3572
+ </span><span id="__span-17-3"><a id="__codelineno-17-3" name="__codelineno-17-3" href="#__codelineno-17-3"></a><span class="w"> </span><span class="n">stream_from</span><span class="w"> </span><span class="s2">&quot;chat_</span><span class="si">#{</span><span class="n">params</span><span class="o">[</span><span class="ss">:session_id</span><span class="o">]</span><span class="si">}</span><span class="s2">&quot;</span>
3573
+ </span><span id="__span-17-4"><a id="__codelineno-17-4" name="__codelineno-17-4" href="#__codelineno-17-4"></a><span class="w"> </span><span class="k">end</span>
3574
+ </span><span id="__span-17-5"><a id="__codelineno-17-5" name="__codelineno-17-5" href="#__codelineno-17-5"></a>
3575
+ </span><span id="__span-17-6"><a id="__codelineno-17-6" name="__codelineno-17-6" href="#__codelineno-17-6"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">receive</span><span class="p">(</span><span class="n">data</span><span class="p">)</span>
3576
+ </span><span id="__span-17-7"><a id="__codelineno-17-7" name="__codelineno-17-7" href="#__codelineno-17-7"></a><span class="w"> </span><span class="n">message</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">data</span><span class="o">[</span><span class="s2">&quot;message&quot;</span><span class="o">]</span>
3577
+ </span><span id="__span-17-8"><a id="__codelineno-17-8" name="__codelineno-17-8" href="#__codelineno-17-8"></a><span class="w"> </span><span class="n">session_id</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">data</span><span class="o">[</span><span class="s2">&quot;session_id&quot;</span><span class="o">]</span>
3578
+ </span><span id="__span-17-9"><a id="__codelineno-17-9" name="__codelineno-17-9" href="#__codelineno-17-9"></a>
3579
+ </span><span id="__span-17-10"><a id="__codelineno-17-10" name="__codelineno-17-10" href="#__codelineno-17-10"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3580
+ </span><span id="__span-17-11"><a id="__codelineno-17-11" name="__codelineno-17-11" href="#__codelineno-17-11"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3581
+ </span><span id="__span-17-12"><a id="__codelineno-17-12" name="__codelineno-17-12" href="#__codelineno-17-12"></a>
3582
+ </span><span id="__span-17-13"><a id="__codelineno-17-13" name="__codelineno-17-13" href="#__codelineno-17-13"></a><span class="w"> </span><span class="no">ActionCable</span><span class="o">.</span><span class="n">server</span><span class="o">.</span><span class="n">broadcast</span><span class="p">(</span>
3583
+ </span><span id="__span-17-14"><a id="__codelineno-17-14" name="__codelineno-17-14" href="#__codelineno-17-14"></a><span class="w"> </span><span class="s2">&quot;chat_</span><span class="si">#{</span><span class="n">session_id</span><span class="si">}</span><span class="s2">&quot;</span><span class="p">,</span>
3584
+ </span><span id="__span-17-15"><a id="__codelineno-17-15" name="__codelineno-17-15" href="#__codelineno-17-15"></a><span class="w"> </span><span class="p">{</span>
3585
+ </span><span id="__span-17-16"><a id="__codelineno-17-16" name="__codelineno-17-16" href="#__codelineno-17-16"></a><span class="w"> </span><span class="ss">event</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;complete&quot;</span><span class="p">,</span>
3586
+ </span><span id="__span-17-17"><a id="__codelineno-17-17" name="__codelineno-17-17" href="#__codelineno-17-17"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3587
+ </span><span id="__span-17-18"><a id="__codelineno-17-18" name="__codelineno-17-18" href="#__codelineno-17-18"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">robot_name</span>
3588
+ </span><span id="__span-17-19"><a id="__codelineno-17-19" name="__codelineno-17-19" href="#__codelineno-17-19"></a><span class="w"> </span><span class="p">}</span>
3589
+ </span><span id="__span-17-20"><a id="__codelineno-17-20" name="__codelineno-17-20" href="#__codelineno-17-20"></a><span class="w"> </span><span class="p">)</span>
3590
+ </span><span id="__span-17-21"><a id="__codelineno-17-21" name="__codelineno-17-21" href="#__codelineno-17-21"></a><span class="w"> </span><span class="k">end</span>
3591
+ </span><span id="__span-17-22"><a id="__codelineno-17-22" name="__codelineno-17-22" href="#__codelineno-17-22"></a><span class="k">end</span>
3592
+ </span></code></pre></div>
3593
+ <h3 id="javascript-client">JavaScript Client<a class="headerlink" href="#javascript-client" title="Permanent link">&para;</a></h3>
3594
+ <div class="language-javascript highlight"><pre><span></span><code><span id="__span-18-1"><a id="__codelineno-18-1" name="__codelineno-18-1" href="#__codelineno-18-1"></a><span class="kd">const</span><span class="w"> </span><span class="nx">channel</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="nx">consumer</span><span class="p">.</span><span class="nx">subscriptions</span><span class="p">.</span><span class="nx">create</span><span class="p">(</span>
3595
+ </span><span id="__span-18-2"><a id="__codelineno-18-2" name="__codelineno-18-2" href="#__codelineno-18-2"></a><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="nx">channel</span><span class="o">:</span><span class="w"> </span><span class="s2">&quot;ChatChannel&quot;</span><span class="p">,</span><span class="w"> </span><span class="nx">session_id</span><span class="o">:</span><span class="w"> </span><span class="nx">sessionId</span><span class="w"> </span><span class="p">},</span>
3596
+ </span><span id="__span-18-3"><a id="__codelineno-18-3" name="__codelineno-18-3" href="#__codelineno-18-3"></a><span class="w"> </span><span class="p">{</span>
3597
+ </span><span id="__span-18-4"><a id="__codelineno-18-4" name="__codelineno-18-4" href="#__codelineno-18-4"></a><span class="w"> </span><span class="nx">received</span><span class="p">(</span><span class="nx">data</span><span class="p">)</span><span class="w"> </span><span class="p">{</span>
3598
+ </span><span id="__span-18-5"><a id="__codelineno-18-5" name="__codelineno-18-5" href="#__codelineno-18-5"></a><span class="w"> </span><span class="k">if</span><span class="w"> </span><span class="p">(</span><span class="nx">data</span><span class="p">.</span><span class="nx">event</span><span class="w"> </span><span class="o">===</span><span class="w"> </span><span class="s2">&quot;complete&quot;</span><span class="p">)</span><span class="w"> </span><span class="p">{</span>
3599
+ </span><span id="__span-18-6"><a id="__codelineno-18-6" name="__codelineno-18-6" href="#__codelineno-18-6"></a><span class="w"> </span><span class="nx">displayMessage</span><span class="p">(</span><span class="nx">data</span><span class="p">.</span><span class="nx">response</span><span class="p">);</span>
3600
+ </span><span id="__span-18-7"><a id="__codelineno-18-7" name="__codelineno-18-7" href="#__codelineno-18-7"></a><span class="w"> </span><span class="p">}</span>
3601
+ </span><span id="__span-18-8"><a id="__codelineno-18-8" name="__codelineno-18-8" href="#__codelineno-18-8"></a><span class="w"> </span><span class="p">}</span>
3602
+ </span><span id="__span-18-9"><a id="__codelineno-18-9" name="__codelineno-18-9" href="#__codelineno-18-9"></a><span class="w"> </span><span class="p">}</span>
3603
+ </span><span id="__span-18-10"><a id="__codelineno-18-10" name="__codelineno-18-10" href="#__codelineno-18-10"></a><span class="p">);</span>
3604
+ </span><span id="__span-18-11"><a id="__codelineno-18-11" name="__codelineno-18-11" href="#__codelineno-18-11"></a>
3605
+ </span><span id="__span-18-12"><a id="__codelineno-18-12" name="__codelineno-18-12" href="#__codelineno-18-12"></a><span class="nx">channel</span><span class="p">.</span><span class="nx">send</span><span class="p">({</span><span class="w"> </span><span class="nx">message</span><span class="o">:</span><span class="w"> </span><span class="s2">&quot;Hello!&quot;</span><span class="p">,</span><span class="w"> </span><span class="nx">session_id</span><span class="o">:</span><span class="w"> </span><span class="nx">sessionId</span><span class="w"> </span><span class="p">});</span>
3606
+ </span></code></pre></div>
3607
+ <h2 id="background-jobs">Background Jobs<a class="headerlink" href="#background-jobs" title="Permanent link">&para;</a></h2>
3608
+ <h3 id="robotrunjob-generated">RobotRunJob (Generated)<a class="headerlink" href="#robotrunjob-generated" title="Permanent link">&para;</a></h3>
3609
+ <p>The install generator creates <code>app/jobs/robot_run_job.rb</code> — an ActiveJob class that wraps robot execution with result persistence and optional Turbo Stream broadcasting.</p>
3610
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-19-1"><a id="__codelineno-19-1" name="__codelineno-19-1" href="#__codelineno-19-1"></a><span class="c1"># Enqueue from a controller</span>
3611
+ </span><span id="__span-19-2"><a id="__codelineno-19-2" name="__codelineno-19-2" href="#__codelineno-19-2"></a><span class="no">RobotRunJob</span><span class="o">.</span><span class="n">perform_later</span><span class="p">(</span>
3612
+ </span><span id="__span-19-3"><a id="__codelineno-19-3" name="__codelineno-19-3" href="#__codelineno-19-3"></a><span class="w"> </span><span class="ss">robot_class</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;SupportRobot&quot;</span><span class="p">,</span>
3613
+ </span><span id="__span-19-4"><a id="__codelineno-19-4" name="__codelineno-19-4" href="#__codelineno-19-4"></a><span class="w"> </span><span class="ss">message</span><span class="p">:</span><span class="w"> </span><span class="n">params</span><span class="o">[</span><span class="ss">:message</span><span class="o">]</span><span class="p">,</span>
3614
+ </span><span id="__span-19-5"><a id="__codelineno-19-5" name="__codelineno-19-5" href="#__codelineno-19-5"></a><span class="w"> </span><span class="ss">thread_id</span><span class="p">:</span><span class="w"> </span><span class="n">session_id</span>
3615
+ </span><span id="__span-19-6"><a id="__codelineno-19-6" name="__codelineno-19-6" href="#__codelineno-19-6"></a><span class="p">)</span>
3616
+ </span><span id="__span-19-7"><a id="__codelineno-19-7" name="__codelineno-19-7" href="#__codelineno-19-7"></a>
3617
+ </span><span id="__span-19-8"><a id="__codelineno-19-8" name="__codelineno-19-8" href="#__codelineno-19-8"></a><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">status</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;processing&quot;</span><span class="w"> </span><span class="p">}</span>
3618
+ </span></code></pre></div>
3619
+ <p>The job:</p>
3620
+ <ol>
3621
+ <li>Finds or creates a <code>RobotLabThread</code> by <code>thread_id</code></li>
3622
+ <li>Resolves the robot class via <code>constantize.build</code></li>
3623
+ <li>Wires Turbo Stream callbacks when <code>turbo-rails</code> is available (graceful no-op otherwise)</li>
3624
+ <li>Runs the robot and persists the result to <code>RobotLabResult</code></li>
3625
+ <li>Broadcasts completion or error events via Turbo Streams</li>
3626
+ </ol>
3627
+ <p>Customize the generated job to change queue name, retry policy, or error handling.</p>
3628
+ <h3 id="turbo-stream-token-streaming">Turbo Stream Token Streaming<a class="headerlink" href="#turbo-stream-token-streaming" title="Permanent link">&para;</a></h3>
3629
+ <p>When <code>turbo-rails</code> is installed, <code>RobotRunJob</code> automatically streams content tokens and tool call badges to the browser in real time.</p>
3630
+ <h4 id="view-setup">View Setup<a class="headerlink" href="#view-setup" title="Permanent link">&para;</a></h4>
3631
+ <p>Subscribe to the thread's Turbo Stream channel in your view:</p>
3632
+ <div class="language-erb highlight"><pre><span></span><code><span id="__span-20-1"><a id="__codelineno-20-1" name="__codelineno-20-1" href="#__codelineno-20-1"></a><span class="x">&lt;%%= turbo_stream_from &quot;robot_lab_thread_#{@thread_id}&quot; </span><span class="err">%&gt;</span>
3633
+ </span><span id="__span-20-2"><a id="__codelineno-20-2" name="__codelineno-20-2" href="#__codelineno-20-2"></a>
3634
+ </span><span id="__span-20-3"><a id="__codelineno-20-3" name="__codelineno-20-3" href="#__codelineno-20-3"></a><span class="x">&lt;div id=&quot;robot_response&quot;&gt;&lt;/div&gt;</span>
3635
+ </span><span id="__span-20-4"><a id="__codelineno-20-4" name="__codelineno-20-4" href="#__codelineno-20-4"></a><span class="x">&lt;div id=&quot;robot_tools&quot;&gt;&lt;/div&gt;</span>
3636
+ </span><span id="__span-20-5"><a id="__codelineno-20-5" name="__codelineno-20-5" href="#__codelineno-20-5"></a><span class="x">&lt;div id=&quot;robot_status&quot;&gt;Processing...&lt;/div&gt;</span>
3637
+ </span><span id="__span-20-6"><a id="__codelineno-20-6" name="__codelineno-20-6" href="#__codelineno-20-6"></a><span class="x">&lt;div id=&quot;robot_errors&quot;&gt;&lt;/div&gt;</span>
3638
+ </span></code></pre></div>
3639
+ <p>As the robot generates tokens, they are appended to <code>#robot_response</code>. Tool calls appear as badges in <code>#robot_tools</code>. On completion, <code>#robot_status</code> is replaced with "Complete".</p>
3640
+ <h4 id="turbostreamcallbacks-api">TurboStreamCallbacks API<a class="headerlink" href="#turbostreamcallbacks-api" title="Permanent link">&para;</a></h4>
3641
+ <p><code>RobotLab::Rails::TurboStreamCallbacks</code> is a stateless utility module for building callback Procs. Use it outside of <code>RobotRunJob</code> for custom streaming setups:</p>
3642
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-21-1"><a id="__codelineno-21-1" name="__codelineno-21-1" href="#__codelineno-21-1"></a><span class="c1"># Check if Turbo Streams is available</span>
3643
+ </span><span id="__span-21-2"><a id="__codelineno-21-2" name="__codelineno-21-2" href="#__codelineno-21-2"></a><span class="no">RobotLab</span><span class="o">::</span><span class="no">Rails</span><span class="o">::</span><span class="no">TurboStreamCallbacks</span><span class="o">.</span><span class="n">available?</span>
3644
+ </span><span id="__span-21-3"><a id="__codelineno-21-3" name="__codelineno-21-3" href="#__codelineno-21-3"></a>
3645
+ </span><span id="__span-21-4"><a id="__codelineno-21-4" name="__codelineno-21-4" href="#__codelineno-21-4"></a><span class="c1"># Build a content streaming callback</span>
3646
+ </span><span id="__span-21-5"><a id="__codelineno-21-5" name="__codelineno-21-5" href="#__codelineno-21-5"></a><span class="n">on_content</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Rails</span><span class="o">::</span><span class="no">TurboStreamCallbacks</span><span class="o">.</span><span class="n">build_content_callback</span><span class="p">(</span>
3647
+ </span><span id="__span-21-6"><a id="__codelineno-21-6" name="__codelineno-21-6" href="#__codelineno-21-6"></a><span class="w"> </span><span class="ss">stream_name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;robot_lab_thread_</span><span class="si">#{</span><span class="n">thread_id</span><span class="si">}</span><span class="s2">&quot;</span><span class="p">,</span>
3648
+ </span><span id="__span-21-7"><a id="__codelineno-21-7" name="__codelineno-21-7" href="#__codelineno-21-7"></a><span class="w"> </span><span class="ss">target</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;robot_response&quot;</span><span class="w"> </span><span class="c1"># default</span>
3649
+ </span><span id="__span-21-8"><a id="__codelineno-21-8" name="__codelineno-21-8" href="#__codelineno-21-8"></a><span class="p">)</span>
3650
+ </span><span id="__span-21-9"><a id="__codelineno-21-9" name="__codelineno-21-9" href="#__codelineno-21-9"></a>
3651
+ </span><span id="__span-21-10"><a id="__codelineno-21-10" name="__codelineno-21-10" href="#__codelineno-21-10"></a><span class="c1"># Build a tool call badge callback</span>
3652
+ </span><span id="__span-21-11"><a id="__codelineno-21-11" name="__codelineno-21-11" href="#__codelineno-21-11"></a><span class="n">on_tool_call</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Rails</span><span class="o">::</span><span class="no">TurboStreamCallbacks</span><span class="o">.</span><span class="n">build_tool_call_callback</span><span class="p">(</span>
3653
+ </span><span id="__span-21-12"><a id="__codelineno-21-12" name="__codelineno-21-12" href="#__codelineno-21-12"></a><span class="w"> </span><span class="ss">stream_name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;robot_lab_thread_</span><span class="si">#{</span><span class="n">thread_id</span><span class="si">}</span><span class="s2">&quot;</span><span class="p">,</span>
3654
+ </span><span id="__span-21-13"><a id="__codelineno-21-13" name="__codelineno-21-13" href="#__codelineno-21-13"></a><span class="w"> </span><span class="ss">target</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;robot_tools&quot;</span><span class="w"> </span><span class="c1"># default</span>
3655
+ </span><span id="__span-21-14"><a id="__codelineno-21-14" name="__codelineno-21-14" href="#__codelineno-21-14"></a><span class="p">)</span>
3656
+ </span><span id="__span-21-15"><a id="__codelineno-21-15" name="__codelineno-21-15" href="#__codelineno-21-15"></a>
3657
+ </span><span id="__span-21-16"><a id="__codelineno-21-16" name="__codelineno-21-16" href="#__codelineno-21-16"></a><span class="c1"># Wire into a robot at build time</span>
3658
+ </span><span id="__span-21-17"><a id="__codelineno-21-17" name="__codelineno-21-17" href="#__codelineno-21-17"></a><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span><span class="p">(</span><span class="ss">on_content</span><span class="p">:</span><span class="w"> </span><span class="n">on_content</span><span class="p">,</span><span class="w"> </span><span class="ss">on_tool_call</span><span class="p">:</span><span class="w"> </span><span class="n">on_tool_call</span><span class="p">)</span>
3659
+ </span><span id="__span-21-18"><a id="__codelineno-21-18" name="__codelineno-21-18" href="#__codelineno-21-18"></a><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3660
+ </span></code></pre></div>
3661
+ <p>The stream name convention is <code>"robot_lab_thread_#{thread_id}"</code>, matching the <code>RobotLabThread.session_id</code> pattern.</p>
3662
+ <h3 id="custom-background-job">Custom Background Job<a class="headerlink" href="#custom-background-job" title="Permanent link">&para;</a></h3>
3663
+ <p>For full control, write your own job instead of using the generated one:</p>
3664
+ <div class="language-ruby highlight"><span class="filename">app/jobs/process_message_job.rb</span><pre><span></span><code><span id="__span-22-1"><a id="__codelineno-22-1" name="__codelineno-22-1" href="#__codelineno-22-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ProcessMessageJob</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationJob</span>
3665
+ </span><span id="__span-22-2"><a id="__codelineno-22-2" name="__codelineno-22-2" href="#__codelineno-22-2"></a><span class="w"> </span><span class="n">queue_as</span><span class="w"> </span><span class="ss">:default</span>
3666
+ </span><span id="__span-22-3"><a id="__codelineno-22-3" name="__codelineno-22-3" href="#__codelineno-22-3"></a>
3667
+ </span><span id="__span-22-4"><a id="__codelineno-22-4" name="__codelineno-22-4" href="#__codelineno-22-4"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">perform</span><span class="p">(</span><span class="ss">session_id</span><span class="p">:,</span><span class="w"> </span><span class="ss">message</span><span class="p">:,</span><span class="w"> </span><span class="ss">user_id</span><span class="p">:)</span>
3668
+ </span><span id="__span-22-5"><a id="__codelineno-22-5" name="__codelineno-22-5" href="#__codelineno-22-5"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3669
+ </span><span id="__span-22-6"><a id="__codelineno-22-6" name="__codelineno-22-6" href="#__codelineno-22-6"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3670
+ </span><span id="__span-22-7"><a id="__codelineno-22-7" name="__codelineno-22-7" href="#__codelineno-22-7"></a>
3671
+ </span><span id="__span-22-8"><a id="__codelineno-22-8" name="__codelineno-22-8" href="#__codelineno-22-8"></a><span class="w"> </span><span class="no">ActionCable</span><span class="o">.</span><span class="n">server</span><span class="o">.</span><span class="n">broadcast</span><span class="p">(</span>
3672
+ </span><span id="__span-22-9"><a id="__codelineno-22-9" name="__codelineno-22-9" href="#__codelineno-22-9"></a><span class="w"> </span><span class="s2">&quot;chat_</span><span class="si">#{</span><span class="n">session_id</span><span class="si">}</span><span class="s2">&quot;</span><span class="p">,</span>
3673
+ </span><span id="__span-22-10"><a id="__codelineno-22-10" name="__codelineno-22-10" href="#__codelineno-22-10"></a><span class="w"> </span><span class="p">{</span>
3674
+ </span><span id="__span-22-11"><a id="__codelineno-22-11" name="__codelineno-22-11" href="#__codelineno-22-11"></a><span class="w"> </span><span class="ss">event</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;complete&quot;</span><span class="p">,</span>
3675
+ </span><span id="__span-22-12"><a id="__codelineno-22-12" name="__codelineno-22-12" href="#__codelineno-22-12"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3676
+ </span><span id="__span-22-13"><a id="__codelineno-22-13" name="__codelineno-22-13" href="#__codelineno-22-13"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">robot_name</span>
3677
+ </span><span id="__span-22-14"><a id="__codelineno-22-14" name="__codelineno-22-14" href="#__codelineno-22-14"></a><span class="w"> </span><span class="p">}</span>
3678
+ </span><span id="__span-22-15"><a id="__codelineno-22-15" name="__codelineno-22-15" href="#__codelineno-22-15"></a><span class="w"> </span><span class="p">)</span>
3679
+ </span><span id="__span-22-16"><a id="__codelineno-22-16" name="__codelineno-22-16" href="#__codelineno-22-16"></a><span class="w"> </span><span class="k">end</span>
3680
+ </span><span id="__span-22-17"><a id="__codelineno-22-17" name="__codelineno-22-17" href="#__codelineno-22-17"></a><span class="k">end</span>
3681
+ </span></code></pre></div>
3682
+ <h2 id="testing">Testing<a class="headerlink" href="#testing" title="Permanent link">&para;</a></h2>
3683
+ <h3 id="test-configuration">Test Configuration<a class="headerlink" href="#test-configuration" title="Permanent link">&para;</a></h3>
3684
+ <p>Use <code>config/robot_lab.yml</code> to configure the test environment with a faster, cheaper model:</p>
3685
+ <div class="language-yaml highlight"><span class="filename">config/robot_lab.yml</span><pre><span></span><code><span id="__span-23-1"><a id="__codelineno-23-1" name="__codelineno-23-1" href="#__codelineno-23-1"></a><span class="nt">test</span><span class="p">:</span>
3686
+ </span><span id="__span-23-2"><a id="__codelineno-23-2" name="__codelineno-23-2" href="#__codelineno-23-2"></a><span class="w"> </span><span class="nt">max_iterations</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">3</span>
3687
+ </span><span id="__span-23-3"><a id="__codelineno-23-3" name="__codelineno-23-3" href="#__codelineno-23-3"></a><span class="w"> </span><span class="nt">streaming_enabled</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">false</span>
3688
+ </span><span id="__span-23-4"><a id="__codelineno-23-4" name="__codelineno-23-4" href="#__codelineno-23-4"></a><span class="w"> </span><span class="nt">ruby_llm</span><span class="p">:</span>
3689
+ </span><span id="__span-23-5"><a id="__codelineno-23-5" name="__codelineno-23-5" href="#__codelineno-23-5"></a><span class="w"> </span><span class="nt">model</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">claude-3-haiku-20240307</span>
3690
+ </span><span id="__span-23-6"><a id="__codelineno-23-6" name="__codelineno-23-6" href="#__codelineno-23-6"></a><span class="w"> </span><span class="nt">request_timeout</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">30</span>
3691
+ </span><span id="__span-23-7"><a id="__codelineno-23-7" name="__codelineno-23-7" href="#__codelineno-23-7"></a><span class="w"> </span><span class="nt">max_retries</span><span class="p">:</span><span class="w"> </span><span class="l l-Scalar l-Scalar-Plain">1</span>
3692
+ </span></code></pre></div>
3693
+ <h3 id="robot-tests">Robot Tests<a class="headerlink" href="#robot-tests" title="Permanent link">&para;</a></h3>
3694
+ <div class="language-ruby highlight"><span class="filename">test/robots/support_robot_test.rb</span><pre><span></span><code><span id="__span-24-1"><a id="__codelineno-24-1" name="__codelineno-24-1" href="#__codelineno-24-1"></a><span class="nb">require</span><span class="w"> </span><span class="s2">&quot;test_helper&quot;</span>
3695
+ </span><span id="__span-24-2"><a id="__codelineno-24-2" name="__codelineno-24-2" href="#__codelineno-24-2"></a>
3696
+ </span><span id="__span-24-3"><a id="__codelineno-24-3" name="__codelineno-24-3" href="#__codelineno-24-3"></a><span class="k">class</span><span class="w"> </span><span class="nc">SupportRobotTest</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ActiveSupport</span><span class="o">::</span><span class="no">TestCase</span>
3697
+ </span><span id="__span-24-4"><a id="__codelineno-24-4" name="__codelineno-24-4" href="#__codelineno-24-4"></a><span class="w"> </span><span class="nb">test</span><span class="w"> </span><span class="s2">&quot;builds valid robot&quot;</span><span class="w"> </span><span class="k">do</span>
3698
+ </span><span id="__span-24-5"><a id="__codelineno-24-5" name="__codelineno-24-5" href="#__codelineno-24-5"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3699
+ </span><span id="__span-24-6"><a id="__codelineno-24-6" name="__codelineno-24-6" href="#__codelineno-24-6"></a><span class="w"> </span><span class="n">assert_equal</span><span class="w"> </span><span class="s2">&quot;support&quot;</span><span class="p">,</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">name</span>
3700
+ </span><span id="__span-24-7"><a id="__codelineno-24-7" name="__codelineno-24-7" href="#__codelineno-24-7"></a><span class="w"> </span><span class="k">end</span>
3701
+ </span><span id="__span-24-8"><a id="__codelineno-24-8" name="__codelineno-24-8" href="#__codelineno-24-8"></a>
3702
+ </span><span id="__span-24-9"><a id="__codelineno-24-9" name="__codelineno-24-9" href="#__codelineno-24-9"></a><span class="w"> </span><span class="nb">test</span><span class="w"> </span><span class="s2">&quot;robot has correct model&quot;</span><span class="w"> </span><span class="k">do</span>
3703
+ </span><span id="__span-24-10"><a id="__codelineno-24-10" name="__codelineno-24-10" href="#__codelineno-24-10"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3704
+ </span><span id="__span-24-11"><a id="__codelineno-24-11" name="__codelineno-24-11" href="#__codelineno-24-11"></a><span class="w"> </span><span class="n">assert_equal</span><span class="w"> </span><span class="s2">&quot;claude-sonnet-4&quot;</span><span class="p">,</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">model</span>
3705
+ </span><span id="__span-24-12"><a id="__codelineno-24-12" name="__codelineno-24-12" href="#__codelineno-24-12"></a><span class="w"> </span><span class="k">end</span>
3706
+ </span><span id="__span-24-13"><a id="__codelineno-24-13" name="__codelineno-24-13" href="#__codelineno-24-13"></a>
3707
+ </span><span id="__span-24-14"><a id="__codelineno-24-14" name="__codelineno-24-14" href="#__codelineno-24-14"></a><span class="w"> </span><span class="nb">test</span><span class="w"> </span><span class="s2">&quot;robot has local tools&quot;</span><span class="w"> </span><span class="k">do</span>
3708
+ </span><span id="__span-24-15"><a id="__codelineno-24-15" name="__codelineno-24-15" href="#__codelineno-24-15"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3709
+ </span><span id="__span-24-16"><a id="__codelineno-24-16" name="__codelineno-24-16" href="#__codelineno-24-16"></a><span class="w"> </span><span class="n">tool_names</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">local_tools</span><span class="o">.</span><span class="n">map</span><span class="p">(</span><span class="o">&amp;</span><span class="ss">:name</span><span class="p">)</span>
3710
+ </span><span id="__span-24-17"><a id="__codelineno-24-17" name="__codelineno-24-17" href="#__codelineno-24-17"></a><span class="w"> </span><span class="n">assert_includes</span><span class="w"> </span><span class="n">tool_names</span><span class="p">,</span><span class="w"> </span><span class="s2">&quot;order_lookup&quot;</span>
3711
+ </span><span id="__span-24-18"><a id="__codelineno-24-18" name="__codelineno-24-18" href="#__codelineno-24-18"></a><span class="w"> </span><span class="k">end</span>
3712
+ </span><span id="__span-24-19"><a id="__codelineno-24-19" name="__codelineno-24-19" href="#__codelineno-24-19"></a><span class="k">end</span>
3713
+ </span></code></pre></div>
3714
+ <h3 id="integration-tests">Integration Tests<a class="headerlink" href="#integration-tests" title="Permanent link">&para;</a></h3>
3715
+ <div class="language-ruby highlight"><span class="filename">test/integration/chat_test.rb</span><pre><span></span><code><span id="__span-25-1"><a id="__codelineno-25-1" name="__codelineno-25-1" href="#__codelineno-25-1"></a><span class="nb">require</span><span class="w"> </span><span class="s2">&quot;test_helper&quot;</span>
3716
+ </span><span id="__span-25-2"><a id="__codelineno-25-2" name="__codelineno-25-2" href="#__codelineno-25-2"></a>
3717
+ </span><span id="__span-25-3"><a id="__codelineno-25-3" name="__codelineno-25-3" href="#__codelineno-25-3"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatTest</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ActionDispatch</span><span class="o">::</span><span class="no">IntegrationTest</span>
3718
+ </span><span id="__span-25-4"><a id="__codelineno-25-4" name="__codelineno-25-4" href="#__codelineno-25-4"></a><span class="w"> </span><span class="nb">test</span><span class="w"> </span><span class="s2">&quot;processes chat message&quot;</span><span class="w"> </span><span class="k">do</span>
3719
+ </span><span id="__span-25-5"><a id="__codelineno-25-5" name="__codelineno-25-5" href="#__codelineno-25-5"></a><span class="w"> </span><span class="no">VCR</span><span class="o">.</span><span class="n">use_cassette</span><span class="p">(</span><span class="s2">&quot;chat_response&quot;</span><span class="p">)</span><span class="w"> </span><span class="k">do</span>
3720
+ </span><span id="__span-25-6"><a id="__codelineno-25-6" name="__codelineno-25-6" href="#__codelineno-25-6"></a><span class="w"> </span><span class="n">post</span><span class="w"> </span><span class="n">chat_path</span><span class="p">,</span><span class="w"> </span><span class="ss">params</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">message</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;Hello&quot;</span><span class="w"> </span><span class="p">}</span>
3721
+ </span><span id="__span-25-7"><a id="__codelineno-25-7" name="__codelineno-25-7" href="#__codelineno-25-7"></a><span class="w"> </span><span class="n">assert_response</span><span class="w"> </span><span class="ss">:success</span>
3722
+ </span><span id="__span-25-8"><a id="__codelineno-25-8" name="__codelineno-25-8" href="#__codelineno-25-8"></a>
3723
+ </span><span id="__span-25-9"><a id="__codelineno-25-9" name="__codelineno-25-9" href="#__codelineno-25-9"></a><span class="w"> </span><span class="n">json</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">JSON</span><span class="o">.</span><span class="n">parse</span><span class="p">(</span><span class="n">response</span><span class="o">.</span><span class="n">body</span><span class="p">)</span>
3724
+ </span><span id="__span-25-10"><a id="__codelineno-25-10" name="__codelineno-25-10" href="#__codelineno-25-10"></a><span class="w"> </span><span class="n">assert</span><span class="w"> </span><span class="n">json</span><span class="o">[</span><span class="s2">&quot;response&quot;</span><span class="o">].</span><span class="n">present?</span>
3725
+ </span><span id="__span-25-11"><a id="__codelineno-25-11" name="__codelineno-25-11" href="#__codelineno-25-11"></a><span class="w"> </span><span class="k">end</span>
3726
+ </span><span id="__span-25-12"><a id="__codelineno-25-12" name="__codelineno-25-12" href="#__codelineno-25-12"></a><span class="w"> </span><span class="k">end</span>
3727
+ </span><span id="__span-25-13"><a id="__codelineno-25-13" name="__codelineno-25-13" href="#__codelineno-25-13"></a><span class="k">end</span>
3728
+ </span></code></pre></div>
3729
+ <h2 id="models">Models<a class="headerlink" href="#models" title="Permanent link">&para;</a></h2>
3730
+ <h3 id="thread-model">Thread Model<a class="headerlink" href="#thread-model" title="Permanent link">&para;</a></h3>
3731
+ <div class="language-ruby highlight"><span class="filename">app/models/robot_lab_thread.rb</span><pre><span></span><code><span id="__span-26-1"><a id="__codelineno-26-1" name="__codelineno-26-1" href="#__codelineno-26-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">RobotLabThread</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationRecord</span>
3732
+ </span><span id="__span-26-2"><a id="__codelineno-26-2" name="__codelineno-26-2" href="#__codelineno-26-2"></a><span class="w"> </span><span class="n">has_many</span><span class="w"> </span><span class="ss">:results</span><span class="p">,</span>
3733
+ </span><span id="__span-26-3"><a id="__codelineno-26-3" name="__codelineno-26-3" href="#__codelineno-26-3"></a><span class="w"> </span><span class="ss">class_name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;RobotLabResult&quot;</span><span class="p">,</span>
3734
+ </span><span id="__span-26-4"><a id="__codelineno-26-4" name="__codelineno-26-4" href="#__codelineno-26-4"></a><span class="w"> </span><span class="ss">foreign_key</span><span class="p">:</span><span class="w"> </span><span class="ss">:session_id</span><span class="p">,</span>
3735
+ </span><span id="__span-26-5"><a id="__codelineno-26-5" name="__codelineno-26-5" href="#__codelineno-26-5"></a><span class="w"> </span><span class="ss">primary_key</span><span class="p">:</span><span class="w"> </span><span class="ss">:session_id</span><span class="p">,</span>
3736
+ </span><span id="__span-26-6"><a id="__codelineno-26-6" name="__codelineno-26-6" href="#__codelineno-26-6"></a><span class="w"> </span><span class="ss">dependent</span><span class="p">:</span><span class="w"> </span><span class="ss">:destroy</span>
3737
+ </span><span id="__span-26-7"><a id="__codelineno-26-7" name="__codelineno-26-7" href="#__codelineno-26-7"></a>
3738
+ </span><span id="__span-26-8"><a id="__codelineno-26-8" name="__codelineno-26-8" href="#__codelineno-26-8"></a><span class="w"> </span><span class="n">validates</span><span class="w"> </span><span class="ss">:session_id</span><span class="p">,</span><span class="w"> </span><span class="ss">presence</span><span class="p">:</span><span class="w"> </span><span class="kp">true</span><span class="p">,</span><span class="w"> </span><span class="ss">uniqueness</span><span class="p">:</span><span class="w"> </span><span class="kp">true</span>
3739
+ </span><span id="__span-26-9"><a id="__codelineno-26-9" name="__codelineno-26-9" href="#__codelineno-26-9"></a>
3740
+ </span><span id="__span-26-10"><a id="__codelineno-26-10" name="__codelineno-26-10" href="#__codelineno-26-10"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nc">self</span><span class="o">.</span><span class="nf">find_or_create_by_session_id</span><span class="p">(</span><span class="nb">id</span><span class="p">)</span>
3741
+ </span><span id="__span-26-11"><a id="__codelineno-26-11" name="__codelineno-26-11" href="#__codelineno-26-11"></a><span class="w"> </span><span class="n">find_or_create_by</span><span class="p">(</span><span class="ss">session_id</span><span class="p">:</span><span class="w"> </span><span class="nb">id</span><span class="p">)</span>
3742
+ </span><span id="__span-26-12"><a id="__codelineno-26-12" name="__codelineno-26-12" href="#__codelineno-26-12"></a><span class="w"> </span><span class="k">end</span>
3743
+ </span><span id="__span-26-13"><a id="__codelineno-26-13" name="__codelineno-26-13" href="#__codelineno-26-13"></a>
3744
+ </span><span id="__span-26-14"><a id="__codelineno-26-14" name="__codelineno-26-14" href="#__codelineno-26-14"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">last_result</span>
3745
+ </span><span id="__span-26-15"><a id="__codelineno-26-15" name="__codelineno-26-15" href="#__codelineno-26-15"></a><span class="w"> </span><span class="n">results</span><span class="o">.</span><span class="n">order</span><span class="p">(</span><span class="ss">sequence_number</span><span class="p">:</span><span class="w"> </span><span class="ss">:desc</span><span class="p">)</span><span class="o">.</span><span class="n">first</span>
3746
+ </span><span id="__span-26-16"><a id="__codelineno-26-16" name="__codelineno-26-16" href="#__codelineno-26-16"></a><span class="w"> </span><span class="k">end</span>
3747
+ </span><span id="__span-26-17"><a id="__codelineno-26-17" name="__codelineno-26-17" href="#__codelineno-26-17"></a><span class="k">end</span>
3748
+ </span></code></pre></div>
3749
+ <h3 id="result-model">Result Model<a class="headerlink" href="#result-model" title="Permanent link">&para;</a></h3>
3750
+ <div class="language-ruby highlight"><span class="filename">app/models/robot_lab_result.rb</span><pre><span></span><code><span id="__span-27-1"><a id="__codelineno-27-1" name="__codelineno-27-1" href="#__codelineno-27-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">RobotLabResult</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationRecord</span>
3751
+ </span><span id="__span-27-2"><a id="__codelineno-27-2" name="__codelineno-27-2" href="#__codelineno-27-2"></a><span class="w"> </span><span class="n">belongs_to</span><span class="w"> </span><span class="ss">:thread</span><span class="p">,</span>
3752
+ </span><span id="__span-27-3"><a id="__codelineno-27-3" name="__codelineno-27-3" href="#__codelineno-27-3"></a><span class="w"> </span><span class="ss">class_name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;RobotLabThread&quot;</span><span class="p">,</span>
3753
+ </span><span id="__span-27-4"><a id="__codelineno-27-4" name="__codelineno-27-4" href="#__codelineno-27-4"></a><span class="w"> </span><span class="ss">foreign_key</span><span class="p">:</span><span class="w"> </span><span class="ss">:session_id</span><span class="p">,</span>
3754
+ </span><span id="__span-27-5"><a id="__codelineno-27-5" name="__codelineno-27-5" href="#__codelineno-27-5"></a><span class="w"> </span><span class="ss">primary_key</span><span class="p">:</span><span class="w"> </span><span class="ss">:session_id</span>
3755
+ </span><span id="__span-27-6"><a id="__codelineno-27-6" name="__codelineno-27-6" href="#__codelineno-27-6"></a>
3756
+ </span><span id="__span-27-7"><a id="__codelineno-27-7" name="__codelineno-27-7" href="#__codelineno-27-7"></a><span class="w"> </span><span class="n">validates</span><span class="w"> </span><span class="ss">:session_id</span><span class="p">,</span><span class="w"> </span><span class="ss">presence</span><span class="p">:</span><span class="w"> </span><span class="kp">true</span>
3757
+ </span><span id="__span-27-8"><a id="__codelineno-27-8" name="__codelineno-27-8" href="#__codelineno-27-8"></a><span class="w"> </span><span class="n">validates</span><span class="w"> </span><span class="ss">:robot_name</span><span class="p">,</span><span class="w"> </span><span class="ss">presence</span><span class="p">:</span><span class="w"> </span><span class="kp">true</span>
3758
+ </span><span id="__span-27-9"><a id="__codelineno-27-9" name="__codelineno-27-9" href="#__codelineno-27-9"></a>
3759
+ </span><span id="__span-27-10"><a id="__codelineno-27-10" name="__codelineno-27-10" href="#__codelineno-27-10"></a><span class="w"> </span><span class="n">default_scope</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="n">order</span><span class="p">(</span><span class="ss">sequence_number</span><span class="p">:</span><span class="w"> </span><span class="ss">:asc</span><span class="p">)</span><span class="w"> </span><span class="p">}</span>
3760
+ </span><span id="__span-27-11"><a id="__codelineno-27-11" name="__codelineno-27-11" href="#__codelineno-27-11"></a>
3761
+ </span><span id="__span-27-12"><a id="__codelineno-27-12" name="__codelineno-27-12" href="#__codelineno-27-12"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">to_robot_result</span>
3762
+ </span><span id="__span-27-13"><a id="__codelineno-27-13" name="__codelineno-27-13" href="#__codelineno-27-13"></a><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">RobotResult</span><span class="o">.</span><span class="n">new</span><span class="p">(</span>
3763
+ </span><span id="__span-27-14"><a id="__codelineno-27-14" name="__codelineno-27-14" href="#__codelineno-27-14"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">robot_name</span><span class="p">,</span>
3764
+ </span><span id="__span-27-15"><a id="__codelineno-27-15" name="__codelineno-27-15" href="#__codelineno-27-15"></a><span class="w"> </span><span class="ss">output</span><span class="p">:</span><span class="w"> </span><span class="p">(</span><span class="n">output_data</span><span class="w"> </span><span class="o">||</span><span class="w"> </span><span class="o">[]</span><span class="p">)</span><span class="o">.</span><span class="n">map</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="o">|</span><span class="n">d</span><span class="o">|</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Message</span><span class="o">.</span><span class="n">from_hash</span><span class="p">(</span><span class="n">d</span><span class="o">.</span><span class="n">symbolize_keys</span><span class="p">)</span><span class="w"> </span><span class="p">},</span>
3765
+ </span><span id="__span-27-16"><a id="__codelineno-27-16" name="__codelineno-27-16" href="#__codelineno-27-16"></a><span class="w"> </span><span class="ss">tool_calls</span><span class="p">:</span><span class="w"> </span><span class="p">(</span><span class="n">tool_calls_data</span><span class="w"> </span><span class="o">||</span><span class="w"> </span><span class="o">[]</span><span class="p">)</span><span class="o">.</span><span class="n">map</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="o">|</span><span class="n">d</span><span class="o">|</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Message</span><span class="o">.</span><span class="n">from_hash</span><span class="p">(</span><span class="n">d</span><span class="o">.</span><span class="n">symbolize_keys</span><span class="p">)</span><span class="w"> </span><span class="p">},</span>
3766
+ </span><span id="__span-27-17"><a id="__codelineno-27-17" name="__codelineno-27-17" href="#__codelineno-27-17"></a><span class="w"> </span><span class="ss">stop_reason</span><span class="p">:</span><span class="w"> </span><span class="n">stop_reason</span>
3767
+ </span><span id="__span-27-18"><a id="__codelineno-27-18" name="__codelineno-27-18" href="#__codelineno-27-18"></a><span class="w"> </span><span class="p">)</span>
3768
+ </span><span id="__span-27-19"><a id="__codelineno-27-19" name="__codelineno-27-19" href="#__codelineno-27-19"></a><span class="w"> </span><span class="k">end</span>
3769
+ </span><span id="__span-27-20"><a id="__codelineno-27-20" name="__codelineno-27-20" href="#__codelineno-27-20"></a><span class="k">end</span>
3770
+ </span></code></pre></div>
3771
+ <h2 id="best-practices">Best Practices<a class="headerlink" href="#best-practices" title="Permanent link">&para;</a></h2>
3772
+ <h3 id="1-use-service-objects">1. Use Service Objects<a class="headerlink" href="#1-use-service-objects" title="Permanent link">&para;</a></h3>
3773
+ <div class="language-ruby highlight"><span class="filename">app/services/chat_service.rb</span><pre><span></span><code><span id="__span-28-1"><a id="__codelineno-28-1" name="__codelineno-28-1" href="#__codelineno-28-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatService</span>
3774
+ </span><span id="__span-28-2"><a id="__codelineno-28-2" name="__codelineno-28-2" href="#__codelineno-28-2"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">initialize</span><span class="p">(</span><span class="ss">user</span><span class="p">:)</span>
3775
+ </span><span id="__span-28-3"><a id="__codelineno-28-3" name="__codelineno-28-3" href="#__codelineno-28-3"></a><span class="w"> </span><span class="vi">@user</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">user</span>
3776
+ </span><span id="__span-28-4"><a id="__codelineno-28-4" name="__codelineno-28-4" href="#__codelineno-28-4"></a><span class="w"> </span><span class="k">end</span>
3777
+ </span><span id="__span-28-5"><a id="__codelineno-28-5" name="__codelineno-28-5" href="#__codelineno-28-5"></a>
3778
+ </span><span id="__span-28-6"><a id="__codelineno-28-6" name="__codelineno-28-6" href="#__codelineno-28-6"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">process</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3779
+ </span><span id="__span-28-7"><a id="__codelineno-28-7" name="__codelineno-28-7" href="#__codelineno-28-7"></a><span class="w"> </span><span class="n">robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3780
+ </span><span id="__span-28-8"><a id="__codelineno-28-8" name="__codelineno-28-8" href="#__codelineno-28-8"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">robot</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3781
+ </span><span id="__span-28-9"><a id="__codelineno-28-9" name="__codelineno-28-9" href="#__codelineno-28-9"></a>
3782
+ </span><span id="__span-28-10"><a id="__codelineno-28-10" name="__codelineno-28-10" href="#__codelineno-28-10"></a><span class="w"> </span><span class="p">{</span>
3783
+ </span><span id="__span-28-11"><a id="__codelineno-28-11" name="__codelineno-28-11" href="#__codelineno-28-11"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3784
+ </span><span id="__span-28-12"><a id="__codelineno-28-12" name="__codelineno-28-12" href="#__codelineno-28-12"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">robot_name</span>
3785
+ </span><span id="__span-28-13"><a id="__codelineno-28-13" name="__codelineno-28-13" href="#__codelineno-28-13"></a><span class="w"> </span><span class="p">}</span>
3786
+ </span><span id="__span-28-14"><a id="__codelineno-28-14" name="__codelineno-28-14" href="#__codelineno-28-14"></a><span class="w"> </span><span class="k">end</span>
3787
+ </span><span id="__span-28-15"><a id="__codelineno-28-15" name="__codelineno-28-15" href="#__codelineno-28-15"></a>
3788
+ </span><span id="__span-28-16"><a id="__codelineno-28-16" name="__codelineno-28-16" href="#__codelineno-28-16"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">process_with_network</span><span class="p">(</span><span class="n">message</span><span class="p">)</span>
3789
+ </span><span id="__span-28-17"><a id="__codelineno-28-17" name="__codelineno-28-17" href="#__codelineno-28-17"></a><span class="w"> </span><span class="n">support_robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">SupportRobot</span><span class="o">.</span><span class="n">build</span>
3790
+ </span><span id="__span-28-18"><a id="__codelineno-28-18" name="__codelineno-28-18" href="#__codelineno-28-18"></a><span class="w"> </span><span class="n">billing_robot</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">BillingRobot</span><span class="o">.</span><span class="n">build</span>
3791
+ </span><span id="__span-28-19"><a id="__codelineno-28-19" name="__codelineno-28-19" href="#__codelineno-28-19"></a>
3792
+ </span><span id="__span-28-20"><a id="__codelineno-28-20" name="__codelineno-28-20" href="#__codelineno-28-20"></a><span class="w"> </span><span class="n">network</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">.</span><span class="n">create_network</span><span class="p">(</span><span class="nb">name</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;customer_service&quot;</span><span class="p">)</span><span class="w"> </span><span class="k">do</span>
3793
+ </span><span id="__span-28-21"><a id="__codelineno-28-21" name="__codelineno-28-21" href="#__codelineno-28-21"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:support</span><span class="p">,</span><span class="w"> </span><span class="n">support_robot</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:none</span>
3794
+ </span><span id="__span-28-22"><a id="__codelineno-28-22" name="__codelineno-28-22" href="#__codelineno-28-22"></a><span class="w"> </span><span class="n">task</span><span class="w"> </span><span class="ss">:billing</span><span class="p">,</span><span class="w"> </span><span class="n">billing_robot</span><span class="p">,</span><span class="w"> </span><span class="ss">depends_on</span><span class="p">:</span><span class="w"> </span><span class="ss">:optional</span>
3795
+ </span><span id="__span-28-23"><a id="__codelineno-28-23" name="__codelineno-28-23" href="#__codelineno-28-23"></a><span class="w"> </span><span class="k">end</span>
3796
+ </span><span id="__span-28-24"><a id="__codelineno-28-24" name="__codelineno-28-24" href="#__codelineno-28-24"></a>
3797
+ </span><span id="__span-28-25"><a id="__codelineno-28-25" name="__codelineno-28-25" href="#__codelineno-28-25"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="n">network</span><span class="o">.</span><span class="n">run</span><span class="p">(</span><span class="ss">message</span><span class="p">:</span><span class="w"> </span><span class="n">message</span><span class="p">,</span><span class="w"> </span><span class="ss">user_id</span><span class="p">:</span><span class="w"> </span><span class="vi">@user</span><span class="o">.</span><span class="n">id</span><span class="p">)</span>
3798
+ </span><span id="__span-28-26"><a id="__codelineno-28-26" name="__codelineno-28-26" href="#__codelineno-28-26"></a>
3799
+ </span><span id="__span-28-27"><a id="__codelineno-28-27" name="__codelineno-28-27" href="#__codelineno-28-27"></a><span class="w"> </span><span class="p">{</span>
3800
+ </span><span id="__span-28-28"><a id="__codelineno-28-28" name="__codelineno-28-28" href="#__codelineno-28-28"></a><span class="w"> </span><span class="ss">response</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">value</span><span class="o">.</span><span class="n">last_text_content</span><span class="p">,</span>
3801
+ </span><span id="__span-28-29"><a id="__codelineno-28-29" name="__codelineno-28-29" href="#__codelineno-28-29"></a><span class="w"> </span><span class="ss">robot_name</span><span class="p">:</span><span class="w"> </span><span class="n">result</span><span class="o">.</span><span class="n">value</span><span class="o">.</span><span class="n">robot_name</span>
3802
+ </span><span id="__span-28-30"><a id="__codelineno-28-30" name="__codelineno-28-30" href="#__codelineno-28-30"></a><span class="w"> </span><span class="p">}</span>
3803
+ </span><span id="__span-28-31"><a id="__codelineno-28-31" name="__codelineno-28-31" href="#__codelineno-28-31"></a><span class="w"> </span><span class="k">end</span>
3804
+ </span><span id="__span-28-32"><a id="__codelineno-28-32" name="__codelineno-28-32" href="#__codelineno-28-32"></a><span class="k">end</span>
3805
+ </span></code></pre></div>
3806
+ <h3 id="2-handle-errors">2. Handle Errors<a class="headerlink" href="#2-handle-errors" title="Permanent link">&para;</a></h3>
3807
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-29-1"><a id="__codelineno-29-1" name="__codelineno-29-1" href="#__codelineno-29-1"></a><span class="k">def</span><span class="w"> </span><span class="nf">create</span>
3808
+ </span><span id="__span-29-2"><a id="__codelineno-29-2" name="__codelineno-29-2" href="#__codelineno-29-2"></a><span class="w"> </span><span class="n">result</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">ChatService</span><span class="o">.</span><span class="n">new</span><span class="p">(</span><span class="ss">user</span><span class="p">:</span><span class="w"> </span><span class="n">current_user</span><span class="p">)</span><span class="o">.</span><span class="n">process</span><span class="p">(</span><span class="n">params</span><span class="o">[</span><span class="ss">:message</span><span class="o">]</span><span class="p">)</span>
3809
+ </span><span id="__span-29-3"><a id="__codelineno-29-3" name="__codelineno-29-3" href="#__codelineno-29-3"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="n">result</span>
3810
+ </span><span id="__span-29-4"><a id="__codelineno-29-4" name="__codelineno-29-4" href="#__codelineno-29-4"></a><span class="k">rescue</span><span class="w"> </span><span class="no">RobotLab</span><span class="o">::</span><span class="no">Error</span><span class="w"> </span><span class="o">=&gt;</span><span class="w"> </span><span class="n">e</span>
3811
+ </span><span id="__span-29-5"><a id="__codelineno-29-5" name="__codelineno-29-5" href="#__codelineno-29-5"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">error</span><span class="p">:</span><span class="w"> </span><span class="n">e</span><span class="o">.</span><span class="n">message</span><span class="w"> </span><span class="p">},</span><span class="w"> </span><span class="ss">status</span><span class="p">:</span><span class="w"> </span><span class="ss">:unprocessable_entity</span>
3812
+ </span><span id="__span-29-6"><a id="__codelineno-29-6" name="__codelineno-29-6" href="#__codelineno-29-6"></a><span class="k">rescue</span><span class="w"> </span><span class="no">StandardError</span><span class="w"> </span><span class="o">=&gt;</span><span class="w"> </span><span class="n">e</span>
3813
+ </span><span id="__span-29-7"><a id="__codelineno-29-7" name="__codelineno-29-7" href="#__codelineno-29-7"></a><span class="w"> </span><span class="no">Rails</span><span class="o">.</span><span class="n">logger</span><span class="o">.</span><span class="n">error</span><span class="p">(</span><span class="s2">&quot;Chat error: </span><span class="si">#{</span><span class="n">e</span><span class="o">.</span><span class="n">message</span><span class="si">}</span><span class="s2">&quot;</span><span class="p">)</span>
3814
+ </span><span id="__span-29-8"><a id="__codelineno-29-8" name="__codelineno-29-8" href="#__codelineno-29-8"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">error</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;An error occurred&quot;</span><span class="w"> </span><span class="p">},</span><span class="w"> </span><span class="ss">status</span><span class="p">:</span><span class="w"> </span><span class="ss">:internal_server_error</span>
3815
+ </span><span id="__span-29-9"><a id="__codelineno-29-9" name="__codelineno-29-9" href="#__codelineno-29-9"></a><span class="k">end</span>
3816
+ </span></code></pre></div>
3817
+ <h3 id="3-rate-limiting">3. Rate Limiting<a class="headerlink" href="#3-rate-limiting" title="Permanent link">&para;</a></h3>
3818
+ <div class="language-ruby highlight"><pre><span></span><code><span id="__span-30-1"><a id="__codelineno-30-1" name="__codelineno-30-1" href="#__codelineno-30-1"></a><span class="k">class</span><span class="w"> </span><span class="nc">ChatController</span><span class="w"> </span><span class="o">&lt;</span><span class="w"> </span><span class="no">ApplicationController</span>
3819
+ </span><span id="__span-30-2"><a id="__codelineno-30-2" name="__codelineno-30-2" href="#__codelineno-30-2"></a><span class="w"> </span><span class="n">before_action</span><span class="w"> </span><span class="ss">:check_rate_limit</span>
3820
+ </span><span id="__span-30-3"><a id="__codelineno-30-3" name="__codelineno-30-3" href="#__codelineno-30-3"></a>
3821
+ </span><span id="__span-30-4"><a id="__codelineno-30-4" name="__codelineno-30-4" href="#__codelineno-30-4"></a><span class="w"> </span><span class="kp">private</span>
3822
+ </span><span id="__span-30-5"><a id="__codelineno-30-5" name="__codelineno-30-5" href="#__codelineno-30-5"></a>
3823
+ </span><span id="__span-30-6"><a id="__codelineno-30-6" name="__codelineno-30-6" href="#__codelineno-30-6"></a><span class="w"> </span><span class="k">def</span><span class="w"> </span><span class="nf">check_rate_limit</span>
3824
+ </span><span id="__span-30-7"><a id="__codelineno-30-7" name="__codelineno-30-7" href="#__codelineno-30-7"></a><span class="w"> </span><span class="n">key</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="s2">&quot;chat_rate:</span><span class="si">#{</span><span class="n">current_user</span><span class="o">.</span><span class="n">id</span><span class="si">}</span><span class="s2">&quot;</span>
3825
+ </span><span id="__span-30-8"><a id="__codelineno-30-8" name="__codelineno-30-8" href="#__codelineno-30-8"></a><span class="w"> </span><span class="n">count</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="no">Rails</span><span class="o">.</span><span class="n">cache</span><span class="o">.</span><span class="n">increment</span><span class="p">(</span><span class="n">key</span><span class="p">,</span><span class="w"> </span><span class="mi">1</span><span class="p">,</span><span class="w"> </span><span class="ss">expires_in</span><span class="p">:</span><span class="w"> </span><span class="mi">1</span><span class="o">.</span><span class="n">minute</span><span class="p">)</span>
3826
+ </span><span id="__span-30-9"><a id="__codelineno-30-9" name="__codelineno-30-9" href="#__codelineno-30-9"></a>
3827
+ </span><span id="__span-30-10"><a id="__codelineno-30-10" name="__codelineno-30-10" href="#__codelineno-30-10"></a><span class="w"> </span><span class="k">if</span><span class="w"> </span><span class="n">count</span><span class="w"> </span><span class="o">&gt;</span><span class="w"> </span><span class="mi">10</span>
3828
+ </span><span id="__span-30-11"><a id="__codelineno-30-11" name="__codelineno-30-11" href="#__codelineno-30-11"></a><span class="w"> </span><span class="n">render</span><span class="w"> </span><span class="ss">json</span><span class="p">:</span><span class="w"> </span><span class="p">{</span><span class="w"> </span><span class="ss">error</span><span class="p">:</span><span class="w"> </span><span class="s2">&quot;Rate limit exceeded&quot;</span><span class="w"> </span><span class="p">},</span><span class="w"> </span><span class="ss">status</span><span class="p">:</span><span class="w"> </span><span class="ss">:too_many_requests</span>
3829
+ </span><span id="__span-30-12"><a id="__codelineno-30-12" name="__codelineno-30-12" href="#__codelineno-30-12"></a><span class="w"> </span><span class="k">end</span>
3830
+ </span><span id="__span-30-13"><a id="__codelineno-30-13" name="__codelineno-30-13" href="#__codelineno-30-13"></a><span class="w"> </span><span class="k">end</span>
3831
+ </span><span id="__span-30-14"><a id="__codelineno-30-14" name="__codelineno-30-14" href="#__codelineno-30-14"></a><span class="k">end</span>
3832
+ </span></code></pre></div>
3833
+ <h2 id="next-steps">Next Steps<a class="headerlink" href="#next-steps" title="Permanent link">&para;</a></h2>
3834
+ <ul>
3835
+ <li><a href="../building-robots/">Building Robots</a> - Robot patterns</li>
3836
+ <li><a href="../creating-networks/">Creating Networks</a> - Network configuration</li>
3837
+ </ul>
3838
+
3839
+
3840
+
3841
+
3842
+
3843
+
3844
+
3845
+
3846
+
3847
+
3848
+
3849
+
3850
+
3851
+ <form class="md-feedback" name="feedback" hidden>
3852
+ <fieldset>
3853
+ <legend class="md-feedback__title">
3854
+ Was this page helpful?
3855
+ </legend>
3856
+ <div class="md-feedback__inner">
3857
+ <div class="md-feedback__list">
3858
+
3859
+ <button class="md-feedback__icon md-icon" type="submit" title="This page was helpful" data-md-value="1">
3860
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M20 12a8 8 0 0 0-8-8 8 8 0 0 0-8 8 8 8 0 0 0 8 8 8 8 0 0 0 8-8m2 0a10 10 0 0 1-10 10A10 10 0 0 1 2 12 10 10 0 0 1 12 2a10 10 0 0 1 10 10M10 9.5c0 .8-.7 1.5-1.5 1.5S7 10.3 7 9.5 7.7 8 8.5 8s1.5.7 1.5 1.5m7 0c0 .8-.7 1.5-1.5 1.5S14 10.3 14 9.5 14.7 8 15.5 8s1.5.7 1.5 1.5m-5 7.73c-1.75 0-3.29-.73-4.19-1.81L9.23 14c.45.72 1.52 1.23 2.77 1.23s2.32-.51 2.77-1.23l1.42 1.42c-.9 1.08-2.44 1.81-4.19 1.81"/></svg>
3861
+ </button>
3862
+
3863
+ <button class="md-feedback__icon md-icon" type="submit" title="This page could be improved" data-md-value="0">
3864
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M20 12a8 8 0 0 0-8-8 8 8 0 0 0-8 8 8 8 0 0 0 8 8 8 8 0 0 0 8-8m2 0a10 10 0 0 1-10 10A10 10 0 0 1 2 12 10 10 0 0 1 12 2a10 10 0 0 1 10 10m-6.5-4c.8 0 1.5.7 1.5 1.5s-.7 1.5-1.5 1.5-1.5-.7-1.5-1.5.7-1.5 1.5-1.5M10 9.5c0 .8-.7 1.5-1.5 1.5S7 10.3 7 9.5 7.7 8 8.5 8s1.5.7 1.5 1.5m2 4.5c1.75 0 3.29.72 4.19 1.81l-1.42 1.42C14.32 16.5 13.25 16 12 16s-2.32.5-2.77 1.23l-1.42-1.42C8.71 14.72 10.25 14 12 14"/></svg>
3865
+ </button>
3866
+
3867
+ </div>
3868
+ <div class="md-feedback__note">
3869
+
3870
+ <div data-md-value="1" hidden>
3871
+
3872
+
3873
+
3874
+
3875
+
3876
+
3877
+
3878
+
3879
+
3880
+ Thanks for your feedback!
3881
+ </div>
3882
+
3883
+ <div data-md-value="0" hidden>
3884
+
3885
+
3886
+
3887
+
3888
+
3889
+
3890
+
3891
+
3892
+
3893
+ Thanks for your feedback! Help us improve by creating an issue.
3894
+ </div>
3895
+
3896
+ </div>
3897
+ </div>
3898
+ </fieldset>
3899
+ </form>
3900
+
3901
+
3902
+
3903
+ </article>
3904
+ </div>
3905
+
3906
+
3907
+ <script>var tabs=__md_get("__tabs");if(Array.isArray(tabs))e:for(var set of document.querySelectorAll(".tabbed-set")){var labels=set.querySelector(".tabbed-labels");for(var tab of tabs)for(var label of labels.getElementsByTagName("label"))if(label.innerText.trim()===tab){var input=document.getElementById(label.htmlFor);input.checked=!0;continue e}}</script>
3908
+
3909
+ <script>var target=document.getElementById(location.hash.slice(1));target&&target.name&&(target.checked=target.name.startsWith("__tabbed_"))</script>
3910
+ </div>
3911
+
3912
+ <button type="button" class="md-top md-icon" data-md-component="top" hidden>
3913
+
3914
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M13 20h-2V8l-5.5 5.5-1.42-1.42L12 4.16l7.92 7.92-1.42 1.42L13 8z"/></svg>
3915
+ Back to top
3916
+ </button>
3917
+
3918
+ </main>
3919
+
3920
+ <footer class="md-footer">
3921
+
3922
+
3923
+
3924
+ <nav class="md-footer__inner md-grid" aria-label="Footer" >
3925
+
3926
+
3927
+ <a href="../memory/" class="md-footer__link md-footer__link--prev" aria-label="Previous: Memory System">
3928
+ <div class="md-footer__button md-icon">
3929
+
3930
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M20 11v2H8l5.5 5.5-1.42 1.42L4.16 12l7.92-7.92L13.5 5.5 8 11z"/></svg>
3931
+ </div>
3932
+ <div class="md-footer__title">
3933
+ <span class="md-footer__direction">
3934
+ Previous
3935
+ </span>
3936
+ <div class="md-ellipsis">
3937
+ Memory System
3938
+ </div>
3939
+ </div>
3940
+ </a>
3941
+
3942
+
3943
+
3944
+ <a href="../../api/" class="md-footer__link md-footer__link--next" aria-label="Next: API Reference">
3945
+ <div class="md-footer__title">
3946
+ <span class="md-footer__direction">
3947
+ Next
3948
+ </span>
3949
+ <div class="md-ellipsis">
3950
+ API Reference
3951
+ </div>
3952
+ </div>
3953
+ <div class="md-footer__button md-icon">
3954
+
3955
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M4 11v2h12l-5.5 5.5 1.42 1.42L19.84 12l-7.92-7.92L10.5 5.5 16 11z"/></svg>
3956
+ </div>
3957
+ </a>
3958
+
3959
+ </nav>
3960
+
3961
+
3962
+ <div class="md-footer-meta md-typeset">
3963
+ <div class="md-footer-meta__inner md-grid">
3964
+ <div class="md-copyright">
3965
+
3966
+ <div class="md-copyright__highlight">
3967
+ Copyright &copy; 2025 Dewayne VanHoozer
3968
+ </div>
3969
+
3970
+
3971
+ Made with
3972
+ <a href="https://squidfunk.github.io/mkdocs-material/" target="_blank" rel="noopener">
3973
+ Material for MkDocs
3974
+ </a>
3975
+
3976
+ </div>
3977
+
3978
+
3979
+ <div class="md-social">
3980
+
3981
+
3982
+
3983
+
3984
+
3985
+ <a href="https://github.com/madbomber/robot_lab" target="_blank" rel="noopener" title="RobotLab on GitHub" class="md-social__link">
3986
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><!--! Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2025 Fonticons, Inc.--><path d="M173.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3.3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6m-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5.3-6.2 2.3m44.2-1.7c-2.9.7-4.9 2.6-4.6 4.9.3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9M252.8 8C114.1 8 8 113.3 8 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C436.2 457.8 504 362.9 504 252 504 113.3 391.5 8 252.8 8M105.2 352.9c-1.3 1-1 3.3.7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1m-10.8-8.1c-.7 1.3.3 2.9 2.3 3.9 1.6 1 3.6.7 4.3-.7.7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3.7m32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3.7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1m-11.4-14.7c-1.6 1-1.6 3.6 0 5.9s4.3 3.3 5.6 2.3c1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2"/></svg>
3987
+ </a>
3988
+
3989
+
3990
+
3991
+
3992
+
3993
+ <a href="https://rubygems.org/gems/robot_lab" target="_blank" rel="noopener" title="RobotLab on RubyGems" class="md-social__link">
3994
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 512 512"><!--! Font Awesome Free 7.1.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License) Copyright 2025 Fonticons, Inc.--><path d="M116.7 33.8c4.5-6.1 11.7-9.8 19.3-9.8h240c7.6 0 14.8 3.6 19.3 9.8l112 152c6.8 9.2 6.1 21.9-1.5 30.4l-232 256c-4.5 5-11 7.9-17.8 7.9s-13.2-2.9-17.8-7.9l-232-256c-7.7-8.5-8.3-21.2-1.5-30.4zm38.5 39.8c-3.3 2.5-4.2 7-2.1 10.5l57.4 95.7L63.3 192c-4.1.3-7.3 3.8-7.3 8s3.2 7.6 7.3 8l192 16h1.3l192-16c4.1-.3 7.3-3.8 7.3-8s-3.2-7.6-7.3-8l-147.2-12.3 57.4-95.6c2.1-3.5 1.2-8.1-2.1-10.5s-7.9-2-10.7 1l-90 97.6-90.1-97.6c-2.8-3-7.4-3.4-10.7-1"/></svg>
3995
+ </a>
3996
+
3997
+ </div>
3998
+
3999
+ </div>
4000
+ </div>
4001
+ </footer>
4002
+
4003
+ </div>
4004
+ <div class="md-dialog" data-md-component="dialog">
4005
+ <div class="md-dialog__inner md-typeset"></div>
4006
+ </div>
4007
+
4008
+
4009
+
4010
+
4011
+
4012
+ <script id="__config" type="application/json">{"annotate": null, "base": "../..", "features": ["navigation.instant", "navigation.tracking", "navigation.tabs", "navigation.tabs.sticky", "navigation.path", "navigation.indexes", "navigation.top", "navigation.footer", "toc.follow", "search.suggest", "search.highlight", "search.share", "header.autohide", "content.code.copy", "content.code.annotate", "content.tabs.link", "content.tooltips", "content.action.edit", "content.action.view"], "search": "../../assets/javascripts/workers/search.2c215733.min.js", "tags": null, "translations": {"clipboard.copied": "Copied to clipboard", "clipboard.copy": "Copy to clipboard", "search.result.more.one": "1 more on this page", "search.result.more.other": "# more on this page", "search.result.none": "No matching documents", "search.result.one": "1 matching document", "search.result.other": "# matching documents", "search.result.placeholder": "Type to start searching", "search.result.term.missing": "Missing", "select.version": "Select version"}, "version": null}</script>
4013
+
4014
+
4015
+ <script src="../../assets/javascripts/bundle.79ae519e.min.js"></script>
4016
+
4017
+
4018
+ </body>
4019
+ </html>