Skip to main content

PhoTorch is Python-based software for robust fitting of photosynthesis and stomatal conductance models based on leaf-level gas exchange data.

Project description

PhoTorch

PhoTorch is a robust and generalized photosynthesis biochemical model fitting package based on PyTorch. Read more about PhoTorch in our paper: https://link.springer.com/article/10.1007/s11120-025-01136-7.

Note: The latest version 1.3.0 includes changes to the file structure and function names within the package.

Currently, the package includes the Farquhar, von Caemmerer, and Berry (FvCB) model and stomatal conductance models including Buckley Mott Farquhar (BMF), Medlyn (MED), and Ball Woodrow Berry (BWB) versions. The Ball Berry Leuning (BBL) stomatal conductance model and the PROSPECT models are under development.

Installation of dependencies

pip install torch
pip install numpy
pip install scipy
pip install pandas
pip install matplotlib

Next, download the repository, and try running the examples in the testphotorch.py file.

Repository download

git clone https://github.com/GEMINI-Breeding/photorch.git

or

pip install photorch

1. FvCB model usage

Create a python file in the PhoTorch directory and import necessary packages.

from photorch import *
import pandas as pd
import torch

Load data

Load the example CSV file. The loaded data frame should have columns with titles 'CurveID', 'FittingGroup', 'Ci', 'A', 'Qin', and 'Tleaf'. Each A/Ci curve should have a unique 'CurveID'. If no 'Qin' and 'Tleaf' are available, it will be automatically set to 2000 and 25, respectively.

The data to be loaded should be:

CurveID FittingGroup Ci A Qin Tleaf
1 1 200 20 2000 25
1 1 400 30 2000 25
1 1 600 40 2000 25
2 1 200 25 2000 30
2 1 400 35 2000 30
2 1 700 55 2000 30
dftest = pd.read_csv('data/dfMAGIC043_lr.csv')

Initialize the data

Then, specify the ID of the light response curve. If there is no light response curve in the dataset, ignore it (default is None).

# Specify the list of light response curve IDs, if no light response curve, input "lightresp_id = None" or ignore it.
lcd = fvcb.initLicordata(dftest, preprocess=True, lightresp_id = [118])

Define the device

Default device is 'cpu'. If you have an NVIDIA GPU, set 'device_fit' to 'cuda' and execute the 'lcd.todevice(torch.device(device_fit))' line.

device_fit = 'cpu'
lcd.todevice(torch.device(device_fit)) # if device is cuda, then execute this line

Initialize FvCB model

If 'onefit' is set to 'True', all curves in a fitting group will share the same set of Vcmax25, Jmax25, TPU25, and Rd25. Otherwise, each curve will have its own set of these four main parameters but share the same light and temperature response parameters for the fitting group.

If no light response curve is specified, set 'LightResp_type' to 0.

LightResp_type 0: J is equal to Jmax.

LightResp_type 1: using equation $J = \frac{\alpha Q J_{max}}{\alpha Q + J_{max}}$ and fitting $\alpha$.

LightResp_type 2: using equation $J = \frac{\alpha Q + J_{max} - \sqrt{(\alpha Q + J_{max})^2 - 4 \theta \alpha Q J_{max}}}{2 \theta}$ and fitting $\alpha$ and $\theta$.

TempResp_type 0: Vcmax, Jmax, TPU, and Rd are equal to the Vcmax25, Jmax25, TPU25, and Rd25, respectively.

TempResp_type 1: using equation $k = k_{25} \exp{\left[\frac{\Delta{H_a}}{R}\left(\frac{1}{298}-\frac{1}{T_{leaf}}\right)\right]}$ and fitting $\Delta{H_a}$ for Vcmax, Jmax, and TPU.

TempResp_type 2: using equation $k = k_{25} \exp\left[\frac{\Delta H_a}{R} \left(\frac{1}{298}-\frac{1}{T_{leaf}}\right)\right] \frac{f\left(298\right)}{f\left(T_{leaf}\right)}$, where $f(T) = 1+\exp \left[\frac{\Delta H_d}{R}\left(\frac{1}{T_{opt}}-\frac{1}{T} \right)-\ln \left(\frac{\Delta H_d}{\Delta H_a}-1 \right) \right]$, and fitting $\Delta{H_a}$ and $T_{opt}$ for Vcmax, Jmax, and TPU.

# initialize the model
fvcbm = fvcb.model(lcd, LightResp_type = 2, TempResp_type = 2, onefit = False)

More fitting options

fitRd: option to fit $R_{d25}$, default is True. If set to False, $R_{d}$ will be fixed to 1% of $V_{cmax}$.

fitRdratio: option to fit $R_{d}$-to- $V_{cmax}$ ratio, default is False, the range is 0.01 to 0.02.

fitag: option to fit $\alpha_g$, default is False, the range is 0 to 1.

