All checks were successful
Build + deploy releases / build-and-deploy (push) Successful in 8s
Schema-driven form renderer plus zddc-server endpoints that turn any
<name>.form.yaml into a working data-collection form at <path>/<name>.form.html.
Submissions land in <path>/<name>/<YYYY-MM-DD>-<email-sanitized>.yaml,
ACL-gated by the existing .zddc cascade. The form posts back to its own URL;
the server strips ".html" and routes by what's underneath, so create and
update use the same client-side code path.
Form spec dialect: JSON Schema 2020-12 + RJSF-style ui:* hints, written in
YAML. Chosen for LLM authorability — it's the canonical structured-output
target for OpenAI/Anthropic, and the ui:* convention is the most-trained UI
hint vocabulary. Supported subset for v0: type (string/number/integer/boolean/
array/object), enum, min/max, minLength/maxLength, required, additionalProperties:
false, properties, items, format (date, email). Round-trip mode is form-as-truth:
submission YAML is regenerated each save, comments are not preserved (the v1
file-as-truth mode for hand-edited files like .zddc itself is deferred).
New components:
* form/ — sixth single-file HTML tool, vanilla JS renderer (~760 LoC)
* zddc/internal/jsonschema/ — focused JSON Schema validator covering only
the v0 keyword subset. Match-implementation-cost-to-surface-used: a full
library brings 70%+ surface we don't use; revisit when v1 adds $ref +
oneOf + if/then/else.
* zddc/internal/handler/formhandler.go — RecognizeFormRequest / ServeForm,
capability-URL re-edit, atomic submission writes via the new
zddc.WriteAtomic helper extracted from writer.go.
* dispatch() in zddc-server/main.go now intercepts *.form.html and
*.yaml.html before the static-file path; spec existence is the trigger.
Build pipeline: form joins ZDDC_RELEASE_TOOLS in lockstep, gets its own
embedded copy in handler/form.html (separate from the apps cascade —
the form renderer is fixed, not subject to per-folder version overrides).
Tests: 5 new Playwright specs (form-safety) + 14 new Go tests across the
validator and handler. All 172 Playwright tests + 10 Go packages green.
End-to-end manual verification: GET empty → POST 201 + capability URL →
GET re-edit (pre-filled) → POST update → 200, raw YAML browsable, ACL
deny → 403.
Docs: form/ section added to AGENTS.md and ARCHITECTURE.md. AGENTS.md
also documents the implementation-vs-dependency policy. CLAUDE.md repo-shape
list extended.
Deferred (v1+): .zddc editor migration onto this system, file-as-truth
lossless YAML round-trip, ui:show-when conditional visibility, oneOf/anyOf,
apps-cascade preview hook, cascade-fetched form definitions.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
109 lines
4 KiB
JavaScript
109 lines
4 KiB
JavaScript
(function (app) {
|
|
'use strict';
|
|
|
|
const u = app.modules.util;
|
|
|
|
function makeObject(schema, ui, path, value, options) {
|
|
const fs = u.h('fieldset', { className: 'form-fieldset' });
|
|
const label = (ui && ui['ui:title']) || schema.title || options.fieldName;
|
|
if (label) {
|
|
fs.appendChild(u.h('legend', { className: 'form-fieldset__legend' }, label));
|
|
}
|
|
if (schema.description) {
|
|
fs.appendChild(u.h('div', { className: 'form-field__description' }, schema.description));
|
|
}
|
|
const errEl = u.h('div', { className: 'form-field__error', hidden: true });
|
|
fs.appendChild(errEl);
|
|
|
|
const props = schema.properties || {};
|
|
const requiredSet = {};
|
|
(schema.required || []).forEach(function (n) { requiredSet[n] = true; });
|
|
|
|
// Resolve render order: ui:order first (with '*' as "everything else"),
|
|
// then fall back to declaration order.
|
|
const declared = Object.keys(props);
|
|
const uiOrder = (ui && ui['ui:order']) || null;
|
|
const ordered = [];
|
|
const seen = {};
|
|
if (uiOrder && Array.isArray(uiOrder)) {
|
|
for (let i = 0; i < uiOrder.length; i++) {
|
|
const name = uiOrder[i];
|
|
if (name === '*') {
|
|
for (let j = 0; j < declared.length; j++) {
|
|
const dn = declared[j];
|
|
if (!seen[dn] && uiOrder.indexOf(dn) < 0) {
|
|
ordered.push(dn);
|
|
seen[dn] = true;
|
|
}
|
|
}
|
|
} else if (props[name] && !seen[name]) {
|
|
ordered.push(name);
|
|
seen[name] = true;
|
|
}
|
|
}
|
|
// Append anything declared but not mentioned in ui:order (and no '*' was used).
|
|
for (let j = 0; j < declared.length; j++) {
|
|
if (!seen[declared[j]]) {
|
|
ordered.push(declared[j]);
|
|
seen[declared[j]] = true;
|
|
}
|
|
}
|
|
} else {
|
|
for (let j = 0; j < declared.length; j++) {
|
|
ordered.push(declared[j]);
|
|
}
|
|
}
|
|
|
|
const children = {};
|
|
const dataObj = (value && typeof value === 'object' && !Array.isArray(value)) ? value : {};
|
|
|
|
for (let i = 0; i < ordered.length; i++) {
|
|
const name = ordered[i];
|
|
const childSchema = props[name];
|
|
const childUi = (ui && ui[name]) || {};
|
|
const childPath = u.ptrPush(path, name);
|
|
const childValue = dataObj[name];
|
|
const childWidget = app.modules.render.create(childSchema, childUi, childPath, childValue, {
|
|
fieldName: u.humanize(name),
|
|
required: !!requiredSet[name]
|
|
});
|
|
children[name] = childWidget;
|
|
fs.appendChild(childWidget.el);
|
|
}
|
|
|
|
return {
|
|
el: fs,
|
|
path: path,
|
|
type: 'object',
|
|
read: function () {
|
|
const out = {};
|
|
const keys = Object.keys(children);
|
|
for (let i = 0; i < keys.length; i++) {
|
|
const k = keys[i];
|
|
const v = children[k].read();
|
|
if (v !== undefined) {
|
|
out[k] = v;
|
|
}
|
|
}
|
|
return out;
|
|
},
|
|
setError: function (msg) {
|
|
errEl.textContent = msg;
|
|
errEl.hidden = false;
|
|
},
|
|
clearErrors: function () {
|
|
errEl.textContent = '';
|
|
errEl.hidden = true;
|
|
const keys = Object.keys(children);
|
|
for (let i = 0; i < keys.length; i++) {
|
|
children[keys[i]].clearErrors();
|
|
}
|
|
},
|
|
child: function (name) {
|
|
return children[name] || null;
|
|
}
|
|
};
|
|
}
|
|
|
|
app.modules.object = { makeObject: makeObject };
|
|
})(window.formApp);
|