Saturday, 14 May 2011

PyRun: An advanced process runner with IronPython extension

I have been using mercurial quite a bit for source control, in this article Mercurial is my motivation to do this, Mercurial is a source control system very similar to git in the way it works with only subtle differences, I chose it as it was well supported on Windows but mostly due to TortoiseHg which is an awesome product.

Now I create many repositories and work in different environments. Now it can be really hard to manage so many repositories and no what to check keep current.

Note (This system could work much the same for git)

So I wanted to completely automate this process and end up with this:

image

image

To get this to work I needed to execute a few commands and then check the console outputs.

I decided to make a really extensible project that you could change and handle outputs differently with ease.

So I thought about a command like this:
   1: PyRun -script "Mercurial-autoupdate.py"  -command "hg" -args "summary" -workingDirectory <directory> -pushremote1 <remotedir>

This would execute the command “hg” which is the mercurial command and the argument “summary” to report the status of the working directory.



It would then run a few callback methods in the script i specify in the script parameters, where I can then check outputs, handle some events, and spawn sub processes.



So I created a PyRunner class which I start like this:




   1: _runner = new PyRunner(cmd.ParamAfterSwitch("command"),
   2:     cmd.ParamAfterSwitch("script"),
   3:     cmd.ParamAfterSwitch("args"),
   4:     cmd.ParamAfterSwitch("workingDirectory"), cmd);
   5: _runner.Run();

PyRunner will utilise the hosting script controller in an earlier post:

http://thewalkingdev.blogspot.com/2011/05/building-reusable-ironpython-hosting.html






   1: public void Run()
   2:         {
   3:             // var code = new StreamReader(Resources.ResourceManager.GetStream("TextFile1.py")).ReadToEnd();
   4:             _controller = new ScriptController(false);
   5:             _controller.OnMessage += (sender, message) =>
   6:                 {
   7:                     //Console.WriteLine(message);
   8:                    // Console.WriteLine(String.Format("[{0:HH:MM:ss:fff}][SCRIPT]:{1}", DateTime.Now, message.Trim()));
   9:                     Console.WriteLine(String.Format("{0}", message.Trim()));
  10:                    
  11:                 };
  12:             var code = Resources.ResourceManager.GetString("ScriptHeader1");
  13:             dynamic handler = null;
  14:             IScriptContext ctx;
  15:             try
  16:             {
  17:  
  18:                 if (string.IsNullOrEmpty(script))
  19:                 {
  20:                     script = code;
  21:                     ctx = _controller.CreateScriptContextFromString(script);
  22:                     ctx.Execute();
  23:                 }
  24:                 else
  25:                 {
  26:                     ctx = _controller.CreateScriptContextFromFile(script);
  27:                     ctx.ExecuteString(code);
  28:                     ctx.Execute();
  29:                 }
  30:              
  31:            
  32:                
  33:                 
  34:                 handler = ctx.Scope.CreateHandler();
  35:             }
  36:             catch (Exception e)
  37:             {
  38:                 Console.WriteLine(e.Message);
  39:                 return;
  40:             }
  41:         
  42:  
  43:             _mainProcess = new PyProcessContext(ctx, handler,
  44:                 _processFactory,
  45:                 command,
  46:                 CommandArguments,
  47:                 workingDirectory,
  48:                 _commandParameters);
  49:  
  50:             
  51:             _mainProcess.Start();
  52:  
  53:             ctx.FlushBuffer();
  54:            // Console.ReadKey();
  55:         }

