Skip to content

Commit

Permalink
Refactor + add density GM prototype
Browse files Browse the repository at this point in the history
  • Loading branch information
briochemc committed Oct 17, 2024
1 parent cbd2945 commit 1cada5b
Show file tree
Hide file tree
Showing 11 changed files with 390 additions and 314 deletions.
5 changes: 3 additions & 2 deletions Project.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
name = "OceanTransportMatrixBuilder"
uuid = "c2b4a04e-6049-4fc4-aa6a-5508a29a1e1c"
authors = ["Benoit Pasquier <[email protected]> and contributors"]
version = "0.2.8"
version = "0.3.0"

[deps]
Distances = "b4f34e82-e78d-54a5-968a-f98e89d6e8f7"
Expand All @@ -18,6 +18,7 @@ julia = "1.10"
CSV = "336ed68f-0bac-5ca0-87d4-7b16caf5d00b"
DataFrames = "a93c6f00-e57d-5684-b7b6-d8193f3e46c0"
DimensionalData = "0703355e-b756-11e9-17c0-8b28908087d0"
Downloads = "f43a241f-c20a-4ad4-852c-f6b1247861c6"
GLMakie = "e9467ef8-e4e7-5192-8a1a-b1aee30e663a"
GibbsSeaWater = "9a22fb26-0b63-4589-b28e-8f9d0b5c3d05"
LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e"
Expand All @@ -32,4 +33,4 @@ YAXArrays = "c21b50f5-aa40-41ea-b809-c0f5e47bfa5c"
Zarr = "0a941bbe-ad1d-11e8-39d9-ab76183a1d99"

[targets]
test = ["DimensionalData", "GibbsSeaWater", "GLMakie", "LinearAlgebra", "Makie", "NaNStatistics", "NetCDF", "Test", "TestItemRunner", "TestItems", "Unitful", "YAXArrays", "CSV", "DataFrames", "Zarr"]
test = ["Downloads", "DimensionalData", "GibbsSeaWater", "GLMakie", "LinearAlgebra", "Makie", "NaNStatistics", "NetCDF", "Test", "TestItemRunner", "TestItems", "Unitful", "YAXArrays", "CSV", "DataFrames", "Zarr"]
4 changes: 2 additions & 2 deletions src/OceanTransportMatrixBuilder.jl
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,8 @@ using Distances: haversine

include("preprocessing.jl")
include("matrixbuilding.jl")
include("grids.jl")
include("topology.jl")
include("gridcellgeometry.jl")
include("gridtopology.jl")
include("derivatives.jl")
include("extratools.jl") # <- I think this should be in a separate "base" repo

