inst/tests/src/runit.NonLinDecompOp_with_linear_fluxes.R

# test the constructor
test.function_by_PoolIndex=function(){
    poolNames=c('barrel','glass','belly')
    timeSymbol='t'
    funcOfVars=function(belly,glass,barrel,t){ # we mix up the order
        # we return a vector for test reasons only
        # in the application the result will be a scalar
        # but it is easier to check that the result is correct this way
        c(belly,glass,barrel,t) 
    }
    funcOfVec=by_PoolIndex(funcOfVars,poolNames,timeSymbol)
    belly=1;glass=2;barrel=3;t=4
    rev=c(belly,glass,barrel,t) # same as funcOf Vars returns

    res_vars=funcOfVars(belly,glass,barrel,t)
    checkEquals(res_vars,rev)

    vec=c(barrel,glass,belly) # in order of the poolNames
    # now apply the two functions to equivalent arguments and check that the result is the same
    res_vec=funcOfVec(vec,t)
    checkEquals(res_vec,rev)

    # another example with a function that depends only on a 
    # single argument
    leaf_resp=function(leaf_pool_content){leaf_pool_content*4}
    leaf_resp(1)
    poolNames=c(
       'some_thing'
      ,'some_thing_else'
      ,'some_thing_altogther'
      ,'leaf_pool_content'
    )
    # create a version of the function f(vec,t) 
    leaf_resp_vec=by_PoolIndex(leaf_resp,poolNames,timeSymbol='t')
    # The result is the same since the only the forth position in the vector
    # is equal to our original leaf_pool_content=1
    leaf_resp_vec(c(1,27,3,1),5) 
}

