-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDailyInso.py
151 lines (122 loc) · 5.21 KB
/
DailyInso.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
# -*- coding: utf-8 -*-
"""
Compute daily insolation
Created on Thu Jun 17 15:20:13 2021
@Author: Didier Paillard. Adapted by Olivier Marti
"""
import numpy as np
## Astronomical parameters
ECC = 0.0167024 # Excentricity 0kBP (1950 CE)
OBL = 23.4393 # Obliquity 0kBP (degrees)
PRE = 102.918 # Climatic precession (degrees)
EQUINOX = 80. # Day of March equinox
## Deliberate difference with Berger et al. (2010), who consider the SIDERAL_YEAR
YEAR_SIDERAL = 365.25636
YEAR_TROPICAL = 365.24219876
YEAR_365 = 365.0
YEAR_366 = 366.0
YEAR_LENGTH = YEAR_365
SOLAR = 1365.0 # Solar constant (W/m^2)
mth_length = np.array ( [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] )
mth_start = np.array ( [ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334] )
mth_end = np.array ( [31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365] )
# Internal parameters for solving the Kepler equation
NITER = 7 ; PREC = 1.0E-7
# Astronomical functions
def daily_inso_longitude (lat, lon, ecc=ECC, obl=OBL, pre=PRE, solar=SOLAR) :
'''
Daily insolation for a given latitude and for true solar longitude
All angles in degrees
Input :
lat : latitude
lon : true longitude of the Sun (orbital position)
'''
lat_r = deg2rad (lat)
lon_r = deg2rad (lon)
obl_r = deg2rad (obl)
sin_lat = np.sin (lat_r)
sin_lon = np.sin (lon_r)
sin_obl = np.sin (obl_r)
p = sin_lat*sin_lon*sin_obl
s = np.maximum (0., 1. - sin_lat*sin_lat - sin_lon*sin_lon*sin_obl*sin_obl)
d = np.sqrt (s+p*p)
# d>0 in almost all cases. If d=0 then p = s = 0 et g = 0
g = np.where (d>0.0, p * np.arccos (-p/np.maximum (d, 1.E-99)) + np.sqrt (s), 0.0)
v_r = lon_r - np.pi - deg2rad (pre) # True anomaly in radian (angle from perihelion)
ar = (1.0+ecc*np.cos(v_r))/(1.0-ecc*ecc) # (a/r) = inverse of the distance to Sun (=focus) in astronomical units U.A.
ar2 = ar*ar
dayly_inso = solar * ar2 * g/np.pi
if isinstance (dayly_inso, xr.core.dataarray.DataArray) :
declinaison.attrs.update ({'units':'W.m-2', 'standard_name':'daily_insolation', 'long_name':'Daily insolation',} )
return dayly_inso
def solve_Kepler_iter (m, ecc=ECC, niter=NITER) :
'''
Solve Kepler equation : E - e sinE = m
Use a fixed number of iterations
'''
E = m
for i in range (niter) : # For a weak eccentricity (like for the Earth, <0.06) a few iterations are enough (error < 5.10e-11 for 7 iterations; <10e-14 for 10 iterations)
E = m + ecc * np.sin (E)
return E
def solve_Kepler_prec (m, ecc=ECC, prec=PREC) :
'''
Solve Kepler equation : E - e sinE = m
Iterate until error < prec
'''
E = m
error = 1.0
while error > prec :
E = m + ecc * np.sin (E)
error = np.max ( np.abs (m - (E - ecc*np.sin(E))) )
return E
def solve_Kepler (m, ecc=ECC, niter=None, prec=None) :
'''
Solve Kepler equation : E - e sinE = m
input :
m
ecc : ecccentricity. Default=ECC
niter : number of iterations (default value: NITER)
prec : absolute precision
niter and prec are mutually exclusive. If none is given, niter=NITER is used
'''
if niter != None and prec != None :
print ( 'Error in solve_Kepler' )
print ( 'Only one of the two parameters niter and prec is allowed' )
return None
else :
if niter == None and prec == None :
niter = NITER
if niter == None :
solver_Kepler = solve_Kepler_prec (m, ecc=ECC, prec=prec )
if prec == None :
solver_Kepler = solve_Kepler_iter (m, ecc=ECC, niter=niter)
return solver_Kepler
def daily_inso_time (lat, t, ecc=ECC, obl=OBL, pre=PRE, solar=SOLAR):
'''
Daily insolation for a given latitute as the fraction of the year t, counted from the March equinox
All angles in degrees
Input :
lat : latitude
t : time in fraction of year (in [0,1])
'''
t_r = t*2.0*np.pi # t in [0, 1] (t_r in [0, 2pi]). For instance t = (day-80)/365 for a 365 day calendar, with equinox at day "80"
v0_r = -np.pi - deg2rad (pre) # true anomaly for longitude 0 (March equinox)
a = np.sqrt ((1.0-ecc)/(1.0+ecc))
E0 = 2.*np.arctan (a*np.tan(v0_r/2)) # eccentric anomaly for lon=0
t0 = E0 - ecc*np.sin (E0) # mean anomaly (time from perihelion) for lon=0
E = solve_Kepler (t_r + t0) # eccentric anomaly at time t
v_r = 2.*np.arctan (np.tan(E/2.0)/a) # true anomaly
lon_r = v_r + np.pi + deg2rad (pre) # true longitude
lon = rad2deg (lon_r)
daily_inso_time = daily_inso_longitude (lat, lon, ecc=ecc, obl=obl, pre=pre, solar=solar)
if isinstance (daily_inso_time, xr.core.dataarray.DataArray) :
declinaison.attrs.update ({'units':'W.m-2', 'standard_name':'daily_insolation', 'long_name':'Daily insolation',} )
return daily_inso_time
def simple_calendar (day, equinox=EQUINOX, year_length=YEAR_LENGTH) :
'''
365 day calendar, with equinox at day "80"
Input
day : day of year (in [0,YEAR])
'''
simple_calendar = (day-equinox)/year_length
return simple_calendar