Expand Down
55 changes: 31 additions & 24 deletions src/derivatives.jl
Original file line number Diff line number Diff line change
Expand Up @@ -12,11 +12,11 @@ Returns the discrete difference
# forward derivative in the i direction for centered data (A-grid)
function ∂ᵢ₊(χ, modelgrid)
χ = χ |> Array
(; lon, lat, gridtype) = modelgrid
(; lon, lat, arakawagrid) = modelgrid
∂ᵢ₊χ = zeros(size(χ))
for I in CartesianIndices(lon)
P = (lon[I], lat[I])
Iᵢ₊₁ = i₊₁(I, gridtype)
Iᵢ₊₁ = i₊₁(I, arakawagrid)
Pᵢ₊₁ = (lon[Iᵢ₊₁], lat[Iᵢ₊₁])
d = haversine(P, Pᵢ₊₁)
∂ᵢ₊χ[I, :] = (χ[Iᵢ₊₁, :] - χ[I, :]) / d
Expand All @@ -26,11 +26,11 @@ end
# backward derivative in the i direction for centered data (A-grid)
function ∂ᵢ₋(χ, modelgrid)
χ = χ |> Array
(; lon, lat, gridtype) = modelgrid
(; lon, lat, arakawagrid) = modelgrid
∂ᵢ₋χ = zeros(size(χ))
for I in CartesianIndices(lon)
P = (lon[I], lat[I])
Iᵢ₋₁ = i₋₁(I, gridtype)
Iᵢ₋₁ = i₋₁(I, arakawagrid)
Pᵢ₋₁ = (lon[Iᵢ₋₁], lat[Iᵢ₋₁])
d = haversine(P, Pᵢ₋₁)
∂ᵢ₋χ[I, :] = (χ[I, :] - χ[Iᵢ₋₁, :]) / d
Expand All @@ -40,11 +40,11 @@ end
# forward derivative in the j direction
function ∂ⱼ₊(χ, modelgrid)
χ = χ |> Array
(; lon, lat, gridtype) = modelgrid
(; lon, lat, arakawagrid) = modelgrid
∂ⱼ₊χ = zeros(size(χ))
for I in CartesianIndices(lon)
P = (lon[I], lat[I])
Iᵢ₊₁ = j₊₁(I, gridtype)
Iᵢ₊₁ = j₊₁(I, arakawagrid)
Pᵢ₊₁ = (lon[Iᵢ₊₁], lat[Iᵢ₊₁])
d = haversine(P, Pᵢ₊₁)
∂ⱼ₊χ[I, :] = (χ[Iᵢ₊₁, :] - χ[I, :]) / d
Expand All @@ -54,11 +54,11 @@ end
# backward derivative in the j direction
function ∂ⱼ₋(χ, modelgrid)
χ = χ |> Array
(; lon, lat, gridtype) = modelgrid
(; lon, lat, arakawagrid) = modelgrid
∂ⱼ₋χ = zeros(size(χ))
for I in CartesianIndices(lon)
P = (lon[I], lat[I])
Iᵢ₋₁ = j₋₁(I, gridtype)
Iᵢ₋₁ = j₋₁(I, arakawagrid)
Pᵢ₋₁ = (lon[Iᵢ₋₁], lat[Iᵢ₋₁])
d = haversine(P, Pᵢ₋₁)
∂ⱼ₋χ[I, :] = (χ[I, :] - χ[Iᵢ₋₁, :]) / d
Expand All @@ -68,75 +68,82 @@ end
# derivative in the k direction
function ∂ₖ₊(χ, modelgrid)
χ = χ |> Array
(; zt, DZT3d, gridtype) = modelgrid
(; zt, thkcello, arakawagrid) = modelgrid
∂ₖ₊χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₊₁ = k₊₁(I, gridtype)
Iᵢ₊₁ = k₊₁(I, arakawagrid)
isnothing(Iᵢ₊₁) && continue
h = (DZT3d[I] + DZT3d[Iᵢ₊₁]) / 2
h = (thkcello[I] + thkcello[Iᵢ₊₁]) / 2
∂ₖ₊χ[I] = (χ[Iᵢ₊₁] - χ[I]) / h
end
return ∂ₖ₊χ
end
# backward derivative in the k direction
function ∂ₖ₋(χ, modelgrid)
χ = χ |> Array
(; zt, DZT3d, gridtype) = modelgrid
(; zt, thkcello, arakawagrid) = modelgrid
∂ₖ₋χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₋₁ = k₋₁(I, gridtype)
Iᵢ₋₁ = k₋₁(I, arakawagrid)
isnothing(Iᵢ₋₁) && continue
h = (DZT3d[I] + DZT3d[Iᵢ₋₁]) / 2
h = (thkcello[I] + thkcello[Iᵢ₋₁]) / 2
∂ₖ₋χ[I] = (χ[I] - χ[Iᵢ₋₁]) / h
end
return ∂ₖ₋χ
end

