@dewtech/dare-cli 3.0.0 → 3.1.0

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 (453) hide show
  1. package/README.md +45 -39
  2. package/dist/bin/dare.js +1 -3
  3. package/dist/bin/dare.js.map +1 -1
  4. package/dist/commands/__tests__/init.integration.spec.d.ts +2 -0
  5. package/dist/commands/__tests__/init.integration.spec.d.ts.map +1 -0
  6. package/dist/commands/__tests__/init.integration.spec.js +134 -0
  7. package/dist/commands/__tests__/init.integration.spec.js.map +1 -0
  8. package/dist/commands/init.d.ts.map +1 -1
  9. package/dist/commands/init.js +84 -1
  10. package/dist/commands/init.js.map +1 -1
  11. package/dist/commands/new.d.ts.map +1 -1
  12. package/dist/commands/new.js +2 -1
  13. package/dist/commands/new.js.map +1 -1
  14. package/dist/mcp-server/bin/server.js +0 -0
  15. package/dist/stacks/__tests__/dna-emitter.spec.d.ts +2 -0
  16. package/dist/stacks/__tests__/dna-emitter.spec.d.ts.map +1 -0
  17. package/dist/stacks/__tests__/dna-emitter.spec.js +207 -0
  18. package/dist/stacks/__tests__/dna-emitter.spec.js.map +1 -0
  19. package/dist/stacks/__tests__/dna.spec.d.ts +2 -0
  20. package/dist/stacks/__tests__/dna.spec.d.ts.map +1 -0
  21. package/dist/stacks/__tests__/dna.spec.js +211 -0
  22. package/dist/stacks/__tests__/dna.spec.js.map +1 -0
  23. package/dist/stacks/__tests__/parity-rails.fixture.json +228 -0
  24. package/dist/stacks/__tests__/parity-rails.spec.d.ts +2 -0
  25. package/dist/stacks/__tests__/parity-rails.spec.d.ts.map +1 -0
  26. package/dist/stacks/__tests__/parity-rails.spec.js +99 -0
  27. package/dist/stacks/__tests__/parity-rails.spec.js.map +1 -0
  28. package/dist/stacks/__tests__/registry.spec.d.ts +2 -0
  29. package/dist/stacks/__tests__/registry.spec.d.ts.map +1 -0
  30. package/dist/stacks/__tests__/registry.spec.js +101 -0
  31. package/dist/stacks/__tests__/registry.spec.js.map +1 -0
  32. package/dist/stacks/__tests__/template-engine.spec.d.ts +2 -0
  33. package/dist/stacks/__tests__/template-engine.spec.d.ts.map +1 -0
  34. package/dist/stacks/__tests__/template-engine.spec.js +149 -0
  35. package/dist/stacks/__tests__/template-engine.spec.js.map +1 -0
  36. package/dist/stacks/dna-emitter.d.ts +45 -0
  37. package/dist/stacks/dna-emitter.d.ts.map +1 -0
  38. package/dist/stacks/dna-emitter.js +267 -0
  39. package/dist/stacks/dna-emitter.js.map +1 -0
  40. package/dist/stacks/go-gin/__tests__/scaffold.spec.d.ts +2 -0
  41. package/dist/stacks/go-gin/__tests__/scaffold.spec.d.ts.map +1 -0
  42. package/dist/stacks/go-gin/__tests__/scaffold.spec.js +221 -0
  43. package/dist/stacks/go-gin/__tests__/scaffold.spec.js.map +1 -0
  44. package/dist/stacks/go-gin/scaffold.d.ts +3 -0
  45. package/dist/stacks/go-gin/scaffold.d.ts.map +1 -0
  46. package/dist/stacks/go-gin/scaffold.js +105 -0
  47. package/dist/stacks/go-gin/scaffold.js.map +1 -0
  48. package/dist/stacks/go-stdlib/__tests__/scaffold.spec.d.ts +2 -0
  49. package/dist/stacks/go-stdlib/__tests__/scaffold.spec.d.ts.map +1 -0
  50. package/dist/stacks/go-stdlib/__tests__/scaffold.spec.js +215 -0
  51. package/dist/stacks/go-stdlib/__tests__/scaffold.spec.js.map +1 -0
  52. package/dist/stacks/go-stdlib/scaffold.d.ts +3 -0
  53. package/dist/stacks/go-stdlib/scaffold.d.ts.map +1 -0
  54. package/dist/stacks/go-stdlib/scaffold.js +106 -0
  55. package/dist/stacks/go-stdlib/scaffold.js.map +1 -0
  56. package/dist/stacks/mcp-go/__tests__/scaffold.spec.d.ts +2 -0
  57. package/dist/stacks/mcp-go/__tests__/scaffold.spec.d.ts.map +1 -0
  58. package/dist/stacks/mcp-go/__tests__/scaffold.spec.js +203 -0
  59. package/dist/stacks/mcp-go/__tests__/scaffold.spec.js.map +1 -0
  60. package/dist/stacks/mcp-go/scaffold.d.ts +3 -0
  61. package/dist/stacks/mcp-go/scaffold.d.ts.map +1 -0
  62. package/dist/stacks/mcp-go/scaffold.js +94 -0
  63. package/dist/stacks/mcp-go/scaffold.js.map +1 -0
  64. package/dist/stacks/mcp-node-ts/__tests__/scaffold.spec.d.ts +2 -0
  65. package/dist/stacks/mcp-node-ts/__tests__/scaffold.spec.d.ts.map +1 -0
  66. package/dist/stacks/mcp-node-ts/__tests__/scaffold.spec.js +236 -0
  67. package/dist/stacks/mcp-node-ts/__tests__/scaffold.spec.js.map +1 -0
  68. package/dist/stacks/mcp-node-ts/scaffold.d.ts +3 -0
  69. package/dist/stacks/mcp-node-ts/scaffold.d.ts.map +1 -0
  70. package/dist/stacks/mcp-node-ts/scaffold.js +95 -0
  71. package/dist/stacks/mcp-node-ts/scaffold.js.map +1 -0
  72. package/dist/stacks/mcp-python/__tests__/scaffold.spec.d.ts +2 -0
  73. package/dist/stacks/mcp-python/__tests__/scaffold.spec.d.ts.map +1 -0
  74. package/dist/stacks/mcp-python/__tests__/scaffold.spec.js +228 -0
  75. package/dist/stacks/mcp-python/__tests__/scaffold.spec.js.map +1 -0
  76. package/dist/stacks/mcp-python/scaffold.d.ts +3 -0
  77. package/dist/stacks/mcp-python/scaffold.d.ts.map +1 -0
  78. package/dist/stacks/mcp-python/scaffold.js +98 -0
  79. package/dist/stacks/mcp-python/scaffold.js.map +1 -0
  80. package/dist/stacks/mcp-rust/__tests__/scaffold.spec.d.ts +2 -0
  81. package/dist/stacks/mcp-rust/__tests__/scaffold.spec.d.ts.map +1 -0
  82. package/dist/stacks/mcp-rust/__tests__/scaffold.spec.js +213 -0
  83. package/dist/stacks/mcp-rust/__tests__/scaffold.spec.js.map +1 -0
  84. package/dist/stacks/mcp-rust/scaffold.d.ts +3 -0
  85. package/dist/stacks/mcp-rust/scaffold.d.ts.map +1 -0
  86. package/dist/stacks/mcp-rust/scaffold.js +98 -0
  87. package/dist/stacks/mcp-rust/scaffold.js.map +1 -0
  88. package/dist/stacks/node-nestjs/__tests__/scaffold.spec.d.ts +2 -0
  89. package/dist/stacks/node-nestjs/__tests__/scaffold.spec.d.ts.map +1 -0
  90. package/dist/stacks/node-nestjs/__tests__/scaffold.spec.js +172 -0
  91. package/dist/stacks/node-nestjs/__tests__/scaffold.spec.js.map +1 -0
  92. package/dist/stacks/node-nestjs/scaffold.d.ts +3 -0
  93. package/dist/stacks/node-nestjs/scaffold.d.ts.map +1 -0
  94. package/dist/stacks/node-nestjs/scaffold.js +117 -0
  95. package/dist/stacks/node-nestjs/scaffold.js.map +1 -0
  96. package/dist/stacks/php-laravel/__tests__/scaffold.spec.d.ts +2 -0
  97. package/dist/stacks/php-laravel/__tests__/scaffold.spec.d.ts.map +1 -0
  98. package/dist/stacks/php-laravel/__tests__/scaffold.spec.js +205 -0
  99. package/dist/stacks/php-laravel/__tests__/scaffold.spec.js.map +1 -0
  100. package/dist/stacks/php-laravel/scaffold.d.ts +3 -0
  101. package/dist/stacks/php-laravel/scaffold.d.ts.map +1 -0
  102. package/dist/stacks/php-laravel/scaffold.js +109 -0
  103. package/dist/stacks/php-laravel/scaffold.js.map +1 -0
  104. package/dist/stacks/python-fastapi/__tests__/scaffold.spec.d.ts +2 -0
  105. package/dist/stacks/python-fastapi/__tests__/scaffold.spec.d.ts.map +1 -0
  106. package/dist/stacks/python-fastapi/__tests__/scaffold.spec.js +168 -0
  107. package/dist/stacks/python-fastapi/__tests__/scaffold.spec.js.map +1 -0
  108. package/dist/stacks/python-fastapi/scaffold.d.ts +3 -0
  109. package/dist/stacks/python-fastapi/scaffold.d.ts.map +1 -0
  110. package/dist/stacks/python-fastapi/scaffold.js +108 -0
  111. package/dist/stacks/python-fastapi/scaffold.js.map +1 -0
  112. package/dist/stacks/registry.d.ts +38 -0
  113. package/dist/stacks/registry.d.ts.map +1 -0
  114. package/dist/stacks/registry.js +153 -0
  115. package/dist/stacks/registry.js.map +1 -0
  116. package/dist/stacks/ruby-rails-8/__tests__/scaffold.spec.d.ts +6 -0
  117. package/dist/stacks/ruby-rails-8/__tests__/scaffold.spec.d.ts.map +1 -0
  118. package/dist/stacks/ruby-rails-8/__tests__/scaffold.spec.js +604 -0
  119. package/dist/stacks/ruby-rails-8/__tests__/scaffold.spec.js.map +1 -0
  120. package/dist/stacks/ruby-rails-8/scaffold.d.ts +91 -0
  121. package/dist/stacks/ruby-rails-8/scaffold.d.ts.map +1 -0
  122. package/dist/stacks/ruby-rails-8/scaffold.js +410 -0
  123. package/dist/stacks/ruby-rails-8/scaffold.js.map +1 -0
  124. package/dist/stacks/rust-axum/__tests__/scaffold.spec.d.ts +2 -0
  125. package/dist/stacks/rust-axum/__tests__/scaffold.spec.d.ts.map +1 -0
  126. package/dist/stacks/rust-axum/__tests__/scaffold.spec.js +203 -0
  127. package/dist/stacks/rust-axum/__tests__/scaffold.spec.js.map +1 -0
  128. package/dist/stacks/rust-axum/scaffold.d.ts +3 -0
  129. package/dist/stacks/rust-axum/scaffold.d.ts.map +1 -0
  130. package/dist/stacks/rust-axum/scaffold.js +105 -0
  131. package/dist/stacks/rust-axum/scaffold.js.map +1 -0
  132. package/dist/stacks/template-engine.d.ts +38 -0
  133. package/dist/stacks/template-engine.d.ts.map +1 -0
  134. package/dist/stacks/template-engine.js +134 -0
  135. package/dist/stacks/template-engine.js.map +1 -0
  136. package/dist/stacks/types.d.ts +69 -0
  137. package/dist/stacks/types.d.ts.map +1 -0
  138. package/dist/stacks/types.js +29 -0
  139. package/dist/stacks/types.js.map +1 -0
  140. package/dist/utils/project-generator.d.ts +1 -1
  141. package/dist/utils/project-generator.d.ts.map +1 -1
  142. package/dist/utils/project-generator.js +16 -7
  143. package/dist/utils/project-generator.js.map +1 -1
  144. package/dist/utils/stack-bootstrap.d.ts +3 -2
  145. package/dist/utils/stack-bootstrap.d.ts.map +1 -1
  146. package/dist/utils/stack-bootstrap.js +46 -16
  147. package/dist/utils/stack-bootstrap.js.map +1 -1
  148. package/package.json +91 -87
  149. package/templates/stacks/go-gin/.dare/skills.yml +11 -0
  150. package/templates/stacks/go-gin/.env.example +24 -0
  151. package/templates/stacks/go-gin/.github/workflows/dare-ci.yml +42 -0
  152. package/templates/stacks/go-gin/README.md.tpl +38 -0
  153. package/templates/stacks/go-gin/cmd/server/main.go.tpl +78 -0
  154. package/templates/stacks/go-gin/db/migrations/0001_create_users.down.sql +2 -0
  155. package/templates/stacks/go-gin/db/migrations/0001_create_users.up.sql +12 -0
  156. package/templates/stacks/go-gin/db/queries/users.sql +23 -0
  157. package/templates/stacks/go-gin/gitignore +7 -0
  158. package/templates/stacks/go-gin/go.mod.tpl +17 -0
  159. package/templates/stacks/go-gin/internal/config/config.go +41 -0
  160. package/templates/stacks/go-gin/internal/db/postgres.go.tpl +25 -0
  161. package/templates/stacks/go-gin/internal/handler/auth_handler.go.tpl +72 -0
  162. package/templates/stacks/go-gin/internal/handler/users_handler.go.tpl +72 -0
  163. package/templates/stacks/go-gin/internal/handler/ws_handler.go +37 -0
  164. package/templates/stacks/go-gin/internal/llm/dummy.go +14 -0
  165. package/templates/stacks/go-gin/internal/llm/provider.go +8 -0
  166. package/templates/stacks/go-gin/internal/middleware/jwt.go.tpl +58 -0
  167. package/templates/stacks/go-gin/internal/middleware/rate_limit.go +55 -0
  168. package/templates/stacks/go-gin/internal/model/user.go +17 -0
  169. package/templates/stacks/go-gin/internal/repository/users_repository.go.tpl +79 -0
  170. package/templates/stacks/go-gin/internal/service/auth_service.go.tpl +55 -0
  171. package/templates/stacks/go-gin/internal/service/users_service.go.tpl +53 -0
  172. package/templates/stacks/go-gin/llms.txt.tpl +54 -0
  173. package/templates/stacks/go-gin/openapi.json.tpl +46 -0
  174. package/templates/stacks/go-gin/sqlc.yaml +14 -0
  175. package/templates/stacks/go-gin/tests/smoke_test.go.tpl +22 -0
  176. package/templates/stacks/go-stdlib/.dare/skills.yml +11 -0
  177. package/templates/stacks/go-stdlib/.env.example +24 -0
  178. package/templates/stacks/go-stdlib/.github/workflows/dare-ci.yml +42 -0
  179. package/templates/stacks/go-stdlib/README.md.tpl +41 -0
  180. package/templates/stacks/go-stdlib/cmd/server/main.go.tpl +82 -0
  181. package/templates/stacks/go-stdlib/db/migrations/0001_create_users.down.sql +2 -0
  182. package/templates/stacks/go-stdlib/db/migrations/0001_create_users.up.sql +12 -0
  183. package/templates/stacks/go-stdlib/db/queries/users.sql +23 -0
  184. package/templates/stacks/go-stdlib/gitignore +6 -0
  185. package/templates/stacks/go-stdlib/go.mod.tpl +15 -0
  186. package/templates/stacks/go-stdlib/internal/config/config.go +41 -0
  187. package/templates/stacks/go-stdlib/internal/db/postgres.go.tpl +24 -0
  188. package/templates/stacks/go-stdlib/internal/handler/auth_handler.go.tpl +71 -0
  189. package/templates/stacks/go-stdlib/internal/handler/users_handler.go.tpl +84 -0
  190. package/templates/stacks/go-stdlib/internal/handler/ws_handler.go +36 -0
  191. package/templates/stacks/go-stdlib/internal/httpx/json.go +32 -0
  192. package/templates/stacks/go-stdlib/internal/llm/dummy.go +14 -0
  193. package/templates/stacks/go-stdlib/internal/llm/provider.go +8 -0
  194. package/templates/stacks/go-stdlib/internal/middleware/chain.go +21 -0
  195. package/templates/stacks/go-stdlib/internal/middleware/cors.go +27 -0
  196. package/templates/stacks/go-stdlib/internal/middleware/jwt.go.tpl +51 -0
  197. package/templates/stacks/go-stdlib/internal/middleware/rate_limit.go +81 -0
  198. package/templates/stacks/go-stdlib/internal/model/user.go +17 -0
  199. package/templates/stacks/go-stdlib/internal/repository/users_repository.go.tpl +75 -0
  200. package/templates/stacks/go-stdlib/internal/service/auth_service.go.tpl +55 -0
  201. package/templates/stacks/go-stdlib/internal/service/users_service.go.tpl +53 -0
  202. package/templates/stacks/go-stdlib/llms.txt.tpl +60 -0
  203. package/templates/stacks/go-stdlib/openapi.json.tpl +46 -0
  204. package/templates/stacks/go-stdlib/sqlc.yaml +14 -0
  205. package/templates/stacks/go-stdlib/tests/smoke_test.go.tpl +45 -0
  206. package/templates/stacks/mcp-go/.dare/skills.yml +8 -0
  207. package/templates/stacks/mcp-go/.env.example +14 -0
  208. package/templates/stacks/mcp-go/.github/workflows/dare-ci.yml +42 -0
  209. package/templates/stacks/mcp-go/README.md.tpl +50 -0
  210. package/templates/stacks/mcp-go/cmd/server/main.go.tpl +62 -0
  211. package/templates/stacks/mcp-go/gitignore +6 -0
  212. package/templates/stacks/mcp-go/go.mod.tpl +9 -0
  213. package/templates/stacks/mcp-go/internal/prompts/summarize.go +9 -0
  214. package/templates/stacks/mcp-go/internal/server/server.go.tpl +80 -0
  215. package/templates/stacks/mcp-go/internal/tools/echo.go +15 -0
  216. package/templates/stacks/mcp-go/internal/transports/http.go.tpl +21 -0
  217. package/templates/stacks/mcp-go/internal/transports/sse.go.tpl +17 -0
  218. package/templates/stacks/mcp-go/internal/transports/stdio.go.tpl +14 -0
  219. package/templates/stacks/mcp-go/llms.txt.tpl +60 -0
  220. package/templates/stacks/mcp-go/openapi.json.tpl +31 -0
  221. package/templates/stacks/mcp-go/tests/echo_test.go.tpl +37 -0
  222. package/templates/stacks/mcp-node-ts/.dare/skills.yml +8 -0
  223. package/templates/stacks/mcp-node-ts/.env.example +16 -0
  224. package/templates/stacks/mcp-node-ts/.github/workflows/dare-ci.yml +54 -0
  225. package/templates/stacks/mcp-node-ts/README.md.hbs +49 -0
  226. package/templates/stacks/mcp-node-ts/gitignore +7 -0
  227. package/templates/stacks/mcp-node-ts/llms.txt.hbs +61 -0
  228. package/templates/stacks/mcp-node-ts/openapi.json.hbs +39 -0
  229. package/templates/stacks/mcp-node-ts/package.json.hbs +35 -0
  230. package/templates/stacks/mcp-node-ts/src/cli.ts.hbs +71 -0
  231. package/templates/stacks/mcp-node-ts/src/prompts/index.ts +36 -0
  232. package/templates/stacks/mcp-node-ts/src/server.ts.hbs +45 -0
  233. package/templates/stacks/mcp-node-ts/src/tools/echo.ts +23 -0
  234. package/templates/stacks/mcp-node-ts/src/tools/index.ts +18 -0
  235. package/templates/stacks/mcp-node-ts/src/transports/http.ts +68 -0
  236. package/templates/stacks/mcp-node-ts/src/transports/sse.ts +58 -0
  237. package/templates/stacks/mcp-node-ts/src/transports/stdio.ts +5 -0
  238. package/templates/stacks/mcp-node-ts/tests/echo.test.ts +50 -0
  239. package/templates/stacks/mcp-node-ts/tsconfig.json +17 -0
  240. package/templates/stacks/mcp-python/.dare/skills.yml +8 -0
  241. package/templates/stacks/mcp-python/.env.example +14 -0
  242. package/templates/stacks/mcp-python/.github/workflows/dare-ci.yml +42 -0
  243. package/templates/stacks/mcp-python/README.md.j2 +49 -0
  244. package/templates/stacks/mcp-python/gitignore +12 -0
  245. package/templates/stacks/mcp-python/llms.txt.j2 +56 -0
  246. package/templates/stacks/mcp-python/openapi.json.j2 +33 -0
  247. package/templates/stacks/mcp-python/pyproject.toml.j2 +37 -0
  248. package/templates/stacks/mcp-python/src/__init__.py +0 -0
  249. package/templates/stacks/mcp-python/src/cli.py.j2 +68 -0
  250. package/templates/stacks/mcp-python/src/prompts/__init__.py +0 -0
  251. package/templates/stacks/mcp-python/src/prompts/summarize.py +10 -0
  252. package/templates/stacks/mcp-python/src/server.py.j2 +28 -0
  253. package/templates/stacks/mcp-python/src/tools/__init__.py +0 -0
  254. package/templates/stacks/mcp-python/src/tools/echo.py +12 -0
  255. package/templates/stacks/mcp-python/src/transports/__init__.py +0 -0
  256. package/templates/stacks/mcp-python/src/transports/http.py +12 -0
  257. package/templates/stacks/mcp-python/src/transports/sse.py +13 -0
  258. package/templates/stacks/mcp-python/src/transports/stdio.py +6 -0
  259. package/templates/stacks/mcp-python/tests/__init__.py +0 -0
  260. package/templates/stacks/mcp-python/tests/test_echo.py +28 -0
  261. package/templates/stacks/mcp-rust/.dare/skills.yml +8 -0
  262. package/templates/stacks/mcp-rust/.env.example +14 -0
  263. package/templates/stacks/mcp-rust/.github/workflows/dare-ci.yml +38 -0
  264. package/templates/stacks/mcp-rust/Cargo.toml.tera +35 -0
  265. package/templates/stacks/mcp-rust/README.md.tera +50 -0
  266. package/templates/stacks/mcp-rust/gitignore +5 -0
  267. package/templates/stacks/mcp-rust/llms.txt.tera +60 -0
  268. package/templates/stacks/mcp-rust/openapi.json.tera +31 -0
  269. package/templates/stacks/mcp-rust/src/cli.rs.tera +33 -0
  270. package/templates/stacks/mcp-rust/src/lib.rs +6 -0
  271. package/templates/stacks/mcp-rust/src/main.rs.tera +30 -0
  272. package/templates/stacks/mcp-rust/src/prompts/mod.rs +1 -0
  273. package/templates/stacks/mcp-rust/src/prompts/summarize.rs +5 -0
  274. package/templates/stacks/mcp-rust/src/server.rs.tera +38 -0
  275. package/templates/stacks/mcp-rust/src/tools/echo.rs +18 -0
  276. package/templates/stacks/mcp-rust/src/tools/mod.rs +22 -0
  277. package/templates/stacks/mcp-rust/src/transports/http.rs +27 -0
  278. package/templates/stacks/mcp-rust/src/transports/mod.rs +3 -0
  279. package/templates/stacks/mcp-rust/src/transports/sse.rs +33 -0
  280. package/templates/stacks/mcp-rust/src/transports/stdio.rs +14 -0
  281. package/templates/stacks/mcp-rust/tests/echo_test.rs.tera +27 -0
  282. package/templates/stacks/node-nestjs/.dare/skills.yml +11 -0
  283. package/templates/stacks/node-nestjs/.env.example +21 -0
  284. package/templates/stacks/node-nestjs/.github/workflows/dare-ci.yml +54 -0
  285. package/templates/stacks/node-nestjs/README.md.hbs +35 -0
  286. package/templates/stacks/node-nestjs/gitignore +7 -0
  287. package/templates/stacks/node-nestjs/llms.txt.hbs +47 -0
  288. package/templates/stacks/node-nestjs/nest-cli.json +16 -0
  289. package/templates/stacks/node-nestjs/openapi.json.hbs +75 -0
  290. package/templates/stacks/node-nestjs/package.json.hbs +57 -0
  291. package/templates/stacks/node-nestjs/prisma/schema.prisma +25 -0
  292. package/templates/stacks/node-nestjs/prisma/seed.ts.hbs +25 -0
  293. package/templates/stacks/node-nestjs/src/app.module.ts +39 -0
  294. package/templates/stacks/node-nestjs/src/auth/auth.controller.ts +29 -0
  295. package/templates/stacks/node-nestjs/src/auth/auth.module.ts +25 -0
  296. package/templates/stacks/node-nestjs/src/auth/auth.service.ts +36 -0
  297. package/templates/stacks/node-nestjs/src/auth/dto/login-response.dto.ts +9 -0
  298. package/templates/stacks/node-nestjs/src/auth/dto/login.dto.ts +17 -0
  299. package/templates/stacks/node-nestjs/src/auth/jwt.strategy.ts +25 -0
  300. package/templates/stacks/node-nestjs/src/common/filters/problem-details.filter.ts +38 -0
  301. package/templates/stacks/node-nestjs/src/common/interceptors/json-response.interceptor.ts +13 -0
  302. package/templates/stacks/node-nestjs/src/main.ts.hbs +44 -0
  303. package/templates/stacks/node-nestjs/src/prisma/prisma.module.ts +9 -0
  304. package/templates/stacks/node-nestjs/src/prisma/prisma.service.ts +9 -0
  305. package/templates/stacks/node-nestjs/src/users/dto/create-user.dto.ts +22 -0
  306. package/templates/stacks/node-nestjs/src/users/dto/user.dto.ts +15 -0
  307. package/templates/stacks/node-nestjs/src/users/users.controller.ts +41 -0
  308. package/templates/stacks/node-nestjs/src/users/users.module.ts +11 -0
  309. package/templates/stacks/node-nestjs/src/users/users.repository.ts +38 -0
  310. package/templates/stacks/node-nestjs/src/users/users.service.ts +38 -0
  311. package/templates/stacks/node-nestjs/tsconfig.build.json +4 -0
  312. package/templates/stacks/node-nestjs/tsconfig.json +28 -0
  313. package/templates/stacks/php-laravel/.dare/skills.yml +11 -0
  314. package/templates/stacks/php-laravel/.env.example +41 -0
  315. package/templates/stacks/php-laravel/.github/workflows/dare-ci.yml +43 -0
  316. package/templates/stacks/php-laravel/README.md.hbs +36 -0
  317. package/templates/stacks/php-laravel/app/Http/Controllers/Api/AuthController.php +36 -0
  318. package/templates/stacks/php-laravel/app/Http/Controllers/Api/UsersController.php +33 -0
  319. package/templates/stacks/php-laravel/app/Http/Requests/CreateUserRequest.php +26 -0
  320. package/templates/stacks/php-laravel/app/Http/Requests/LoginRequest.php +34 -0
  321. package/templates/stacks/php-laravel/app/Llm/Contracts/LlmProvider.php +12 -0
  322. package/templates/stacks/php-laravel/app/Llm/Providers/DummyProvider.php +13 -0
  323. package/templates/stacks/php-laravel/app/Llm/Providers/OpenAiProvider.php +33 -0
  324. package/templates/stacks/php-laravel/app/Models/User.php +44 -0
  325. package/templates/stacks/php-laravel/app/Repositories/UsersRepository.php +32 -0
  326. package/templates/stacks/php-laravel/app/Services/AuthService.php +37 -0
  327. package/templates/stacks/php-laravel/app/Services/UsersService.php +57 -0
  328. package/templates/stacks/php-laravel/artisan +12 -0
  329. package/templates/stacks/php-laravel/bootstrap/app.php +29 -0
  330. package/templates/stacks/php-laravel/bootstrap/providers.php +5 -0
  331. package/templates/stacks/php-laravel/composer.json.hbs +58 -0
  332. package/templates/stacks/php-laravel/config/l5-swagger.php +41 -0
  333. package/templates/stacks/php-laravel/config/reverb.php +34 -0
  334. package/templates/stacks/php-laravel/config/sanctum.php +15 -0
  335. package/templates/stacks/php-laravel/database/migrations/2026_06_01_000001_create_users_table.php +27 -0
  336. package/templates/stacks/php-laravel/database/seeders/DatabaseSeeder.php +21 -0
  337. package/templates/stacks/php-laravel/gitignore +23 -0
  338. package/templates/stacks/php-laravel/llms.txt.hbs +53 -0
  339. package/templates/stacks/php-laravel/openapi.json.hbs +43 -0
  340. package/templates/stacks/php-laravel/phpstan.neon +9 -0
  341. package/templates/stacks/php-laravel/routes/api.php +13 -0
  342. package/templates/stacks/php-laravel/routes/channels.php +7 -0
  343. package/templates/stacks/php-laravel/tests/Feature/AuthTest.php +35 -0
  344. package/templates/stacks/php-laravel/tests/Feature/UsersTest.php +30 -0
  345. package/templates/stacks/php-laravel/tests/Pest.php +5 -0
  346. package/templates/stacks/python-fastapi/.dare/skills.yml +11 -0
  347. package/templates/stacks/python-fastapi/.env.example +21 -0
  348. package/templates/stacks/python-fastapi/.github/workflows/dare-ci.yml +43 -0
  349. package/templates/stacks/python-fastapi/README.md.j2 +35 -0
  350. package/templates/stacks/python-fastapi/alembic/env.py +46 -0
  351. package/templates/stacks/python-fastapi/alembic/script.py.mako +26 -0
  352. package/templates/stacks/python-fastapi/alembic/versions/0001_create_users.py.j2 +37 -0
  353. package/templates/stacks/python-fastapi/alembic.ini.j2 +39 -0
  354. package/templates/stacks/python-fastapi/app/__init__.py +0 -0
  355. package/templates/stacks/python-fastapi/app/core/__init__.py +0 -0
  356. package/templates/stacks/python-fastapi/app/core/config.py +24 -0
  357. package/templates/stacks/python-fastapi/app/core/security.py +34 -0
  358. package/templates/stacks/python-fastapi/app/db/__init__.py +0 -0
  359. package/templates/stacks/python-fastapi/app/db/session.py +22 -0
  360. package/templates/stacks/python-fastapi/app/main.py.j2 +36 -0
  361. package/templates/stacks/python-fastapi/app/models/__init__.py +3 -0
  362. package/templates/stacks/python-fastapi/app/models/user.py +30 -0
  363. package/templates/stacks/python-fastapi/app/repositories/__init__.py +0 -0
  364. package/templates/stacks/python-fastapi/app/repositories/user_repository.py +34 -0
  365. package/templates/stacks/python-fastapi/app/routers/__init__.py +0 -0
  366. package/templates/stacks/python-fastapi/app/routers/auth.py +37 -0
  367. package/templates/stacks/python-fastapi/app/routers/users.py +46 -0
  368. package/templates/stacks/python-fastapi/app/schemas/__init__.py +0 -0
  369. package/templates/stacks/python-fastapi/app/schemas/user.py +56 -0
  370. package/templates/stacks/python-fastapi/app/services/__init__.py +0 -0
  371. package/templates/stacks/python-fastapi/app/services/auth_service.py +22 -0
  372. package/templates/stacks/python-fastapi/app/services/user_service.py +31 -0
  373. package/templates/stacks/python-fastapi/gitignore +12 -0
  374. package/templates/stacks/python-fastapi/llms.txt.j2 +53 -0
  375. package/templates/stacks/python-fastapi/openapi.json.j2 +43 -0
  376. package/templates/stacks/python-fastapi/pyproject.toml.j2 +45 -0
  377. package/templates/stacks/python-fastapi/tests/__init__.py +0 -0
  378. package/templates/stacks/python-fastapi/tests/test_auth.py +22 -0
  379. package/templates/stacks/ruby-rails-8/.dare/skills.yml +50 -0
  380. package/templates/stacks/ruby-rails-8/.env.example +20 -0
  381. package/templates/stacks/ruby-rails-8/.github/workflows/dare-ci.yml +112 -0
  382. package/templates/stacks/ruby-rails-8/Gemfile.erb +61 -0
  383. package/templates/stacks/ruby-rails-8/app/channels/application_cable/channel.rb +11 -0
  384. package/templates/stacks/ruby-rails-8/app/channels/application_cable/connection.rb +34 -0
  385. package/templates/stacks/ruby-rails-8/app/channels/dare_updates_channel.rb +18 -0
  386. package/templates/stacks/ruby-rails-8/app/channels/user_updates_channel.rb +23 -0
  387. package/templates/stacks/ruby-rails-8/app/controllers/application_controller.rb +44 -0
  388. package/templates/stacks/ruby-rails-8/app/controllers/concerns/problem_details.rb +93 -0
  389. package/templates/stacks/ruby-rails-8/app/handlers/summarize_handler.rb +33 -0
  390. package/templates/stacks/ruby-rails-8/app/handlers/users_handler.rb +68 -0
  391. package/templates/stacks/ruby-rails-8/app/llm/cache/llm_cache.rb +44 -0
  392. package/templates/stacks/ruby-rails-8/app/llm/prompts/prompt_loader.rb +54 -0
  393. package/templates/stacks/ruby-rails-8/app/llm/prompts/summarize_v1.jinja2 +12 -0
  394. package/templates/stacks/ruby-rails-8/app/llm/providers/dummy_provider.rb +35 -0
  395. package/templates/stacks/ruby-rails-8/app/llm/providers/llm_provider.rb +67 -0
  396. package/templates/stacks/ruby-rails-8/app/llm/providers/openai_provider.rb +62 -0
  397. package/templates/stacks/ruby-rails-8/app/llm/rate_limit/token_bucket.rb +82 -0
  398. package/templates/stacks/ruby-rails-8/app/llm/validators/summarize_output_schema.json +21 -0
  399. package/templates/stacks/ruby-rails-8/app/llm/validators/validator.rb +52 -0
  400. package/templates/stacks/ruby-rails-8/app/models/user.rb +36 -0
  401. package/templates/stacks/ruby-rails-8/app/presenters/user_presenter.rb +48 -0
  402. package/templates/stacks/ruby-rails-8/app/repositories/document_repository.rb +57 -0
  403. package/templates/stacks/ruby-rails-8/app/repositories/user_repository.rb +73 -0
  404. package/templates/stacks/ruby-rails-8/app/services/create_user_service.rb +67 -0
  405. package/templates/stacks/ruby-rails-8/app/services/realtime_service.rb +53 -0
  406. package/templates/stacks/ruby-rails-8/app/services/summarize_document_service.rb +57 -0
  407. package/templates/stacks/ruby-rails-8/config/dare.yml +42 -0
  408. package/templates/stacks/ruby-rails-8/config/initializers/dare.rb +31 -0
  409. package/templates/stacks/ruby-rails-8/config/initializers/rack_attack.rb +64 -0
  410. package/templates/stacks/ruby-rails-8/config/initializers/rswag_api.rb +12 -0
  411. package/templates/stacks/ruby-rails-8/lib/tasks/dare.rake +159 -0
  412. package/templates/stacks/ruby-rails-8/llms.txt.erb +69 -0
  413. package/templates/stacks/ruby-rails-8/spec/api/summarize_spec.rb +56 -0
  414. package/templates/stacks/ruby-rails-8/spec/api/users_spec.rb +72 -0
  415. package/templates/stacks/ruby-rails-8/spec/channels/dare_updates_channel_spec.rb +61 -0
  416. package/templates/stacks/ruby-rails-8/spec/channels/user_updates_channel_spec.rb +56 -0
  417. package/templates/stacks/ruby-rails-8/spec/factories/users.rb +27 -0
  418. package/templates/stacks/ruby-rails-8/spec/handlers/users_handler_spec.rb +88 -0
  419. package/templates/stacks/ruby-rails-8/spec/rails_helper.rb +31 -0
  420. package/templates/stacks/ruby-rails-8/spec/services/create_user_service_spec.rb +88 -0
  421. package/templates/stacks/ruby-rails-8/spec/services/summarize_document_service_spec.rb +142 -0
  422. package/templates/stacks/ruby-rails-8/spec/swagger_helper.rb +73 -0
  423. package/templates/stacks/rust-axum/.dare/skills.yml +11 -0
  424. package/templates/stacks/rust-axum/.env.example +26 -0
  425. package/templates/stacks/rust-axum/.github/workflows/dare-ci.yml +40 -0
  426. package/templates/stacks/rust-axum/Cargo.toml.tera +53 -0
  427. package/templates/stacks/rust-axum/README.md.tera +37 -0
  428. package/templates/stacks/rust-axum/gitignore +5 -0
  429. package/templates/stacks/rust-axum/llms.txt.tera +54 -0
  430. package/templates/stacks/rust-axum/migrations/0001_create_users.sql +13 -0
  431. package/templates/stacks/rust-axum/openapi.json.tera +46 -0
  432. package/templates/stacks/rust-axum/src/config.rs +45 -0
  433. package/templates/stacks/rust-axum/src/errors.rs +48 -0
  434. package/templates/stacks/rust-axum/src/handlers/auth.rs +48 -0
  435. package/templates/stacks/rust-axum/src/handlers/mod.rs +3 -0
  436. package/templates/stacks/rust-axum/src/handlers/users.rs +81 -0
  437. package/templates/stacks/rust-axum/src/handlers/ws.rs +24 -0
  438. package/templates/stacks/rust-axum/src/lib.rs +19 -0
  439. package/templates/stacks/rust-axum/src/llm/mod.rs +1 -0
  440. package/templates/stacks/rust-axum/src/llm/provider.rs +48 -0
  441. package/templates/stacks/rust-axum/src/main.rs.tera +64 -0
  442. package/templates/stacks/rust-axum/src/middleware/auth.rs +20 -0
  443. package/templates/stacks/rust-axum/src/middleware/mod.rs +2 -0
  444. package/templates/stacks/rust-axum/src/middleware/rate_limit.rs +27 -0
  445. package/templates/stacks/rust-axum/src/models/mod.rs +1 -0
  446. package/templates/stacks/rust-axum/src/models/user.rs +13 -0
  447. package/templates/stacks/rust-axum/src/repositories/mod.rs +1 -0
  448. package/templates/stacks/rust-axum/src/repositories/user_repository.rs +62 -0
  449. package/templates/stacks/rust-axum/src/services/auth_service.rs +50 -0
  450. package/templates/stacks/rust-axum/src/services/mod.rs +2 -0
  451. package/templates/stacks/rust-axum/src/services/user_service.rs +53 -0
  452. package/templates/stacks/rust-axum/tests/integration_test.rs.tera +13 -0
  453. package/LICENSE +0 -21
