Skip to content

Commit

Permalink
Merge pull request #71 from FourierFlows/EnhanceForcedBetaplane
Browse files Browse the repository at this point in the history
Simplifies examples + adds binder/nbviewer intro
  • Loading branch information
navidcy authored May 5, 2020
2 parents 832a559 + 12ad5e2 commit 505f89f
Show file tree
Hide file tree
Showing 12 changed files with 125 additions and 67 deletions.
2 changes: 1 addition & 1 deletion docs/Project.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,4 +9,4 @@ Printf = "de0858da-6303-5e67-8744-51eddeeeb8d7"
PyPlot = "d330b81b-6aea-500a-939a-2ce795aea3ee"

[compat]
FFTW = "≥ 0.4.1"
FourierFlows = "≥ 0.4.1"
2 changes: 1 addition & 1 deletion docs/make.jl
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ end
Timer(t -> println(" "), 0, interval=240)

format = Documenter.HTML(
collapselevel = 1,
collapselevel = 2,
prettyurls = get(ENV, "CI", nothing) == "true",
canonical = "https://fourierflows.github.io/GeophysicalFlows.jl/dev/"
)
Expand Down
13 changes: 8 additions & 5 deletions examples/README.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,20 @@
# GeophysicalFlows.jl/examples


These are some basic examples of the various modules included in GeophysicalFlows.jl.
These are some basic examples of the various modules included in GeophysicalFlows.jl. The best way to go through the examples is via the <a href="https://fourierflows.github.io/GeophysicalFlows.jl/dev/">documentation <img src="https://img.shields.io/badge/docs-dev-blue.svg"></a>.

## Usage

Run
## Run the examples

You can run the examples in an executable environment via [Binder](https://mybinder.org) by clicking on the [![badge](https://img.shields.io/badge/binder-badge-579ACA.svg?logo=)](https://mybinder.org)
at the top of each example page in the documentation.

Alternatively, you can run these scripts directly. Run
```julia
] instantiate
```
to install dependencies. Then you can run any of the examples by
to install dependencies. Then run any of the examples by
```julia
include("example_script.jl")
```


13 changes: 7 additions & 6 deletions examples/barotropicqg_acc.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqgtopography.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqgtopography.ipynb)

# # Barotropic QG beta-plane turbulence over topography
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqgtopography.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqgtopography.ipynb).
#
# An idealized version of the Southern Ocean. We solve the barotropic
# quasi-geostrophic eddy dynamics in a flud with variable depth $H-h(x,y)$. We
# also include an "Antarctic Circumpolar Current," i.e., a domain-average zonal
Expand Down Expand Up @@ -76,11 +76,11 @@ BarotropicQG.set_zeta!(prob, 0*x)

# ## Diagnostics

# Create Diagnostics -- `energy` and `enstrophy` functions are imported at the top.
# Create Diagnostics -- `energy`, `meanenergy`, `enstrophy`, and `meanenstrophy` functions are imported at the top.
E = Diagnostic(energy, prob; nsteps=nsteps)
Q = Diagnostic(enstrophy, prob; nsteps=nsteps)
Emean = Diagnostic(meanenergy, prob; nsteps=nsteps)
Qmean = Diagnostic(meanenergy, prob; nsteps=nsteps)
Qmean = Diagnostic(meanenstrophy, prob; nsteps=nsteps)
diags = [E, Emean, Q, Qmean]
nothing # hide

Expand Down Expand Up @@ -173,6 +173,7 @@ fig, axs = subplots(ncols=3, nrows=1, figsize=(15, 4), dpi=200)
plot_output(prob, fig, axs; drawcolorbar=true)
gcf() # hide

# and finally save the figure
# Note that since mean flow enstrophy is $Q_U = \beta U$ it can attain negative values.
# Finally we save the figure.
savename = @sprintf("%s_%09d.png", joinpath(plotpath, plotname), cl.step)
savefig(savename)
24 changes: 10 additions & 14 deletions examples/barotropicqg_betadecay.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqg_betadecay.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqg_betadecay.ipynb)

