For this example we will use the data that were recorded during the
Plant Functional Traits Course 6 (PFTC6) in Norway in 2022 at the site
called Liahovden.  The
CO2 concentration data as well as air and soil temperature
and photosynthetically active radiations (PAR) were recorded in a
dataframe named co2_liahovden. The metadata for each
measurements are in a dataframe called record_liahovden.
This dataframe contains the starting time of each measurements, the type
of measurement and the unique ID for each turf. The type of measurement
describes if it was net ecosystem exchange (NEE), measured with a
transparent chamber, or ecosystem respiration (ER), measured with a dark
chamber.
We use the flux_match function to slice the data from
co2_liahovden into each measurement and discard what was
recorded in between.
library(fluxible)
library(dplyr)
str(record_liahovden)
#> tibble [138 × 4] (S3: tbl_df/tbl/data.frame)
#>  $ turfID: chr [1:138] "4 AN1C 4" "4 AN1C 4" "27 AN3C 27" "27 AN3C 27" ...
#>  $ type  : chr [1:138] "NEE" "ER" "NEE" "ER" ...
#>  $ round : num [1:138] 1 1 1 1 1 1 2 2 2 2 ...
#>  $ start : POSIXct[1:138], format: "2022-07-27 05:37:30" "2022-07-27 05:42:00" ...
str(co2_liahovden)
#> tibble [89,692 × 5] (S3: tbl_df/tbl/data.frame)
#>  $ datetime : POSIXct[1:89692], format: "2022-07-27 05:34:49" "2022-07-27 05:34:50" ...
#>  $ temp_air : num [1:89692] 3 NA NA NA NA NA NA NA NA NA ...
#>  $ temp_soil: num [1:89692] 2.96 NA NA NA NA NA NA NA NA NA ...
#>  $ conc     : num [1:89692] 468 469 468 468 468 ...
#>  $ PAR      : num [1:89692] 2.59 NA NA NA NA NA NA NA NA NA ...
conc_liahovden <- flux_match(
  co2_liahovden,
  record_liahovden,
  datetime,
  start,
  conc,
  startcrop = 0,
  measurement_length = 220,
  ratio_threshold = 0.5,
  time_diff = 0
)
str(conc_liahovden)
#> tibble [30,281 × 16] (S3: tbl_df/tbl/data.frame)
#>  $ datetime    : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air    : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil   : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc        : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR         : num [1:30281] NA NA NA NA NA ...
#>  $ turfID      : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type        : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round       : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start       : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start     : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid    : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end       : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_n_conc    : int [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_length    : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio     : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match: chr [1:30281] NA NA NA NA ...Before calculating fluxes we need to fit a model to each measurement
and estimate a slope of the concentration changing rate. We use the
flux_fitting function with the model provided by Zhao et al. (2018). The function
flux_fitting also provides a quadratic and a linear
fit.
slopes_exp_liahovden <- flux_fitting(
  conc_liahovden,
  conc,
  datetime,
  fit_type = "exponential"
)
#> Cutting measurements...
#> Estimating starting parameters for optimization...
#> Optimizing fitting parameters...
#> Calculating fits and slopes...
#> Done.
#> Warning in flux_fitting(conc_liahovden, conc, datetime, fit_type = "exponential"): 
#>  fluxID 77 : slope was estimated on 194 points out of 220 seconds
#>  fluxID 81 : slope was estimated on 217 points out of 220 seconds
#>  fluxID 83 : slope was estimated on 215 points out of 220 seconds
#>  fluxID 85 : slope was estimated on 175 points out of 220 seconds
str(slopes_exp_liahovden)
#> tibble [30,281 × 32] (S3: tbl_df/tbl/data.frame)
#>  $ datetime         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air         : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil        : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc             : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR              : num [1:30281] NA NA NA NA NA ...
#>  $ turfID           : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type             : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round            : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start            : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start          : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid         : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end            : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length         : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio          : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match     : chr [1:30281] NA NA NA NA ...
#>  $ f_time           : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut            : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared_lm    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared_lm: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope_lm       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept_lm   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue_lm      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit_lm         : num [1:30281] 465 465 464 464 464 ...
#>  $ f_Cz             : num [1:30281] 467 467 467 467 467 ...
#>  $ f_Cm             : num [1:30281] 372 372 372 372 372 ...
#>  $ f_a              : num [1:30281] -0.402 -0.402 -0.402 -0.402 -0.402 ...
#>  $ f_b              : num [1:30281] -0.00229 -0.00229 -0.00229 -0.00229 -0.00229 ...
#>  $ f_tz             : num [1:30281] 13.2 13.2 13.2 13.2 13.2 ...
#>  $ f_slope          : num [1:30281] -0.184 -0.184 -0.184 -0.184 -0.184 ...
#>  $ f_fit            : num [1:30281] 470 470 470 469 469 ...
#>  $ f_fit_slope      : num [1:30281] 470 470 470 469 469 ...
#>  $ f_start_z        : POSIXct[1:30281], format: "2022-07-27 05:37:43" "2022-07-27 05:37:43" ...
#>  - attr(*, "fit_type")= chr "exp_zhao18"
slopes_qua_liahovden <- flux_fitting(
  conc_liahovden,
  conc,
  datetime,
  fit_type = "quadratic"
)
#> Warning in flux_fitting(conc_liahovden, conc, datetime, fit_type = "quadratic"): 
#>  fluxID 77 : slope was estimated on 194 points out of 220 seconds
#>  fluxID 81 : slope was estimated on 217 points out of 220 seconds
#>  fluxID 83 : slope was estimated on 215 points out of 220 seconds
#>  fluxID 85 : slope was estimated on 175 points out of 220 seconds
str(slopes_qua_liahovden)
#> tibble [30,281 × 33] (S3: tbl_df/tbl/data.frame)
#>  $ datetime         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air         : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil        : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc             : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR              : num [1:30281] NA NA NA NA NA ...
#>  $ turfID           : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type             : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round            : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start            : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start          : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid         : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end            : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length         : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio          : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match     : chr [1:30281] NA NA NA NA ...
#>  $ f_time           : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut            : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared_lm    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared_lm: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope_lm       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept_lm   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue_lm      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit_lm         : num [1:30281] 465 465 464 464 464 ...
#>  $ f_param1         : num [1:30281] -0.11 -0.11 -0.11 -0.11 -0.11 ...
#>  $ f_param2         : num [1:30281] 3.9e-06 3.9e-06 3.9e-06 3.9e-06 3.9e-06 ...
#>  $ f_rsquared       : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared   : num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_intercept      : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue         : Named num [1:30281] 9.07e-161 9.07e-161 9.07e-161 9.07e-161 9.07e-161 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_slope          : num [1:30281] -0.11 -0.11 -0.11 -0.11 -0.11 ...
#>  $ f_fit            : num [1:30281] 465 465 464 464 464 ...
#>  $ f_fit_slope      : num [1:30281] 465 465 464 464 464 ...
#>  $ f_start_z        : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  - attr(*, "fit_type")= chr "quadratic"
slopes_lin_liahovden <- flux_fitting(
  conc_liahovden,
  conc,
  datetime,
  fit_type = "linear"
)
#> Warning in flux_fitting(conc_liahovden, conc, datetime, fit_type = "linear"): 
#>  fluxID 77 : slope was estimated on 194 points out of 220 seconds
#>  fluxID 81 : slope was estimated on 217 points out of 220 seconds
#>  fluxID 83 : slope was estimated on 215 points out of 220 seconds
#>  fluxID 85 : slope was estimated on 175 points out of 220 seconds
str(slopes_lin_liahovden)
#> tibble [30,281 × 23] (S3: tbl_df/tbl/data.frame)
#>  $ datetime      : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air      : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil     : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc          : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR           : num [1:30281] NA NA NA NA NA ...
#>  $ turfID        : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type          : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round         : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start       : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid      : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end         : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length      : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio       : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match  : chr [1:30281] NA NA NA NA ...
#>  $ f_time        : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut         : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit         : num [1:30281] 465 465 464 464 464 ...
#>  - attr(*, "fit_type")= chr "linear"The function flux_quality is used to provide diagnostics
about the quality of the fit, eventually advising to discard some
measurements or replace them by zero.
slopes_exp_liahovden_flag <- flux_quality(
  slopes_exp_liahovden,
  conc
)
#> 
#>  Total number of measurements: 138
#> 
#>  ok   109     79 %
#>  zero     27      20 %
#>  discard      2   1 %
#>  force_discard    0   0 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
str(slopes_exp_liahovden_flag)
#> tibble [30,281 × 39] (S3: tbl_df/tbl/data.frame)
#>  $ datetime         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air         : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil        : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc             : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR              : num [1:30281] NA NA NA NA NA ...
#>  $ turfID           : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type             : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round            : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start            : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start          : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid         : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end            : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length         : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio          : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match     : chr [1:30281] NA NA NA NA ...
#>  $ f_time           : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut            : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared_lm    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared_lm: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope_lm       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept_lm   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue_lm      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit_lm         : num [1:30281] 465 465 464 464 464 ...
#>  $ f_Cz             : num [1:30281] 467 467 467 467 467 ...
#>  $ f_Cm             : num [1:30281] 372 372 372 372 372 ...
#>  $ f_a              : num [1:30281] -0.402 -0.402 -0.402 -0.402 -0.402 ...
#>  $ f_b              : num [1:30281] -0.00229 -0.00229 -0.00229 -0.00229 -0.00229 ...
#>  $ f_tz             : num [1:30281] 13.2 13.2 13.2 13.2 13.2 ...
#>  $ f_slope          : num [1:30281] -0.184 -0.184 -0.184 -0.184 -0.184 ...
#>  $ f_fit            : num [1:30281] 470 470 470 469 469 ...
#>  $ f_fit_slope      : num [1:30281] 470 470 470 469 469 ...
#>  $ f_start_z        : POSIXct[1:30281], format: "2022-07-27 05:37:43" "2022-07-27 05:37:43" ...
#>  $ f_min_slope      : num [1:30281] 0.0457 0.0457 0.0457 0.0457 0.0457 ...
#>  $ f_cor_coef       : num [1:30281] -0.983 -0.983 -0.983 -0.983 -0.983 ...
#>  $ f_RMSE           : num [1:30281] 2.25 2.25 2.25 2.25 2.25 ...
#>  $ f_gfactor        : num [1:30281] 1.69 1.69 1.69 1.69 1.69 ...
#>  $ f_quality_flag   : chr [1:30281] "ok" "ok" "ok" "ok" ...
#>  $ f_slope_corr     : num [1:30281] -0.184 -0.184 -0.184 -0.184 -0.184 ...
#>  $ f_start_error    : chr [1:30281] NA NA NA NA ...
#>  - attr(*, "fit_type")= chr "exp_zhao18"
slopes_qua_liahovden_flag <- flux_quality(
  slopes_qua_liahovden,
  conc
)
#> 
#>  Total number of measurements: 138
#> 
#>  ok   82      59 %
#>  zero     43      31 %
#>  discard      13      9 %
#>  force_discard    0   0 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
str(slopes_qua_liahovden_flag)
#> tibble [30,281 × 38] (S3: tbl_df/tbl/data.frame)
#>  $ datetime         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air         : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil        : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc             : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR              : num [1:30281] NA NA NA NA NA ...
#>  $ turfID           : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type             : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round            : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start            : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start          : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid         : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end            : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length         : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio          : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match     : chr [1:30281] NA NA NA NA ...
#>  $ f_time           : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut            : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared_lm    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared_lm: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope_lm       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept_lm   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue_lm      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit_lm         : num [1:30281] 465 465 464 464 464 ...
#>  $ f_param1         : num [1:30281] -0.11 -0.11 -0.11 -0.11 -0.11 ...
#>  $ f_param2         : num [1:30281] 3.9e-06 3.9e-06 3.9e-06 3.9e-06 3.9e-06 ...
#>  $ f_rsquared       : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared   : num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_intercept      : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue         : Named num [1:30281] 9.07e-161 9.07e-161 9.07e-161 9.07e-161 9.07e-161 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_slope          : num [1:30281] -0.11 -0.11 -0.11 -0.11 -0.11 ...
#>  $ f_fit            : num [1:30281] 465 465 464 464 464 ...
#>  $ f_fit_slope      : num [1:30281] 465 465 464 464 464 ...
#>  $ f_start_z        : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_min_slope      : num [1:30281] 0.0457 0.0457 0.0457 0.0457 0.0457 ...
#>  $ f_gfactor        : num [1:30281] 1.01 1.01 1.01 1.01 1.01 ...
#>  $ f_quality_flag   : chr [1:30281] "ok" "ok" "ok" "ok" ...
#>  $ f_slope_corr     : num [1:30281] -0.11 -0.11 -0.11 -0.11 -0.11 ...
#>  $ f_start_error    : chr [1:30281] NA NA NA NA ...
#>  - attr(*, "fit_type")= chr "quadratic"
slopes_lin_liahovden_flag <- flux_quality(
  slopes_lin_liahovden,
  conc
)
#> 
#>  Total number of measurements: 138
#> 
#>  ok   52      38 %
#>  zero     52      38 %
#>  discard      34      25 %
#>  force_discard    0   0 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
str(slopes_lin_liahovden_flag)
#> tibble [30,281 × 27] (S3: tbl_df/tbl/data.frame)
#>  $ datetime      : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:31" ...
#>  $ temp_air      : num [1:30281] NA NA NA NA NA NA NA NA NA 3 ...
#>  $ temp_soil     : num [1:30281] NA NA NA NA NA NA NA NA NA 6.83 ...
#>  $ conc          : num [1:30281] 468 467 467 467 467 ...
#>  $ PAR           : num [1:30281] NA NA NA NA NA ...
#>  $ turfID        : chr [1:30281] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" ...
#>  $ type          : chr [1:30281] "NEE" "NEE" "NEE" "NEE" ...
#>  $ round         : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ start         : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_start       : POSIXct[1:30281], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ f_fluxid      : Factor w/ 138 levels "1","2","3","4",..: 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_end         : POSIXct[1:30281], format: "2022-07-27 05:41:10" "2022-07-27 05:41:10" ...
#>  $ f_length      : num [1:30281] 220 220 220 220 220 220 220 220 220 220 ...
#>  $ f_ratio       : num [1:30281] 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_flag_match  : chr [1:30281] NA NA NA NA ...
#>  $ f_time        : num [1:30281] 0 1 2 3 4 5 6 7 8 9 ...
#>  $ f_cut         : Factor w/ 1 level "keep": 1 1 1 1 1 1 1 1 1 1 ...
#>  $ f_rsquared    : num [1:30281] 0.967 0.967 0.967 0.967 0.967 ...
#>  $ f_adj_rsquared: num [1:30281] 0.966 0.966 0.966 0.966 0.966 ...
#>  $ f_slope       : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_intercept   : num [1:30281] 465 465 465 465 465 ...
#>  $ f_pvalue      : Named num [1:30281] 9.23e-163 9.23e-163 9.23e-163 9.23e-163 9.23e-163 ...
#>   ..- attr(*, "names")= chr [1:30281] "value" "value" "value" "value" ...
#>  $ f_fit         : num [1:30281] 465 465 464 464 464 ...
#>  $ f_min_slope   : num [1:30281] 0.0457 0.0457 0.0457 0.0457 0.0457 ...
#>  $ f_quality_flag: chr [1:30281] "ok" "ok" "ok" "ok" ...
#>  $ f_slope_corr  : num [1:30281] -0.109 -0.109 -0.109 -0.109 -0.109 ...
#>  $ f_start_error : chr [1:30281] NA NA NA NA ...
#>  - attr(*, "fit_type")= chr "linear"The function flux_plot provides plots for a visual
assessment of the measurements, explicitly displaying the quality flags
from flux_quality and the cuts from
flux_fitting.
slopes_exp_liahovden_flag |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Part of the fit will not be displayed
#>     because f_ylim_upper is too low.
#> Part of the fit will not be displayed
#>     because f_ylim_lower is too high.
#> Plotting in progress
slopes_qua_liahovden_flag |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Plotting in progress
slopes_lin_liahovden_flag |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Plotting in progressBased on the quality flags and the plots, the user can decide to run
flux_fitting again with different arguments. Here we will
do it while cutting the last 60 seconds of the fluxes (cutting the last
third). We also detected fluxes that do not look correct. Sometimes some
measurements will pass the automated quality control but are obviously
wrong for an experience user. That is what the
force_discard argument is for. For the sake of
reproducibility, this argument should be the last option and be
accompanied with a justification.
slopes_exp_liahovden_flag_60 <- conc_liahovden |>
  flux_fitting(
    conc,
    datetime,
    fit_type = "exp_zhao18",
    end_cut = 60
  ) |>
  flux_quality(
    conc,
    force_discard = c(
      51, # slope is much steeper than the flux because t zero was estimated
      # at the very start of the measurement
      101 # plot starts with a high peak: accumulation in the canopy?
    )
  )
