apimock_config/view.rs
1//! Read-only views on workspace state, and the command + result types
2//! the editing API uses.
3//!
4//! # 5.1.0 — spec alignment
5//!
6//! In 5.0.0 this module carried a placeholder shape defined only by
7//! rustdoc; 5.1.0 re-aligns it with the 5.1 spec:
8//!
9//! - `WorkspaceSnapshot { files, routes, diagnostics }` (spec §4.2)
10//! - each node carries `id: NodeId` + `source_file` + `toml_path` +
11//! `display_name` + `kind` + `validation`
12//! - `EditCommand` is eight variants covering every editable action
13//! (spec §4.3)
14//! - `ApplyResult { changed_nodes, diagnostics, requires_reload }`
15//! (spec §4.4)
16//! - `SaveResult { changed_files, diff_summary, requires_reload }`
17//! (spec §4.5) — populated in Step 4
18//! - `ValidationReport { diagnostics, is_valid }` (spec §4.6)
19//! - `Diagnostic { node_id, file, severity, message }` (spec §4.7)
20//!
21//! # Why UUIDs and not positional IDs
22//!
23//! The spec's §4.3 says "すべて NodeId で対象を指定". Positional IDs
24//! (`rule_sets[0].rules[3]`) would shift on every insert / delete /
25//! move, forcing the GUI to re-index its selection set after every
26//! edit. UUIDs are stable within a `Workspace` instance regardless of
27//! reordering.
28
29use serde::{Deserialize, Serialize};
30use serde_json::Value as JsonValue;
31use uuid::Uuid;
32
33use std::path::PathBuf;
34
35use apimock_routing::view::RouteCatalogSnapshot;
36
37/// Stable identifier for an editable node.
38///
39/// # Stability contract
40///
41/// Stable within one `Workspace` instance — that is, across any
42/// sequence of `apply()` calls. IDs are reassigned on fresh `load()`,
43/// which matches spec §10 "Workspace はメモリ上に独立インスタンスを持つ".
44#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)]
45#[serde(transparent)]
46pub struct NodeId(pub Uuid);
47
48impl NodeId {
49 pub fn new() -> Self {
50 Self(Uuid::new_v4())
51 }
52}
53
54impl Default for NodeId {
55 fn default() -> Self {
56 Self::new()
57 }
58}
59
60impl std::fmt::Display for NodeId {
61 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
62 self.0.fmt(f)
63 }
64}
65
66/// Complete snapshot of the workspace state.
67///
68/// Shape matches spec §4.2 exactly. Consumed read-only by the GUI;
69/// mutated indirectly via `Workspace::apply`.
70#[derive(Clone, Debug, Serialize)]
71#[non_exhaustive]
72pub struct WorkspaceSnapshot {
73 /// All editable TOML files in the workspace, flattened. Each file
74 /// carries its own list of editable nodes.
75 pub files: Vec<ConfigFileView>,
76 /// Route overview pulled from the routing crate.
77 pub routes: RouteCatalogSnapshot,
78 /// Workspace-scoped issues (e.g. a root file that failed to load).
79 /// Per-node diagnostics live inside each `ConfigNodeView.validation`.
80 pub diagnostics: Vec<Diagnostic>,
81}
82
83impl WorkspaceSnapshot {
84 pub fn empty() -> Self {
85 Self {
86 files: Vec::new(),
87 routes: RouteCatalogSnapshot::empty(),
88 diagnostics: Vec::new(),
89 }
90 }
91}
92
93/// One TOML file inside the workspace.
94#[derive(Clone, Debug, Serialize)]
95#[non_exhaustive]
96pub struct ConfigFileView {
97 /// Absolute path on disk.
98 pub path: PathBuf,
99 /// Display name — typically the file name. Used as a tab title in
100 /// the GUI.
101 pub display_name: String,
102 /// What kind of file this is (root config, rule set, middleware).
103 pub kind: ConfigFileKind,
104 /// Editable nodes extracted from the file.
105 pub nodes: Vec<ConfigNodeView>,
106}
107
108#[derive(Clone, Copy, Debug, Serialize)]
109pub enum ConfigFileKind {
110 Root,
111 RuleSet,
112 Middleware,
113}
114
115/// One editable value inside a `ConfigFileView`.
116///
117/// Each node carries the six fields spec §4.2 makes mandatory.
118#[derive(Clone, Debug, Serialize)]
119#[non_exhaustive]
120pub struct ConfigNodeView {
121 /// Stable identifier — survives moves / renames within a Workspace
122 /// instance.
123 pub id: NodeId,
124 /// File the node was loaded from.
125 pub source_file: PathBuf,
126 /// Dotted TOML path inside `source_file` (e.g. `"listener.port"`,
127 /// `"rules[2].respond"`).
128 pub toml_path: String,
129 /// Human-readable label for UI list rendering (e.g. the rule's
130 /// `url_path` value, or `"Rule #3"` for a rule without one).
131 pub display_name: String,
132 /// Shape of the underlying value.
133 pub kind: NodeKind,
134 /// Per-node validation results.
135 pub validation: NodeValidation,
136}
137
138/// What shape of value a node holds. The variants are what the
139/// spec-defined `EditCommand` variants act on.
140#[derive(Clone, Copy, Debug, Serialize)]
141pub enum NodeKind {
142 /// Root config node — listener / log / service fields.
143 RootSetting,
144 /// One rule set loaded from a referenced TOML file.
145 RuleSet,
146 /// One rule inside a rule set.
147 Rule,
148 /// The `respond` block of a rule.
149 Respond,
150 /// File-based response node (fallback dir entry).
151 FileNode,
152 /// Script / middleware route.
153 Script,
154}
155
156/// Per-node validation result.
157///
158/// # Why validation is a field on the node and not a separate pass
159///
160/// GUIs render validation inline ("this field has a red underline").
161/// Keeping the validation result stapled to the node the GUI is about
162/// to render avoids a second lookup step in every render frame.
163#[derive(Clone, Debug, Default, Serialize)]
164pub struct NodeValidation {
165 /// Convenience flag — true iff `issues` is empty.
166 pub ok: bool,
167 /// Human-readable issues scoped to this node.
168 pub issues: Vec<ValidationIssue>,
169}
170
171impl NodeValidation {
172 pub fn ok() -> Self {
173 Self {
174 ok: true,
175 issues: Vec::new(),
176 }
177 }
178}
179
180#[derive(Clone, Debug, Serialize)]
181pub struct ValidationIssue {
182 pub severity: Severity,
183 pub message: String,
184}
185
186/// Structured edit command applied via `Workspace::apply`.
187///
188/// # Shape comes straight from spec §4.3
189///
190/// Each variant targets a node by NodeId (never by positional index).
191/// This guarantees edits remain well-defined across previous inserts /
192/// removes in the same GUI session.
193#[derive(Clone, Debug)]
194#[non_exhaustive]
195pub enum EditCommand {
196 /// Add a rule set file to the workspace.
197 ///
198 /// `path` is relative to the root config's directory — the same
199 /// convention as the value stored in `service.rule_sets`.
200 AddRuleSet {
201 path: String,
202 },
203 /// Remove a rule set by its NodeId. The underlying TOML file is
204 /// NOT deleted from disk — the workspace only removes the reference.
205 RemoveRuleSet {
206 id: NodeId,
207 },
208 /// Add a rule to an existing rule set.
209 AddRule {
210 parent: NodeId,
211 rule: RulePayload,
212 },
213 /// Update a rule's `when` / `respond` block.
214 ///
215 /// # Preservation of unspecified fields
216 ///
217 /// `RulePayload` carries `url_path`, `method`, and `respond` —
218 /// the fields a stage-1 GUI form exposes. A rule may also carry
219 /// `headers` and `body.json` match conditions that aren't part of
220 /// the payload shape. Those clauses are **preserved** across an
221 /// `UpdateRule`: the new rule keeps whatever headers / body
222 /// conditions the previous rule had, even though the payload
223 /// doesn't mention them.
224 ///
225 /// Without this preservation, every `UpdateRule` would silently
226 /// strip the unsurfaced clauses, which is a save-time bug when a
227 /// GUI re-saves a rule it loaded from a hand-edited TOML file.
228 UpdateRule {
229 id: NodeId,
230 rule: RulePayload,
231 },
232 /// Remove a rule by NodeId.
233 DeleteRule {
234 id: NodeId,
235 },
236 /// Reorder a rule within its parent rule set.
237 MoveRule {
238 id: NodeId,
239 new_index: usize,
240 },
241 /// Update the `respond` block of a rule.
242 UpdateRespond {
243 id: NodeId,
244 respond: RespondPayload,
245 },
246 /// Update a root-level setting (listener, log, service-level flags).
247 UpdateRootSetting {
248 key: RootSettingKey,
249 value: EditValue,
250 },
251}
252
253/// Payload for `AddRule` / `UpdateRule`.
254///
255/// # Preservation of unspecified fields (5.5.0 guarantee)
256///
257/// Fields set to `None` are preserved from the existing rule when this
258/// is an `UpdateRule` call. The `headers` and `body` fields use
259/// `Option<Vec<_>>` to distinguish three states:
260/// - `None` — preserve existing conditions.
261/// - `Some(vec![])` — clear all conditions.
262/// - `Some(vec![…])` — replace with the given set.
263///
264/// # URL path operator (RFC 001)
265///
266/// `url_path_op` controls which operator the routing crate uses to
267/// match the given `url_path` value. When `url_path_op` is `None` and
268/// `url_path` is `Some(_)`, the operator defaults to `Equal` (5.7.0
269/// behaviour). When `url_path` is `None`, both fields are ignored.
270///
271/// # Header and body conditions (RFC 002)
272///
273/// `headers` and `body` are optional lists of conditions. Each `None`
274/// preserves the existing rule's conditions; each `Some(_)` replaces
275/// them wholesale (an empty `Vec` clears them).
276#[derive(Clone, Debug, Default)]
277pub struct RulePayload {
278 pub url_path: Option<String>,
279 /// URL path match operator (RFC 001). `None` defaults to `Equal`.
280 pub url_path_op: Option<UrlPathOp>,
281 pub method: Option<String>,
282 /// Header conditions (RFC 002). `None` = preserve; `Some([])` = clear.
283 pub headers: Option<Vec<HeaderConditionPayload>>,
284 /// Body conditions (RFC 002). `None` = preserve; `Some([])` = clear.
285 pub body: Option<Vec<BodyConditionPayload>>,
286 pub respond: RespondPayload,
287}
288
289// ── RFC 001 — URL path operator ───────────────────────────────────────
290
291/// Operator for the URL path match in [`RulePayload`].
292///
293/// Mirrors the routing crate's internal operator set but lives in
294/// `apimock-config` to decouple the GUI-facing payload type from
295/// routing-internal types.
296#[derive(Clone, Copy, Debug, PartialEq, Eq)]
297pub enum UrlPathOp {
298 Equal,
299 StartsWith,
300 Contains,
301 EndsWith,
302 /// Glob wildcard match.
303 WildCard,
304 /// Regular expression match.
305 NotEqual,
306}
307
308// ── RFC 002 — Header and body condition payloads ──────────────────────
309
310/// One header condition in a [`RulePayload`].
311#[derive(Clone, Debug)]
312pub struct HeaderConditionPayload {
313 /// Header name (case-insensitive at match time).
314 pub name: String,
315 pub op: HeaderOp,
316 /// Required for all operators except `Exists` / `Absent`.
317 pub value: Option<String>,
318}
319
320/// Operator for a header condition.
321#[derive(Clone, Copy, Debug, PartialEq, Eq)]
322pub enum HeaderOp {
323 Equal,
324 Contains,
325 StartsWith,
326 EndsWith,
327 Regex,
328 /// Header must be present (any value).
329 Exists,
330 /// Header must be absent.
331 Absent,
332 NotEqual,
333 WildCard,
334}
335
336/// One body condition in a [`RulePayload`].
337#[derive(Clone, Debug)]
338pub struct BodyConditionPayload {
339 /// Currently only `Json`.
340 pub kind: BodyConditionKind,
341 /// Dotted path into the JSON body (not canonical JSONPath).
342 pub path: String,
343 pub op: BodyOp,
344 /// Configured comparison value.
345 pub value: serde_json::Value,
346}
347
348/// Body condition kind — currently only JSON.
349#[derive(Clone, Copy, Debug, PartialEq, Eq)]
350pub enum BodyConditionKind {
351 Json,
352}
353
354/// Operator for a body condition (RFC 002 / RFC 008 combined set).
355#[derive(Clone, Copy, Debug, PartialEq, Eq)]
356pub enum BodyOp {
357 // string-style
358 Equal,
359 EqualString,
360 Contains,
361 StartsWith,
362 EndsWith,
363 Regex,
364 // type-aware
365 EqualTyped,
366 // numeric
367 EqualNumber,
368 GreaterThan,
369 LessThan,
370 GreaterOrEqual,
371 LessOrEqual,
372 // presence
373 Exists,
374 Absent,
375 // array
376 ArrayLengthEqual,
377 ArrayLengthAtLeast,
378 ArrayContains,
379 // exact integer (RFC 010)
380 EqualInteger,
381}
382
383/// Payload for `UpdateRespond`.
384///
385/// The three fields are mutually specialised: exactly one of
386/// `file_path` / `text` / `status` should be populated. Validation
387/// catches cases that violate this.
388#[derive(Clone, Debug, Default)]
389pub struct RespondPayload {
390 pub file_path: Option<String>,
391 pub text: Option<String>,
392 pub status: Option<u16>,
393 pub delay_milliseconds: Option<u32>,
394}
395
396/// Enumerated root-level setting. Typed enum rather than free-form
397/// path so the apply-layer can exhaustively match without parsing.
398///
399/// # RFC 003 — TLS and Log variants
400///
401/// Seven new variants cover TLS configuration and log settings.
402/// Changes to TLS and listener fields require a full process restart
403/// (`HardRestart`); log-level and strategy changes only need a soft
404/// config reload (`SoftReload`).
405#[derive(Clone, Copy, Debug)]
406#[non_exhaustive]
407pub enum RootSettingKey {
408 // ── listener ──────────────────────────────────────────────────────
409 ListenerIpAddress,
410 ListenerPort,
411 // ── service ───────────────────────────────────────────────────────
412 ServiceFallbackRespondDir,
413 ServiceStrategy,
414 // ── TLS (RFC 003) ─────────────────────────────────────────────────
415 TlsEnabled,
416 TlsCertFile,
417 TlsKeyFile,
418 // ── log (RFC 003) ─────────────────────────────────────────────────
419 LogLevel,
420 LogFile,
421 LogFormat,
422 // ── file tree view (RFC 012) ──────────────────────────────────────
423 FileTreeShowHidden,
424 FileTreeBuiltinExcludes,
425 /// Value: `EditValue::StringList`
426 FileTreeExtraExcludes,
427 /// Value: `EditValue::StringList`
428 FileTreeInclude,
429}
430
431/// Value provided with an edit command.
432#[derive(Clone, Debug)]
433#[non_exhaustive]
434pub enum EditValue {
435 String(String),
436 Integer(i64),
437 Boolean(bool),
438 StringList(Vec<String>),
439 /// For settings whose domain is a small enum value (e.g.
440 /// `ServiceStrategy` → `"first_match"`).
441 Enum(String),
442 /// For completeness — callers can pass a raw JSON value when the
443 /// spec-defined key set is extended by stage-3 tooling. Currently
444 /// reserved; no stage-1 setting uses it.
445 Json(JsonValue),
446}
447
448/// Outcome of a successful `apply`.
449#[derive(Clone, Debug, Serialize)]
450#[non_exhaustive]
451pub struct ApplyResult {
452 /// Node IDs whose content (or position) changed.
453 pub changed_nodes: Vec<NodeId>,
454 /// Issues surfaced by applying the command (validation during apply
455 /// may add diagnostics — e.g. a new rule pointing at a missing file).
456 pub diagnostics: Vec<Diagnostic>,
457 /// `true` iff the server should reload to see this change. An edit
458 /// that changes the listener port needs a restart, not just a
459 /// reload — see `Workspace::save` for the richer `ReloadHint`.
460 pub requires_reload: bool,
461}
462
463/// Outcome of `Workspace::save`.
464#[derive(Clone, Debug, Serialize)]
465#[non_exhaustive]
466pub struct SaveResult {
467 /// TOML files actually written to disk.
468 pub changed_files: Vec<PathBuf>,
469 /// One entry per node that changed since last load.
470 pub diff_summary: Vec<DiffItem>,
471 pub requires_reload: bool,
472}
473
474/// One summary row in a `SaveResult::diff_summary`.
475#[derive(Clone, Debug, Serialize)]
476pub struct DiffItem {
477 pub kind: DiffKind,
478 pub target: NodeId,
479 pub summary: String,
480}
481
482#[derive(Clone, Copy, Debug, Serialize)]
483pub enum DiffKind {
484 Added,
485 Updated,
486 Removed,
487}
488
489/// Workspace-wide validation result. Mirrors spec §4.6.
490#[derive(Clone, Debug, Serialize)]
491pub struct ValidationReport {
492 pub diagnostics: Vec<Diagnostic>,
493 pub is_valid: bool,
494}
495
496impl ValidationReport {
497 pub fn ok() -> Self {
498 Self {
499 diagnostics: Vec::new(),
500 is_valid: true,
501 }
502 }
503}
504
505/// Human-readable notice about the workspace.
506#[derive(Clone, Debug, Serialize)]
507pub struct Diagnostic {
508 /// Target node, if any. `None` means "workspace-wide".
509 pub node_id: Option<NodeId>,
510 /// Target file, if the diagnostic is best reported at file level
511 /// (e.g. "could not read apimock-rule-set.toml"). May be `None` for
512 /// purely in-memory errors.
513 pub file: Option<PathBuf>,
514 pub severity: Severity,
515 pub message: String,
516}
517
518#[derive(Clone, Copy, Debug, Serialize)]
519pub enum Severity {
520 Error,
521 Warning,
522 Info,
523}
524
525// ---------------------------------------------------------------------------
526// Reload hint — spec §9. The same enum shape was defined in 5.0.0;
527// 5.1 reuses it unchanged so existing consumers keep working.
528// ---------------------------------------------------------------------------
529
530/// Advisory indicating what, if anything, the server needs to do in
531/// response to a config change.
532///
533/// # RFC 003 — Reload semantics
534///
535/// | Key group | Hint |
536/// |-------------------------------|----------------|
537/// | `ListenerIpAddress/Port` | `HardRestart` |
538/// | `TlsEnabled`, `TlsCert*` | `HardRestart` |
539/// | `LogFile` | `HardRestart` |
540/// | `ServiceFallbackRespondDir` | `SoftReload` |
541/// | `ServiceStrategy` | `SoftReload` |
542/// | `LogLevel`, `LogFormat` | `SoftReload` |
543///
544/// The hint is advisory — the server does not auto-restart. The GUI
545/// surfaces it to the user.
546#[derive(Clone, Copy, Debug, Default, Serialize)]
547pub struct ReloadHint {
548 /// Server can re-read config without rebinding the listener.
549 pub requires_reload: bool,
550 /// Process must restart (rebind socket, reload TLS, reopen log file).
551 pub requires_restart: bool,
552}
553
554impl ReloadHint {
555 pub fn none() -> Self {
556 Self::default()
557 }
558
559 /// Config can be hot-reloaded without a process restart.
560 pub fn reload() -> Self {
561 Self {
562 requires_reload: true,
563 requires_restart: false,
564 }
565 }
566
567 /// Process must fully restart for this change to take effect.
568 pub fn restart() -> Self {
569 Self {
570 requires_reload: false,
571 requires_restart: true,
572 }
573 }
574
575 /// Return the hint appropriate for the given [`RootSettingKey`].
576 pub fn for_key(key: RootSettingKey) -> Self {
577 use RootSettingKey::*;
578 match key {
579 ListenerIpAddress | ListenerPort | TlsEnabled | TlsCertFile | TlsKeyFile
580 | LogFile => Self::restart(),
581 ServiceFallbackRespondDir | ServiceStrategy | LogLevel | LogFormat
582 | FileTreeShowHidden | FileTreeBuiltinExcludes | FileTreeExtraExcludes
583 | FileTreeInclude => Self::reload(),
584 }
585 }
586}