# # Decaying barotropic QG beta-plane turbulence
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqg_betadecay.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqg_betadecay.ipynb).
#
# An example of decaying barotropic quasi-geostrophic turbulence on a beta plane.

using FourierFlows, PyPlot, Printf, Random
Expand Down Expand Up @@ -37,7 +37,7 @@ Lx = 2π # domain size
= 1 # viscosity order
β = 15.0 # planetary PV gradient
μ = 0.0 # bottom drag

nothing # hide

# ## Problem setup
# We initialize a `Problem` by providing a set of keyword arguments,
Expand All @@ -57,16 +57,12 @@ nothing # hide

Random.seed!(1234)
E0 = 0.1
modk = ones(g.nkr, g.nl)
modk[real.(g.Krsq).<(8*2*pi/g.Lx)^2] .= 0
modk[real.(g.Krsq).>(10*2*pi/g.Lx)^2] .= 0
modk[1, :] .= 0
psih = (randn(Float64, size(sol)) .+ im*randn(Float64, size(sol))).*modk
psih = @. psih*prob.timestepper.filter
Ein = real(sum(g.Krsq.*abs2.(psih)/(g.nx*g.ny)^2))
psih = psih*sqrt(E0/Ein)
qi = -irfft(g.Krsq.*psih, g.nx)
E0 = FourierFlows.parsevalsum(g.Krsq.*abs2.(psih), g)
qih = randn(Complex{Float64}, size(sol))
qih[ g.Krsq .< (8*2π /g.Lx)^2] .= 0
qih[ g.Krsq .> (10*2π/g.Lx)^2] .= 0
Ein = energy(qih, g) # compute energy of qi
qih = qih*sqrt(E0/Ein) # normalize qi to have energy E0
qi = irfft(qih, g.nx)

BarotropicQG.set_zeta!(prob, qi)
nothing #hide
Expand Down
30 changes: 22 additions & 8 deletions examples/barotropicqg_betaforced.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqg_betaforced.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqg_betaforced.ipynb)

# # Forced-dissipative barotropic QG beta-plane turbulence
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqg_betaforced.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqg_betaforced.ipynb).
#
# A simulation of forced-dissipative barotropic quasi-geostrophic turbulence on
# a beta plane. The dynamics include linear drag and stochastic excitation.

Expand Down Expand Up @@ -90,6 +90,20 @@ sol, cl, v, p, g = prob.sol, prob.clock, prob.vars, prob.params, prob.grid
nothing # hide


# First let's see how a forcing realization looks like.
calcFq!(v.Fqh, sol, 0.0, cl, v, p, g)

fig = figure(figsize=(3, 2), dpi=150)
pcolormesh(x, y, irfft(v.Fqh, g.nx))
axis("square")
xticks(-3:1:3)
yticks(-3:1:3)
title("a forcing realization")
colorbar()
clim(-8, 8)
gcf() # hide


# ## Setting initial conditions

# Our initial condition is simply fluid at rest.
Expand All @@ -98,7 +112,7 @@ BarotropicQG.set_zeta!(prob, 0*x)

# ## Diagnostics

# Create Diagnostic -- "energy" and "enstrophy" are functions imported at the top.
# Create Diagnostic -- `energy` and `enstrophy` are functions imported at the top.
E = Diagnostic(energy, prob; nsteps=nsteps)
Z = Diagnostic(enstrophy, prob; nsteps=nsteps)
diags = [E, Z] # A list of Diagnostics types passed to "stepforward!" will be updated every timestep.
Expand Down Expand Up @@ -139,8 +153,8 @@ function plot_output(prob, fig, axs; drawcolorbar=false)
cla()
pcolormesh(x, y, v.q)
axis("square")
xticks(-2:2:2)
yticks(-2:2:2)
xticks(-3:1:3)
yticks(-3:1:3)
title(L"vorticity $\zeta = \partial_x v - \partial_y u$")
if drawcolorbar==true
colorbar()
Expand All @@ -153,8 +167,8 @@ function plot_output(prob, fig, axs; drawcolorbar=false)
contour(x, y, v.psi, colors="k")
end
axis("square")
xticks(-2:2:2)
yticks(-2:2:2)
xticks(-3:1:3)
yticks(-3:1:3)
title(L"streamfunction $\psi$")
if drawcolorbar==true
colorbar()
Expand Down
20 changes: 17 additions & 3 deletions examples/barotropicqgql_betaforced.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqgql_betaforced.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqgql_betaforced.ipynb)

