Author: kwsutter
Date: Fri Feb 24 19:25:19 2012
New Revision: 1293389

URL: http://svn.apache.org/viewvc?rev=1293389&view=rev
Log:
Integrating the patchoj.py tool for ease of creating patches.  It already 
existed in later releases, so I'm just bringing it backwards.

Added:
    openjpa/branches/2.1.x/patchoj.py

Added: openjpa/branches/2.1.x/patchoj.py
URL: 
http://svn.apache.org/viewvc/openjpa/branches/2.1.x/patchoj.py?rev=1293389&view=auto
==============================================================================
--- openjpa/branches/2.1.x/patchoj.py (added)
+++ openjpa/branches/2.1.x/patchoj.py Fri Feb 24 19:25:19 2012
@@ -0,0 +1,179 @@
+#!/usr/bin/python
+"""
+ Licensed to the Apache Software Foundation (ASF) under one
+ or more contributor license agreements.  See the NOTICE file
+ distributed with this work for additional information
+ regarding copyright ownership.  The ASF licenses this file
+ to you under the Apache License, Version 2.0 (the
+ "License"); you may not use this file except in compliance
+ with the License.  You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing,
+ software distributed under the License is distributed on an
+ "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ KIND, either express or implied.  See the License for the
+ specific language governing permissions and limitations
+ under the License.
+"""
+
+from optparse import OptionParser
+from zipfile import ZipFile
+from subprocess import Popen,PIPE
+from glob import glob
+import os,re
+
+def main() :
+    usage= "usage: %prog [optons]"
+    description = "Generate a jar file which contains the OpenJPA class files 
which have been changed. The output from svn info and svn diff may also be 
included. A typical \
+            use is to run from the same directory as the parent pom. This will 
automatically include all changed files with a .java extension."
+    version = "$prog 0.5"
+    parser = OptionParser(usage=usage, version=version, description = 
description) 
+    parser.add_option("-f", "--files", dest="files", help="list of filenames 
to include")
+    parser.add_option("-p", "--patchfile", dest="patchfile", help="patch file 
name", default="patch.jar")
+    parser.add_option("-i", "--noinfo", action="store_false", 
dest="includeInfo", help="exclude output from svn info", default=True)
+    parser.add_option("-d", "--nodiff", action="store_false", 
dest="includeDiff", help="exclude output from svn diff",default=True)
+    parser.add_option("-v", "--verbose", action="store_true", dest="verbose", 
help="print debug information",default=False)
+    # parser.add_option("-p", "--pattern", dest="pattern", help="regex of 
filenames to match")
+
+    global options
+    (options,args) = parser.parse_args()
+
+    genZip(options)
+
+def genZip(options):
+    """ generate the zip file """
+    if options.files == None:
+        files = getAllFiles() 
+    else :
+        files = getFilesFromList(options.files)
+
+    zip = ZipFile(options.patchfile, 'w')
+
+    if options.includeInfo: 
+        writeInfo(zip, files)
+    if options.includeDiff:
+        writeDiff(zip, files)
+
+    writeClasses(zip, files)
+
+    zip.close()
+
+    print 'Wrote patch to %s. ' % (options.patchfile) 
+    if options.verbose:
+        print 'Files in patch: %s' % ("\n".join(files))
+
+def getAllFiles():
+    """ get the list of all modified files. Only Added or Modified .java, 
.properties files will be considered """
+    if options.verbose:
+        print ' > getAllFiles'
+    files = []
+
+    commandOutput =  Popen(['svn', 'stat'], 
stdout=PIPE).communicate()[0].splitlines()
+    prog = re.compile('([A|M] *)(.*\.java)')
+    for candidate in commandOutput: 
+        match = prog.match(candidate)
+        if match : 
+            files.append(match.group(2))
+
+    prog = re.compile('([A|M] *)(.*\.properties)')
+    for candidate in commandOutput: 
+        match = prog.match(candidate)
+        if match : 
+            files.append(match.group(2))
+
+
+    if options.verbose:
+        print 'Found modified files : ' + str(files)
+        print ' < getAllFiles'
+    return files
+
+def getFilesFromList(files) :
+    """ get a list of modified files from a comma separated list usually from 
the command line """
+
+    if options.verbose :
+        print ' > getFilesFromList'
+        print ' returning ' + str(files.split(','))
+        print ' < getFilesFromList' 
+
+    return files.split(',')
+
+def writeInfo(zip, files=None):
+    """ write the output of svn info to a temp file and store in a zip """
+    if options.verbose : 
+        print ' > writeInfo'
+
+    patchFile = open('info.txt', 'w')
+    args = ['svn', 'info']
+    
+    if files: 
+        args.extend(files)
+
+    Popen(args=args, stdout=patchFile).communicate()[0] 
+
+    zip.write(patchFile.name)
+    patchFile.close()
+    os.remove(patchFile.name)
+
+    if options.verbose: 
+        print ' < writeInfo' 
+
+def writeDiff(zip, files=None):
+    """ Write the output of svn diff to a temp file and store in a zip """
+    if options.verbose:
+        print ' > writeDiff'
+
+    patchFile = open('patch.txt', 'w')
+    args = ['svn', 'diff']
+    if files: 
+        args.extend(files)
+
+    Popen(args=args, stdout=patchFile).communicate()[0] 
+
+    zip.write(patchFile.name)
+    patchFile.close()
+    os.remove(patchFile.name)
+
+    if options.verbose: 
+        print ' < writeDiff'
+
+def javaToClass(file) :
+    """ simple helper function, converts a string from svn stat (or command 
line) to its corresponding
+    .class file
+    """
+
+    rval = 
file.replace('src','target').replace('main','classes').replace('test','test-classes').replace('.java',
 '.class').replace('java','').replace('\\\\','\\')
+    return rval;
+
+def javaToInnerClass(file):
+    """ helper function, converts .java file to a glob pattern that matches 
inner classes """
+    return javaToClass(file).replace('.class', '$*.class')
+
+def writeClasses(zip, files=None): 
+    """ Write class files to a zip """
+   
+    prog = re.compile('(.*classes.)(.*)')
+    propertiesProg = re.compile('(.*resources.)(.*)')
+    
+    for file in files :
+        if str(file).endswith('.java'):
+            for globMatch in glob(javaToClass(file)) :
+                if(prog.match(globMatch)):
+                    target = prog.match(globMatch).group(2)
+                    zip.write(os.path.realpath(globMatch), target)
+
+            # match again on inner classes, not sure if glob supports optional 
matches. 
+            for globMatch in glob(javaToInnerClass(file)) : 
+                if(prog.match(globMatch)): 
+                    target = prog.match(globMatch).group(2)
+                    zip.write(os.path.realpath(globMatch), target)
+        elif str(file).endswith('.properties'):
+            for globMatch in glob(file) :
+                 if(propertiesProg.match(globMatch)): 
+                    target = propertiesProg.match(globMatch).group(2)
+                    zip.write(os.path.realpath(globMatch), target)
+
+if __name__ == "__main__":
+    main()
+


Reply via email to