#> Cutting measurements...
#> Estimating starting parameters for optimization...
#> Optimizing fitting parameters...
#> Calculating fits and slopes...
#> Done.
#> 
#>  Total number of measurements: 138
#> 
#>  ok   127     92 %
#>  zero     8   6 %
#>  force_discard    2   1 %
#>  discard      1   1 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
slopes_exp_liahovden_flag_60 |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Part of the fit will not be displayed
#>     because f_ylim_lower is too high.
#> Plotting in progressWe also apply a cut on the dataset that was fitted with a quadratic model. At this point it is up to the user to decide which model works the best for the entire dataset. The function flux_quality provides a count of the quality flags that can help to take a decision.
slopes_qua_liahovden_flag_60 <- conc_liahovden |>
  flux_fitting(
    conc,
    datetime,
    fit_type = "quadratic",
    end_cut = 60
  ) |>
  flux_quality(
    conc
  )
#> 
#>  Total number of measurements: 138
#> 
#>  ok   120     87 %
#>  zero     14      10 %
#>  discard      4   3 %
#>  force_discard    0   0 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
slopes_qua_liahovden_flag_60 |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Plotting in progressWhen using a linear fit it is common to take only a short section of the measurement close to the start. Here we will cut 120 seconds at the end, effectively keeping only the first 90 seconds.
slopes_lin_liahovden_flag_120 <- conc_liahovden |>
  flux_fitting(
    conc,
    datetime,
    fit_type = "linear",
    end_cut = 120
  ) |>
  flux_quality(
    conc
  )