fitKc: option to fit $k_{25}$, default is False.

fitKo: option to fit $k_{25}$, default is False.

fitgamma: option to fit $\Gamma^*_{25}$, default is False.

fitgm: option to fit $g_m$, default is False.

fvcbm = fvcb.model(lcd, LightResp_type = 0, TempResp_type = 1, onefit = False, fitRd = True, fitRdratio = False, fitag = False, fitgm= False, fitgamma=False, fitKo=False, fitKc=False, allparams=allparams)

Specify default fixed or learnable parameters

allparams = fvcb.allparameters()
allparams.dHa_Vcmax = torch.tensor(40.0).to(device_fit) # If the device is cuda, then execute ".to(device_fit)"
fvcbm = fvcb.model(lcd, LightResp_type = 0, TempResp_type = 1, onefit = False, fitag = False, fitgm= False, fitgamma=False, fitKo=False, fitKc=False, allparams=allparams)

Fit A/Ci curves

fitresult = fvcb.fit(fvcbm, learn_rate= 0.08, maxiteration = 20000, minloss= 1, recordweightsTF=False)
fvcbm = fitresult.model

Get fitted parameters by ID

The main parameters are stored in the 'fvcbm'. The temperature response parameters are in 'fvcbm.TempResponse', just like the light response parameters.

id_index = 0
id = int(lcd.IDs[id_index]) # target curve ID
fg_index =  int(lcd.FGs_idx[id_index]) # index of the corresponding fitting group
if not fvcbm.onefit:
    Vcmax25_id = fvcbm.Vcmax25[id_index]
    Jmax25_id = fvcbm.Jmax25[id_index]
else:
    Vcmax25_id = fvcbm.Vcmax25[fg_index]
    Jmax25_id = fvcbm.Jmax25[fg_index]

dHa_Vcmax_id = fvcbm.TempResponse.dHa_Vcmax[fg_index]
alpha_id = fvcbm.LightResponse.alpha[fg_index]

Get fitted A/Ci curves

A, Ac, Aj, Ap = fvcbm()

Get fitted A/Ci curves by ID

id_index = 0
id = lcd.IDs[id_index]
indices_id = lcd.getIndicesbyID(id)
A_id = A[indices_id]
Ac_id = Ac[indices_id]

Get the (preprocessed) photosynthesis data by ID

A_id_mea, Ci_id, Q_id, Tlf_id = lcd.getDatabyID(lcd.IDs[id_index])

A Notebook Interface for Routine Single Group Fitting, Plotting, and Evaluating

For routine fitting of single fitting groups, an easy to use jupyter notebook is available. First, install jupyterlab with

pip install jupyterlab

Once installed, open a terminal window, traverse to your PhoTorch directory and open the notebook with

jupyter-lab PhotosynthesisFitting.ipynb

which should open the notebook and the file directory in an interface using your default web browser. From here, one can easily change model settings in the User Settings section of each code block. The first block will process your data (compile all response curves) and the second block with fit the processed data with PhoTorch.

# Process ACi Data and Verify the Printed Curves are Those Desired to Fit

################# User Settings #################

fitting_group_folder_path = "photorch/data/fvcb/curves/iceberg"
species_to_fit = "Iceberg"
species_variety = "Calmar"

#################################################
from photorch.src.util import *

compiledDataPath = compileACiFiles(fitting_group_folder_path)
# Fit Compiled Data to FvCB Model using PhoTorch, Save Parameters, and Plot Results

######### User Settings ##########
LightResponseType = 2
TemperatureResponseType = 2
Fitgm = False
FitGamma = False
FitKc = False
FitKo = False
saveParameters = True
plotResultingFit = True
#### Advanced Hyper Parameters ####
learningRate = 0.08
iterations = 10000
###################################

The file structure is used in the processing and plotting pipeline so familiarize yourself with it. Data for the FvCB model fitting is housed under "data/fvcb/curves/{FittingGroup}" for the response curves, and "data/fvcb/survey{FittingGroup}" for any auxillary survey measurements. E.g.

data/fvcb/curves/iceberg/

