Skip to content

Commit 4171188

Browse files
Added NumSpinUpDays for averarge soil flux computations
1 parent 784db2f commit 4171188

File tree

132 files changed

+14892
-21836
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

132 files changed

+14892
-21836
lines changed

src/lisflood/hydrological_modules/soil.py

+35-24
Original file line numberDiff line numberDiff line change
@@ -18,20 +18,23 @@
1818
from __future__ import absolute_import, print_function
1919

2020
import numpy as np
21+
import warnings
22+
import xarray as xr
23+
import numpy as np
24+
import scipy
2125

2226
from ..global_modules.settings import MaskInfo, LisSettings
2327
from ..global_modules.add1 import loadmap, defsoil
2428
from . import HydroModule
29+
from ..global_modules.errors import LisfloodWarning, LisfloodError
2530
from collections import OrderedDict
2631
#from global_modules.add1 import *
27-
import xarray as xr
28-
import numpy as np ##### 2024
29-
import scipy ##### 2024
30-
from scipy.optimize import least_squares ##### 2024
3132

32-
def function_estimate(satdegrthetastart,*data): ##### 2024
33-
SeepTopToSubBPixelAv, KSat2,GenuInvM2, GenuM2 = data
34-
return SeepTopToSubBPixelAv - KSat2 * np.sqrt(satdegrthetastart) * (1. - (1. - satdegrthetastart ** GenuInvM2) ** GenuM2) ** 2
33+
from scipy.optimize import least_squares
34+
35+
def function_estimate(satdegrthetastart,*data):
36+
SeepTopToSubBAv, KSat2,GenuInvM2, GenuM2 = data
37+
return SeepTopToSubBAv - KSat2 * np.sqrt(satdegrthetastart) * (1. - (1. - satdegrthetastart ** GenuInvM2) ** GenuM2) ** 2
3538

3639
def pressure2SoilMoistureFun(residual_sm, sat_sm, GenuA, GenuN, GenuM):
3740
"""Generate a function to compute soil moisture values corresponding to characteristic pressure head levels [cm].
@@ -78,8 +81,8 @@ def initial(self):
7881
""" initial part of the soil module
7982
"""
8083
maskinfo = MaskInfo.instance()
81-
self.var.cumSeepTopToSubBAv = self.var.allocateVariableAllVegetation() #######2024
82-
self.var.SeepTopToSubBAv = self.var.allocateVariableAllVegetation() #######2024
84+
self.var.cumSeepTopToSubBAv = self.var.allocateVariableAllVegetation()
85+
self.var.SeepTopToSubBAv = self.var.allocateVariableAllVegetation()
8386

8487
def splitlanduse(array1, array2=None, array3=None):
8588
""" splits maps into the 3 different land use types - other , forest, irrigation
@@ -273,7 +276,7 @@ def splitlanduse(array1, array2=None, array3=None):
273276
# Set to zero if soil depth is zero.
274277
# IMPORTANT: WInit1 and WInit2 represent the soil moisture in the *permeable* fraction of the pixel *only*
275278
# (since all soil moisture-related calculations are done for permeable fraction only!).
276-
if not option['InitLisflood']:
279+
if not option['InitLisflood'] and not option['WarmStart']:
277280
self.var.SeepTopToSubBAv[0] = loadmap('SeepTopToSubBAverageOtherMap')
278281
self.var.SeepTopToSubBAv[1] = loadmap('SeepTopToSubBAverageForestMap')
279282
self.var.SeepTopToSubBAv[2] = loadmap('SeepTopToSubBAverageIrrigationMap')
@@ -290,19 +293,20 @@ def splitlanduse(array1, array2=None, array3=None):
290293
self.var.W2[iveg] = np.where(self.var.PoreSpaceNotZero2[iluse], ini_2, 0)
291294

292295