#> 
#>  Total number of measurements: 138
#> 
#>  ok   109     79 %
#>  zero     24      17 %
#>  discard      5   4 %
#>  force_discard    0   0 %
#>  start_error      0   0 %
#>  no_data      0   0 %
#>  force_ok     0   0 %
#>  force_zero   0   0 %
#>  force_lm     0   0 %
slopes_lin_liahovden_flag_120 |>
  # we just show a sample of the plots to avoid slowing down the example
  dplyr::filter(f_fluxid %in% c(54, 95, 100, 101)) |>
  flux_plot(
    conc,
    datetime,
    f_ylim_upper = 600,
    f_ylim_lower = 350,
    y_text_position = 450,
    facet_wrap_args = list(
      nrow = 2,
      ncol = 2,
      scales = "free"
    )
  )
#> Plotting in progressOnce we are satisfied with the fit, we can calculate fluxes with
flux_calc. Here the volume is defined as a constant for all
the measurements but it is also possible to provide a specific volume
for each plot in case that is different.
fluxes_exp_liahovden_60 <- slopes_exp_liahovden_flag_60 |>
  flux_calc(
    f_slope_corr, # we use the slopes provided by flux_quality
    datetime,
    temp_air,
    conc_unit = "ppm",
    flux_unit = "mmol",
    chamber_volume = 24.5,
    tube_volume = 0.075,
    atm_pressure = 1,
    plot_area = 0.0625,
    cols_keep = c("turfID", "type", "round"),
    cols_ave = c("temp_soil", "PAR")
  )
