Source code for pycropml.transpiler.antlr_py.dssat.run

from __future__ import absolute_import
from __future__ import print_function
import os
from pycropml.composition import Description
from pycropml.transpiler.antlr_py.to_CASG import to_CASG, to_dictASG
from pycropml.transpiler.antlr_py.fortran.fortranExtraction import FortranExtraction
from pycropml.transpiler.ast_transform import transform_to_syntax_tree
from pycropml.transpiler.antlr_py.generateCyml import writeCyml
from pycropml.transpiler.antlr_py.fortran import f90_cyml
from pycropml.transpiler.antlr_py.cmake.cmakeTransformer import retrievefiles
from collections import OrderedDict
from pycropml.transpiler.antlr_py.extraction import ExtractComments   
from pycropml.transpiler.antlr_py.codeExtraction import extraction
from pycropml.transpiler.antlr_py.fortran.fortran_preprocessing import Declarations, Attr, Assignment, Call_stmt, Implicit_return, Local
from pycropml.transpiler.antlr_py.to_specification import extractMetaInfo, createObjectModel, extractcomments, createObjectCompo
from pycropml.transpiler.antlr_py.createXml import Pl2Crop2ml, create_repo

import tempfile
import copy

types = ["float", "str", "list", "array", "int", "boolean"]

model_tags = ["!%%CyML Model Begin%%", "!%%CyML Model End%%"]
init_tags=["!%%CyML Init Begin%%", "!%%CyML Init End%%"]
rates_tags = ["!%%CyML Rate Begin%%", "!%%CyML Rate End%%"]
states_tags = ["!%%CyML State Begin%%", "!%%CyML State End%%"]
compute_tags = ["!%%CyML Compute Begin%%", "!%%CyML Compute End%%"]
ignore_tags = ["!%%CyML Ignore Begin%%", "!%%CyML Ignore End%%"]
composition_tags = ["!%%CyML Composition Begin%%", "!%%CyML Composition End%%"]

start_linecom = ["!", ["C", 1]] # if item is a list: first is the tag and scond is the porition of the tag
default_mltCom = ['%%%%', '%%%%%'] # use '%%%%' if the language didn't provide multi lines comments

language = 'f90'

endconstruct = "end"

endline = "\n"

cyml_ext = ".pyx"

output_folder = "temp"

