A freezing bucket

A common laboratory experiment freezes an insultated bucket of water from the top down, using a metal lid to keep the top of the bucket at some constant, very cold temperature. In this example, we simulate such a scenario using SlabSeaIceModel. Here, the bucket is perfectly insulated and infinitely deep, like many buckets are: if the Simulation is run for longer, the ice will keep freezing, and freezing, and will never run out of water. Also, the water in the infinite bucket is (somehow) all at the same temperature, in equilibrium with the ice-water interface (and therefore fixed at the melting temperature). Yep, this kind of thing happens all the time.

We start by using Oceananigans to bring in functions for building grids and Simulations and the like.

using Oceananigans
using Oceananigans.Units

Next we using ClimaSeaIce to get some ice-specific names.

using ClimaSeaIce

An infinitely deep bucket with a single grid point

Perhaps surprisingly, we need just one grid point to model an possibly infinitely thick slab of ice with SlabSeaIceModel. We would only need more than 1 grid point if our boundary conditions vary in the horizontal direction.

grid = RectilinearGrid(size=(), topology=(Flat, Flat, Flat))
1×1×1 RectilinearGrid{Float64, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat} on Oceananigans.Architectures.CPU with 0×0×0 halo
├── Flat x 
├── Flat y 
└── Flat z 

Next, we build our model of an ice slab freezing into a bucket. We start by defining a constant internal ConductiveFlux with ice_conductivity

conductivity = 2 # kg m s⁻³ K⁻¹
internal_heat_flux = ConductiveFlux(; conductivity)
ConductiveFlux{Float64}(2.0)

Note that other units besides Celsius can be used, but that requires setting model.phase_transitions` with appropriate parameters. We set the ice heat capacity and density as well,

ice_heat_capacity = 2100 # J kg⁻¹ K⁻¹
ice_density = 900 # kg m⁻³
phase_transitions = PhaseTransitions(; ice_heat_capacity, ice_density)
PhaseTransitions{Float64, ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}}(900.0, 2100.0, 999.8, 4186.0, 334000.0, 0.0, ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}(0.0, 0.054))

We set the top ice temperature,

top_temperature = -10 # ᵒC
top_heat_boundary_condition = PrescribedTemperature(-10)
PrescribedTemperature{Int64}(-10)

Construct the thermodynamics of sea ice, for this we use a simple slab sea ice representation of thermodynamics

ice_thermodynamics = SlabSeaIceThermodynamics(grid;
                                              internal_heat_flux,
                                              phase_transitions,
                                              top_heat_boundary_condition)
SlabSeaIceThermodynamics
├── top_surface_temperature: ConstantField(-10)
└── minimium_ice_thickness: ConstantField(0.0)

Then we assemble it all into a model,

model = SeaIceModel(grid; ice_thermodynamics)
SeaIceModel{Oceananigans.Architectures.CPU, Oceananigans.Grids.RectilinearGrid}(time = 0 seconds, iteration = 0)
├── grid: 1×1×1 RectilinearGrid{Float64, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat, Oceananigans.Grids.Flat} on Oceananigans.Architectures.CPU with 0×0×0 halo
├── ice_thermodynamics: SlabThermodynamics
├── advection: Nothing
└── external_heat_fluxes: 
    ├── top: FluxFunction of slab_internal_heat_flux with parameters (flux=ConductiveFlux{Float64}, liquidus=ClimaSeaIce.SeaIceThermodynamics.LinearLiquidus{Float64}, bottom_heat_boundary_condition=ClimaSeaIce.SeaIceThermodynamics.HeatBoundaryConditions.IceWaterThermalEquilibrium{Int64})
    └── bottom: 0

Note that the default bottom heat boundary condition for SlabSeaIceThermodynamics is IceWaterThermalEquilibrium with freshwater. That's what we want!

model.ice_thermodynamics.heat_boundary_conditions.bottom
ClimaSeaIce.SeaIceThermodynamics.HeatBoundaryConditions.IceWaterThermalEquilibrium{Int64}(0)

Ok, we're ready to freeze the bucket for 10 straight days with an initial ice thickness of 1 cm,

simulation = Simulation(model, Δt=10minute, stop_time=10days)

set!(model, h=0.01)

Collecting data and running the simulation

Before simulating the freezing bucket, we set up a Callback to create a timeseries of the ice thickness saved at every time step.

# Container to hold the data
timeseries = []

# Callback function to collect the data from the `sim`ulation
function accumulate_timeseries(sim)
    h = sim.model.ice_thickness
    push!(timeseries, (time(sim), first(h)))
end

# Add the callback to `simulation`
simulation.callbacks[:save] = Callback(accumulate_timeseries)
Callback of accumulate_timeseries on IterationInterval(1)

Now we're ready to hit the Big Red Button (it should run pretty quick):

run!(simulation)
[ Info: Initializing simulation...
[ Info:     ... simulation initialization complete (1.590 seconds)
[ Info: Executing initial time step...
[ Info:     ... initial time step complete (1.570 seconds).
[ Info: Simulation is stopping after running for 3.617 seconds.
[ Info: Simulation time 10 days equals or exceeds stop time 10 days.

Visualizing the result

It'd be a shame to run such a "cool" simulation without looking at the results. We'll visualize it with Makie.

using CairoMakie

timeseries is a Vector of Tuple. So we have to do a bit of processing to build Vectors of time t and thickness h. It's not much work though:

t = [datum[1] for datum in timeseries]
h = [datum[2] for datum in timeseries]
1441-element Vector{Float64}:
 0.01
 0.013115723787646819
 0.015618315847984355
 0.01777126422864224
 0.019689999447000853
 0.02143760767315016
 0.023053076484693194
 0.024562507006759427
 0.02598439207428873
 0.0273324015606709
 ⋮
 0.3209657543453097
 0.32107769484037885
 0.32118959632212435
 0.32130145883130834
 0.3214132824086221
 0.3215250670946861
 0.32163681293005025
 0.321748519955194
 0.3218601882105266

Just for fun, we also compute the velocity of the ice-water interface:

dhdt = @. (h[2:end] - h[1:end-1]) / simulation.Δt
1440-element Vector{Float64}:
 5.192872979411364e-6
 4.170986767229228e-6
 3.588247301096473e-6
 3.1978920305976893e-6
 2.9126803769155148e-6
 2.6924480192383877e-6
 2.5157175367770553e-6
 2.3698084458821715e-6
 2.246682477303619e-6
 2.1409626325971827e-6
 ⋮
 1.8663258204348886e-7
 1.8656749178188234e-7
 1.8650246957583697e-7
 1.8643751530665145e-7
 1.8637262885627206e-7
 1.863078101066451e-7
 1.8624305894027193e-7
 1.8617837523956146e-7
 1.8611375888766267e-7

All that's left, really, is to put those lines! in an Axis:

set_theme!(Theme(fontsize=24, linewidth=4))

fig = Figure(size=(1200, 600))

axh = Axis(fig[1, 1], xlabel="Time (days)", ylabel="Ice thickness (cm)")
axd = Axis(fig[1, 2], xlabel="Ice thickness (cm)", ylabel="Freezing rate (μm s⁻¹)")

lines!(axh, t ./ day, 1e2 .* h)
lines!(axd, 1e2 .* h[1:end-1], 1e6 .* dhdt)

fig
Example block output

If you want more ice, you can increase simulation.stop_time and run!(simulation) again (or just re-run the whole script).


This page was generated using Literate.jl.