Source code for soil

# -*- coding: utf-8 -*-

# =============================================================================
# This file is part of WaterGAP.

# WaterGAP is an opensource software which computes water flows and storages as
# well as water withdrawals and consumptive uses on all continents.

# You should have received a copy of the LGPLv3 License along with WaterGAP.
# if not see <https://www.gnu.org/licenses/lgpl-3.0>
# =============================================================================

"""Soil Storage."""

# =============================================================================
# This module computes soil water balance, including soil storage and related
# fluxes for all grid cells based on section 4.4 of
# (Müller Schmied et al. (2021)). After, interception and snow, runoff from
# urban areas or immediate runoff (R1) is computed.
# Effective precipitation (computes as throufall - snowfall + snowmelt) is
# reduced by the immediate runoff. After, runoff from soil water overflow (R2)
# is computed.Then, daily runoff (R3), Bergström (1995), is calculated by using
# soil saturation, effective precipitation and runoff coefficient.
# Actual evapotranspiration is calculated and the soil storage is
# updated. Afterwards, ground water recharge is calculated based on the
# soil runoff (R3). Then, total daily runoff from land (RL) is computed as
# ( daily runoff (R3) + immediate runoff (R1) + soil water overflow (R2) )
# (see Corrigendum equation 18a-c of Müller Schmied et al. (2021)).

# For runoff from soil water overflow (R2),  we check if previous and current
# storage exceed maximum soil water and update runoff from soil water overflow
# accordingly.

# Note: Total daily runoff (RL) is corrected with areal correction factor (CFA)
# (if gamma is not sufficeint to fit simulated discharge). To conserve water
# balance , evapotranspiration is also corrected with CFA.

# After evapotranspiration correction, soil storage, total daily runoff and
#  groundwater recharge are adjusted as well. Finally Surface runoff is then
# calculated as total daily runoff minus ground water recharge.
# =============================================================================

import numpy as np
from numba import njit

# @njit is a decorator from numba to optimised the python code for speed.


@njit(cache=True)
def immediate_runoff(effective_precipitation, runoff_frac_builtup,
                     builtup_area_frac):
    """
    Compute immediate runoff and effective precipitation.

    Parameters
    ----------
    effective_precipitation : float
        Effective precipitation based on [1]_, Units: [mm/day]
    runoff_frac_builtup:float
        Fraction of effective_precipitation that directly becomes runoff
        (specifically in urban areas) based on [1]_, Units: [-]
    builtup_area_frac: float
        Built up (Urban) area fraction, Units: [-]

    Returns
    -------
    effective_precipitation : array
        Effective precipitation based on [1]_, Units: [mm/day]
    immediate_runoff : array
        Immediate runoff, Units: [mm/day]

    References.

    .. [1] Müller Schmied, H., Cáceres, D., Eisner, S., Flörke, M.,
                Herbert, C., Niemann, C., Peiris, T. A., Popat, E.,
                Portmann, F. T., Reinecke, R., Schumacher, M., Shadkam, S.,
                Telteu, C.E., Trautmann, T., & Döll, P. (2021).
                The global water resources and use model WaterGAP v2.2d:
                model description and evaluation. Geoscientific Model
                Development, 14(2), 1037–1079.
                https://doi.org/10.5194/gmd-14-1037-2021

    """
    # Runoff from urban areas or immediate runoff (R1). see eq.18b in H.
    # Müller Schmied et al 2021(Corrigendum)

    # Note !!! runoff_frac_builtup = 0.5, Which is the fraction of
    # effective_precipitation that directly becomes runoff (specifically
    # in urban areas)
    if builtup_area_frac > 0:
        immediate_runoff = runoff_frac_builtup * effective_precipitation * \
            builtup_area_frac

        # Reducing effective precipitation by immediate runoff since 50% is
        # left.
        effective_precipitation -= immediate_runoff
    else:
        immediate_runoff = 0

    return effective_precipitation, immediate_runoff


