@hivehub/rulebook 5.5.2 → 5.7.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 (325) hide show
  1. package/.claude/commands/rulebook-learn-capture.md +41 -48
  2. package/.claude/commands/rulebook-learn-list.md +13 -13
  3. package/README.md +332 -394
  4. package/dist/cli/commands/context-intelligence.d.ts +0 -1
  5. package/dist/cli/commands/context-intelligence.d.ts.map +1 -1
  6. package/dist/cli/commands/context-intelligence.js +12 -33
  7. package/dist/cli/commands/context-intelligence.js.map +1 -1
  8. package/dist/cli/commands/index.d.ts +4 -7
  9. package/dist/cli/commands/index.d.ts.map +1 -1
  10. package/dist/cli/commands/index.js +4 -7
  11. package/dist/cli/commands/index.js.map +1 -1
  12. package/dist/cli/commands/init.d.ts.map +1 -1
  13. package/dist/cli/commands/init.js +40 -81
  14. package/dist/cli/commands/init.js.map +1 -1
  15. package/dist/cli/commands/mcp.d.ts +0 -1
  16. package/dist/cli/commands/mcp.d.ts.map +1 -1
  17. package/dist/cli/commands/mcp.js +1 -7
  18. package/dist/cli/commands/mcp.js.map +1 -1
  19. package/dist/cli/commands/memory.d.ts +7 -1
  20. package/dist/cli/commands/memory.d.ts.map +1 -1
  21. package/dist/cli/commands/memory.js +51 -57
  22. package/dist/cli/commands/memory.js.map +1 -1
  23. package/dist/cli/commands/misc.d.ts +1 -15
  24. package/dist/cli/commands/misc.d.ts.map +1 -1
  25. package/dist/cli/commands/misc.js +36 -215
  26. package/dist/cli/commands/misc.js.map +1 -1
  27. package/dist/cli/commands/plans.d.ts +0 -6
  28. package/dist/cli/commands/plans.d.ts.map +1 -1
  29. package/dist/cli/commands/plans.js +9 -77
  30. package/dist/cli/commands/plans.js.map +1 -1
  31. package/dist/cli/commands/skills.js +6 -6
  32. package/dist/cli/commands/skills.js.map +1 -1
  33. package/dist/cli/commands/task.js +4 -4
  34. package/dist/cli/commands/task.js.map +1 -1
  35. package/dist/cli/commands/update.d.ts.map +1 -1
  36. package/dist/cli/commands/update.js +122 -52
  37. package/dist/cli/commands/update.js.map +1 -1
  38. package/dist/cli/prompts.d.ts.map +1 -1
  39. package/dist/cli/prompts.js +1 -78
  40. package/dist/cli/prompts.js.map +1 -1
  41. package/dist/core/claude/claude-mcp.d.ts +59 -0
  42. package/dist/core/claude/claude-mcp.d.ts.map +1 -0
  43. package/dist/core/claude/claude-mcp.js +220 -0
  44. package/dist/core/claude/claude-mcp.js.map +1 -0
  45. package/dist/core/claude/claude-md-generator.d.ts +52 -0
  46. package/dist/core/claude/claude-md-generator.d.ts.map +1 -0
  47. package/dist/core/claude/claude-md-generator.js +104 -0
  48. package/dist/core/claude/claude-md-generator.js.map +1 -0
  49. package/dist/core/claude/claude-settings-manager.d.ts +44 -0
  50. package/dist/core/claude/claude-settings-manager.d.ts.map +1 -0
  51. package/dist/core/claude/claude-settings-manager.js +194 -0
  52. package/dist/core/claude/claude-settings-manager.js.map +1 -0
  53. package/dist/core/console/cli-bridge.d.ts +113 -0
  54. package/dist/core/console/cli-bridge.d.ts.map +1 -0
  55. package/dist/core/console/cli-bridge.js +1094 -0
  56. package/dist/core/console/cli-bridge.js.map +1 -0
  57. package/dist/core/detect/detector.d.ts +35 -0
  58. package/dist/core/detect/detector.d.ts.map +1 -0
  59. package/dist/core/detect/detector.js +541 -0
  60. package/dist/core/detect/detector.js.map +1 -0
  61. package/dist/core/docs/docs-generator.d.ts +9 -0
  62. package/dist/core/docs/docs-generator.d.ts.map +1 -0
  63. package/dist/core/docs/docs-generator.js +531 -0
  64. package/dist/core/docs/docs-generator.js.map +1 -0
  65. package/dist/core/docs/mcp-reference-generator.d.ts +13 -0
  66. package/dist/core/docs/mcp-reference-generator.d.ts.map +1 -0
  67. package/dist/core/docs/mcp-reference-generator.js +66 -0
  68. package/dist/core/docs/mcp-reference-generator.js.map +1 -0
  69. package/dist/core/generators/generator.d.ts +54 -0
  70. package/dist/core/generators/generator.d.ts.map +1 -0
  71. package/dist/core/generators/generator.js +1041 -0
  72. package/dist/core/generators/generator.js.map +1 -0
  73. package/dist/core/generators/gitignore-generator.d.ts +13 -0
  74. package/dist/core/generators/gitignore-generator.d.ts.map +1 -0
  75. package/dist/core/generators/gitignore-generator.js +307 -0
  76. package/dist/core/generators/gitignore-generator.js.map +1 -0
  77. package/dist/core/generators/minimal-scaffolder.d.ts +8 -0
  78. package/dist/core/generators/minimal-scaffolder.d.ts.map +1 -0
  79. package/dist/core/generators/minimal-scaffolder.js +51 -0
  80. package/dist/core/generators/minimal-scaffolder.js.map +1 -0
  81. package/dist/core/generators/rules-generator.d.ts +73 -0
  82. package/dist/core/generators/rules-generator.d.ts.map +1 -0
  83. package/dist/core/generators/rules-generator.js +202 -0
  84. package/dist/core/generators/rules-generator.js.map +1 -0
  85. package/dist/core/generators/workflow-generator.d.ts +15 -0
  86. package/dist/core/generators/workflow-generator.d.ts.map +1 -0
  87. package/dist/core/generators/workflow-generator.js +390 -0
  88. package/dist/core/generators/workflow-generator.js.map +1 -0
  89. package/dist/core/ide/multi-tool-generator.d.ts +59 -0
  90. package/dist/core/ide/multi-tool-generator.d.ts.map +1 -0
  91. package/dist/core/ide/multi-tool-generator.js +157 -0
  92. package/dist/core/ide/multi-tool-generator.js.map +1 -0
  93. package/dist/core/ide/opencode-generator.d.ts +72 -0
  94. package/dist/core/ide/opencode-generator.d.ts.map +1 -0
  95. package/dist/core/ide/opencode-generator.js +450 -0
  96. package/dist/core/ide/opencode-generator.js.map +1 -0
  97. package/dist/core/merger.d.ts +1 -1
  98. package/dist/core/merger.d.ts.map +1 -1
  99. package/dist/core/merger.js +5 -5
  100. package/dist/core/merger.js.map +1 -1
  101. package/dist/core/migrator.d.ts +0 -1
  102. package/dist/core/migrator.d.ts.map +1 -1
  103. package/dist/core/migrator.js +4 -29
  104. package/dist/core/migrator.js.map +1 -1
  105. package/dist/core/quality/coverage-checker.d.ts +14 -0
  106. package/dist/core/quality/coverage-checker.d.ts.map +1 -0
  107. package/dist/core/quality/coverage-checker.js +176 -0
  108. package/dist/core/quality/coverage-checker.js.map +1 -0
  109. package/dist/core/quality/dependency-checker.d.ts +21 -0
  110. package/dist/core/quality/dependency-checker.d.ts.map +1 -0
  111. package/dist/core/quality/dependency-checker.js +247 -0
  112. package/dist/core/quality/dependency-checker.js.map +1 -0
  113. package/dist/core/quality/doctor.d.ts +19 -0
  114. package/dist/core/quality/doctor.d.ts.map +1 -0
  115. package/dist/core/quality/doctor.js +163 -0
  116. package/dist/core/quality/doctor.js.map +1 -0
  117. package/dist/core/quality/validator.d.ts +21 -0
  118. package/dist/core/quality/validator.d.ts.map +1 -0
  119. package/dist/core/quality/validator.js +177 -0
  120. package/dist/core/quality/validator.js.map +1 -0
  121. package/dist/core/skills/skills-manager.d.ts +126 -0
  122. package/dist/core/skills/skills-manager.d.ts.map +1 -0
  123. package/dist/core/skills/skills-manager.js +630 -0
  124. package/dist/core/skills/skills-manager.js.map +1 -0
  125. package/dist/core/state/config-manager.d.ts +86 -0
  126. package/dist/core/state/config-manager.d.ts.map +1 -0
  127. package/dist/core/state/config-manager.js +562 -0
  128. package/dist/core/state/config-manager.js.map +1 -0
  129. package/dist/core/state/override-manager.d.ts +23 -0
  130. package/dist/core/state/override-manager.d.ts.map +1 -0
  131. package/dist/core/state/override-manager.js +82 -0
  132. package/dist/core/state/override-manager.js.map +1 -0
  133. package/dist/core/state/state-writer.d.ts +34 -0
  134. package/dist/core/state/state-writer.d.ts.map +1 -0
  135. package/dist/core/state/state-writer.js +78 -0
  136. package/dist/core/state/state-writer.js.map +1 -0
  137. package/dist/core/state/version-bumper.d.ts +19 -0
  138. package/dist/core/state/version-bumper.d.ts.map +1 -0
  139. package/dist/core/state/version-bumper.js +180 -0
  140. package/dist/core/state/version-bumper.js.map +1 -0
  141. package/dist/core/tasks/decision-manager.d.ts +25 -0
  142. package/dist/core/tasks/decision-manager.d.ts.map +1 -0
  143. package/dist/core/tasks/decision-manager.js +183 -0
  144. package/dist/core/tasks/decision-manager.js.map +1 -0
  145. package/dist/core/tasks/knowledge-manager.d.ts +24 -0
  146. package/dist/core/tasks/knowledge-manager.d.ts.map +1 -0
  147. package/dist/core/tasks/knowledge-manager.js +173 -0
  148. package/dist/core/tasks/knowledge-manager.js.map +1 -0
  149. package/dist/core/tasks/learn-manager.d.ts +27 -0
  150. package/dist/core/tasks/learn-manager.d.ts.map +1 -0
  151. package/dist/core/tasks/learn-manager.js +121 -0
  152. package/dist/core/tasks/learn-manager.js.map +1 -0
  153. package/dist/core/tasks/plans-manager.d.ts +46 -0
  154. package/dist/core/tasks/plans-manager.d.ts.map +1 -0
  155. package/dist/core/tasks/plans-manager.js +158 -0
  156. package/dist/core/tasks/plans-manager.js.map +1 -0
  157. package/dist/core/tasks/task-manager.d.ts +127 -0
  158. package/dist/core/tasks/task-manager.d.ts.map +1 -0
  159. package/dist/core/tasks/task-manager.js +607 -0
  160. package/dist/core/tasks/task-manager.js.map +1 -0
  161. package/dist/core/workspace/project-worker.d.ts +6 -6
  162. package/dist/core/workspace/project-worker.d.ts.map +1 -1
  163. package/dist/core/workspace/project-worker.js +6 -6
  164. package/dist/core/workspace/project-worker.js.map +1 -1
  165. package/dist/index.d.ts +1 -1
  166. package/dist/index.d.ts.map +1 -1
  167. package/dist/index.js +19 -176
  168. package/dist/index.js.map +1 -1
  169. package/dist/mcp/rulebook-server.d.ts.map +1 -1
  170. package/dist/mcp/rulebook-server.js +16 -960
  171. package/dist/mcp/rulebook-server.js.map +1 -1
  172. package/dist/memory/file-search.d.ts +43 -0
  173. package/dist/memory/file-search.d.ts.map +1 -0
  174. package/dist/memory/file-search.js +228 -0
  175. package/dist/memory/file-search.js.map +1 -0
  176. package/dist/memory/file-store.d.ts +99 -0
  177. package/dist/memory/file-store.d.ts.map +1 -0
  178. package/dist/memory/file-store.js +615 -0
  179. package/dist/memory/file-store.js.map +1 -0
  180. package/dist/memory/legacy-migrator.d.ts +27 -0
  181. package/dist/memory/legacy-migrator.d.ts.map +1 -0
  182. package/dist/memory/legacy-migrator.js +185 -0
  183. package/dist/memory/legacy-migrator.js.map +1 -0
  184. package/dist/memory/memory-manager.d.ts +25 -24
  185. package/dist/memory/memory-manager.d.ts.map +1 -1
  186. package/dist/memory/memory-manager.js +97 -140
  187. package/dist/memory/memory-manager.js.map +1 -1
  188. package/dist/memory/memory-types.d.ts +1 -1
  189. package/dist/memory/memory-types.d.ts.map +1 -1
  190. package/dist/types.d.ts +8 -119
  191. package/dist/types.d.ts.map +1 -1
  192. package/package.json +1 -6
  193. package/templates/agents/context-intelligence.md +50 -52
  194. package/templates/cli/OPENCODE.md +85 -18
  195. package/templates/commands/rulebook-learn-capture.md +41 -48
  196. package/templates/commands/rulebook-learn-list.md +13 -13
  197. package/templates/core/AGENTS_LEAN.md +0 -14
  198. package/templates/hooks/check-context-and-handoff.sh +8 -10
  199. package/templates/hooks/enforce-pre-tool.sh +70 -0
  200. package/templates/hooks/terse-mode-tracker.sh +146 -143
  201. package/templates/ides/OPENCODE.md +63 -0
  202. package/templates/skills/cli/opencode/SKILL.md +82 -28
  203. package/.claude/commands/ralph-config.md +0 -112
  204. package/.claude/commands/ralph-history.md +0 -110
  205. package/.claude/commands/ralph-init.md +0 -72
  206. package/.claude/commands/ralph-pause-resume.md +0 -105
  207. package/.claude/commands/ralph-run.md +0 -101
  208. package/.claude/commands/ralph-status.md +0 -76
  209. package/templates/core/RALPH.md +0 -471
  210. package/templates/frameworks/ANGULAR.md +0 -36
  211. package/templates/frameworks/DJANGO.md +0 -83
  212. package/templates/frameworks/ELECTRON.md +0 -147
  213. package/templates/frameworks/FLASK.md +0 -38
  214. package/templates/frameworks/FLUTTER.md +0 -55
  215. package/templates/frameworks/JQUERY.md +0 -32
  216. package/templates/frameworks/LARAVEL.md +0 -38
  217. package/templates/frameworks/NESTJS.md +0 -43
  218. package/templates/frameworks/NEXTJS.md +0 -127
  219. package/templates/frameworks/NUXT.md +0 -40
  220. package/templates/frameworks/RAILS.md +0 -66
  221. package/templates/frameworks/REACT.md +0 -38
  222. package/templates/frameworks/REACT_NATIVE.md +0 -47
  223. package/templates/frameworks/SPRING.md +0 -39
  224. package/templates/frameworks/SYMFONY.md +0 -36
  225. package/templates/frameworks/VUE.md +0 -36
  226. package/templates/frameworks/ZEND.md +0 -35
  227. package/templates/hooks/enforce-mcp-for-tasks.sh +0 -31
  228. package/templates/hooks/enforce-no-deferred.sh +0 -21
  229. package/templates/hooks/enforce-no-shortcuts.sh +0 -31
  230. package/templates/ides/COPILOT.md +0 -37
  231. package/templates/ides/CURSOR.md +0 -43
  232. package/templates/ides/JETBRAINS_AI.md +0 -35
  233. package/templates/ides/REPLIT.md +0 -36
  234. package/templates/ides/TABNINE.md +0 -29
  235. package/templates/ides/VSCODE.md +0 -40
  236. package/templates/ides/WINDSURF.md +0 -36
  237. package/templates/ides/ZED.md +0 -32
  238. package/templates/ides/cursor-mdc/go.mdc +0 -24
  239. package/templates/ides/cursor-mdc/python.mdc +0 -24
  240. package/templates/ides/cursor-mdc/quality.mdc +0 -25
  241. package/templates/ides/cursor-mdc/ralph.mdc +0 -39
  242. package/templates/ides/cursor-mdc/rulebook.mdc +0 -38
  243. package/templates/ides/cursor-mdc/rust.mdc +0 -24
  244. package/templates/ides/cursor-mdc/typescript.mdc +0 -25
  245. package/templates/ralph/ralph-history.bat +0 -4
  246. package/templates/ralph/ralph-history.sh +0 -5
  247. package/templates/ralph/ralph-init.bat +0 -5
  248. package/templates/ralph/ralph-init.sh +0 -5
  249. package/templates/ralph/ralph-pause.bat +0 -5
  250. package/templates/ralph/ralph-pause.sh +0 -5
  251. package/templates/ralph/ralph-run.bat +0 -5
  252. package/templates/ralph/ralph-run.sh +0 -5
  253. package/templates/ralph/ralph-status.bat +0 -4
  254. package/templates/ralph/ralph-status.sh +0 -5
  255. package/templates/services/AZURE_BLOB.md +0 -184
  256. package/templates/services/CASSANDRA.md +0 -239
  257. package/templates/services/DATADOG.md +0 -26
  258. package/templates/services/DOCKER.md +0 -124
  259. package/templates/services/DOCKER_COMPOSE.md +0 -168
  260. package/templates/services/DYNAMODB.md +0 -308
  261. package/templates/services/ELASTICSEARCH.md +0 -347
  262. package/templates/services/GCS.md +0 -178
  263. package/templates/services/HELM.md +0 -194
  264. package/templates/services/INFLUXDB.md +0 -265
  265. package/templates/services/KAFKA.md +0 -341
  266. package/templates/services/KUBERNETES.md +0 -208
  267. package/templates/services/MARIADB.md +0 -183
  268. package/templates/services/MEMCACHED.md +0 -242
  269. package/templates/services/MINIO.md +0 -201
  270. package/templates/services/MONGODB.md +0 -268
  271. package/templates/services/MYSQL.md +0 -358
  272. package/templates/services/NEO4J.md +0 -247
  273. package/templates/services/OPENTELEMETRY.md +0 -25
  274. package/templates/services/ORACLE.md +0 -290
  275. package/templates/services/PINO.md +0 -24
  276. package/templates/services/POSTGRESQL.md +0 -326
  277. package/templates/services/PROMETHEUS.md +0 -33
  278. package/templates/services/RABBITMQ.md +0 -286
  279. package/templates/services/REDIS.md +0 -292
  280. package/templates/services/S3.md +0 -298
  281. package/templates/services/SENTRY.md +0 -23
  282. package/templates/services/SQLITE.md +0 -294
  283. package/templates/services/SQLSERVER.md +0 -294
  284. package/templates/services/WINSTON.md +0 -30
  285. package/templates/skills/frameworks/angular/SKILL.md +0 -46
  286. package/templates/skills/frameworks/django/SKILL.md +0 -93
  287. package/templates/skills/frameworks/electron/SKILL.md +0 -157
  288. package/templates/skills/frameworks/flask/SKILL.md +0 -48
  289. package/templates/skills/frameworks/flutter/SKILL.md +0 -65
  290. package/templates/skills/frameworks/jquery/SKILL.md +0 -42
  291. package/templates/skills/frameworks/laravel/SKILL.md +0 -48
  292. package/templates/skills/frameworks/nestjs/SKILL.md +0 -53
  293. package/templates/skills/frameworks/nextjs/SKILL.md +0 -137
  294. package/templates/skills/frameworks/nuxt/SKILL.md +0 -50
  295. package/templates/skills/frameworks/rails/SKILL.md +0 -76
  296. package/templates/skills/frameworks/react/SKILL.md +0 -48
  297. package/templates/skills/frameworks/react-native/SKILL.md +0 -57
  298. package/templates/skills/frameworks/spring/SKILL.md +0 -49
  299. package/templates/skills/frameworks/symfony/SKILL.md +0 -46
  300. package/templates/skills/frameworks/vue/SKILL.md +0 -46
  301. package/templates/skills/frameworks/zend/SKILL.md +0 -45
  302. package/templates/skills/services/azure-blob/SKILL.md +0 -194
  303. package/templates/skills/services/cassandra/SKILL.md +0 -249
  304. package/templates/skills/services/dynamodb/SKILL.md +0 -318
  305. package/templates/skills/services/elasticsearch/SKILL.md +0 -357
  306. package/templates/skills/services/gcs/SKILL.md +0 -188
  307. package/templates/skills/services/influxdb/SKILL.md +0 -275
  308. package/templates/skills/services/kafka/SKILL.md +0 -351
  309. package/templates/skills/services/mariadb/SKILL.md +0 -193
  310. package/templates/skills/services/memcached/SKILL.md +0 -252
  311. package/templates/skills/services/minio/SKILL.md +0 -211
  312. package/templates/skills/services/mongodb/SKILL.md +0 -278
  313. package/templates/skills/services/mysql/SKILL.md +0 -368
  314. package/templates/skills/services/neo4j/SKILL.md +0 -257
  315. package/templates/skills/services/oracle/SKILL.md +0 -300
  316. package/templates/skills/services/postgresql/SKILL.md +0 -336
  317. package/templates/skills/services/rabbitmq/SKILL.md +0 -296
  318. package/templates/skills/services/redis/SKILL.md +0 -302
  319. package/templates/skills/services/s3/SKILL.md +0 -308
  320. package/templates/skills/services/sqlite/SKILL.md +0 -304
  321. package/templates/skills/services/sqlserver/SKILL.md +0 -304
  322. package/templates/skills/workflows/ralph/SETUP.md +0 -228
  323. package/templates/skills/workflows/ralph/SKILL.md +0 -309
  324. package/templates/skills/workflows/ralph/install.sh +0 -87
  325. package/templates/skills/workflows/ralph/manifest.json +0 -158
