4008401c9b
It passed "make check" on Linux Change-Id: I3fb2bb8cfe22fd905a3badd2a90a9aef497bfcf7 Reviewed-on: https://gerrit.libreoffice.org/c/core/+/103448 Tested-by: Jenkins Reviewed-by: Julien Nabet <serval2412@yahoo.fr>
486 lines
18 KiB
Python
486 lines
18 KiB
Python
#!/usr/bin/env python # -*- tab-width: 4; indent-tabs-mode: nil; py-indent-offset: 4 -*-
|
|
#
|
|
# Version: MPL 1.1 / GPLv3+ / LGPLv3+
|
|
#
|
|
# The contents of this file are subject to the Mozilla Public License Version
|
|
# 1.1 (the "License"); you may not use this file except in compliance with
|
|
# the License or as specified alternatively below. You may obtain a copy of
|
|
# the License at http://www.mozilla.org/MPL/
|
|
#
|
|
# Software distributed under the License is distributed on an "AS IS" basis,
|
|
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
# for the specific language governing rights and limitations under the
|
|
# License.
|
|
#
|
|
# Major Contributor(s):
|
|
# Copyright (C) 2012 Red Hat, Inc., Michael Stahl <mstahl@redhat.com>
|
|
# (initial developer)
|
|
#
|
|
# All Rights Reserved.
|
|
#
|
|
# For minor contributions see the git repository.
|
|
#
|
|
# Alternatively, the contents of this file may be used under the terms of
|
|
# either the GNU General Public License Version 3 or later (the "GPLv3+"), or
|
|
# the GNU Lesser General Public License Version 3 or later (the "LGPLv3+"),
|
|
# in which case the provisions of the GPLv3+ or the LGPLv3+ are applicable
|
|
# instead of those above.
|
|
|
|
# Simple script to load a bunch of documents and export them as Flat ODF
|
|
#
|
|
# Personally I run it like this:
|
|
# ~/lo/master-suse/instdir/program/python ~/lo/master-suse/bin/benchmark-document-loading --soffice=path:/home/tml/lo/master-suse/instdir/program/soffice --outdir=file://$PWD/out --userdir=file:///tmp/test $PWD/docs
|
|
#
|
|
|
|
import argparse
|
|
import datetime
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
import threading
|
|
import time
|
|
import urllib
|
|
try:
|
|
from urllib.parse import quote
|
|
except ImportError:
|
|
from urllib import quote
|
|
import uuid
|
|
|
|
try:
|
|
import pyuno
|
|
import uno
|
|
import unohelper
|
|
except ImportError:
|
|
print("pyuno not found: try to set PYTHONPATH and URE_BOOTSTRAP variables")
|
|
print("PYTHONPATH=/installation/opt/program")
|
|
print("URE_BOOTSTRAP=file:///installation/opt/program/fundamentalrc")
|
|
raise
|
|
|
|
try:
|
|
from com.sun.star.beans import PropertyValue
|
|
from com.sun.star.document import XDocumentEventListener
|
|
from com.sun.star.io import IOException, XOutputStream
|
|
except ImportError:
|
|
print("UNO API class not found: try to set URE_BOOTSTRAP variable")
|
|
print("URE_BOOTSTRAP=file:///installation/opt/program/fundamentalrc")
|
|
raise
|
|
|
|
validCalcFileExtensions = [ ".xlsx", ".xls", ".ods", ".fods" ]
|
|
validWriterFileExtensions = [ ".docx" , ".rtf", ".odt", ".fodt", ".doc" ]
|
|
validImpressFileExtensions = [ ".ppt", ".pptx", ".odp", ".fodp" ]
|
|
validDrawFileExtensions = [ ".odg", ".fodg" ]
|
|
validReverseFileExtensions = [ ".vsd", ".vdx", ".cdr", ".pub", ".wpd" ]
|
|
validFileExtensions = {"calc": validCalcFileExtensions,
|
|
"writer": validWriterFileExtensions,
|
|
"impress": validImpressFileExtensions,
|
|
"draw": validDrawFileExtensions,
|
|
"reverse": validReverseFileExtensions}
|
|
flatODFTypes = {"calc": (".fods", "OpenDocument Spreadsheet Flat XML"),
|
|
"writer": (".fodt", "OpenDocument Text Flat XML"),
|
|
"impress": (".fodp", "OpenDocument Presentation Flat XML"),
|
|
"draw": (".fodg", "OpenDocument Drawing Flat XML")}
|
|
|
|
outdir = ""
|
|
|
|
def partition(list, pred):
|
|
left = []
|
|
right = []
|
|
for e in list:
|
|
if pred(e):
|
|
left.append(e)
|
|
else:
|
|
right.append(e)
|
|
return (left, right)
|
|
|
|
def filelist(directory, suffix):
|
|
if not directory:
|
|
raise Exception("filelist: empty directory")
|
|
if directory[-1] != "/":
|
|
directory += "/"
|
|
files = [directory + f for f in os.listdir(directory)]
|
|
# print(files)
|
|
return [f for f in files
|
|
if os.path.isfile(f) and os.path.splitext(f)[1] == suffix]
|
|
|
|
def getFiles(dirs, suffix):
|
|
# print( dirs )
|
|
files = []
|
|
for d in dirs:
|
|
files += filelist(d, suffix)
|
|
return files
|
|
|
|
### UNO utilities ###
|
|
|
|
class OutputStream( unohelper.Base, XOutputStream ):
|
|
def __init__( self ):
|
|
self.closed = 0
|
|
|
|
def closeOutput(self):
|
|
self.closed = 1
|
|
|
|
def writeBytes( self, seq ):
|
|
sys.stdout.write( seq.value )
|
|
|
|
def flush( self ):
|
|
pass
|
|
|
|
class OfficeConnection:
|
|
def __init__(self, args):
|
|
self.args = args
|
|
self.soffice = None
|
|
self.socket = None
|
|
self.xContext = None
|
|
self.pro = None
|
|
def setUp(self):
|
|
(method, sep, rest) = self.args.soffice.partition(":")
|
|
if sep != ":":
|
|
raise Exception("soffice parameter does not specify method")
|
|
if method == "path":
|
|
socket = "pipe,name=pytest" + str(uuid.uuid1())
|
|
userdir = self.args.userdir
|
|
if not userdir:
|
|
raise Exception("'path' method requires --userdir")
|
|
if not userdir.startswith("file://"):
|
|
raise Exception("--userdir must be file URL")
|
|
self.soffice = self.bootstrap(rest, userdir, socket)
|
|
elif method == "connect":
|
|
socket = rest
|
|
else:
|
|
raise Exception("unsupported connection method: " + method)
|
|
self.xContext = self.connect(socket)
|
|
|
|
def bootstrap(self, soffice, userdir, socket):
|
|
argv = [ soffice, "--accept=" + socket + ";urp",
|
|
"-env:UserInstallation=" + userdir,
|
|
"--quickstart=no",
|
|
"--norestore", "--nologo", "--headless" ]
|
|
if self.args.valgrind:
|
|
argv.append("--valgrind")
|
|
os.putenv("SAL_LOG", "-INFO-WARN")
|
|
os.putenv("LIBO_ONEWAY_STABLE_ODF_EXPORT", "YES")
|
|
self.pro = subprocess.Popen(argv)
|
|
# print(self.pro.pid)
|
|
|
|
def connect(self, socket):
|
|
xLocalContext = uno.getComponentContext()
|
|
xUnoResolver = xLocalContext.ServiceManager.createInstanceWithContext("com.sun.star.bridge.UnoUrlResolver", xLocalContext)
|
|
url = "uno:" + socket + ";urp;StarOffice.ComponentContext"
|
|
# print("OfficeConnection: connecting to: " + url)
|
|
while True:
|
|
try:
|
|
xContext = xUnoResolver.resolve(url)
|
|
return xContext
|
|
# except com.sun.star.connection.NoConnectException
|
|
except pyuno.getClass("com.sun.star.connection.NoConnectException"):
|
|
# print("NoConnectException: sleeping...")
|
|
time.sleep(1)
|
|
|
|
def tearDown(self):
|
|
if self.soffice:
|
|
if self.xContext:
|
|
try:
|
|
# print("tearDown: calling terminate()...")
|
|
xMgr = self.xContext.ServiceManager
|
|
xDesktop = xMgr.createInstanceWithContext("com.sun.star.frame.Desktop", self.xContext)
|
|
xDesktop.terminate()
|
|
# print("...done")
|
|
# except com.sun.star.lang.DisposedException:
|
|
except pyuno.getClass("com.sun.star.beans.UnknownPropertyException"):
|
|
# print("caught UnknownPropertyException while TearDown")
|
|
pass # ignore, also means disposed
|
|
except pyuno.getClass("com.sun.star.lang.DisposedException"):
|
|
# print("caught DisposedException while TearDown")
|
|
pass # ignore
|
|
else:
|
|
self.soffice.terminate()
|
|
ret = self.soffice.wait()
|
|
self.xContext = None
|
|
self.socket = None
|
|
self.soffice = None
|
|
if ret != 0:
|
|
raise Exception("Exit status indicates failure: " + str(ret))
|
|
# return ret
|
|
def kill(self):
|
|
command = "kill " + str(self.pro.pid)
|
|
with open("killFile.log", "a") as killFile:
|
|
killFile.write(command + "\n")
|
|
# print("kill")
|
|
# print(command)
|
|
os.system(command)
|
|
|
|
class PersistentConnection:
|
|
def __init__(self, args):
|
|
self.args = args
|
|
self.connection = None
|
|
def getContext(self):
|
|
return self.connection.xContext
|
|
def setUp(self):
|
|
assert(not self.connection)
|
|
conn = OfficeConnection(self.args)
|
|
conn.setUp()
|
|
self.connection = conn
|
|
def preTest(self):
|
|
assert(self.connection)
|
|
def postTest(self):
|
|
assert(self.connection)
|
|
def tearDown(self):
|
|
if self.connection:
|
|
try:
|
|
self.connection.tearDown()
|
|
finally:
|
|
self.connection = None
|
|
def kill(self):
|
|
if self.connection:
|
|
self.connection.kill()
|
|
|
|
def simpleInvoke(connection, test):
|
|
try:
|
|
connection.preTest()
|
|
test.run(connection.getContext(), connection)
|
|
finally:
|
|
connection.postTest()
|
|
|
|
def runConnectionTests(connection, invoker, tests):
|
|
try:
|
|
connection.setUp()
|
|
for test in tests:
|
|
invoker(connection, test)
|
|
finally:
|
|
pass
|
|
#connection.tearDown()
|
|
|
|
class EventListener(XDocumentEventListener,unohelper.Base):
|
|
def __init__(self):
|
|
self.layoutFinished = False
|
|
def documentEventOccured(self, event):
|
|
# print(str(event.EventName))
|
|
if event.EventName == "OnLayoutFinished":
|
|
self.layoutFinished = True
|
|
def disposing(event):
|
|
pass
|
|
|
|
def mkPropertyValue(name, value):
|
|
return uno.createUnoStruct("com.sun.star.beans.PropertyValue",
|
|
name, 0, value, 0)
|
|
|
|
### tests ###
|
|
|
|
def logTimeSpent(url, startTime):
|
|
print(os.path.basename(urllib.parse.urlparse(url).path) + "\t" + str(time.time()-startTime))
|
|
|
|
def loadFromURL(xContext, url, t, component):
|
|
xDesktop = xContext.ServiceManager.createInstanceWithContext("com.sun.star.frame.Desktop", xContext)
|
|
props = [("Hidden", True), ("ReadOnly", True)] # FilterName?
|
|
loadProps = tuple([mkPropertyValue(name, value) for (name, value) in props])
|
|
xListener = None
|
|
if component == "writer":
|
|
xListener = EventListener()
|
|
xGEB = xContext.getValueByName(
|
|
"/singletons/com.sun.star.frame.theGlobalEventBroadcaster")
|
|
xGEB.addDocumentEventListener(xListener)
|
|
try:
|
|
xDoc = None
|
|
startTime = time.time()
|
|
xDoc = xDesktop.loadComponentFromURL(url, "_blank", 0, loadProps)
|
|
if component == "calc":
|
|
try:
|
|
if xDoc:
|
|
xDoc.calculateAll()
|
|
except AttributeError:
|
|
pass
|
|
t.cancel()
|
|
logTimeSpent(url, startTime)
|
|
return xDoc
|
|
elif component == "writer":
|
|
time_ = 0
|
|
t.cancel()
|
|
while time_ < 30:
|
|
if xListener.layoutFinished:
|
|
logTimeSpent(url, startTime)
|
|
return xDoc
|
|
# print("delaying...")
|
|
time_ += 1
|
|
time.sleep(1)
|
|
else:
|
|
t.cancel()
|
|
logTimeSpent(url, startTime)
|
|
return xDoc
|
|
with open("file.log", "a") as fh:
|
|
fh.write("layout did not finish\n")
|
|
return xDoc
|
|
except pyuno.getClass("com.sun.star.beans.UnknownPropertyException"):
|
|
xListener = None
|
|
raise # means crashed, handle it later
|
|
except pyuno.getClass("com.sun.star.lang.DisposedException"):
|
|
xListener = None
|
|
raise # means crashed, handle it later
|
|
except pyuno.getClass("com.sun.star.lang.IllegalArgumentException"):
|
|
pass # means could not open the file, ignore it
|
|
except:
|
|
if xDoc:
|
|
# print("CLOSING")
|
|
xDoc.close(True)
|
|
raise
|
|
finally:
|
|
if xListener:
|
|
xGEB.removeDocumentEventListener(xListener)
|
|
|
|
def exportToODF(xContext, xDoc, baseName, t, component):
|
|
exportFileName = outdir + "/" + os.path.splitext(baseName)[0] + flatODFTypes[component][0]
|
|
print("exportToODF " + baseName + " => " + exportFileName)
|
|
props = [("FilterName", flatODFTypes[component][1]),
|
|
("Overwrite", True)]
|
|
storeProps = tuple([mkPropertyValue(name, value) for (name, value) in props])
|
|
xDoc.storeToURL(exportFileName, tuple(storeProps))
|
|
|
|
def handleCrash(file, disposed):
|
|
# print("File: " + file + " crashed")
|
|
with open("crashlog.txt", "a") as crashLog:
|
|
crashLog.write('Crash:' + file + ' ')
|
|
if disposed == 1:
|
|
crashLog.write('through disposed\n')
|
|
# crashed_files.append(file)
|
|
# add here the remaining handling code for crashed files
|
|
|
|
def alarm_handler(args):
|
|
args.kill()
|
|
|
|
class HandleFileTest:
|
|
def __init__(self, file, state, component):
|
|
self.file = file
|
|
self.state = state
|
|
self.component = component
|
|
def run(self, xContext, connection):
|
|
# print("Loading document: " + self.file)
|
|
t = None
|
|
args = None
|
|
try:
|
|
url = "file://" + quote(self.file)
|
|
with open("file.log", "a") as fh:
|
|
fh.write(url + "\n")
|
|
xDoc = None
|
|
args = [connection]
|
|
t = threading.Timer(60, alarm_handler, args)
|
|
t.start()
|
|
xDoc = loadFromURL(xContext, url, t, self.component)
|
|
self.state.goodFiles.append(self.file)
|
|
exportToODF(xContext, xDoc, os.path.basename(urllib.parse.urlparse(url).path), t, self.component)
|
|
except pyuno.getClass("com.sun.star.beans.UnknownPropertyException"):
|
|
# print("caught UnknownPropertyException " + self.file)
|
|
if not t.is_alive():
|
|
# print("TIMEOUT!")
|
|
self.state.timeoutFiles.append(self.file)
|
|
else:
|
|
t.cancel()
|
|
handleCrash(self.file, 0)
|
|
self.state.badPropertyFiles.append(self.file)
|
|
connection.tearDown()
|
|
connection.setUp()
|
|
except pyuno.getClass("com.sun.star.lang.DisposedException"):
|
|
# print("caught DisposedException " + self.file)
|
|
if not t.is_alive():
|
|
# print("TIMEOUT!")
|
|
self.state.timeoutFiles.append(self.file)
|
|
else:
|
|
t.cancel()
|
|
handleCrash(self.file, 1)
|
|
self.state.badDisposedFiles.append(self.file)
|
|
connection.tearDown()
|
|
connection.setUp()
|
|
finally:
|
|
if t.is_alive():
|
|
t.cancel()
|
|
try:
|
|
if xDoc:
|
|
t = threading.Timer(10, alarm_handler, args)
|
|
t.start()
|
|
xDoc.close(True)
|
|
t.cancel()
|
|
except pyuno.getClass("com.sun.star.beans.UnknownPropertyException"):
|
|
print("caught UnknownPropertyException while closing")
|
|
self.state.badPropertyFiles.append(self.file)
|
|
connection.tearDown()
|
|
connection.setUp()
|
|
except pyuno.getClass("com.sun.star.lang.DisposedException"):
|
|
print("caught DisposedException while closing")
|
|
if t.is_alive():
|
|
t.cancel()
|
|
else:
|
|
self.state.badDisposedFiles.append(self.file)
|
|
connection.tearDown()
|
|
connection.setUp()
|
|
# print("...done with: " + self.file)
|
|
|
|
class State:
|
|
def __init__(self):
|
|
self.goodFiles = []
|
|
self.badDisposedFiles = []
|
|
self.badPropertyFiles = []
|
|
self.timeoutFiles = []
|
|
|
|
|
|
def write_state_report(files_list, start_time, report_filename, description):
|
|
with open(report_filename, "w") as fh:
|
|
fh.write("%s:\n" % description)
|
|
fh.write("Starttime: %s\n" % start_time.isoformat())
|
|
for f in files_list:
|
|
fh.write("%s\n" % f)
|
|
|
|
|
|
def writeReport(state, startTime):
|
|
write_state_report(state.goodFiles, startTime, "goodFiles.log",
|
|
"Files which loaded perfectly")
|
|
write_state_report(state.badDisposedFiles, startTime, "badDisposedFiles.log",
|
|
"Files which crashed with DisposedException")
|
|
write_state_report(state.badPropertyFiles, startTime, "badPropertyFiles.log",
|
|
"Files which crashed with UnknownPropertyException")
|
|
write_state_report(state.timeoutFiles, startTime, "timeoutFiles.log",
|
|
"Files which timed out")
|
|
|
|
def runHandleFileTests(opts):
|
|
startTime = datetime.datetime.now()
|
|
connection = PersistentConnection(opts)
|
|
global outdir
|
|
outdir = os.path.join(opts.outdir, startTime.strftime('%Y%m%d.%H%M%S'))
|
|
try:
|
|
tests = []
|
|
state = State()
|
|
# print("before map")
|
|
for component, validExtension in validFileExtensions.items():
|
|
files = []
|
|
for suffix in validExtension:
|
|
files.extend(getFiles(opts.dirs, suffix))
|
|
files.sort()
|
|
tests.extend( (HandleFileTest(file, state, component) for file in files) )
|
|
runConnectionTests(connection, simpleInvoke, tests)
|
|
finally:
|
|
connection.kill()
|
|
writeReport(state, startTime)
|
|
|
|
def parseArgs(argv):
|
|
epilog = "'location' is a pathname, not a URL. 'outdir' and 'userdir' are URLs.\n" \
|
|
"The 'directory' parameters should be full absolute pathnames, not URLs."
|
|
|
|
parser = argparse.ArgumentParser(formatter_class=argparse.RawTextHelpFormatter,
|
|
epilog=epilog)
|
|
parser.add_argument('--soffice', metavar='method:location', required=True,
|
|
help="specify soffice instance to connect to\n"
|
|
"supported methods: 'path', 'connect'")
|
|
parser.add_argument('--outdir', metavar='URL', required=True,
|
|
help="specify the output directory for flat ODF exports")
|
|
parser.add_argument('--userdir', metavar='URL',
|
|
help="specify user installation directory for 'path' method")
|
|
parser.add_argument('--valgrind', action='store_true',
|
|
help="pass --valgrind to soffice for 'path' method")
|
|
parser.add_argument('dirs', metavar='directory', nargs='+')
|
|
|
|
args = parser.parse_args(argv[1:])
|
|
|
|
return args
|
|
|
|
|
|
if __name__ == "__main__":
|
|
opts = parseArgs(sys.argv)
|
|
runHandleFileTests(opts)
|
|
|
|
# vim:set shiftwidth=4 softtabstop=4 expandtab:
|