#> Cutting data according to 'keep_arg'...
#> Averaging air temperature for each flux...
#> Creating a df with the columns from 'cols_keep' argument...
#> Creating a df with the columns from 'cols_ave' argument...
#> Calculating fluxes...
#> R constant set to 0.082057
#> Concentration was measured in ppm
#> Fluxes are in mmol/m2/h
fluxes_qua_liahovden_60 <- slopes_qua_liahovden_flag_60 |>
  flux_calc(
    f_slope_corr, # we use the slopes provided by flux_quality
    datetime,
    temp_air,
    conc_unit = "ppm",
    flux_unit = "mmol",
    chamber_volume = 24.5,
    tube_volume = 0.075,
    atm_pressure = 1,
    plot_area = 0.0625,
    cols_keep = c("turfID", "type", "round"),
    cols_ave = c("temp_soil", "PAR")
  )
#> Cutting data according to 'keep_arg'...
#> Averaging air temperature for each flux...
#> Creating a df with the columns from 'cols_keep' argument...
#> Creating a df with the columns from 'cols_ave' argument...
#> Calculating fluxes...
#> R constant set to 0.082057
#> Concentration was measured in ppm
#> Fluxes are in mmol/m2/h
fluxes_lin_liahovden_120 <- slopes_lin_liahovden_flag_120 |>
  flux_calc(
    f_slope_corr, # we use the slopes provided by flux_quality
    datetime,
    temp_air,
    conc_unit = "ppm",
    flux_unit = "mmol",
    chamber_volume = 24.5,
    tube_volume = 0.075,
    atm_pressure = 1,
    plot_area = 0.0625,
    cols_keep = c("turfID", "type", "round"),
    cols_ave = c("temp_soil", "PAR")
  )