# interpolation in the i direction
function itpₖ₊(χ, modelgrid)
(; gridtype) = modelgrid
(; arakawagrid) = modelgrid
χ = χ |> Array
itpₖ₊χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₊₁ = k₊₁(I, gridtype)
Iᵢ₊₁ = k₊₁(I, arakawagrid)
isnothing(Iᵢ₊₁) && continue
itpₖ₊χ[I] = (χ[Iᵢ₊₁] + χ[I]) / 2
end
return itpₖ₊χ
end
function itpₖ₋(χ, modelgrid)
(; gridtype) = modelgrid
(; arakawagrid) = modelgrid
χ = χ |> Array
itpₖ₋χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₋₁ = k₋₁(I, gridtype)
Iᵢ₋₁ = k₋₁(I, arakawagrid)
isnothing(Iᵢ₋₁) && continue
itpₖ₋χ[I] = (χ[Iᵢ₋₁] + χ[I]) / 2
end
return itpₖ₋χ
end
function itpᵢ₊(χ, modelgrid)
(; gridtype) = modelgrid
(; arakawagrid) = modelgrid
χ = χ |> Array
itpᵢ₊χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₊₁ = i₊₁(I, gridtype)
Iᵢ₊₁ = i₊₁(I, arakawagrid)
itpᵢ₊χ[I] = (χ[Iᵢ₊₁] + χ[I]) / 2
end
return itpᵢ₊χ
end
function itpⱼ₊(χ, modelgrid)
(; gridtype) = modelgrid
(; arakawagrid) = modelgrid
χ = χ |> Array
itpⱼ₊χ = zeros(size(χ))
for I in CartesianIndices(χ)
Iᵢ₊₁ = j₊₁(I, gridtype)
Iᵢ₊₁ = j₊₁(I, arakawagrid)
itpⱼ₊χ[I] = (χ[Iᵢ₊₁] + χ[I]) / 2
end
return itpⱼ₊χ
end

"""
bolus_GM_velocity(σ, modelgrid; κGM = 600, maxslope = 0.01)
Returns the bolus velocity field due to the Gent-McWilliams parameterization,
computed from the neutral density field `σ` (or potential density, ρθ in kg/m³).
Note: This is experimental at this stage.
"""
function bolus_GM_velocity(σ, modelgrid; κGM = 600, maxslope = 0.01)
# σ is neutral density (or potential density, ρθ in kg/m³)
σ = replace(σ, missing => 0, NaN => 0)
Expand Down Expand Up @@ -193,13 +200,13 @@ end


# function ∂ᵢ(χ, modelgrid, shift=0, d=:f)
# (;lon, lat, gridtype) = modelgrid
# (;lon, lat, arakawagrid) = modelgrid
# m = 1
# ∂ᵢχ = zeros(size(χ))
# # Draw a line along coordinate i and compute the distances
# d = dᵢ₊(lon, lat)
# for I in CartesianIndices(lon)
# movingI = SVector(ishift(I, gridtype, xxx))
# movingI = SVector(ishift(I, arakawagrid, xxx))
# i, j = I.I
# x = SVector(0, d[i])
# w = stencil(x, 0, χ)
Expand Down
103 changes: 54 additions & 49 deletions src/grids.jl → src/gridcellgeometry.jl
Original file line number Diff line number Diff line change
@@ -1,19 +1,25 @@


abstract type ArakawaGrid end
abstract type ArakawaGridCell end

struct AGrid <: ArakawaGrid
struct AGridCell <: ArakawaGridCell
u_pos::Symbol
v_pos::Symbol
end

struct BGrid <: ArakawaGrid
struct BGridCell <: ArakawaGridCell
u_pos::Symbol
v_pos::Symbol
end

struct CGrid <: ArakawaGrid
struct CGridCell <: ArakawaGridCell
u_pos::Symbol
v_pos::Symbol
end


"""
arakawa, uo_pos, vo_pos, uo_relerr, vo_relerr = gridtype(uo_lon, uo_lat, vo_lon, vo_lat, modelgrid)
arakawa = getarakawagrid(u_lon, u_lat, v_lon, v_lat, modelgrid)
Returns the type of the grid (A, B, or C), the grid position of the velocity points,
and the error of that position relative to the perimeter of the cell.
Expand Down Expand Up @@ -43,14 +49,14 @@ The different Arakawa grids recongnized here are:
where each grid is centered in C.
Also returns the distances from the velocity points to the grid points.
"""
function gridtype(uo_lon, uo_lat, vo_lon, vo_lat, modelgrid)
function getarakawagrid(u_lon, u_lat, v_lon, v_lat, modelgrid)

