| # subprocess does not kill the child daemon when a test case fails by raising |
| # an exception. So use pexpect instead. |
| import infra |
| |
| import pexpect |
| |
| |
| GIT_REMOTE_PORT_INITIAL = 9418 |
| GIT_REMOTE_PORT_LAST = GIT_REMOTE_PORT_INITIAL + 99 |
| |
| |
| class GitRemote(object): |
| def __init__(self, builddir, serveddir, logtofile): |
| """ |
| Start a local git server. |
| |
| In order to support test cases in parallel, select the port the |
| server will listen to in runtime. Since there is no reliable way |
| to allocate the port prior to starting the server (another |
| process in the host machine can use the port between it is |
| selected from a list and it is really allocated to the server) |
| try to start the server in a port and in the case it is already |
| in use, try the next one in the allowed range. |
| """ |
| self.daemon = None |
| self.port = None |
| self.logfile = infra.open_log_file(builddir, "gitremote", logtofile) |
| |
| daemon_cmd = ["git", "daemon", "--reuseaddr", "--verbose", |
| "--listen=localhost", "--export-all", |
| "--base-path={}".format(serveddir)] |
| for port in range(GIT_REMOTE_PORT_INITIAL, GIT_REMOTE_PORT_LAST + 1): |
| cmd = daemon_cmd + ["--port={port}".format(port=port)] |
| self.logfile.write("> starting git remote with '{}'\n".format(" ".join(cmd))) |
| self.daemon = pexpect.spawn(cmd[0], cmd[1:], logfile=self.logfile, |
| encoding='utf-8') |
| ret = self.daemon.expect(["Ready to rumble", |
| "Address already in use"]) |
| if ret == 0: |
| self.port = port |
| return |
| raise SystemError("Could not find a free port to run git remote") |
| |
| def stop(self): |
| if self.daemon is None: |
| return |
| self.daemon.terminate(force=True) |