| 
									
										
										
										
											2013-11-27 14:53:57 +01:00
										 |  |  | #!/usr/bin/python2 | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Authors: | 
					
						
							|  |  |  | #   Jason Gerard DeRose <jderose@redhat.com> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Copyright (C) 2008  Red Hat | 
					
						
							|  |  |  | # see file 'COPYING' for use and warranty information | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2010-12-09 13:59:11 +01:00
										 |  |  | # This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  | # it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | # the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | # (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | # | 
					
						
							|  |  |  | # This program 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 General Public License for more details. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # You should have received a copy of the GNU General Public License | 
					
						
							| 
									
										
										
										
											2010-12-09 13:59:11 +01:00
										 |  |  | # along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | In-tree paste-based test server. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This uses the *Python Paste* WSGI server.  For more info, see: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     http://pythonpaste.org/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Unfortunately, SSL support is broken under Python 2.6 with paste 1.7.2, see: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     http://trac.pythonpaste.org/pythonpaste/ticket/314 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  | from os import path, getcwd | 
					
						
							| 
									
										
										
										
											2016-11-23 10:04:43 +01:00
										 |  |  | import optparse  # pylint: disable=deprecated-module | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | from paste import httpserver | 
					
						
							|  |  |  | import paste.gzipper | 
					
						
							|  |  |  | from paste.urlmap import URLMap | 
					
						
							|  |  |  | from ipalib import api | 
					
						
							| 
									
										
										
										
											2015-07-20 16:04:07 +02:00
										 |  |  | from subprocess import check_output, CalledProcessError | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Ugly hack for test purposes only. GSSAPI has no way to get default ccache | 
					
						
							|  |  |  | # name, but we don't need it outside test server | 
					
						
							|  |  |  | def get_default_ccache_name(): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         out = check_output(['klist']) | 
					
						
							|  |  |  |     except CalledProcessError: | 
					
						
							|  |  |  |         raise RuntimeError("Default ccache not found. Did you kinit?") | 
					
						
							|  |  |  |     match = re.match(r'^Ticket cache:\s*(\S+)', out) | 
					
						
							|  |  |  |     if not match: | 
					
						
							|  |  |  |         raise RuntimeError("Cannot obtain ccache name") | 
					
						
							|  |  |  |     return match.group(1) | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class KRBCheater(object): | 
					
						
							|  |  |  |     def __init__(self, app): | 
					
						
							|  |  |  |         self.app = app | 
					
						
							|  |  |  |         self.url = app.url | 
					
						
							| 
									
										
										
										
											2015-07-20 16:04:07 +02:00
										 |  |  |         self.ccname = get_default_ccache_name() | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __call__(self, environ, start_response): | 
					
						
							|  |  |  |         environ['KRB5CCNAME'] = self.ccname | 
					
						
							|  |  |  |         return self.app(environ, start_response) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  | class WebUIApp(object): | 
					
						
							| 
									
										
										
										
											2010-11-18 20:17:14 -06:00
										 |  |  |     INDEX_FILE = 'index.html' | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  |     EXTENSION_TO_MIME_MAP = { | 
					
						
							|  |  |  |         'xhtml': 'text/html', | 
					
						
							|  |  |  |         'html': 'text/html', | 
					
						
							|  |  |  |         'js': 'text/javascript', | 
					
						
							|  |  |  |         'inc': 'text/html', | 
					
						
							|  |  |  |         'css': 'text/css', | 
					
						
							|  |  |  |         'png': 'image/png', | 
					
						
							|  |  |  |         'json': 'text/javascript', | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         self.url = '/ipa/ui' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __call__(self, environ, start_response): | 
					
						
							|  |  |  |         path_info = environ['PATH_INFO'].lstrip('/') | 
					
						
							|  |  |  |         if path_info == '': | 
					
						
							|  |  |  |             path_info = self.INDEX_FILE | 
					
						
							| 
									
										
										
										
											2011-01-19 12:26:14 -05:00
										 |  |  |         requested_file = path.join(getcwd(), 'install/ui/', path_info) | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  |         extension = requested_file.rsplit('.', 1)[-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if extension not in self.EXTENSION_TO_MIME_MAP: | 
					
						
							|  |  |  |             start_response('404 Not Found', [('Content-Type', 'text/plain')]) | 
					
						
							|  |  |  |             return ['NOT FOUND'] | 
					
						
							|  |  |  |         mime_type = self.EXTENSION_TO_MIME_MAP[extension] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         f = None | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             f = open(requested_file, 'r') | 
					
						
							| 
									
										
										
										
											2011-12-02 14:27:47 -05:00
										 |  |  |             api.log.info('Request file %s' % requested_file) | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  |             start_response('200 OK', [('Content-Type', mime_type)]) | 
					
						
							|  |  |  |             return [f.read()] | 
					
						
							|  |  |  |         except IOError: | 
					
						
							|  |  |  |             start_response('404 Not Found', [('Content-Type', 'text/plain')]) | 
					
						
							|  |  |  |             return ['NOT FOUND'] | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             if f is not None: | 
					
						
							|  |  |  |                 f.close() | 
					
						
							| 
									
										
										
										
											2011-12-02 14:27:47 -05:00
										 |  |  |             api.log.info('Request done') | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     parser = optparse.OptionParser() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parser.add_option('--dev', | 
					
						
							|  |  |  |         help='Run WebUI in development mode (requires FireBug)', | 
					
						
							|  |  |  |         default=True, | 
					
						
							|  |  |  |         action='store_false', | 
					
						
							|  |  |  |         dest='prod', | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_option('--host', | 
					
						
							|  |  |  |         help='Listen on address HOST (default 127.0.0.1)', | 
					
						
							|  |  |  |         default='127.0.0.1', | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     parser.add_option('--port', | 
					
						
							|  |  |  |         help='Listen on PORT (default 8888)', | 
					
						
							|  |  |  |         default=8888, | 
					
						
							|  |  |  |         type='int', | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     api.env.in_server = True | 
					
						
							| 
									
										
										
										
											2010-06-25 13:37:27 -04:00
										 |  |  |     api.env.startup_traceback = True | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  |     (options, args) = api.bootstrap_with_global_options(parser, context='lite') | 
					
						
							|  |  |  |     api.env._merge( | 
					
						
							|  |  |  |         lite_port=options.port, | 
					
						
							|  |  |  |         lite_host=options.host, | 
					
						
							|  |  |  |         webui_prod=options.prod, | 
					
						
							|  |  |  |         lite_pem=api.env._join('dot_ipa', 'lite.pem'), | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     api.finalize() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     urlmap = URLMap() | 
					
						
							|  |  |  |     apps = [ | 
					
						
							| 
									
										
										
										
											2012-02-15 10:26:42 -05:00
										 |  |  |         ('IPA', KRBCheater(api.Backend.wsgi_dispatch)), | 
					
						
							| 
									
										
										
										
											2010-08-25 13:06:15 -04:00
										 |  |  |         ('webUI', KRBCheater(WebUIApp())), | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  |     ] | 
					
						
							|  |  |  |     for (name, app) in apps: | 
					
						
							| 
									
										
										
										
											2010-02-23 10:53:47 -07:00
										 |  |  |         urlmap[app.url] = app | 
					
						
							| 
									
										
										
										
											2009-10-13 11:28:00 -06:00
										 |  |  |         api.log.info('Mounting %s at %s', name, app.url) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if path.isfile(api.env.lite_pem): | 
					
						
							|  |  |  |         pem = api.env.lite_pem | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         api.log.info('To enable SSL, place PEM file at %r', api.env.lite_pem) | 
					
						
							|  |  |  |         pem = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     httpserver.serve(paste.gzipper.middleware(urlmap), | 
					
						
							|  |  |  |         host=api.env.lite_host, | 
					
						
							|  |  |  |         port=api.env.lite_port, | 
					
						
							|  |  |  |         ssl_pem=pem, | 
					
						
							|  |  |  |     ) |