@@ -1,302 +0,0 @@
1
- ---
2
- name: "Redis"
3
- description: "Use Redis for high-performance caching, session storage, pub/sub messaging, and real-time features."
4
- version: "1.0.0"
5
- category: "services"
6
- author: "Rulebook"
7
- tags: ["services", "caching"]
8
- dependencies: []
9
- conflicts: []
10
- ---
11
- <!-- REDIS:START -->
12
- # Redis Cache Instructions
13
-
14
- **CRITICAL**: Use Redis for high-performance caching, session storage, pub/sub messaging, and real-time features.
15
-
16
- ## Core Features
17
-
18
- ### Connection
19
- ```typescript
20
- // Using redis (Node.js)
21
- import { createClient } from 'redis'
22
-
23
- const client = createClient({
24
- url: process.env.REDIS_URL || 'redis://localhost:6379',
25
- socket: {
26
- reconnectStrategy: (retries) => {
27
- if (retries > 10) {
28
- return new Error('Too many reconnection attempts')
29
- }
30
- return Math.min(retries * 100, 3000)
31
- },
32
- },
33
- })
34
-
35
- await client.connect()
36
-
37
- // Using ioredis
38
- import Redis from 'ioredis'
39
-
40
- const redis = new Redis({
41
- host: process.env.REDIS_HOST || 'localhost',
42
- port: parseInt(process.env.REDIS_PORT || '6379'),
43
- password: process.env.REDIS_PASSWORD,
44
- retryStrategy: (times) => Math.min(times * 50, 2000),
45
- maxRetriesPerRequest: 3,
46
- })
47
- ```
48
-
49
- ### Basic Operations
50
- ```typescript
51
- // String operations
52
- await client.set('user:1', JSON.stringify({ name: 'John', email: 'john@example.com' }))
53
- await client.set('user:1', 'value', { EX: 3600 }) // Expire in 1 hour
54
- const user = await client.get('user:1')
55
- await client.del('user:1')
56
-
57
- // Multiple operations
58
- await client.mSet({
59
- 'key1': 'value1',
60
- 'key2': 'value2',
61
- })
62
- const values = await client.mGet(['key1', 'key2'])
63
-
64
- // Increment/Decrement
65
- await client.incr('counter')
66
- await client.incrBy('counter', 5)
67
- await client.decr('counter')
68
- ```
69
-
70
- ### Data Structures
71
- ```typescript
72
- // Lists
73
- await client.lPush('tasks', 'task1', 'task2')
74
- await client.rPush('tasks', 'task3')
75
- const task = await client.lPop('tasks')
76
- const tasks = await client.lRange('tasks', 0, -1)
77
-
78
- // Sets
79
- await client.sAdd('tags', 'javascript', 'typescript', 'nodejs')
80
- const tags = await client.sMembers('tags')
81
- const exists = await client.sIsMember('tags', 'javascript')
82
- await client.sRem('tags', 'javascript')
83
-
84
- // Sorted Sets
85
- await client.zAdd('leaderboard', {
86
- score: 100,
87
- value: 'player1',
88
- })
89
- const topPlayers = await client.zRange('leaderboard', 0, 9, { REV: true })
90
-
91
- // Hashes
92
- await client.hSet('user:1', {
93
- name: 'John',
94
- email: 'john@example.com',
95
- age: '30',
96
- })
97
- const user = await client.hGetAll('user:1')
98
- await client.hIncrBy('user:1', 'age', 1)
99
- ```
100
-
101
- ### Advanced Features
102
- ```typescript
103
- // Pub/Sub
104
- const publisher = client.duplicate()
105
- await publisher.connect()
106
-
107
- const subscriber = client.duplicate()
108
- await subscriber.connect()
109
-
110
- await subscriber.subscribe('notifications', (message) => {
111
- console.log('Received:', message)
112
- })
113
-
114
- await publisher.publish('notifications', JSON.stringify({ type: 'alert', message: 'Hello' }))
115
-
116
- // Streams
117
- await client.xAdd('events', '*', {
118
- type: 'user_login',
119
- userId: '123',
120
- timestamp: Date.now().toString(),
121
- })
122
-
123
- const events = await client.xRead({
124
- key: 'events',
125
- id: '0',
126
- }, {
127
- COUNT: 10,
128
- BLOCK: 1000,
129
- })
130
-
131
- // Transactions
132
- const multi = client.multi()
133
- multi.set('key1', 'value1')
134
- multi.set('key2', 'value2')
135
- multi.incr('counter')
136
- await multi.exec()
137
- ```
138
-
139
- ## Common Patterns
140
-
141
- ### Caching
142
- ```typescript
143
- async function getCachedUser(userId: string) {
144
- const cached = await client.get(`user:${userId}`)
145
- if (cached) {
146
- return JSON.parse(cached)
147
- }
148
-
149
- const user = await fetchUserFromDatabase(userId)
150
- await client.set(`user:${userId}`, JSON.stringify(user), { EX: 3600 })
151
- return user
152
- }
153
-
154
- // Cache with tags (for invalidation)
155
- async function cacheWithTags(key: string, value: any, tags: string[], ttl: number) {
156
- await client.set(key, JSON.stringify(value), { EX: ttl })
157
- for (const tag of tags) {
158
- await client.sAdd(`tag:${tag}`, key)
159
- }
160
- }
161
-
162
- async function invalidateByTag(tag: string) {
163
- const keys = await client.sMembers(`tag:${tag}`)
164
- if (keys.length > 0) {
165
- await client.del(...keys)
166
- }
167
- await client.del(`tag:${tag}`)
168
- }
169
- ```
170
-
171
- ### Rate Limiting
172
- ```typescript
173
- async function checkRateLimit(userId: string, limit: number, window: number): Promise<boolean> {
174
- const key = `ratelimit:${userId}`
175
- const current = await client.incr(key)
176
-
177
- if (current === 1) {
178
- await client.expire(key, window)
179
- }
180
-
181
- return current <= limit
182
- }
183
- ```
184
-
185
- ### Session Storage
186
- ```typescript
187
- async function setSession(sessionId: string, data: any, ttl: number) {
188
- await client.set(`session:${sessionId}`, JSON.stringify(data), { EX: ttl })
189
- }
190
-
191
- async function getSession(sessionId: string) {
192
- const data = await client.get(`session:${sessionId}`)
193
- return data ? JSON.parse(data) : null
194
- }
195
-
196
- async function deleteSession(sessionId: string) {
197
- await client.del(`session:${sessionId}`)
198
- }
199
- ```
200
-
201
- ### Distributed Locks
202
- ```typescript
203
- async function acquireLock(key: string, ttl: number): Promise<boolean> {
204
- const result = await client.set(key, 'locked', {
205
- EX: ttl,
206
- NX: true, // Only set if not exists
207
- })
208
- return result === 'OK'
209
- }
210
-
211
- async function releaseLock(key: string) {
212
- await client.del(key)
213
- }
214
- ```
215
-
216
- ## Best Practices
217
-
218
- ✅ **DO:**
219
- - Use connection pooling
220
- - Set appropriate TTL for cached data
221
- - Use pipelining for multiple operations
222
- - Use transactions (MULTI/EXEC) for atomic operations
223
- - Monitor memory usage
224
- - Use appropriate data structures (hash for objects, set for unique values)
225
- - Implement cache invalidation strategies
226
- - Use Redis Cluster for high availability
227
- - Enable persistence (RDB or AOF) for production
228
- - Use Redis Sentinel for failover
229
-
230
- ❌ **DON'T:**
231
- - Store large values (> 100KB, use compression or external storage)
232
- - Use Redis as primary database (it's a cache)
233
- - Skip error handling
234
- - Ignore memory limits
235
- - Hardcode connection strings
236
- - Use KEYS command in production (use SCAN instead)
237
- - Store sensitive data without encryption
238
- - Skip connection retry logic
239
- - Ignore eviction policies
240
- - Use blocking operations without timeouts
241
-
242
- ## Configuration
243
-
244
- ### Environment Variables
245
- ```bash
246
- REDIS_URL=redis://localhost:6379
247
- REDIS_URL=redis://:password@host:6379
248
- REDIS_URL=rediss://host:6380 # SSL
249
- REDIS_HOST=localhost
250
- REDIS_PORT=6379
251
- REDIS_PASSWORD=securepassword
252
- ```
253
-
254
- ### Docker Compose
255
- ```yaml
256
- services:
257
- redis:
258
- image: redis:7-alpine
259
- ports:
260
- - "6379:6379"
261
- command: redis-server --requirepass securepassword --appendonly yes
262
- volumes:
263
- - redis_data:/data
264
- healthcheck:
265
- test: ["CMD", "redis-cli", "ping"]
266
- interval: 10s
267
- timeout: 5s
268
- retries: 5
269
-
270
- volumes:
271
- redis_data:
272
- ```
273
-
274
- ## Integration with Development
275
-
276
- ### Testing
277
- ```typescript
278
- // Use test Redis instance
279
- const testClient = createClient({
280
- url: 'redis://localhost:6380', // Different port for tests
281
- })
282
-
283
- // Clean up after tests
284
- afterEach(async () => {
285
- await testClient.flushDb()
286
- })
287
- ```
288
-
289
- ### Health Checks
290
- ```typescript
291
- async function checkRedisHealth(): Promise<boolean> {
292
- try {
293
- await client.ping()
294
- return true
295
- } catch {
296
- return false
297
- }
298
- }
299
- ```
300
-
301
- <!-- REDIS:END -->
302
-
@@ -1,308 +0,0 @@
1
- ---
2
- name: "S3"
3
- description: "Use AWS S3 for object storage, file uploads, static assets, and backup storage with high availability."
4
- version: "1.0.0"
5
- category: "services"
6
- author: "Rulebook"
7
- tags: ["services", "storage"]
8
- dependencies: []
9
- conflicts: []
10
- ---
11
- <!-- S3:START -->
12
- # AWS S3 Storage Instructions
13
-
14
- **CRITICAL**: Use AWS S3 for object storage, file uploads, static assets, and backup storage with high availability.
15
-
16
- ## Core Features
17
-
18
- ### Connection
19
- ```typescript
20
- // Using @aws-sdk/client-s3
21
- import { S3Client, PutObjectCommand, GetObjectCommand, DeleteObjectCommand } from '@aws-sdk/client-s3'
22
-
23
- const s3Client = new S3Client({
24
- region: process.env.AWS_REGION || 'us-east-1',
25
- credentials: {
26
- accessKeyId: process.env.AWS_ACCESS_KEY_ID || '',
27
- secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY || '',
28
- },
29
- })
30
-
31
- // Using AWS SDK v2
32
- import AWS from 'aws-sdk'
33
-
34
- const s3 = new AWS.S3({
35
- accessKeyId: process.env.AWS_ACCESS_KEY_ID,
36
- secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY,
37
- region: process.env.AWS_REGION || 'us-east-1',
38
- })
39
- ```
40
-
41
- ### Basic Operations
42
- ```typescript
43
- // Upload file
44
- const uploadParams = {
45
- Bucket: process.env.S3_BUCKET || 'my-bucket',
46
- Key: 'path/to/file.jpg',
47
- Body: fileBuffer,
48
- ContentType: 'image/jpeg',
49
- ACL: 'private', // or 'public-read'
50
- }
51
-
52
- await s3Client.send(new PutObjectCommand(uploadParams))
53
-
54
- // Upload with metadata
55
- await s3Client.send(new PutObjectCommand({
56
- ...uploadParams,
57
- Metadata: {
58
- userId: '123',
59
- originalName: 'photo.jpg',
60
- },
61
- TagSet: [
62
- { Key: 'category', Value: 'profile' },
63
- ],
64
- }))
65
-
66
- // Get file
67
- const getParams = {
68
- Bucket: process.env.S3_BUCKET,
69
- Key: 'path/to/file.jpg',
70
- }
71
-
72
- const response = await s3Client.send(new GetObjectCommand(getParams))
73
- const fileContent = await response.Body?.transformToByteArray()
74
-
75
- // Delete file
76
- await s3Client.send(new DeleteObjectCommand({
77
- Bucket: process.env.S3_BUCKET,
78
- Key: 'path/to/file.jpg',
79
- }))
80
- ```
81
-
82
- ### Advanced Features
83
- ```typescript
84
- // Presigned URL for upload
85
- import { getSignedUrl } from '@aws-sdk/s3-request-presigner'
86
-
87
- const command = new PutObjectCommand({
88
- Bucket: process.env.S3_BUCKET,
89
- Key: 'uploads/file.jpg',
90
- ContentType: 'image/jpeg',
91
- })
92
-
93
- const presignedUrl = await getSignedUrl(s3Client, command, { expiresIn: 3600 })
94
-
95
- // Presigned URL for download
96
- const getCommand = new GetObjectCommand({
97
- Bucket: process.env.S3_BUCKET,
98
- Key: 'path/to/file.jpg',
99
- })
100
-
101
- const downloadUrl = await getSignedUrl(s3Client, getCommand, { expiresIn: 3600 })
102
-
103
- // Multipart upload (for large files)
104
- import { CreateMultipartUploadCommand, UploadPartCommand, CompleteMultipartUploadCommand } from '@aws-sdk/client-s3'
105
-
106
- const createCommand = new CreateMultipartUploadCommand({
107
- Bucket: process.env.S3_BUCKET,
108
- Key: 'large-file.zip',
109
- ContentType: 'application/zip',
110
- })
111
-
112
- const { UploadId } = await s3Client.send(createCommand)
113
-
114
- // Upload parts
115
- const part1 = await s3Client.send(new UploadPartCommand({
116
- Bucket: process.env.S3_BUCKET,
117
- Key: 'large-file.zip',
118
- PartNumber: 1,
119
- UploadId,
120
- Body: part1Buffer,
121
- }))
122
-
123
- // Complete upload
124
- await s3Client.send(new CompleteMultipartUploadCommand({
125
- Bucket: process.env.S3_BUCKET,
126
- Key: 'large-file.zip',
127
- UploadId,
128
- MultipartUpload: {
129
- Parts: [
130
- { PartNumber: 1, ETag: part1.ETag },
131
- ],
132
- },
133
- }))
134
-
135
- // List objects
136
- import { ListObjectsV2Command } from '@aws-sdk/client-s3'
137
-
138
- const listCommand = new ListObjectsV2Command({
139
- Bucket: process.env.S3_BUCKET,
140
- Prefix: 'uploads/',
141
- MaxKeys: 100,
142
- })
143
-
144
- const { Contents } = await s3Client.send(listCommand)
145
- ```
146
-
147
- ## Common Patterns
148
-
149
- ### File Upload Handler
150
- ```typescript
151
- async function uploadFile(file: Buffer, filename: string, userId: string) {
152
- const key = `users/${userId}/${Date.now()}-${filename}`
153
-
154
- await s3Client.send(new PutObjectCommand({
155
- Bucket: process.env.S3_BUCKET,
156
- Key: key,
157
- Body: file,
158
- ContentType: getContentType(filename),
159
- Metadata: {
160
- userId,
161
- originalName: filename,
162
- uploadedAt: new Date().toISOString(),
163
- },
164
- }))
165
-
166
- return {
167
- key,
168
- url: `https://${process.env.S3_BUCKET}.s3.${process.env.AWS_REGION}.amazonaws.com/${key}`,
169
- }
170
- }
171
- ```
172
-
173
- ### Temporary File Access
174
- ```typescript
175
- async function generateTemporaryDownloadUrl(key: string, expiresIn: number = 3600) {
176
- const command = new GetObjectCommand({
177
- Bucket: process.env.S3_BUCKET,
178
- Key: key,
179
- })
180
-
181
- return await getSignedUrl(s3Client, command, { expiresIn })
182
- }
183
- ```
184
-
185
- ### File Cleanup
186
- ```typescript
187
- async function deleteFilesByPrefix(prefix: string) {
188
- const listCommand = new ListObjectsV2Command({
189
- Bucket: process.env.S3_BUCKET,
190
- Prefix: prefix,
191
- })
192
-
193
- let continuationToken: string | undefined
194
-
195
- do {
196
- const response = await s3Client.send({
197
- ...listCommand,
198
- ContinuationToken: continuationToken,
199
- })
200
-
201
- if (response.Contents) {
202
- for (const object of response.Contents) {
203
- if (object.Key) {
204
- await s3Client.send(new DeleteObjectCommand({
205
- Bucket: process.env.S3_BUCKET,
206
- Key: object.Key,
207
- }))
208
- }
209
- }
210
- }
211
-
212
- continuationToken = response.NextContinuationToken
213
- } while (continuationToken)
214
- }
215
- ```
216
-
217
- ## Best Practices
218
-
219
- ✅ **DO:**
220
- - Use presigned URLs for client uploads
221
- - Set appropriate Content-Type headers
222
- - Use versioning for important files
223
- - Implement lifecycle policies for cleanup
224
- - Use multipart upload for files > 5MB
225
- - Enable encryption (SSE-S3 or SSE-KMS)
226
- - Use appropriate storage classes (Standard, IA, Glacier)
227
- - Implement proper error handling
228
- - Use bucket policies for access control
229
- - Monitor bucket usage and costs
230
-
231
- ❌ **DON'T:**
232
- - Store sensitive data without encryption
233
- - Use public-read ACL unnecessarily
234
- - Hardcode credentials
235
- - Ignore error handling
236
- - Skip content-type validation
237
- - Store large files without multipart upload
238
- - Ignore lifecycle policies
239
- - Skip access logging
240
- - Use default bucket policies
241
- - Ignore cost optimization
242
-
243
- ## Configuration
244
-
245
- ### Environment Variables
246
- ```bash
247
- AWS_REGION=us-east-1
248
- AWS_ACCESS_KEY_ID=your-access-key
249
- AWS_SECRET_ACCESS_KEY=your-secret-key
250
- S3_BUCKET=my-bucket
251
- ```
252
-
253
- ### IAM Policy
254
- ```json
255
- {
256
- "Version": "2012-10-17",
257
- "Statement": [
258
- {
259
- "Effect": "Allow",
260
- "Action": [
261
- "s3:PutObject",
262
- "s3:GetObject",
263
- "s3:DeleteObject"
264
- ],
265
- "Resource": "arn:aws:s3:::my-bucket/*"
266
- }
267
- ]
268
- }
269
- ```
270
-
271
- ## Integration with Development
272
-
273
- ### Testing
274
- ```typescript
275
- // Use test bucket or LocalStack
276
- const testS3Client = new S3Client({
277
- endpoint: 'http://localhost:4566', // LocalStack
278
- region: 'us-east-1',
279
- credentials: {
280
- accessKeyId: 'test',
281
- secretAccessKey: 'test',
282
- },
283
- forcePathStyle: true,
284
- })
285
-
286
- // Clean up after tests
287
- afterEach(async () => {
288
- // Delete test files
289
- })
290
- ```
291
-
292
- ### Health Checks
293
- ```typescript
294
- async function checkS3Health(): Promise<boolean> {
295
- try {
296
- await s3Client.send(new ListObjectsV2Command({
297
- Bucket: process.env.S3_BUCKET,
298
- MaxKeys: 1,
299
- }))
300
- return true
301
- } catch {
302
- return false
303
- }
304
- }
305
- ```
306
-
307
- <!-- S3:END -->
308
-