293-
if not option['InitLisflood'] and not option['WarmStart']: ## 2024
294-
295-
check_var_coldstart = np.min(self.var.SeepTopToSubBAv[0]) ######################### TO BE CORRECTED!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
296-
297-
if check_var_coldstart < 0.0:
298-
warnings.warn(LisfloodWarning('WARNING: soil moisture end states OR average fluxes not provided/erroneous. Soil moisture states are initialized at field capacity'))
296+
if not option['InitLisflood'] and not option['WarmStart']:
297+
check_prerun_results1 = np.min(ThetaInit2Value[iveg])
298+
check_prerun_results2 = np.min(self.var.SeepTopToSubBAv[iluse])
299+
check_prerun_results = np.min([check_prerun_results1,check_prerun_results2])
300+
301+
if check_prerun_results < 0.0:
302+
warnings.warn(LisfloodWarning('WARNING: soil moisture end state for bottom layer OR average fluxes not provided/erroneous. Soil moisture states are initialized at field capacity'))
299303
else:
300-
SOLVED = ((ThetaInit2Value[iveg] * self.var.SoilDepth2[iluse])-self.var.WRes2[iluse])/(self.var.WS2[iluse]-self.var.WRes2[iluse]) #### for extra safety - stef
304+
SOLVED = ((ThetaInit2Value[iveg] * self.var.SoilDepth2[iluse])-self.var.WRes2[iluse])/(self.var.WS2[iluse]-self.var.WRes2[iluse])
301305
for ii in np.arange(len(ThetaInit2Value[iveg])):
302306
data = []
303307
analyticalcheckzero = []
304308
data=(self.var.SeepTopToSubBAv[iluse][ii], self.var.KSat2[iluse][ii], self.var.GenuInvM2[iluse][ii], self.var.GenuM2[iluse][ii])
305-
analyticalcheckzero = least_squares(function_estimate,((ThetaInit2Value[iveg][ii] * self.var.SoilDepth2[iluse][ii])-self.var.WRes2[iluse][ii])/(self.var.WS2[iluse][ii]-self.var.WRes2[iluse][ii]),bounds=(self.var.WFC2[iluse][ii]*0,self.var.WFC2[iluse][ii]*0+1.0), args=data)
309+
analyticalcheckzero = least_squares(function_estimate,((ThetaInit2Value[iveg][ii] * self.var.SoilDepth2[iluse][ii])-self.var.WRes2[iluse][ii])/(self.var.WS2[iluse][ii]-self.var.WRes2[iluse][ii]),bounds=(self.var.WFC2[iluse][ii]*0,self.var.WFC2[iluse][ii]*0+1.0), gtol = 10e-12, args = data)
306310
SOLVED[ii]=analyticalcheckzero.x
307311
ini_2 = SOLVED*(self.var.WS2[iluse]-self.var.WRes2[iluse])+self.var.WRes2[iluse]
308312
self.var.W2[iveg] = np.where(self.var.PoreSpaceNotZero2[iluse], ini_2, 0)
@@ -504,6 +508,10 @@ def dynamic_perpixel(self):
504508
""" dynamic part of the soil module
505509
Calculation per Pixel
506510
"""
511+
settings = LisSettings.instance()
512+
option = settings.options
513+
binding = settings.binding
514+
507515
self.var.TaInterceptionAll = self.var.deffraction(self.var.TaInterception) + self.var.DirectRunoffFraction * self.var.TASealed
508516
self.var.TaInterceptionCUM += self.var.TaInterceptionAll
509517
self.var.TaInterceptionWB = self.var.TaInterceptionAll
@@ -540,12 +548,15 @@ def dynamic_perpixel(self):
540548
# Pixel-average seepage values in [mm] per timestep
541549
# (no seepage from direct runoff fraction)
542550

543-
self.var.cumSeepTopToSubBAv[0] += self.var.SeepTopToSubB[0] #### 2024
544-
self.var.SeepTopToSubBAv[0] = (self.var.cumSeepTopToSubBAv[0] * self.var.InvDtDay) / (self.var.TimeSinceStart) #### 2024
545-
self.var.cumSeepTopToSubBAv[1] += self.var.SeepTopToSubB[1] #### 2024
546-
self.var.SeepTopToSubBAv[1] = (self.var.cumSeepTopToSubBAv[1] * self.var.InvDtDay) / (self.var.TimeSinceStart) #### 2024
547-
self.var.cumSeepTopToSubBAv[2] += self.var.SeepTopToSubB[2] #### 2024
548-
self.var.SeepTopToSubBAv[2] = (self.var.cumSeepTopToSubBAv[2] * self.var.InvDtDay) / (self.var.TimeSinceStart) #### 2024
551+
if option['InitLisflood']:
552+
NumDaysSpinUp = float(binding['NumDaysSpinUp'])
553+
if (self.var.TimeSinceStart > np.round(NumDaysSpinUp/self.var.DtDay)) :
554+
self.var.cumSeepTopToSubBAv[0] += self.var.SeepTopToSubB[0]
555+
self.var.SeepTopToSubBAv[0] = (self.var.cumSeepTopToSubBAv[0] * self.var.InvDtDay) / (self.var.TimeSinceStart - np.round(NumDaysSpinUp/self.var.DtDay))
556+
self.var.cumSeepTopToSubBAv[1] += self.var.SeepTopToSubB[1]
557+
self.var.SeepTopToSubBAv[1] = (self.var.cumSeepTopToSubBAv[1] * self.var.InvDtDay) / (self.var.TimeSinceStart - np.round(NumDaysSpinUp/self.var.DtDay))
558+
self.var.cumSeepTopToSubBAv[2] += self.var.SeepTopToSubB[2]
559+
self.var.SeepTopToSubBAv[2] = (self.var.cumSeepTopToSubBAv[2] * self.var.InvDtDay) / (self.var.TimeSinceStart - np.round(NumDaysSpinUp/self.var.DtDay))
549560