# Unpack modelgrid
(; lon, lat, lon_vertices, lat_vertices) = modelgrid

i = j = 1
uo_point = (uo_lon[i, j], uo_lat[i, j])
vo_point = (vo_lon[i, j], vo_lat[i, j])
u_point = (u_lon[i, j], u_lat[i, j])
v_point = (v_lon[i, j], v_lat[i, j])

C = (lon[i, j], lat[i, j])
SW = (lon_vertices[1, i, j], lat_vertices[1, i, j])
Expand All @@ -64,75 +70,74 @@ function gridtype(uo_lon, uo_lat, vo_lon, vo_lat, modelgrid)

cell = (; C, SW, SE, NE, NW, S, N, W, E)

uo_distances = (; (k => haversine(P, uo_point) for (k,P) in pairs(cell))...)
vo_distances = (; (k => haversine(P, vo_point) for (k,P) in pairs(cell))...)
u_distances = (; (k => haversine(P, u_point) for (k,P) in pairs(cell))...)
v_distances = (; (k => haversine(P, v_point) for (k,P) in pairs(cell))...)

uo_distance, uo_pos = findmin(uo_distances)
vo_distance, vo_pos = findmin(vo_distances)
u_distance, u_pos = findmin(u_distances)
v_distance, v_pos = findmin(v_distances)

# Arakawa grid type
if uo_pos == vo_pos == :C
arakawa = AGrid()
elseif uo_pos == vo_pos && uo_pos (:NE, :NW, :SE, :SW)
arakawa = BGrid()
elseif uo_pos (:E, :W) && vo_pos (:N, :S)
arakawa = CGrid()
if u_pos == v_pos == :C
arakawa = AGridCell(u_pos, v_pos)
elseif u_pos == v_pos && u_pos (:NE, :NW, :SE, :SW)
arakawa = BGridCell(u_pos, v_pos)
elseif u_pos (:E, :W) && v_pos (:N, :S)
arakawa = CGridCell(u_pos, v_pos)
else
error("Unknown Arakawa grid type")
end

cellperimeter = haversine(SW, SE) + haversine(SE, NE) + haversine(NE, NW) + haversine(NW, SW)
uo_relerr = uo_distance / cellperimeter
vo_relerr = vo_distance / cellperimeter
relerr = (u_distance + v_distance) / cellperimeter

return arakawa, uo_pos, vo_pos, uo_relerr, vo_relerr
relerr > 0.01 && warn("Relative error in grid positions in $arakawa is $relerr")

return arakawa

end

"""
uo, uo_lon, uo_lat, vo, vo_lon, vo_lat = interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid)
u, u_lon, u_lat, v, v_lon, v_lat = interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid)
Interpolates the velocity fields `uo` and `vo` from B- or C-grid
Interpolates the velocity fields `u` and `v` from B- or C-grid
onto the default C-grid (centered on the cell faces).
"""
interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid) = interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid, gridtype(uo_lon, uo_lat, vo_lon, vo_lat, modelgrid))
interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid, ::CGrid) = uo, uo_lon, uo_lat, vo, vo_lon, vo_lat
interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid, ::AGrid) = error("Interpolation not implemented for this grid type")
# TODO this is clumsy to have to pass the gridtype but then recompute it
function interpolateontodefaultCgrid(uo, uo_lon, uo_lat, vo, vo_lon, vo_lat, modelgrid, arakawa::BGrid)

arakawa, uo_pos, vo_pos, uo_relerr, vo_relerr = gridtype(uo_lon, uo_lat, vo_lon, vo_lat, modelgrid)
interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid) = interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid, getarakawagrid(u_lon, u_lat, v_lon, v_lat, modelgrid))
interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid, ::CGridCell) = u, u_lon, u_lat, v, v_lon, v_lat
interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid, ::AGridCell) = error("Interpolation not implemented for A-grid type")
function interpolateontodefaultCgrid(u, u_lon, u_lat, v, v_lon, v_lat, modelgrid, arakawa::BGridCell)