# # Quasi-Linear forced-dissipative barotropic QG beta-plane turbulence
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/barotropicqgql_betaforced.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/barotropicqgql_betaforced.ipynb).
#
# A simulation of forced-dissipative barotropic quasi-geostrophic turbulence on
# a beta plane under the *quasi-linear approximation*. The dynamics include
# linear drag and stochastic excitation.
Expand Down Expand Up @@ -84,6 +84,20 @@ sol, cl, v, p, g = prob.sol, prob.clock, prob.vars, prob.params, prob.grid
nothing # hide


# First let's see how a forcing realization looks like.
calcF!(v.Fh, sol, 0.0, cl, v, p, g)

fig = figure(figsize=(3, 2), dpi=150)
pcolormesh(x, y, irfft(v.Fh, g.nx))
axis("square")
xticks(-3:1:3)
yticks(-3:1:3)
title("a forcing realization")
colorbar()
clim(-8, 8)
gcf() # hide


# ## Setting initial conditions

# Our initial condition is simply fluid at rest.
Expand Down
10 changes: 5 additions & 5 deletions examples/multilayerqg_2layer.jl
Original file line number Diff line number Diff line change
@@ -1,16 +1,16 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/multilayerqg_2layer.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/multilayerqg_2layer.ipynb)

# # Phillips model of Baroclinic Instability
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/multilayerqg_2layer.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/multilayerqg_2layer.ipynb).
#
# A simulation of the growth of barolinic instability in the Phillips 2-layer model
# when we impose a vertical mean flow shear as a difference $\Delta U$ in the
# imposed, domain-averaged, zonal flow at each layer.

using FourierFlows, PyPlot, Printf

import GeophysicalFlows.MultilayerQG
import GeophysicalFlows.MultilayerQG: energies, fluxes
import GeophysicalFlows.MultilayerQG: energies


# ## Numerical parameters and time-stepping parameters
Expand Down Expand Up @@ -61,7 +61,7 @@ nothing # hide

# ## Diagnostics

# Create Diagnostics -- `energy` function is imported at the top.
# Create Diagnostics -- `energies` function is imported at the top.
E = Diagnostic(energies, prob; nsteps=nsteps)
diags = [E] # A list of Diagnostics types passed to "stepforward!" will be updated every timestep.
nothing # hide
Expand Down
6 changes: 3 additions & 3 deletions examples/twodnavierstokes_decaying.jl
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/twodnavierstokes_decaying.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/twodnavierstokes_decaying.ipynb)

# # 2D decaying turbulence
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/twodnavierstokes_decaying.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/twodnavierstokes_decaying.ipynb).
#
# A simulations of decaying two-dimensional turbulence.

using FourierFlows, PyPlot, Printf, Random
Expand Down
29 changes: 23 additions & 6 deletions examples/twodnavierstokes_stochasticforcing.jl
Original file line number Diff line number Diff line change
@@ -1,14 +1,15 @@
#md # [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/twodnavierstokes_stochasticforcing.ipynb)
#md # [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/twodnavierstokes_stochasticforcing.ipynb)

# # 2D forced-dissipative turbulence
#
#md # This example can be run online via [![](https://mybinder.org/badge_logo.svg)](@__BINDER_ROOT_URL__/generated/twodnavierstokes_stochasticforcing.ipynb).
#md # Also, it can be viewed as a Jupyter notebook via [![](https://img.shields.io/badge/show-nbviewer-579ACA.svg)](@__NBVIEWER_ROOT_URL__/generated/twodnavierstokes_stochasticforcing.ipynb).
#
# A simulation of forced-dissipative two-dimensional turbulence. We solve the
# two-dimensional vorticity equation with linear drag and stochastic excitation.