550561

551562
# the variables below were added to report catchment-averaged soil moisture profiles

src/lisflood/hydrological_modules/soilloop.py

+6-5
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,7 @@ def soilColumnsWaterBalance(index_landuse_all, is_irrigated, is_paddy_irrig, pad
116116
sim_pixels = np.arange(num_pixs)
117117
landuse = index_landuse_all[veg]
118118
NoSubS = np.empty(sim_pixels.size, dtype=np.int_)
119-
DtSub = np.empty(sim_pixels.size) ### merge 3 stef
119+
DtSub = np.empty(sim_pixels.size)
120120
KUnSat1a, KUnSat1b, KUnSat2 = np.empty(sim_pixels.size), np.empty(sim_pixels.size), np.empty(sim_pixels.size) #| REMOVE BEFORE NEW
121121
AvailableWater1a, AvailableWater1b, AvailableWater2 = np.empty(sim_pixels.size), np.empty(sim_pixels.size), np.empty(sim_pixels.size) #| CALIBRATION
122122
CapacityLayer1, CapacityLayer2 = np.empty(sim_pixels.size), np.empty(sim_pixels.size) #| (SEE [*] BELOW) Stef WHY??
@@ -275,8 +275,8 @@ def soilColumnsWaterBalance(index_landuse_all, is_irrigated, is_paddy_irrig, pad
275275
SeepSubToGW[veg,pix] = 0.
276276
# Initialize fluxes out of subsoil (accumulated value for all sub-steps)
277277
# Start iterating
278-
DtSub[q] = DtDay / NoSubS[q] ### merge 3 (NoSubSteps)
279-
for i in range(NoSubS[q]): ### merge 3 (NoSubSteps)
278+
DtSub[q] = DtDay / NoSubS[q]
279+
for i in range(NoSubS[q]):
280280
if i > 0:
281281
KUnSat1a[q] = unsaturatedConductivity(WTemp1a, PoreSpaceNotZero1a[landuse,pix], WRes1a[landuse,pix], WS1a[landuse,pix],
282282
KSat1a[landuse,pix], GenuInvM1a[landuse,pix], GenuM1a[landuse,pix])
@@ -319,10 +319,11 @@ def soilColumnsWaterBalance(index_landuse_all, is_irrigated, is_paddy_irrig, pad
319319
W1a[veg,pix] -= SeepTopToSubA[veg,pix]
320320
W1b[veg,pix] = W1b[veg,pix] + SeepTopToSubA[veg,pix] - SeepTopToSubB[veg,pix]
321321
W2[veg,pix] = W2[veg,pix] + SeepTopToSubB[veg,pix] - SeepSubToGW[veg,pix]
322-
W1[veg,pix] = W1a[veg,pix] + W1b[veg,pix] # SHOULD THIS BE MOVED AFTER W1a ADJUSTMENT BELOW?
322+
323323
# Update soil moisture amounts in top- and sub soil
324-
Infiltration[veg,pix] -= max(W1a[veg,pix] - WS1a[landuse,pix], 0.)
325324
W1a[veg,pix] = min(W1a[veg,pix], WS1a[landuse,pix])
325+
Infiltration[veg,pix] -= max(W1a[veg,pix] - WS1a[landuse,pix], 0.)
326+
W1[veg,pix] = W1a[veg,pix] + W1b[veg,pix]
326327
# Compute the amount of water that could not infiltrate and add this water to the surface runoff
327328
# Remove the excess of water in the top layer
328329
# Calculate: volumetric soil moisture contents of top- and sub soil [V/V]; and saturation with respect to the WP and FC values

src/lisfloodSettings_reference.xml

+12
Original file line numberDiff line numberDiff line change
@@ -299,6 +299,12 @@ Number of last time step in simulation
299299
</comment>
300300
</textvar>
301301

302+
<textvar name="NumDaysSpinUp" value="1095">
303+
<comment>
304+
Recommended value > 730 (2 years)
305+
</comment>
306+
</textvar>
307+
302308
<textvar name="ReportSteps" value="1..999999">
303309
<comment>
304310
1..9999
@@ -1773,6 +1779,12 @@ Number of last time step in simulation
17731779
</comment>
17741780
</textvar>
17751781

1782+
<textvar name="NumDaysSpinUp" value="$(NumDaysSpinUp)">
1783+
<comment>
1784+
Number of days used for internal spin-up (fluxes computations during prerun)
1785+
</comment>
1786+
</textvar>
1787+
17761788
</group>
17771789

17781790
<group>
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.

0 commit comments

Comments
 (0)