#!/usr/bin/env python # # Copyright 2010 Google Inc. All Rights Reserved. """Playback driver.""" import cgi import simplejson as json import os import string import sys import threading import urlparse START_PAGE = """ """ START_PAGE_POPUP = """ """ DATA_JS = 'Benchmark.data = $data;' def ReadFile(file_name, mode='r'): f = open(file_name, mode) data = f.read() f.close() return data def ReadJSON(file_name): f = open(file_name, 'r') data = json.load(f) f.close() return data class PlaybackRequestHandler(object): """This class is used to process HTTP requests during test playback. Attributes: test_dir: directory containing test files. test_callback: function to be called when the test is finished. script_dir: directory where javascript files are located. """ def __init__(self, test_dir, test_callback=None, script_dir=os.getcwd()): self.test_dir = test_dir self.test_callback = test_callback self.script_dir = script_dir def ProcessRequest(self, handler): "Processes single HTTP request." parse_result = urlparse.urlparse(handler.path) if parse_result.path.endswith('/benchmark/'): query = cgi.parse_qs(parse_result.query) if 'run_test' in query: run_count = 1 if 'run_count' in query: run_count = query['run_count'][0] self._StartTest(handler, self.test_dir, run_count) elif 'resource' in query: self._GetBenchmarkResource(query['resource'][0], handler) else: self._ProcessBenchmarkReport(handler.body, handler) else: self._GetApplicationResource(handler) def _StartTest(self, handler, test_dir, run_count): "Sends test start page to browser." cache_data = ReadJSON(os.path.join(test_dir, 'cache.json')) # Load cached responses. self.cache = {} responses_dir = os.path.join(test_dir, 'responses') for request in cache_data['requests']: response_file = os.path.join(responses_dir, request['response_file']) response = ReadFile(response_file, 'rb') key = (request['method'], request['path']) self.cache[key] = {'response': response, 'headers': request['headers']} # Load benchmark scripts. self.benchmark_resources = {} data = ReadFile(os.path.join(test_dir, 'data.json')) data = string.Template(DATA_JS).substitute(data=data) self.benchmark_resources['data.js'] = {'data': data, 'type': 'application/javascript'} for resource in ('common.js', 'playback.js'): resource_file = os.path.join(self.script_dir, resource) self.benchmark_resources[resource] = {'data': ReadFile(resource_file), 'type': 'application/javascript'} # Format start page. parse_result = urlparse.urlparse(cache_data['start_url']) target_origin = '%s://%s' % (parse_result.scheme, parse_result.netloc) start_page = string.Template(START_PAGE).substitute( run_count=run_count, target_origin=target_origin, width=cache_data['width'], height=cache_data['height']) self.benchmark_resources['start_page'] = { 'data': start_page, 'type': 'text/html; charset=UTF-8' } start_page_popup = string.Template(START_PAGE_POPUP).substitute( start_url=cache_data['start_url'], width=cache_data['width'], height=cache_data['height']) self.benchmark_resources['start_page_popup'] = { 'data': start_page_popup, 'type': 'text/html; charset=UTF-8' } self._GetBenchmarkResource('start_page', handler) def _GetBenchmarkResource(self, resource, handler): "Sends requested resource to browser." if resource in self.benchmark_resources: resource = self.benchmark_resources[resource] handler.send_response(200) handler.send_header('content-length', len(resource['data'])) handler.send_header('content-type', resource['type']) handler.end_headers() handler.wfile.write(resource['data']) else: handler.send_response(404) handler.end_headers() def _ProcessBenchmarkReport(self, content, handler): "Reads benchmark score from report content and invokes callback." handler.send_response(204) handler.end_headers() content = json.loads(content) if 'results' in content: results = content['results'] sys.stdout.write('Results: %s\n' % results) if self.test_callback: self.test_callback(results) elif 'error' in content: sys.stderr.write('Error: %s\n' % content['error']) def _GetApplicationResource(self, handler): "Searches for response in cache. If not found, responds with 204." key = (handler.command, handler.path) if key in self.cache: sys.stdout.write('%s %s -> found\n' % key) handler.wfile.write(self.cache[key]['response']) else: sys.stderr.write('%s %s -> not found\n' % key) handler.send_response(204, "not in cache") handler.end_headers()