ZDDC/transmittal/js
ZDDC 3115e388fc feat(server): authenticated CRUD + verb-based RBAC with WORM archive folders
Replaces the binary acl.allow/deny model with five permission verbs
(r/w/c/d/a) and first-class roles, and adds an authenticated file API
(PUT/DELETE/POST move/mkdir) so the HTML tools can edit-in-place over
HTTP. Closes the AC-3(7) and AC-6 federal-readiness gaps.

File API (zddc/internal/handler/fileapi.go)
  - PUT <new>      → action c
  - PUT <existing> → action w
  - PUT <.zddc>    → action a (CanEditZddc strict-ancestor rule)
  - DELETE         → action d
  - POST mkdir     → action c (auto-writes creator-owned .zddc when the
                     parent is Incoming/Working/Staging)
  - POST move      → action w on src + c on dst, atomic via os.Rename
  - Optional If-Match for optimistic concurrency, --max-write-bytes cap,
    audit log emits a structured file_write event per operation.

Permission model (zddc/internal/zddc/{acl,file,roles,cascade_mode}.go)
  - acl.permissions: { principal → verb-set } map; principals are email
    patterns or role names. Empty verb set is an explicit deny.
  - roles: { name → members } definitions, available at the level they
    declare and all descendants. Closer-to-leaf shadows ancestor.
  - Legacy acl.allow/deny still work; they fold into permissions at
    parse time (allow → "rwcd", deny → "").
  - Cascade walks leaf→root; first level with any matching entry wins;
    the union of matching verb sets at that level decides.
  - --cascade-mode=strict adds a root→leaf ancestor-deny pre-pass so an
    ancestor explicit-deny is absolute (NIST AC-6). Default delegated
    preserves the existing commercial behavior.

Special folders (zddc/internal/zddc/special.go)
  - Incoming / Working / Staging: mkdir auto-writes a .zddc into the new
    subdir granting created_by + that email rwcda directly. Same form
    operators write by hand; creator can edit it later to add others.
  - Issued / Received: server-enforced WORM split. Cascade grants
    inherited from above the WORM folder are masked to r only; grants
    placed at-or-below the WORM folder retain r,c. Operators grant
    write-once (cr) to the doc controller via an explicit .zddc at the
    Issued/Received folder. Admins exempt — only escape hatch.

Browser polyfill (shared/zddc-source.js)
  - HttpDirectoryHandle + HttpFileHandle implement the FS Access API
    surface (values, getFileHandle, createWritable, removeEntry,
    queryPermission/requestPermission) over zddc-server's listing JSON
    and file API. Existing tools written against showDirectoryPicker
    work unchanged.
  - detectServerRoot() returns { handle, status }: tools auto-load on
    HTTP, surface a clear "no permission to list" message on 403, and
    fall back to the welcome screen on 0.
  - classifier renames take the atomic POST move path on HTTP-backed
    handles; mdedit and transmittal route reads/writes through the
    polyfill so prior FS-API code paths cover both modes.

Tests
  - zddc/internal/zddc/{cascade_mode,roles,special,acl}_test.go cover
    delegated vs strict, role membership / shadowing / legacy fallback,
    WORM split semantics, verb-set parser round-trip.
  - zddc/internal/handler/fileapi_test.go now also covers role-based
    vendor scenarios, WORM blocking vendor & doc controller writes,
    explicit Issued .zddc unlocking the cr drop-box, admin bypass,
    auto-ownership on mkdir, and strict-mode lockouts.

Docs
  - ARCHITECTURE.md + zddc/README.md document the verb model, role
    syntax, special-folder behaviors, cascade-mode flag, and full file
    API surface. Federal-readiness gap analysis strikes AC-3(7) and
    AC-6.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-05 15:58:04 -05:00
..
app.js Initial commit 2026-04-27 11:05:47 -05:00
data.js Initial commit 2026-04-27 11:05:47 -05:00
dom.js Initial commit 2026-04-27 11:05:47 -05:00
drop-zones.js Initial commit 2026-04-27 11:05:47 -05:00
email-tags.js Initial commit 2026-04-27 11:05:47 -05:00
files-archive.js Initial commit 2026-04-27 11:05:47 -05:00
files-preview.js perf(tools): vendor jszip + docx-preview for archive/transmittal/classifier 2026-05-04 07:49:17 -05:00
files-render.js Initial commit 2026-04-27 11:05:47 -05:00
files.js feat(server): authenticated CRUD + verb-based RBAC with WORM archive folders 2026-05-05 15:58:04 -05:00
filters.js Initial commit 2026-04-27 11:05:47 -05:00
focus.js Initial commit 2026-04-27 11:05:47 -05:00
hydrate.js Initial commit 2026-04-27 11:05:47 -05:00
json.js Initial commit 2026-04-27 11:05:47 -05:00
live-digest.js Initial commit 2026-04-27 11:05:47 -05:00
logos.js Initial commit 2026-04-27 11:05:47 -05:00
main.js Initial commit 2026-04-27 11:05:47 -05:00
markdown-editor.js Initial commit 2026-04-27 11:05:47 -05:00
markdown.js Initial commit 2026-04-27 11:05:47 -05:00
mode.js Initial commit 2026-04-27 11:05:47 -05:00
publish-modal.js Initial commit 2026-04-27 11:05:47 -05:00
publish.js Initial commit 2026-04-27 11:05:47 -05:00
reactive.js Initial commit 2026-04-27 11:05:47 -05:00
reset.js Initial commit 2026-04-27 11:05:47 -05:00
security.js Initial commit 2026-04-27 11:05:47 -05:00
state.js Initial commit 2026-04-27 11:05:47 -05:00
util.js Initial commit 2026-04-27 11:05:47 -05:00
validation.js Initial commit 2026-04-27 11:05:47 -05:00
verification.js Initial commit 2026-04-27 11:05:47 -05:00
visibility.js Initial commit 2026-04-27 11:05:47 -05:00