The {FittingGroup}s are species, varieties, or any collection of data used to fit one set of parameters with. The {FittingGroup} folder expects raw LI-6800 ASCII files (without extensions) that are output alongside the optional .xlsx files (which pandas currently cannot import). The processing step converts these raw files to .txt before extracting and compiling the curves. The species_to_fit (required) and species_variety (can be left as an empty string, $``"$) variables are just to name the saved parameters and plots, and are not required to match the path given.

Parameters will optionally be saved to

results/parameters/{species_to_fit}{species_variety}_FvCB_Parameters.csv

and figures will optionally be saved to

results/figures/{species_to_fit}{species_variety}_FvCB_{figure_name}.png

The normalization of response curves from different leaves using survey data is currently under development.


2. Stomatal conductance model usage

Three stomatal conductance model is currently available: Buckley Mott Farquhar (BMF), Medlyn (MED), and Ball Woodrow Berry (BWB). The Ball Berry Leuning (BBL) model is under development. More details about these four models can be found at: https://baileylab.ucdavis.edu/software/helios/_stomatal_doc.html.

Create a python file in the PhoTorch directory and import necessary packages.

import stomatal
import pandas as pd
import torch

Initialize the stomatal conductance data

The data to be loaded should be:

CurveID gsw VPDleaf A Qin Tleaf RHcham
0 0.34 11.32 55 2000 21.81 21.0
0 0.34 18.33 55 2000 22.71 30.02
0 0.35 29.57 51 2000 20.02 38.01
0 0.38 15.4 54 2000 22.6 26.99
0 0.32 15.44 54 1200 19.97 27.0
1 0.23 29.03 49 2000 17.93 35.92
1 0.29 20.51 50 2000 20.51 29.96
1 0.28 11.77 49 2000 18.61 19.99

'A' is not necessary for BMF model.

datasc = pd.read_csv('data/steadystate_stomatalconductance.csv')
scd = stomatal.initscdata(datasc)

Initialize the BMF model and fit the parameters Emerson effect (Em), quantum yield of electron transport (i0), curvature factor (k), and intercept (b).

scm = stomatal.BMF(scd)
#scm = stomatal.BWB(scd) 
#scm = stomatal.MED(scd)
fitresult = stomatal.fit(scm, learnrate = 0.5, maxiteration =20000)
scm = fitresult.model

Get the fitted and measured stomatal conductance

gsw = scm()
gsw_mea = scd.gsw

Get the fitted stomatal conductance by ID

id_index = 0
id = scd.IDs[id_index]
indices_id = scd.getIndicesbyID(id)
gsw_id = gsw[indices_id]

Get the fitted parameters by ID

id_index = 0
id = int(scd.IDs[id_index])
Em_id = scm.Em[id_index]
i0_id = scm.i0[id_index]
k_id = scm.k[id_index]
b_id = scm.b[id_index]

A Notebook Interface for Routine Single Group Fitting, Plotting, and Evaluating

For routine fitting of single fitting groups, an easy to use jupyter notebook is available. First, install jupyterlab with

pip install jupyterlab

Once installed, open a terminal window, traverse to your PhoTorch directory and open the notebook with

jupyter-lab PhotosynthesisFitting.ipynb

which should open the notebook and the file directory in an interface using your default web browser. From here, one can easily change model settings in the User Settings section of the code block.

# Process Stomatal Data and Observe the Model Fit
################# User Settings #################

fitting_group_folder_path = "photorch/data/stomatal/survey/iceberg/Iceberg_poro"
species_to_fit = "Iceberg"
species_variety = "Calmar"

#################################################

Data of steady-state response curves or non-steady-state survey measurements can be fit to and are housed in their respective directories

data/stomatal/curves/{FittingGroup}
data/stomatal/survey/{FittingGroup}

The species_to_fit (required) and species_variety (can be left as an empty string, $``"$) variables are just to name the saved parameters and plots, and are not required to match the path given for the input data.

The inclusive of the use of stomatal models within this notebook interface is under development, but is of course available to program manually with PhoTorch.

3. PROSPECT-X model usage

The PROSPECT-X model is under development.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

photorch-1.4.1.tar.gz (18.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

photorch-1.4.1-py3-none-any.whl (18.0 kB view details)

Uploaded Python 3

File details

Details for the file photorch-1.4.1.tar.gz.

File metadata

  • Download URL: photorch-1.4.1.tar.gz
  • Upload date:
  • Size: 18.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.10.9

File hashes

Hashes for photorch-1.4.1.tar.gz
Algorithm Hash digest
SHA256 c0177b94f57d0ab2fada9a27dded5a3cf6ae8323923f27f87a3c2303140e989b
MD5 e36ac14498adfab45bcaf47a9f8c7627
BLAKE2b-256 f3cc8df4a41bd6064ac3b2581f17676cbc10a9131dea08e35b7e6f3b65c0319c

See more details on using hashes here.

File details

Details for the file photorch-1.4.1-py3-none-any.whl.

File metadata

  • Download URL: photorch-1.4.1-py3-none-any.whl
  • Upload date:
  • Size: 18.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.10.9

File hashes

Hashes for photorch-1.4.1-py3-none-any.whl
Algorithm Hash digest
SHA256 c41681f9cc8cc9b550e6a5f3ccbfbca9d49a6bf8acac3ef3c056c0624f88c4d8
MD5 3e1094e266c9f5b4c51d1eb0e6e7d61c
BLAKE2b-256 8941292e7c1afba29c312e2865052540d1e2884b2847e5d3e045b0f3d05708a1

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page