local_rietveld.py 4.8 KB
Newer Older
1
#!/usr/bin/env python
2
# Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 4 5 6 7 8 9 10 11 12
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.

"""Setups a local Rietveld instance to test against a live server for
integration tests.

It makes sure Google AppEngine SDK is found, download Rietveld and Django code
if necessary and starts the server on a free inbound TCP port.
"""

13
import optparse
14
import os
15
import shutil
16
import socket
17
import sys
18 19
import time

20 21 22 23 24 25
try:
  import subprocess2
except ImportError:
  sys.path.append(
      os.path.join(os.path.dirname(os.path.abspath(__file__)), '..'))
  import subprocess2
26

27 28

class Failure(Exception):
29
  pass
30 31 32


def test_port(port):
33 34 35 36 37
  s = socket.socket()
  try:
    return s.connect_ex(('127.0.0.1', port)) == 0
  finally:
    s.close()
38 39 40


def find_free_port():
41 42 43 44 45 46 47 48
  # Test to find an available port starting at 8080.
  port = 8080
  max_val = (2<<16)
  while test_port(port) and port < max_val:
    port += 1
  if port == max_val:
    raise Failure('Having issues finding an available port')
  return port
49 50 51


class LocalRietveld(object):
52 53 54 55 56 57 58
  """Downloads everything needed to run a local instance of Rietveld."""

  def __init__(self, base_dir=None):
    # Paths
    self.base_dir = base_dir
    if not self.base_dir:
      self.base_dir = os.path.dirname(os.path.abspath(__file__))
59 60 61
    # TODO(maruel): This should be in /tmp but that would mean having to fetch
    # everytime. This test is already annoyingly slow.
    self.rietveld = os.path.join(self.base_dir, '_rietveld')
62 63 64
    self.test_server = None
    self.port = None

65 66 67 68 69 70 71 72 73 74 75
    # Find the GAE SDK
    previous_dir = ''
    self.sdk_path = ''
    base_dir = self.base_dir
    while base_dir != previous_dir:
      previous_dir = base_dir
      self.sdk_path = os.path.join(base_dir, 'google_appengine')
      if not os.path.isfile(os.path.join(self.sdk_path, 'VERSION')):
        base_dir = os.path.dirname(base_dir)
    self.dev_app = os.path.join(self.sdk_path, 'dev_appserver.py')

76 77 78
  def install_prerequisites(self):
    # First, verify the Google AppEngine SDK is available.
    if not os.path.isfile(self.dev_app):
79 80
      raise Failure(
          'Install google_appengine sdk in %s or higher up' % self.base_dir)
81

82 83 84 85
    if os.path.isdir(os.path.join(self.rietveld, '.svn')):
      # Left over from subversion. Delete it.
      shutil.rmtree(self.rietveld)

86
    # Second, checkout rietveld if not available.
87
    rev = '9349cab9a3bb'
88 89
    if not os.path.isdir(self.rietveld):
      print('Checking out rietveld...')
90 91
      try:
        subprocess2.check_call(
92 93
            [ 'hg', 'clone', '-q', '-u', rev, '-r', rev,
              'https://code.google.com/p/rietveld/', self.rietveld])
94
      except (OSError, subprocess2.CalledProcessError), e:
95 96 97
        raise Failure(
            'Failed to checkout rietveld. Do you have mercurial installed?\n'
            '%s' % e)
98 99
    else:
      print('Syncing rietveld...')
100 101
      try:
        subprocess2.check_call(
102
            ['hg', 'co', '-q', '-C', rev], cwd=self.rietveld)
103 104
      except (OSError, subprocess2.CalledProcessError), e:
        raise Failure('Failed to sync rietveld\n%s' % e)
105 106 107 108 109

  def start_server(self, verbose=False):
    self.install_prerequisites()
    self.port = find_free_port()
    if verbose:
110
      pipe = None
111
    else:
112
      pipe = subprocess2.VOID
113
    cmd = [
114
        sys.executable,
115 116 117 118 119 120
        self.dev_app,
        '--skip_sdk_update_check',
        '.',
        '--port=%d' % self.port,
        '--datastore_path=' + os.path.join(self.rietveld, 'tmp.db'),
        '-c']
121 122 123 124 125 126 127 128

    # CHEAP TRICK
    # By default you only want to bind on loopback but I'm testing over a
    # headless computer so it's useful to be able to access the test instance
    # remotely.
    if os.environ.get('GAE_LISTEN_ALL', '') == 'true':
      cmd.extend(('-a', '0.0.0.0'))

129
    self.test_server = subprocess2.Popen(
130
        cmd, stdout=pipe, stderr=pipe, cwd=self.rietveld)
131 132 133 134 135 136 137
    # Loop until port 127.0.0.1:port opens or the process dies.
    while not test_port(self.port):
      self.test_server.poll()
      if self.test_server.returncode is not None:
        raise Failure(
            'Test rietveld instance failed early on port %s' %
            self.port)
138
      time.sleep(0.01)
139 140 141 142

  def stop_server(self):
    if self.test_server:
      self.test_server.kill()
143
      self.test_server.wait()
144 145
      self.test_server = None
      self.port = None
146 147 148


def main():
149 150 151 152 153 154 155 156 157 158 159 160 161
  parser = optparse.OptionParser()
  parser.add_option('-v', '--verbose', action='store_true')
  options, args = parser.parse_args()
  if args:
    parser.error('Unknown arguments: %s' % ' '.join(args))
  instance = LocalRietveld()
  try:
    instance.start_server(verbose=options.verbose)
    print 'Local rietveld instance started on port %d' % instance.port
    while True:
      time.sleep(0.1)
  finally:
    instance.stop_server()
162 163 164


if __name__ == '__main__':
165
  main()