openlp/openlp/core/utils/__init__.py

516 lines
19 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
###############################################################################
# OpenLP - Open Source Lyrics Projection #
# --------------------------------------------------------------------------- #
2010-12-26 11:04:47 +00:00
# Copyright (c) 2008-2011 Raoul Snyman #
# Portions copyright (c) 2008-2011 Tim Bentley, Jonathan Corwin, Michael #
2011-03-24 19:04:02 +00:00
# Gorven, Scott Guerrieri, Matthias Hub, Meinert Jordan, Armin Köhler, #
# Andreas Preikschat, Mattias Põldaru, Christian Richter, Philip Ridout, #
# Maikel Stuivenberg, Martin Thompson, Jon Tibble, Frode Woldsund #
# --------------------------------------------------------------------------- #
# 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; version 2 of the License. #
# #
# 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, write to the Free Software Foundation, Inc., 59 #
# Temple Place, Suite 330, Boston, MA 02111-1307 USA #
###############################################################################
2010-06-10 19:45:02 +00:00
"""
2010-12-10 05:09:03 +00:00
The :mod:`openlp.core.utils` module provides the utility libraries for OpenLP.
2010-06-10 19:45:02 +00:00
"""
2010-07-30 22:48:09 +00:00
import logging
import os
2010-07-30 22:48:09 +00:00
import re
import sys
2010-07-30 22:48:09 +00:00
import time
2009-10-12 04:43:02 +00:00
import urllib2
2010-06-10 19:45:02 +00:00
from datetime import datetime
2011-03-24 19:12:27 +00:00
from subprocess import Popen, PIPE
from PyQt4 import QtGui, QtCore
2011-01-18 04:32:24 +00:00
if sys.platform != u'win32' and sys.platform != u'darwin':
try:
from xdg import BaseDirectory
XDG_BASE_AVAILABLE = True
except ImportError:
XDG_BASE_AVAILABLE = False
import openlp
2011-03-24 16:43:08 +00:00
from openlp.core.lib import Receiver, translate, check_directory_exists
2009-10-14 17:52:50 +00:00
log = logging.getLogger(__name__)
2011-03-25 16:29:39 +00:00
APPLICATION_VERSION = {}
2011-01-18 04:32:24 +00:00
IMAGES_FILTER = None
2011-01-21 22:28:34 +00:00
UNO_CONNECTION_TYPE = u'pipe'
#UNO_CONNECTION_TYPE = u'socket'
2009-10-14 17:52:50 +00:00
2010-07-30 22:48:09 +00:00
class VersionThread(QtCore.QThread):
"""
A special Qt thread class to fetch the version of OpenLP from the website.
This is threaded so that it doesn't affect the loading time of OpenLP.
"""
2011-03-25 16:29:39 +00:00
def __init__(self, parent):
2010-07-30 22:48:09 +00:00
QtCore.QThread.__init__(self, parent)
self.version_splitter = re.compile(
r'([0-9]+).([0-9]+).([0-9]+)(?:-bzr([0-9]+))?')
def run(self):
"""
Run the thread.
"""
time.sleep(1)
Receiver.send_message(u'maindisplay_blank_check')
2011-03-25 16:29:39 +00:00
app_version = get_application_version()
version = check_latest_version(app_version)
2010-07-30 22:48:09 +00:00
remote_version = {}
local_version = {}
match = self.version_splitter.match(version)
if match:
remote_version[u'major'] = int(match.group(1))
remote_version[u'minor'] = int(match.group(2))
remote_version[u'release'] = int(match.group(3))
if len(match.groups()) > 3 and match.group(4):
remote_version[u'revision'] = int(match.group(4))
else:
return
2011-03-25 16:29:39 +00:00
match = self.version_splitter.match(app_version[u'full'])
2010-07-30 22:48:09 +00:00
if match:
local_version[u'major'] = int(match.group(1))
local_version[u'minor'] = int(match.group(2))
local_version[u'release'] = int(match.group(3))
if len(match.groups()) > 3 and match.group(4):
local_version[u'revision'] = int(match.group(4))
else:
return
2010-07-30 22:48:09 +00:00
if remote_version[u'major'] > local_version[u'major'] or \
remote_version[u'minor'] > local_version[u'minor'] or \
remote_version[u'release'] > local_version[u'release']:
Receiver.send_message(u'openlp_version_check', u'%s' % version)
elif remote_version.get(u'revision') and \
local_version.get(u'revision') and \
remote_version[u'revision'] > local_version[u'revision']:
Receiver.send_message(u'openlp_version_check', u'%s' % version)
2011-04-30 07:42:20 +00:00
class DelayStartThread(QtCore.QThread):
"""
A special Qt thread class to build things after OpenLP has started
"""
def __init__(self, parent):
QtCore.QThread.__init__(self, parent)
def run(self):
"""
Run the thread.
"""
2011-04-29 16:27:27 +00:00
Receiver.send_message(u'openlp_phonon_creation')
2010-07-30 22:48:09 +00:00
class AppLocation(object):
"""
The :class:`AppLocation` class is a static class which retrieves a
directory based on the directory type.
"""
AppDir = 1
ConfigDir = 2
DataDir = 3
PluginsDir = 4
VersionDir = 5
2010-06-22 12:32:15 +00:00
CacheDir = 6
LanguageDir = 7
@staticmethod
def get_directory(dir_type=1):
"""
Return the appropriate directory according to the directory type.
``dir_type``
The directory type you want, for instance the data directory.
"""
if dir_type == AppLocation.AppDir:
2011-01-18 04:32:24 +00:00
return _get_frozen_path(
os.path.abspath(os.path.split(sys.argv[0])[0]),
os.path.split(openlp.__file__)[0])
elif dir_type == AppLocation.PluginsDir:
app_path = os.path.abspath(os.path.split(sys.argv[0])[0])
2011-01-18 04:32:24 +00:00
return _get_frozen_path(os.path.join(app_path, u'plugins'),
os.path.join(os.path.split(openlp.__file__)[0], u'plugins'))
elif dir_type == AppLocation.VersionDir:
2011-01-18 04:32:24 +00:00
return _get_frozen_path(
os.path.abspath(os.path.split(sys.argv[0])[0]),
os.path.split(openlp.__file__)[0])
2011-01-22 11:33:56 +00:00
elif dir_type == AppLocation.LanguageDir:
2011-01-18 04:32:24 +00:00
app_path = _get_frozen_path(
os.path.abspath(os.path.split(sys.argv[0])[0]),
_get_os_dir_path(dir_type))
return os.path.join(app_path, u'i18n')
2011-01-22 11:33:56 +00:00
else:
2011-01-23 00:27:29 +00:00
return _get_os_dir_path(dir_type)
2010-04-27 16:27:57 +00:00
@staticmethod
def get_data_path():
2011-01-18 04:32:24 +00:00
"""
Return the path OpenLP stores all its data under.
"""
2010-04-27 16:27:57 +00:00
path = AppLocation.get_directory(AppLocation.DataDir)
2011-03-24 16:43:08 +00:00
check_directory_exists(path)
2010-04-27 16:27:57 +00:00
return path
@staticmethod
def get_section_data_path(section):
2011-01-18 04:32:24 +00:00
"""
Return the path a particular module stores its data under.
"""
2010-04-27 16:27:57 +00:00
data_path = AppLocation.get_data_path()
path = os.path.join(data_path, section)
2011-03-24 16:43:08 +00:00
check_directory_exists(path)
2010-04-27 16:27:57 +00:00
return path
2011-01-23 00:27:29 +00:00
def _get_os_dir_path(dir_type):
2011-01-18 04:32:24 +00:00
"""
Return a path based on which OS and environment we are running in.
"""
encoding = sys.getfilesystemencoding()
2011-01-18 04:32:24 +00:00
if sys.platform == u'win32':
2011-02-11 17:50:34 +00:00
if dir_type == AppLocation.DataDir:
return os.path.join(unicode(os.getenv(u'APPDATA'), encoding),
u'openlp', u'data')
elif dir_type == AppLocation.LanguageDir:
return os.path.split(openlp.__file__)[0]
return os.path.join(unicode(os.getenv(u'APPDATA'), encoding),
u'openlp')
2011-01-18 04:32:24 +00:00
elif sys.platform == u'darwin':
2011-01-22 11:33:56 +00:00
if dir_type == AppLocation.DataDir:
return os.path.join(unicode(os.getenv(u'HOME'), encoding),
u'Library', u'Application Support', u'openlp', u'Data')
elif dir_type == AppLocation.LanguageDir:
return os.path.split(openlp.__file__)[0]
return os.path.join(unicode(os.getenv(u'HOME'), encoding),
u'Library', u'Application Support', u'openlp')
2011-01-18 04:32:24 +00:00
else:
if dir_type == AppLocation.LanguageDir:
return os.path.join(u'/usr', u'share', u'openlp')
2011-01-18 04:32:24 +00:00
if XDG_BASE_AVAILABLE:
2011-01-22 11:33:56 +00:00
if dir_type == AppLocation.ConfigDir:
return os.path.join(unicode(BaseDirectory.xdg_config_home,
encoding), u'openlp')
2011-01-22 11:33:56 +00:00
elif dir_type == AppLocation.DataDir:
return os.path.join(
unicode(BaseDirectory.xdg_data_home, encoding), u'openlp')
2011-01-22 11:33:56 +00:00
elif dir_type == AppLocation.CacheDir:
return os.path.join(unicode(BaseDirectory.xdg_cache_home,
encoding), u'openlp')
2011-02-11 17:55:41 +00:00
if dir_type == AppLocation.DataDir:
return os.path.join(unicode(os.getenv(u'HOME'), encoding),
u'.openlp', u'data')
return os.path.join(unicode(os.getenv(u'HOME'), encoding), u'.openlp')
2011-01-18 04:32:24 +00:00
def _get_frozen_path(frozen_option, non_frozen_option):
"""
Return a path based on the system status.
"""
if hasattr(sys, u'frozen') and sys.frozen == 1:
return frozen_option
2011-02-11 17:50:34 +00:00
return non_frozen_option
2010-04-27 16:27:57 +00:00
2011-03-25 16:29:39 +00:00
def get_application_version():
2011-03-24 19:12:27 +00:00
"""
Returns the application version of the running instance of OpenLP::
{u'full': u'1.9.4-bzr1249', u'version': u'1.9.4', u'build': u'bzr1249'}
"""
global APPLICATION_VERSION
if APPLICATION_VERSION:
return APPLICATION_VERSION
2011-03-25 16:29:39 +00:00
if u'--dev-version' in sys.argv or u'-d' in sys.argv:
2011-03-24 19:12:27 +00:00
# If we're running the dev version, let's use bzr to get the version.
try:
# If bzrlib is available, use it.
from bzrlib.branch import Branch
b = Branch.open_containing('.')[0]
b.lock_read()
try:
# Get the branch's latest revision number.
revno = b.revno()
# Convert said revision number into a bzr revision id.
revision_id = b.dotted_revno_to_revision_id((revno,))
# Get a dict of tags, with the revision id as the key.
tags = b.tags.get_reverse_tag_dict()
# Check if the latest
if revision_id in tags:
full_version = u'%s' % tags[revision_id][0]
else:
full_version = '%s-bzr%s' % \
(sorted(b.tags.get_tag_dict().keys())[-1], revno)
finally:
b.unlock()
except:
# Otherwise run the command line bzr client.
bzr = Popen((u'bzr', u'tags', u'--sort', u'time'), stdout=PIPE)
output, error = bzr.communicate()
code = bzr.wait()
if code != 0:
raise Exception(u'Error running bzr tags')
lines = output.splitlines()
if len(lines) == 0:
tag = u'0.0.0'
revision = u'0'
else:
tag, revision = lines[-1].split()
bzr = Popen((u'bzr', u'log', u'--line', u'-r', u'-1'), stdout=PIPE)
output, error = bzr.communicate()
code = bzr.wait()
if code != 0:
raise Exception(u'Error running bzr log')
latest = output.split(u':')[0]
full_version = latest == revision and tag or \
u'%s-bzr%s' % (tag, latest)
else:
# We're not running the development version, let's use the file.
filepath = AppLocation.get_directory(AppLocation.VersionDir)
filepath = os.path.join(filepath, u'.version')
fversion = None
try:
fversion = open(filepath, u'r')
full_version = unicode(fversion.read()).rstrip()
except IOError:
log.exception('Error in version file.')
full_version = u'0.0.0-bzr000'
finally:
if fversion:
fversion.close()
bits = full_version.split(u'-')
APPLICATION_VERSION = {
u'full': full_version,
u'version': bits[0],
u'build': bits[1] if len(bits) > 1 else None
}
if APPLICATION_VERSION[u'build']:
log.info(u'Openlp version %s build %s',
APPLICATION_VERSION[u'version'], APPLICATION_VERSION[u'build'])
else:
log.info(u'Openlp version %s' % APPLICATION_VERSION[u'version'])
return APPLICATION_VERSION
2010-04-27 16:27:57 +00:00
def check_latest_version(current_version):
"""
Check the latest version of OpenLP against the version file on the OpenLP
site.
``current_version``
The current version of OpenLP.
"""
version_string = current_version[u'full']
# set to prod in the distribution config file.
2010-04-28 14:17:42 +00:00
settings = QtCore.QSettings()
settings.beginGroup(u'general')
last_test = unicode(settings.value(u'last version test',
QtCore.QVariant(datetime.now().date())).toString())
2009-11-30 18:29:22 +00:00
this_test = unicode(datetime.now().date())
2010-04-28 14:17:42 +00:00
settings.setValue(u'last version test', QtCore.QVariant(this_test))
settings.endGroup()
2009-11-30 18:29:22 +00:00
if last_test != this_test:
if current_version[u'build']:
2010-04-23 16:00:32 +00:00
req = urllib2.Request(
u'http://www.openlp.org/files/dev_version.txt')
else:
req = urllib2.Request(u'http://www.openlp.org/files/version.txt')
req.add_header(u'User-Agent', u'OpenLP/%s' % current_version[u'full'])
2010-08-26 00:37:25 +00:00
remote_version = None
2009-10-12 04:43:02 +00:00
try:
2010-08-26 00:37:25 +00:00
remote_version = unicode(urllib2.urlopen(req, None).read()).strip()
2011-01-18 04:32:24 +00:00
except IOError:
log.exception(u'Failed to download the latest OpenLP version file')
2010-08-26 00:37:25 +00:00
if remote_version:
version_string = remote_version
2009-10-12 04:43:02 +00:00
return version_string
2010-04-23 16:00:32 +00:00
def add_actions(target, actions):
"""
Adds multiple actions to a menu or toolbar in one command.
``target``
The menu or toolbar to add actions to.
``actions``
2011-02-25 17:05:01 +00:00
The actions to be added. An action consisting of the keyword 'None'
2010-04-23 19:42:51 +00:00
will result in a separator being inserted into the target.
2010-04-23 16:00:32 +00:00
"""
for action in actions:
if action is None:
target.addSeparator()
else:
target.addAction(action)
def get_filesystem_encoding():
"""
Returns the name of the encoding used to convert Unicode filenames into
system file names.
"""
encoding = sys.getfilesystemencoding()
if encoding is None:
encoding = sys.getdefaultencoding()
return encoding
def get_images_filter():
"""
Returns a filter string for a file dialog containing all the supported
image formats.
"""
2011-01-18 04:32:24 +00:00
global IMAGES_FILTER
if not IMAGES_FILTER:
log.debug(u'Generating images filter.')
2010-06-19 18:42:49 +00:00
formats = [unicode(fmt)
for fmt in QtGui.QImageReader.supportedImageFormats()]
2010-06-19 18:42:49 +00:00
visible_formats = u'(*.%s)' % u'; *.'.join(formats)
actual_formats = u'(*.%s)' % u' *.'.join(formats)
2011-01-18 04:32:24 +00:00
IMAGES_FILTER = u'%s %s %s' % (translate('OpenLP', 'Image Files'),
visible_formats, actual_formats)
2011-01-18 04:32:24 +00:00
return IMAGES_FILTER
def split_filename(path):
2011-01-18 04:32:24 +00:00
"""
Return a list of the parts in a given path.
"""
path = os.path.abspath(path)
if not os.path.isfile(path):
return path, u''
else:
return os.path.split(path)
2011-01-14 18:58:47 +00:00
def delete_file(file_path_name):
"""
Deletes a file from the system.
``file_path_name``
The file, including path, to delete.
"""
if not file_path_name:
return False
try:
if os.path.exists(file_path_name):
os.remove(file_path_name)
return True
except (IOError, OSError):
log.exception("Unable to delete file %s" % file_path_name)
return False
def get_web_page(url, header=None, update_openlp=False):
2011-01-10 01:46:47 +00:00
"""
Attempts to download the webpage at url and returns that page or None.
``url``
The URL to be downloaded.
``header``
An optional HTTP header to pass in the request to the web server.
2011-01-10 01:46:47 +00:00
``update_openlp``
Tells OpenLP to update itself if the page is successfully downloaded.
Defaults to False.
"""
2011-02-25 17:05:01 +00:00
# TODO: Add proxy usage. Get proxy info from OpenLP settings, add to a
2011-01-10 01:46:47 +00:00
# proxy_handler, build into an opener and install the opener into urllib2.
# http://docs.python.org/library/urllib2.html
if not url:
return None
req = urllib2.Request(url)
if header:
req.add_header(header[0], header[1])
2011-01-10 01:46:47 +00:00
page = None
log.debug(u'Downloading URL = %s' % url)
try:
page = urllib2.urlopen(req)
2011-01-10 01:46:47 +00:00
log.debug(u'Downloaded URL = %s' % page.geturl())
except urllib2.URLError:
log.exception(u'The web page could not be downloaded')
if not page:
return None
if update_openlp:
Receiver.send_message(u'openlp_process_events')
2011-03-10 13:15:49 +00:00
log.debug(page)
2011-01-10 01:46:47 +00:00
return page
def file_is_unicode(filename):
"""
Checks if a file is valid unicode and returns the unicode decoded file or
None.
``filename``
File to check is valid unicode.
"""
if not filename:
return None
ucsfile = None
try:
ucsfile = filename.decode(u'utf-8')
except UnicodeDecodeError:
log.exception(u'Filename "%s" is not valid UTF-8' %
filename.decode(u'utf-8', u'replace'))
if not ucsfile:
return None
return ucsfile
2011-01-13 17:55:29 +00:00
def string_is_unicode(test_string):
"""
Makes sure a string is unicode.
``test_string``
The string to confirm is unicode.
"""
return_string = u''
if not test_string:
return return_string
if isinstance(test_string, unicode):
return_string = test_string
if not isinstance(test_string, unicode):
try:
return_string = unicode(test_string, u'utf-8')
except UnicodeError:
log.exception("Error encoding string to unicode")
return return_string
2011-01-21 22:28:34 +00:00
def get_uno_command():
"""
Returns the UNO command to launch an openoffice.org instance.
"""
2011-02-14 20:06:32 +00:00
COMMAND = u'soffice'
2011-03-12 19:00:23 +00:00
OPTIONS = u'-nologo -norestore -minimized -nodefault -nofirststartwizard'
2011-01-21 22:28:34 +00:00
if UNO_CONNECTION_TYPE == u'pipe':
CONNECTION = u'"-accept=pipe,name=openlp_pipe;urp;"'
2011-01-21 22:28:34 +00:00
else:
CONNECTION = u'"-accept=socket,host=localhost,port=2002;urp;"'
2011-02-14 20:06:32 +00:00
return u'%s %s %s' % (COMMAND, OPTIONS, CONNECTION)
2011-01-21 22:28:34 +00:00
def get_uno_instance(resolver):
"""
Returns a running openoffice.org instance.
``resolver``
The UNO resolver to use to find a running instance.
"""
log.debug(u'get UNO Desktop Openoffice - resolve')
if UNO_CONNECTION_TYPE == u'pipe':
return resolver.resolve(u'uno:pipe,name=openlp_pipe;' \
+ u'urp;StarOffice.ComponentContext')
else:
return resolver.resolve(u'uno:socket,host=localhost,port=2002;' \
+ u'urp;StarOffice.ComponentContext')
2010-04-16 22:06:28 +00:00
from languagemanager import LanguageManager
2010-10-28 05:21:45 +00:00
from actions import ActionList
2011-03-29 16:44:36 +00:00
__all__ = [u'AppLocation', u'get_application_version', u'check_latest_version',
u'add_actions', u'get_filesystem_encoding', u'LanguageManager',
u'ActionList', u'get_web_page', u'file_is_unicode', u'string_is_unicode',
2011-01-21 22:28:34 +00:00
u'get_uno_command', u'get_uno_instance', u'delete_file']