test.NonLinDecompOp_with_linear_fluxes_by_Index=function(){
  #     barrel
  #  X= glass
  #     belly
  n<-3
  k<-3
  intfs=c(
    InternalFlux_by_PoolIndex(
         sourceIndex=1
        ,destinationIndex=2
        ,func=function(X,t){
            barrel=X[[1]]
            k*barrel # just a linear donor dependent flux although we could use the glass argument 
        }
    )
  )
  ofs=c(
    OutFlux_by_PoolIndex(
         sourceIndex=1
        ,func=function(X,t){
            barrel=X[[1]]
            k*barrel # just a linear donor dependent flux 
        }
    )
  )
  #BFunc=UnBoundNonLinDecompOp(
  #  chi_func
  #  ,normalized_internal_fluxes=intfs
  #  ,normalized_out_fluxes=ofs
  #  ,numberOfPools=3
  #  ,timeSymbol='t'
  #)@matFunc
  BFunc=getCompartmentalMatrixFunc(
      UnBoundNonLinDecompOp(
      internal_fluxes=intfs
      ,out_fluxes=ofs
      ,numberOfPools=n
      ,timeSymbol='t'
    )
  )
  # initial values
  iv<-c(barrel=1,glass=1,belly=1)
  B<-BFunc(iv,t=0)
  print(B)
  checkEquals(
     B
    ,matrix(
       nrow=n
      ,ncol=n
      ,byrow=TRUE
      ,c( 
       -2*k,0,0
         ,k,0,0
         ,0,0,0
      )
    )
  )
}
#--------------------------------------------------------------
test.NonLinDecompOp_with_linear_fluxes_by_Name=function(){
    # formulate a linear autonomous model
    # (constant matrix and constant influxes)
    iv<-c(barrel=1,glass=1,belly=1)
    state_variable_names=names(iv)
    timeSymbol='t'
    times=0:10
    n<-3
    k<-.02
    B_ref=matrix(
         nrow=n
        ,ncol=n
        ,byrow=TRUE
        ,c( 
         -2*k,0,0
           ,k,0,0
           ,0,0,0
        )
    )
    ifrs=ConstantInternalFluxRateList_by_PoolName(
        list(
            ConstantInternalFluxRate_by_PoolName(sourceName='barrel',destinationName='glass',rate_constant=k)
            )
    )
    ofrs=ConstantOutFluxRateList_by_PoolName(
        list(
            ConstantOutFluxRate_by_PoolName(sourceName='barrel',rate_constant=k)
         )
    )
    cop=ConstLinDecompOp(
        internal_flux_rates=ifrs
        ,out_flux_rates=ofrs
        ,poolNames=state_variable_names
    )
    ifl_cl=ConstInFluxes(c(1,0,0))
    # initial values
    mod_cl=GeneralModel(
          t=times
          ,A=cop
          ,ivList=iv
          ,inputFluxes=ifl_cl
          ,timeSymbol='t'
    )
    I_func_cl=getFunctionDefinition(ifl_cl)
    I_cl_0=I_func_cl(0)
    pe(I_cl_0)
    rhs_cl=getRightHandSideOfODE(mod_cl)
    rhs_cl_0=rhs_cl(iv,0)
    pe(rhs_cl_0)
 
    # now we formulate the same Model as (possibly) nonlinear Model
    # which does not change the solution but hides the information
    # of linearity from SoilR

    intfs=InternalFluxList_by_PoolName(
      list(
          InternalFlux_by_PoolName(
              sourceName='barrel'
              ,destinationName='glass'
              ,func=function(barrel,t){
                  # just a linear donor dependent flux 
                  # although we could use the glass argument 
                  k*barrel 
              }
          )
      )
    )
      
    ofs=OutFluxList_by_PoolName(
          c(
              OutFlux_by_PoolName(
                  sourceName='barrel'
                  ,func=function(barrel,t){
                      k*barrel 
                      # just a linear donor dependent 
                      # flux the second argument is fake but here for the test
                  }
              )
          )
    )
    ifs=InFluxList_by_PoolName(
          c(
              InFlux_by_PoolName(
                  destinationName='barrel'
                  ,func=function(barrel,glass,t){
                      # ignore arguments
                      #res=(barrel+glass)*(1+sin(t))/100
                      1
                  }
              )
          )
    )
    #BFunc=UnBoundNonLinDecompOp(
    #  chi_func
    #  ,normalized_internal_fluxes=intfs
    #  ,normalized_out_fluxes=ofs
    #  ,numberOfPools=3
    #  ,timeSymbol='t'
    #)@matFunc
    
    obn<- UnBoundNonLinDecompOp_by_PoolNames(
      internal_fluxes=intfs
      ,out_fluxes=ofs
      ,timeSymbol='t'
    )
    BFunc=getCompartmentalMatrixFunc(
      obn
      ,timeSymbol
      ,state_variable_names
      )
    # initial values
    iv<-c(barrel=1,glass=1,belly=1)
    B_0<-BFunc(iv,t=0)
    pp('B_0')
    B_0_cl=cop@mat
    pp('B_0_cl')
    checkEquals(B_0,B_ref)
    mod=GeneralModel(
          t=times
          ,A=obn
          ,ivList=iv 
          ,inputFluxes=ifs
          ,timeSymbol='t'
    )
    mod_cl=GeneralModel(
          t=times
          ,A=cop
          ,ivList=iv
          ,inputFluxes=ifl_cl
          ,timeSymbol='t'
    )

    I_func=getFunctionDefinition(
      mod@inputFluxes
      ,timeSymbol=timeSymbol
      ,poolNames=names(iv)
    )
    I_func_cl=getFunctionDefinition(
      mod_cl@inputFluxes
    )
    I_0=I_func(iv,0)
    I_0_cl=I_func_cl(0)
    rhs=getRightHandSideOfODE(mod)
    rhs_0=rhs(iv,0)
    sol=getC(mod)
    co2=getReleaseFlux(mod)
    sol_cl=getC(mod_cl)
    co2_cl=getReleaseFlux(mod_cl)
    #plot(x=times,y=sol[,2])
    #lines(x=times,y=sol_cl[,2],col='red')
    plot(x=times,y=co2[,1])
    lines(x=times,y=co2_cl[,1],col='red')

}

Try the SoilR package in your browser

Any scripts or data that you put into this service are public.

SoilR documentation built on Oct. 13, 2023, 5:06 p.m.