uo_pos == vo_pos == :NE || error("Interpolation not implemented for this B-grid type")
(; u_pos, v_pos) = arakawa
u_pos == v_pos == :NE || error("Interpolation not implemented for this B-grid($u_pos,$v_pos) type")

_FillValue = uo.properties["_FillValue"]
_FillValue = u.properties["_FillValue"]

# Make sure uo and vo are in memory
uo = uo |> Array
vo = vo |> Array
# Make sure u and v are in memory
u = u |> Array
v = v |> Array

nx, ny, nz = size(uo)
nx, ny, nz = size(u)

# unpack modelgrid
(; lon_vertices, lat_vertices) = modelgrid

# It seems that uo/vo is NaN on boundaries (for ACCESS-ESM1-5)
# and that umo/vmo were computed as if uo/vo were 0 on boundaries
# It seems that u/v is NaN on boundaries (for ACCESS-ESM1-5)
# and that umo/vmo were computed as if u/v were 0 on boundaries
# so that's what we do here
uo2 = replace(uo, _FillValue => 0.0)
vo2 = replace(vo, _FillValue => 0.0)
uo2 = 0.5(uo2 + [fill(0.0, nx, 1, nz);; uo2[:, 1:end-1, :]])
vo2 = 0.5(vo2 + [fill(0.0, 1, ny, nz); vo2[1:end-1, :, :]])
u2 = replace(u, _FillValue => 0.0)
v2 = replace(v, _FillValue => 0.0)
u2 = 0.5(u2 + [fill(0.0, nx, 1, nz);; u2[:, 1:end-1, :]])
v2 = 0.5(v2 + [fill(0.0, 1, ny, nz); v2[1:end-1, :, :]])
SE_points = [(lon, lat) for (lon, lat) in zip(lon_vertices[2, :, :], lat_vertices[2, :, :])]
NE_points = [(lon, lat) for (lon, lat) in zip(lon_vertices[3, :, :], lat_vertices[3, :, :])]
NW_points = [(lon, lat) for (lon, lat) in zip(lon_vertices[4, :, :], lat_vertices[4, :, :])]
uo2_points = [midpointonsphere(SE, NE) for (SE, NE) in zip(NE_points, SE_points)]
vo2_points = [midpointonsphere(NE, NW) for (NE, NW) in zip(NW_points, NE_points)]
uo2_lon = [P[1] for P in uo2_points]
uo2_lat = [P[2] for P in uo2_points]
vo2_lon = [P[1] for P in vo2_points]
vo2_lat = [P[2] for P in vo2_points]
return uo2, uo2_lon, uo2_lat, vo2, vo2_lon, vo2_lat
u2_points = [midpointonsphere(SE, NE) for (SE, NE) in zip(NE_points, SE_points)]
v2_points = [midpointonsphere(NE, NW) for (NE, NW) in zip(NW_points, NE_points)]
u2_lon = [P[1] for P in u2_points]
u2_lat = [P[2] for P in u2_points]
v2_lon = [P[1] for P in v2_points]
v2_lat = [P[2] for P in v2_points]
return u2, u2_lon, u2_lat, v2, v2_lon, v2_lat

end

Expand Down
Loading

2 comments on commit 1cada5b

@briochemc
Copy link
Member Author

Choose a reason for hiding this comment

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

@JuliaRegistrator register

Release notes:

  • Add some prototype code for derivatives and GM velocities (WIP)
  • Refactor some of the codebase (also WIP)
  • Add variable density feature for advective transport

@JuliaRegistrator
Copy link

Choose a reason for hiding this comment

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

Registration pull request created: JuliaRegistries/General/117539

Tagging

After the above pull request is merged, it is recommended that a tag is created on this repository for the registered package version.

This will be done automatically if the Julia TagBot GitHub Action is installed, or can be done manually through the github interface, or via:

git tag -a v0.3.0 -m "<description of version>" 1cada5b2c38e075f08d6310eb6aa8248e20c1c7c
git push origin v0.3.0

Please sign in to comment.