Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: allow dot idents to resolve to local names #6602

Merged
merged 2 commits into from
Jan 12, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 8 additions & 5 deletions src/Lean/Elab/App.lean
Original file line number Diff line number Diff line change
Expand Up @@ -1474,7 +1474,7 @@ where
| field::fields, false => .fieldName field field.getId.getString! none fIdent :: toLVals fields false

/-- Resolve `(.$id:ident)` using the expected type to infer namespace. -/
private partial def resolveDotName (id : Syntax) (expectedType? : Option Expr) : TermElabM Name := do
private partial def resolveDotName (id : Syntax) (expectedType? : Option Expr) : TermElabM Expr := do
tryPostponeIfNoneOrMVar expectedType?
let some expectedType := expectedType?
| throwError "invalid dotted identifier notation, expected type must be known"
Expand All @@ -1489,17 +1489,20 @@ where
withForallBody body k
else
k body
go (resultType : Expr) (expectedType : Expr) (previousExceptions : Array Exception) : TermElabM Name := do
go (resultType : Expr) (expectedType : Expr) (previousExceptions : Array Exception) : TermElabM Expr := do
let resultType ← instantiateMVars resultType
let resultTypeFn := resultType.cleanupAnnotations.getAppFn
try
tryPostponeIfMVar resultTypeFn
let .const declName .. := resultTypeFn.cleanupAnnotations
| throwError "invalid dotted identifier notation, expected type is not of the form (... → C ...) where C is a constant{indentExpr expectedType}"
let idNew := declName ++ id.getId.eraseMacroScopes
unless (← getEnv).contains idNew do
if (← getEnv).contains idNew then
mkConst idNew
else if let some (fvar, []) ← resolveLocalName idNew then
return fvar
else
throwError "invalid dotted identifier notation, unknown identifier `{idNew}` from expected type{indentExpr expectedType}"
return idNew
catch
| ex@(.error ..) =>
match (← unfoldDefinition? resultType) with
Expand Down Expand Up @@ -1548,7 +1551,7 @@ private partial def elabAppFn (f : Syntax) (lvals : List LVal) (namedArgs : Arra
| `(_) => throwError "placeholders '_' cannot be used where a function is expected"
| `(.$id:ident) =>
addCompletionInfo <| CompletionInfo.dotId f id.getId (← getLCtx) expectedType?
let fConst ← mkConst (← resolveDotName id expectedType?)
let fConst ← resolveDotName id expectedType?
let s ← observing do
-- Use (force := true) because we want to record the result of .ident resolution even in patterns
let fConst ← addTermInfo f fConst expectedType? (force := true)
Expand Down
4 changes: 2 additions & 2 deletions src/Lean/Parser/Basic.lean
Original file line number Diff line number Diff line change
Expand Up @@ -1583,8 +1583,8 @@ namespace TokenMap

def insert (map : TokenMap α) (k : Name) (v : α) : TokenMap α :=
match map.find? k with
| none => .insert map k [v]
| some vs => .insert map k (v::vs)
| none => RBMap.insert map k [v]
| some vs => RBMap.insert map k (v::vs)

instance : Inhabited (TokenMap α) where
default := RBMap.empty
Expand Down
2 changes: 1 addition & 1 deletion src/Lean/Server/Requests.lean
Original file line number Diff line number Diff line change
Expand Up @@ -97,7 +97,7 @@ abbrev RequestT m := ReaderT RequestContext <| ExceptT RequestError m
/-- Workers execute request handlers in this monad. -/
abbrev RequestM := ReaderT RequestContext <| EIO RequestError

abbrev RequestTask.pure (a : α) : RequestTask α := .pure (.ok a)
abbrev RequestTask.pure (a : α) : RequestTask α := Task.pure (.ok a)

instance : MonadLift IO RequestM where
monadLift x := do
Expand Down
2 changes: 1 addition & 1 deletion src/Std/Time/Date/Unit/Month.lean
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ Creates an `Offset` from a natural number.
-/
@[inline]
def ofNat (data : Nat) : Offset :=
.ofNat data
Int.ofNat data

/--
Creates an `Offset` from an integer.
Expand Down
2 changes: 1 addition & 1 deletion src/Std/Time/Date/Unit/Year.lean
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ Creates an `Offset` from a natural number.
-/
@[inline]
def ofNat (data : Nat) : Offset :=
.ofNat data
Int.ofNat data

/--
Creates an `Offset` from an integer.
Expand Down
26 changes: 26 additions & 0 deletions tests/lean/6601.lean
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
/-! Dot ident notation should resolve mutual definitions -/

mutual

inductive Even
| zero
| succ (n : Odd)
deriving Inhabited

inductive Odd
| succ (n : Even)
deriving Inhabited

end

mutual

def Even.ofNat : Nat → Even
| 0 => .zero
| n + 1 => .succ (.ofNat n)

def Odd.ofNat : Nat → Odd
| 0 => panic! "invalid input"
| n + 1 => .succ (.ofNat n)

end
Empty file.
Loading