From 8df6d98c1b7f06ba1d394908c6e8c333c7c5d517 Mon Sep 17 00:00:00 2001 From: Folker Bernitt Date: Thu, 22 Oct 2015 13:38:01 +0200 Subject: Move interstitial to root resource - Issue #501 - Moves assets to startup-assets - No more switching between twisted resources at runtime --- service/pixelated/application.py | 31 ++++++++---------- service/pixelated/assets/Interstitial.html | 6 ++-- service/pixelated/resources/loading_page.py | 49 ---------------------------- service/pixelated/resources/root_resource.py | 24 +++++++++++++- service/test/unit/test_application.py | 36 ++++++++++++++------ 5 files changed, 66 insertions(+), 80 deletions(-) delete mode 100644 service/pixelated/resources/loading_page.py (limited to 'service') diff --git a/service/pixelated/application.py b/service/pixelated/application.py index dfeb8d82..082f45ee 100644 --- a/service/pixelated/application.py +++ b/service/pixelated/application.py @@ -25,7 +25,6 @@ from pixelated.config.services import Services from pixelated.config.leap import initialize_leap from pixelated.config import logger from pixelated.config.site import PixelatedSite -from pixelated.resources.loading_page import LoadingResource from pixelated.resources.root_resource import RootResource from leap.common.events import ( @@ -35,26 +34,18 @@ from leap.common.events import ( @defer.inlineCallbacks -def start_user_agent(loading_app, host, port, sslkey, sslcert, leap_home, leap_session): - yield loading_app.stopListening() +def start_user_agent(root_resource, leap_home, leap_session): services = Services(leap_home, leap_session) yield services.setup(leap_home, leap_session) - resource = RootResource() - - resource.initialize( + root_resource.initialize( services.keymanager, services.search_engine, services.mail_service, services.draft_service, services.feedback_service) - if sslkey and sslcert: - reactor.listenSSL(port, PixelatedSite(resource), _ssl_options(sslkey, sslcert), interface=host) - else: - reactor.listenTCP(port, PixelatedSite(resource), interface=host) - # soledad needs lots of threads reactor.threadpool.adjustPoolsize(5, 15) @@ -77,7 +68,9 @@ def _ssl_options(sslkey, sslcert): def initialize(): args = arguments.parse_user_agent_args() logger.init(debug=args.debug) - loading_app = reactor.listenTCP(args.port, PixelatedSite(LoadingResource()), interface=args.host) + resource = RootResource() + + start_site(args, resource) deferred = initialize_leap(args.leap_provider_cert, args.leap_provider_cert_fingerprint, @@ -87,11 +80,7 @@ def initialize(): deferred.addCallback( lambda leap_session: start_user_agent( - loading_app, - args.host, - args.port, - args.sslkey, - args.sslcert, + resource, args.leap_home, leap_session)) @@ -107,3 +96,11 @@ def initialize(): deferred.addErrback(_quit_on_error) reactor.run() + + +def start_site(config, resource): + if config.sslkey and config.sslcert: + reactor.listenSSL(config.port, PixelatedSite(resource), _ssl_options(config.sslkey, config.sslcert), + interface=config.host) + else: + reactor.listenTCP(config.port, PixelatedSite(resource), interface=config.host) diff --git a/service/pixelated/assets/Interstitial.html b/service/pixelated/assets/Interstitial.html index e98080be..bc6cc738 100644 --- a/service/pixelated/assets/Interstitial.html +++ b/service/pixelated/assets/Interstitial.html @@ -4,15 +4,15 @@ - - + +
- + diff --git a/service/pixelated/resources/loading_page.py b/service/pixelated/resources/loading_page.py deleted file mode 100644 index 84388068..00000000 --- a/service/pixelated/resources/loading_page.py +++ /dev/null @@ -1,49 +0,0 @@ -# -# Copyright (c) 2014 ThoughtWorks, Inc. -# -# Pixelated is free software: you can redistribute it and/or modify -# it under the terms of the GNU Affero General Public License as published by -# the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# Pixelated is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU Affero General Public License for more details. -# -# You should have received a copy of the GNU Affero General Public License -# along with Pixelated. If not, see . - -import os -from twisted.internet.threads import deferToThread -from twisted.web.resource import Resource -from twisted.web.server import NOT_DONE_YET -from twisted.web.static import File - - -class LoadingResource(Resource): - - def __init__(self): - Resource.__init__(self) - self._path = os.path.dirname(os.path.abspath(__file__)) - self.putChild('assets', File(os.path.join(self._path, '..', 'assets'))) - - def render_GET(self, request): - - def open_html(): - return open(os.path.join(self._path, '..', 'assets', 'Interstitial.html')).read() - - def close_request(html): - request.responseHeaders.addRawHeader("Connection", "close") - request.write(html) - request.finish() - - d = deferToThread(open_html) - d.addCallback(close_request) - - return NOT_DONE_YET - - def getChild(self, path, request): - if path == '': - return self - return Resource.getChild(self, path, request) diff --git a/service/pixelated/resources/root_resource.py b/service/pixelated/resources/root_resource.py index 8b536450..e6046eae 100644 --- a/service/pixelated/resources/root_resource.py +++ b/service/pixelated/resources/root_resource.py @@ -12,11 +12,21 @@ from twisted.web.resource import Resource from twisted.web.static import File +MODE_STARTUP = 1 +MODE_RUNNING = 2 + + class RootResource(Resource): def __init__(self): Resource.__init__(self) + self._startup_assets_folder = self._get_startup_folder() self._static_folder = self._get_static_folder() + self._startup_mode() + + def _startup_mode(self): + self.putChild('startup-assets', File(self._startup_assets_folder)) + self._mode = MODE_STARTUP def getChild(self, path, request): if path == '': @@ -34,6 +44,12 @@ class RootResource(Resource): self.putChild('mail', MailResource(mail_service)) self.putChild('feedback', FeedbackResource(feedback_service)) + self._mode = MODE_RUNNING + + def _get_startup_folder(self): + path = os.path.dirname(os.path.abspath(__file__)) + return os.path.join(path, '..', 'assets') + def _get_static_folder(self): static_folder = os.path.abspath(os.path.join(os.path.abspath(__file__), "..", "..", "..", "web-ui", "app")) # this is a workaround for packaging @@ -44,5 +60,11 @@ class RootResource(Resource): static_folder = os.path.join('/', 'usr', 'share', 'pixelated-user-agent') return static_folder + def _is_starting(self): + return self._mode == MODE_STARTUP + def render_GET(self, request): - return open(os.path.join(self._static_folder, 'index.html')).read() + if self._is_starting(): + return open(os.path.join(self._startup_assets_folder, 'Interstitial.html')).read() + else: + return open(os.path.join(self._static_folder, 'index.html')).read() diff --git a/service/test/unit/test_application.py b/service/test/unit/test_application.py index 16317ee5..2f9ee742 100644 --- a/service/test/unit/test_application.py +++ b/service/test/unit/test_application.py @@ -14,30 +14,49 @@ class ApplicationTest(unittest.TestCase): self.sslcert = sslcert self.home = 'leap_home' + @patch('leap.common.events.client') + @patch('pixelated.application.reactor') + def test_that_start_site_binds_to_tcp_port_if_no_ssl_options(self, reactor_mock, _): + app_mock = MagicMock() + config = ApplicationTest.MockConfig(12345, '127.0.0.1') + + pixelated.application.start_site(config, app_mock) + + reactor_mock.listenTCP.assert_called_once_with(12345, ANY, interface='127.0.0.1') + + @patch('leap.common.events.client') + @patch('pixelated.application.reactor') + def test_that_start_site_binds_to_ssl_if_ssl_options(self, reactor_mock, _): + app_mock = MagicMock() + pixelated.application._ssl_options = lambda x, y: 'options' + + config = ApplicationTest.MockConfig(12345, '127.0.0.1', sslkey="sslkey", sslcert="sslcert") + + pixelated.application.start_site(config, app_mock) + + reactor_mock.listenSSL.assert_called_once_with(12345, ANY, 'options', interface='127.0.0.1') + @patch('leap.common.events.client') @patch('pixelated.application.reactor') @patch('pixelated.application.Services') - def test_that_create_app_binds_to_tcp_port_if_no_ssl_options(self, services_mock, reactor_mock, _): + def test_that_start_user_agent_binds_to_tcp_port_if_no_ssl_options(self, services_mock, reactor_mock, _): # FIXME patch something closer, instead of leap.common app_mock = MagicMock() leap_session = MagicMock() config = ApplicationTest.MockConfig(12345, '127.0.0.1', leap_session) - d = pixelated.application.start_user_agent(app_mock, config.host, config.port, config.sslkey, config.sslcert, config.home, leap_session) + d = pixelated.application.start_user_agent(app_mock, config.home, leap_session) def _assert(_): services_mock.assert_called_once_with(config.home, leap_session) - reactor_mock.listenTCP.assert_called_once_with(12345, ANY, interface='127.0.0.1') - app_mock.stopListening.assert_called() - d.addCallback(_assert) return d @patch('leap.common.events.client') @patch('pixelated.application.reactor') @patch('pixelated.application.Services') - def test_that_create_app_binds_to_ssl_if_ssl_options(self, services_mock, reactor_mock, _): + def test_that_start_user_agent_binds_to_ssl_if_ssl_options(self, services_mock, reactor_mock, _): # FIXME patch something closer, instead of leap.common app_mock = MagicMock() leap_session = MagicMock() @@ -45,13 +64,10 @@ class ApplicationTest(unittest.TestCase): config = ApplicationTest.MockConfig(12345, '127.0.0.1', sslkey="sslkey", sslcert="sslcert") - d = pixelated.application.start_user_agent(app_mock, config.host, config.port, config.sslkey, config.sslcert, config.home, leap_session) + d = pixelated.application.start_user_agent(app_mock, config.home, leap_session) def _assert(_): services_mock.assert_called_once_with(config.home, leap_session) - reactor_mock.listenSSL.assert_called_once_with(12345, ANY, 'options', interface='127.0.0.1') - app_mock.stopListening.assert_called() - d.addCallback(_assert) return d -- cgit v1.2.3