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 21:54:56 +01:00
|
|
|
import marshal, popen2
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 09:39:20 +01:00
|
|
|
if len(sys.argv) != 2:
|
2007-01-31 21:54:56 +01:00
|
|
|
print "usage: %s //depot/path[@revRange]" % sys.argv[0]
|
|
|
|
|
print "\n example:"
|
|
|
|
|
print " %s //depot/my/project/ -- to import everything"
|
|
|
|
|
print " %s //depot/my/project/@1,6 -- to import only from revision 1 to 6"
|
|
|
|
|
print ""
|
|
|
|
|
print " (a ... is not needed in the path p4 specification, it's added implicitly)"
|
|
|
|
|
print ""
|
2007-01-31 09:39:20 +01:00
|
|
|
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 22:19:18 +01:00
|
|
|
def p4CmdList(cmd):
|
2007-01-31 20:48:39 +01:00
|
|
|
pipe = os.popen("p4 -G %s" % cmd, "rb")
|
2007-01-31 22:19:18 +01:00
|
|
|
result = []
|
2007-01-31 20:48:39 +01:00
|
|
|
try:
|
|
|
|
|
while True:
|
|
|
|
|
entry = marshal.load(pipe)
|
2007-01-31 22:19:18 +01:00
|
|
|
result.append(entry)
|
2007-01-31 20:48:39 +01:00
|
|
|
except EOFError:
|
|
|
|
|
pass
|
|
|
|
|
pipe.close()
|
|
|
|
|
return result
|
|
|
|
|
|
2007-01-31 22:19:18 +01:00
|
|
|
def p4Cmd(cmd):
|
|
|
|
|
list = p4CmdList(cmd)
|
|
|
|
|
result = {}
|
|
|
|
|
for entry in list:
|
|
|
|
|
result.update(entry)
|
|
|
|
|
return result;
|
|
|
|
|
|
2007-01-31 00:16:59 +01:00
|
|
|
def describe(change):
|
2007-01-31 22:13:17 +01:00
|
|
|
describeOutput = p4Cmd("describe %s" % change)
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 22:13:17 +01:00
|
|
|
author = describeOutput["user"]
|
|
|
|
|
epoch = describeOutput["time"]
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 22:13:17 +01:00
|
|
|
log = describeOutput["desc"]
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
|
changed = []
|
|
|
|
|
removed = []
|
|
|
|
|
|
2007-01-31 22:13:17 +01:00
|
|
|
i = 0
|
|
|
|
|
while describeOutput.has_key("depotFile%s" % i):
|
|
|
|
|
path = describeOutput["depotFile%s" % i]
|
|
|
|
|
rev = describeOutput["rev%s" % i]
|
|
|
|
|
action = describeOutput["action%s" % i]
|
|
|
|
|
path = path + "#" + rev
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 22:13:17 +01:00
|
|
|
if action == "delete":
|
2007-01-31 00:16:59 +01:00
|
|
|
removed.append(path)
|
|
|
|
|
else:
|
|
|
|
|
changed.append(path)
|
|
|
|
|
|
2007-01-31 22:13:17 +01:00
|
|
|
i = i + 1
|
|
|
|
|
|
2007-01-31 16:39:46 +01:00
|
|
|
return author, log, epoch, changed, removed
|
2007-01-31 00:16:59 +01:00
|
|
|
|
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 = {}
|
|
|
|
|
|
2007-01-31 22:19:18 +01:00
|
|
|
for output in p4CmdList("users"):
|
|
|
|
|
if not output.has_key("User"):
|
|
|
|
|
continue
|
|
|
|
|
users[output["User"]] = output["FullName"] + " <" + output["Email"] + ">"
|
2007-01-31 00:16:59 +01:00
|
|
|
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 21:54:56 +01:00
|
|
|
gitOutput, gitStream, gitError = popen2.popen3("git-fast-import")
|
|
|
|
|
|
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 21:54:56 +01:00
|
|
|
sys.stdout.write("\rimporting revision %s (%s%%)" % (change, cnt * 100 / len(changes)))
|
2007-01-31 00:16:59 +01:00
|
|
|
cnt = cnt + 1
|
|
|
|
|
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("commit refs/heads/master\n")
|
2007-01-31 00:16:59 +01:00
|
|
|
if author in users:
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("committer %s %s %s\n" % (users[author], epoch, tz))
|
2007-01-31 00:16:59 +01:00
|
|
|
else:
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("committer %s <a@b> %s %s\n" % (author, epoch, tz))
|
|
|
|
|
gitStream.write("data <<EOT\n")
|
2007-01-31 22:13:17 +01:00
|
|
|
gitStream.write(log)
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("EOT\n\n")
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("M %s inline %s\n" % (mode, stripRevision(relpath)))
|
|
|
|
|
gitStream.write("data %s\n" % fileSize)
|
|
|
|
|
gitStream.write(os.popen("p4 print -q \"%s\"" % f).read())
|
|
|
|
|
gitStream.write("\n")
|
2007-01-31 00:16:59 +01:00
|
|
|
|
|
|
|
|
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):]
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("D %s\n" % stripRevision(relpath))
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.write("\n")
|
2007-01-31 00:16:59 +01:00
|
|
|
|
2007-01-31 21:54:56 +01:00
|
|
|
gitStream.close()
|
|
|
|
|
gitOutput.close()
|
|
|
|
|
gitError.close()
|
2007-01-31 16:39:46 +01:00
|
|
|
|
2007-01-31 21:54:56 +01:00
|
|
|
print ""
|