It will create a handler that will be passed to the process context:




   1: public class PyProcessContext
   2:     {
   3:         dynamic _processHandler;
   4:         ProcessContext _processContext;
   5:  
   6:         public ProcessContext ProcessContext
   7:         {
   8:             get { return _processContext; }
   9:            
  10:         }
  11:         ProcessFactory _processFactory;
  12:         string _command;
  13:         string _commandArguments;
  14:         string _workingDirectory;
  15:         IScriptContext _scriptContext;
  16:         ICommandParameters _commandParameters;
  17:  
  18:         public PyProcessContext(IScriptContext scriptContext,
  19:             dynamic processHandler, 
  20:             ProcessFactory processFactory, 
  21:             string command,
  22:             string commandArguments,
  23:             string workingDirectory,
  24:             ICommandParameters commandParameters)
  25:         {
  26:             _commandParameters = commandParameters;
  27:             _scriptContext = scriptContext;
  28:             _command = command;
  29:             _commandArguments = commandArguments;
  30:             _workingDirectory = workingDirectory;
  31:             _processFactory = processFactory;
  32:             var processContext = _processFactory.CreateProcessContext(command, 
  33:                 commandArguments, 
  34:                 workingDirectory);
  35:             _processHandler = processHandler;
  36:             processContext.OnMessage += (message) =>
  37:             {
  38:                 _processHandler.OnMessage(message);
  39:                 //  Console.WriteLine(message);
  40:             };
  41:             processContext.OnError += (message) =>
  42:             {
  43:                 _processHandler.OnError(message);
  44:             };
  45:             _processHandler.OnInit(this, _commandParameters);
  46:             _processContext = processContext;
  47:         }
  48:         public void Start()
  49:         {
  50:             _processHandler.OnCommandStarting();
  51:             _processContext.Start();
  52:             _processHandler.OnCommandCompleted();
  53:         }
  54:  
  55:         public PyProcessContext CreateSpawnedProcess(string command,
  56:             string commandArguments)
  57:         {
  58:             return CreateSpawnedProcess(this._command,
  59:                 this._commandArguments,
  60:                 this._workingDirectory,
  61:                 this._processHandler);
  62:         }
  63:  
  64:         public PyProcessContext CreateSpawnedProcess(string command,
  65:             string commandArguments,
  66:             string workingDirectory,
  67:             dynamic processHandler)
  68:         {
  69:             if (workingDirectory == null)
  70:             {
  71:                 workingDirectory = _workingDirectory;
  72:             }
  73:             var cmd = new CmdLineHelper();
  74:             cmd.ParseString(commandArguments);
  75:             return new PyProcessContext(_scriptContext, 
  76:                 processHandler, 
  77:                 this._processFactory,
  78:                 command,
  79:                 commandArguments,
  80:                 workingDirectory,
  81:                 cmd);
  82:         }
  83:     }



Here is how to build the handler that each script must implement:




   1: class BaseHandler:
   2:         
   3:     def OnInit(self, processContext, args):
   4:         self._processContext = processContext
   5:         self._commandArgs = args
   6:         pass
   7:         
   8:     def OnError(self, message):
   9:         pass
  10:         
  11:     def OnMessage(self, message):
  12:         pass        
  13:         
  14:     def OnCommandStarting(self):
  15:         pass        
  16:         
  17:         
  18:     def OnCommandCompleted(self):
  19:         pass        
  20:         
  21:     def Grep(self, input, pattern):
  22:         #print "matching: %s pattern: %s" % (input, pattern)
  23:         return Regex(pattern).Match(input)
  24:     
  25:     def GetStdOut(self):
  26:         return self._processContext.ProcessContext.StandardOutput.GetValue()
  27:  
  28:     def GetErrorOut(self):
  29:         return self._processContext.ProcessContext.ErrorOutput.GetValue()
  30:     
  31:     def GrepStdOut(self, pattern):
  32:         return self.Grep(self._processContext.ProcessContext.StandardOutput.GetValue(), pattern)
  33:  
  34:  
  35:     def GrepStdOut(self, pattern):
  36:         
  37:         return self.Grep(self._processContext.ProcessContext.ErrorOutput.GetValue(), pattern)
  38:  
  39:  
  40: def CreateHandler():
  41:     return BaseHandler()

These events will be called as they occur with the process, you can only use OnCommandCompleted if you want to handle the outputs once completed.



Getting the mercurial summary of the working directory:

Spawn a process and read the output


   1: def HgGetSummary(self):
   2:     h = SubHandler()
   3:     self._processContext.CreateSpawnedProcess("hg", "summary", None, h).Start()
   4:     return h.GetStdOut()



Pull from remote and check if there are changes:




   1: def HgPull(self):
   2:     for remote in self.getPullRemotes():
   3:         print "Pulling remote: " + remote                        
   4:         h = SubHandler()
   5:         self._processContext.CreateSpawnedProcess("hg", 'pull "%s"' % remote, None, h).Start()    
   6:         self.PrintPushPullClean(h, "Pull: CLEAN", "Pull: NOT CLEAN, output:")



