summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorLars-Dominik Braun <lars@6xq.net>2018-05-01 15:43:34 +0200
committerLars-Dominik Braun <lars@6xq.net>2018-05-04 16:00:05 +0200
commitde2b4036ac7ce2d242d49fe37337db890c30da3b (patch)
tree9bb0b6bb5813a5c3e6df684f4098a53760671ff7
parenteb818f0c6eb86461a0db1845876f2a0b39b99b7f (diff)
downloadcrocoite-de2b4036ac7ce2d242d49fe37337db890c30da3b.tar.gz
crocoite-de2b4036ac7ce2d242d49fe37337db890c30da3b.tar.bz2
crocoite-de2b4036ac7ce2d242d49fe37337db890c30da3b.zip
browser: Replace context manager decorator
Use an actual class that supports multiple invokations.
-rw-r--r--crocoite/browser.py117
1 files changed, 66 insertions, 51 deletions
diff --git a/crocoite/browser.py b/crocoite/browser.py
index 7250b11..9e59556 100644
--- a/crocoite/browser.py
+++ b/crocoite/browser.py
@@ -340,64 +340,79 @@ class AccountingSiteLoader (SiteLoader):
import subprocess
from tempfile import mkdtemp
-from contextlib import contextmanager
import socket, shutil
-@contextmanager
-def ChromeService (binary='google-chrome-stable', host='localhost', port=9222, windowSize=(1920, 1080)):
+class ChromeService:
"""
Start Chrome with socket activation (i.e. pass listening socket). Polling
is not required with this method, since reads will block until Chrome is
ready.
"""
- while True:
- s = socket.socket ()
- s.setsockopt (socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
- try:
- s.bind ((host, port))
- break
- except OSError:
- # try different port
- if port < 65000:
- port += 1
- else:
- raise
- s.listen (10)
- userDataDir = mkdtemp ()
- args = [binary,
- '--window-size={},{}'.format (*windowSize),
- '--user-data-dir={}'.format (userDataDir), # use temporory user dir
- '--no-default-browser-check',
- '--no-first-run', # don’t show first run screen
- '--disable-breakpad', # no error reports
- '--disable-extensions',
- '--disable-infobars',
- '--disable-notifications', # no libnotify
- '--headless',
- '--disable-gpu',
- '--hide-scrollbars', # hide scrollbars on screenshots
- '--mute-audio', # don’t play any audio
- '--remote-debugging-socket-fd={}'.format (s.fileno ()),
- '--homepage=about:blank',
- 'about:blank']
- # start new session, so ^C does not affect subprocess
- p = subprocess.Popen (args, pass_fds=[s.fileno()], start_new_session=True,
- stdin=subprocess.DEVNULL, stdout=subprocess.DEVNULL,
- stderr=subprocess.DEVNULL)
- s.close ()
-
- # must be wrapped in try-finally, otherwise code in __exit__/finally is not
- # executed
- try:
- yield 'http://{}:{}'.format (host, port)
- finally:
- p.terminate ()
- p.wait ()
- shutil.rmtree (userDataDir)
-
-@contextmanager
-def NullService (url):
- yield url
+
+ def __init__ (self, binary='google-chrome-stable', host='localhost', port=9222, windowSize=(1920, 1080)):
+ self.binary = binary
+ self.host = host
+ self.port = port
+ self.windowSize = windowSize
+ self.p = None
+
+ def __enter__ (self):
+ assert self.p is None
+
+ port = self.port
+ while True:
+ s = socket.socket ()
+ s.setsockopt (socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
+ try:
+ s.bind ((self.host, port))
+ break
+ except OSError:
+ # try different port
+ if port < 65000:
+ port += 1
+ else:
+ raise
+ s.listen (10)
+ self.userDataDir = mkdtemp ()
+ args = [self.binary,
+ '--window-size={},{}'.format (*self.windowSize),
+ '--user-data-dir={}'.format (self.userDataDir), # use temporory user dir
+ '--no-default-browser-check',
+ '--no-first-run', # don’t show first run screen
+ '--disable-breakpad', # no error reports
+ '--disable-extensions',
+ '--disable-infobars',
+ '--disable-notifications', # no libnotify
+ '--headless',
+ '--disable-gpu',
+ '--hide-scrollbars', # hide scrollbars on screenshots
+ '--mute-audio', # don’t play any audio
+ '--remote-debugging-socket-fd={}'.format (s.fileno ()),
+ '--homepage=about:blank',
+ 'about:blank']
+ # start new session, so ^C does not affect subprocess
+ self.p = subprocess.Popen (args, pass_fds=[s.fileno()], start_new_session=True,
+ stdin=subprocess.DEVNULL, stdout=subprocess.DEVNULL,
+ stderr=subprocess.DEVNULL)
+ s.close ()
+
+ return 'http://{}:{}'.format (self.host, port)
+
+ def __exit__ (self, *exc):
+ self.p.terminate ()
+ self.p.wait ()
+ shutil.rmtree (self.userDataDir)
+ self.p = None
+
+class NullService:
+ def __init__ (self, url):
+ self.url = url
+
+ def __enter__ (self):
+ return url
+
+ def __exit__ (self, *exc):
+ pass
### tests ###