Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

111 set air temperature as local leaf property #114

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 21 additions & 7 deletions src/hydroshoot/energy.py
Original file line number Diff line number Diff line change
Expand Up @@ -127,7 +127,7 @@ def set_leaf_temperature_to_air_temperature(air_temperature, leaf_ids):
return {vid: air_temperature for vid in leaf_ids}


def set_wind_speed(g, meteo, leaf_lbl_prefix='L'):
def set_local_wind_speed(g, meteo, leaf_lbl_prefix='L') -> dict:
"""Basic model for wind speed at leaf level, considered equal to air wind speed for all leaves

Args:
Expand All @@ -144,6 +144,22 @@ def set_wind_speed(g, meteo, leaf_lbl_prefix='L'):
return {vid: u for vid in leaves}


def set_local_air_temperature(g, meteo, leaf_lbl_prefix='L') -> dict:
"""Basic model for air temperature at the leaf level, considered equal to air temperature at reference height for
all leaves.

Args:
g: a multiscale tree graph object
meteo (DataFrame): forcing meteorological variables
leaf_lbl_prefix (str): the prefix of the leaf label

Returns:
(dict): keys are leaves vertices ids and their values are all equal to air temperature at reference height

"""
return {vid: meteo['Tac'].iloc[0] for vid in get_leaves(g=g, leaf_lbl_prefix=leaf_lbl_prefix)}


def _gbH(leaf_length, wind_speed):
"""Computes boundary layer conductance to heat

Expand Down Expand Up @@ -187,12 +203,11 @@ def calc_heat_boundary_layer_conductance(g, leaf_ids, unit_scene_length):
return g


def calc_leaf_temperature(g, meteo, t_soil, t_sky_eff, leaf_ids, max_iter=100, t_error_crit=0.01, t_step=0.5):
def calc_leaf_temperature(g, t_soil, t_sky_eff, leaf_ids, max_iter=100, t_error_crit=0.01, t_step=0.5):
"""Computes the temperature of each individual leaf and soil elements.

Args:
g: a multiscale tree graph object
meteo (DataFrame): forcing meteorological variables
t_soil (float): [°C] soil surface temperature
t_sky_eff (float): [°C] effective sky temperature
solo (bool):
Expand All @@ -211,7 +226,6 @@ def calc_leaf_temperature(g, meteo, t_soil, t_sky_eff, leaf_ids, max_iter=100, t
"""

temp_sky = utils.celsius_to_kelvin(t_sky_eff)
temp_air = utils.celsius_to_kelvin(meteo['Tac'])
temp_soil = utils.celsius_to_kelvin(t_soil)

