2007-01-31 00:16:59 +01:00
|
|
|
#!/usr/bin/python
|
|
|
|
#
|
|
|
|
# p4-fast-export.py
|
|
|
|
#
|
|
|
|
# Author: Simon Hausmann <hausmann@kde.org>
|
|
|
|
# License: MIT <http://www.opensource.org/licenses/mit-license.php>
|
|
|
|
#
|
2007-01-31 20:16:26 +01:00
|
|
|
# TODO:
|
|
|
|
# - support integrations (at least p4i)
|
2007-01-31 00:16:59 +01:00
|
|
|
# - support incremental imports
|
|
|
|
# - create tags
|
|
|
|
# - instead of reading all files into a variable try to pipe from
|
|
|
|
# - support p4 submit (hah!)
|
|
|
|
# - don't hardcode the import to master
|
|
|
|
#
|
2007-01-31 16:39:46 +01:00
|
|
|
import os, string, sys, time
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 09:39:20 +01:00
|
|
|
if len(sys.argv) != 2:
|
|
|
|
sys.stderr.write("usage: %s //depot/path[@revRange]\n" % sys.argv[0]);
|
|
|
|
sys.stderr.write("\n example:\n");
|
2007-01-31 09:49:41 +01:00
|
|
|
sys.stderr.write(" %s //depot/my/project/ -- to import everything\n");
|
|
|
|
sys.stderr.write(" %s //depot/my/project/@1,6 -- to import only from revision 1 to 6\n");
|
|
|
|
sys.stderr.write("\n");
|
|
|
|
sys.stderr.write(" (a ... is not needed in the path p4 specification, it's added implicitly)\n");
|
2007-01-31 09:39:20 +01:00
|
|
|
sys.stderr.write("\n");
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
prefix = sys.argv[1]
|
2007-01-31 00:16:59 +01:00
|
|
|
changeRange = ""
|
2007-01-31 09:39:20 +01:00
|
|
|
try:
|
|
|
|
atIdx = prefix.index("@")
|
|
|
|
changeRange = prefix[atIdx:]
|
|
|
|
prefix = prefix[0:atIdx]
|
|
|
|
except ValueError:
|
|
|
|
changeRange = ""
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 09:49:41 +01:00
|
|
|
if not prefix.endswith("/"):
|
|
|
|
prefix += "/"
|
|
|
|
|
2007-01-31 00:16:59 +01:00
|
|
|
def describe(change):
|
|
|
|
output = os.popen("p4 describe %s" % change).readlines()
|
|
|
|
|
|
|
|
firstLine = output[0]
|
|
|
|
|
2007-01-31 16:39:46 +01:00
|
|
|
splitted = firstLine.split(" ")
|
|
|
|
author = splitted[3]
|
2007-01-31 00:16:59 +01:00
|
|
|
author = author[:author.find("@")]
|
2007-01-31 19:43:16 +01:00
|
|
|
tm = time.strptime(splitted[5] + " " + splitted[6], "%Y/%m/%d %H:%M:%S ")
|
2007-01-31 16:39:46 +01:00
|
|
|
epoch = int(time.mktime(tm))
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
filesSection = 0
|
|
|
|
try:
|
|
|
|
filesSection = output.index("Affected files ...\n")
|
|
|
|
except ValueError:
|
|
|
|
sys.stderr.write("Change %s doesn't seem to affect any files. Weird.\n" % change)
|
2007-01-31 16:39:46 +01:00
|
|
|
return [], [], [], [], []
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
differencesSection = 0
|
|
|
|
try:
|
|
|
|
differencesSection = output.index("Differences ...\n")
|
|
|
|
except ValueError:
|
|
|
|
sys.stderr.write("Change %s doesn't seem to have a differences section. Weird.\n" % change)
|
2007-01-31 16:39:46 +01:00
|
|
|
return [], [], [], [], []
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
log = output[2:filesSection - 1]
|
|
|
|
|
|
|
|
lines = output[filesSection + 2:differencesSection - 1]
|
|
|
|
|
|
|
|
changed = []
|
|
|
|
removed = []
|
|
|
|
|
|
|
|
for line in lines:
|
|
|
|
# chop off "... " and trailing newline
|
|
|
|
line = line[4:len(line) - 1]
|
|
|
|
|
|
|
|
lastSpace = line.rfind(" ")
|
|
|
|
if lastSpace == -1:
|
|
|
|
sys.stderr.write("trouble parsing line %s, skipping!\n" % line)
|
|
|
|
continue
|
|
|
|
|
|
|
|
operation = line[lastSpace + 1:]
|
|
|
|
path = line[:lastSpace]
|
|
|
|
|
|
|
|
if operation == "delete":
|
|
|
|
removed.append(path)
|
|
|
|
else:
|
|
|
|
changed.append(path)
|
|
|
|
|
2007-01-31 16:39:46 +01:00
|
|
|
return author, log, epoch, changed, removed
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
def p4cat(path):
|
|
|
|
return os.popen("p4 print -q \"%s\"" % path).read()
|
|
|
|
|
2007-01-31 20:16:26 +01:00
|
|
|
def p4Stat(path):
|
|
|
|
output = os.popen("p4 fstat -Ol \"%s\"" % path).readlines()
|
|
|
|
fileSize = 0
|
|
|
|
mode = 644
|
|
|
|
for line in output:
|
|
|
|
if line.startswith("... headType x"):
|
|
|
|
mode = 755
|
|
|
|
elif line.startswith("... fileSize "):
|
|
|
|
fileSize = long(line[12:])
|
|
|
|
return mode, fileSize
|
|
|
|
|
2007-01-31 00:16:59 +01:00
|
|
|
def stripRevision(path):
|
|
|
|
hashPos = path.rindex("#")
|
|
|
|
return path[:hashPos]
|
|
|
|
|
|
|
|
def getUserMap():
|
|
|
|
users = {}
|
|
|
|
output = os.popen("p4 users")
|
|
|
|
for line in output:
|
|
|
|
firstSpace = line.index(" ")
|
|
|
|
secondSpace = line.index(" ", firstSpace + 1)
|
|
|
|
key = line[:firstSpace]
|
|
|
|
email = line[firstSpace + 1:secondSpace]
|
|
|
|
openParenPos = line.index("(", secondSpace)
|
|
|
|
closedParenPos = line.index(")", openParenPos)
|
|
|
|
name = line[openParenPos + 1:closedParenPos]
|
|
|
|
|
|
|
|
users[key] = name + " " + email
|
|
|
|
|
|
|
|
return users
|
|
|
|
|
|
|
|
users = getUserMap()
|
|
|
|
|
|
|
|
output = os.popen("p4 changes %s...%s" % (prefix, changeRange)).readlines()
|
|
|
|
|
|
|
|
changes = []
|
|
|
|
for line in output:
|
|
|
|
changeNum = line.split(" ")[1]
|
|
|
|
changes.append(changeNum)
|
|
|
|
|
|
|
|
changes.reverse()
|
|
|
|
|
|
|
|
sys.stderr.write("\n")
|
|
|
|
|
2007-01-31 19:43:16 +01:00
|
|
|
tz = - time.timezone / 36
|
|
|
|
|
2007-01-31 16:39:46 +01:00
|
|
|
cnt = 1
|
2007-01-31 00:16:59 +01:00
|
|
|
for change in changes:
|
2007-01-31 16:39:46 +01:00
|
|
|
[ author, log, epoch, changedFiles, removedFiles ] = describe(change)
|
2007-01-31 00:16:59 +01:00
|
|
|
sys.stderr.write("\rimporting revision %s (%s%%)" % (change, cnt * 100 / len(changes)))
|
|
|
|
cnt = cnt + 1
|
|
|
|
|
|
|
|
print "commit refs/heads/master"
|
|
|
|
if author in users:
|
2007-01-31 19:43:16 +01:00
|
|
|
print "committer %s %s %s" % (users[author], epoch, tz)
|
2007-01-31 00:16:59 +01:00
|
|
|
else:
|
2007-01-31 19:43:16 +01:00
|
|
|
print "committer %s <a@b> %s %s" % (author, epoch, tz)
|
2007-01-31 00:16:59 +01:00
|
|
|
print "data <<EOT"
|
|
|
|
for l in log:
|
|
|
|
print l[:len(l) - 1]
|
|
|
|
print "EOT"
|
|
|
|
|
|
|
|
print ""
|
|
|
|
|
|
|
|
for f in changedFiles:
|
|
|
|
if not f.startswith(prefix):
|
|
|
|
sys.stderr.write("\nchanged files: ignoring path %s outside of %s in change %s\n" % (f, prefix, change))
|
|
|
|
continue
|
|
|
|
relpath = f[len(prefix):]
|
2007-01-31 20:16:26 +01:00
|
|
|
|
|
|
|
[mode, fileSize] = p4Stat(f)
|
|
|
|
|
|
|
|
print "M %s inline %s" % (mode, stripRevision(relpath))
|
|
|
|
print "data %s" % fileSize
|
|
|
|
sys.stdout.flush();
|
|
|
|
os.system("p4 print -q \"%s\"" % f)
|
2007-01-31 00:16:59 +01:00
|
|
|
print ""
|
|
|
|
|
|
|
|
for f in removedFiles:
|
|
|
|
if not f.startswith(prefix):
|
|
|
|
sys.stderr.write("\ndeleted files: ignoring path %s outside of %s in change %s\n" % (f, prefix, change))
|
|
|
|
continue
|
|
|
|
relpath = f[len(prefix):]
|
|
|
|
print "D %s" % stripRevision(relpath)
|
|
|
|
|
|
|
|
print ""
|
|
|
|
|
2007-01-31 16:39:46 +01:00
|
|
|
sys.stderr.write("\n")
|
|
|
|
|