[docs] @njit(cache=True) def soil_water_balance(soil_water_content, pet_to_soil, current_landarea_frac, landareafrac_ratio, max_temp_elev, canopy_evap, effective_precipitation, precipitation, immediate_runoff, land_storage_change_sum, sublimation, daily_storage_transfer, snow_freeze_temp, gamma, max_daily_pet, humid_arid, soil_texture, drainage_direction, max_groundwater_recharge, groundwater_recharge_factor, critcal_gw_precipitation, max_soil_water_content, areal_corr_factor, minstorage_volume, x, y): """ Compute daily soil balance. Parameters ---------- soil_water_content : float Soil water content, Units: [mm] pet_to_soil : float Remaining energy for addtional evaporation from soil, Units: [mm] current_landarea_frac : float Land area fraction of current time step, Units: [-] landareafrac_ratio : float Ratio of land area fraction of previous to current time step, Units: [-] max_temp_elev : float Maximum temperature from the 1st(lowest) elevation from snow algorithm. , Units: [K] canopy_evap : float Canopy evaporation, Units: [mm/day] effective_precipitation : float Effective precipitation based on Müller Schmied et al 2021, Units: [mm/day] precipitation : float Daily precipitation, Units: [mm/day] immediate_runoff : float Immediate runoff, Units: [mm/day] land_storage_change_sum : float Sum of change in vertical balance storages, Units: [mm] sublimation : float Sublimation, Units: [mm/day] daily_storage_transfer : float Storage to be transfered to runoff when land area fraction of current time step is zero, Units: [mm] snow_freeze_temp : float Snow freeze temperature , Units: [K] gamma : float Runoff coefficient , Units: [-] max_daily_pet : float Maximum daily potential evapotranspiration, Units: [mm/day] humid_arid : float Humid-arid calssification based on Müller Schmied et al. 2021 soil_texture : float Soil texture classification based on Müller Schmied et al. 2021 drainage_direction : float Drainage direction based on Müller Schmied et al. 2021 max_groundwater_recharge : float Maximum groundwater recharge from soil, Units: [mm/day] groundwater_recharge_factor : float Groundwater recharge factor, Units: [-] critcal_gw_precipitation : float critical precipitation for groundwater recharge, Units: [mm/day] max_soil_water_content : float Maximum soil water content , Units: [mm] areal_corr_factor : float Areal correction factor-CFA for correcting runoff, Units: [-] minstorage_volume : float Volume at which storage is set to zero, Units: [km3] x, y : Latititude and longitude indexes of grid cells. Returns ------- soil_water_content : float Updated soil water content , Units: [mm] groundwater_recharge_from_soil_mm : float Groundwater recharge from soil, Units: [mm/day] actual_soil_evap : float Actual evapotranspiration from the soil, Units: [mm/day] soil_saturation : float Soil saturation, Units: [-] surface_runoff : float Surface runoff from land, Units: [mm/day] daily_storage_transfer : float Updated storage to be transfered to runoff when land area fraction of current time step is zero, Units: [mm] total_daily_runoff : float Total daily runoff from land (RL) (runoff + immediate runoff + soil water overflow), Units: [mm/day] daily_runoff : float daily runoff (R3), Bergström (1995), Units: [mm/day] soil_water_overflow : float Soil water overflow (R2), Units: [mm/day] immediate_runoff : float runoff from urban areas or immediate runoff (R1), Units: [mm/day] """ # Index (x, y) to print out varibales of interest # e.g. if x==65 and y==137: print(current_landarea_frac) if current_landarea_frac > 0: # ===================================================================== # Calculating soil water overflow (R2) (mm) and soil water content(mm). # ===================================================================== # Adapting soil_water_content since land area fraction is dynamic soil_water_content *= landareafrac_ratio # Initial storage to calulate change in soil storage. initial_storage = soil_water_content # Soil_water_overflow (R2): see eq.18c in H. Müller Schmied et al 2021 # (Corrigendum). Soil_water_overflow will be used to calulate # Total daily runoff (runoff + immediate runoff + soil water overflow) soil_water_overflow = 0 if soil_water_content > max_soil_water_content: soil_water_overflow = soil_water_content - max_soil_water_content soil_water_content = max_soil_water_content if max_temp_elev > snow_freeze_temp: if max_soil_water_content > 0: # ============================================================= # Calculating daily soil runoff (R3) (mm/day). # ============================================================= # Runoff (R3): see eq.18d in H. Müller Schmied et al 2021 # (Corrigendum) soil_saturation = soil_water_content / max_soil_water_content # gamma = Runoff coefficient (-) daily_runoff = effective_precipitation * \ (soil_saturation)**gamma # ============================================================= # Calculating actual soil evapotranspiration (mm/day) and # updating soil water storage (mm). # ============================================================= # Equation for actual soil evapotranspiration taken from # Eq. 17 in H. Müller Schmied et al 2021. # max_daily_pet = maximum daily potential evapotransipration # (mm/day) actual_soil_evap = \ np.minimum(pet_to_soil, (max_daily_pet - canopy_evap) * soil_saturation) # Updating soil water content # (eq.15 in H. Müller Schmied et al 2021) soil_water_content += (effective_precipitation - actual_soil_evap - daily_runoff) # minimal storage volume =1e15 (smaller volumes set to zero) # to counter numerical inaccuracies if np.abs(soil_water_content) <= minstorage_volume: soil_water_content = 0 # ============================================================= # I dont know why effective prctipiatation is set to zero after # addition to the soil water content if you won't use it for # further calculations. This was done in the old code (c++). # I just wrote it here for safety (will removethis soon) # # effective_precipitation = 0 # ============================================================= # Note !!! when too much water has been taken out of the # soil water storage (negative soil storage), soil water # content is set to zero. Daily actual soil evaporation is # now corrected by reducing with the soil water content # (negative soil water content) if soil_water_content < 0: actual_soil_evap += soil_water_content soil_water_content = 0 # ============================================================= # Correcting runoff and immediate runoff with areal_corr_factor # Areal correction factor-CFA (-) # ============================================================= daily_runoff *= areal_corr_factor immediate_runoff *= areal_corr_factor # ============================================================= # Calculating ground water recharge from soil(mm/day), # potential recharge (mm/day) # Order of calculation is important!!! # ============================================================= # Note: the 'mm' at the end of # groundwater_recharge_from_soil_mm means it is a vertical # water balance variable.This distinguish it from # ground water recharge in the lateral water balance # Groundwater recharge equation is calulated based on Eq. 19 # from H. Müller Schmied et al 2021 if (humid_arid == 1) & (soil_texture < 21) & \ (drainage_direction >= 0): groundwater_recharge_from_soil_mm = \ np.minimum(max_groundwater_recharge, groundwater_recharge_factor * daily_runoff) # For (semi) arid cells groundwater recharge becomes zero # when critical precipitation for groundwater recharge # (default = 12.5 mm/day) is not exceeded. if precipitation <= critcal_gw_precipitation: potential_gw_recharge = \ groundwater_recharge_from_soil_mm groundwater_recharge_from_soil_mm = 0 else: potential_gw_recharge = 0 groundwater_recharge_from_soil_mm = \ np.minimum(max_groundwater_recharge, groundwater_recharge_factor * daily_runoff) # ============================================================= # Updating runoff and soil water content with remaning water in # the soil. # ============================================================= # As potential recharge is water that remains in the soil for # (seimi)arid under the prior stated conditions , # it is subtracted from runoff and added to storage. # Remove double counting of CFA as recharge is computed from # corrected daily runoff. daily_runoff -= potential_gw_recharge soil_water_content += (potential_gw_recharge / areal_corr_factor) # Updating soil_water_overflow (R2) and soil water content. # if the updated soil_water_content > maximum, the excess # becomes overflow. This is to prevent the storage of the # current time step to exceed maximum soil storage. if soil_water_content > max_soil_water_content: soil_water_overflow += soil_water_content - \ max_soil_water_content soil_water_content = max_soil_water_content # correct runoff R2 with areal correction factor soil_water_overflow *= areal_corr_factor # Total daily runoff (RL) is calculated as runoff + # immediate runoff + updated soil water overflow # See eq. 18a in H. Müller Schmied et al 2021 (Corrigendum) total_daily_runoff = daily_runoff + immediate_runoff + \ soil_water_overflow else: total_daily_runoff = 0 groundwater_recharge_from_soil_mm = 0 else: soil_water_overflow *= areal_corr_factor total_daily_runoff = soil_water_overflow groundwater_recharge_from_soil_mm = 0 actual_soil_evap = 0 # computing change in soil_water_content soil_water_content_change = soil_water_content - initial_storage # Sum of change in vertical balance storages (canopy, snow, soil) land_storage_change_sum += soil_water_content_change # ===================================================================== # Calulating corrected actual total evaporation for land (mm/day) # ===================================================================== # Actual total evaporation from land is corrected such that when areal # correction factor is increased or reduced , actual total evaporation # will also be reduced or increased respectively. This is then # consistent with cell-corrected runoff. # ---------------------------------------------------------------- # R(L) = P - AET - ds (dt=1) eqn. 1 # R(L) * CFA = P - AET_corr - ds (dt=1) eqn. 2 # eqn2 into egn 1 # P - AET_corr - ds = CFA ( P - AET - ds) # AET_corr = ds(CFA-1) - P(CFA-1) + AET(CFA) # /////////////////////////////////////////////////////////// # R(L): runoff from land (mm/day), P: precipitation (mm/day) # AET: actual total evaporation for land (mm/day) # ds: change in soil moisture storage (mm/day) # CFA: areal correction factor # ///////////////////////////////////////////////////////////// corr_land_aet =\ land_storage_change_sum * (areal_corr_factor - 1.0) - \ precipitation * (areal_corr_factor - 1.0) + \ (actual_soil_evap + canopy_evap + sublimation) * areal_corr_factor # Avoid negative values for corrected actual total evaporation for land # This negative values are stored in neg_land_aet if corr_land_aet < 0: neg_land_aet = corr_land_aet corr_land_aet = 0 else: # ================================================================= # Check if current_landarea_frac == 0 , then add previous storage # to daily_storage_tranfer. This storage will then added to runoff # (e.g. island) # ================================================================= daily_storage_transfer += soil_water_content daily_storage_transfer *= areal_corr_factor soil_water_content = 0 groundwater_recharge_from_soil_mm = 0 total_daily_runoff = 0 corr_land_aet = 0 # ===================================================================== # Surface Runoff # ===================================================================== # Correcting total_daily_runoff with neg_land_aet if neg_land_aet < 0: total_daily_runoff = total_daily_runoff + neg_land_aet # Avoid negative total daily runoff in case neg_land_aet is large. if total_daily_runoff < 0: total_daily_runoff = 0 # Compute deficit surface runoff (neg_runoff) from total_daily_runoff # and groundwater recharge from soil [after evapotranspiration # correction]. The idea here is to reduce soil water content with # deficit surface runoff when groundwater recharge from soil is larger # than total daily runoff. Also groundwater recharge from soil is # limited to total daily runoff. if total_daily_runoff - groundwater_recharge_from_soil_mm < 0: neg_runoff = total_daily_runoff - groundwater_recharge_from_soil_mm # Ensure that groundwater recharge from soil is not greater than # total daily runoff groundwater_recharge_from_soil_mm = total_daily_runoff soil_water_content += neg_runoff if soil_water_content < 0: soil_water_content = 0 # Finally surface runoff is calculated as follows: surface_runoff = total_daily_runoff - groundwater_recharge_from_soil_mm return soil_water_content, groundwater_recharge_from_soil_mm, \ actual_soil_evap, soil_saturation, surface_runoff, \ daily_storage_transfer, total_daily_runoff, daily_runoff, \ soil_water_overflow, immediate_runoff, corr_land_aet