t_prev = g.property('Tlc')
Expand All @@ -231,6 +245,7 @@ def calc_leaf_temperature(g, meteo, t_soil, t_sky_eff, leaf_ids, max_iter=100, t
gb_h = mtg_node.properties()['gbH']
evap = mtg_node.properties()['E']
t_leaf = utils.celsius_to_kelvin(mtg_node.properties()['Tlc'])
temp_air = utils.celsius_to_kelvin(mtg_node.properties()['Tac'])

longwave_gain_from_leaves = ff_leaves * cst.sigma * t_leaf ** 4

Expand Down Expand Up @@ -264,13 +279,12 @@ def _VineEnergyX(t_leaf):
return t_new, it


def calc_leaf_temperature2(g, meteo, t_soil, t_sky_eff, leaf_ids):
def calc_leaf_temperature2(g, t_soil, t_sky_eff, leaf_ids):
"""Computes the temperature of each individual leaf and soil elements whilst considering explicitly each other
leaf temperature energy in a matrix solution using symbolic solver.

Args:
g: a multiscale tree graph object
meteo (DataFrame): forcing meteorological variables
t_soil (float): [°C] soil surface temperature
t_sky_eff (float): [°C] effective sky temperature
leaf_ids (list of int): leaf ids
Expand All @@ -282,7 +296,6 @@ def calc_leaf_temperature2(g, meteo, t_soil, t_sky_eff, leaf_ids):
"""

temp_sky = utils.celsius_to_kelvin(t_sky_eff)
temp_air = utils.celsius_to_kelvin(meteo['Tac'])
temp_soil = utils.celsius_to_kelvin(t_soil)

t_prev = g.property('Tlc')
Expand All @@ -299,6 +312,7 @@ def calc_leaf_temperature2(g, meteo, t_soil, t_sky_eff, leaf_ids):
ff_sky = mtg_node.properties()['ff_sky']
ff_leaves = mtg_node.properties()['ff_leaves']
ff_soil = mtg_node.properties()['ff_soil']
temp_air = utils.celsius_to_kelvin(mtg_node.properties()['Tac'])

gb_h = mtg_node.properties()['gbH']
evap = mtg_node.properties()['E']
Expand Down
14 changes: 7 additions & 7 deletions src/hydroshoot/exchange.py
Original file line number Diff line number Diff line change
Expand Up @@ -533,7 +533,7 @@ def calc_gas_exchange_rates(leaf_water_potential, leaf_temperature, carboxylatio
leaf_ids (list): leaf ids (default None)
photo_params (dict): values at 25 °C of Farquhar's model
gs_params (dict): parameters of the stomatal conductance model (model, g0, m0, psi0, D0, n)
air_temperature (float): [°C] air temperature
air_temperature (dict): [°C] air temperature
relative_humidity (float): (%) air relative humidity (between 0 and 100 for dry and saturated air, respectively)
air_co2 (float): [ppm] air CO2 concentration
atmospheric_pressure (float): [kPa] atmospheric pressure
Expand Down Expand Up @@ -581,7 +581,7 @@ def calc_gas_exchange_rates(leaf_water_potential, leaf_temperature, carboxylatio
dHd=entalpy_deactivation[vid]))

a_n, c_c, c_i, gs = an_gs_ci(
air_temperature=air_temperature,
air_temperature=air_temperature[vid],
absorbed_ppfd=absorbed_ppfd[vid],
relative_humidity=relative_humidity,
leaf_temperature=temperature_leaf,
Expand All @@ -595,13 +595,13 @@ def calc_gas_exchange_rates(leaf_water_potential, leaf_temperature, carboxylatio
leaf_length=leaf_length[vid],
wind_speed=wind_speed[vid],
atm_pressure=atmospheric_pressure,
air_temp=air_temperature,
air_temp=air_temperature[vid],
ideal_gas_cst=r)

# Transpiration
e = calc_transpiration_rate(
leaf_temperature=temperature_leaf,
ea=utils.calc_air_vapor_pressure(air_temperature=air_temperature, relative_humidity=relative_humidity),
ea=utils.calc_air_vapor_pressure(air_temperature=air_temperature[vid], relative_humidity=relative_humidity),
gs=gs,
gb=gb,
atm_pressure=atmospheric_pressure)
Expand All @@ -616,15 +616,14 @@ def calc_gas_exchange_rates(leaf_water_potential, leaf_temperature, carboxylatio
boundary_layer_conductance_rate, transpiration_rate)


def set_gas_exchange_rates(g, photo_params, gs_params, air_temperature, relative_humidity, air_co2,
def set_gas_exchange_rates(g, photo_params, gs_params, relative_humidity, air_co2,
atmospheric_pressure, E_type2, leaf_ids, rbt=2. / 3.):
"""Sets gas exchange fluxes at the leaf scale analytically.

Args:
g: a multiscale tree graph object
photo_params (dict): values at 25 °C of Farquhar's model
gs_params (dict): parameters of the stomatal conductance model (model, g0, m0, psi0, D0, n)
air_temperature (float): [°C] air temperature
relative_humidity (float): (%) air relative humidity (between 0 and 100 for dry and saturated air, respectively)
air_co2 (float): [ppm] air CO2 concentration
atmospheric_pressure (float): [kPa] atmospheric pressure
Expand Down Expand Up @@ -656,6 +655,7 @@ def set_gas_exchange_rates(g, photo_params, gs_params, air_temperature, relative
all_absorbed_ppfd = g.property(E_type2)
all_leaf_length = g.property('Length')
all_wind_speed = g.property('u')
all_air_temperature = g.property('Tac')

(g.properties()['An'], g.properties()['Ci'], g.properties()['gs'], g.properties()['gb'],
g.properties()['E']) = calc_gas_exchange_rates(
Expand All @@ -672,7 +672,7 @@ def set_gas_exchange_rates(g, photo_params, gs_params, air_temperature, relative
leaf_ids=leaf_ids,
photo_params=photo_params,
gs_params=gs_params,
air_temperature=air_temperature,
air_temperature=all_air_temperature,
relative_humidity=relative_humidity,
air_co2=air_co2,
atmospheric_pressure=atmospheric_pressure,
Expand Down
10 changes: 7 additions & 3 deletions src/hydroshoot/initialisation.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

from hydroshoot import soil
from hydroshoot.architecture import get_mtg_base, add_soil_surface_mesh, get_leaves
from hydroshoot.energy import set_form_factors_simplified, set_wind_speed
from hydroshoot.energy import set_form_factors_simplified, set_local_wind_speed, set_local_air_temperature
from hydroshoot.exchange import leaf_Na
from hydroshoot.io import HydroShootInputs, HydroShootHourlyInputs
from hydroshoot.irradiance import irradiance_distribution, hsCaribu, set_optical_properties
Expand Down Expand Up @@ -169,8 +169,12 @@ def init_hourly(g: MTG, inputs_hourly: HydroShootHourlyInputs, leaf_ppfd: dict,
# Add a date index to g
g.date = datetime.strftime(inputs_hourly.date, "%Y%m%d%H%M%S")

# initiate wind speed
g.properties()['u'] = set_wind_speed(
# initiate local wind speed
g.properties()['u'] = set_local_wind_speed(
g=g, meteo=inputs_hourly.weather, leaf_lbl_prefix=params.mtg_api.leaf_lbl_prefix)

# initiate local air temperature
g.properties()['Tac'] = set_local_air_temperature(
g=g, meteo=inputs_hourly.weather, leaf_lbl_prefix=params.mtg_api.leaf_lbl_prefix)

if leaf_ppfd is not None:
Expand Down
8 changes: 4 additions & 4 deletions src/hydroshoot/solver.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ def solve_interactions(g, meteo, psi_soil, t_soil, t_sky_eff, params, calc_colla
# Compute gas-exchange fluxes. Leaf T and Psi are from prev calc loop
exchange.set_gas_exchange_rates(
g=g, photo_params=photosynthesis_params, gs_params=stomatal_conductance_params,
air_temperature=meteo['Tac'], relative_humidity=meteo['hs'], air_co2=meteo['Ca'],
relative_humidity=meteo['hs'], air_co2=meteo['Ca'],
atmospheric_pressure=meteo['Pa'], E_type2=irradiance_type2, leaf_ids=leaf_ids,
rbt=turbulence_resistance)

Expand Down Expand Up @@ -135,7 +135,7 @@ def solve_interactions(g, meteo, psi_soil, t_soil, t_sky_eff, params, calc_colla
# Compute gas-exchange fluxes. Leaf T and Psi are from prev calc loop
exchange.set_gas_exchange_rates(
g=g, photo_params=photosynthesis_params, gs_params=stomatal_conductance_params,
air_temperature=meteo['Tac'], relative_humidity=meteo['hs'], air_co2=meteo['Ca'],
relative_humidity=meteo['hs'], air_co2=meteo['Ca'],
atmospheric_pressure=meteo['Pa'], E_type2=irradiance_type2, leaf_ids=leaf_ids,
rbt=turbulence_resistance)

Expand All @@ -150,11 +150,11 @@ def solve_interactions(g, meteo, psi_soil, t_soil, t_sky_eff, params, calc_colla
it += 1
if params.energy.solo:
g.properties()['Tlc'], t_iter = energy.calc_leaf_temperature(
g=g, meteo=meteo, t_soil=t_soil, t_sky_eff=t_sky_eff, leaf_ids=leaf_ids,
g=g, t_soil=t_soil, t_sky_eff=t_sky_eff, leaf_ids=leaf_ids,
max_iter=max_iter, t_error_crit=temp_error_threshold, t_step=temp_step)
else:
g.properties()['Tlc'], t_iter = energy.calc_leaf_temperature2(
g=g, meteo=meteo, t_soil=t_soil, t_sky_eff=t_sky_eff, leaf_ids=leaf_ids)
g=g, t_soil=t_soil, t_sky_eff=t_sky_eff, leaf_ids=leaf_ids)

# t_iter_list.append(t_iter)
t_new = deepcopy(g.property('Tlc'))
Expand Down
2 changes: 1 addition & 1 deletion src/hydroshoot/version.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

major = 5
minor = 1
post = 0
post = 1

__version__ = ".".join([str(s) for s in (major, minor, post)])
# #}
5 changes: 3 additions & 2 deletions test/test_energy.py
Original file line number Diff line number Diff line change
Expand Up @@ -65,9 +65,10 @@ def test_leaf_temperature():
node.ff_soil = 0.4
node.gbH = 1
node.E = 0.
node.Tlc = met.Tac.values[0]
node.Tac = met.Tac.values[0]
node.Tlc = node.Tac

tleaf, it = calc_leaf_temperature(g, met, tsoil, tsky, get_leaves(g=g, leaf_lbl_prefix='L'))
tleaf, it = calc_leaf_temperature(g, tsoil, tsky, get_leaves(g=g, leaf_lbl_prefix='L'))
assert len(tleaf) == 46
first = list(tleaf.keys())[0]
for vid in tleaf:
Expand Down