Versions Compared

Key

  • This line was added.
  • This line was removed.
  • Formatting was changed.

Here is an example of a script as a local native python task:

There are several ways of doing this. The following are examples.

  •  There are two ways of accessing the ecflow child commands(init,event,meter, label,abort,complete). 
     We can either call the child commands directly using ecflow extension, or we can call system command to access ecflow_clent
     The following examples will use ecflow python extension. However this requires that PYTHONPATH is set directory where ecflow.so extension was installed.
  • definition file:
    The default ECF_MICRO is %, this may interfere with your python scripts. In this case you either redefine it, in the task definition or directly in the python script.
    task python
      edit ECF_MICRO '^$'
      edit ECF_JOB_CMD '^ECFpython $ECF_JOB^JOB$ > ^ECF$ECF_JOBOUT^JOBOUT$ 2>&1'
      label info "none"
      meter step -1 100 100
      event 1
      event 2
  • task wrapper:Header

    Code Block
    languagepy
    titlepython header
    import os, time, signal, sys
    import ecflow
     
    print "PYTHONPATH====================================================="
    print os.environ['PYTHONPATH'].split(os.pathsep)
    class Client(object):
       def __init__(self#!/usr/env python
    #python.ecf
    ^include <head.py>
    ^manual
    one liner manual ...
    ^end
    
    ^comment
    one liner
    ^end
    
    xlabel("info", "start")
    xevent("1")
    
    for step in range(0,101):
          print "Creating stepClient"
          xmeter("step", step)
    
    xevent("2")
    xlabel("info", "completion")
    ^include <tail.py>

    headers:

    Code Block
    #!/usr/local/bin/python
    # head.py
    import os, sys, signal
    
    ECF_PORT = ^ECF_PORT:0^
    XSVR = "ecflow_client --port %s --host ^ECF_NODE:0^ --" % ECF_PORT
    pid =  self.ci = ecflow.Client()
          self.ci.set_host_port("%ECF_NODE%","%ECF_PORT%")
          self.ci.set_child_pid(os.getpid())
    
    def xinit(): 
        os.system(XSVR + "init %s"%pid)
    def xabort(): 
    self.ci.set_child_path("%ECF_NAME%")
          os.system(XSVR + "abort %s"%pid)
    def xcomplete(): 
    self.ci.set_child_password("%ECF_PASS%")
          os.system(XSVR + "complete")self.ci.set_child_try_no(%ECF_TRYNO%)
       
    def  SigHandler(signum, frame): 
      print  xabort(); sys.exit(0)
    
    signal.signal (signal.SIGINT,  SigHandler); # ...
    os.environ['ECF_PORT'] = "^ECF_PORT:0^"
    os.environ['ECF_NAME'] = "^ECF_NAME:0^"
    os.environ['ECF_NODE'] = "^ECF_NODE:0^"
    os.environ['ECF_PASS'] = "^ECF_PASS:0^"
    
    def xmeter(name, step):
       os.system(XSVR + "meter %s %s"%(name,step))  
    def xevent(name):
       os.system(XSVR + "event %s"%name)  
    def xlabel(name, info):
       os.system(XSVR + "label %s %s"%(name,info))
    
    signal.signal "Only wait 20 seconds, if the server cannot be contacted (note default is 24 hours) before failing"
          self.ci.set_child_timeout(20) 
        
          signal.signal(signal.SIGINT,  self.signal_handler) 
          signal.signal(signal.SIGHUP,  SigHandlerself.signal_handler)
          signal.signal (signal.SIGQUIT, SigHandlerself.signal_handler)
          signal.signal (signal.SIGILL,  SigHandlerself.signal_handler)
          signal.signal (signal.SIGTRAP, SigHandlerself.signal_handler)
          signal.signal (signal.SIGIOT,  SigHandlerself.signal_handler)
          signal.signal (signal.SIGBUS,  SigHandlerself.signal_handler)
          signal.signal (signal.SIGFPE,  SigHandlerself.signal_handler)
          signal.signal (signal.SIGUSR1, SigHandlerself.signal_handler)
          signal.signal (signal.SIGUSR2, SigHandlerself.signal_handler)
          signal.signal (signal.SIGPIPE, SigHandlerself.signal_handler)
          signal.signal (signal.SIGTERM, SigHandlerself.signal_handler)
          signal.signal (signal.SIGXCPU, SigHandlerself.signal_handler)
          signal.signal (signal.SIGPWR,  SigHandler)
    
    print 'start'
    xinit()self.signal_handler)
        
       def signal_handler(self,signum, frame):
          print 'Aborting: Signal handler called with signal ', signum
          self.ci.child_abort("Signal handler called with signal " + str(signum));
        
       def __enter__(self):
          self.ci.child_init()
          return self.ci
        
       def __exit__(self,ex_type,value,tb):
          print "Client:__exit__: ex_type:" + str(ex_type) + " value:" + str(value) + "\n" + str(tb)
          if ex_type != None:
             self.ci.child_abort("Aborted with exception type " + str(ex_type) + str(value))
          self.ci.child_complete()
          return False

    This header will automatically handle calling the child command init() and complete(), it will also handle any signals, and exceptions. In which case it call abort automatically

    task wrapper:

    Code Block
    languagepython
    $include <head.py>
    $manual
       This is the manual section. Instead of calling python from the ECF_JOB_CMD we could alternatively place,
           #!/bin/env/python
       on the first line of this file.
    $end
    
    $comment
       Note: We do not need a tail,
    $end
     
    # This will also handle call to sys.exit(), i.e Client.__exit__ will still be called.
    with Client() as ci:
       for i in range(1,100):
          ci.child_meter('step',i)
          ci.child_label('info', "value_" + str(i))
          time.sleep(1)
          
       ci.child_event('event_fred')
       print "Finished event,meter and label child commands"
  •  


     

     

     

    Code Block# tail.py xcomplete()