1   
  2   
  3   
  4   
  5   
  6   
  7   
  8   
  9   
 10   
 11   
 12   
 13   
 14   
 15   
 16   
 17   
 18   
 19   
 20  """Enables creation of build command list in several formats. 
 21   
 22  This module implements class that represent shell commands. 
 23  It supports build stage and command parallelization (depends of the output format). 
 24  CommandList can be generated in different format: ant, make, ebs, batch. 
 25   
 26  Example: 
 27  from mc.buildtools import CommandList, Convert 
 28  list = CommandList() 
 29  list.addCommand("\\epoc32\\rombuild", "make_fpsx.bat..yy...", "build_xx_rom") 
 30  list.addCommand("\\epoc32\\rombuild", "make_fpsx.bat..xx...", "build_yy_rom") 
 31  list.addCommand("\\epoc32\\rombuild", "copy \\foo \\bar", "simple copy", False) 
 32   
 33  convert(list, "outputfile.mk", "make") 
 34  convert(list, "outputfile.ant.xml", "ant") 
 35  convert(list, "outputfile.ebs.xml", "ebs") 
 36  convert(list, "outputfile.bat", "bat") 
 37   
 38  """ 
 39  import os 
 40  import types 
 41  import xml.dom.minidom 
 42  import sys 
 43   
 45      """ This class implements an abstract prebuilder. 
 46          A prebuilder takes a configurationset as input and generates a build file. 
 47      """ 
 52   
 54          """ Converting a task list into output format and writing it into buildFilePath file. """ 
 55          writer = None 
 56           
 57               
 58           
 59           
 60           
 61           
 62          print self.config.keys() 
 63          buildFileDir = os.path.dirname(buildFilePath) 
 64          if not os.path.exists(buildFileDir): 
 65              os.makedirs(buildFileDir) 
 66          writer = get_writer(output, open(buildFilePath, 'w')) 
 67          writer.write(taskList) 
   68   
 70      """ Abstract Task object. """ 
 71      pass 
  72           
 74      """ 
 75          This class implements a command definition. 
 76          It handles command id and stage. 
 77          All command from one stage should be finished before starting the next stage. 
 78      """ 
 79 -    def __init__(self, executable, path, args=None, name=''): 
  80          Task.__init__(self) 
 81          if args == None: 
 82              args = [] 
 83          self._id  = 1 
 84          self._stage = 1 
 85          self._name = name 
 86          self._executable = executable 
 87          self._path = path 
 88          self._args = args 
  89   
 91          """ Set the command id. """ 
 92          self._id = idn 
  93   
 95          """ Set the command stage. """ 
 96          self._stage = stage 
  97   
 99          """ Get the command id. """ 
100          return self._id 
 101   
103          """ Get the command stage. """ 
104          return self._stage 
 105   
107          """ Get the command name. """ 
108          return self._name 
 109   
111          """ Get the command executable. """ 
112          return self._executable 
 113   
115          """ Get the command path. """ 
116          return self._path 
 117   
119          """ Get the command line. """ 
120          return ' '.join(self._args) 
 121   
123          """ Add a command line argument. """ 
124          self._args.append(arg) 
 125   
127          argsString = ' '.join(self._args) 
128          return "%s: %s: %s" % (self.name(), self.path(), argsString) 
  129   
131      """ Interface that defines supports for an Ant task rendering. """ 
132       
134          """ Override this method to convert a specific command into Ant command. 
135              e.g: Delete Class will use delete task from Ant, else convert into perl ... remove filename.__getCommandByStage 
136          """  
137          pass 
  138       
139 -class Delete(AntTask, Command): 
 140      """ Implements file/directory deleletion mechanism. """ 
141       
142 -    def __init__(self, filename=None, dirname=None): 
 143          Command.__init__(self, "perl", "") 
144          AntTask.__init__(self) 
145          self._filename = filename 
146          self._dir = dirname 
147          self._args.append("-MExtUtils::Command") 
148          self._args.append("-e") 
149          if self._filename != None: 
150              self._args.append("rm_f") 
151              self._args.append('"' + self._filename + '"') 
152          elif self._dir != None: 
153              self._args.append("rm_rf") 
154              self._args.append('"' + self._dir + '"') 
 155   
