Files
shopdb-flask/CONTRIBUTING.md
cproudlock d6725c08e0 Phase 0: lock platform contract, naming convention, and style enforcement
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>
2026-05-08 14:47:30 -04:00

7.4 KiB

Contributing to ShopDB Flask

Coding Standards

Naming principles

  1. Spell it out. If a non-technical user looking at a report would not understand a column at a glance, rename it.
  2. No project-specific shorthand. A name either uses a word in full, or uses an acronym from the allowed list below.
  3. Consistency beats cleverness. Once a name is committed and used in code, do not rename it for marginal improvement.
  4. 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
  • Column names: lowercase, concatenated, singular. No dashes, no underscores.
    • Good: machineid, machinenumber, lastzabbixsync, isactive
    • Bad: machine_id, last_zabbix_sync, IsActive
  • Foreign keys: referenced table name (singular) + id.
    • Good: locationid references locations, vendorid references vendors
  • Booleans: prefix with is or has.
    • Good: isactive, isshopfloor, hasprinter
  • Index names: idx_<table>_<column>. Underscores allowed here only because indexes are infrastructure, not data.
    • Good: idx_machines_locationid

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_bp are fine. The prefix names what blueprint, the _bp suffix is conventional for Flask blueprints.
  • request_obj is fine when disambiguating from imported request.
  • bp alone 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 variable machineid, attribute Machine.machineid, dict key {"machineid": 1}
    • DB column lastzabbixsync -> Python variable lastzabbixsync
  • 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()
  • Classes: PascalCase. Spell out fully.
    • Good: PrinterData, AssetType, NetworkDevice
  • Modules / file names: lowercase. Underscores allowed where it improves readability (PEP 8 permits this for modules).
    • Good: employee_service.py, asset_routes.py

JavaScript / Vue

  • Variables: camelCase per JS convention.
    • Good: currentUser, isLoading
  • Variables holding API field values: match the API field name exactly. Do NOT convert to camelCase.
    • API returns {"machineid": 1} -> JS response.machineid, NOT response.machineId
    • API returns {"lastzabbixsync": "..."} -> JS device.lastzabbixsync
  • Components: PascalCase. Spell out fully.
    • Good: AssetDetail.vue, MachineForm.vue, PrinterList.vue
  • CSS classes: lowercase with dashes.
    • Good: asset-detail, machine-form

API

  • Endpoints: lowercase, plural nouns. No underscores or dashes.
    • Good: /api/machines, /api/networkdevices, /api/businessunits
  • Query parameters: lowercase, concatenated. Match column names where applicable.
    • Good: ?locationid=5, ?isactive=true
  • Response keys: match database column names exactly.
    • Good: {"machineid": 1, "lastzabbixsync": "2026-01-12T10:00:00Z"}

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:

  1. Create directory plugins/<name>/. Use the scaffold (flask plugin new <name>) when available.
  2. Include manifest.json with metadata. This is the single source of truth for plugin name, version, dependencies, api_prefix.
  3. Extend BasePlugin class.
  4. Follow naming conventions above.
  5. 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 BasePlugin interface and contract version
  • Tests included for new functionality