Establishes the framework's foundation as a multi-site adoptable platform. ADRs (migrations/adr/): - ADR-001 (ACCEPTED): Asset is the platform contract; Machine retires. Three relationship types (partof, controls, connectedto) with free-text label, position-resolution chain (asset > related > location), hierarchical locations, sibling-bay propagation. - ADR-002 (ACCEPTED): Plugin contract semver via __contract_version__. - ADR-003 (ACCEPTED): Hybrid plugin distribution (in-tree bundled + filesystem-based external). - ADR-004 (ACCEPTED): Per-site instances, not multi-tenant. - ADR-005 (ACCEPTED): Equipment plugin (manufacturing) split from measuringtools plugin (metrology). Subtype-table pattern for protocol data (FOCAS, CLM, MTConnect). - ADR-006 (ACCEPTED): Plugin collector contract via get_collector_schema hook with API-key auth and identity-based upsert. Naming convention v1 (CONTRIBUTING.md): - DB tables/columns: lowercase concatenated, no underscores or dashes - DB-mirrored Python/JS variables match column names exactly; pure code follows host-language convention (PEP 8 / camelCase) - Closed acronym allowlist (universal + shop-floor domain), banned shorthand list with suffix exception (printers_bp etc allowed) - Plain ASCII everywhere: chat, docs, comments, string literals Style enforcement (scripts/check-naming-and-style.sh): - Pre-commit-runnable check script: non-ASCII, banned shorthand, snake_case DB names, snake_case API params in frontend - Fixes 14 violations across 11 files (Unicode arrows, snake_case params, ctx -> canvasContext, res -> response, req -> request_obj) Project state (CLAUDE.md, README.md, frontend/CLAUDE.md): - De-staled CLAUDE.md to reflect actual current state - README unifies DB story (MySQL canonical, SQLite test-only) - frontend/CLAUDE.md points at root convention Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
7.4 KiB
Contributing to ShopDB Flask
Coding Standards
Naming principles
- Spell it out. If a non-technical user looking at a report would not understand a column at a glance, rename it.
- No project-specific shorthand. A name either uses a word in full, or uses an acronym from the allowed list below.
- Consistency beats cleverness. Once a name is committed and used in code, do not rename it for marginal improvement.
- When unsure, ask: "would a new shop-floor IT person understand this?"
Database
- Table names: lowercase, concatenated, plural. No dashes, no underscores.
- Good:
machines,pctypes,networkdevices,businessunits,auditlogs - Bad:
machine_types,network-devices,BusinessUnits
- Good:
- Column names: lowercase, concatenated, singular. No dashes, no underscores.
- Good:
machineid,machinenumber,lastzabbixsync,isactive - Bad:
machine_id,last_zabbix_sync,IsActive
- Good:
- Foreign keys: referenced table name (singular) +
id.- Good:
locationidreferenceslocations,vendoridreferencesvendors
- Good:
- Booleans: prefix with
isorhas.- Good:
isactive,isshopfloor,hasprinter
- Good:
- Index names:
idx_<table>_<column>. Underscores allowed here only because indexes are infrastructure, not data.- Good:
idx_machines_locationid
- Good:
Allowed acronyms (closed list, expand by PR)
- Universal:
id,url,api,http,https,json,jwt,sql,os,ip,dns,csv,pdf,cors,ttl,uuid,html,css,orm - Domain (shop floor / IT infra):
cmm,cnc,pc,usb,vnc,winrm,ssh,ssl,tls,tcp,udp,smtp,ldap,vlan,sso,dnc,focas,clm,mtconnect
Any acronym not on these lists must be spelled out. Proposing a new acronym is a separate PR that updates this list.
Third-party imports are exempt. Library and stdlib module paths cannot be renamed. from sqlalchemy.orm import ..., import importlib.util, from werkzeug.utils import ... are all fine even if they contain words that would otherwise be banned.
Banned shorthand
cfg, ctx, mgr, req, res, env, util, helper, and any other domain-specific shortening. Spell it out: config, context, manager, request, response, environment, utilities. Note: db as a standalone variable name is banned, but db as a prefix in database or as part of project name shopdb is fine.
Suffix exception. Banned shorthand applies to standalone identifiers. As a suffix where the prefix establishes meaning, it is acceptable:
printers_bp,auth_bp,network_bpare fine. The prefix names what blueprint, the_bpsuffix is conventional for Flask blueprints.request_objis fine when disambiguating from importedrequest.bpalone as a variable name is not fine. Always pair with a meaningful prefix.
Code
Python
- Variables and functions holding a database value: match the column name exactly. No conversion to snake_case.
- DB column
machineid-> Python variablemachineid, attributeMachine.machineid, dict key{"machineid": 1} - DB column
lastzabbixsync-> Python variablelastzabbixsync
- DB column
- Pure code variables and functions (loop counters, helpers, anything that does not mirror a DB field): snake_case per PEP 8.
- Good:
loop_count,current_user,validate_input()
- Good:
- Classes: PascalCase. Spell out fully.
- Good:
PrinterData,AssetType,NetworkDevice
- Good:
- Modules / file names: lowercase. Underscores allowed where it improves readability (PEP 8 permits this for modules).
- Good:
employee_service.py,asset_routes.py
- Good:
JavaScript / Vue
- Variables: camelCase per JS convention.
- Good:
currentUser,isLoading
- Good:
- Variables holding API field values: match the API field name exactly. Do NOT convert to camelCase.
- API returns
{"machineid": 1}-> JSresponse.machineid, NOTresponse.machineId - API returns
{"lastzabbixsync": "..."}-> JSdevice.lastzabbixsync
- API returns
- Components: PascalCase. Spell out fully.
- Good:
AssetDetail.vue,MachineForm.vue,PrinterList.vue
- Good:
- CSS classes: lowercase with dashes.
- Good:
asset-detail,machine-form
- Good:
API
- Endpoints: lowercase, plural nouns. No underscores or dashes.
- Good:
/api/machines,/api/networkdevices,/api/businessunits
- Good:
- Query parameters: lowercase, concatenated. Match column names where applicable.
- Good:
?locationid=5,?isactive=true
- Good:
- Response keys: match database column names exactly.
- Good:
{"machineid": 1, "lastzabbixsync": "2026-01-12T10:00:00Z"}
- Good:
Examples
# Good - DB-mirrored fields keep column names; pure code uses snake_case
class PrinterData(db.Model):
__tablename__ = 'printerdata'
machineid = db.Column(db.Integer, db.ForeignKey('machines.machineid'))
lastzabbixsync = db.Column(db.DateTime)
isnetworkprinter = db.Column(db.Boolean)
def list_printers_by_location(locationid: int) -> list[dict]:
printers = PrinterData.query.filter_by(locationid=locationid).all()
result_count = len(printers)
return [{"machineid": p.machineid, "isnetworkprinter": p.isnetworkprinter} for p in printers]
# Bad - PEP 8-style underscores on DB fields, banned shorthand, project acronym not on allowed list
class PrinterData(db.Model):
__tablename__ = 'printer_data'
machine_id = db.Column(db.Integer)
last_zabbix_sync = db.Column(db.DateTime)
is_net_prn = db.Column(db.Boolean)
def list_prns_by_loc(loc_id):
bp_ctx = get_context()
cfg = load_cfg()
return PrinterData.query.filter_by(loc_id=loc_id).all()
Style policy (non-naming)
- No emojis in code, comments, documentation, string literals, or UI.
- No em-dashes (U+2014), en-dashes (U+2013), Unicode arrows, or smart quotes anywhere. Plain ASCII only.
- Comments default to none. Add one only when the WHY is non-obvious. Keep comments terse (lite caveman style is fine for inline
#and//comments). Docstrings stay normal English. - Dark theme is default. Keep UI functional and professional.
File structure
- Core models:
shopdb/core/models/ - Core API routes:
shopdb/core/api/ - Core services:
shopdb/core/services/ - Plugin code:
plugins/<plugin>/{models,api,services,schemas}/
Plugin development
When creating a new plugin:
- Create directory
plugins/<name>/. Use the scaffold (flask plugin new <name>) when available. - Include
manifest.jsonwith metadata. This is the single source of truth for plugin name, version, dependencies, api_prefix. - Extend
BasePluginclass. - Follow naming conventions above.
- Run
flask plugin install <name>to install.
See docs/PLUGIN-QUICKSTART.md (when published) for the 30-minute walkthrough.
Testing
Run tests with:
pytest tests/
Tests are required for:
- New routes (smoke + happy path + 1-2 edge cases)
- New plugin hooks (contract test)
- Schema migrations (before/after data integrity)
Code review checklist
- No underscores in table or column names
- No banned shorthand or unapproved acronyms in any new identifier
- DB-mirrored Python and JS variables match column names exactly
- API response keys match column names exactly
- No em-dashes, en-dashes, smart quotes, Unicode arrows, or emojis (run
grep -rP '[^\x00-\x7F]' --include='*.py' --include='*.vue' --include='*.js' .) - Plugin follows
BasePlugininterface and contract version - Tests included for new functionality