• Editorial pass per user direction: stop justifying the architecture.
    For every "why this design works" passage, name the costs the design
    imposes — not as a parenthetical aside but as substantive critical
    analysis. Each major architectural-claim page now carries an explicit
    drawbacks/costs section.
    
    Pages revised:
    
    concepts/thesis.md
    - "The reward" → "What the design enables (and what each enabler still costs)":
      for each promised benefit (single codebase, PMs evolve without
      engineering, customisations layered cleanly), name the limit. Added
      closing observation that data-driven design redistributes complexity
      to people and tools the framework can't compile-check.
    - "When it breaks down": rewrote to call out that "bypassing the
      framework" via 18 customer dirs makes the data-driven thesis
      partial, not complete.
    
    concepts/semantic-fk.md
    - "Why xly disabled FKs": added critical analysis. Both reasons
      could be addressed surgically; the chosen "no FKs anywhere" is the
      trade for DB-enforced integrity, paid every day the system runs.
    
    concepts/master-slave.md
    - "Slave naming caveat": stop framing retention as wise pragmatism.
      The naming was a poor choice; preservation has a real ongoing cost.
    
    concepts/modules-forms-vtables.md
    - "Three nouns, one engine": the universal dispatch path concentrates
      3,500+ lines + edge cases + special-case hardcodes in one class.
      Naming the trade.
    
    concepts/multi-tenancy.md
    - "How the design scales" → "How the design scales — and where it
      doesn't": shared schema = shared contention; tenant-filter index
      discipline; no physical hard-delete; rigid (sBrandsId,
      sSubsidiaryId) tenancy unit.
    
    concepts/customization-channels.md
    - Soften "90%+ should live here" claim — that's an aspirational
      target, not a measured fact. The 18 customer override directories
      are evidence the channel-2 demand is non-trivial.
    
    concepts/api-surface.md
    - "Why three tiers, not one" → "Why three tiers (and what splitting
      them costs)": three WARs to deploy, duplicate code, no shared
      session, three reverse-proxy entries. Note the alternative
      (single-WAR with package boundaries) and what that would cost
      vs gain.
    
    reference/maintainer/proc-dispatch.md
    - "Why dynamic proc dispatch matters": added five concrete costs
      (no compile-time check, no type safety, no call-site discoverability,
      no static analysis, broken stack traces). Reframed: dynamic
      dispatch made it cheap to keep adding procs, which made the pile
      grow, which made the pile harder to audit.
    
    reference/maintainer/cache-invalidation.md
    - New "Drawbacks of this design" section: confusing co-named systems,
      eviction in same transaction as write (silent corruption on
      Redis outage), allEntries=true blunt eviction, no batching,
      direct DB writes bypass everything. Also fixed the "if cache is
      local" hedge in section 3 (we've now empirically confirmed Redis-
      backed, so cache is shared).
    
    reference/maintainer/bi-engine.md
    - New "Drawbacks of the homebrewed approach" section: every chart
      needs a SQL author, charts run heavy SQL on OLTP DB, no semantic
      consistency between charts, no drill-down, customer-divergent KPI
      logic. Also dedup'd the duplicated "What this is not" section.
    
    reference/maintainer/sql-templates.md
    - "Why this is a 'template' library and not a code generator" →
      added costs: no enforcement, no regeneration, no template-origin
      tracking, customer overrides drift from scaffold. The 1,687 procs
      the schema carries are the evidence that "discipline rather than
      enforcement" doesn't fully hold.
    
    reference/maintainer/activiti.md
    - "Why this design works for xly's audience" → "Why xly avoided
      Activiti — and what that costs": scattered workflow logic, no
      central audit trail, no parallel-branch/reassignment, invisible
      flow-graph evolution, idle Activiti engine paying boot cost
      anyway.
    - "Why xly bothered with Activiti at all" → "Why xly bothered with
      Activiti — and whether it was worth it": named the costs (second
      engine, second schema, second auth surface, modeler UI to learn)
      and the damning fact that on this dev DB the engine is idle. A
      future cleanup could plausibly remove Activiti entirely.
    
    reference/maintainer/runtime.md
    - New "What 'universal CRUD' means in practice" section: 3,500-line
      single-point-of-failure class, no type system on Map<String,Object>,
      poor discoverability ("what endpoints write to table X" is
      unanswerable). The trade: adding a module is essentially free,
      touching the runtime essentially never is.
    - Updated cache-invalidation cross-link to drop the "open question"
      hedge (now empirically resolved).
    
    slices/04-custom-field.md
    - "Why it works without code changes" → "Why it works without code
      changes — and what that costs": merge runs on every request,
      three near-empty tables on every schema, display-only extension
      (real persisted fields still need ALTER TABLE), debuggability
      requires diffing 3 overlay tables.
    
    slices/05-customer-sql-override.md
    - Added drawbacks: no version control on the deployed body, no
      type-safety bridge, compounds the BI problem. Reframed the
      "right rule of thumb": 18 customer override directories suggest
      the channel-2 demand is structural, not exceptional — that's
      evidence the metadata model isn't expressive enough, not a
      celebration of the escape hatch.
    
    slices/06-hardware.md
    - "The cleanest story xly tells about an awkward problem" →
      removed the "cleanest" framing. Added costs of "DB as the only
      contract": no backpressure, no request/response, bridge-side
      state invisible to the framework, three layers of polling
      multiply latency, hardest code (byte protocols) gets least CI.
      A real-time-aware architecture would use streaming end-to-end;
      xly's choice trades latency, observability, flow control for
      operational simplicity. Liveable for press tempo, not for
      faster shop-floor signals.
    zichun authored
     
    Browse Code »

  • Live findings from BACK + FROUNT browser session:
    - /business/addSysLocking: undocumented optimistic-lock endpoint
      (POST, called when entering edit mode; handler at
      BusinessBaseController.java:400-407). Added to internal.md universal-CRUD
      table.
    - BACK admin sidebar fully enumerated (10 modules) and mapped to
      backing form-master sTbName + owning service. 8/10 are framework
      primitives covered elsewhere; 常用操作配置 has no
      gdsconfigformmaster row in this dev DB (SPA-side admin
      special-case); 图表配置 is gdsconfigcharmaster/slave (added to
      the gds prefix entry in modules-forms-vtables.md).
    - FROUNT login confirmed; this dev DB exposes only 4 sidebar
      modules (2 are AI/LLM out-of-scope). Production tenants would
      carry the full business-module catalog.
    - Slice 1 save endpoint live-corroborated: clicking 新增 → 保存
      fired POST /xlyEntry/business/addUpdateDelBusinessData?sModelsId=13
      → 200 OK. (Audit-tag mutation didn't actually land because the
      Vue model didn't pick up our scripted input change — endpoint
      flowed but with original row state.) Updated slice-01 open-
      verification item with the partial verification.
    
    Dead-source flag (sql-templates.md):
    - Two `FileSqlUtil` classes coexist. xlyFlow's loader points at the
      8 templates that exist on disk. xlyApi's loader names 7
      templates (sInSqlStrTemple, sOutSqlStrTemple, sDataSqlTmp[Def],
      sJsonSqlTmp[Out], sDbPro) — NONE of them exist as files.
      Documented as broken-by-default; either restore templates or
      remove the loader.
    
    Auto-catalog regen verified accurate (gdsmodule.md columns match
    live DB).
    zichun authored
     
    Browse Code »
  • Audit every concrete claim in the 41 hand-written en pages against the
    three primary sources (DB, source on cleanup branch, source-tree
    inventory). Fix divergent claims in place; preserve framing where
    verified.
    
    Substantive corrections:
    
    - request-lifecycle / runtime / slice 01: the metadata read sources from
      five tables/families (gdsconfigformmaster + overlays, gdsformconst,
      sysjurisdiction, sysbillnosettings, sysreport), not four. The map key
      `gdsjurisdiction` is misleading — the per-user grant read queries
      `sysjurisdiction`; `gdsjurisdiction` is the builder-side action
      catalogue. `gdsformconst`, `gdsconfigformmaster`, `gdsconfigformslave`
      are NOT tenant-scoped; they filter by form-id only.
    - multi-tenancy: four metadata tables (gdsformconst, gdsmodule,
      gdsconfigformmaster, gdsconfigformslave) are an explicit exception to
      the "every table tenant-scoped" promise — `sTableNameList` strips
      sBrandsId/sSubsidiaryId from writes against them.
    - sSaveProName / sSaveProNameBefore are pre/post-save HOOKS on top of
      the always-running base path (BusinessBaseServiceImpl.add/update),
      not either/or branches. Default add/update path is in
      BusinessBaseServiceImpl, not AddDelUpdCommonServiceImpl.
    - cache-invalidation: redis cache is cleared synchronously in BACK via
      @CacheEvict on CleanRedisServiceImpl during save. The JMS
      CHANGE_GDS_MODULE queue triggers PRO_ERPMERGEBASEGDSMODULE (base-data
      merge), NOT cache invalidation despite the name. Cross-node
      coherence open question (no custom CacheManager bean configured).
    - messaging: enumerate all 24 P2pQueue destinations grouped by intent;
      fix CHANGE_GDS_MODULE description; clarify single Consumer.java with
      24 @JmsListener methods (not 24 listener classes).
    - API paths: /checkflow lowercase (mapping value, not class name);
      /procedureCall/doGenericProcedureCall (not /business/genericProcedureCall*).
    - tech-stack: Druid 6 java imports + 16 yml mentions (was 25 conflated);
      fastjson per-module xlyInterface 9 (was 10); commons-lang3 39 (was 41);
      @Document classes 20 PLAT_* + 2 DIKE_TEST* (was "all PLAT_*"); xlyPersist
      activiti hit is IdGen.java (was BaseDao.java); add Springfox to
      declared-but-no-imports table; reconcile module list to 11 framework
      core + xlyPlc plugin + xlyPlatConstant utility.
    - index.md: clarify xlyFace as "in build, not documented"; add xlyErpTask
      / xlyPlatTask scheduler bullet; correct MongoDB framing (caller is in
      xlyPersist with no consumers, not xlyPlat*); add xlyPlc note; extend
      backup-table OOS to cover *_copy1 / *_history / *YYYYMMDD[HHMMSS].
    - deployment.md: split deployable Boot apps from library modules;
      enumerate 12 commented-out includes (was 3); remove xlyPlatConstant from
      out-of-scope Plat* list; split profile permutations by service.
    - activiti.md: add xlyApi to 5.17 dependency list; replace speculative
      BPMN path hint with verified state; name actual ActivitiConfig.java;
      note act_id_* are views projecting xly users into Activiti shapes.
    - api-reference/external.md: fix bearer-token validation flow (sysapibrand
      via AES-decrypted corpid, not sysapithirdtoken); /online/* are page
      renders not API execution; /pro/* mostly returns Thymeleaf views; mark
      sysapidbtodb as xlyFlow-owned; /token/getToken accepts GET and POST.
    - api-reference/webhooks.md: add Swagger Docket caveat (UI shell ships but
      no Docket bean → /v2/api-docs effectively empty); flag /send/sendQw as
      stub (returns "ok").
    - slices/03-report.md: fix dir path xlyEntry/com/xly/report/ →
      xlyEntry/com/xly/web/report/; reframe PrintReportControllerOld as dead
      source (file body fully commented out).
    - concepts/modules-forms-vtables.md: add 22-prefix glossary table
      (gds/sys/sis/sft/ele/mft/sal/quo/acc/pur/ops/cah/sgd/ept/mit/pit/qly/
      kpi/udf/viw_/plat_/ai_/act_/qrtz_) so a maintainer can enumerate
      business-data domains at a glance.
    - concepts/master-slave.md: disambiguate document-row pattern from
      DataSource master/slave (different concept, name overlap).
    - proc-dispatch.md: add proc-name molds (Sp_*_BeforeSave/AfterSave/
      SaveReturn, sp_btn_*, PRO_ERPMERGE*) + function-layer paragraph
      (Fun_*/Fn_*/get_*; SQL-called, not Java-dispatched).
    - concepts/index.md: schema label MySQL\nxlyweberp → xlyweberp_*.
    
    Pass E (live behavioural traces) deferred — source/DB-side audit was
    thorough; live traces best done as a follow-up sweep against a running
    instance.
    zichun authored
     
    Browse Code »

  • Verified every factual claim in the hand-written prose against the
    codebase under ../xly/ and the live xlyweberp_saas_ai schema, and fixed
    the drift:
    
    - semantic-fk: clarified — zero FKs on xly tables; the ones that exist
      are all on bundled Activiti / Quartz schemas, which the runtime
      doesn't join through.
    - multi-tenancy / runtime / slice 2: corrected RequestAddParamUtil shape
      (16 keys, 56 lines), noted the parallel xlyApi copy.
    - slice 1: corrected line ranges for addUpdateDelBusinessData;
      clarified that sTableNameList is a cache-invalidation gate, not an
      authorisation gate; corrected the gdsroute claim — unregistered paths
      are NOT 404'd server-side, the SPA uses gdsroute as a client-side
      whitelist (web-verified).
    - runtime: fixed controller package paths
      (Gdsmodule/Gdsconfigform/Gdsconfigtb live in systemweb/, not
      businessweb/); added CheckFlowController.
    - sql-templates / proc-dispatch: 8 scaffolds, not 7 (sSqlStr.sql).
    - master-slave: removed table names that don't exist (accOrderCostAnalysisMaster,
      quoQuotationCalc, mftWorkOrderCalc, mftWorkOrderSlaveMoney);
      added the *_tmp family that does.
    - permissions: full rewrite. The original page described
      gdsjurisdiction as a per-(module, role, button) rule table backed by
      empty plat_base_authority_* lookups; reality is gdsjurisdiction is a
      per-module catalog of buttons/actions, with sysjurisdiction carrying
      the actual role/user grants.
    - Removed environment-specific facts (row counts, sizes, timestamps,
      per-DB enumerations) so the wiki documents the framework, not one
      particular DB snapshot.
    
    Auto-catalog generator (en/scripts/gen_catalog.py):
    - Rewritten to query MySQL directly via ~/.my.cnf (replaces the
      recon/*.tsv pipeline).
    - Stripped ephemeral fields from output (Rows, Data size, Created,
      Updated on tables; Created, Last altered on routines; Definer on
      views).
    - Strips the schema-name prefix from VIEW_DEFINITION so view bodies
      are portable across deployments.
    - Wipes and rewrites docs/auto-catalog/{tables,views,procedures,functions}/
      on each run.
    - Auto-catalog regenerated.
    
    New API Reference chapter (5):
    - concepts/api-surface.md introduces the three-tier design (xlyEntry
      internal, xlyApi external, xlyInterface webhooks).
    - api-reference/{index,internal,external,webhooks,messaging}.md cover
      each surface, with the data-driven /api/invoke{sApiCode} pattern,
      sysapi schema, token flow, and the SpringFox Swagger UI location on
      xlyInterface.
    - mkdocs nav: chapters renumbered (Auto-Catalog → 6, Glossary → 7,
      Contributing → 8); glossary/contributing wrapped as section indexes
      to render correctly under navigation.sections.
    
    Other:
    - Bilingual top-level README (Chinese + English).
    - requirements.txt: pymysql added for the new generator.
    zichun authored
     
    Browse Code »
  • zichun authored
     
    Browse Code »
  • Documents the xly (小羚羊) printing-industry ERP framework. Built with
    MkDocs Material; CJK search via jieba; 3,076 auto-generated catalog
    pages from recon/*.tsv plus hand-written prose for the framework's
    core mental model and end-to-end vertical slices.
    
    Phase 0 recon: stack, schema shape, framework metadata layer, scope.
    Phase 1 wiki: scaffold + auto-catalog + Slices 1-6 (Slice 7 deferred).
    
    Slice coverage:
      1. CRUD module (Hello World) — observed network + cited source
      2. Multi-tenancy & product editions — sBrandsId/sSubsidiaryId/sVersionFlowId
      3. View-backed module (read-only report)
      4. Custom field overlay (gdsconfigformcustomslave)
      5. Per-customer SQL override (script/客户/<customer>/)
      6. Hardware integration (xlyPlc, optional)
      7. Workflow (deferred — Activiti tables empty in dev DB)
    
    Concepts: thesis, modules-forms-vtables, master/slave, semantic-FK,
    customization channels & layers, multi-tenancy, request lifecycle.
    
    Reference (Builder): define-form, define-vtable, permissions,
    attach-workflow (deferred).
    
    Reference (Maintainer): runtime, proc-dispatch, cache-invalidation,
    sql-templates, deployment, activiti.
    reporkey authored
     
    Browse Code »