157          """ Render the delete as an Ant task. """ 
158          node = doc.createElementNS("", "delete") 
159          node.setAttributeNS("", "verbose", "true") 
160          node.setAttributeNS("", "failonerror", "false") 
161          if self._filename != None: 
162              node.setAttributeNS("", "file", self._filename) 
163          elif self._dir != None: 
164              node.setAttributeNS("", "dir", self._dir) 
165          return node 
  166   
167   
168 -class Copy(AntTask, Command): 
 169      """ Implement copy command. """ 
171          Command.__init__(self, "perl", os.path.dirname(srcFile)) 
172          AntTask.__init__(self) 
173          self.srcFile = srcFile 
174          self.todir = todir 
175          self._args.append("-MExtUtils::Command") 
176          self._args.append("-e") 
177          self._args.append("cp") 
178          self._args.append('"' + self.srcFile + '"') 
179          self._args.append('"' + os.path.join(self.todir, os.path.basename(self.srcFile)) + '"') 
 180           
182          """ Render the copy as an Ant task. """ 
183          node = doc.createElementNS("", "copy") 
184          node.setAttributeNS("", "verbose", "true") 
185          node.setAttributeNS("", "failonerror", "false") 
186          node.setAttributeNS("", "file", self.srcFile) 
187          node.setAttributeNS("", "todir", self.todir) 
188          return node 
  189            
190   
192      """ 
193          This class allows to safely handle Command object into lists 
194      """ 
197   
199          """ Returns all command list. """ 
200          return self.__cmds 
 201   
 215   
216   
218      """Base class which contains define an AbstractOutputWriter. 
219   
220      The subclass must implement a convert method which compute a command list into 
221      some output file. 
222      """ 
224          if isinstance(fileOut, types.StringType): 
225              self._fileOut = open(fileOut, 'w') 
226          else: 
227              self._fileOut = fileOut 
 228   
229 -    def write(self, cmdList): 
 230          """ Method to override to implement format specific output. """ 
 231 -    def writeTopLevel(self, config_list, spec_name, output_path, xml_file): 
 232          """ Method to override to implement top level commands. """ 
 233   
236   
238          """ Close the output stream. """ 
239          self._fileOut.close() 
  240   
241   
243      """ Implements a Writer which is able to directly write to the output stream. """ 
244       
247   
248 -    def write(self, content): 
 249          """ Write content to the output. """ 
250          self._fileOut.write(content) 
  251   
252   
254      """ Implements EBS XML output format. """ 
255       
258   
259 -    def write(self, cmdList): 
 260          """ Write the command list to EBS format. """ 
261          doc = xml.dom.minidom.Document() 
262          productnode = doc.createElementNS("", "Product") 
263          cmdsnode = doc.createElementNS("", "Commands") 
264          productnode.appendChild(cmdsnode) 
265          doc.appendChild(productnode) 
266   
267          for cmd in cmdList.allCommands(): 
268              cmdsnode.appendChild(self.__commandToXml(doc, cmd)) 
269   
270          self._fileOut.write(doc.toprettyxml()) 
 271   
272      @staticmethod 
274          """ Convert a Command into an EBS command. """ 
275           
276          cmdsnode = doc.createElementNS("", "Execute") 
277          cmdsnode.setAttributeNS("", "ID", "%d" % cmd.jobId()) 
278          cmdsnode.setAttributeNS("", "Stage", "%d" % cmd.stage()) 
279          cmdsnode.setAttributeNS("", "Component", cmd.name()) 
280          cmdsnode.setAttributeNS("", "Cwd", cmd.path()) 
281          cmdsnode.setAttributeNS("", "CommandLine", cmd.executable()+" "+cmd.cmd()) 
282          return cmdsnode 
  283   
284   
286      """ Implements Ant XML output format. """ 
287       
290   
291 -    def writeTopLevel(self, config_list, spec_name, output_path, xml_file): 
 292          doc = xml.dom.minidom.Document() 
