home / skills / secondsky / claude-skills / api-filtering-sorting
This skill helps you implement flexible API filtering and sorting by parsing query parameters, validating input, and securing endpoints.
npx playbooks add skill secondsky/claude-skills --skill api-filtering-sortingReview the files below or copy the command above to add this skill to your agents.
---
name: api-filtering-sorting
description: Builds flexible API filtering and sorting systems with query parameter parsing, validation, and security. Use when implementing search endpoints, building data grids, or creating dynamic query APIs.
---
# API Filtering & Sorting
Build flexible filtering and sorting systems that handle complex queries efficiently.
## Query Parameter Syntax
```
GET /products?category=electronics&price[gte]=100&price[lte]=500&sort=-price,name
```
## Implementation (Node.js)
```javascript
const allowedFilters = ['category', 'status', 'price', 'createdAt'];
const allowedSorts = ['name', 'price', 'createdAt'];
app.get('/products', async (req, res) => {
const filter = {};
const sort = {};
// Parse filters
for (const [key, value] of Object.entries(req.query)) {
if (key === 'sort') continue;
const match = key.match(/^(\w+)\[(\w+)\]$/);
if (match) {
const [, field, operator] = match;
if (!allowedFilters.includes(field)) continue;
filter[field] = { [`$${operator}`]: parseValue(value) };
} else if (allowedFilters.includes(key)) {
filter[key] = value;
}
}
// Parse sort
if (req.query.sort) {
for (const field of req.query.sort.split(',')) {
const direction = field.startsWith('-') ? -1 : 1;
const name = field.replace(/^-/, '');
if (allowedSorts.includes(name)) sort[name] = direction;
}
}
const products = await Product.find(filter).sort(sort);
res.json({ data: products });
});
function parseValue(value) {
if (value === 'true') return true;
if (value === 'false') return false;
if (!isNaN(value)) return Number(value);
return value;
}
```
## Filter Operators
| Operator | Meaning | Example |
|----------|---------|---------|
| eq | Equals | `?status=active` |
| ne | Not equals | `?status[ne]=deleted` |
| gt/gte | Greater than | `?price[gte]=100` |
| lt/lte | Less than | `?price[lte]=500` |
| in | In array | `?status[in]=active,pending` |
| like | Contains | `?name[like]=phone` |
## Security
- Whitelist allowed filter fields
- Validate input types per field
- Index frequently-filtered columns
- Limit query complexity
- Prevent SQL/NoSQL injection
## Best Practices
- Support common operators
- Cache filter option lists
- Monitor query performance
- Provide sensible defaults
This skill builds flexible API filtering and sorting systems with robust query parameter parsing, validation, and security safeguards. It provides a production-oriented approach for search endpoints, data grids, and dynamic query APIs using TypeScript and Node.js patterns. The implementation focuses on whitelisting, operator support, and safe value parsing to avoid injection and performance issues.
The skill parses query parameters into filter and sort objects, supporting operators like eq, ne, gt/gte, lt/lte, in, and like. It enforces allowed filter and sort lists, converts string values to proper types (boolean, number, string), and maps operator syntax (e.g., price[gte]) into driver-friendly query clauses. It also recommends indexing, input validation, and query complexity limits to protect runtime performance and security.
How do you prevent injection attacks?
Whitelist fields, validate types and operators, and never concatenate raw strings into SQL/NoSQL queries; use parameterized queries or the database driver's query builder.
What operators should I support first?
Start with eq, ne, gt/gte, lt/lte, in, and like — they cover common needs without adding unnecessary complexity.
How should I handle arrays for the in operator?
Accept comma-separated values and parse them into typed arrays, validating element types against the field schema.