scc-universal 1.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 (271) hide show
  1. package/.claude-plugin/plugin.json +44 -0
  2. package/.cursor/agents/deep-researcher.md +142 -0
  3. package/.cursor/agents/doc-updater.md +219 -0
  4. package/.cursor/agents/eval-runner.md +335 -0
  5. package/.cursor/agents/learning-engine.md +210 -0
  6. package/.cursor/agents/loop-operator.md +245 -0
  7. package/.cursor/agents/refactor-cleaner.md +119 -0
  8. package/.cursor/agents/sf-admin-agent.md +127 -0
  9. package/.cursor/agents/sf-agentforce-agent.md +126 -0
  10. package/.cursor/agents/sf-apex-agent.md +117 -0
  11. package/.cursor/agents/sf-architect.md +426 -0
  12. package/.cursor/agents/sf-aura-reviewer.md +369 -0
  13. package/.cursor/agents/sf-bugfix-agent.md +101 -0
  14. package/.cursor/agents/sf-flow-agent.md +155 -0
  15. package/.cursor/agents/sf-integration-agent.md +141 -0
  16. package/.cursor/agents/sf-lwc-agent.md +123 -0
  17. package/.cursor/agents/sf-review-agent.md +357 -0
  18. package/.cursor/agents/sf-visualforce-reviewer.md +465 -0
  19. package/.cursor/hooks/adapter.js +81 -0
  20. package/.cursor/hooks/after-file-edit.js +26 -0
  21. package/.cursor/hooks/after-mcp-execution.js +12 -0
  22. package/.cursor/hooks/after-shell-execution.js +30 -0
  23. package/.cursor/hooks/after-tab-file-edit.js +12 -0
  24. package/.cursor/hooks/before-mcp-execution.js +11 -0
  25. package/.cursor/hooks/before-read-file.js +13 -0
  26. package/.cursor/hooks/before-shell-execution.js +29 -0
  27. package/.cursor/hooks/before-submit-prompt.js +23 -0
  28. package/.cursor/hooks/pre-compact.js +7 -0
  29. package/.cursor/hooks/session-end.js +10 -0
  30. package/.cursor/hooks/session-start.js +10 -0
  31. package/.cursor/hooks/stop.js +18 -0
  32. package/.cursor/hooks/subagent-start.js +10 -0
  33. package/.cursor/hooks/subagent-stop.js +10 -0
  34. package/.cursor/hooks.json +107 -0
  35. package/.cursor/skills/aside/SKILL.md +115 -0
  36. package/.cursor/skills/checkpoint/SKILL.md +50 -0
  37. package/.cursor/skills/configure-scc/SKILL.md +160 -0
  38. package/.cursor/skills/continuous-agent-loop/SKILL.md +260 -0
  39. package/.cursor/skills/mcp-server-patterns/SKILL.md +142 -0
  40. package/.cursor/skills/model-route/SKILL.md +81 -0
  41. package/.cursor/skills/prompt-optimizer/SKILL.md +366 -0
  42. package/.cursor/skills/refactor-clean/SKILL.md +133 -0
  43. package/.cursor/skills/resume-session/SKILL.md +111 -0
  44. package/.cursor/skills/save-session/SKILL.md +183 -0
  45. package/.cursor/skills/search-first/SKILL.md +140 -0
  46. package/.cursor/skills/security-scan/SKILL.md +142 -0
  47. package/.cursor/skills/sessions/SKILL.md +124 -0
  48. package/.cursor/skills/sf-agentforce-development/SKILL.md +449 -0
  49. package/.cursor/skills/sf-apex-async-patterns/SKILL.md +324 -0
  50. package/.cursor/skills/sf-apex-best-practices/SKILL.md +421 -0
  51. package/.cursor/skills/sf-apex-constraints/SKILL.md +79 -0
  52. package/.cursor/skills/sf-apex-cursor/SKILL.md +336 -0
  53. package/.cursor/skills/sf-apex-enterprise-patterns/SKILL.md +344 -0
  54. package/.cursor/skills/sf-apex-testing/SKILL.md +407 -0
  55. package/.cursor/skills/sf-api-design/SKILL.md +237 -0
  56. package/.cursor/skills/sf-approval-processes/SKILL.md +312 -0
  57. package/.cursor/skills/sf-aura-development/SKILL.md +260 -0
  58. package/.cursor/skills/sf-build-fix/SKILL.md +120 -0
  59. package/.cursor/skills/sf-data-modeling/SKILL.md +274 -0
  60. package/.cursor/skills/sf-debugging/SKILL.md +362 -0
  61. package/.cursor/skills/sf-deployment/SKILL.md +291 -0
  62. package/.cursor/skills/sf-deployment-constraints/SKILL.md +153 -0
  63. package/.cursor/skills/sf-devops-ci-cd/SKILL.md +322 -0
  64. package/.cursor/skills/sf-docs-lookup/SKILL.md +100 -0
  65. package/.cursor/skills/sf-e2e-testing/SKILL.md +321 -0
  66. package/.cursor/skills/sf-experience-cloud/SKILL.md +248 -0
  67. package/.cursor/skills/sf-flow-development/SKILL.md +376 -0
  68. package/.cursor/skills/sf-governor-limits/SKILL.md +319 -0
  69. package/.cursor/skills/sf-harness-audit/SKILL.md +139 -0
  70. package/.cursor/skills/sf-help/SKILL.md +156 -0
  71. package/.cursor/skills/sf-integration/SKILL.md +479 -0
  72. package/.cursor/skills/sf-lwc-constraints/SKILL.md +128 -0
  73. package/.cursor/skills/sf-lwc-development/SKILL.md +302 -0
  74. package/.cursor/skills/sf-lwc-testing/SKILL.md +387 -0
  75. package/.cursor/skills/sf-metadata-management/SKILL.md +285 -0
  76. package/.cursor/skills/sf-platform-events-cdc/SKILL.md +372 -0
  77. package/.cursor/skills/sf-quickstart/SKILL.md +170 -0
  78. package/.cursor/skills/sf-security/SKILL.md +330 -0
  79. package/.cursor/skills/sf-security-constraints/SKILL.md +125 -0
  80. package/.cursor/skills/sf-soql-constraints/SKILL.md +129 -0
  81. package/.cursor/skills/sf-soql-optimization/SKILL.md +353 -0
  82. package/.cursor/skills/sf-tdd-workflow/SKILL.md +332 -0
  83. package/.cursor/skills/sf-testing-constraints/SKILL.md +198 -0
  84. package/.cursor/skills/sf-trigger-constraints/SKILL.md +88 -0
  85. package/.cursor/skills/sf-trigger-frameworks/SKILL.md +343 -0
  86. package/.cursor/skills/sf-visualforce-development/SKILL.md +259 -0
  87. package/.cursor/skills/strategic-compact/SKILL.md +205 -0
  88. package/.cursor/skills/update-docs/SKILL.md +162 -0
  89. package/.cursor/skills/update-platform-docs/SKILL.md +86 -0
  90. package/.cursor-plugin/plugin.json +26 -0
  91. package/LICENSE +21 -0
  92. package/README.md +522 -0
  93. package/agents/deep-researcher.md +145 -0
  94. package/agents/doc-updater.md +222 -0
  95. package/agents/eval-runner.md +340 -0
  96. package/agents/learning-engine.md +211 -0
  97. package/agents/loop-operator.md +247 -0
  98. package/agents/refactor-cleaner.md +122 -0
  99. package/agents/sf-admin-agent.md +131 -0
  100. package/agents/sf-agentforce-agent.md +132 -0
  101. package/agents/sf-apex-agent.md +124 -0
  102. package/agents/sf-architect.md +435 -0
  103. package/agents/sf-aura-reviewer.md +372 -0
  104. package/agents/sf-bugfix-agent.md +105 -0
  105. package/agents/sf-flow-agent.md +159 -0
  106. package/agents/sf-integration-agent.md +146 -0
  107. package/agents/sf-lwc-agent.md +127 -0
  108. package/agents/sf-review-agent.md +366 -0
  109. package/agents/sf-visualforce-reviewer.md +468 -0
  110. package/assets/logo.svg +18 -0
  111. package/docs/ARCHITECTURE.md +133 -0
  112. package/docs/authoring-guide.md +373 -0
  113. package/docs/hook-development.md +578 -0
  114. package/docs/token-optimization.md +139 -0
  115. package/docs/workflow-examples.md +645 -0
  116. package/examples/agentforce-action/README.md +227 -0
  117. package/examples/apex-trigger-handler/README.md +114 -0
  118. package/examples/devops-pipeline/README.md +325 -0
  119. package/examples/flow-automation/README.md +188 -0
  120. package/examples/integration-pattern/README.md +416 -0
  121. package/examples/lwc-component/README.md +180 -0
  122. package/examples/platform-events/README.md +492 -0
  123. package/examples/scratch-org-setup/README.md +138 -0
  124. package/examples/security-audit/README.md +244 -0
  125. package/examples/visualforce-migration/README.md +314 -0
  126. package/hooks/hooks.json +338 -0
  127. package/hooks/memory-persistence/README.md +73 -0
  128. package/manifests/install-modules.json +217 -0
  129. package/manifests/install-profiles.json +17 -0
  130. package/mcp-configs/mcp-servers.json +19 -0
  131. package/package.json +89 -0
  132. package/schemas/hooks.schema.json +123 -0
  133. package/schemas/install-modules.schema.json +76 -0
  134. package/schemas/install-profiles.schema.json +28 -0
  135. package/schemas/install-state.schema.json +73 -0
  136. package/schemas/package-manager.schema.json +18 -0
  137. package/schemas/plugin.schema.json +112 -0
  138. package/schemas/scc-install-config.schema.json +29 -0
  139. package/schemas/state-store.schema.json +111 -0
  140. package/scripts/cli/install-apply.js +170 -0
  141. package/scripts/cli/uninstall.js +193 -0
  142. package/scripts/hooks/check-console-log.js +101 -0
  143. package/scripts/hooks/check-hook-enabled.js +17 -0
  144. package/scripts/hooks/check-platform-docs-age.js +48 -0
  145. package/scripts/hooks/cost-tracker.js +78 -0
  146. package/scripts/hooks/doc-file-warning.js +63 -0
  147. package/scripts/hooks/evaluate-session.js +98 -0
  148. package/scripts/hooks/governor-check.js +220 -0
  149. package/scripts/hooks/learning-observe.sh +206 -0
  150. package/scripts/hooks/mcp-health-check.js +588 -0
  151. package/scripts/hooks/post-bash-build-complete.js +34 -0
  152. package/scripts/hooks/post-bash-pr-created.js +43 -0
  153. package/scripts/hooks/post-edit-console-warn.js +61 -0
  154. package/scripts/hooks/post-edit-format.js +79 -0
  155. package/scripts/hooks/post-edit-typecheck.js +98 -0
  156. package/scripts/hooks/post-write.js +168 -0
  157. package/scripts/hooks/pre-bash-git-push-reminder.js +35 -0
  158. package/scripts/hooks/pre-bash-tmux-reminder.js +47 -0
  159. package/scripts/hooks/pre-compact.js +51 -0
  160. package/scripts/hooks/pre-tool-use.js +163 -0
  161. package/scripts/hooks/pre-write-doc-warn.js +9 -0
  162. package/scripts/hooks/quality-gate.js +251 -0
  163. package/scripts/hooks/run-with-flags-shell.sh +32 -0
  164. package/scripts/hooks/run-with-flags.js +135 -0
  165. package/scripts/hooks/session-end-marker.js +29 -0
  166. package/scripts/hooks/session-end.js +311 -0
  167. package/scripts/hooks/session-start.js +202 -0
  168. package/scripts/hooks/sfdx-scanner-check.js +142 -0
  169. package/scripts/hooks/sfdx-validate.js +119 -0
  170. package/scripts/hooks/stop-hook.js +170 -0
  171. package/scripts/hooks/suggest-compact.js +67 -0
  172. package/scripts/lib/agent-adapter.js +82 -0
  173. package/scripts/lib/apex-analysis.js +194 -0
  174. package/scripts/lib/hook-flags.js +74 -0
  175. package/scripts/lib/install-config.js +73 -0
  176. package/scripts/lib/install-executor.js +363 -0
  177. package/scripts/lib/install-state.js +121 -0
  178. package/scripts/lib/orchestration-session.js +299 -0
  179. package/scripts/lib/package-manager.js +124 -0
  180. package/scripts/lib/project-detect.js +228 -0
  181. package/scripts/lib/schema-validator.js +190 -0
  182. package/scripts/lib/skill-adapter.js +100 -0
  183. package/scripts/lib/state-store.js +376 -0
  184. package/scripts/lib/tmux-worktree-orchestrator.js +598 -0
  185. package/scripts/lib/utils.js +313 -0
  186. package/scripts/scc.js +164 -0
  187. package/skills/_reference/AGENTFORCE_PATTERNS.md +112 -0
  188. package/skills/_reference/APEX_CURSOR.md +159 -0
  189. package/skills/_reference/API_VERSIONS.md +78 -0
  190. package/skills/_reference/APPROVAL_PROCESSES.md +105 -0
  191. package/skills/_reference/ASYNC_PATTERNS.md +163 -0
  192. package/skills/_reference/AURA_COMPONENTS.md +146 -0
  193. package/skills/_reference/DATA_MIGRATION_PATTERNS.md +151 -0
  194. package/skills/_reference/DATA_MODELING.md +124 -0
  195. package/skills/_reference/DEBUGGING_TOOLS.md +140 -0
  196. package/skills/_reference/DEPLOYMENT_CHECKLIST.md +87 -0
  197. package/skills/_reference/DEPRECATIONS.md +79 -0
  198. package/skills/_reference/DOCKER_CI_PATTERNS.md +138 -0
  199. package/skills/_reference/ENTERPRISE_PATTERNS.md +122 -0
  200. package/skills/_reference/EXPERIENCE_CLOUD.md +143 -0
  201. package/skills/_reference/FLOW_PATTERNS.md +113 -0
  202. package/skills/_reference/GOVERNOR_LIMITS.md +77 -0
  203. package/skills/_reference/INTEGRATION_PATTERNS.md +105 -0
  204. package/skills/_reference/LWC_PATTERNS.md +79 -0
  205. package/skills/_reference/METADATA_TYPES.md +115 -0
  206. package/skills/_reference/NAMING_CONVENTIONS.md +84 -0
  207. package/skills/_reference/PACKAGE_DEVELOPMENT.md +150 -0
  208. package/skills/_reference/PLATFORM_EVENTS.md +121 -0
  209. package/skills/_reference/REPORTING_API.md +143 -0
  210. package/skills/_reference/SCRATCH_ORG_PATTERNS.md +126 -0
  211. package/skills/_reference/SECURITY_PATTERNS.md +127 -0
  212. package/skills/_reference/SHARING_MODEL.md +120 -0
  213. package/skills/_reference/SOQL_PATTERNS.md +119 -0
  214. package/skills/_reference/TESTING_STANDARDS.md +96 -0
  215. package/skills/_reference/TRIGGER_PATTERNS.md +114 -0
  216. package/skills/_reference/VISUALFORCE_PATTERNS.md +121 -0
  217. package/skills/aside/SKILL.md +118 -0
  218. package/skills/checkpoint/SKILL.md +53 -0
  219. package/skills/configure-scc/SKILL.md +163 -0
  220. package/skills/continuous-agent-loop/SKILL.md +264 -0
  221. package/skills/mcp-server-patterns/SKILL.md +146 -0
  222. package/skills/model-route/SKILL.md +84 -0
  223. package/skills/prompt-optimizer/SKILL.md +369 -0
  224. package/skills/refactor-clean/SKILL.md +136 -0
  225. package/skills/resume-session/SKILL.md +114 -0
  226. package/skills/save-session/SKILL.md +186 -0
  227. package/skills/search-first/SKILL.md +144 -0
  228. package/skills/security-scan/SKILL.md +146 -0
  229. package/skills/sessions/SKILL.md +127 -0
  230. package/skills/sf-agentforce-development/SKILL.md +450 -0
  231. package/skills/sf-apex-async-patterns/SKILL.md +326 -0
  232. package/skills/sf-apex-best-practices/SKILL.md +425 -0
  233. package/skills/sf-apex-constraints/SKILL.md +81 -0
  234. package/skills/sf-apex-cursor/SKILL.md +338 -0
  235. package/skills/sf-apex-enterprise-patterns/SKILL.md +348 -0
  236. package/skills/sf-apex-testing/SKILL.md +409 -0
  237. package/skills/sf-api-design/SKILL.md +238 -0
  238. package/skills/sf-approval-processes/SKILL.md +315 -0
  239. package/skills/sf-aura-development/SKILL.md +263 -0
  240. package/skills/sf-build-fix/SKILL.md +121 -0
  241. package/skills/sf-data-modeling/SKILL.md +278 -0
  242. package/skills/sf-debugging/SKILL.md +363 -0
  243. package/skills/sf-deployment/SKILL.md +295 -0
  244. package/skills/sf-deployment-constraints/SKILL.md +155 -0
  245. package/skills/sf-devops-ci-cd/SKILL.md +325 -0
  246. package/skills/sf-docs-lookup/SKILL.md +103 -0
  247. package/skills/sf-e2e-testing/SKILL.md +324 -0
  248. package/skills/sf-experience-cloud/SKILL.md +249 -0
  249. package/skills/sf-flow-development/SKILL.md +377 -0
  250. package/skills/sf-governor-limits/SKILL.md +323 -0
  251. package/skills/sf-harness-audit/SKILL.md +142 -0
  252. package/skills/sf-help/SKILL.md +159 -0
  253. package/skills/sf-integration/SKILL.md +483 -0
  254. package/skills/sf-lwc-constraints/SKILL.md +130 -0
  255. package/skills/sf-lwc-development/SKILL.md +303 -0
  256. package/skills/sf-lwc-testing/SKILL.md +388 -0
  257. package/skills/sf-metadata-management/SKILL.md +288 -0
  258. package/skills/sf-platform-events-cdc/SKILL.md +375 -0
  259. package/skills/sf-quickstart/SKILL.md +173 -0
  260. package/skills/sf-security/SKILL.md +334 -0
  261. package/skills/sf-security-constraints/SKILL.md +127 -0
  262. package/skills/sf-soql-constraints/SKILL.md +131 -0
  263. package/skills/sf-soql-optimization/SKILL.md +354 -0
  264. package/skills/sf-tdd-workflow/SKILL.md +336 -0
  265. package/skills/sf-testing-constraints/SKILL.md +200 -0
  266. package/skills/sf-trigger-constraints/SKILL.md +90 -0
  267. package/skills/sf-trigger-frameworks/SKILL.md +347 -0
  268. package/skills/sf-visualforce-development/SKILL.md +260 -0
  269. package/skills/strategic-compact/SKILL.md +208 -0
  270. package/skills/update-docs/SKILL.md +165 -0
  271. package/skills/update-platform-docs/SKILL.md +90 -0
