You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

87 lines
2.3 KiB

  1. """Basic http server for tests to simulate PyPI or custom indexes"""
  2. import http.server
  3. import os
  4. import threading
  5. import time
  6. import urllib.parse
  7. import urllib.request
  8. class IndexServer(http.server.HTTPServer):
  9. """Basic single-threaded http server simulating a package index
  10. You can use this server in unittest like this::
  11. s = IndexServer()
  12. s.start()
  13. index_url = s.base_url() + 'mytestindex'
  14. # do some test requests to the index
  15. # The index files should be located in setuptools/tests/indexes
  16. s.stop()
  17. """
  18. def __init__(
  19. self,
  20. server_address=('', 0),
  21. RequestHandlerClass=http.server.SimpleHTTPRequestHandler,
  22. ):
  23. http.server.HTTPServer.__init__(self, server_address, RequestHandlerClass)
  24. self._run = True
  25. def start(self):
  26. self.thread = threading.Thread(target=self.serve_forever)
  27. self.thread.start()
  28. def stop(self):
  29. "Stop the server"
  30. # Let the server finish the last request and wait for a new one.
  31. time.sleep(0.1)
  32. self.shutdown()
  33. self.thread.join()
  34. self.socket.close()
  35. def base_url(self):
  36. port = self.server_port
  37. return f'http://127.0.0.1:{port}/setuptools/tests/indexes/'
  38. class RequestRecorder(http.server.BaseHTTPRequestHandler):
  39. def do_GET(self):
  40. requests = vars(self.server).setdefault('requests', [])
  41. requests.append(self)
  42. self.send_response(200, 'OK')
  43. class MockServer(http.server.HTTPServer, threading.Thread):
  44. """
  45. A simple HTTP Server that records the requests made to it.
  46. """
  47. def __init__(self, server_address=('', 0), RequestHandlerClass=RequestRecorder):
  48. http.server.HTTPServer.__init__(self, server_address, RequestHandlerClass)
  49. threading.Thread.__init__(self)
  50. self.daemon = True
  51. self.requests = []
  52. def run(self):
  53. self.serve_forever()
  54. @property
  55. def netloc(self):
  56. return f'localhost:{self.server_port}'
  57. @property
  58. def url(self):
  59. return f'http://{self.netloc}/'
  60. def path_to_url(path, authority=None):
  61. """Convert a path to a file: URL."""
  62. path = os.path.normpath(os.path.abspath(path))
  63. base = 'file:'
  64. if authority is not None:
  65. base += '//' + authority
  66. return urllib.parse.urljoin(base, urllib.request.pathname2url(path))