165 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
	
| import atexit
 | |
| import BaseHTTPServer
 | |
| import errno
 | |
| import os
 | |
| import shutil
 | |
| import signal
 | |
| import socket
 | |
| import subprocess
 | |
| import sys
 | |
| import tempfile
 | |
| import threading
 | |
| import time
 | |
| 
 | |
| 
 | |
| class ToSServerThread(threading.Thread):
 | |
|     class ToSHandler(BaseHTTPServer.BaseHTTPRequestHandler):
 | |
|         def do_GET(self):
 | |
|             self.send_response(200)
 | |
|             self.end_headers()
 | |
|             self.wfile.write("Do What Ye Will (An it Harm None).\n")
 | |
|     def run(self):
 | |
|         try:
 | |
|             BaseHTTPServer.HTTPServer(("localhost", 4001), self.ToSHandler).serve_forever()
 | |
|         except Exception as e:
 | |
|             print "Problem starting ToSServer: %s" % e
 | |
|             sys.exit(1)
 | |
| 
 | |
| 
 | |
| default_config = os.environ.get('BOULDER_CONFIG')
 | |
| if default_config is None:
 | |
|     default_config = 'test/boulder-config.json'
 | |
| processes = []
 | |
| 
 | |
| 
 | |
| def install(race_detection):
 | |
|     # Pass empty BUILD_TIME and BUILD_ID flags to avoid constantly invalidating the
 | |
|     # build cache with new BUILD_TIMEs, or invalidating it on merges with a new
 | |
|     # BUILD_ID.
 | |
|     cmd = "make GO_BUILD_FLAGS=''  "
 | |
|     if race_detection:
 | |
|         cmd = "make GO_BUILD_FLAGS=-race"
 | |
| 
 | |
|     return subprocess.call(cmd, shell=True) == 0
 | |
| 
 | |
| def run(binary, race_detection, config=default_config):
 | |
|     # Note: Must use exec here so that killing this process kills the command.
 | |
|     cmd = """GORACE="halt_on_error=1" exec ./bin/%s --config %s""" % (binary, config)
 | |
|     p = subprocess.Popen(cmd, shell=True)
 | |
|     p.cmd = cmd
 | |
|     print('started %s with pid %d' % (p.cmd, p.pid))
 | |
|     return p
 | |
| 
 | |
| def start(race_detection):
 | |
|     """Return True if everything builds and starts.
 | |
| 
 | |
|     Give up and return False if anything fails to build, or dies at
 | |
|     startup. Anything that did start before this point can be cleaned
 | |
|     up explicitly by calling stop(), or automatically atexit.
 | |
|     """
 | |
|     global processes
 | |
|     forward()
 | |
|     t = ToSServerThread()
 | |
|     t.daemon = True
 | |
|     t.start()
 | |
|     progs = [
 | |
|         'activity-monitor',
 | |
|         'boulder-wfe',
 | |
|         'boulder-ra',
 | |
|         'boulder-sa',
 | |
|         'boulder-ca',
 | |
|         'boulder-va',
 | |
|         'boulder-publisher',
 | |
|         'ocsp-updater',
 | |
|         'ocsp-responder',
 | |
|         'ct-test-srv',
 | |
|         'dns-test-srv'
 | |
|     ]
 | |
|     if not install(race_detection):
 | |
|         return False
 | |
|     for prog in progs:
 | |
|         try:
 | |
|             processes.append(run(prog, race_detection))
 | |
|         except Exception as e:
 | |
|             print(e)
 | |
|             return False
 | |
|         if not check():
 | |
|             # Don't keep building stuff if a server has already died.
 | |
|             return False
 | |
| 
 | |
|     # Additionally run the issuer-ocsp-responder, which is not amenable to the
 | |
|     # above `run` pattern because it uses a different config file.
 | |
|     try:
 | |
|         processes.append(run('ocsp-responder', race_detection, 'test/issuer-ocsp-responder.json'))
 | |
|     except Exception as e:
 | |
|         print(e)
 | |
|         return False
 | |
| 
 | |
|     # Wait until all servers are up before returning to caller. This means
 | |
|     # checking each server's debug port until it's available.
 | |
|     # seconds.
 | |
|     while True:
 | |
|         try:
 | |
|             # If one of the servers has died, quit immediately.
 | |
|             if not check():
 | |
|                 return False
 | |
|             ports = range(8000, 8005) + [4000]
 | |
|             for debug_port in ports:
 | |
|                 s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
 | |
|                 s.connect(('localhost', debug_port))
 | |
|                 s.close()
 | |
|             break
 | |
|         except socket.error as e:
 | |
|             if e.errno == errno.ECONNREFUSED:
 | |
|                 print "Waiting for debug port %d" % debug_port
 | |
|             else:
 | |
|                 raise
 | |
|         time.sleep(1)
 | |
| 
 | |
|     # Some servers emit extra text after their debug server is open. Sleep 1
 | |
|     # second so the "servers running" message comes last.
 | |
|     time.sleep(1)
 | |
|     print "All servers running. Hit ^C to kill."
 | |
|     return True
 | |
| 
 | |
| def forward():
 | |
|     """Add a TCP forwarder between Boulder and RabbitMQ to simulate failures."""
 | |
|     cmd = """exec listenbuddy -listen :5673 -speak localhost:5672"""
 | |
|     p = subprocess.Popen(cmd, shell=True)
 | |
|     p.cmd = cmd
 | |
|     print('started %s with pid %d' % (p.cmd, p.pid))
 | |
|     global processes
 | |
|     processes.insert(0, p)
 | |
| 
 | |
| def bounce_forward():
 | |
|     """Kill all forwarded TCP connections."""
 | |
|     global processes
 | |
|     processes[0].send_signal(signal.SIGUSR1)
 | |
| 
 | |
| def check():
 | |
|     """Return true if all started processes are still alive.
 | |
| 
 | |
|     Log about anything that died.
 | |
|     """
 | |
|     global processes
 | |
|     busted = []
 | |
|     stillok = []
 | |
|     for p in processes:
 | |
|         if p.poll() is None:
 | |
|             stillok.append(p)
 | |
|         else:
 | |
|             busted.append(p)
 | |
|     if busted:
 | |
|         print "\n\nThese processes exited early (check above for their output):"
 | |
|         for p in busted:
 | |
|             print "\t'%s' with pid %d exited %d" % (p.cmd, p.pid, p.returncode)
 | |
|     processes = stillok
 | |
|     return not busted
 | |
| 
 | |
| 
 | |
| @atexit.register
 | |
| def stop():
 | |
|     for p in processes:
 | |
|         if p.poll() is None:
 | |
|             p.kill()
 |