|
| 1 | +using Test |
| 2 | +using AGNI |
| 3 | + |
| 4 | +# Access the guillot submodule through setpt |
| 5 | +const guillot = AGNI.setpt.guillot |
| 6 | + |
| 7 | +# Guillot (2010) analytical grey-gas T(p) profile module. |
| 8 | +# Constants inside the module: |
| 9 | +# κ_th = 7e-2 * 10 = 0.7 m2/kg (LW opacity) |
| 10 | +# κ_vs = 4e-3 * 10 = 0.04 m2/kg (SW opacity) |
| 11 | +# γ = κ_vs/κ_th = 0.04/0.7 (opacity ratio) |
| 12 | + |
| 13 | +@testset "guillot" begin |
| 14 | + |
| 15 | + # ------------- |
| 16 | + # Optical depth: τ = p * κ_th / g |
| 17 | + # ------------- |
| 18 | + @testset "eval_tau" begin |
| 19 | + p_test = [1e3, 1e5, 1e7 ] # pressure [Pa] |
| 20 | + g_test = [10.0, 10.0, 25.0 ] # gravity [m s-2] |
| 21 | + v_expt = [70.0, 7000.0, 280000.0] # τ = p * 0.7 / g [dimensionless] |
| 22 | + for i in eachindex(p_test) |
| 23 | + @test isapprox(guillot.eval_tau(p_test[i], g_test[i]), v_expt[i]; rtol=1e-10) |
| 24 | + end |
| 25 | + # τ must be linear in pressure at fixed gravity |
| 26 | + @test isapprox(guillot.eval_tau(2e5, 10.0), 2 * guillot.eval_tau(1e5, 10.0); rtol=1e-10) |
| 27 | + end |
| 28 | + |
| 29 | + # ------------- |
| 30 | + # Stellar temperatures: geometric relations |
| 31 | + # eval_Tirr(Tstar, Rstar, sep) = Tstar * sqrt(Rstar / sep) |
| 32 | + # eval_Teqm(Tstar, Rstar, sep) = Tstar * sqrt(Rstar / (2*sep)) |
| 33 | + # So Tirr / Teqm == sqrt(2) exactly. |
| 34 | + # ------------- |
| 35 | + @testset "stellar_temps" begin |
| 36 | + Tstar = 5778.0 # solar effective temperature [K] |
| 37 | + Rstar = 6.957e8 # solar radius [m] |
| 38 | + sep = 1.496e11 # Earth-Sun distance [m] |
| 39 | + |
| 40 | + Tirr = guillot.eval_Tirr(Tstar, Rstar, sep) |
| 41 | + Teqm = guillot.eval_Teqm(Tstar, Rstar, sep) |
| 42 | + |
| 43 | + @test Tirr > 0.0 |
| 44 | + @test Teqm > 0.0 |
| 45 | + @test Teqm < Tirr |
| 46 | + # exact geometric relationship between the two temperatures |
| 47 | + @test isapprox(Tirr / Teqm, sqrt(2.0); rtol=1e-10) |
| 48 | + end |
| 49 | + |
| 50 | + # ------------- |
| 51 | + # T^4 functions must return positive values for physical inputs |
| 52 | + # Note: eval_T4_avg uses E2(γτ) which diverges at τ=0 (τ>0 only). |
| 53 | + # eval_T4_cos does not use E1/E2 so τ=0 is fine. |
| 54 | + # ------------- |
| 55 | + @testset "T4_positive" begin |
| 56 | + τ_vals_avg = [0.01, 0.1, 1.0, 10.0, 100.0] # τ > 0 for avg (E1 singularity at 0) |
| 57 | + τ_vals_cos = [0.0, 0.1, 1.0, 10.0, 100.0] # τ = 0 is fine for cos |
| 58 | + Tint = 100.0 # internal temperature [K] |
| 59 | + Teqm = 800.0 # equilibrium temperature [K] |
| 60 | + Tirr = 1000.0 # irradiation temperature [K] |
| 61 | + θ = 45.0 # zenith angle [degrees] |
| 62 | + for τ in τ_vals_avg |
| 63 | + @test guillot.eval_T4_avg(τ, Tint, Teqm) > 0.0 |
| 64 | + end |
| 65 | + for τ in τ_vals_cos |
| 66 | + @test guillot.eval_T4_cos(τ, Tint, Tirr, θ) > 0.0 |
| 67 | + end |
| 68 | + end |
| 69 | + |
| 70 | + # ------------- |
| 71 | + # T^4 must increase monotonically with optical depth (deeper = hotter) |
| 72 | + # for the average profile at sensible parameters |
| 73 | + # ------------- |
| 74 | + @testset "T4_monotone" begin |
| 75 | + τ_lo, τ_hi = 0.1, 10.0 |
| 76 | + Tint, Teqm, Tirr, θ = 200.0, 1200.0, 1500.0, 30.0 |
| 77 | + @test guillot.eval_T4_avg(τ_hi, Tint, Teqm) > guillot.eval_T4_avg(τ_lo, Tint, Teqm) |
| 78 | + @test guillot.eval_T4_cos(τ_hi, Tint, Tirr, θ) > guillot.eval_T4_cos(τ_lo, Tint, Tirr, θ) |
| 79 | + end |
| 80 | + |
| 81 | + # ------------- |
| 82 | + # Profile helpers: correct length and all-positive temperatures |
| 83 | + # ------------- |
| 84 | + @testset "profiles" begin |
| 85 | + grav = 10.0 # [m s-2] |
| 86 | + Tint = 100.0 # [K] |
| 87 | + Teqm = 900.0 # [K] |
| 88 | + Tirr = 1200.0 # [K] |
| 89 | + θ = 45.0 # [degrees] |
| 90 | + p_arr = [1e1, 1e2, 1e3, 1e4, 1e5, 1e6] # pressure grid [Pa] |
| 91 | + |
| 92 | + T_avg = guillot.calc_profile_avg(p_arr, grav, Tint, Teqm) |
| 93 | + @test length(T_avg) == length(p_arr) |
| 94 | + @test all(T_avg .> 0.0) |
| 95 | + @test issorted(T_avg) # temperature increases towards higher pressures |
| 96 | + |
| 97 | + T_cos = guillot.calc_profile_cos(p_arr, grav, Tint, Tirr, θ) |
| 98 | + @test length(T_cos) == length(p_arr) |
| 99 | + @test all(T_cos .> 0.0) |
| 100 | + @test issorted(T_cos) |
| 101 | + end |
| 102 | + |
| 103 | +end |
0 commit comments