diff options
| author | Andreas Grois <andi@grois.info> | 2024-09-19 00:14:18 +0200 |
|---|---|---|
| committer | Andreas Grois <andi@grois.info> | 2024-09-19 00:14:18 +0200 |
| commit | 2ceb539bf306992e726280c24ce8d7b69e4c8b5e (patch) | |
| tree | 3433e5bb28f65f475dcf2ee17544dc1ff2e0d612 /Common/Parsing.lean | |
| parent | 2c1f2362c69ab20b10b61bdbebf8e5121eca38b1 (diff) | |
Day 11 Parsing
Diffstat (limited to 'Common/Parsing.lean')
| -rw-r--r-- | Common/Parsing.lean | 110 |
1 files changed, 86 insertions, 24 deletions
diff --git a/Common/Parsing.lean b/Common/Parsing.lean index 4967385..d0e9427 100644 --- a/Common/Parsing.lean +++ b/Common/Parsing.lean @@ -1,31 +1,28 @@ /- This file contains various parsing helpers. Started _after_ Day10. -/ import Common.List +import Common.Nat namespace Parsing -structure RectangularGrid (Element : Type) where +structure MaybeEmptyRectangularGrid (Element : Type) where width : Nat height : Nat elements : Array Element - valid : elements.size = width * height + size_valid : elements.size = width * height + +instance : Inhabited (MaybeEmptyRectangularGrid Element) where + default := { width := 0, height := 0, elements := Array.empty, size_valid := rfl} + +structure RectangularGrid (Element : Type) extends MaybeEmptyRectangularGrid Element where + not_empty : width > 0 ∧ height > 0 structure RectangularGrid.Coordinate (grid : RectangularGrid Element) where x : Fin grid.width y : Fin grid.height private def RectangularGrid.Coordinate.toIndex {grid : RectangularGrid Element} (coordinate : grid.Coordinate) : Fin (grid.width * grid.height) := - ⟨coordinate.x + grid.width * coordinate.y, - Nat.le_pred_of_lt coordinate.y.isLt - |> Nat.mul_le_mul_left grid.width - |> Nat.add_le_add_iff_right.mpr - |> (Nat.mul_pred grid.width grid.height).subst (motive :=λx↦grid.width * coordinate.y.val + grid.width ≤ x + grid.width) - |> (Nat.sub_add_cancel (Nat.le_mul_of_pos_right grid.width (Nat.zero_lt_of_lt coordinate.y.isLt))).subst - |> (Nat.add_comm _ _).subst (motive := λx↦x ≤ grid.width*grid.height) - |> Nat.le_sub_of_add_le - |> Nat.lt_of_lt_of_le coordinate.x.isLt - |> λx↦(Nat.add_lt_add_right) x (grid.width * coordinate.y.val) - |> (Nat.sub_add_cancel (Nat.le_of_lt ((Nat.mul_lt_mul_left (Nat.zero_lt_of_lt coordinate.x.isLt)).mpr coordinate.y.isLt))).subst⟩ + ⟨coordinate.x + grid.width * coordinate.y, Nat.two_d_coordinate_to_index_lt_size coordinate.x.isLt coordinate.y.isLt⟩ private def RectangularGrid.Coordinate.ofIndex (grid : RectangularGrid Element) (index : Fin (grid.width * grid.height)) : grid.Coordinate := have : grid.width > 0 := @@ -42,11 +39,27 @@ theorem Coordinate.toIndex_inv_fromIndex {grid : RectangularGrid Element} (index simp only [RectangularGrid.Coordinate.toIndex, RectangularGrid.Coordinate.ofIndex, Nat.mod_add_div, Fin.eta] def RectangularGrid.Get {grid : RectangularGrid Element} (coordinate : grid.Coordinate) : Element := - grid.elements[coordinate.toIndex]'(grid.valid.substr coordinate.toIndex.isLt) + grid.elements[coordinate.toIndex]'(grid.size_valid.substr coordinate.toIndex.isLt) instance : GetElem (RectangularGrid Element) (RectangularGrid.Coordinate grid) Element (λg _ ↦ g = grid) where getElem := λ g c h ↦ g.Get (h▸c) +instance [ToString Element] : ToString (MaybeEmptyRectangularGrid Element) where + toString := λe ↦ Id.run do + let mut r := s!"Width: {e.width}, Height: {e.height}" + for h₂ : y in [0:e.height] do + r := r.push '\n' + for h₁ : x in [0:e.width] do + have : x + e.width *y < e.elements.size := by + simp[Membership.mem, inferInstance, Std.instMembershipNatRange] at h₁ h₂ + rw[e.size_valid] + exact Nat.two_d_coordinate_to_index_lt_size h₁ h₂ + r := r ++ (ToString.toString e.elements[x+e.width*y]) + r + +instance [ToString Element] : ToString (RectangularGrid Element) where + toString := ToString.toString ∘ RectangularGrid.toMaybeEmptyRectangularGrid + inductive RectangularGrid.ParseError (ParseCharError : Type) | NoInput : RectangularGrid.ParseError ParseCharError | NotRectangular : RectangularGrid.ParseError ParseCharError @@ -58,21 +71,70 @@ instance [ToString ParseCharError] : ToString (RectangularGrid.ParseError ParseC | .NotRectangular => "Input is not rectangular." | .CharacterParsingError e => ToString.toString e -def RectangularGrid.parseSingleLine (parseCharacter : Char → Except ParseCharError Element) (alreadyParsed : Array Element) (remainingLine : Substring) : Except (RectangularGrid.ParseError ParseCharError) (Array Element × Nat) := - sorry +/-- Internal helper. Not private so it's usable in proofs. You probably want ofSubstring instead. -/ +protected def RectangularGrid.parseSingleLine (parseCharacter : Char → Except ParseCharError Element) (alreadyParsed : Array Element) (remainingLine : Substring) : Except (RectangularGrid.ParseError ParseCharError) (Array Element) := + if _h₁ : remainingLine.isEmpty then + Except.ok alreadyParsed + else + let c := remainingLine.front + match parseCharacter c with + | Except.error e => Except.error $ RectangularGrid.ParseError.CharacterParsingError e + | Except.ok element => + RectangularGrid.parseSingleLine parseCharacter (alreadyParsed.push element) (remainingLine.drop 1) +termination_by remainingLine.bsize +decreasing_by + simp_wf + simp only [Substring.isEmpty, Substring.bsize, Nat.sub_eq, beq_iff_eq] at _h₁ + simp only [Substring.drop, Substring.bsize, Substring.nextn, Substring.next, String.Pos.add_byteIdx, Nat.sub_eq] + split + case isTrue h₂ => + simp[←(congrArg String.Pos.byteIdx h₂)] at _h₁ + case isFalse h₂ => + simp[String.next, Char.utf8Size] + split <;> try split <;> try split + all_goals omega -def RectangularGrid.parseLines (parseCharacter : Char → Except ParseCharError Element) (alreadyParsed : Array Element) (remainingLines : List Substring) : Except (RectangularGrid.ParseError ParseCharError) (Array Element × Nat × Nat) := - sorry +/-- Internal helper. Not private so it's usable in proofs. You probably want ofSubstring instead. -/ +protected def RectangularGrid.parseLines (parseCharacter : Char → Except ParseCharError Element) (alreadyParsed : MaybeEmptyRectangularGrid Element) (remainingLines : List Substring) : Except (RectangularGrid.ParseError ParseCharError) (RectangularGrid Element) := + match remainingLines with + | [] => + if h₁ : alreadyParsed.width = 0 ∨ alreadyParsed.height = 0 then + Except.error RectangularGrid.ParseError.NoInput + else + Except.ok {alreadyParsed with not_empty := ⟨Nat.pos_of_ne_zero (not_or.mp h₁).left, Nat.pos_of_ne_zero (not_or.mp h₁).right⟩} + | l :: ls => + let currentSize := alreadyParsed.elements.size + match RectangularGrid.parseSingleLine parseCharacter alreadyParsed.elements l with + | Except.error e => + Except.error e + | Except.ok elements => + let newSize := elements.size + if alreadyParsed.height = 0 then -- first line + RectangularGrid.parseLines parseCharacter { + width := newSize + height := 1 + elements + size_valid := (Nat.mul_one _).substr rfl + } ls + else + if h₂ : newSize = currentSize + alreadyParsed.width then + RectangularGrid.parseLines parseCharacter { + width := alreadyParsed.width + height := alreadyParsed.height + 1 + elements + size_valid := + alreadyParsed.size_valid.subst (motive := λx↦_=x+_) h₂ + |> (Nat.mul_succ (alreadyParsed.width) (alreadyParsed.height)).substr + } ls + else + Except.error RectangularGrid.ParseError.NotRectangular def RectangularGrid.ofSubstring (parseCharacter : Char → Except ParseCharError Element) (input : Substring) : Except (RectangularGrid.ParseError ParseCharError) (RectangularGrid Element) := let input := input.trim let lines := input.splitOn "\n" - if h₁ : lines.isEmpty then - throw RectangularGrid.ParseError.NoInput - else - --let height := lines.length - --let width := lines[0]'(List.not_empty_iff_size_gt_zero.mp $ (Bool.not_eq_true _).mp h₁) - sorry + let lines := lines.map Substring.trim + let lines := lines.filter $ not ∘ Substring.isEmpty + RectangularGrid.parseLines parseCharacter Inhabited.default lines def RectangularGrid.ofString (parseCharacter : Char → Except ParseCharError Element) (input : String) : Except (RectangularGrid.ParseError ParseCharError) (RectangularGrid Element) := ofSubstring parseCharacter input.toSubstring |