#> Cutting data according to 'keep_arg'...
#> Averaging air temperature for each flux...
#> Creating a df with the columns from 'cols_keep' argument...
#> Creating a df with the columns from 'cols_ave' argument...
#> Calculating fluxes...
#> R constant set to 0.082057
#> Concentration was measured in ppm
#> Fluxes are in mmol/m2/hThe output is in mmol/m2/h and the calculation used is as follow:
\[ \text{flux}=\text{slope}\times \frac{P\times V}{R\times T\times A} \]
where
flux: the flux of gas at the surface of the plot (mmol/m2/h)
slope: slope estimate (ppm*s-1)
P: pressure, assumed (atm)
V: volume of the chamber and tubing (L)
R: gas constant (0.082057 L*atm*K-1*mol-1)
T: chamber air temperature (K)
A: area of chamber frame base (m2)
The conversion from micromol/m2/s to mmol/m2/h is included in the function.
Fluxes were calculated in five steps from raw gas concentration data and the process is entirely reproducible.
Next we can calculated gross ecosystem production (GEP) and plot the results:
fluxes_exp_liahovden_60_gep <- fluxes_exp_liahovden_60 |>
  flux_gep(
    type,
    datetime,
    id_cols = c("round", "turfID"),
    cols_keep = c("temp_soil", "f_model", "PAR")
  )
