Initial commit. Six modules extracted from
cubical-transport-hott-lean4/CubicalTransport/Algebra/ and
re-namespaced as `Infoductor`:
- Foundation/Meta.lean — MetaCType, MetaClassifier,
MetaArtifact, MetaPosition
(with manual mutual decEq for
MetaClassifier's recursive
lattice arms)
- Foundation/Edit.lean — Edit monad + Context comonad +
distributive law +
MetaClassifier.atPosition
with lattice laws as theorems
- Foundation/Restructure.lean — universal `restructure` (5-field
comp-shaped operation), 6 frozen
aliases, headless apply +
brokenRefs / selfConsistent /
Edit.guarded
- Foundation/MacroAlias.lean — @[macroAlias] attribute +
EnvExtension registry
- Foundation/MetaPath.lean — @[metaPath src target]
attribute + registry +
findPathsFromSource/ToTarget
- Foundation/Methodology.lean — @[methodology classifier]
attribute + registry +
deriveByTransport (walks
metaPath registry) +
tryEntryAsClosed primitive +
cubical_search reference
demonstrator
Builds clean on lean4 v4.30.0-rc2, ten oleans. No deps beyond
Lean stdlib. Sub-libs Tactics / Pantograph / Runner are planned
under the same lakefile.toml but not yet implemented.
Pairs with Pantograph as the conductor sits atop the pantograph
on an electric train. "Info-ductor" — conducts information
(structure / classifications / methodology) through a Lean
codebase.
Forgejo: http://maxgit.wg:3000/max/infoductor
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
242 lines
11 KiB
Text
242 lines
11 KiB
Text
/-
|
||
Infoductor.Foundation.Meta — meta-mirror types
|
||
=================================================
|
||
Phase A of `docs/ALGEBRA_PLAN.md`. Defines the meta-level
|
||
vocabulary on which the universal `restructure` macro is built —
|
||
the meta-mirror of the cubical AST (`CType`, `FaceFormula`,
|
||
`CTerm`).
|
||
|
||
These types are pure data; no semantic content is committed in
|
||
this module. Their semantics — how `restructure` consumes them
|
||
to emit `MakeEditLinkProps.ofReplaceRange` calls in the `Edit`
|
||
monad — lives in `Algebra/Restructure.lean`.
|
||
|
||
| Cubical-AST type | Meta-mirror | Role |
|
||
|------------------|-------------|--------------------------------|
|
||
| `CType` | `MetaCType` | meta-types of source artifacts |
|
||
| `FaceFormula` | `MetaClassifier` | "where in the codebase" |
|
||
| `CTerm` | `MetaArtifact` | the new content / fallback |
|
||
|
||
The point: every restructuring operation has the *same five
|
||
fields* as `comp i A φ u t`, with each field promoted from the
|
||
cubical CTerm world to the meta-Lean-source world.
|
||
|
||
See `docs/ALGEBRA_PLAN.md` §2.2 for the formal table.
|
||
-/
|
||
|
||
import Lean.Syntax
|
||
import Lean.Data.Name
|
||
|
||
namespace Infoductor
|
||
|
||
-- ── MetaCType — the meta-mirror of `CType` ──────────────────────────────────
|
||
-- Every artifact in a Lean source has a meta-type that classifies its
|
||
-- structural role. These are the "cubical-types" of the source-code
|
||
-- universe: a theorem is an artifact whose meta-type is `theorem`,
|
||
-- a `def` is an artifact whose meta-type is `definition`, etc.
|
||
|
||
/-- Meta-mirror of `CType`: classifies the structural role of an
|
||
artifact in Lean source code.
|
||
|
||
Each constructor names a kind of declaration / structural unit
|
||
that `restructure` can operate on. The list is closed: extending
|
||
it requires updating the universal macro to handle the new
|
||
artifact kind.
|
||
|
||
See `docs/ALGEBRA_PLAN.md` §2.2 for the design rationale. -/
|
||
inductive MetaCType where
|
||
/-- A `theorem` declaration: `theorem foo : T := proof`. -/
|
||
| theorem_ : MetaCType
|
||
/-- A `def` declaration: `def foo := body`. -/
|
||
| definition : MetaCType
|
||
/-- An `instance` declaration. -/
|
||
| instance_ : MetaCType
|
||
/-- A `structure` declaration. -/
|
||
| structure_ : MetaCType
|
||
/-- A Lean `inductive` declaration. -/
|
||
| inductive_ : MetaCType
|
||
/-- An entire file. -/
|
||
| file : MetaCType
|
||
/-- A `namespace` block. -/
|
||
| namespace_ : MetaCType
|
||
/-- A `class` declaration. -/
|
||
| class_ : MetaCType
|
||
/-- A logical "set of classifiers" — abstract collection used by
|
||
methodology dispatch. -/
|
||
| classifierSet : MetaCType
|
||
/-- An edge in the dependency graph between declarations. Used
|
||
by `transp`-along-MetaPath to identify what gets reorganised
|
||
together. -/
|
||
| dependencyEdge : MetaCType
|
||
/-- A custom-attribute annotation site (e.g., the position of a
|
||
`@[simp]` or `@[methodology]` annotation on a declaration). -/
|
||
| attributeSite : MetaCType
|
||
deriving Repr, Inhabited, DecidableEq
|
||
|
||
-- ── MetaClassifier — the meta-mirror of `FaceFormula` ──────────────────────
|
||
-- "Where in the codebase does this restructuring apply?" The face
|
||
-- lattice on dimensions has its analogue at the meta level: face = a
|
||
-- predicate over dimension-coordinates picking out a sub-cube; meta-
|
||
-- classifier = a predicate over codebase-coordinates picking out a
|
||
-- sub-region.
|
||
|
||
/-- Meta-mirror of `FaceFormula`: a predicate over codebase
|
||
positions. Combined via `meet` and `join` to form composite
|
||
"where in the codebase" predicates.
|
||
|
||
The lattice structure mirrors the cubical face lattice:
|
||
`meet` is intersection, `join` is union; `always` is the top
|
||
(`.top` analogue), `never` is the bottom (`.bot` analogue). -/
|
||
inductive MetaClassifier where
|
||
/-- "Everywhere" — the top of the meta-classifier lattice;
|
||
analogue of `FaceFormula.top`. -/
|
||
| always : MetaClassifier
|
||
/-- "Nowhere" — the bottom; analogue of `FaceFormula.bot`. -/
|
||
| never : MetaClassifier
|
||
/-- "At this declaration"; analogue of `FaceFormula.eq0`/`eq1`
|
||
on a specific dim coordinate. -/
|
||
| atDecl : Lean.Name → MetaClassifier
|
||
/-- "In this file." -/
|
||
| inFile : String → MetaClassifier
|
||
/-- "Under this attribute" — anywhere a particular attribute is
|
||
attached. -/
|
||
| underAttribute : Lean.Name → MetaClassifier
|
||
/-- "In any declaration that depends on this one." -/
|
||
| dependencyOf : Lean.Name → MetaClassifier
|
||
/-- "In the same namespace as this name." -/
|
||
| inNamespace : Lean.Name → MetaClassifier
|
||
/-- Conjunction; analogue of `FaceFormula.meet`. -/
|
||
| meet : MetaClassifier → MetaClassifier → MetaClassifier
|
||
/-- Disjunction; analogue of `FaceFormula.join`. -/
|
||
| join : MetaClassifier → MetaClassifier → MetaClassifier
|
||
deriving Repr, Inhabited
|
||
|
||
-- DecidableEq for MetaClassifier — manual mutual decision because
|
||
-- the type is recursive (meet/join arms) and mixes Lean.Name / String
|
||
-- carriers.
|
||
|
||
def MetaClassifier.decEq : (a b : MetaClassifier) → Decidable (a = b)
|
||
| .always, .always => isTrue rfl
|
||
| .never, .never => isTrue rfl
|
||
| .atDecl n, .atDecl m =>
|
||
if h : n = m then isTrue (by rw [h])
|
||
else isFalse (fun heq => h (by cases heq; rfl))
|
||
| .inFile s, .inFile t =>
|
||
if h : s = t then isTrue (by rw [h])
|
||
else isFalse (fun heq => h (by cases heq; rfl))
|
||
| .underAttribute n, .underAttribute m =>
|
||
if h : n = m then isTrue (by rw [h])
|
||
else isFalse (fun heq => h (by cases heq; rfl))
|
||
| .dependencyOf n, .dependencyOf m =>
|
||
if h : n = m then isTrue (by rw [h])
|
||
else isFalse (fun heq => h (by cases heq; rfl))
|
||
| .inNamespace n, .inNamespace m =>
|
||
if h : n = m then isTrue (by rw [h])
|
||
else isFalse (fun heq => h (by cases heq; rfl))
|
||
| .meet a₁ b₁, .meet a₂ b₂ =>
|
||
match MetaClassifier.decEq a₁ a₂, MetaClassifier.decEq b₁ b₂ with
|
||
| isTrue ha, isTrue hb => isTrue (by rw [ha, hb])
|
||
| isFalse h, _ => isFalse (fun heq => h (by cases heq; rfl))
|
||
| _, isFalse h => isFalse (fun heq => h (by cases heq; rfl))
|
||
| .join a₁ b₁, .join a₂ b₂ =>
|
||
match MetaClassifier.decEq a₁ a₂, MetaClassifier.decEq b₁ b₂ with
|
||
| isTrue ha, isTrue hb => isTrue (by rw [ha, hb])
|
||
| isFalse h, _ => isFalse (fun heq => h (by cases heq; rfl))
|
||
| _, isFalse h => isFalse (fun heq => h (by cases heq; rfl))
|
||
| .always, .never | .always, .atDecl _ | .always, .inFile _
|
||
| .always, .underAttribute _ | .always, .dependencyOf _
|
||
| .always, .inNamespace _ | .always, .meet _ _ | .always, .join _ _
|
||
| .never, .always | .never, .atDecl _ | .never, .inFile _
|
||
| .never, .underAttribute _ | .never, .dependencyOf _
|
||
| .never, .inNamespace _ | .never, .meet _ _ | .never, .join _ _
|
||
| .atDecl _, .always | .atDecl _, .never | .atDecl _, .inFile _
|
||
| .atDecl _, .underAttribute _ | .atDecl _, .dependencyOf _
|
||
| .atDecl _, .inNamespace _ | .atDecl _, .meet _ _ | .atDecl _, .join _ _
|
||
| .inFile _, .always | .inFile _, .never | .inFile _, .atDecl _
|
||
| .inFile _, .underAttribute _ | .inFile _, .dependencyOf _
|
||
| .inFile _, .inNamespace _ | .inFile _, .meet _ _ | .inFile _, .join _ _
|
||
| .underAttribute _, .always | .underAttribute _, .never
|
||
| .underAttribute _, .atDecl _ | .underAttribute _, .inFile _
|
||
| .underAttribute _, .dependencyOf _ | .underAttribute _, .inNamespace _
|
||
| .underAttribute _, .meet _ _ | .underAttribute _, .join _ _
|
||
| .dependencyOf _, .always | .dependencyOf _, .never
|
||
| .dependencyOf _, .atDecl _ | .dependencyOf _, .inFile _
|
||
| .dependencyOf _, .underAttribute _ | .dependencyOf _, .inNamespace _
|
||
| .dependencyOf _, .meet _ _ | .dependencyOf _, .join _ _
|
||
| .inNamespace _, .always | .inNamespace _, .never
|
||
| .inNamespace _, .atDecl _ | .inNamespace _, .inFile _
|
||
| .inNamespace _, .underAttribute _ | .inNamespace _, .dependencyOf _
|
||
| .inNamespace _, .meet _ _ | .inNamespace _, .join _ _
|
||
| .meet _ _, .always | .meet _ _, .never | .meet _ _, .atDecl _
|
||
| .meet _ _, .inFile _ | .meet _ _, .underAttribute _
|
||
| .meet _ _, .dependencyOf _ | .meet _ _, .inNamespace _
|
||
| .meet _ _, .join _ _
|
||
| .join _ _, .always | .join _ _, .never | .join _ _, .atDecl _
|
||
| .join _ _, .inFile _ | .join _ _, .underAttribute _
|
||
| .join _ _, .dependencyOf _ | .join _ _, .inNamespace _
|
||
| .join _ _, .meet _ _ =>
|
||
isFalse (fun heq => by cases heq)
|
||
|
||
instance : DecidableEq MetaClassifier := MetaClassifier.decEq
|
||
|
||
-- ── MetaArtifact — the meta-mirror of `CTerm` ───────────────────────────────
|
||
-- The "content" half of restructure. An artifact is what gets put in
|
||
-- the source: a chunk of raw text, a parsed syntax tree, a reference
|
||
-- to another decl, or "remove this" (the empty artifact).
|
||
|
||
/-- Meta-mirror of `CTerm`: the content placed at a meta-position by a
|
||
restructuring operation.
|
||
|
||
Four shapes: raw source text (string), parsed syntax (Lean.Syntax),
|
||
a reference to an existing decl by `Name`, and the empty artifact
|
||
(used for deletion).
|
||
|
||
`Lean.Syntax` is opaque-by-default in Lean 4 — its internal
|
||
structure is large. We don't derive `DecidableEq` on `MetaArtifact`
|
||
because comparing arbitrary Syntax trees structurally is heavy and
|
||
not needed by `restructure`'s dispatch logic. Use `MetaArtifact.toString`
|
||
for printable comparisons. -/
|
||
inductive MetaArtifact where
|
||
/-- Raw Lean source text (will be parsed at apply time). -/
|
||
| source : String → MetaArtifact
|
||
/-- A parsed Lean syntax tree. -/
|
||
| declAt : Lean.Syntax → MetaArtifact
|
||
/-- A reference to an existing declaration by name; resolved at
|
||
apply time. -/
|
||
| refTo : Lean.Name → MetaArtifact
|
||
/-- The empty artifact — "remove this position." -/
|
||
| empty : MetaArtifact
|
||
deriving Inhabited
|
||
|
||
/-- A printable summary of an artifact, useful for diagnostics and
|
||
widget rendering. -/
|
||
def MetaArtifact.toString : MetaArtifact → String
|
||
| .source s => s!"source({s})"
|
||
| .declAt _ => "declAt(<syntax>)"
|
||
| .refTo n => s!"refTo({n})"
|
||
| .empty => "empty"
|
||
|
||
instance : ToString MetaArtifact := ⟨MetaArtifact.toString⟩
|
||
|
||
-- ── MetaPosition — where an artifact lives in source ───────────────────────
|
||
-- The first field of `restructure i (Context = MetaCType) φ witness fallback`
|
||
-- — analogue of the dim-binder `i` in `comp i A φ u t`.
|
||
|
||
/-- The position of an artifact within a Lean source file or
|
||
project. Used as the `MetaPosition` argument of `restructure`,
|
||
analogous to the dim-binder `i` of `comp`. -/
|
||
structure MetaPosition where
|
||
/-- The fully qualified declaration name (or namespace) the
|
||
position lives under. `Name.anonymous` for file-level. -/
|
||
declName : Lean.Name
|
||
/-- Path to the source file (or empty for in-memory). -/
|
||
filePath : String
|
||
/-- Optional byte offset / range info; `none` if positional via
|
||
declName alone is sufficient. -/
|
||
range : Option (Nat × Nat)
|
||
deriving Inhabited
|
||
|
||
instance : Repr MetaPosition where
|
||
reprPrec p _ := s!"MetaPosition(declName={p.declName}, filePath={p.filePath}, range={repr p.range})"
|
||
|
||
end Infoductor
|