''' This file is part of the PyPhantomJS project. Copyright (C) 2011 James Roe 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. 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 along with this program. If not, see . ''' import os import sys from PyQt4.QtCore import pyqtProperty, pyqtSlot, QObject from PyQt4.QtGui import QApplication from PyQt4.QtNetwork import QNetworkProxy, QNetworkProxyFactory from __init__ import __version_info__ from encoding import Encode from filesystem import FileSystem from plugincontroller import do_action from utils import QPyFile from webpage import injectJsInFrame, WebPage from webserver import WebServer class Phantom(QObject): def __init__(self, parent, args): super(Phantom, self).__init__(parent) # variable declarations self.m_defaultPageSettings = {} self.m_pages = [] self.m_servers = [] self.m_verbose = args.verbose self.m_page = WebPage(self, args) self.m_returnValue = 0 self.m_terminated = False # setup the values from args self.app_args = args self.m_scriptFile = args.script self.m_args = args.script_args self.m_scriptEncoding = Encode(args.script_encoding, 'utf-8') self.m_outputEncoding = Encode(args.output_encoding, sys.stdout.encoding_sys) self.m_pages.append(self.m_page) do_action('PhantomInitPre') if args.proxy is None: QNetworkProxyFactory.setUseSystemConfiguration(True) else: proxy = QNetworkProxy(args.proxy_type, args.proxy[0], int(args.proxy[1])) QNetworkProxy.setApplicationProxy(proxy) self.m_page.javaScriptConsoleMessageSent.connect(self.printConsoleMessage) self.m_defaultPageSettings['loadImages'] = args.load_images self.m_defaultPageSettings['loadPlugins'] = args.load_plugins self.m_defaultPageSettings['javascriptEnabled'] = True self.m_defaultPageSettings['XSSAuditingEnabled'] = False self.m_defaultPageSettings['userAgent'] = self.m_page.userAgent() self.m_defaultPageSettings['localToRemoteUrlAccessEnabled'] = args.local_to_remote_url_access self.m_page.applySettings(self.m_defaultPageSettings) self.libraryPath = os.path.dirname(os.path.abspath(self.m_scriptFile)) # inject our properties and slots into javascript self.m_page.mainFrame().addToJavaScriptWindowObject('phantom', self) with QPyFile(':/bootstrap.js') as f: self.m_page.mainFrame().evaluateJavaScript(f.readAll()) do_action('PhantomInitPost') def execute(self): injectJsInFrame(self.m_scriptFile, self.m_scriptEncoding.encoding, os.path.dirname(os.path.abspath(__file__)), self.m_page.mainFrame(), True) return not self.m_terminated def printConsoleMessage(self, message, lineNumber, source): if source: message = '%s:%d %s' % (source, lineNumber, message) print message def returnValue(self): return self.m_returnValue ## # Properties and methods exposed to JavaScript ## @pyqtProperty('QStringList') def args(self): return self.m_args @pyqtSlot(result=FileSystem) def createFilesystem(self): return FileSystem(self) @pyqtSlot(result=WebPage) def createWebPage(self): page = WebPage(self, self.app_args) self.m_pages.append(page) page.applySettings(self.m_defaultPageSettings) page.libraryPath = os.path.dirname(os.path.abspath(self.m_scriptFile)) return page @pyqtSlot(result=WebServer) def createWebServer(self): server = WebServer(self) self.m_servers.append(server) # :TODO: # page.applySettings(self.m_defaultPageSettings) # page.libraryPath = os.path.dirname(os.path.abspath(self.m_scriptFile) return server @pyqtProperty('QVariantMap') def defaultPageSettings(self): return self.m_defaultPageSettings @pyqtSlot() @pyqtSlot(int) def exit(self, code=0): self.m_terminated = True self.m_returnValue = code # stop javascript execution in start script; # release all pages, then clear # the page list, and empty the Phantom page for page in self.m_pages: page.release() del self.m_pages[:] self.m_page = None for server in self.m_servers: server.release() del self.m_servers[:] # not needed, but I'd rather be thorough QApplication.instance().exit(code) @pyqtSlot(str, result=bool) def injectJs(self, filePath): return injectJsInFrame(filePath, self.m_scriptEncoding.encoding, self.libraryPath, self.m_page.mainFrame()) @pyqtSlot(str, result=str) def loadModuleSource(self, name): moduleSourceFilePath = ':/modules/%s.js' % name with QPyFile(moduleSourceFilePath) as f: return f.readAll() @pyqtProperty(str) def libraryPath(self): return self.m_page.libraryPath @libraryPath.setter def libraryPath(self, dirPath): self.m_page.libraryPath = dirPath @pyqtProperty(str) def outputEncoding(self): return self.m_outputEncoding.name @outputEncoding.setter def outputEncoding(self, encoding): self.m_outputEncoding = Encode(encoding, self.m_outputEncoding.encoding) sys.stdout.encoding = self.m_outputEncoding.encoding sys.stdout.encode_to = self.m_outputEncoding.encoding sys.stderr.encoding = self.m_outputEncoding.encoding sys.stdout.encode_to = self.m_outputEncoding.encoding @pyqtProperty(str) def scriptName(self): return os.path.basename(self.m_scriptFile) @pyqtProperty('QVariantMap') def version(self): version = { 'major': __version_info__[0], 'minor': __version_info__[1], 'patch': __version_info__[2] } return version do_action('Phantom')