@@ -0,0 +1,129 @@
1
+ ---
2
+ name: sf-soql-constraints
3
+ description: >-
4
+ Enforce SOQL/SOSL safety rules, selectivity, and governor limit compliance. Use when writing or reviewing ANY SOQL query, SOSL search, or database operations. Do NOT use for Apex structure, LWC, or Flow.
5
+ ---
6
+
7
+ # SOQL Constraints
8
+
9
+ ## When to Use
10
+
11
+ This skill auto-activates when writing, reviewing, or optimizing any SOQL query, SOSL search, or Apex database operation. It enforces query safety rules, selectivity requirements, and governor limit compliance for all database operations.
12
+
13
+ Hard rules for every SOQL query, SOSL search, and Apex database operation.
14
+ Violations cause governor limit failures, security vulnerabilities, or
15
+ production outages. See @../_reference/SOQL_PATTERNS.md for selectivity thresholds and
16
+ @../_reference/GOVERNOR_LIMITS.md for per-transaction budgets.
17
+
18
+ ## Never Rules
19
+
20
+ 1. **Never place SOQL or SOSL inside a loop.** Every iteration consumes one
21
+ of the per-transaction SOQL query budget (see @../_reference/GOVERNOR_LIMITS.md). Query
22
+ once before the loop, store results in a `Map<Id, SObject>`, then iterate.
23
+
24
+ 2. **Never write a non-selective query on objects with >200,000 rows.** The
25
+ query optimizer will full-table-scan and the query fails in trigger context.
26
+ Every WHERE clause must target an indexed field below the selectivity
27
+ threshold (see @../_reference/SOQL_PATTERNS.md, Selectivity Thresholds table).
28
+
29
+ 3. **Never use `FIELDS(ALL)` or `FIELDS(CUSTOM)` in triggers, service classes,
30
+ or production paths.** Select only the fields the calling code actually
31
+ reads. `FIELDS()` directives are for exploration and debugging only.
32
+
33
+ 4. **Never hardcode Salesforce record IDs.** IDs differ between orgs and
34
+ sandboxes. Use `Schema.SObjectType`, Custom Metadata, or Custom Labels to
35
+ resolve IDs at runtime.
36
+
37
+ 5. **Never omit `LIMIT` on unbounded queries.** Any query that could return an
38
+ unknown number of rows must include a `LIMIT` clause to stay within the
39
+ per-transaction row limit (see @../_reference/GOVERNOR_LIMITS.md).
40
+
41
+ 6. **Never concatenate user input into dynamic SOQL strings.** This creates
42
+ SOQL injection vulnerabilities. Use bind variables (`:variable`) for inline
43
+ SOQL or `Database.queryWithBinds()` for dynamic SOQL.
44
+
45
+ 7. **Never use a leading wildcard in `LIKE` filters** (`LIKE '%term%'`).
46
+ Leading wildcards prevent index use and cause full table scans. Use SOSL
47
+ (`FIND`) for full-text search instead.
48
+
49
+ 8. **Never use `!=` or `NOT IN` as the sole WHERE filter.** These operators
50
+ are non-optimizable and always produce table scans (see @../_reference/SOQL_PATTERNS.md,
51
+ Optimizable vs Non-Optimizable Operators).
52
+
53
+ 9. **Never omit security enforcement on user-facing queries.** Queries
54
+ triggered by user actions (LWC, Aura, VF, REST endpoints) must include
55
+ `WITH USER_MODE` or equivalent FLS/CRUD enforcement.
56
+
57
+ 10. **Never load all records just to count them.** Use `SELECT COUNT() FROM`
58
+ or aggregate queries instead of querying records and calling `.size()`.
59
+
60
+ ## Always Rules
61
+
62
+ 1. **Always bulkify database operations.** Collect IDs in a `Set<Id>`, query
63
+ once with `WHERE Id IN :idSet`, and store results in a `Map<Id, SObject>`.
64
+
65
+ 2. **Always use bind variables** (`:variable`) in inline SOQL. For dynamic
66
+ SOQL, always use `Database.queryWithBinds()` with a bind map.
67
+
68
+ 3. **Always use `WITH USER_MODE`** (see @../_reference/API_VERSIONS.md for minimum version) on queries executed in
69
+ user-facing contexts (LWC controllers, Aura controllers, VF controllers,
70
+ REST resources). Use `WITH SYSTEM_MODE` only for documented system
71
+ processes (batch jobs, integrations) with explicit justification.
72
+
73
+ 4. **Always filter on indexed fields.** Prefer `Id`, `Name`, `OwnerId`,
74
+ `RecordTypeId`, `CreatedDate`, `SystemModstamp`, lookup/master-detail
75
+ fields, or External ID fields. See @../_reference/SOQL_PATTERNS.md, Standard Indexed
76
+ Fields table for the full list.
77
+
78
+ 5. **Always add `LIMIT` when only one record is expected** (`LIMIT 1`) or
79
+ when displaying a bounded list.
80
+
81
+ 6. **Always use relationship queries** (parent-to-child subqueries or
82
+ child-to-parent dot notation) instead of separate queries when fetching
83
+ related data. Subqueries do not count as separate SOQL queries.
84
+
85
+ 7. **Always use SOSL instead of `LIKE` for text search across objects.**
86
+ SOSL uses the search index and is far more efficient than `LIKE` on
87
+ large-volume objects.
88
+
89
+ 8. **Always validate object and field names** via `Schema.getGlobalDescribe()`
90
+ before building dynamic SOQL. Never trust external input for object or
91
+ field names.
92
+
93
+ 9. **Always test triggers and services with 200 records** (the standard
94
+ trigger batch size) to validate bulk safety against governor limits.
95
+
96
+ ## Anti-Pattern Table
97
+
98
+ | Problem | Correct Pattern |
99
+ |---|---|
100
+ | SOQL inside `for` loop | Query before loop, store in `Map<Id, SObject>` |
101
+ | `SELECT FIELDS(ALL) FROM Account` in service class | `SELECT Id, Name FROM Account` -- explicit fields only |
102
+ | `WHERE Description LIKE '%keyword%'` | `FIND 'keyword' IN ALL FIELDS RETURNING Account(Id, Name)` (SOSL) |
103
+ | `WHERE Custom_Field__c = 'value'` on non-indexed field (LDV) | Add indexed field to WHERE, or request custom index |
104
+ | `String query = '...WHERE Name = \'' + input + '\''` | `[SELECT Id FROM Account WHERE Name = :input]` or `Database.queryWithBinds()` |
105
+ | `List<Account> all = [SELECT Id FROM Account]; Integer c = all.size();` | `Integer c = [SELECT COUNT() FROM Account];` |
106
+ | Hardcoded `WHERE Id = '001xx000003DGXXX'` | `WHERE Id = :accountId` with runtime-resolved variable |
107
+ | No `WITH USER_MODE` on LWC controller query | Add `WITH USER_MODE` to enforce FLS + sharing |
108
+ | Separate queries for parent and child records | Use subquery: `SELECT Id, (SELECT Id FROM Contacts) FROM Account` |
109
+ | `WHERE Status__c != 'Closed'` as only filter | Add a selective indexed filter: `WHERE RecordTypeId = :rtId AND Status__c != 'Closed'` |
110
+
111
+ ## Limit Budgets (Quick Reference)
112
+
113
+ Do not memorize raw numbers -- always check @../_reference/GOVERNOR_LIMITS.md for the
114
+ authoritative table. Key constraint categories that shape every query decision:
115
+
116
+ - **SOQL query limit** per synchronous/asynchronous transaction (see @../_reference/GOVERNOR_LIMITS.md)
117
+ - **Total rows returned** per transaction (see @../_reference/GOVERNOR_LIMITS.md)
118
+ - **Heap size** synchronous/asynchronous (see @../_reference/GOVERNOR_LIMITS.md)
119
+ - **CPU time** synchronous/asynchronous (see @../_reference/GOVERNOR_LIMITS.md)
120
+
121
+ Use `Limits.getQueries()` / `Limits.getLimitQueries()` to check remaining
122
+ budget at runtime before issuing additional queries.
123
+
124
+ ## Related
125
+
126
+ - **sf-soql-optimization** -- Action skill for interactive query optimization,
127
+ index strategy guidance, and Query Plan Tool usage.
128
+ - **sf-apex-constraints** -- Apex-level constraint rules (bulkification, DML
129
+ patterns, CPU/heap management).
@@ -0,0 +1,353 @@
1
+ ---
2
+ name: sf-soql-optimization
3
+ description: >-
4
+ SOQL optimization — selective queries, index strategy, query plans, relationship efficiency, large data volumes. Use when queries hit limits or designing for 100K+ records. Do NOT use for general Apex, LWC, or Flow.
5
+ ---
6
+
7
+ # SOQL Optimization
8
+
9
+ Poorly written SOQL is the most common cause of governor limit exceptions and slow page loads in Salesforce. This skill covers optimization procedures, query plan analysis, and index strategy.
10
+
11
+ @../_reference/SOQL_PATTERNS.md
12
+ @../_reference/GOVERNOR_LIMITS.md
13
+
14
+ ## When to Use
15
+
16
+ - SOQL queries hitting governor limits (100-query limit, CPU time, or heap size)
17
+ - Investigating slow page loads, timeout errors, or CPU violations in Apex code
18
+ - Adding new SOQL queries to triggers, batch jobs, or service classes
19
+ - Auditing code for SOQL-in-loops anti-patterns during code review
20
+ - Designing queries for large data sets (100K+ records) requiring selective filters
21
+ - Troubleshooting `System.LimitException: Too many SOQL queries` errors
22
+
23
+ ## Selectivity Fundamentals
24
+
25
+ The query optimizer decides whether to use an index or perform a full table scan based on **selectivity** — the percentage of records a query expects to return. See @../_reference/SOQL_PATTERNS.md for threshold tables.
26
+
27
+ ```apex
28
+ // Potentially NOT selective — if more than 10% of Contacts have Status = 'Active'
29
+ List<Contact> contacts = [SELECT Id FROM Contact WHERE Status__c = 'Active'];
30
+
31
+ // Selective — CreatedDate is indexed, LAST_N_DAYS:30 likely returns < 10%
32
+ List<Contact> recentContacts = [
33
+ SELECT Id FROM Contact
34
+ WHERE CreatedDate = LAST_N_DAYS:30
35
+ ];
36
+ ```
37
+
38
+ > Use the Query Plan Tool in Developer Console to verify actual query plans. These thresholds are guidelines; the optimizer considers data distribution, available indexes, and org-specific factors.
39
+
40
+ ---
41
+
42
+ ## SOQL in Loops — The Fix
43
+
44
+ Every iteration of a loop containing a SOQL query multiplies the query count toward the 100/200 limit.
45
+
46
+ ### The Fix — Query Once, Store in Map
47
+
48
+ ```apex
49
+ // Collect all IDs first
50
+ List<Account> accounts = [SELECT Id, OwnerId FROM Account WHERE Type = 'Customer'];
51
+
52
+ Set<Id> ownerIds = new Set<Id>();
53
+ for (Account acc : accounts) {
54
+ ownerIds.add(acc.OwnerId);
55
+ }
56
+
57
+ // Single query for all related records
58
+ Map<Id, User> ownerMap = new Map<Id, User>(
59
+ [SELECT Id, Name, Email FROM User WHERE Id IN :ownerIds]
60
+ );
61
+
62
+ // Now iterate — zero SOQL queries in this loop
63
+ for (Account acc : accounts) {
64
+ User owner = ownerMap.get(acc.OwnerId);
65
+ if (owner != null) {
66
+ sendWelcomeEmail(acc, owner);
67
+ }
68
+ }
69
+ ```
70
+
71
+ ### Trigger Context Pattern
72
+
73
+ ```apex
74
+ trigger OpportunityTrigger on Opportunity (after update) {
75
+ Set<Id> oppIds = Trigger.newMap.keySet();
76
+
77
+ Map<Id, List<OpportunityLineItem>> itemsByOppId = new Map<Id, List<OpportunityLineItem>>();
78
+ for (OpportunityLineItem item : [
79
+ SELECT Id, OpportunityId, Quantity, UnitPrice
80
+ FROM OpportunityLineItem
81
+ WHERE OpportunityId IN :oppIds
82
+ ]) {
83
+ if (!itemsByOppId.containsKey(item.OpportunityId)) {
84
+ itemsByOppId.put(item.OpportunityId, new List<OpportunityLineItem>());
85
+ }
86
+ itemsByOppId.get(item.OpportunityId).add(item);
87
+ }
88
+
89
+ for (Opportunity opp : Trigger.new) {
90
+ List<OpportunityLineItem> items = itemsByOppId.get(opp.Id);
91
+ if (items == null) items = new List<OpportunityLineItem>();
92
+ // Process items...
93
+ }
94
+ }
95
+ ```
96
+
97
+ ---
98
+
99
+ ## Query Optimization Checklist
100
+
101
+ ### Use Indexed Fields in WHERE Clause
102
+
103
+ ```apex
104
+ // Filter on RecordTypeId (indexed) rather than custom non-indexed fields
105
+ [SELECT Id FROM Account WHERE RecordTypeId = :RETAIL_RECORD_TYPE_ID]
106
+
107
+ // Combine indexed + non-indexed for compound selectivity
108
+ [SELECT Id FROM Account
109
+ WHERE Type = 'Customer'
110
+ AND Custom_Category__c = 'Retail']
111
+ ```
112
+
113
+ ### Select Only Fields You Need
114
+
115
+ ```apex
116
+ // Select only what the calling code actually uses
117
+ List<Account> accounts = [SELECT Id, Name FROM Account WHERE Id IN :accountIds];
118
+ ```
119
+
120
+ ### Use LIMIT When Possible
121
+
122
+ ```apex
123
+ Account acc = [SELECT Id, Name FROM Account WHERE Name = 'Acme' LIMIT 1];
124
+
125
+ List<Account> recentAccounts = [
126
+ SELECT Id, Name, CreatedDate
127
+ FROM Account
128
+ ORDER BY CreatedDate DESC
129
+ LIMIT 50
130
+ ];
131
+ ```
132
+
133
+ ### Avoid LIKE with % Prefix
134
+
135
+ ```apex
136
+ // Trailing wildcard CAN use the index
137
+ [SELECT Id FROM Account WHERE Name LIKE 'Acme%']
138
+
139
+ // For full-text search — use SOSL instead
140
+ List<List<SObject>> results = [FIND 'Corp' IN NAME FIELDS RETURNING Account(Id, Name)];
141
+ ```
142
+
143
+ ### Use Bind Variables
144
+
145
+ ```apex
146
+ // Bind variables are safe and performant
147
+ List<Account> accounts = [SELECT Id FROM Account WHERE Name = :accountName];
148
+
149
+ // For dynamic SOQL — use Database.queryWithBinds
150
+ Map<String, Object> bindVars = new Map<String, Object>{
151
+ 'accountName' => accountName,
152
+ 'minRevenue' => minimumRevenue
153
+ };
154
+ String safeQuery = 'SELECT Id, Name FROM Account WHERE Name = :accountName AND AnnualRevenue >= :minRevenue';
155
+ List<Account> accounts2 = Database.queryWithBinds(safeQuery, bindVars, AccessLevel.USER_MODE);
156
+ ```
157
+
158
+ ---
159
+
160
+ ## Relationship Queries
161
+
162
+ ### Parent-to-Child (Subquery)
163
+
164
+ ```apex
165
+ List<Account> accounts = [
166
+ SELECT Id, Name,
167
+ (SELECT Id, FirstName, LastName, Email
168
+ FROM Contacts
169
+ WHERE Email != null
170
+ ORDER BY LastName)
171
+ FROM Account
172
+ WHERE Type = 'Customer'
173
+ WITH USER_MODE
174
+ ];
175
+ ```
176
+
177
+ **Limits:** Max 1 level of subquery depth. Child subqueries do NOT count as separate SOQL queries. Max 20 subqueries per query.
178
+
179
+ ### Child-to-Parent (Dot Notation)
180
+
181
+ ```apex
182
+ List<Contact> contacts = [
183
+ SELECT Id, FirstName, LastName,
184
+ Account.Name,
185
+ Account.Owner.Name,
186
+ Account.Owner.Email
187
+ FROM Contact
188
+ WHERE AccountId != null
189
+ WITH USER_MODE
190
+ ];
191
+ ```
192
+
193
+ **Limits:** Max 5 levels of parent traversal. Max 35 relationship traversals total per query.
194
+
195
+ ---
196
+
197
+ ## Aggregate Queries
198
+
199
+ ```apex
200
+ // Use COUNT() instead of loading records to count
201
+ Integer customerCount = [SELECT COUNT() FROM Account WHERE Type = 'Customer'];
202
+
203
+ // AggregateResult for grouped/summed data
204
+ List<AggregateResult> results = [
205
+ SELECT Type, COUNT(Id) recordCount, SUM(AnnualRevenue) totalRevenue
206
+ FROM Account
207
+ WHERE Type != null
208
+ GROUP BY Type
209
+ HAVING COUNT(Id) > 5
210
+ ORDER BY COUNT(Id) DESC
211
+ ];
212
+ ```
213
+
214
+ ---
215
+
216
+ ## Dynamic SOQL — Safe Pattern
217
+
218
+ ```apex
219
+ public List<SObject> buildDynamicQuery(
220
+ String objectName, List<String> fields,
221
+ String whereClause, Integer maxRecords
222
+ ) {
223
+ Schema.DescribeSObjectResult describe =
224
+ Schema.getGlobalDescribe().get(objectName)?.getDescribe();
225
+ if (describe == null) {
226
+ throw new InvalidQueryException('Unknown object: ' + objectName);
227
+ }
228
+
229
+ Map<String, Schema.SObjectField> fieldMap = describe.fields.getMap();
230
+ List<String> validatedFields = new List<String>();
231
+ for (String field : fields) {
232
+ if (fieldMap.containsKey(field.toLowerCase())) {
233
+ validatedFields.add(field);
234
+ }
235
+ }
236
+ if (validatedFields.isEmpty()) validatedFields.add('Id');
237
+
238
+ String soql = 'SELECT ' + String.join(validatedFields, ', ') +
239
+ ' FROM ' + objectName;
240
+ if (String.isNotBlank(whereClause)) {
241
+ soql += ' WHERE ' + whereClause;
242
+ }
243
+ soql += ' LIMIT ' + Math.min(maxRecords, 2000);
244
+
245
+ return Database.queryWithBinds(soql, new Map<String, Object>(), AccessLevel.USER_MODE);
246
+ }
247
+ ```
248
+
249
+ ---
250
+
251
+ ## Large Data Volume Patterns
252
+
253
+ ### Custom Indexes
254
+
255
+ For heavily-queried fields not indexed by default, request a custom index from Salesforce Support. Provide object name, field API name, and typical query pattern.
256
+
257
+ ### Skinny Tables
258
+
259
+ For objects with >10 million records, Salesforce Support can create a skinny table — a narrow copy with only the most-queried fields.
260
+
261
+ ### Batch Apex for LDV Processing
262
+
263
+ ```apex
264
+ public class LargeDataProcessingBatch implements Database.Batchable<SObject> {
265
+ public Database.QueryLocator start(Database.BatchableContext bc) {
266
+ return Database.getQueryLocator([
267
+ SELECT Id, Status__c FROM Account
268
+ WHERE CreatedDate < :Date.today().addYears(-5)
269
+ AND Status__c = 'Active'
270
+ ]);
271
+ }
272
+ public void execute(Database.BatchableContext bc, List<Account> scope) { /* ... */ }
273
+ public void finish(Database.BatchableContext bc) {}
274
+ }
275
+ ```
276
+
277
+ ### SOSL for Full-Text Search
278
+
279
+ ```apex
280
+ String searchTerm = 'Acme Holdings';
281
+ List<List<SObject>> searchResults = [
282
+ FIND :searchTerm
283
+ IN ALL FIELDS
284
+ RETURNING
285
+ Account(Id, Name, Type WHERE Type = 'Customer'),
286
+ Contact(Id, FirstName, LastName, AccountId)
287
+ LIMIT 50
288
+ ];
289
+ ```
290
+
291
+ ---
292
+
293
+ ## Modern SOQL Features
294
+
295
+ ### TYPEOF for Polymorphic Lookups
296
+
297
+ ```apex
298
+ List<Task> tasks = [
299
+ SELECT Id, Subject,
300
+ TYPEOF What
301
+ WHEN Account THEN Name, Phone
302
+ WHEN Opportunity THEN Amount, StageName
303
+ END
304
+ FROM Task
305
+ WHERE OwnerId = :userId
306
+ WITH USER_MODE
307
+ ];
308
+ ```
309
+
310
+ Use when querying objects with polymorphic lookups (Task.WhoId, Task.WhatId, Event.WhoId, Event.WhatId) to avoid multiple queries or instanceof checks.
311
+
312
+ ### Semi-Join and Anti-Join Subqueries
313
+
314
+ ```soql
315
+ -- Semi-join: accounts that have contacts with emails
316
+ SELECT Id, Name FROM Account
317
+ WHERE Id IN (SELECT AccountId FROM Contact WHERE Email != null)
318
+
319
+ -- Anti-join: accounts with no closed opportunities
320
+ SELECT Id, Name FROM Account
321
+ WHERE Id NOT IN (SELECT AccountId FROM Opportunity WHERE IsClosed = true)
322
+ ```
323
+
324
+ Inner subquery can only return one field. Max one level of nesting. More efficient than querying both objects and filtering in Apex.
325
+
326
+ ---
327
+
328
+ ## Query Plan Tool
329
+
330
+ Use the Developer Console Query Plan to understand query execution before deploying.
331
+
332
+ 1. Open Developer Console > Query Editor
333
+ 2. Enable Query Plan: Help menu > Preferences > Enable Query Plan
334
+ 3. Write query and click "Query Plan" instead of "Execute"
335
+
336
+ | Result | Meaning |
337
+ |---|---|
338
+ | `TableScan` | Full table scan — potentially slow on large objects |
339
+ | `Index` | Index used — fast and selective |
340
+ | `Cost` | Estimated relative cost — lower is better |
341
+
342
+ If you see `TableScan` on a large object:
343
+
344
+ 1. Add a more selective condition using an indexed field
345
+ 2. Request a custom index from Salesforce Support
346
+ 3. Restructure the query to use SOSL
347
+
348
+ ---
349
+
350
+ ## Related
351
+
352
+ - **Agent**: `sf-apex-agent` — For interactive, in-depth guidance
353
+ - **Constraints**: `sf-soql-constraints` — Hard rules for SOQL safety and compliance