@@ -0,0 +1,49 @@
1
+ # {{projectName}}
2
+
3
+ DARE-shaped MCP (Model Context Protocol) server in TypeScript.
4
+ Default transport: **{{defaultTransport}}**.
5
+
6
+ ## Setup
7
+
8
+ ```bash
9
+ cp .env.example .env
10
+ pnpm install
11
+ pnpm build
12
+ pnpm start # uses {{defaultTransport}} transport (default)
13
+ ```
14
+
15
+ ## Test it with MCP Inspector
16
+
17
+ ```bash
18
+ pnpm inspect # spawns the server + opens the inspector UI
19
+ ```
20
+
21
+ ## Tools
22
+
23
+ - `echo` — returns its input. The canonical smoke test for an MCP server.
24
+
25
+ ## Prompts
26
+
27
+ - `summarize` — placeholder prompt template that takes a `{text}` variable.
28
+
29
+ ## Transports
30
+
31
+ Pick at runtime via `--transport stdio|sse|http` or `MCP_TRANSPORT` env:
32
+
33
+ | Transport | When to use |
34
+ |---|---|
35
+ | `stdio` (default) | Local CLI agents (Claude Code, Cursor, etc.) |
36
+ | `sse` | Web clients via Server-Sent Events |
37
+ | `http` | Streamable HTTP — newest MCP transport |
38
+
39
+ ## Commands
40
+
41
+ - `pnpm dev` — watch mode via tsx
42
+ - `pnpm build` — TypeScript → `dist/`
43
+ - `pnpm test` — vitest
44
+ - `pnpm lint` — ESLint
45
+ - `pnpm audit` — `pnpm audit --audit-level=high`
46
+
47
+ ## DARE Method
48
+
49
+ See `.dare/skills.yml` and `llms.txt`.
@@ -0,0 +1,7 @@
1
+ node_modules
2
+ dist
3
+ coverage
4
+ .env
5
+ .env.local
6
+ *.log
7
+ .DS_Store
@@ -0,0 +1,61 @@
1
+ # {{projectName}}
2
+
3
+ DARE-shaped MCP server (TypeScript). Default transport: **{{defaultTransport}}**.
4
+
5
+ ## Setup
6
+
7
+ ```bash
8
+ cp .env.example .env
9
+ pnpm install
10
+ pnpm build
11
+ pnpm start
12
+ ```
13
+
14
+ ## Commands
15
+
16
+ - `pnpm dev` — tsx watch
17
+ - `pnpm build` — tsc to dist/
18
+ - `pnpm test` — vitest
19
+ - `pnpm lint` — ESLint
20
+ - `pnpm audit` — pnpm audit --audit-level=high
21
+ - `pnpm inspect` — MCP Inspector for the running server
22
+
23
+ ## Tools (registered in src/tools/)
24
+
25
+ | Name | Input | Output | Description |
26
+ |---|---|---|---|
27
+ | `echo` | `{ text: string }` | `{ text: string }` | Returns input verbatim. Smoke test. |
28
+
29
+ ## Prompts (registered in src/prompts/)
30
+
31
+ | Name | Args | Description |
32
+ |---|---|---|
33
+ | `summarize` | `{ text }` | Summarize the input text. Placeholder. |
34
+
35
+ ## Transports
36
+
37
+ Choose with `--transport stdio|sse|http` or `MCP_TRANSPORT`:
38
+
39
+ - `stdio` (default) — local agents. No port. 1 client at a time.
40
+ - `sse` — SSE-over-HTTP on `MCP_PORT` (default 3001).
41
+ - `http` — Streamable HTTP on `MCP_PORT` (default 3001). Modern MCP HTTP.
42
+
43
+ ## Architecture
44
+
45
+ - `src/server.ts` — MCP `Server` instance, tools and prompts registered here.
46
+ - `src/cli.ts` — entrypoint. Resolves transport from CLI flag or env.
47
+ - `src/transports/{stdio,sse,http}.ts` — one factory per transport.
48
+ - `src/tools/`, `src/prompts/` — registries the server reads from.
49
+
50
+ ## Env vars
51
+
52
+ See `.env.example`. None are required for stdio.
53
+
54
+ ## CLI inventory
55
+
56
+ `node dist/cli.js --json --list-tools` returns the tool registry as JSON
57
+ (M-03 — Agent eXperience).
58
+
59
+ ## DARE skills
60
+
61
+ See `.dare/skills.yml`.
@@ -0,0 +1,39 @@
1
+ {
2
+ "openapi": "3.1.0",
3
+ "info": {
4
+ "title": "{{projectName}}",
5
+ "version": "0.1.0",
6
+ "description": "MCP server JSON-RPC surface. MCP itself is not a REST API; this document describes the HTTP/SSE transport envelope. Stdio transport is out of band."
7
+ },
8
+ "paths": {
9
+ "/mcp/v1/sse": {
10
+ "get": {
11
+ "summary": "MCP SSE transport — long-lived event stream",
12
+ "responses": {
13
+ "200": { "description": "text/event-stream with MCP JSON-RPC frames" }
14
+ }
15
+ }
16
+ },
17
+ "/mcp/v1/messages": {
18
+ "post": {
19
+ "summary": "MCP JSON-RPC message (HTTP transport)",
20
+ "requestBody": {
21
+ "required": true,
22
+ "content": {
23
+ "application/json": {
24
+ "schema": { "type": "object", "description": "JSON-RPC 2.0 request" }
25
+ }
26
+ }
27
+ },
28
+ "responses": {
29
+ "200": {
30
+ "description": "JSON-RPC 2.0 response",
31
+ "content": {
32
+ "application/json": { "schema": { "type": "object" } }
33
+ }
34
+ }
35
+ }
36
+ }
37
+ }
38
+ }
39
+ }
@@ -0,0 +1,35 @@
1
+ {
2
+ "name": "{{projectName}}",
3
+ "version": "0.1.0",
4
+ "private": true,
5
+ "description": "DARE-shaped MCP server (Node.js / TypeScript)",
6
+ "type": "module",
7
+ "bin": {
8
+ "{{projectName}}": "dist/cli.js"
9
+ },
10
+ "main": "dist/server.js",
11
+ "scripts": {
12
+ "build": "tsc",
13
+ "start": "node dist/cli.js",
14
+ "dev": "tsx watch src/cli.ts",
15
+ "test": "vitest run",
16
+ "lint": "eslint src --ext .ts",
17
+ "audit": "pnpm audit --audit-level=high",
18
+ "inspect": "npx @modelcontextprotocol/inspector node dist/cli.js"
19
+ },
20
+ "dependencies": {
21
+ "@modelcontextprotocol/sdk": "^1.0.0",
22
+ "express": "^4.21.0",
23
+ "zod": "^3.23.0"
24
+ },
25
+ "devDependencies": {
26
+ "@types/express": "^4.17.0",
27
+ "@types/node": "^20.11.0",
28
+ "@typescript-eslint/eslint-plugin": "^7.0.0",
29
+ "@typescript-eslint/parser": "^7.0.0",
30
+ "eslint": "^8.57.0",
31
+ "tsx": "^4.16.0",
32
+ "typescript": "^5.4.5",
33
+ "vitest": "^1.6.0"
34
+ }
35
+ }
@@ -0,0 +1,71 @@
1
+ #!/usr/bin/env node
2
+ import { TOOLS } from './tools/index.js';
3
+ import { createServer } from './server.js';
4
+ import { createStdioTransport } from './transports/stdio.js';
5
+ import { startSseTransport } from './transports/sse.js';
6
+ import { startHttpTransport } from './transports/http.js';
7
+
8
+ type Transport = 'stdio' | 'sse' | 'http';
9
+
10
+ interface Args {
11
+ transport: Transport;
12
+ listToolsJson: boolean;
13
+ }
14
+
15
+ function parseArgs(argv: string[]): Args {
16
+ let transport: Transport =
17
+ (process.env.MCP_TRANSPORT as Transport | undefined) ?? '{{defaultTransport}}';
18
+ let listToolsJson = false;
19
+ let json = false;
20
+
21
+ for (let i = 0; i < argv.length; i++) {
22
+ const a = argv[i];
23
+ if (a === '--transport' && argv[i + 1]) {
24
+ const next = argv[++i];
25
+ if (next === 'stdio' || next === 'sse' || next === 'http') {
26
+ transport = next;
27
+ } else {
28
+ throw new Error(`invalid --transport '${next}' (valid: stdio, sse, http)`);
29
+ }
30
+ } else if (a === '--json') {
31
+ json = true;
32
+ } else if (a === '--list-tools') {
33
+ listToolsJson = true;
34
+ }
35
+ }
36
+
37
+ return { transport, listToolsJson: listToolsJson && json };
38
+ }
39
+
40
+ async function main(): Promise<void> {
41
+ const args = parseArgs(process.argv.slice(2));
42
+
43
+ if (args.listToolsJson) {
44
+ process.stdout.write(JSON.stringify({ tools: TOOLS }, null, 2) + '\n');
45
+ return;
46
+ }
47
+
48
+ const server = createServer();
49
+ const port = Number(process.env.MCP_PORT ?? 3001);
50
+
51
+ switch (args.transport) {
52
+ case 'stdio': {
53
+ const transport = createStdioTransport();
54
+ await server.connect(transport);
55
+ return;
56
+ }
57
+ case 'sse': {
58
+ await startSseTransport(server, port);
59
+ return;
60
+ }
61
+ case 'http': {
62
+ await startHttpTransport(server, port);
63
+ return;
64
+ }
65
+ }
66
+ }
67
+
68
+ main().catch((err) => {
69
+ process.stderr.write(`[mcp] fatal: ${(err as Error).message}\n`);
70
+ process.exit(1);
71
+ });
@@ -0,0 +1,36 @@
1
+ import type { Prompt } from '@modelcontextprotocol/sdk/types.js';
2
+
3
+ export const PROMPTS: Prompt[] = [
4
+ {
5
+ name: 'summarize',
6
+ description: 'Summarize the given text in 1-2 sentences.',
7
+ arguments: [
8
+ {
9
+ name: 'text',
10
+ description: 'Text to summarize',
11
+ required: true,
12
+ },
13
+ ],
14
+ },
15
+ ];
16
+
17
+ export async function getPrompt(
18
+ name: string,
19
+ args: Record<string, unknown>,
20
+ ): Promise<{ messages: Array<{ role: 'user'; content: { type: 'text'; text: string } }> }> {
21
+ if (name === 'summarize') {
22
+ const text = String(args.text ?? '');
23
+ return {
24
+ messages: [
25
+ {
26
+ role: 'user',
27
+ content: {
28
+ type: 'text',
29
+ text: `Summarize the following text in 1-2 sentences.\n\n${text}`,
30
+ },
31
+ },
32
+ ],
33
+ };
34
+ }
35
+ throw new Error(`Unknown prompt: ${name}`);
36
+ }
@@ -0,0 +1,45 @@
1
+ import { Server } from '@modelcontextprotocol/sdk/server/index.js';
2
+ import {
3
+ CallToolRequestSchema,
4
+ GetPromptRequestSchema,
5
+ ListPromptsRequestSchema,
6
+ ListToolsRequestSchema,
7
+ } from '@modelcontextprotocol/sdk/types.js';
8
+
9
+ import { TOOLS, callTool } from './tools/index.js';
10
+ import { PROMPTS, getPrompt } from './prompts/index.js';
11
+
12
+ export function createServer(): Server {
13
+ const server = new Server(
14
+ {
15
+ name: '{{projectName}}',
16
+ version: '0.1.0',
17
+ },
18
+ {
19
+ capabilities: {
20
+ tools: {},
21
+ prompts: {},
22
+ },
23
+ },
24
+ );
25
+
26
+ server.setRequestHandler(ListToolsRequestSchema, async () => ({
27
+ tools: TOOLS,
28
+ }));
29
+
30
+ server.setRequestHandler(CallToolRequestSchema, async (req) => {
31
+ const { name, arguments: args } = req.params;
32
+ return callTool(name, args ?? {});
33
+ });
34
+
35
+ server.setRequestHandler(ListPromptsRequestSchema, async () => ({
36
+ prompts: PROMPTS,
37
+ }));
38
+
39
+ server.setRequestHandler(GetPromptRequestSchema, async (req) => {
40
+ const { name, arguments: args } = req.params;
41
+ return getPrompt(name, args ?? {});
42
+ });
43
+
44
+ return server;
45
+ }
@@ -0,0 +1,23 @@
1
+ import { z } from 'zod';
2
+ import type { Tool } from '@modelcontextprotocol/sdk/types.js';
3
+
4
+ export const echoInput = z.object({
5
+ text: z.string().min(1, 'text is required'),
6
+ });
7
+
8
+ export const echoTool: Tool = {
9
+ name: 'echo',
10
+ description: 'Returns its input. Canonical smoke test for an MCP server.',
11
+ inputSchema: {
12
+ type: 'object',
13
+ required: ['text'],
14
+ properties: {
15
+ text: { type: 'string', minLength: 1, description: 'Text to echo back' },
16
+ },
17
+ },
18
+ };
19
+
20
+ export async function runEcho(args: unknown): Promise<string> {
21
+ const parsed = echoInput.parse(args);
22
+ return parsed.text;
23
+ }
@@ -0,0 +1,18 @@
1
+ import type { Tool } from '@modelcontextprotocol/sdk/types.js';
2
+ import { echoTool, runEcho } from './echo.js';
3
+
4
+ export const TOOLS: Tool[] = [echoTool];
5
+
6
+ export async function callTool(
7
+ name: string,
8
+ args: Record<string, unknown>,
9
+ ): Promise<{ content: Array<{ type: 'text'; text: string }>; isError?: boolean }> {
10
+ if (name === echoTool.name) {
11
+ const text = await runEcho(args);
12
+ return { content: [{ type: 'text', text }] };
13
+ }
14
+ return {
15
+ content: [{ type: 'text', text: `Unknown tool: ${name}` }],
16
+ isError: true,
17
+ };
18
+ }
@@ -0,0 +1,68 @@
1
+ import express from 'express';
2
+ import { randomUUID } from 'node:crypto';
3
+ import { Server } from '@modelcontextprotocol/sdk/server/index.js';
4
+ import { StreamableHTTPServerTransport } from '@modelcontextprotocol/sdk/server/streamableHttp.js';
5
+
6
+ /**
7
+ * Streamable HTTP — modern MCP HTTP transport. Each client gets its own
8
+ * session keyed by Mcp-Session-Id header (auto-generated on initialize).
9
+ */
10
+ export async function startHttpTransport(server: Server, port: number): Promise<void> {
11
+ const app = express();
12
+ app.use(express.json());
13
+
14
+ // Rate limit (same shape as SSE) — apply if RATE_LIMIT_RPM is set.
15
+ const rpm = Number(process.env.RATE_LIMIT_RPM ?? 0);
16
+ if (rpm > 0) {
17
+ const seen = new Map<string, { count: number; resetAt: number }>();
18
+ app.use((req, res, next) => {
19
+ const ip = req.ip ?? 'unknown';
20
+ const now = Date.now();
21
+ const entry = seen.get(ip);
22
+ if (!entry || entry.resetAt < now) {
23
+ seen.set(ip, { count: 1, resetAt: now + 60_000 });
24
+ return next();
25
+ }
26
+ if (entry.count >= rpm) {
27
+ res.status(429).json({ error: 'rate limit exceeded' });
28
+ return;
29
+ }
30
+ entry.count += 1;
31
+ next();
32
+ });
33
+ }
34
+
35
+ const transports = new Map<string, StreamableHTTPServerTransport>();
36
+
37
+ app.post('/mcp/v1/messages', async (req, res) => {
38
+ const sessionId = req.header('mcp-session-id');
39
+ let transport = sessionId ? transports.get(sessionId) : undefined;
40
+ if (!transport) {
41
+ transport = new StreamableHTTPServerTransport({
42
+ sessionIdGenerator: () => randomUUID(),
43
+ onsessioninitialized: (id: string) => {
44
+ transports.set(id, transport!);
45
+ },
46
+ });
47
+ await server.connect(transport);
48
+ }
49
+ await transport.handleRequest(req, res, req.body);
50
+ });
51
+
52
+ app.get('/mcp/v1/messages', async (req, res) => {
53
+ const sessionId = req.header('mcp-session-id');
54
+ const transport = sessionId ? transports.get(sessionId) : undefined;
55
+ if (!transport) {
56
+ res.status(400).json({ error: 'no active session' });
57
+ return;
58
+ }
59
+ await transport.handleRequest(req, res);
60
+ });
61
+
62
+ await new Promise<void>((resolve) => {
63
+ app.listen(port, () => {
64
+ process.stderr.write(`[mcp/http] listening on :${port}\n`);
65
+ resolve();
66
+ });
67
+ });
68
+ }
@@ -0,0 +1,58 @@
1
+ import express from 'express';
2
+ import { Server } from '@modelcontextprotocol/sdk/server/index.js';
3
+ import { SSEServerTransport } from '@modelcontextprotocol/sdk/server/sse.js';
4
+
5
+ /**
6
+ * Legacy MCP SSE transport. Holds one SSE connection at a time; multiple
7
+ * clients require running multiple server instances behind a load balancer.
8
+ *
9
+ * For new deployments prefer the modern Streamable HTTP transport
10
+ * (transports/http.ts) — SSE is kept here for compatibility with older
11
+ * MCP clients that only speak the SSE wire format.
12
+ */
13
+ export async function startSseTransport(server: Server, port: number): Promise<void> {
14
+ const app = express();
15
+
16
+ // Rate limit knob — apply if RATE_LIMIT_RPM is set. Simple token bucket per IP.
17
+ const rpm = Number(process.env.RATE_LIMIT_RPM ?? 0);
18
+ if (rpm > 0) {
19
+ const seen = new Map<string, { count: number; resetAt: number }>();
20
+ app.use((req, res, next) => {
21
+ const ip = req.ip ?? 'unknown';
22
+ const now = Date.now();
23
+ const entry = seen.get(ip);
24
+ if (!entry || entry.resetAt < now) {
25
+ seen.set(ip, { count: 1, resetAt: now + 60_000 });
26
+ return next();
27
+ }
28
+ if (entry.count >= rpm) {
29
+ res.status(429).json({ error: 'rate limit exceeded' });
30
+ return;
31
+ }
32
+ entry.count += 1;
33
+ next();
34
+ });
35
+ }
36
+
37
+ let transport: SSEServerTransport | null = null;
38
+
39
+ app.get('/mcp/v1/sse', async (_req, res) => {
40
+ transport = new SSEServerTransport('/mcp/v1/messages', res);
41
+ await server.connect(transport);
42
+ });
43
+
44
+ app.post('/mcp/v1/messages', express.json(), async (req, res) => {
45
+ if (!transport) {
46
+ res.status(400).json({ error: 'no active SSE session' });
47
+ return;
48
+ }
49
+ await transport.handlePostMessage(req, res, req.body);
50
+ });
51
+
52
+ await new Promise<void>((resolve) => {
53
+ app.listen(port, () => {
54
+ process.stderr.write(`[mcp/sse] listening on :${port}\n`);
55
+ resolve();
56
+ });
57
+ });
58
+ }
@@ -0,0 +1,5 @@
1
+ import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';
2
+
3
+ export function createStdioTransport(): StdioServerTransport {
4
+ return new StdioServerTransport();
5
+ }
@@ -0,0 +1,50 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { runEcho, echoTool } from '../src/tools/echo.js';
3
+ import { TOOLS, callTool } from '../src/tools/index.js';
4
+ import { PROMPTS, getPrompt } from '../src/prompts/index.js';
5
+
6
+ describe('echo tool', () => {
7
+ it('is registered in TOOLS', () => {
8
+ expect(TOOLS).toContain(echoTool);
9
+ });
10
+
11
+ it('returns its input verbatim', async () => {
12
+ const out = await runEcho({ text: 'hello' });
13
+ expect(out).toBe('hello');
14
+ });
15
+
16
+ it('rejects empty text', async () => {
17
+ await expect(runEcho({ text: '' })).rejects.toThrow();
18
+ });
19
+
20
+ it('rejects missing text', async () => {
21
+ await expect(runEcho({})).rejects.toThrow();
22
+ });
23
+
24
+ it('callTool routes to echo', async () => {
25
+ const res = await callTool('echo', { text: 'hi' });
26
+ expect(res.content[0].text).toBe('hi');
27
+ expect(res.isError).toBeFalsy();
28
+ });
29
+
30
+ it('callTool returns isError for unknown tool', async () => {
31
+ const res = await callTool('unknown', {});
32
+ expect(res.isError).toBe(true);
33
+ });
34
+ });
35
+
36
+ describe('summarize prompt', () => {
37
+ it('is registered in PROMPTS', () => {
38
+ expect(PROMPTS.some((p) => p.name === 'summarize')).toBe(true);
39
+ });
40
+
41
+ it('returns a user message with the input text', async () => {
42
+ const res = await getPrompt('summarize', { text: 'long text here' });
43
+ expect(res.messages[0].role).toBe('user');
44
+ expect(res.messages[0].content.text).toContain('long text here');
45
+ });
46
+
47
+ it('throws on unknown prompt', async () => {
48
+ await expect(getPrompt('unknown', {})).rejects.toThrow();
49
+ });
50
+ });
@@ -0,0 +1,17 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "outDir": "./dist",
7
+ "rootDir": "./src",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "resolveJsonModule": true,
11
+ "declaration": true,
12
+ "skipLibCheck": true,
13
+ "forceConsistentCasingInFileNames": true
14
+ },
15
+ "include": ["src/**/*"],
16
+ "exclude": ["node_modules", "dist", "tests"]
17
+ }
@@ -0,0 +1,8 @@
1
+ version: 1
2
+ skills:
3
+ - id: skill-mcp-server
4
+ source: skill-mcp-server
5
+ description: MCP server patterns (tools, prompts, transports)
6
+ - id: dare-ax
7
+ source: dare-ax
8
+ description: Agent eXperience invariants (llms.txt, OpenAPI, --json --list-tools, rate limit)
@@ -0,0 +1,14 @@
1
+ # MCP transport — overrides --transport flag if set
2
+ # Valid: stdio | sse | http
3
+ MCP_TRANSPORT=stdio
4
+
5
+ # HTTP/SSE transport port (ignored on stdio)
6
+ MCP_PORT=3001
7
+ MCP_HOST=0.0.0.0
8
+
9
+ # Logging
10
+ LOG_LEVEL=info
11
+
12
+ # LLM provider (optional)
13
+ LLM_PROVIDER=dummy
14
+ # OPENAI_API_KEY=
@@ -0,0 +1,42 @@
1
+ name: DARE CI
2
+
3
+ on:
4
+ push:
5
+ branches: [main]
6
+ pull_request:
7
+ branches: [main]
8
+
9
+ jobs:
10
+ audit:
11
+ name: pip-audit
12
+ runs-on: ubuntu-latest
13
+ steps:
14
+ - uses: actions/checkout@v4
15
+ - uses: actions/setup-python@v5
16
+ with:
17
+ python-version: "3.11"
18
+ - run: pip install pip-audit
19
+ - run: pip install -e ".[dev]"
20
+ - run: pip-audit --strict
21
+
22
+ lint:
23
+ name: ruff
24
+ runs-on: ubuntu-latest
25
+ steps:
26
+ - uses: actions/checkout@v4
27
+ - uses: actions/setup-python@v5
28
+ with:
29
+ python-version: "3.11"
30
+ - run: pip install ruff
31
+ - run: ruff check .
32
+
33
+ test:
34
+ name: pytest
35
+ runs-on: ubuntu-latest
36
+ steps:
37
+ - uses: actions/checkout@v4
38
+ - uses: actions/setup-python@v5
39
+ with:
40
+ python-version: "3.11"
41
+ - run: pip install -e ".[dev]"
42
+ - run: pytest