frappe-testing
Frappe Testing
Write and run tests for Frappe applications using the built-in testing framework.
When to use
- Writing unit tests for DocType controllers
- Writing integration tests for workflows and APIs
- Running existing test suites
- Debugging test failures
- Setting up CI pipelines for Frappe apps
Inputs required
- App name to test
- Site name for test execution
- Specific module/DocType to test (optional)
- Test environment (dev site, dedicated test site)
Procedure
0) Setup test environment
# Install dev dependencies
bench setup requirements --dev
# Ensure site is ready
bench --site <site> migrate
1) Run tests
# Run all tests for an app
bench --site <site> run-tests --app my_app
# Run tests for specific module
bench --site <site> run-tests --module my_app.my_module.tests
# Run tests for specific DocType
bench --site <site> run-tests --doctype "My DocType"
# Verbose output
bench --site <site> run-tests --app my_app -v
# Run single test file
bench --site <site> run-tests --module my_app.doctype.sample_doc.test_sample_doc
2) Write DocType tests
Create test_<doctype_name>.py alongside the DocType:
# my_app/doctype/sample_doc/test_sample_doc.py
import frappe
from frappe.tests.utils import FrappeTestCase
class TestSampleDoc(FrappeTestCase):
def setUp(self):
# Create test data
self.doc = frappe.get_doc({
"doctype": "Sample Doc",
"title": "Test Document"
}).insert()
def tearDown(self):
# Cleanup
frappe.delete_doc("Sample Doc", self.doc.name, force=True)
def test_creation(self):
self.assertEqual(self.doc.title, "Test Document")
def test_validation(self):
doc = frappe.get_doc({
"doctype": "Sample Doc",
"title": "" # Invalid - required field
})
self.assertRaises(frappe.ValidationError, doc.insert)
def test_workflow(self):
self.doc.status = "Approved"
self.doc.save()
self.assertEqual(self.doc.status, "Approved")
3) Write API tests
# my_app/tests/test_api.py
import frappe
from frappe.tests.utils import FrappeTestCase
class TestAPI(FrappeTestCase):
def test_whitelist_method(self):
from my_app.api import process_order
# Create test order
order = frappe.get_doc({
"doctype": "Sales Order",
"customer": "_Test Customer"
}).insert()
# Test the API
result = process_order(order.name, "approve")
self.assertEqual(result["status"], "success")
# Cleanup
frappe.delete_doc("Sales Order", order.name, force=True)
def test_permission_denied(self):
# Test as restricted user
frappe.set_user("guest@example.com")
from my_app.api import sensitive_action
self.assertRaises(frappe.PermissionError, sensitive_action, "doc-001")
# Reset user
frappe.set_user("Administrator")
4) Test permissions
def test_role_permissions(self):
# Create user with specific role
user = frappe.get_doc({
"doctype": "User",
"email": "test_user@example.com",
"roles": [{"role": "Sales User"}]
}).insert()
frappe.set_user("test_user@example.com")
# Test permission
self.assertTrue(frappe.has_permission("Sales Order", "read"))
self.assertFalse(frappe.has_permission("Sales Order", "delete"))
# Cleanup
frappe.set_user("Administrator")
frappe.delete_doc("User", user.name, force=True)
5) Use fixtures
# my_app/doctype/sample_doc/test_records.json
[
{
"doctype": "Sample Doc",
"title": "Test Record 1"
},
{
"doctype": "Sample Doc",
"title": "Test Record 2"
}
]
Reference in tests:
class TestSampleDoc(FrappeTestCase):
def test_fixture_loaded(self):
doc = frappe.get_doc("Sample Doc", "Test Record 1")
self.assertIsNotNone(doc)
6) Run UI tests (Cypress)
# Run UI tests for an app
bench --site <site> run-ui-tests my_app
# Headless mode
bench --site <site> run-ui-tests my_app --headless
Verification
- All tests pass:
bench --site <site> run-tests --app my_app - No test pollution (tests are isolated)
- Tests run in < 5 minutes for fast feedback
- CI pipeline runs tests on each commit
Failure modes / debugging
- Test not found: Ensure filename starts with
test_and class/method names follow conventions - Database errors: Tests may not be isolated—check for missing cleanup
- Permission errors in tests: Use
frappe.set_user("Administrator")in setup - Slow tests: Avoid unnecessary fixtures, mock external services
Escalation
- For complex fixtures, see references/fixtures.md
- For UI testing patterns, see references/cypress.md
- For CI setup, see references/ci-testing.md
References
- references/test-patterns.md - Common test patterns
- references/fixtures.md - Test data management
- references/cypress.md - UI testing
Guardrails
- Use test fixtures: Load test data via fixtures, not manual creation in each test
- Clean up test data: Delete created records in
tearDown()or usefrappe.db.rollback() - Mock external services: Never call real APIs in tests; mock HTTP calls
- Isolate tests: Each test should be independent; no reliance on test execution order
- Set user context explicitly: Use
frappe.set_user()to test as specific users
Common Mistakes
| Mistake | Why It Fails | Fix |
|---|---|---|
Not using FrappeTestCase |
Missing test setup/teardown | Extend frappe.tests.utils.FrappeTestCase |
| Missing db rollback | Test pollution, flaky tests | Use frappe.db.rollback() in tearDown or transactions |
| Flaky async tests | Intermittent failures | Use frappe.tests.utils.run_until() or proper async handling |
| Testing implementation not behavior | Brittle tests | Test outcomes, not internal method calls |
| Hardcoded test data | Conflicts with existing data | Use unique names like _Test Record {uuid} |
| Skipping permission tests | Security holes | Test with different user roles, not just Administrator |
More from lubusin/agent-skills
frappe-frontend-development
Build modern Vue 3 frontend apps using Frappe UI with components, data fetching, and portal pages. Use when creating custom frontends, SPAs, or portal interfaces for Frappe applications.
94frappe-app-development
Scaffold and architect custom Frappe apps including app structure, hooks, background jobs, service layers, and production hardening. Use when creating new apps, setting up app architecture, or implementing cross-cutting patterns like caching, logging, and error handling.
85frappe-router
Route to the appropriate Frappe skill based on task type. Use as the entry point when working on Frappe projects to determine which specialized skill to apply.
82frappe-api-development
Build REST and RPC APIs in Frappe including whitelisted methods, authentication, and permission handling. Use when creating custom endpoints, integrating with external systems, or exposing business logic via API.
82frappe-desk-customization
Customize Frappe Desk UI with form scripts, list view scripts, report scripts, dialogs, and client-side JavaScript APIs. Use when building interactive Desk experiences, adding custom buttons, or scripting form behavior.
80frappe-doctype-development
Create and modify Frappe DocTypes including schema design, controllers, child tables, and customization. Use when building data models, adding fields, or implementing document lifecycle logic.
78