293          projectnode = doc.createElementNS("", "project") 
294          projectnode.setAttributeNS("", "name", '') 
295          projectnode.setAttributeNS("", "default", "all") 
296          doc.appendChild(projectnode) 
297          target = doc.createElementNS("", "target") 
298          target.setAttributeNS("", "name", "all") 
299          projectnode.appendChild(target) 
300   
301          parallel = doc.createElementNS("", "parallel") 
302          parallel.setAttributeNS("", "threadCount", "${number.of.threads}") 
303          target.appendChild(parallel) 
304          index = 0 
305          script_loc = os.path.normpath(os.path.join(os.environ['HELIUM_HOME'], 'tools/common/python/lib/CreateZipInput.py')) 
306          for config in config_list: 
307              sequential = doc.createElementNS("", "sequential") 
308              outputfile = os.path.normpath(os.path.join(output_path, config + ".xml")) 
309              exec_element = doc.createElementNS("", "exec") 
310              exec_element.setAttributeNS("", "executable", "python") 
311   
312              args = doc.createElementNS("", "arg") 
313              args.setAttributeNS("", "value", "%s" % script_loc) 
314              exec_element.appendChild(args) 
315   
316              args = doc.createElementNS("", "arg") 
317              args.setAttributeNS("", "line", "--output=%s" % outputfile) 
318              exec_element.appendChild(args) 
319              args = doc.createElementNS("", "arg") 
320              args.setAttributeNS("", "line", "--config=%s" % spec_name) 
321              exec_element.appendChild(args) 
322              args = doc.createElementNS("", "arg") 
323              args.setAttributeNS("", "line", "--filename=%s" % xml_file) 
324              exec_element.appendChild(args) 
325              args = doc.createElementNS("", "arg") 
326              args.setAttributeNS("", "line", "--id=%d" % index) 
327              exec_element.appendChild(args) 
328              args = doc.createElementNS("", "arg") 
329              args.setAttributeNS("", "line", "--writertype=ant") 
330              exec_element.appendChild(args) 
331              sequential.appendChild(exec_element) 
332              index += 1 
333              ant_exec = doc.createElementNS("", "ant") 
334              ant_exec.setAttributeNS("", "antfile", outputfile) 
335              sequential.appendChild(ant_exec) 
336              parallel.appendChild(sequential) 
337           
338          self._fileOut.write(doc.toprettyxml()) 
 339 -    def write(self, cmdList): 
 340          """ Writes the command list to Ant format. """ 
341          doc = xml.dom.minidom.Document() 
342          projectnode = doc.createElementNS("", "project") 
343          projectnode.setAttributeNS("", "name", '') 
344          projectnode.setAttributeNS("", "default", "all") 
345          doc.appendChild(projectnode) 
346   
347          stages = self.__getCommandByStage(cmdList) 
348   
349          for stage in stages.keys(): 
350              projectnode.appendChild(self.__stageToTarget(doc, stage, stages[stage])) 
351   
352          target = doc.createElementNS("", "target") 
353          target.setAttributeNS("", "name", "all") 
354          def __toStage(stage): 
355              """ Convert the stage id into and Ant target name. """ 
356              return "stage%s" % stage 
 357          target.setAttributeNS("", "depends", ','.join([__toStage(stage) for stage in stages.keys()])) 
358          projectnode.appendChild(target) 
359   
360          self._fileOut.write(doc.toprettyxml()) 
 361   
363          """ Convert a stage into an Ant target. """ 
364          target = doc.createElementNS("", "target") 
365          target.setAttributeNS("", "name", "stage%s" % stage) 
366          parallel = doc.createElementNS("", "parallel") 
367          parallel.setAttributeNS("", "threadCount", "${number.of.threads}") 
368          target.appendChild(parallel) 
369   
370          for cmd in cmds: 
371              parallel.appendChild(self.__commandToAnt(doc, cmd)) 
372          return target 
 373   
374      @staticmethod 
376          """ Convert a command into an Ant task. """ 
377           
378           
379          if issubclass(type(cmd), AntTask): 
380              return cmd.toAntTask(doc) 
381          else: 
382              execnode = doc.createElementNS("", "exec") 
383              execnode.setAttributeNS("", "executable", cmd.executable()) 
384              execnode.setAttributeNS("", "dir", cmd.path()) 
385              arg = doc.createElementNS("", "arg") 
386              arg.setAttributeNS("", "line", cmd.cmd()) 
387              execnode.appendChild(arg) 
388              return execnode 
 389   
