Code

Initial import of a python script to import changesets from Perforce into git.
[git.git] / contrib / fast-import / p4-fast-export.py
1 #!/usr/bin/python
2 #
3 # p4-fast-export.py
4 #
5 # Author: Simon Hausmann <hausmann@kde.org>
6 # License: MIT <http://www.opensource.org/licenses/mit-license.php>
7 #
8 # TODO: - fix date parsing (how hard can it be?)
9 #       - support integrations (at least p4i)
10 #       - support incremental imports
11 #       - create tags
12 #       - instead of reading all files into a variable try to pipe from
13 #       - p4 print directly to stdout. need to figure out file size somehow
14 #         though.
15 #       - support p4 submit (hah!)
16 #       - don't hardcode the import to master
17 #
18 import os, string, sys
20 # yep, that's hardcoded right. will fix to a commandline option rsn :)
21 prefix = "//depot/qt/main/"
22 # that's in revision range syntax, for example @2342,523634
23 changeRange = ""
25 def describe(change):
26     output = os.popen("p4 describe %s" % change).readlines()
28     firstLine = output[0]
30     author = firstLine.split(" ")[3]
31     author = author[:author.find("@")]
33     filesSection = 0
34     try:
35         filesSection = output.index("Affected files ...\n")
36     except ValueError:
37         sys.stderr.write("Change %s doesn't seem to affect any files. Weird.\n" % change)
38         return [], [], [], []
40     differencesSection = 0
41     try:
42         differencesSection = output.index("Differences ...\n")
43     except ValueError:
44         sys.stderr.write("Change %s doesn't seem to have a differences section. Weird.\n" % change)
45         return [], [], [], []
47     log = output[2:filesSection - 1]
49     lines = output[filesSection + 2:differencesSection - 1]
51     changed = []
52     removed = []
54     for line in lines:
55         # chop off "... " and trailing newline
56         line = line[4:len(line) - 1]
58         lastSpace = line.rfind(" ")
59         if lastSpace == -1:
60             sys.stderr.write("trouble parsing line %s, skipping!\n" % line)
61             continue
63         operation = line[lastSpace + 1:]
64         path = line[:lastSpace]
66         if operation == "delete":
67             removed.append(path)
68         else:
69             changed.append(path)
71     return author, log, changed, removed
73 def p4cat(path):
74     return os.popen("p4 print -q \"%s\"" % path).read()
76 def stripRevision(path):
77     hashPos = path.rindex("#")
78     return path[:hashPos]
80 def getUserMap():
81     users = {}
82     output = os.popen("p4 users")
83     for line in output:
84         firstSpace = line.index(" ")
85         secondSpace = line.index(" ", firstSpace + 1)
86         key = line[:firstSpace]
87         email = line[firstSpace + 1:secondSpace]
88         openParenPos = line.index("(", secondSpace)
89         closedParenPos = line.index(")", openParenPos)
90         name = line[openParenPos + 1:closedParenPos]
92         users[key] = name + " " + email
94     return users
97 users = getUserMap()
99 output = os.popen("p4 changes %s...%s" % (prefix, changeRange)).readlines()
101 changes = []
102 for line in output:
103     changeNum = line.split(" ")[1]
104     changes.append(changeNum)
106 changes.reverse()
108 sys.stderr.write("\n")
110 cnt = 0
111 for change in changes:
112     [ author, log, changedFiles, removedFiles ] = describe(change)
113     sys.stderr.write("\rimporting revision %s (%s%%)" % (change, cnt * 100 / len(changes)))
114     cnt = cnt + 1
115 #    sys.stderr.write("%s\n" % log)
116 #    sys.stderr.write("%s\n" % changedFiles)
117 #    sys.stderr.write("%s\n" % removedFiles)
119     print "commit refs/heads/master"
120     if author in users:
121         print "committer %s 1 2" % users[author]
122     else:
123         print "committer %s <a@b> 1 2" % author
124     print "data <<EOT"
125     for l in log:
126         print l[:len(l) - 1]
127     print "EOT"
129     print ""
131     for f in changedFiles:
132         if not f.startswith(prefix):
133             sys.stderr.write("\nchanged files: ignoring path %s outside of %s in change %s\n" % (f, prefix, change))
134             continue
135         relpath = f[len(prefix):]
136         print "M 644 inline %s" % stripRevision(relpath)
137         data = p4cat(f)
138         print "data %s" % len(data)
139         sys.stdout.write(data)
140         print ""
142     for f in removedFiles:
143         if not f.startswith(prefix):
144             sys.stderr.write("\ndeleted files: ignoring path %s outside of %s in change %s\n" % (f, prefix, change))
145             continue
146         relpath = f[len(prefix):]
147         print "D %s" % stripRevision(relpath)
149     print ""