Similarly push and see if there are changes:


   1: def HgPush(self):
   2:     for remote in self.getPushRemotes():
   3:         print "Pushing remote: " + remote                        
   4:         h = SubHandler()
   5:         self._processContext.CreateSpawnedProcess("hg", 'push "%s"' % remote, None, h).Start()
   6:         self.PrintPushPullClean(h, "Push: CLEAN", "Push: NOT CLEAN, output:")



The function that check whether the call to the remote had no changes:


   1: def CheckPushPullClean(self, output):        
   2:     match = self.Grep(output, r"searching for changes[\r\n\s]+?no changes found")
   3:     return match.Success

Then check if anything needs to be committed:




   1: def CheckSummaryCommitClean(self, output):        
   2:     match = self.Grep(output, r"commit: \(clean\)")
   3:     return match.Success



Check if an update is required:




   1: def CheckSummaryUpdateClean(self, output):
   2:     #print "CHECKING IF UPDATE CLEAN: [[[[[[%s]]]]]]]] " % output        
   3:     match = self.Grep(output, r"update: \(current\)")
   4:     return match.Success

Here is a function that performs the commit:


   1: def HgCommit(self):
   2:     return self._processContext.CreateSpawnedProcess("thg", "commit", None, SubHandler()).Start()

Here is the function that performs an update:




All commits and updates run the visual version, however I added an update non visual flag, this will only perform hg update which has no UI so it is more automated.


   1: def HgUpdate(self):
   2:     if self.updateNonVisual():
   3:         print "Updating in non visual mode"
   4:         h = SubHandler()
   5:         updateResult = self._processContext.CreateSpawnedProcess("hg", "update", None, h).Start()
   6:         print h.GetStdOut()            
   7:         #self.PrintSummary(self.HgGetSummary())
   8:         return updateResult
   9:     else:
  10:         return self._processContext.CreateSpawnedProcess("thg", "update", None, SubHandler()).Start()
  11:       



Or just fire up the repository explorer:




   1: def HgRepoExplorer(self):
   2:     return self._processContext.CreateSpawnedProcess("thg", "log", None, SubHandler()).Start()



So now lets look at the AutoUpdate function with will apply all this logic to the working directory, to check the remote status, working directory status and push/pull/commit and update if needed.




   1: def AutoUpdate(self):        
   2:     summary = self.GetStdOut()
   3:     #self.PrintSummary(summary)
   4:     commitClean = self.CheckSummaryCommitClean(summary)
   5:     updateClean = self.CheckSummaryUpdateClean(summary)
   6:     if (commitClean):
   7:         print "Commit: CLEAN"           
   8:     else:            
   9:         print "Commit is NOT Clean"
  10:         self.HgCommit()
  11:         if self.hasRemotes():
  12:             if not self.pushNonVisual():
  13:                 print "You specified remote repositores, take a moment and view your changes with the explorer, a push will commence afterwards"
  14:                 self.HgRepoExplorer()
  15:                 self.HgPush()
  16:             else:
  17:                 print "You specified remote repositories to push in background"
  18:                 self.HgPush()
  19:             
  20:     if (not updateClean):
  21:         print "Update needed"
  22:         self.HgUpdate()
  23:         summary = self.HgGetSummary()
  24:         self.checkMustBeClean(summary)        
  25:     else:
  26:         print "Update: CLEAN"                       
  27:     
  28:     if self.hasRemotes():    
  29:         if (not self.CheckSummaryUpdateClean(self.HgGetSummary()) or not self.CheckSummaryCommitClean(self.HgGetSummary())):
  30:             raise Exception("Cannot pull from the remote repository because the working directory is still not clean")
  31:         self.HgPull()
  32:         if (not self.CheckSummaryUpdateClean(self.HgGetSummary())):
  33:             print "New revisions were detected after pull, update required"
  34:             self.HgUpdate()
  35:             summary = self.HgGetSummary()
  36:             self.checkMustBeClean(summary)
  37:         self.HgPush()
  38:     else:
  39:         print "No remotes, skipping pull"
  40:         
  41:     summary = self.HgGetSummary()
  42:     self.checkMustBeClean(summary)        
  43:  



Now I have a fully automated process that will check in, push/pull update in one click.



https://github.com/TheWalkingDev/Projects/tree/master/Processes/PyRun/PyRun



Library files can be found here:



https://github.com/TheWalkingDev/ACSR

No comments:

Post a Comment