frappe-ops-app-lifecycle
$
npx mdskill add Impertio-Studio/Frappe_Claude_Skill_Package/frappe-ops-app-lifecycleAutomate end-to-end Frappe app creation, configuration, and deployment.
- Prevents broken app structure and missing setup.py fields.
- Integrates with bench CLI, Git, and Frappe marketplace.
- Executes commands based on app lifecycle stage requirements.
- Delivers success by generating working apps ready for production.
SKILL.md
.github/skills/frappe-ops-app-lifecycleView on GitHub ↗
---
name: frappe-ops-app-lifecycle
description: >
Use when scaffolding a new Frappe app, configuring app settings, building assets, running tests, deploying, updating, or publishing to marketplace.
Prevents broken app structure from incorrect scaffolding, missing setup.py fields, and failed builds.
Covers bench new-app, app directory structure, setup.py/pyproject.toml, hooks.py config, bench build, bench run-tests, app publishing.
Keywords: app lifecycle, new-app, scaffolding, setup.py, pyproject.toml, hooks.py, bench build, app publishing, marketplace, create app, publish app, app structure, how to start new app, app directory layout..
license: MIT
compatibility: "Claude Code, Claude.ai Projects, Claude API. Frappe v14-v16."
metadata:
author: OpenAEC-Foundation
version: "2.0"
---
# App Lifecycle Management
## Quick Reference
| Command | Purpose | When to Use |
|---|---|---|
| `bench new-app` | Scaffold new app | Starting a new project |
| `bench get-app URL` | Clone from Git | Installing existing app |
| `bench --site SITE install-app` | Install on site | After get-app or new-app |
| `bench --site SITE remove-app` | Uninstall from site | Removing app from site |
| `bench remove-app` | Remove from bench | Removing app entirely |
| `bench --site SITE migrate` | Run patches + sync | After code changes |
| `bench build` | Compile assets | After JS/CSS changes |
| `bench --site SITE console` | Python REPL | Debugging |
| `bench start` | Start dev server | Development |
| `bench setup production` | Configure nginx+supervisor | Deploying to production |
## 1. Scaffolding: bench new-app
```bash
bench new-app my_custom_app
```
Interactive prompts:
- App Title → Human-readable name
- App Description → One-line summary
- App Publisher → Company/author name
- App Email → Contact email
- App Icon → Default: `octicon octicon-file-directory`
- App Color → Default: `grey`
- App License → Default: `MIT`
### Generated Directory Structure
```
apps/my_custom_app/
├── MANIFEST.in # Files included in Python package
├── README.md # Project readme
├── license.txt # License file
├── requirements.txt # Python dependencies
├── dev-requirements.txt # Dev-only Python deps (v15+)
├── package.json # Node.js dependencies
├── setup.py # Python package config (v14)
├── pyproject.toml # Python package config (v15+)
├── my_custom_app/
│ ├── __init__.py # App version string
│ ├── hooks.py # Framework integration hooks
│ ├── modules.txt # List of app modules
│ ├── patches.txt # Migration patches list
│ ├── config/
│ │ ├── __init__.py
│ │ ├── desktop.py # Desktop/workspace config
│ │ └── docs.py # Documentation config
│ ├── public/ # Static assets → /assets/my_custom_app/
│ │ ├── css/
│ │ └── js/
│ ├── templates/ # Jinja templates
│ └── www/ # Portal pages (URL = path)
```
### What Each Core File Does
| File | Purpose | NEVER Forget |
|---|---|---|
| `__init__.py` | Defines `__version__` | ALWAYS update before release |
| `hooks.py` | ALL framework integration | Entry point for everything |
| `modules.txt` | Declares app modules | ALWAYS add new modules here |
| `patches.txt` | Migration patch registry | ALWAYS add patches in order |
| `requirements.txt` | Python deps installed on setup | Add pip packages here |
| `public/` | Static files served by nginx | Accessible at `/assets/app_name/` |
| `www/` | Portal pages | Filename = URL path |
## 2. Development Cycle
```
Code → Migrate → Build → Test → Commit
```
### Step-by-Step
```bash
# 1. Make code changes (DocTypes, reports, APIs, etc.)
# 2. Migrate — sync DocType schema + run patches
bench --site mysite migrate
# 3. Build — compile JS/CSS assets
bench build --app my_custom_app
# 4. Test — run Python tests
bench --site mysite run-tests --app my_custom_app
# 5. Commit
git -C apps/my_custom_app add -A && git -C apps/my_custom_app commit -m "feat: add feature"
```
ALWAYS run `bench migrate` after modifying DocType JSON files.
ALWAYS run `bench build` after modifying JS/CSS files.
## 3. Getting Apps from Git
```bash
# Public repo
bench get-app https://github.com/org/my_app
# Specific branch
bench get-app https://github.com/org/my_app --branch develop
# Private repo via SSH
bench get-app git@github.com:org/private_app.git
# Private repo via token (v15+)
bench get-app https://TOKEN@github.com/org/private_app.git
```
After `get-app`, ALWAYS install on the target site:
```bash
bench --site mysite install-app my_app
```
`get-app` clones to `apps/` and adds to `apps.txt`.
`install-app` creates database tables and runs `after_install` hooks.
## 4. Installing and Removing Apps
### Installation Order Matters
Apps are installed in order listed in `apps.txt`. If App B depends on App A, App A MUST be listed first.
```bash
# Install
bench --site mysite install-app my_app
# Verify
bench --site mysite list-apps
# Output: frappe, erpnext, my_app
# Remove from site (keeps code in apps/)
bench --site mysite remove-app my_app
# Remove from bench entirely (deletes code)
bench remove-app my_app
```
### App Dependencies (v14+)
Declare in `hooks.py`:
```python
required_apps = ["frappe", "erpnext"]
```
Frappe ALWAYS checks `required_apps` during installation and blocks if dependencies are missing.
## 5. Debugging with bench console
```bash
bench --site mysite console
```
Opens an IPython REPL with Frappe context:
```python
# Query data
frappe.db.sql("SELECT name, status FROM `tabSales Invoice` LIMIT 5", as_dict=True)
# Get a document
doc = frappe.get_doc("Sales Invoice", "SINV-00001")
print(doc.grand_total)
# Test a whitelisted method
from my_app.api import my_function
result = my_function(param="value")
# Check configuration
frappe.get_site_config()
# Auto-reload on code changes (v15+)
# Start with: bench --site mysite console --autoreload
```
ALWAYS use `bench console` for debugging — NEVER modify production data with raw SQL.
## 6. Development Mode vs Production Mode
### Development Mode
```bash
# Enable
bench set-config -g developer_mode 1
# Start dev server (Procfile: web + worker + redis + socketio)
bench start
```
Development mode enables:
- DocType editing in Desk
- "Is Standard" option for reports/scripts
- Auto-reload on Python file changes
- Detailed error tracebacks in browser
- `dev-requirements.txt` dependencies installed
### Production Mode
```bash
# Disable developer mode
bench set-config -g developer_mode 0
# Setup production (nginx + supervisor)
sudo bench setup production USERNAME
# Restart
sudo supervisorctl restart all
# or
sudo systemctl restart supervisor
```
Production mode:
- Serves via nginx (port 80/443)
- Background workers via supervisor
- Static files served directly by nginx
- Errors logged to files, not browser
- NEVER enable `developer_mode` on production sites
## 7. Asset Building
### v15+ (esbuild)
```bash
# Build all apps
bench build
# Build specific app
bench build --app my_custom_app
# Watch mode (auto-rebuild on changes)
bench watch
```
### v14 (build.json)
v14 uses `build.json` in the app root to map source files to bundles:
```json
{
"css/my_app.css": [
"public/css/style.css"
],
"js/my_app.js": [
"public/js/main.js"
]
}
```
### Asset Include in hooks.py
```python
# Desk (backend UI)
app_include_js = "my_app.bundle.js" # v15+ bundle syntax
app_include_css = "my_app.bundle.css"
# Portal (website)
web_include_js = "my_app_web.bundle.js"
web_include_css = "my_app_web.bundle.css"
# v14 legacy syntax
app_include_js = "/assets/my_app/js/my_app.js"
app_include_css = "/assets/my_app/css/my_app.css"
```
ALWAYS run `bench build` after changing JS/CSS files.
ALWAYS run `bench clear-cache` if assets are not updating.
## 8. App Versioning
### Version String in __init__.py
```python
# my_custom_app/__init__.py
__version__ = "1.2.0"
```
ALWAYS use semantic versioning: `MAJOR.MINOR.PATCH`
- MAJOR: Breaking changes
- MINOR: New features (backward compatible)
- PATCH: Bug fixes
The version is read by `bench version`, displayed in Desk, and used by the Marketplace.
### Checking Versions
```bash
bench version
# frappe 15.23.0
# erpnext 15.18.0
# my_custom_app 1.2.0
```
## 9. Patches: Data Migrations
### Writing a Patch
```python
# my_app/patches/v1_2/update_customer_status.py
import frappe
def execute():
frappe.reload_doc("module_name", "doctype", "customer_extension")
frappe.db.sql("""
UPDATE `tabCustomer Extension`
SET status = 'Active'
WHERE status IS NULL
""")
frappe.db.commit()
```
### Registering in patches.txt
```
# patches.txt — v14+ supports sections
[pre_model_sync]
my_app.patches.v1_1.fix_old_data
my_app.patches.v1_2.rename_field_before_schema
[post_model_sync]
my_app.patches.v1_2.update_customer_status
my_app.patches.v1_2.migrate_settings
```
**Section timing** (v14+):
- `[pre_model_sync]` — Runs BEFORE DocType schema changes are applied
- `[post_model_sync]` — Runs AFTER schema changes (new fields available)
- No section header — Runs in `[pre_model_sync]` by default
### Patch Rules
- ALWAYS add new patches at the END of their section
- Patches run ONCE — tracked in `tabPatch Log`
- To re-run a patch, append a comment: `my_app.patches.v1_2.fix #2025-03-20`
- ALWAYS call `frappe.reload_doc()` before accessing new/modified DocTypes
- ALWAYS use `[post_model_sync]` for patches that need new fields
- One-liner patches: `execute:frappe.delete_doc("Page", "old-page", ignore_missing=True)`
### Testing a Patch
```bash
# Run all pending patches
bench --site mysite migrate
# Run a specific patch manually in console
bench --site mysite console
>>> from my_app.patches.v1_2.update_customer_status import execute
>>> execute()
>>> frappe.db.commit()
```
## 10. Publishing to Frappe Marketplace
### Prerequisites Checklist
1. App hosted on public GitHub repository
2. `setup.py` or `pyproject.toml` with correct metadata
3. Valid `__version__` in `__init__.py`
4. README.md with installation instructions
5. All tests passing
### setup.py (v14)
```python
from setuptools import setup, find_packages
setup(
name="my_custom_app",
version="1.0.0",
description="My Custom App for ERPNext",
author="Your Name",
author_email="you@example.com",
packages=find_packages(),
zip_safe=False,
include_package_data=True,
install_requires=["frappe"],
)
```
### pyproject.toml (v15+)
```toml
[project]
name = "my_custom_app"
dynamic = ["version"]
requires-python = ">=3.10,<3.13"
dependencies = ["frappe"]
[build-system]
requires = ["flit_core >=3.4,<4"]
build-backend = "flit_core:buildapi"
```
### Publishing Steps
1. Create account at https://frappecloud.com/marketplace
2. Add your GitHub repository
3. Configure supported versions (v14, v15)
4. Submit for review
5. After approval, app appears in Marketplace
## 11. App Update Lifecycle on Client Sites
```bash
# Pull latest code
bench update --pull
# Or update specific app
cd apps/my_custom_app && git pull origin main && cd ../..
# Then migrate (runs patches + syncs schema)
bench --site mysite migrate
# Rebuild assets
bench build --app my_custom_app
# Restart workers
bench restart
```
The `bench update` command wraps: backup → pull → requirements → migrate → build → restart.
ALWAYS take a backup before running `bench update` on production.
ALWAYS test updates on staging before applying to production.
## See Also
- [references/examples.md](references/examples.md) — Complete app scaffolding examples
- [references/anti-patterns.md](references/anti-patterns.md) — Common mistakes
- [references/workflows.md](references/workflows.md) — Step-by-step workflows
- [references/module-workspace-shipping.md](references/module-workspace-shipping.md) — Module Def, modules.txt, and workspace shipping
- `frappe-syntax-hooks` — Complete hooks.py reference
- `frappe-core-database` — Database and migration patterns
- `frappe-impl-workspace` — Workspace builder, components, and customization