cModel.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722
  1. import numpy
  2. import json
  3. import os
  4. import scipy.interpolate
  5. #for partial function specializations
  6. import functools
  7. import function
  8. import importlib
  9. importlib.reload(function)
  10. class model:
  11. def __init__(self):
  12. self.compartments={}
  13. self.seJ={}
  14. self.scaled=[]
  15. def add_input(self,compartmentName,parameterName):
  16. self.compartments[compartmentName]['input']=parameterName
  17. def add_compartment(self,compartmentName):
  18. self.compartments[compartmentName]={}
  19. self.compartments[compartmentName]['targets']={}
  20. self.compartments[compartmentName]['sensTargets']={}
  21. def getTimeUnit(self):
  22. try:
  23. return self.mod['timeUnit']
  24. except KeyError:
  25. return 's'
  26. def bind(self,src,target,qName,pcName):
  27. #establish a flow from source compartment to the target
  28. #the source equation (where we subtract the current)
  29. #in fact, this is the diagonal element
  30. #get volume names
  31. srcVName=self.getVolumePar(src)
  32. #generate coupling object (w/derivatives)
  33. pSrc=self.couplingObject(-1,qName,pcName,srcVName)
  34. #this includes derivatives and value!
  35. self.addValueObject(src,src,pSrc)
  36. #special target which is not part of calculation
  37. if target=='dump':
  38. return
  39. #the target equation (where we add the current)
  40. #get volume names
  41. targetVName=self.getVolumePar(target)
  42. #generate coupling object
  43. pTarget=self.couplingObject(1,qName,pcName,targetVName)
  44. #equation is for target compartment, but binding for source
  45. self.addValueObject(target,src,pTarget)
  46. def addValueObject(self,targetName,srcName,cObject):
  47. #always binds equation id and a variable
  48. targetList=self.compartments[targetName]['targets']
  49. addValue(targetList,srcName,cObject["value"])
  50. der=cObject["derivatives"]
  51. for d in der:
  52. targetSE=self.getSEJ_comp(d,targetName)
  53. addValue(targetSE,srcName,der[d])
  54. def couplingObject(self,sign, qParName, pcParName, vParName):
  55. qPar=self.get(qParName)
  56. pcPar=self.get(pcParName)
  57. vPar=self.get(vParName)
  58. q=qPar['value']
  59. pc=pcPar['value']
  60. v=vPar['value']
  61. DPC=pcPar['derivatives']
  62. DQ=qPar['derivatives']
  63. DV=vPar['derivatives']
  64. if any(['function' in qPar,'function' in pcPar, 'function' in vPar]):
  65. fq=function.to(q)
  66. fpc=function.to(pc)
  67. fv=function.to(v)
  68. f=lambda t,q=fq,pc=fpc,v=fv,s=sign:s*q(t)/v(t)/pc(t)
  69. dfdPC=lambda t,f=f,pc=fpc:-f(t)/pc(t)
  70. dPC=function.generate(dfdPC,DPC)
  71. dfdQ=lambda t,f=f,q=fq: f(t)/q(t)
  72. dQ=function.generate(dfdQ,DQ)
  73. dfdV=lambda t,f=f,v=fv: -f(t)/v(t)
  74. dV=function.generate(dfdV,DV)
  75. return function.Object(f,[dPC,dQ,dV])
  76. else:
  77. f=sign*q/v/pc
  78. return function.derivedObject(sign*q/v/pc,\
  79. [{'df':-f/pc,'D':DPC},\
  80. {'df':sign/v/pc,'D':DQ},\
  81. {'df':-f/v,'D':DV}])
  82. #derivatives is the combination of the above
  83. def getVolumePar(self,compartment):
  84. #returnis volume name, if found and useVolume is directed,
  85. #or a standard parameter one
  86. try:
  87. return self.mod["volumes"][compartment]
  88. #parV=self.mod["parameters"][parVName]
  89. except KeyError:
  90. pass
  91. return "one"
  92. def build(self):
  93. comps=self.compartments
  94. self.n=len(comps)
  95. #numeric representation of the input
  96. self.fu=numpy.zeros((self.n))
  97. #dictionary that holds potential input function objects
  98. self.du={}
  99. self.lut={c:i for (i,c) in zip(range(self.n),comps.keys())}
  100. self.dM={}
  101. self.fM=numpy.zeros((self.n,self.n))
  102. self.inputDerivatives={}
  103. self.uTotal=[]
  104. for c in comps:
  105. comp=comps[c]
  106. if 'input' in comp:
  107. qs=self.get(comp['input'])
  108. self.uTotal.append(qs["value"])
  109. qV=self.getVolumePar(c)
  110. #input is a quotient (amount of exogen per unit time per volume(mass) of input compartment)
  111. qs1=function.ratio(qs,self.get(qV))
  112. if function.isFunction(qs1):
  113. self.du[self.lut[c]]=qs1
  114. else:
  115. self.fu[self.lut[c]]=qs1['value']
  116. #let buildSE know we have to include this derivatives
  117. self.inputDerivatives[c]=qs1['derivatives']
  118. for t in comp['targets']:
  119. arr=comp['targets'][t]
  120. if function.contains(arr):
  121. try:
  122. self.dM[self.lut[c]][self.lut[t]]=\
  123. function.sumArray(arr)
  124. except (KeyError,TypeError):
  125. self.dM[self.lut[c]]={}
  126. self.dM[self.lut[c]][self.lut[t]]=\
  127. function.sumArray(arr)
  128. else:
  129. #just set once
  130. self.fM[self.lut[c],self.lut[t]]=sum(arr)
  131. #generate total from self.uTotal
  132. #ignore derivatives; uTotal is just a scaling shorthand
  133. if function.contains(self.uTotal):
  134. self.du[self.lut['total']]=function.Object(function.sumArray(self.uTotal),[])
  135. else:
  136. self.fu[self.lut['total']]=sum(self.uTotal)
  137. #build SE part
  138. self.buildSE()
  139. def buildSE(self):
  140. #check which parameterst to include
  141. parList=[]
  142. pars=self.parSetup['parameters']
  143. #add derivatives to jacobi terms
  144. parCandidates=list(self.seJ.keys())
  145. #add derivatives of input terms
  146. for x in self.inputDerivatives:
  147. D=self.inputDerivatives[x]
  148. parCandidates.extend(list(D.keys()))
  149. for x in self.du:
  150. D=self.du[x]['derivatives']
  151. parCandidates.extend(list(D.keys()))
  152. #get rid of duplicates
  153. parCandidates=list(set(parCandidates))
  154. for parName in parCandidates:
  155. #print(par)
  156. par=pars[parName]
  157. usePar=calculateDerivative(par)
  158. #print('[{}]: {}'.format(usePar,par))
  159. if not usePar:
  160. continue
  161. parList.append(parName)
  162. #print(parList)
  163. self.m=len(parList)
  164. self.lutSE={c:i for (i,c) in zip(range(self.m),parList)}
  165. w=self.getWeights(self.lutSE)
  166. w=numpy.sqrt(w)
  167. self.qSS={}
  168. self.SS=numpy.zeros((self.m,self.n,self.n))
  169. #elements of SS will be w_p*dM_i,j/dp
  170. for parName in parList:
  171. try:
  172. sources=self.seJ[parName]
  173. except KeyError:
  174. continue
  175. for compartment in sources:
  176. targets=sources[compartment]
  177. for t in targets:
  178. k=self.lutSE[parName]
  179. i=self.lut[compartment]
  180. j=self.lut[t]
  181. #print('[{} {} {}] {}'.format(parName,compartment,t,targets[t]))
  182. arr=targets[t]
  183. if not function.contains(arr):
  184. self.SS[k,i,j]=w[k]*sum(arr)
  185. continue
  186. ft=function.sumArray(arr,w[k])
  187. try:
  188. self.qSS[k][i][j]=ft
  189. except (KeyError,TypeError):
  190. try:
  191. self.qSS[k][i]={}
  192. self.qSS[k][i][j]=ft
  193. except (KeyError,TypeError):
  194. self.qSS[k]={}
  195. self.qSS[k][i]={}
  196. self.qSS[k][i][j]=ft
  197. #derivatives of inputs
  198. #time dependent derivatives are handled in self.Su(t)
  199. self.fSu=numpy.zeros((self.m,self.n))
  200. for x in self.inputDerivatives:
  201. D=self.inputDerivatives[x]
  202. for p in D:
  203. if p in parList:
  204. k=self.lutSE[p]
  205. self.fSu[self.lutSE[p],self.lut[x]]=D[p]*w[k]
  206. #use fM to build static part of fJ
  207. N=self.n*(self.m+1)
  208. self.fJ=numpy.zeros((N,N))
  209. for i in range(self.m+1):
  210. self.fJ[i*self.n:(i+1)*self.n,i*self.n:(i+1)*self.n]=self.fM
  211. def inspect(self):
  212. comps=self.compartments
  213. pars=self.parSetup['parameters']
  214. #pars=self.mod['parameters']
  215. tu=self.getTimeUnit()
  216. print('Time unit: {}'.format(tu))
  217. print('Compartments')
  218. for c in comps:
  219. print('{}/{}:'.format(c,self.lut[c]))
  220. comp=comps[c]
  221. if 'input' in comp:
  222. print('\tinput\n\t\t{}'.format(comp['input']))
  223. print('\ttargets')
  224. for t in comp['targets']:
  225. print('\t\t{}[{},{}]: {}'.format(t,self.lut[c],self.lut[t],\
  226. comp['targets'][t]))
  227. print('Flows')
  228. for f in self.flows:
  229. fName=self.flows[f]
  230. fParName=self.mod['flows'][fName]
  231. fPar=pars[fParName]
  232. print('\t{}[{}]:{} [{}]'.format(f,fName,fParName,self.get(fParName)))
  233. print('Volumes')
  234. for v in self.mod['volumes']:
  235. vParName=self.mod['volumes'][v]
  236. vPar=pars[vParName]
  237. print('\t{}:{} [{}]'.format(v,vParName,self.get(vParName)))
  238. print('Partition coefficients')
  239. for pc in self.mod['partitionCoefficients']:
  240. pcParName=self.mod['partitionCoefficients'][pc]
  241. pcPar=pars[pcParName]
  242. print('\t{}:{} [{}]'.format(pc,pcParName,self.get(pcParName)))
  243. def inspectSE(self):
  244. print('SE parameters')
  245. for p in self.seJ:
  246. print(p)
  247. sources=self.seJ[p]
  248. for compartment in sources:
  249. targets=sources[compartment]
  250. for t in targets:
  251. print('\t SE bind {}/{}:{}'.format(compartment,t,targets[t]))
  252. def parse(self,setupFile,parameterFile):
  253. with open(setupFile,'r') as f:
  254. self.mod=json.load(f)
  255. with open(parameterFile,'r') as f:
  256. self.parSetup=json.load(f)
  257. self.mod['compartments'].append('total')
  258. for m in self.mod['compartments']:
  259. self.add_compartment(m)
  260. for m in self.mod['scaled']:
  261. self.scaled.append(m)
  262. self.add_default_parameters()
  263. #standard parameters such as one,zero etc.
  264. for s in self.mod['inputs']:
  265. #src=mod['inputs'][s]
  266. self.add_input(s,self.mod['inputs'][s])
  267. self.flows={}
  268. #pars=self.mod['parameters']
  269. pars=self.parSetup['parameters']
  270. for f in self.mod['flows']:
  271. #skip comments
  272. if f.find(':')<0:
  273. continue
  274. comps=f.split(':')
  275. c0=splitVector(comps[0])
  276. c1=splitVector(comps[1])
  277. for x in c0:
  278. for y in c1:
  279. pairName='{}:{}'.format(x,y)
  280. self.flows[pairName]=f
  281. for b in self.mod['bindings']['diffusion']:
  282. #whether to scale transfer constants to organ volume
  283. #default is true, but changing here will assume no scaling
  284. comps=b.split('->')
  285. try:
  286. pcParName=self.mod['partitionCoefficients'][b]
  287. except KeyError:
  288. pcParName="one"
  289. kParName=self.mod['bindings']['diffusion'][b]
  290. #operate with names to allow for value/function/derived infrastructure
  291. self.bind(comps[0],comps[1],kParName,pcParName)
  292. for q in self.mod['bindings']['flow']:
  293. comps=q.split('->')
  294. srcs=splitVector(comps[0])
  295. tgts=splitVector(comps[1])
  296. for cs in srcs:
  297. for ct in tgts:
  298. #get partition coefficient
  299. try:
  300. pcParName=self.mod['partitionCoefficients'][cs]
  301. except KeyError:
  302. pcParName="one"
  303. #get flow (direction could be reversed)
  304. try:
  305. qName=self.flows['{}:{}'.format(cs,ct)]
  306. except KeyError:
  307. qName=self.flows['{}:{}'.format(ct,cs)]
  308. flowParName=self.mod['flows'][qName]
  309. #flowPar=pars[flowParName]
  310. self.bind(cs,ct,flowParName,pcParName)
  311. self.build()
  312. def add_default_parameters(self):
  313. pars=self.parSetup['parameters']
  314. pars['one']={'value':1}
  315. pars['zero']={'value':0}
  316. pars['two']={'value':2}
  317. def M(self,t,y=numpy.array([])):
  318. for i in self.dM:
  319. for j in self.dM[i]:
  320. self.fM[i,j]=self.dM[i][j](t)
  321. #create an array and fill it with outputs of function at t
  322. if (y.size==0):
  323. return self.fM
  324. self.set_scaledM(t,y)
  325. return self.fM
  326. def set_scaledM(self,t,y):
  327. #prevent zero division
  328. eps=1e-8
  329. for c in self.scaled:
  330. i=self.lut[c]
  331. it=self.lut['total']
  332. try:
  333. k=numpy.copy(self.originalK[i])
  334. except AttributeError:
  335. k=numpy.copy(self.fM[i,:])
  336. self.originalK={}
  337. self.originalK[i]=k
  338. #make another copy
  339. k=numpy.copy(self.originalK[i])
  340. except KeyError:
  341. k=numpy.copy(self.fM[i,:])
  342. self.originalK[i]=k
  343. #make another copy
  344. k=numpy.copy(self.originalK[i])
  345. k[i]=k[i]-self.u(t)[it]
  346. #scale all inputs by total input mass
  347. for j in range(self.n):
  348. self.fM[i,j]=k[j]/(y[it]+eps)
  349. def u(self,t):
  350. for x in self.du:
  351. self.fu[x]=self.du[x]['value'](t)
  352. #this should be done previously
  353. return self.fu
  354. def Su(self,t):
  355. w=self.getWeights(self.lutSE)
  356. w=numpy.sqrt(w)
  357. #add time dependent values
  358. for x in self.du:
  359. D=self.du[x]['derivatives']
  360. for p in D:
  361. k=self.lutSE[p]
  362. self.fSu[k,self.lut[x]]=w[k]*D[p](t)
  363. return self.fSu
  364. def jacobiFull(self,t):
  365. #update jacobi created during build phase with time dependent values
  366. for i in self.dM:
  367. for j in self.dM[i]:
  368. for k in range(self.m+1):
  369. self.fJ[k*self.n+i,k*self.n+j]=self.dM[i][j](t)
  370. return self.fJ
  371. def fSS(self,t,y=numpy.array([])):
  372. for k in self.qSS:
  373. for i in self.qSS[k]:
  374. for j in self.qSS[k][i]:
  375. #print('[{},{},{}] {}'.format(k,i,j,self.qSS[k][i][j]))
  376. self.SS[k,i,j]=(self.qSS[k][i][j])(t)
  377. if y.size==0:
  378. return self.SS
  379. self.set_scaledSS(t,y)
  380. return self.SS
  381. def set_scaledSS(self,t,y):
  382. #prevent zero division
  383. eps=1e-8
  384. for c in self.scaled:
  385. it=self.lut['total']
  386. i=self.lut[c]
  387. try:
  388. dkdp=numpy.copy(self.originalSS[i])
  389. except AttributeError:
  390. dkdp=numpy.copy(self.SS[:,i,:])
  391. self.originalSS={}
  392. self.originalSS[i]=dkdp
  393. dkdp=numpy.copy(self.originalSS[i])
  394. except KeyError:
  395. dkdp=numpy.copy(self.SS[:,i,:])
  396. self.originalSS[i]=dkdp
  397. dkdp=numpy.copy(self.originalSS[i])
  398. self.SS[:,i,:]=dkdp/(y[it]+eps)
  399. #should add error on u!
  400. def fSY(self,y,t):
  401. #M number of sensitivity parameters
  402. #N number of equations
  403. #fSS is MxNxN
  404. #assume a tabulated solution y(t) at t spaced intervals
  405. qS=self.fSS(t,y).dot(y)
  406. #qS is MxN
  407. #but NxM is expected, so do a transpose
  408. #for simultaneous calculation, a Nx(M+1) matrix is expected
  409. tS=numpy.zeros((self.n,self.m+1))
  410. #columns from 2..M+1 are the partial derivatives
  411. tS[:,1:]=numpy.transpose(qS)
  412. #first column is the original function
  413. tS[:,0]=self.u(t)
  414. return tS
  415. def fS(self,t):
  416. #M number of sensitivity parameters
  417. #N number of equations
  418. #fSS is MxNxN
  419. #assume a tabulated solution y(t) at t spaced intervals
  420. qS=self.fSS(t).dot(self.getY(t))
  421. return numpy.transpose(qS)
  422. def getSEJ(self,parName):
  423. #find the sensitivity (SE) derivative of Jacobi with
  424. #respect to parameter
  425. try:
  426. return self.seJ[parName]
  427. except KeyError:
  428. self.seJ[parName]={}
  429. return self.seJ[parName]
  430. def getSEJ_comp(self,parName,compartmentName):
  431. #find equation dictating concentration in compartmentName
  432. #for jacobi-parameter derivative
  433. seJ=self.getSEJ(parName)
  434. try:
  435. return seJ[compartmentName]
  436. except KeyError:
  437. seJ[compartmentName]={}
  438. return seJ[compartmentName]
  439. def setY(self,t,y):
  440. self.tck=[None]*self.n
  441. for i in range(self.n):
  442. self.tck[i] = scipy.interpolate.splrep(t, y[:,i], s=0)
  443. def getY(self,t):
  444. fY=numpy.zeros(self.n)
  445. for i in range(self.n):
  446. fY[i]=scipy.interpolate.splev(t, self.tck[i], der=0)
  447. return fY
  448. def getWeight(self,parName):
  449. pars=self.parSetup['parameters']
  450. par=pars[parName]
  451. #self.get parses the units
  452. v=self.get(parName)["value"]
  453. #if par['dist']=='lognormal':
  454. #this is sigma^2_lnx
  455. #sln2=numpy.log(par["cv"]*par["cv"]+1)
  456. #have to multiplied by value to get the derivative
  457. #with respect to lnx
  458. #return sln2*v*v
  459. #else:
  460. #for Gaussian, cv is sigma/value; get sigma by value multiplication
  461. return par["cv"]*par["cv"]*v*v
  462. def getMax(lutSE):
  463. fm=-1
  464. for x in lutSE:
  465. if int(lutSE[x])>fm:
  466. fm=lutSE[x]
  467. return fm
  468. def getWeights(self,lutSE):
  469. #pars=self.parSetup['parameters']
  470. wts=numpy.zeros((model.getMax(lutSE)+1))
  471. for parName in lutSE:
  472. j=lutSE[parName]
  473. wts[j]=self.getWeight(parName)
  474. return wts
  475. def getVolumes(self):
  476. m=numpy.zeros((len(self.lut)))
  477. for p in self.lut:
  478. m[self.lut[p]]=self.getVolume(p)
  479. return m
  480. def getVolume(self,p):
  481. pV=self.getVolumePar(p)
  482. return self.get(pV)['value']
  483. def getDerivatives(self,se,i):
  484. #return latest point derivatives
  485. fse=se[-1][i]
  486. #fse is an m-vector
  487. return fse*fse
  488. def calculateUncertainty(self,se):
  489. s2out=numpy.zeros(se.shape[1:])
  490. se2=numpy.multiply(se,se)
  491. #w=self.getWeights(self.lutSE)
  492. w=numpy.ones((self.m))
  493. return numpy.sqrt(numpy.dot(se2,w))
  494. def get(self,parName):
  495. pars=self.parSetup['parameters']
  496. par=pars[parName]
  497. par['name']=parName
  498. if "value" in par:
  499. return self.getValue(par)
  500. if "function" in par:
  501. return self.getFunction(par)
  502. if "derived" in par:
  503. return self.getDerived(par)
  504. print('Paramter {} not found!'.format(parName))
  505. def getValue(self,par):
  506. v=par["value"]
  507. parName=par['name']
  508. #convert to seconds
  509. try:
  510. parUnits=par['unit'].split('/')
  511. except (KeyError,IndexError):
  512. #no unit given
  513. return valueObject(v,parName)
  514. timeUnit=self.getTimeUnit()
  515. try:
  516. if parUnits[1]==timeUnit:
  517. return valueObject(v,parName)
  518. except IndexError:
  519. #no / in unit name
  520. return valueObject(v,parName)
  521. if parUnits[1]=='min' and timeUnit=='s':
  522. return valueObject(v/60,parName)
  523. if parUnits[1]=='s' and timeUnit=='min':
  524. return valueObject(60*v,parName)
  525. if parUnits[1]=='day' and timeUnit=='min':
  526. return valueObject(v/24/60,parName)
  527. if parUnits[1]=='hour' and timeUnit=='min':
  528. return valueObject(v/60,parName)
  529. #no idea what to do
  530. return valueObject(v,parName)
  531. def getFunction(self,par):
  532. fName=par['function']
  533. #print('[{}]: getFunction({})'.format(par['name'],par['function']))
  534. df=self.parSetup['functions'][fName]
  535. skip=['type']
  536. par1={x:self.get(df[x]) for x in df if x not in skip}
  537. if df['type']=='linearGrowth':
  538. #print(par1)
  539. return function.linearGrowth(par1)
  540. if df['type']=='linearGrowthFixedSlope':
  541. return function.linearGrowthFixedSlope(par1)
  542. print('Function {} not found!'.format(fName))
  543. def getDerived(self,par):
  544. dName=par['derived']
  545. d=self.parSetup['derivedParameters'][dName]
  546. #print('Derived [{}]: type {}'.format(dName,d['type']))
  547. if d['type']=='product':
  548. return function.product(self.get(d['a']),self.get(d['b']))
  549. if d['type']=='power':
  550. return function.power(self.get(d['a']),self.get(d['n']))
  551. if d['type']=='ratio':
  552. return function.ratio(pA=self.get(d['a']),pB=self.get(d['b']))
  553. if d['type']=='sum':
  554. return function.add(pA=self.get(d['a']),pB=self.get(d['b']))
  555. def calculateDerivative(par):
  556. #add derivatives if dist(short for distribution) is specified
  557. return "dist" in par
  558. def valueObject(v,parName):
  559. #convert everything to functions
  560. d0={parName:1}
  561. return {'value':v,'derivatives':{parName:1}}
  562. def splitVector(v):
  563. if v.find('(')<0:
  564. return [v]
  565. return v[1:-1].split(',')
  566. def addValue(qdict,compName,v):
  567. #add function to compName of dictionary qdict,
  568. #check if compName exists and handle the potential error
  569. #lambda functions can't be summed directly, so qdict is a list
  570. #that will be merged at matrix generation time
  571. try:
  572. qdict[compName].append(v)
  573. except KeyError:
  574. qdict[compName]=[v]
  575. #also add derivatives
  576. #
  577. # for d in dTarget:
  578. # ctarget=self.getSEJ_comp(d,target)
  579. # addValue(ctarget,target,dTarget[d])
  580. def get(timeUnit,par):
  581. v=par["value"]
  582. #convert to seconds
  583. try:
  584. parUnits=par['unit'].split('/')
  585. except (KeyError,IndexError):
  586. #no unit given
  587. return v
  588. try:
  589. if parUnits[1]==timeUnit:
  590. return v
  591. except IndexError:
  592. #no / in unit name
  593. return v
  594. if parUnits[1]=='min' and timeUnit=='s':
  595. return v/60
  596. if parUnits[1]=='s' and timeUnit=='min':
  597. return 60*v
  598. #no idea what to do
  599. return v