390      @staticmethod 
392          """ Reorder a CommandList into a list of stages. """ 
393          stages = {} 
394          for cmd in cmdList.allCommands(): 
395              if not stages.has_key(cmd.stage()): 
396                  stages[cmd.stage()]=[] 
397              stages[cmd.stage()].append(cmd) 
398   
399          return stages 
 400   
401   
403      """ Implements Makefile writer. """ 
404       
407   
408 -    def writeTopLevel(self, config_list, spec_name, output_path, xml_file): 
 409          content = "\n\nall: zip_inputs zip_files\n\n" 
410          index = 0 
411          input_list = "zip_inputs: " 
412          zip_list = "\n\nzip_files: " 
413          full_content = "" 
414          script_path =  os.path.normpath(os.path.join(os.environ['HELIUM_HOME'], 'tools/compile/ec')) 
415          for config in config_list: 
416              outputfile = os.path.normpath(os.path.join(output_path, config + ".mk")) 
417              input_list += " \\\n\t zip_input%d" % index 
418              zip_list += " \\\n\t zip_files%d" % index 
419              content += "\n\nzip_input%d :\n" % index 
420              content += "\t@echo === identifying files for %s\n" % config 
421               
422              content += "\tpython $(HELIUM_HOME)\\tools\\common\\python\\lib\\CreateZipInput.py --config=%s --filename=%s --id=%d --output=%s --writertype=%s\n\n" % (spec_name, xml_file, index, outputfile,'make') 
423              content += "\n\nzip_files%d :zip_input%d\n" % (index, index) 
424              content += "\t@echo === identifying files for %s\n" % config 
425              content += "\t$(MAKE) -f %s" % (outputfile) 
426              index += 1 
427           
428          full_content += input_list 
429          full_content += zip_list 
430          full_content += content 
431          self._fileOut.write(full_content) 
 432 -    def write(self, cmdList): 
 433          """ Converts the list of command into Makefile. """ 
434          stages = {} 
435          for cmd in cmdList.allCommands(): 
436              if not stages.has_key(cmd.stage()): 
437                  stages[cmd.stage()] = [] 
438              stages[cmd.stage()].append(cmd) 
439           
440           
441          def __toStage(stage): 
442              """ Convert stage Id into a target name. """ 
443              return "stage%s" % stage 
 444                   
445           
446          if len(stages.keys()) > 0: 
447              self._fileOut.write("all : stage%s ;\n" % max(stages.keys())) 
448          else: 
449              self._fileOut.write("all: ;\n") 
450               
451          for stage in stages.keys(): 
452               
453              def __toId(cmd): 
454                  """ Convert command Id into a target name. """ 
455                  self.__commandToTarget(cmd) 
456                  return "id%s" % cmd.jobId() 
 457              self._fileOut.write("stage%s : %s\n" % (stage, ' '.join([__toId(task) for task in stages[stage]]))) 
458           
459   
461          """ Converting a Command into a Makefile target. """ 
462          deps = "" 
463          if cmd.stage() > 1: 
464              deps = " stage%s" % (cmd.stage() - 1) 
465          self._fileOut.write("id%s:%s\n" % (cmd.jobId(), deps)) 
466          self._fileOut.write("\t@echo Target %s\n" % cmd.name()) 
467          winargs = "" 
468          if sys.platform == "win32": 
469              winargs = "/d" 
470          self._fileOut.write("\tcd %s %s && %s " % (winargs, cmd.path(), cmd.executable())) 
471          self._fileOut.write("%s\n" % cmd.cmd()) 
472          self._fileOut.write("\n") 
 473   
474   
475  __writerConstructors = { 'ant': AntWriter, 
476                           'make': MakeWriter, 
477                           'ebs': EBSWriter } 
478   
479 -def convert(cmdList, filename, outputtype="ant"): 
 480      """ Helper to directly convert a command list into a specific runnable command format. 
481          e.g: 
482          cmdList = CommandList() 
483          cmdList.addCommand(...) 
484          convert(cmdList, "echo Hello world", "ant") 
485      """ 
486      writer = __writerConstructors[outputtype](filename) 
487      writer(cmdList) 
 488   
492   
493   
497