#> Warning in flux_gep(fluxes_exp_liahovden_60, type, datetime, id_cols = c("round", : 'flux_gep' is deprecated.
#> Use 'flux_gpp' instead.
#> See help("Deprecated")
str(fluxes_exp_liahovden_60_gep)
#> tibble [207 × 8] (S3: tbl_df/tbl/data.frame)
#>  $ datetime : POSIXct[1:207], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ type     : chr [1:207] "GEP" "NEE" "ER" "GEP" ...
#>  $ f_flux   : num [1:207] -18.57 -14.09 4.48 -38.42 -23.22 ...
#>  $ temp_soil: num [1:207] 6.96 6.96 7.01 6.83 6.83 ...
#>  $ f_model  : chr [1:207] "exp_zhao18" "exp_zhao18" "exp_zhao18" "exp_zhao18" ...
#>  $ PAR      : num [1:207] 24.24 24.24 1.05 28.81 28.81 ...
#>  $ round    : num [1:207] 1 1 1 1 1 1 1 1 1 2 ...
#>  $ turfID   : chr [1:207] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "27 AN3C 27" ...
fluxes_qua_liahovden_60_gep <- fluxes_qua_liahovden_60 |>
  flux_gep(
    type,
    datetime,
    id_cols = c("round", "turfID"),
    cols_keep = c("temp_soil", "f_model", "PAR")
  )
