/- Copyright (c) 2024 Lean FRO, LLC. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Authors: Markus Himmel -/ prelude import Std.Data.DHashMap.Raw /-! # Dependent hash maps This file develops the type `Std.Data.DHashMap` of dependent hash maps. The operations `map` and `filterMap` on `Std.Data.DHashMap` are defined in the module `Std.Data.DHashMap.AdditionalOperations`. Lemmas about the operations on `Std.Data.DHashMap` are available in the module `Std.Data.DHashMap.Lemmas`. See the module `Std.Data.DHashMap.Raw` for a variant of this type which is safe to use in nested inductive types. For implementation notes, see the docstring of the module `Std.Data.DHashMap.Internal.Defs`. -/ set_option linter.missingDocs true set_option autoImplicit false universe u v w variable {α : Type u} {β : α → Type v} {δ : Type w} {m : Type w → Type w} [Monad m] namespace Std open DHashMap.Internal DHashMap.Internal.List /-- Dependent hash maps. This is a simple separate-chaining hash table. The data of the hash map consists of a cached size and an array of buckets, where each bucket is a linked list of key-value pais. The number of buckets is always a power of two. The hash map doubles its size upon inserting an element such that the number of elements is more than 75% of the number of buckets. The hash map uses `==` (provided by the `BEq` typeclass) to compare keys and `hash` (provided by the `Hashable` typeclass) to hash them. To ensure that the operations behave as expected, `==` should be an equivalence relation and `a == b` should imply `hash a = hash b` (see also the `EquivBEq` and `LawfulHashable` typeclasses). Both of these conditions are automatic if the BEq instance is lawful, i.e., if `a == b` implies `a = b`. These hash maps contain a bundled well-formedness invariant, which means that they cannot be used in nested inductive types. For these use cases, `Std.Data.DHashMap.Raw` and `Std.Data.DHashMap.Raw.WF` unbundle the invariant from the hash map. When in doubt, prefer `DHashMap` over `DHashMap.Raw`. -/ def DHashMap (α : Type u) (β : α → Type v) [BEq α] [Hashable α] := { m : DHashMap.Raw α β // m.WF } namespace DHashMap @[inline, inherit_doc Raw.empty] def empty [BEq α] [Hashable α] (capacity := 8) : DHashMap α β := ⟨Raw.empty capacity, .empty₀⟩ instance [BEq α] [Hashable α] : EmptyCollection (DHashMap α β) where emptyCollection := empty instance [BEq α] [Hashable α] : Inhabited (DHashMap α β) where default := ∅ @[inline, inherit_doc Raw.insert] def insert [BEq α] [Hashable α] (m : DHashMap α β) (a : α) (b : β a) : DHashMap α β := ⟨Raw₀.insert ⟨m.1, m.2.size_buckets_pos⟩ a b, .insert₀ m.2⟩ @[inline, inherit_doc Raw.insertIfNew] def insertIfNew [BEq α] [Hashable α] (m : DHashMap α β) (a : α) (b : β a) : DHashMap α β := ⟨Raw₀.insertIfNew ⟨m.1, m.2.size_buckets_pos⟩ a b, .insertIfNew₀ m.2⟩ @[inline, inherit_doc Raw.containsThenInsert] def containsThenInsert [BEq α] [Hashable α] (m : DHashMap α β) (a : α) (b : β a) : Bool × DHashMap α β := let m' := Raw₀.containsThenInsert ⟨m.1, m.2.size_buckets_pos⟩ a b ⟨m'.1, ⟨m'.2.1, .containsThenInsert₀ m.2⟩⟩ @[inline, inherit_doc Raw.containsThenInsertIfNew] def containsThenInsertIfNew [BEq α] [Hashable α] (m : DHashMap α β) (a : α) (b : β a) : Bool × DHashMap α β := let m' := Raw₀.containsThenInsertIfNew ⟨m.1, m.2.size_buckets_pos⟩ a b ⟨m'.1, ⟨m'.2.1, .containsThenInsertIfNew₀ m.2⟩⟩ @[inline, inherit_doc Raw.getThenInsertIfNew?] def getThenInsertIfNew? [BEq α] [Hashable α] [LawfulBEq α] (m : DHashMap α β) (a : α) (b : β a) : Option (β a) × DHashMap α β := let m' := Raw₀.getThenInsertIfNew? ⟨m.1, m.2.size_buckets_pos⟩ a b ⟨m'.1, ⟨m'.2.1, .getThenInsertIfNew?₀ m.2⟩⟩ @[inline, inherit_doc Raw.get?] def get? [BEq α] [LawfulBEq α] [Hashable α] (m : DHashMap α β) (a : α) : Option (β a) := Raw₀.get? ⟨m.1, m.2.size_buckets_pos⟩ a @[inline, inherit_doc Raw.contains] def contains [BEq α] [Hashable α] (m : DHashMap α β) (a : α) : Bool := Raw₀.contains ⟨m.1, m.2.size_buckets_pos⟩ a instance [BEq α] [Hashable α] : Membership α (DHashMap α β) where mem a m := m.contains a instance [BEq α] [Hashable α] {m : DHashMap α β} {a : α} : Decidable (a ∈ m) := show Decidable (m.contains a) from inferInstance @[inline, inherit_doc Raw.get] def get [BEq α] [Hashable α] [LawfulBEq α] (m : DHashMap α β) (a : α) (h : a ∈ m) : β a := Raw₀.get ⟨m.1, m.2.size_buckets_pos⟩ a h @[inline, inherit_doc Raw.get!] def get! [BEq α] [Hashable α] [LawfulBEq α] (m : DHashMap α β) (a : α) [Inhabited (β a)] : β a := Raw₀.get! ⟨m.1, m.2.size_buckets_pos⟩ a @[inline, inherit_doc Raw.getD] def getD [BEq α] [Hashable α] [LawfulBEq α] (m : DHashMap α β) (a : α) (fallback : β a) : β a := Raw₀.getD ⟨m.1, m.2.size_buckets_pos⟩ a fallback @[inline, inherit_doc Raw.erase] def erase [BEq α] [Hashable α] (m : DHashMap α β) (a : α) : DHashMap α β := ⟨Raw₀.erase ⟨m.1, m.2.size_buckets_pos⟩ a, .erase₀ m.2⟩ section variable {β : Type v} @[inline, inherit_doc Raw.Const.get?] def Const.get? [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) (a : α) : Option β := Raw₀.Const.get? ⟨m.1, m.2.size_buckets_pos⟩ a @[inline, inherit_doc Raw.Const.get] def Const.get [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) (a : α) (h : a ∈ m) : β := Raw₀.Const.get ⟨m.1, m.2.size_buckets_pos⟩ a h @[inline, inherit_doc Raw.Const.getD] def Const.getD [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) (a : α) (fallback : β) : β := Raw₀.Const.getD ⟨m.1, m.2.size_buckets_pos⟩ a fallback @[inline, inherit_doc Raw.Const.get!] def Const.get! [BEq α] [Hashable α] [Inhabited β] (m : DHashMap α (fun _ => β)) (a : α) : β := Raw₀.Const.get! ⟨m.1, m.2.size_buckets_pos⟩ a @[inline, inherit_doc Raw.Const.getThenInsertIfNew?] def Const.getThenInsertIfNew? [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) (a : α) (b : β) : Option β × DHashMap α (fun _ => β) := let m' := Raw₀.Const.getThenInsertIfNew? ⟨m.1, m.2.size_buckets_pos⟩ a b ⟨m'.1, ⟨m'.2.1, .constGetThenInsertIfNew?₀ m.2⟩⟩ end @[inline, inherit_doc Raw.size] def size [BEq α] [Hashable α] (m : DHashMap α β) : Nat := m.1.size @[inline, inherit_doc Raw.isEmpty] def isEmpty [BEq α] [Hashable α] (m : DHashMap α β) : Bool := m.1.isEmpty section Unverified /-! We currently do not provide lemmas for the functions below. -/ @[inline, inherit_doc Raw.filter] def filter [BEq α] [Hashable α] (f : (a : α) → β a → Bool) (m : DHashMap α β) : DHashMap α β := ⟨Raw₀.filter f ⟨m.1, m.2.size_buckets_pos⟩, .filter₀ m.2⟩ @[inline, inherit_doc Raw.foldM] def foldM [BEq α] [Hashable α] (f : δ → (a : α) → β a → m δ) (init : δ) (b : DHashMap α β) : m δ := b.1.foldM f init @[inline, inherit_doc Raw.fold] def fold [BEq α] [Hashable α] (f : δ → (a : α) → β a → δ) (init : δ) (b : DHashMap α β) : δ := b.1.fold f init @[inline, inherit_doc Raw.forM] def forM [BEq α] [Hashable α] (f : (a : α) → β a → m PUnit) (b : DHashMap α β) : m PUnit := b.1.forM f @[inline, inherit_doc Raw.forIn] def forIn [BEq α] [Hashable α] (f : (a : α) → β a → δ → m (ForInStep δ)) (init : δ) (b : DHashMap α β) : m δ := b.1.forIn f init instance [BEq α] [Hashable α] : ForM m (DHashMap α β) ((a : α) × β a) where forM m f := m.forM (fun a b => f ⟨a, b⟩) instance [BEq α] [Hashable α] : ForIn m (DHashMap α β) ((a : α) × β a) where forIn m init f := m.forIn (fun a b acc => f ⟨a, b⟩ acc) init @[inline, inherit_doc Raw.toList] def toList [BEq α] [Hashable α] (m : DHashMap α β) : List ((a : α) × β a) := m.1.toList @[inline, inherit_doc Raw.toArray] def toArray [BEq α] [Hashable α] (m : DHashMap α β) : Array ((a : α) × β a) := m.1.toArray @[inline, inherit_doc Raw.Const.toList] def Const.toList {β : Type v} [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) : List (α × β) := Raw.Const.toList m.1 @[inline, inherit_doc Raw.Const.toArray] def Const.toArray {β : Type v} [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) : Array (α × β) := Raw.Const.toArray m.1 @[inline, inherit_doc Raw.keys] def keys [BEq α] [Hashable α] (m : DHashMap α β) : List α := m.1.keys @[inline, inherit_doc Raw.keysArray] def keysArray [BEq α] [Hashable α] (m : DHashMap α β) : Array α := m.1.keysArray @[inline, inherit_doc Raw.values] def values {β : Type v} [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) : List β := m.1.values @[inline, inherit_doc Raw.valuesArray] def valuesArray {β : Type v} [BEq α] [Hashable α] (m : DHashMap α (fun _ => β)) : Array β := m.1.valuesArray @[inline, inherit_doc Raw.insertMany] def insertMany [BEq α] [Hashable α] {ρ : Type w} [ForIn Id ρ ((a : α) × β a)] (m : DHashMap α β) (l : ρ) : DHashMap α β := ⟨(Raw₀.insertMany ⟨m.1, m.2.size_buckets_pos⟩ l).1, (Raw₀.insertMany ⟨m.1, m.2.size_buckets_pos⟩ l).2 _ Raw.WF.insert₀ m.2⟩ @[inline, inherit_doc Raw.Const.insertMany] def Const.insertMany {β : Type v} [BEq α] [Hashable α] {ρ : Type w} [ForIn Id ρ (α × β)] (m : DHashMap α (fun _ => β)) (l : ρ) : DHashMap α (fun _ => β) := ⟨(Raw₀.Const.insertMany ⟨m.1, m.2.size_buckets_pos⟩ l).1, (Raw₀.Const.insertMany ⟨m.1, m.2.size_buckets_pos⟩ l).2 _ Raw.WF.insert₀ m.2⟩ @[inline, inherit_doc Raw.Const.insertManyUnit] def Const.insertManyUnit [BEq α] [Hashable α] {ρ : Type w} [ForIn Id ρ α] (m : DHashMap α (fun _ => Unit)) (l : ρ) : DHashMap α (fun _ => Unit) := ⟨(Raw₀.Const.insertManyUnit ⟨m.1, m.2.size_buckets_pos⟩ l).1, (Raw₀.Const.insertManyUnit ⟨m.1, m.2.size_buckets_pos⟩ l).2 _ Raw.WF.insert₀ m.2⟩ @[inline, inherit_doc Raw.ofList] def ofList [BEq α] [Hashable α] (l : List ((a : α) × β a)) : DHashMap α β := insertMany ∅ l @[inline, inherit_doc Raw.Const.ofList] def Const.ofList {β : Type v} [BEq α] [Hashable α] (l : List (α × β)) : DHashMap α (fun _ => β) := Const.insertMany ∅ l @[inline, inherit_doc Raw.Const.unitOfList] def Const.unitOfList [BEq α] [Hashable α] (l : List α) : DHashMap α (fun _ => Unit) := Const.insertManyUnit ∅ l @[inherit_doc Raw.Internal.numBuckets] def Internal.numBuckets [BEq α] [Hashable α] (m : DHashMap α β) : Nat := Raw.Internal.numBuckets m.1 instance [BEq α] [Hashable α] [Repr α] [(a : α) → Repr (β a)] : Repr (DHashMap α β) where reprPrec m prec := Repr.addAppParen ("Std.DHashMap.ofList " ++ reprArg m.toList) prec end Unverified end Std.DHashMap