using PyPlot, FourierFlows, Printf

using Random: seed!
using FFTW: irfft

import GeophysicalFlows.TwoDNavierStokes
import GeophysicalFlows.TwoDNavierStokes: energy, enstrophy, dissipation, work, drag
Expand Down Expand Up @@ -81,6 +82,20 @@ sol, cl, v, p, g = prob.sol, prob.clock, prob.vars, prob.params, prob.grid
nothing # hide


# First let's see how a forcing realization looks like.
calcF!(v.Fh, sol, 0.0, cl, v, p, g)

fig = figure(figsize=(3, 2), dpi=150)
pcolormesh(x, y, irfft(v.Fh, g.nx))
axis("square")
xticks(-3:1:3)
yticks(-3:1:3)
title("a forcing realization")
colorbar()
clim(-200, 200)
gcf() # hide


# ## Setting initial conditions

# Our initial condition is simply fluid at rest.
Expand All @@ -89,7 +104,7 @@ TwoDNavierStokes.set_zeta!(prob, 0*x)

# ## Diagnostics

# Create Diagnostics; the diagnostics here will probe the energy budget.
# Create Diagnostics; the diagnostics are aimed to probe the energy budget.
E = Diagnostic(energy, prob, nsteps=nt) # energy
R = Diagnostic(drag, prob, nsteps=nt) # dissipation by drag
D = Diagnostic(dissipation, prob, nsteps=nt) # dissipation by hyperviscosity
Expand All @@ -111,10 +126,12 @@ function makeplot(prob, diags)
t = round*cl.t, digits=2)
sca(axs[1]); cla()
pcolormesh(x, y, v.zeta)
axis("square")
xticks(-3:1:3)
yticks(-3:1:3)
xlabel(L"$x$")
ylabel(L"$y$")
title("\$\\nabla^2\\psi(x,y,\\mu t= $t )\$")
axis("square")
title("\$\\nabla^2\\psi(x,y,\\mu t= $t)\$")

sca(axs[3]); cla()

Expand Down
23 changes: 16 additions & 7 deletions src/barotropicqg.jl
Original file line number Diff line number Diff line change
Expand Up @@ -316,31 +316,40 @@ set_U!(prob, U::Float64) = set_U!(prob.sol, prob.vars, prob.params, prob.grid, U


"""
Calculate the domain-averaged kinetic energy.
energy(sol, g)
energy(prob)
Returns the domain-averaged kinetic energy of sol.
"""
function energy(prob)
sol, g = prob.sol, prob.grid
function energy(sol, g::AbstractGrid)
0.5*(parsevalsum2(g.kr.*g.invKrsq.*sol, g)
+ parsevalsum2(g.l.*g.invKrsq.*sol, g))/(g.Lx*g.Ly)
end

energy(prob) = energy(prob.sol, prob.grid)

"""
Returns the domain-averaged enstrophy.
enstrophy(sol, g, v)
enstrophy(prob)
Returns the domain-averaged enstrophy of sol.
"""
function enstrophy(prob)
sol, v, g = prob.sol, prob.vars, prob.grid
function enstrophy(sol, g::AbstractGrid, v::AbstractVars)
@. v.uh = sol
v.uh[1, 1] = 0
0.5*parsevalsum2(v.uh, g)/(g.Lx*g.Ly)
end
enstrophy(prob) = enstrophy(prob.sol, prob.grid, prob.vars)

"""
meanenergy(prob)
Returns the energy of the domain-averaged U.
"""
meanenergy(prob) = real(0.5*prob.sol[1, 1].^2)

"""
meanenstrophy(prob)
Returns the enstrophy of the domain-averaged U.
"""
meanenstrophy(prob) = real(prob.params.β*prob.sol[1, 1])
Expand Down
Loading

0 comments on commit 505f89f

Please sign in to comment.