forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			97 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			97 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""
 | 
						|
Utilities helpful for writing tests
 | 
						|
 | 
						|
Provides a UnixSocketServerThread that creates a running server, listening on a
 | 
						|
newly created unix socket.
 | 
						|
 | 
						|
Example usage:
 | 
						|
 | 
						|
.. code-block:: python
 | 
						|
 | 
						|
    def test_unix_domain_adapter_monkeypatch():
 | 
						|
        with UnixSocketServerThread() as usock_thread:
 | 
						|
            with requests_unixsocket.monkeypatch('http+unix://'):
 | 
						|
                urlencoded_usock = quote_plus(usock_process.usock)
 | 
						|
                url = 'http+unix://%s/path/to/page' % urlencoded_usock
 | 
						|
                r = requests.get(url)
 | 
						|
"""
 | 
						|
 | 
						|
import logging
 | 
						|
import os
 | 
						|
import threading
 | 
						|
import time
 | 
						|
import uuid
 | 
						|
import waitress
 | 
						|
 | 
						|
 | 
						|
logger = logging.getLogger(__name__)
 | 
						|
 | 
						|
 | 
						|
class KillThread(threading.Thread):
 | 
						|
    def __init__(self, server, *args, **kwargs):
 | 
						|
        super(KillThread, self).__init__(*args, **kwargs)
 | 
						|
        self.server = server
 | 
						|
 | 
						|
    def run(self):
 | 
						|
        time.sleep(1)
 | 
						|
        logger.debug('Sleeping')
 | 
						|
        self.server._map.clear()
 | 
						|
 | 
						|
 | 
						|
class WSGIApp:
 | 
						|
    server = None
 | 
						|
 | 
						|
    def __call__(self, environ, start_response):
 | 
						|
        logger.debug('WSGIApp.__call__: Invoked for %s', environ['PATH_INFO'])
 | 
						|
        logger.debug('WSGIApp.__call__: environ = %r', environ)
 | 
						|
        status_text = '200 OK'
 | 
						|
        response_headers = [
 | 
						|
            ('X-Transport', 'unix domain socket'),
 | 
						|
            ('X-Socket-Path', environ['SERVER_PORT']),
 | 
						|
            ('X-Requested-Query-String', environ['QUERY_STRING']),
 | 
						|
            ('X-Requested-Path', environ['PATH_INFO'])]
 | 
						|
        body_bytes = b'Hello world!'
 | 
						|
        start_response(status_text, response_headers)
 | 
						|
        logger.debug(
 | 
						|
            'WSGIApp.__call__: Responding with '
 | 
						|
            'status_text = %r; '
 | 
						|
            'response_headers = %r; '
 | 
						|
            'body_bytes = %r',
 | 
						|
            status_text, response_headers, body_bytes)
 | 
						|
        return [body_bytes]
 | 
						|
 | 
						|
 | 
						|
class UnixSocketServerThread(threading.Thread):
 | 
						|
    def __init__(self, *args, **kwargs):
 | 
						|
        super(UnixSocketServerThread, self).__init__(*args, **kwargs)
 | 
						|
        self.usock = self.get_tempfile_name()
 | 
						|
        self.server = None
 | 
						|
        self.server_ready_event = threading.Event()
 | 
						|
 | 
						|
    def get_tempfile_name(self):
 | 
						|
        # I'd rather use tempfile.NamedTemporaryFile but IDNA limits
 | 
						|
        # the hostname to 63 characters and we'll get a "InvalidURL:
 | 
						|
        # URL has an invalid label" error if we exceed that.
 | 
						|
        args = (os.stat(__file__).st_ino, os.getpid(), uuid.uuid4().hex[-8:])
 | 
						|
        return '/tmp/test_requests.%s_%s_%s' % args
 | 
						|
 | 
						|
    def run(self):
 | 
						|
        logger.debug('Call waitress.serve in %r ...', self)
 | 
						|
        wsgi_app = WSGIApp()
 | 
						|
        server = waitress.create_server(wsgi_app, unix_socket=self.usock)
 | 
						|
        wsgi_app.server = server
 | 
						|
        self.server = server
 | 
						|
        self.server_ready_event.set()
 | 
						|
        server.run()
 | 
						|
 | 
						|
    def __enter__(self):
 | 
						|
        logger.debug('Starting %r ...' % self)
 | 
						|
        self.start()
 | 
						|
        logger.debug('Started %r.', self)
 | 
						|
        self.server_ready_event.wait()
 | 
						|
        return self
 | 
						|
 | 
						|
    def __exit__(self, *args):
 | 
						|
        self.server_ready_event.wait()
 | 
						|
        if self.server:
 | 
						|
            KillThread(self.server).start()
 |