server.py 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. # server.py -- Implementation of the server side git protocols
  2. # Copryight (C) 2008 John Carr <john.carr@unrouted.co.uk>
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; version 2
  7. # of the License.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
  17. # MA 02110-1301, USA.
  18. import SocketServer
  19. class Backend(object):
  20. def get_refs(self):
  21. """
  22. Get all the refs in the repository
  23. :return: list of tuple(name, sha)
  24. """
  25. raise NotImplementedError
  26. def has_revision(self, sha):
  27. """
  28. Is a given sha in this repository?
  29. :return: True or False
  30. """
  31. raise NotImplementedError
  32. def apply_pack(self, refs, read):
  33. """ Import a set of changes into a repository and update the refs
  34. :param refs: list of tuple(name, sha)
  35. :param read: callback to read from the incoming pack
  36. """
  37. raise NotImplementedError
  38. def generate_pack(self, want, have, write, progress):
  39. """
  40. Generate a pack containing all commits a client is missing
  41. :param want: is a list of sha's the client desires
  42. :param have: is a list of sha's the client has (allowing us to send the minimal pack)
  43. :param write: is a callback to write pack data to the client
  44. :param progress: is a callback to send progress messages to the client
  45. """
  46. raise NotImplementedError
  47. class Handler(object):
  48. def __init__(self, backend, read, write):
  49. self.backend = backend
  50. self.read = read
  51. self.write = write
  52. def read_pkt_line(self):
  53. """
  54. Reads a 'pkt line' from the remote git process
  55. :return: The next string from the stream
  56. """
  57. sizestr = self.read(4)
  58. if not sizestr:
  59. return None
  60. size = int(sizestr, 16)
  61. if size == 0:
  62. return None
  63. return self.read(size-4)
  64. def write_pkt_line(self, line):
  65. """
  66. Sends a 'pkt line' to the remote git process
  67. :param line: A string containing the data to send
  68. """
  69. self.write("%04x%s" % (len(line)+4, line))
  70. def write_sideband(self, channel, blob):
  71. """
  72. Write data to the sideband (a git multiplexing method)
  73. :param channel: int specifying which channel to write to
  74. :param blob: a blob of data (as a string) to send on this channel
  75. """
  76. # a pktline can be a max of 65535. a sideband line can therefore be
  77. # 65535-5 = 65530
  78. # WTF: Why have the len in ASCII, but the channel in binary.
  79. while blob:
  80. self.write_pkt_line("%s%s" % (chr(channel), blob[:65530]))
  81. blob = blob[65530:]
  82. def handle(self):
  83. """
  84. Deal with the request
  85. """
  86. raise NotImplementedError
  87. class UploadPackHandler(Handler):
  88. def handle(self):
  89. refs = self.backend.get_refs()
  90. if refs:
  91. self.write_pkt_line("%s %s\x00multi_ack side-band-64k thin-pack ofs-delta\n" % (refs[0][1], refs[0][0]))
  92. for i in range(1, len(refs)):
  93. ref = refs[i]
  94. self.write_pkt_line("%s %s\n" % (ref[1], ref[0]))
  95. # i'm done...
  96. self.write("0000")
  97. # Now client will either send "0000", meaning that it doesnt want to pull.
  98. # or it will start sending want want want commands
  99. want = self.read_pkt_line()
  100. if want == None:
  101. return
  102. # Keep reading the list of demands until we hit another "0000"
  103. want_revs = []
  104. while want and want[:4] == 'want':
  105. want_rev = want[5:40]
  106. # FIXME: This check probably isnt needed?
  107. if self.backend.has_revision(want_rev):
  108. want_revs.append(want_rev)
  109. want = self.read_pkt_line()
  110. # Client will now tell us which commits it already has - if we have them we ACK them
  111. # this allows client to stop looking at that commits parents (main reason why git pull is fast)
  112. last_sha = None
  113. have_revs = []
  114. have = self.read_pkt_line()
  115. while have and have[:4] == 'have':
  116. have_ref = have[6:40]
  117. if self.backend.has_revision(hav_rev):
  118. self.write_pkt_line("ACK %s continue\n" % sha)
  119. last_sha = sha
  120. have_revs.append(rev_id)
  121. have = self.read_pkt_line()
  122. # At some point client will stop sending commits and will tell us it is done
  123. assert(have[:4] == "done")
  124. # Oddness: Git seems to resend the last ACK, without the "continue" statement
  125. if last_sha:
  126. self.write_pkt_line("ACK %s\n" % last_sha)
  127. # The exchange finishes with a NAK
  128. self.write_pkt_line("NAK\n")
  129. #if True: # False: #self.no_progress == False:
  130. # self.write_sideband(2, "Bazaar is preparing your pack, plz hold.\n")
  131. # for x in range(1,200)
  132. # self.write_sideband(2, "Counting objects: %d\x0d" % x*2)
  133. # self.write_sideband(2, "Counting objects: 200, done.\n")
  134. # for x in range(1,100):
  135. # self.write_sideband(2, "Compressiong objects: %d (%d/%d)\x0d" % (x, x*2, 200))
  136. # self.write_sideband(2, "Compressing objects: 100% (200/200), done.\n")
  137. self.backend.generate_pack(want_revs, have_revs, self.write, None)
  138. class ReceivePackHandler(Handler):
  139. def handle(self):
  140. refs = self.backend.get_refs()
  141. if refs:
  142. self.write_pkt_line("%s %s\x00multi_ack side-band-64k thin-pack ofs-delta\n" % (refs[0][1], refs[0][0]))
  143. for i in range(1, len(refs)):
  144. ref = refs[i]
  145. self.write_pkt_line("%s %s\n" % (ref[1], ref[0]))
  146. self.write("0000")
  147. client_refs = []
  148. ref = self.read_pkt_line()
  149. while ref:
  150. client_refs.append(ref.split())
  151. ref = self.read_pkt_line()
  152. if len(client_refs) == 0:
  153. return None
  154. self.backend.apply_pack(client_refs, self.read)
  155. class TCPGitRequestHandler(SocketServer.StreamRequestHandler, Handler):
  156. def __init__(self, request, client_address, server):
  157. SocketServer.StreamRequestHandler.__init__(self, request, client_address, server)
  158. def handle(self):
  159. #FIXME: StreamRequestHandler seems to be the thing that calls handle(),
  160. #so we can't call this in a sane place??
  161. Handler.__init__(self, self.server.backend, self.rfile.read, self.wfile.write)
  162. request = self.read_pkt_line()
  163. # up until the space is the command to run, everything after is parameters
  164. splice_point = request.find(' ')
  165. command, params = request[:splice_point], request[splice_point+1:]
  166. # params are null seperated
  167. params = params.split(chr(0))
  168. # switch case to handle the specific git command
  169. if command == 'git-upload-pack':
  170. cls = UploadPackHandler
  171. elif command == 'git-receive-pack':
  172. cls = ReceivePackHandler
  173. else:
  174. return
  175. h = cls(self.backend, self.read, self.write)
  176. h.handle()
  177. class TCPGitServer(SocketServer.TCPServer):
  178. allow_reuse_address = True
  179. serve = SocketServer.TCPServer.serve_forever
  180. def __init__(self, backend, addr):
  181. self.backend = backend
  182. SocketServer.TCPServer.__init__(self, addr, TCPGitRequestHandler)