Agent skill
clojure-review
Review Clojure and ClojureScript code changes for compliance with Metabase coding standards, style violations, and code quality issues. Use when reviewing pull requests or diffs containing Clojure/ClojureScript code.
Install this agent skill to your Project
npx add-skill https://github.com/aiskillstore/marketplace/tree/main/skills/metabase/clojure-review
SKILL.md
Clojure Code Review Skill
Metabase Clojure Style Guide
This guide covers Clojure and ClojureScript coding conventions for Metabase. See also: CLOJURE_STYLE_GUIDE.adoc for the Community Clojure Style Guide.
Naming Conventions
General Naming:
- Acceptable abbreviations:
acc,i,pred,coll,n,s,k,f - Use
kebab-casefor all variables, functions, and constants
Function Naming:
- Pure functions should be nouns describing the value they return (e.g.,
agenotcalculate-ageorget-age) - Functions with side effects must end with
! - Don't repeat namespace alias in function names
Destructuring:
- Map destructuring should use kebab-case local bindings even if the map uses
snake_casekeys
Documentation Standards
Docstrings:
- Every public var in
srcorenterprise/backend/srcmust have docstring - Format using Markdown conventions
- Reference other vars with
[[other-var]]not backticks
Comments:
TODOformat:;; TODO (Name M/D/YY) -- description
Code Organization
Visibility:
- Make everything
^:privateunless it is used elsewhere - Try to organize namespaces to avoid
declare(put public functions near the end)
Size and Structure:
- Break up functions > 20 lines
- Lines ≤ 120 characters
- No blank lines within definition forms (except pairwise
let/cond)
Style Conventions
Keywords and Metadata:
- Prefer namespaced keywords for internal use:
:query-type/normalnot:normal - Tag variables with
:arglistsmetadata if they're functions but wouldn't otherwise have it
Tests
Organization:
- Break large tests into separate
deftestforms for logically separate test cases - Test names should end in
-testor-test-<number>
Performance:
- Mark pure function tests
^:parallel
Modules
OSS Modules:
- Follow
metabase.<module>.*pattern - Source in
src/metabase/<module>/
Enterprise Modules:
- Follow
metabase-enterprise.<module>.*pattern - Source in
enterprise/backend/src/metabase_enterprise/<module>/
Module Structure:
- REST API endpoints go in
<module>.apior<module>.api.*namespaces - Put module public API in
<module>.coreusing Potemkin imports - Put Toucan models in
<module>.models.* - Put settings in
<module>.settings - Put schemas in
<module>.schema
Module Linters:
- Do not cheat module linters with
:clj-kondo/ignore [:metabase/modules]
REST API Endpoints
Required Elements:
- All new endpoints must have response schemas (
:- <schema>after route string) - All endpoints need Malli schemas for parameters (detailed and complete)
- All new REST API endpoints MUST HAVE TESTS
Naming Conventions:
- Query parameters use kebab-case
- Request bodies use
snake_case - Routes use singular nouns (e.g.,
/api/dashboard/:id)
Behavior:
GETendpoints should not have side effects (except analytics)defendpointforms should be small wrappers around Toucan model code
MBQL (Metabase Query Language)
Restrictions:
- No raw MBQL introspection outside of
lib,lib-be, orquery-processormodules - Use Lib and MBQL 5 in new source code; avoid legacy MBQL
Database and Models
Naming:
- Model names and table names should be singular nouns
- Application database uses
snake_caseidentifiers
Best Practices:
- Use
t2/select-one-fninstead of fetching entire rows for one column - Put correct behavior in Toucan methods, not separate helper functions
Drivers
Documentation:
- New driver multimethods must be mentioned in
docs/developers-guide/driver-changelog.md
Implementation:
- Driver implementations should pass
driverargument to other driver multimethods - Don't hardcode driver names in implementations
- Minimize logic inside
read-column-thunkin JDBC-based drivers
Miscellaneous
Examples:
- Example data should be bird-themed if possible
Linter Suppressions:
- Use proper format for kondo suppressions
- No
#_:clj-kondo/ignore(keyword form)
Configurable Options:
- Don't define configurable options that can only be set with environment variables
- Use
:internaldefsettinginstead
Linting and Formatting
- Lint PR:
./bin/mage kondo-updated master(or whatever target branch)- Call the command one time at the beginning, record the results, then work through the problems one at a time.
- If the solution is obvious, then please apply the fix. Otherwise skip it.
- If you fix all the issues (and verify by rerunning the kondo-updated command):
- commit the change with a succinct and descriptive commit message
- Lint File:
./bin/mage kondo <file or files>- Use the linter as a way to know that you are adhering to conventions in place in the codebase
- Lint Changes:
./bin/mage kondo-updated HEAD - Format:
./bin/mage cljfmt-files [path]
Testing
- Run a test:
./bin/mage run-tests namespace/test-name - Run all tests in a namespace:
./bin/mage run-tests namespace - Run all tests for a module:
./bin/mage run-tests test/metabase/notificationBecause the module lives in that directory.
Note: the ./bin/mage run-tests command accepts multiple args, so you can pass
./bin/mage run-tests namespace/test-name namespace/other-test namespace/third-test
to run 3 tests, or
./bin/mage run-tests test/metabase/module1 test/metabase/module2 to run 2 modules.
Code Readability
- Check Code Readability:
./bin/mage -check-readable <file> [line-number]- Run after every change to Clojure code
- Check specific line first, then entire file if readable
REPL Usage
Note: If you have
clojure-mcptools available (check for tools likeclojure_eval), always prefer those over./bin/mage -repl. The MCP tools provide better integration, richer feedback, and avoid shell escaping issues. Only use./bin/mage -replas a fallback when clojure-mcp is not available.
- Evaluating Clojure Code:
./bin/mage -repl '<code>'- See "Sending Code to the REPL" section for more details
Sending Code to the REPL
- Send code to the metabase process REPL using:
./bin/mage -repl '(+ 1 1)'where(+ 1 1)is your Clojure code.- See
./bin/mage -repl -hfor more details. - If the Metabase backend is not running, you'll see an error message with instructions on how to start it.
- See
Working with Files and Namespaces
- Load a file and call functions with fully qualified names:
To call your.namespace/your-function on arg1 and arg2:
./bin/mage -repl --namespace your.namespace '(your-function arg1 arg2)'
DO NOT use "require", "load-file" etc in the code string argument.
Understanding the Response
The ./bin/mage -repl command returns three separate, independent outputs:
value: The return value of the last expression (best for data structures)stdout: Any printed output fromprintlnetc. (best for messages)stderr: Any error messages (best for warnings and errors)
Example call:
./bin/mage -repl '(println "Hello, world!") '\''({0 1, 1 3, 2 0, 3 2} {0 2, 1 0, 2 3, 3 1})'
Example response:
ns: user
session: 32a35206-871c-4553-9bc9-f49491173d1c
value: ({0 1, 1 3, 2 0, 3 2} {0 2, 1 0, 2 3, 3 1})
stdout: Hello, world!
stderr:
For effective REPL usage:
- Return data structures as function return values
- Use
printlnfor human-readable messages - Print errors to stderr
Review guidelines
What to flag:
- Check compliance with the Metabase Clojure style guide (included above)
- If
CLOJURE_STYLE_GUIDE.adocexists in the working directory, also check compliance with the community Clojure style guide - Flag all style guide violations
What NOT to post:
- Do not post comments congratulating someone for trivial changes or for following style guidelines
- Do not post comments confirming things "look good" or telling them they did something correctly
- Only post comments about style violations or potential issues
Example bad code review comments to avoid:
This TODO comment is properly formatted with author and date - nice work!
Good addition of limit 1 to the query - this makes the test more efficient without changing its behavior.
The kondo ignore comment is appropriately placed here
Test name properly ends with -test as required by the style guide.
Special cases:
- Do not post comments about missing parentheses (these will be caught by the linter)
Quick review checklist
Use this to scan through changes efficiently:
Naming
- Descriptive names (no
tbl,zs') - Pure functions named as nouns describing their return value
-
kebab-casefor all variables and functions - Side-effect functions end with
! - No namespace-alias repetition in function names
Documentation
- Public vars in
srcorenterprise/backend/srchave useful docstrings - Docstrings use Markdown conventions
- References use
[[other-var]]not backticks -
TODOcomments include author and date:;; TODO (Name 1/1/25) -- description
Code Organization
- Everything
^:privateunless used elsewhere - No
declarewhen avoidable (public functions near end) - Functions under 20 lines when possible
- No blank lines within definition forms (except pairwise constructs in
let/cond) - Lines ≤ 120 characters
Tests
- Separate
deftestforms for distinct test cases - Pure tests marked
^:parallel - Test names end in
-testor-test-<number>
Modules
- Correct module patterns (OSS:
metabase.<module>.*, EE:metabase-enterprise.<module>.*) - API endpoints in
<module>.apinamespaces - Public API in
<module>.corewith Potemkin - No cheating module linters with
:clj-kondo/ignore [:metabase/modules]
REST API
- Response schemas present (
:- <schema>) - Query params use kebab-case, bodies use
snake_case - Routes use singular nouns (e.g.,
/api/dashboard/:id) -
GEThas no side effects (except analytics) - Malli schemas detailed and complete
- All new endpoints have tests
MBQL
- No raw MBQL manipulation outside
lib,lib-be, orquery-processormodules - Uses Lib and MBQL 5, not legacy MBQL
Database
- Model and table names are singular nouns
- Uses
t2/select-one-fninstead of selecting full rows for one column - Logic in Toucan methods, not helper functions
Drivers
- New multimethods documented in
docs/developers-guide/driver-changelog.md - Passes
driverargument to other driver methods (no hardcoded driver names) - Minimal logic in
read-column-thunk
Miscellaneous
- Example data is bird-themed when possible
- Kondo linter suppressions use proper format (not
#_:clj-kondo/ignorekeyword form)
Pattern matching table
Quick scan for common issues:
| Pattern | Issue |
|---|---|
calculate-age, get-user |
Pure functions should be nouns: age, user |
update-db, save-model |
Missing ! for side effects: update-db!, save-model! |
snake_case_var |
Should use kebab-case |
| Public var without docstring | Add docstring explaining purpose |
;; TODO fix this |
Missing author/date: ;; TODO (Name 1/1/25) -- description |
(defn foo ...) in namespace used elsewhere |
Should be (defn ^:private foo ...) |
| Function > 20 lines | Consider breaking up into smaller functions |
/api/dashboards/:id |
Use singular: /api/dashboard/:id |
Query params with snake_case |
Use kebab-case for query params |
| New API endpoint without tests | Add tests for the endpoint |
Feedback format examples
For style violations:
This pure function should be named as a noun describing its return value. Consider
userinstead ofget-user.
For missing documentation:
This public var needs a docstring explaining its purpose, inputs, and outputs.
For organization issues:
This function is only used in this namespace, so it should be marked
^:private.
For API conventions:
Query parameters should use kebab-case. Change
user_idtouser-id.
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
perigon-backend
Perigon ASP.NET Core + EF Core + Aspire conventions
perigon-agent
Pointers for Copilot/agents to apply Perigon conventions
perigon-angular
Angular 21+ standalone/Material/signal conventions for Perigon WebApp
fastapi-mastery
Comprehensive FastAPI development skill covering REST API creation, routing, request/response handling, validation, authentication, database integration, middleware, and deployment. Use when working with FastAPI projects, building APIs, implementing CRUD operations, setting up authentication/authorization, integrating databases (SQL/NoSQL), adding middleware, handling WebSockets, or deploying FastAPI applications. Triggered by requests involving .py files with FastAPI code, API endpoint creation, Pydantic models, or FastAPI-specific features.
context7-efficient
Token-efficient library documentation fetcher using Context7 MCP with 86.8% token savings through intelligent shell pipeline filtering. Fetches code examples, API references, and best practices for JavaScript, Python, Go, Rust, and other libraries. Use when users ask about library documentation, need code examples, want API usage patterns, are learning a new framework, need syntax reference, or troubleshooting with library-specific information. Triggers include questions like "Show me React hooks", "How do I use Prisma", "What's the Next.js routing syntax", or any request for library/framework documentation.
browser-use
Browser automation using Playwright MCP. Navigate websites, fill forms, click elements, take screenshots, and extract data. Use when tasks require web browsing, form submission, web scraping, UI testing, or any browser interaction.
Didn't find tool you were looking for?