[docs] def translate(node, asgt, imports, inout=[], index=[]): """Transform specific nodes based on class of subnodes of node. It also allows to extract some usefull information. At finish the modified node contains only the constructs of CyML and converted in CyML after applying translate_simple function. Args: node (Node): A subtree of asgt asgt (Node): A tree imports (str): Names of the import Modules used in the code from where the node is generated inout (list, optional): Names of the inputs and outputs of the ModelUnit. Defaults to []. Returns: files: init, algo and external functions files """ attrib = Attr(asgt,imports=imports) node_w_attrib = attrib.process(node) ass = Assignment() node2=ass.process(node_w_attrib) l = Local() res = l.process(node2) pp = l.localvar.difference(set(inout)) # remove input output declarations #zz = pp.intersection(ass.targets) ### to remove local variables that are not used as targets in an assignment if index: zz = pp.union(set(index)) else: zz=pp e = Declarations(localvar=list(zz)) ## pp by zz f = e.process(res) g = Call_stmt(trees=asgt) h = g.process(f) z = Implicit_return() r = z.process(h) return r
[docs] def translate_simple(r, total_tree=None, var = []): """ Transform the ASG of Fortran code in the Common Model Representation that is transformed into CyML Args: r (Node): The modified ASG of Fortran Returns: str: The generated CyML code """ cd = f90_cyml.F90_Cyml_ast(r, treeG=total_tree, var = var) hh = cd.transform() nd = transform_to_syntax_tree(hh) codes = writeCyml(nd) return codes
[docs] def fortrancomments(code ): """ Extract comments inside Fortran code """ return extractcomments(code, ["!", ["C", 1]])
""" Read DSSAT component and infer Initializations, Algorithms and External Functions """
[docs] def run_dssat(component, package): create_repo(package) # Read Cmake files !!!!!!!! modu=os.path.join(component, "CMakeLists.txt") with open(modu, "r") as f: cmakecode = f.read() # Parse Cmake code to retrieve source files files = retrievefiles(cmakecode) fp = tempfile.TemporaryFile(mode="w+t",encoding="utf8") # Merge files in a temporary file for f in files: fil = open(os.path.join(component, f), "r") fp.write(fil.read()) fp.write("\n") fil.close() fp.seek(0) #print(fp.read()) #!!!! # Create the Fortran ASG of the merged files code = fp.read() d_sorted = fortrancomments(code) dictasgt = to_dictASG(code,language, comments=d_sorted) asgt = to_CASG(dictasgt) fp.close() extr = FortranExtraction() # Find the not required functions or subroutines notreq = extr.notRequiredFunc(asgt) for f in files: with open(os.path.join(component,f), 'r') as text: mod = text.read() #comments = fortrancomments(mod) res = extraction(mod, model_tags[0],model_tags[1],ignore_tags[0],ignore_tags[1]) if res: models = [] totalcomments = fortrancomments(mod) nn=0 description = {"Title":"model of soil", "Authors":"Cyrille", "Institution":"INRAE" } head = {} for com in totalcomments: if totalcomments[com].startswith("C="): nn = nn+1 if nn==3: model_name = totalcomments[com+1][1:].strip().split(",")[0] head["name"] = model_name description["Title"] = f"Model of {model_name}" continue if totalcomments[com][1:].strip()=="@ShortDescription": shortdescription=[] while totalcomments[com+1]!="C": shortdescription.append(totalcomments[com+1][1:].strip()) com = com + 1 description["ShortDescription"]=" ".join(shortdescription) continue if totalcomments[com][1:].strip()=="@ExtendedDescription": extdescription=[] while totalcomments[com+1].strip()!="C": extdescription.append(totalcomments[com+1][1:].strip()) com = com + 1 description["ExtendedDescription"]=" ".join(extdescription) continue if totalcomments[com][1:].strip().startswith("@Timestep"): timestep = totalcomments[com][1:].strip().split("@Timestep")[-1].split("day")[0] head["timestep"] = timestep continue if totalcomments[com][1:].strip().startswith("@Version"): version = totalcomments[com][1:].strip().split("@Version")[-1].split("day")[0] head["version"]=version continue if totalcomments[com][1:].strip().startswith("@Authors"): auth = [] while totalcomments[com+1].strip()!="C": auth.append(totalcomments[com+1][1:].strip().replace(")", "")) com = com + 1 description["Authors"] = ", ".join([n.split("(")[0] for n in auth]) description["Institution"] = ", ".join([n.split("(")[-1] for n in auth]) continue if totalcomments[com][1:].strip().startswith("@Reference"): reference = totalcomments[com+1][1:].strip().split("@Reference")[-1].split("day")[0] description["Reference"] = reference break modunit_dictasg = to_dictASG(res[0] , language) modunit_asg = to_CASG(modunit_dictasg) imports = modunit_asg[0].imports attrib = Attr(asgt,imports) node_w_attrib = attrib.process(modunit_asg[0]) zz =attrib.decls attrname = [] newinputs = [] for d in zz: attrname.append(d.decl[0].name) newinputs.append(d.decl[0]) inout = modunit_asg[0].inputs_name + modunit_asg[0].outputs_name + attrname ass = Assignment() node2=ass.process(node_w_attrib) l = Local() res1 = l.process(node2) pp = l.localvar.difference(set(inout)) # remove input output declarations input1 = pp.difference(ass.targets) ### to get local variables that are used as model inputs decl=[] inputs1=[] for m in modunit_asg[0].block: if m and m.type=="declaration": decl.append(m) for inp in m.decl: if inp.name in input1 and inp.name not in modunit_asg[0].indexnames : inputs1.append(inp) initialization = extraction(res[0] , init_tags[0],init_tags[1]) ratecalculation = extraction(res[0] , rates_tags[0],rates_tags[1]) statecalculation = extraction(res[0] , states_tags[0],states_tags[1]) algoPart = extraction(res[0], compute_tags[0],compute_tags[1]) varnotdeclared = modunit_asg[0].notdeclared extr2 = FortranExtraction() if varnotdeclared: res = extr2.getDecl(asgt, imports, varnotdeclared) otherparams = list(res.values()) else: otherparams=[] model_inputs = [m for m in otherparams + newinputs + modunit_asg[0].inputs+ inputs1 if m.type in types] model_outputs = modunit_asg[0].outputs inout = list(set([m.name for m in model_inputs + model_outputs ])) metainfo_init = {} initv = False algop = False ratep = False statep = False init_outputs = [] v = [] if algoPart: algop = True algoPart = algoPart[0] + endline + endconstruct comments_compute = fortrancomments(algoPart) algoPart_dictasg = to_dictASG(algoPart, language,comments_compute, env = modunit_asg[0].env) algoPart_asg = to_CASG(algoPart_dictasg) codes_compute = translate(decl+algoPart_asg,asgt,imports,inout=inout, index = modunit_asg[0].indexnames) if ratecalculation: ratep = True ratecalculation = ratecalculation[0] + endline + endconstruct comments_rate = fortrancomments(ratecalculation) ratecalculation_dictasg = to_dictASG(ratecalculation, language,comments_rate, env = modunit_asg[0].env) ratecalculation_asg = to_CASG(ratecalculation_dictasg) g = Call_stmt(trees=asgt) h = g.process(ratecalculation_asg) from pycropml.transpiler.antlr_py.csharp.csharp_preprocessing import CheckingInOut zz = CheckingInOut( {},isAlgo = True) r_ch = zz.process(h) index = list(set(modunit_asg[0].indexnames).intersection(set(zz.inputs) )) r = translate(decl + ratecalculation_asg,asgt,imports,inout=inout, index = index) var = list(set(zz.inputs) - set(index)) print(var) x = list(set(var)-set([m.name for m in model_inputs])) for i in x: v.append(extr2.getDeclaration(asgt, i)) codes_rates = translate_simple(r, total_tree=asgt, var = var) if statecalculation: statep = True statecalculation = statecalculation[0] + endline + endconstruct comments_state = fortrancomments(statecalculation) statecalculation_dictasg = to_dictASG(statecalculation, language,comments_state, env = modunit_asg[0].env) statecalculation_asg = to_CASG(statecalculation_dictasg) r, codes_states = translate(decl+statecalculation_asg,asgt,imports,inout=inout, index = modunit_asg[0].indexnames) if initialization: initv = True metainfo_init = {"name": f"init.{modunit_asg[0].name}", "language":"Cyml", "filename":f"algo/pyx/init.{modunit_asg[0].name}.pyx"} initialization = initialization[0] +endline + endconstruct comments_init = fortrancomments(initialization) initialization_dictasg = to_dictASG(initialization, language,comments_init, env = modunit_asg[0].env) initialization_asg = to_CASG(initialization_dictasg) from pycropml.transpiler.antlr_py.csharp.csharp_preprocessing import CheckingInOut zz = CheckingInOut( {},isAlgo = True) r_ch = zz.process(initialization_asg) index = list(set(modunit_asg[0].indexnames).intersection(set(zz.inputs) )) init_outputs = zz.outputs r = translate(decl + initialization_asg,asgt,imports,inout=inout, index = index) #var = list(set(zz.inputs) - set(index)) var = [n.name for n in model_inputs+v] codes_init = translate_simple(r, total_tree=asgt, var=var) metainfo = extractMetaInfo(mod, "!") inp_info = [] out_info = [] metainfo2 = copy.deepcopy(metainfo) mm = [] for inp in model_inputs+v: res = {} if inp.name in metainfo.keys() and inp.name not in mm: res["name"] = inp.name mm.append(inp.name) var = metainfo[inp.name] if var["type"] == "parameter": var["parametercategory"] = var["category"] del var["category"] if var["type"] == "variable": var["variablecategory"] = var["category"] del var["category"] var["inputtype"] = var["type"] del var["type"] if (isinstance(inp.pseudo_type, list) and inp.pseudo_type[0]!="array") or not isinstance(inp.pseudo_type, list): del var["len"] var["datatype"] = transform_type(inp.pseudo_type) res.update(var) inp_info.append(res) for out in model_outputs: res = {} if out.name in metainfo2.keys(): res["name"] = out.name var = metainfo2[out.name] var["variablecategory"] = var["category"] del var["category"] del var["type"] if (isinstance(out.pseudo_type, list) and out.pseudo_type[0]!="array") or not isinstance(out.pseudo_type, list): del var["len"] var["datatype"] = transform_type(out.pseudo_type) res.update(var) out_info.append(res) exterfunc = extr.externFunction(modunit_asg[0]) # external functions extfunc = exterfunc.difference(notreq) for ext in extfunc: func = extr.getSubroutine(asgt, ext) extcode = translate_simple(func, total_tree=asgt) exfunc = os.path.join(package, "crop2ml", "algo", "pyx", ext + ".pyx") with open(exfunc, "w") as fi: fi.write(extcode + '\n') model = createObjectModel(head, description, inp_info, out_info, metainfo_init, extfunc) models.append(model) if initv: out_init = os.path.join(package, "crop2ml", "algo", "pyx", "init."+f.split(".")[0] + ".pyx") with open(out_init, "w") as fi: fi.write(codes_init+'\n') if algop: out_compute = os.path.join(package, "crop2ml", "algo", "pyx", f.split(".")[0] + ".pyx") with open(out_compute, "w") as fi: fi.write(codes_compute + '\n') else: if ratep and not statep: out_rates = os.path.join(package, "crop2ml", "algo", "pyx", f.split(".")[0] + ".pyx") with open(out_rates, "w") as fi: fi.write(codes_rates+'\n') if statep and not ratep: out_states = os.path.join(package, "crop2ml", "algo", "pyx", f.split(".")[0] + ".pyx") with open(out_states, "w") as fi: fi.write(codes_states+'\n') xml_ = Pl2Crop2ml(model, "DSSAT_").run_unit() filename = os.path.join(package, "crop2ml", "unit.%s.xml"%(model.name)) with open(filename, "wb") as xml_file: r = '<?xml version="1.0" encoding="UTF-8"?>\n' r += '<!DOCTYPE ModelUnit PUBLIC " " "https://raw.githubusercontent.com/AgriculturalModelExchangeInitiative/crop2ml/master/ModelUnit.dtd">\n' r += xml_.unicode(indent=4)#.encode('utf-8') xml_file.write(r.encode()) res_compo = extraction(code, composition_tags[0],composition_tags[1],ignore_tags[0],ignore_tags[1]) if not res_compo: description["name"]=head["name"]+"_" description["version"]=head["version"] description["timestep"]=head["timestep"] description["url"] = "" mc = createObjectCompo(description,models) else: mc = {} #TODO xml_ = Pl2Crop2ml(mc, "DSSAT_").run_compo() filename = os.path.join(package, "crop2ml", "composition.%s.xml"%(model.name)) with open(filename, "wb") as xml_file: r = '<?xml version="1.0" encoding="UTF-8"?>\n' r += '<!DOCTYPE ModelComposition PUBLIC " " "https://raw.githubusercontent.com/AgriculturalModelExchangeInitiative/crop2ml/master/ModelComposition.dtd">\n' r += xml_.unicode(indent=4)#.encode('utf-8') xml_file.write(r.encode())
dicttype = {"str":"STRING", "int":"INT", "float":"DOUBLE"}
[docs] def transform_type(pseudo_type): if isinstance(pseudo_type, list): newtype = dicttype[pseudo_type[-1]] + pseudo_type[0].upper() else: newtype = dicttype[pseudo_type] return newtype