skills/sf-api-design/SKILL.md
Salesforce API design — custom REST endpoints, batch operations, Composite API, error envelopes, auth. Use when designing APIs exposed from Salesforce. Do NOT use for outbound callouts or Platform Events.
npx skillsauth add jiten-singh-shahi/salesforce-claude-code sf-api-designInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Patterns for designing and implementing custom APIs on the Salesforce platform. Callout limits, Composite API limits, and Named Credential details live in the reference file.
@../_reference/INTEGRATION_PATTERNS.md
@RestResource(urlMapping='/api/accounts/*')
global with sharing class AccountAPI {
@HttpGet
global static void getAccount() {
RestRequest req = RestContext.request;
RestResponse res = RestContext.response;
String accountId = req.requestURI.substringAfterLast('/');
try {
Account acc = [
SELECT Id, Name, Industry, AnnualRevenue
FROM Account WHERE Id = :accountId
WITH USER_MODE LIMIT 1
];
res.statusCode = 200;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(true, acc, null)));
} catch (QueryException e) {
res.statusCode = 404;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null, 'Account not found')));
} catch (Exception e) {
res.statusCode = 500;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null, 'An internal error occurred')));
}
}
@HttpPost
global static void createAccount() {
RestRequest req = RestContext.request;
RestResponse res = RestContext.response;
try {
Object parsed = JSON.deserializeUntyped(req.requestBody.toString());
if (!(parsed instanceof Map<String, Object>)) {
res.statusCode = 400;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null,
'Expected JSON object, got ' +
(parsed instanceof List<Object> ? 'array' : 'primitive')
)));
return;
}
Map<String, Object> body = (Map<String, Object>) parsed;
Account acc = new Account(
Name = (String) body.get('name'),
Industry = (String) body.get('industry')
);
// stripInaccessible — check getRemovedFields() to avoid silent data loss
SObjectAccessDecision decision = Security.stripInaccessible(
AccessType.CREATABLE, new List<Account>{acc});
if (!decision.getRemovedFields().isEmpty()) {
res.statusCode = 403;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null,
'Insufficient field permissions for: ' +
decision.getRemovedFields())));
return;
}
insert decision.getRecords();
res.statusCode = 201;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(true, decision.getRecords()[0], null)));
} catch (Exception e) {
res.statusCode = 400;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null, e.getMessage())));
}
}
global class ApiResponse {
public Boolean success;
public Object data;
public String error;
public ApiResponse(Boolean success, Object data, String error) {
this.success = success;
this.data = data;
this.error = error;
}
}
}
@HttpPost
global static void bulkCreate() {
RestRequest req = RestContext.request;
RestResponse res = RestContext.response;
try {
List<Object> items = (List<Object>) JSON.deserializeUntyped(
req.requestBody.toString());
List<Account> accounts = new List<Account>();
for (Object item : items) {
Map<String, Object> fields = (Map<String, Object>) item;
accounts.add(new Account(
Name = (String) fields.get('name'),
Industry = (String) fields.get('industry')
));
}
SObjectAccessDecision decision = Security.stripInaccessible(
AccessType.CREATABLE, accounts);
List<Database.SaveResult> results =
Database.insert(decision.getRecords(), false);
List<Object> response = new List<Object>();
for (Integer i = 0; i < results.size(); i++) {
Map<String, Object> row = new Map<String, Object>();
row.put('index', i);
row.put('success', results[i].isSuccess());
row.put('id', results[i].isSuccess() ? results[i].getId() : null);
if (!results[i].isSuccess()) {
row.put('errors', results[i].getErrors()[0].getMessage());
}
response.add(row);
}
res.statusCode = 200;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(true, response, null)));
} catch (Exception e) {
res.statusCode = 400;
res.responseBody = Blob.valueOf(JSON.serialize(
new ApiResponse(false, null, e.getMessage())));
}
}
Structured error codes for API consumers:
global class ApiError {
public String code;
public String message;
public String field;
public ApiError(String code, String message, String field) {
this.code = code;
this.message = message;
this.field = field;
}
}
// Standard error codes:
// FIELD_REQUIRED — Missing required field
// RECORD_NOT_FOUND — Record ID doesn't exist or no access
// GOVERNOR_LIMIT — Operation would exceed governor limits
// INSUFFICIENT_ACCESS — User lacks CRUD/FLS permission
// VALIDATION_FAILED — Validation rule or trigger prevented save
// DUPLICATE_VALUE — Unique field constraint violated
| Method | Use When | Setup | |--------|----------|-------| | Named Principal | All API users share one Salesforce user | Connected App + single auth | | Per-User | Each API caller maps to a Salesforce user | Connected App + OAuth per user | | JWT Bearer | Server-to-server, no user interaction | Connected App + X.509 certificate | | API Key (Custom) | Simple external tools | Custom Metadata + header validation |
| Anti-Pattern | Problem | Fix |
|-------------|---------|-----|
| God endpoint (all CRUD in one method) | Hard to maintain and test | One method per operation (@HttpGet, @HttpPost) |
| No pagination | Timeouts, governor limits | Add LIMIT + OFFSET or cursor-based pagination |
| Exposing internal Salesforce IDs | Security risk, breaks across orgs | Use external IDs or custom identifiers |
| No error codes | Consumers can't programmatically handle errors | Return structured error codes |
| No API versioning | Breaking changes affect all consumers | Version via URL path: /api/v1/accounts/ |
| WITHOUT SHARING on API class | Bypasses record-level security | Use WITH SHARING on REST resources |
| Returning all fields | Wastes bandwidth, exposes sensitive data | Return only requested/needed fields |
WITH USER_MODE in SOQL and AccessLevel.USER_MODE in DMLSecurity.stripInaccessible() when you need field-level enforcement on DML -- check getRemovedFields() for critical fields/api/v1/accounts/)Database.insert(records, false) for bulk APIs to support partial successdevelopment
Update Salesforce platform reference docs with latest release features and deprecation announcements. Use when SessionStart hook warns docs are outdated or a new Salesforce release has shipped. Do NOT use for Apex or LWC development.
development
Use when syncing documentation after Salesforce Apex code changes. Update README, API docs, and deploy metadata references to match the current org codebase.
development
Use when managing context during long Salesforce Apex development sessions. Suggests manual compaction at logical intervals to preserve deploy and org context across phases.
tools
Visualforce development — pages, controllers, extensions, ViewState, JS Remoting, LWC migration. Use when maintaining VF pages, building PDFs, or planning VF-to-LWC migration. Do NOT use for LWC, Aura, or Flow.