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

Specify interface for LibGit2 abstract types #36452

Merged
merged 2 commits into from
Jul 13, 2020
Merged
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
36 changes: 36 additions & 0 deletions stdlib/LibGit2/src/types.jl
Original file line number Diff line number Diff line change
Expand Up @@ -966,11 +966,47 @@ function split_cfg_entry(ce::ConfigEntry)
end

# Abstract object types

"""
AbstractGitObject

`AbstractGitObject`s must obey the following interface:
- `obj.owner`, if present, must be a `Union{Nothing,GitRepo,GitTree}`
- `obj.ptr`, if present, must be a `Union{Ptr{Cvoid},Ptr{SignatureStruct}}`
"""
abstract type AbstractGitObject end

function Base.getproperty(obj::AbstractGitObject, name::Symbol)
# These type-assertions enforce the interface requirements above.
# They assist type-inference in cases where the compiler only knows that it
# has an `AbstractGitObject` without being certain about the concrete type.
# See detailed explanation in https://github.com/JuliaLang/julia/pull/36452.
if name === :owner
return getfield(obj, :owner)::Union{Nothing,GitRepo,GitTree}
elseif name === :ptr
return getfield(obj, :ptr)::Union{Ptr{Cvoid},Ptr{SignatureStruct}}
else
return getfield(obj, name)
end
end
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think a comment should be added explaining that this helps inference as on its own the function looks redundant


Base.isempty(obj::AbstractGitObject) = (obj.ptr == C_NULL)

# `GitObject`s must obey the following interface:
# - `obj.owner` must be a `GitRepo`
# - `obj.ptr` must be a Ptr{Cvoid}
abstract type GitObject <: AbstractGitObject end

function Base.getproperty(obj::GitObject, name::Symbol)
if name === :owner
return getfield(obj, :owner)::GitRepo
elseif name === :ptr
return getfield(obj, :ptr)::Ptr{Cvoid}
else
return getfield(obj, name)
end
end

for (typ, owntyp, sup, cname) in [
(:GitRepo, nothing, :AbstractGitObject, :git_repository),
(:GitConfig, :(Union{GitRepo, Nothing}), :AbstractGitObject, :git_config),
Expand Down