#> Warning in flux_gep(fluxes_qua_liahovden_60, type, datetime, id_cols = c("round", : 'flux_gep' is deprecated.
#> Use 'flux_gpp' instead.
#> See help("Deprecated")
str(fluxes_qua_liahovden_60_gep)
#> tibble [207 × 8] (S3: tbl_df/tbl/data.frame)
#>  $ datetime : POSIXct[1:207], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ type     : chr [1:207] "GEP" "NEE" "ER" "GEP" ...
#>  $ f_flux   : num [1:207] -13.94 -10.07 3.87 -34.04 -24.8 ...
#>  $ temp_soil: num [1:207] 6.96 6.96 7.01 6.83 6.83 ...
#>  $ f_model  : chr [1:207] "quadratic" "quadratic" "quadratic" "quadratic" ...
#>  $ PAR      : num [1:207] 24.24 24.24 1.05 28.81 28.81 ...
#>  $ round    : num [1:207] 1 1 1 1 1 1 1 1 1 2 ...
#>  $ turfID   : chr [1:207] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "27 AN3C 27" ...
fluxes_lin_liahovden_120_gep <- fluxes_lin_liahovden_120 |>
  flux_gep(
    type,
    datetime,
    id_cols = c("round", "turfID"),
    cols_keep = c("temp_soil", "f_model", "PAR")
  )
#> Warning in flux_gep(fluxes_lin_liahovden_120, type, datetime, id_cols = c("round", : 'flux_gep' is deprecated.
#> Use 'flux_gpp' instead.
#> See help("Deprecated")
str(fluxes_lin_liahovden_120_gep)
#> tibble [207 × 8] (S3: tbl_df/tbl/data.frame)
#>  $ datetime : POSIXct[1:207], format: "2022-07-27 05:37:30" "2022-07-27 05:37:30" ...
#>  $ type     : chr [1:207] "GEP" "NEE" "ER" "GEP" ...
#>  $ f_flux   : num [1:207] -10.02 -7.97 2.05 -20.55 -14.47 ...
#>  $ temp_soil: num [1:207] 6.93 6.93 7.01 6.82 6.82 ...
#>  $ f_model  : chr [1:207] "linear" "linear" "linear" "linear" ...
#>  $ PAR      : num [1:207] 23.025 23.025 0.795 28.181 28.181 ...
#>  $ round    : num [1:207] 1 1 1 1 1 1 1 1 1 2 ...
#>  $ turfID   : chr [1:207] "4 AN1C 4" "4 AN1C 4" "4 AN1C 4" "27 AN3C 27" ...
library(ggplot2)
bind_rows(
  fluxes_exp_liahovden_60_gep,
  fluxes_qua_liahovden_60_gep,
  fluxes_lin_liahovden_120_gep
) |>
  ggplot(aes(x = datetime, y = f_flux, color = f_model)) +
  geom_point() +
  geom_smooth() +
  labs(
    title = "Net Ecosystem Exchange at Upper Site (Liahovden) during 24 hour",
    x = "Datetime",
    y = bquote(~ CO[2] ~ "flux [mmol/" * m^2 * "/h]"),
    color = "Model used in flux_fitting"
  ) +
  theme(legend.position = "bottom") +
  facet_grid(type ~ ., scales = "free")