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

Some more serialization of stuff around localizations #4773

17 changes: 8 additions & 9 deletions src/Rings/mpolyquo-localizations.jl
Original file line number Diff line number Diff line change
Expand Up @@ -993,15 +993,14 @@ write_as_linear_combination(f::MPolyQuoLocRingElem, g::Vector) = write_as_linear

@doc raw"""
MPolyQuoLocalizedRingHom{
BaseRingType,
BaseRingElemType,
RingType,
RingElemType,
DomainMultSetType,
CodomainMultSetType
} <: AbsLocalizedRingHom{
RingType, RingElemType, DomainMultSetType, CodomainMultSetType
}
DomainType<:MPolyQuoLocRing,
CodomainType<:Ring,
RestrictedMapType<:Map
} <: AbsLocalizedRingHom{
DomainType,
CodomainType,
RestrictedMapType
}

Homomorphisms of localizations of affine algebras

Expand Down
167 changes: 164 additions & 3 deletions src/Serialization/Rings.jl
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,13 @@ const PolyRingUnionType = Union{UniversalPolyRing,
AbstractAlgebra.Generic.LaurentMPolyWrapRing}

const IdealUnionType = Union{MPolyIdeal,
LaurentMPolyIdeal,
FreeAssociativeAlgebraIdeal}
MPolyQuoIdeal,
MPolyLocalizedIdeal,
MPolyQuoLocalizedIdeal,
LaurentMPolyIdeal,
FreeAssociativeAlgebraIdeal,
IdealGens
}

const RelPowerSeriesUnionType = Union{Generic.RelPowerSeriesRing,
QQRelPowerSeriesRing,
Expand Down Expand Up @@ -247,9 +252,11 @@ end

################################################################################
# Polynomial Ideals

@register_serialization_type MPolyIdeal
@register_serialization_type LaurentMPolyIdeal
@register_serialization_type MPolyLocalizedIdeal
@register_serialization_type MPolyQuoLocalizedIdeal
@register_serialization_type MPolyQuoIdeal

function save_object(s::SerializerState, I::T) where T <: IdealUnionType
# we might want to serialize generating_system(I) and I.gb
Expand Down Expand Up @@ -656,3 +663,157 @@ function load_object(s::DeserializerState, ::Type{Orderings.SymbOrdering})
vars = load_object(s, Vector{Int}, :vars) # are these always Vector{Int} ?
return Orderings.SymbOrdering(S, vars)
end


# localizations of polynomial rings
@register_serialization_type MPolyPowersOfElement uses_id

type_params(U::T) where T <: MPolyPowersOfElement = TypeParams(T, ring(U))

function save_object(s::SerializerState, U::MPolyPowersOfElement)
save_object(s, denominators(U))
end

function load_object(s::DeserializerState, ::Type{<:MPolyPowersOfElement}, R::MPolyRing)
dens = load_object(s, Vector{elem_type(R)}, R)
return MPolyPowersOfElement(R, dens)
end


@register_serialization_type MPolyComplementOfPrimeIdeal uses_id

type_params(U::MPolyComplementOfPrimeIdeal) = TypeParams(typeof(U), ring(U))

function save_object(s::SerializerState, U::MPolyComplementOfPrimeIdeal)
save_object(s, prime_ideal(U))
end

function load_object(s::DeserializerState, ::Type{<:MPolyComplementOfPrimeIdeal}, R::MPolyRing)
id = load_object(s, ideal_type(R), R)
return MPolyComplementOfPrimeIdeal(id)
end

@register_serialization_type MPolyLocRing uses_id

type_params(W::T) where {T <: MPolyLocRing} = TypeParams(T, :base_ring => base_ring(W), :mult_set_type => TypeParams(typeof(inverted_set(W)), nothing)) # TODO: make this neater!

function save_object(s::SerializerState, L::MPolyLocRing)
save_object(s, inverted_set(L))
end

function load_object(
s::DeserializerState,
::Type{<:MPolyLocRing}, params::Dict
)
U = params[:mult_set_type]
R = params[:base_ring]
mult_set = load_object(s, U, R)
return MPolyLocRing(R, mult_set)
end

@register_serialization_type MPolyLocRingElem uses_params

type_params(a::MPolyLocRingElem) = TypeParams(MPolyLocRingElem, parent(a))

function save_object(s::SerializerState, a::MPolyLocRingElem)
# Because the `parent` of `a` is a `Ring` the generic implementation
# for `uses_params` above calls `save_type_params` and that stores
# the ring. Hopefully.
save_data_array(s) do
save_object(s, numerator(a))
save_object(s, denominator(a))
end
end

function load_object(s::DeserializerState, ::Type{<:MPolyLocRingElem}, parent::MPolyLocRing)
P = base_ring(parent)
RET = elem_type(P)
num = load_object(s, RET, P, 1)
den = load_object(s, RET, P, 2)
return parent(num, den; check=false)
end

@register_serialization_type MPolyQuoLocRing uses_id

type_params(L::T) where {T <: MPolyQuoLocRing} = TypeParams(T, :base_ring=>base_ring(L), :loc_ring=>localized_ring(L), :quo_ring=>underlying_quotient(L))

function save_object(s::SerializerState, L::MPolyQuoLocRing)
save_data_dict(s) do
# Everything happens in the type_params.
# We still need to do something here, because otherwise
# we get an error. TODO: Make this better!
end
end

function load_object(s::DeserializerState, ::Type{<:MPolyQuoLocRing}, params::Dict)
R = params[:base_ring]::MPolyRing
L = params[:loc_ring]::MPolyLocRing
Q = params[:quo_ring]::MPolyQuoRing
return MPolyQuoLocRing(R, modulus(Q), inverted_set(L), Q, L)
end

@register_serialization_type MPolyQuoLocRingElem uses_params

type_params(a::T) where {T<:MPolyQuoLocRingElem} = TypeParams(T, parent(a))

function save_object(s::SerializerState, a::MPolyQuoLocRingElem)
save_object(s, [lifted_numerator(a), lifted_denominator(a)])
end

function load_object(s::DeserializerState, ::Type{<:MPolyQuoLocRingElem}, parent::MPolyQuoLocRing)
P = base_ring(parent)
RET = elem_type(P)
num = load_object(s, RET, P, 1)
den = load_object(s, RET, P, 2)
return parent(num, den; check=false)
end

@register_serialization_type MPolyComplementOfKPointIdeal uses_id

type_params(U::T) where {T<:MPolyComplementOfKPointIdeal} = TypeParams(T, ring(U))

function save_object(s::SerializerState, U::MPolyComplementOfKPointIdeal)
save_object(s, point_coordinates(U))
end

function load_object(s::DeserializerState, ::Type{<:MPolyComplementOfKPointIdeal}, R::Ring)
kk = coefficient_ring(R)
T = elem_type(kk)
a = load_object(s, Vector{T}, kk)
return MPolyComplementOfKPointIdeal(R, a)
end

### Morphisms of the four types of rings

@register_serialization_type MPolyLocalizedRingHom uses_id

type_params(phi::T) where {T<:MPolyLocalizedRingHom} = TypeParams(T, :domain=>domain(phi), :codomain=>codomain(phi), :restricted_map_params=>type_params(restricted_map(phi)))

function save_object(s::SerializerState, phi::MPolyLocalizedRingHom)
save_object(s, restricted_map(phi))
end

function load_object(s::DeserializerState, ::Type{T}, params::Dict) where {T<:MPolyLocalizedRingHom} # RT is the type of the `restricted_map`
dom = params[:domain]
cod = params[:codomain]
rm_tp = params[:restricted_map_params]
res = load_object(s, MPolyAnyMap, rm_tp)
return MPolyLocalizedRingHom(dom, cod, res; check=false)
end

@register_serialization_type MPolyQuoLocalizedRingHom uses_id

type_params(phi::T) where {T<:MPolyQuoLocalizedRingHom} = TypeParams(T, :domain=>domain(phi), :codomain=>codomain(phi), :restricted_map_params=>type_params(restricted_map(phi)))

function save_object(s::SerializerState, phi::MPolyQuoLocalizedRingHom)
save_object(s, restricted_map(phi))
end

function load_object(s::DeserializerState, ::Type{T}, params::Dict) where {T<:MPolyQuoLocalizedRingHom} # RT is the type of the `restricted_map`
dom = params[:domain]
cod = params[:codomain]
rm_tp = params[:restricted_map_params]
res = load_object(s, MPolyAnyMap, rm_tp)
return MPolyQuoLocalizedRingHom(dom, cod, res; check=false)
end

23 changes: 23 additions & 0 deletions src/Serialization/main.jl
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,29 @@ type(tp::TypeParams) = tp.type

type_params(obj::T) where T = TypeParams(T, nothing)

function Base.show(io::IO, tp::TypeParams{T, Tuple}) where T
if is_terse(io)
print(io, "Type parameters for $T")
else
io = pretty(io)
print(io, "Type parameters for $T")
for param in params(tp)
println(io, "")
print(terse(io), Lowercase(), param)
end
end
end

function Base.show(io::IO, tp::TypeParams{T, S}) where {T, S}
if is_terse(io)
print(io, "Type parameters for $T")
else
io = pretty(io)
print(io, "Type parameter for $T ")
print(terse(io), Lowercase(), params(tp))
end
end

# ATTENTION
# We need to distinguish between data with a globally defined normal form and data where such a normal form depends on some parameters.
# In particular, this does NOT ONLY depend on the type; see, e.g., FqField.
Expand Down
6 changes: 6 additions & 0 deletions src/Serialization/parallel.jl
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,12 @@ function put_type_params(channel::RemoteChannel, tps::Tuple{Vararg{Pair}})
end
end

function put_type_params(channel::RemoteChannel, tps::Tuple{Vararg{TypeParams}})
for tp in tps
put_type_params(channel, params(tp))
end
end

function put_type_params(channel::RemoteChannel, obj::T) where T
# only types that use ids need to be sent to the other processes
if serialize_with_id(T)
Expand Down
33 changes: 33 additions & 0 deletions test/Serialization/PolynomialsSeries.jl
Original file line number Diff line number Diff line change
Expand Up @@ -210,3 +210,36 @@ cases = [
end
end
end

@testset "localizations and quotients" begin
mktempdir() do path
R, (x, y, z) = GF(103)[:x, :y, :z]
for U in [powers_of_element(x),
complement_of_point_ideal(R, GF(103).([1, 2, 3])),
complement_of_prime_ideal(ideal(R, x))
]
L, _ = localization(R, U)
Q, _ = quo(L, ideal(L, y))
Qz = Q(z)
# Test MPolyQuoLocRings and their elements
test_save_load_roundtrip(path, Qz; params=Q) do loaded
@test Qz == loaded
end
# Test ideals in these rings.
J = ideal(Q, z)
test_save_load_roundtrip(path, J; params=Q) do loaded
@test J == loaded
end
# Test MPolyLocalizedRingHom
phi = hom(L, L, gens(L))
test_save_load_roundtrip(path, phi) do loaded
@test phi == loaded
end
# Test MPolyQuoLocalizedRingHom
phi = hom(Q, Q, gens(Q))
test_save_load_roundtrip(path, phi) do loaded
@test phi == loaded
end
end
end
end
Loading