openlp/openlp/plugins/bibles/lib/csvbible.py

220 lines
8.7 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# vim: autoindent shiftwidth=4 expandtab textwidth=80 tabstop=4 softtabstop=4
2008-11-17 21:22:14 +00:00
###############################################################################
# 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 #
2010-07-24 22:10:47 +00:00
# Gorven, Scott Guerrieri, Meinert Jordan, Andreas Preikschat, Christian #
# Richter, Philip Ridout, Maikel Stuivenberg, Martin Thompson, Jon Tibble, #
# Carsten Tinggaard, 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 #
###############################################################################
2011-01-27 03:13:43 +00:00
"""
The :mod:`cvsbible` modules provides a facility to import bibles from a set of
CSV files.
2008-11-17 21:22:14 +00:00
2011-01-27 03:13:43 +00:00
The module expects two mandatory files containing the books and the verses and
will accept an optional third file containing the testaments.
The format of the testament file is:
<testament_id>,<testament_name>
For example:
1,Old Testament
2,New Testament
The format of the books file is:
<book_id>,<testament_id>,<book_name>,<book_abbreviation>
For example
1,1,Genesis,Gen
2,1,Exodus,Exod
...
40,2,Matthew,Matt
2011-01-30 09:43:07 +00:00
There are two acceptable formats of the verses file. They are:
2011-01-27 03:13:43 +00:00
<book_id>,<chapter_number>,<verse_number>,<verse_text>
2011-01-30 09:43:07 +00:00
or
<book_name>,<chapter_number>,<verse_number>,<verse_text>
2011-01-27 03:13:43 +00:00
For example:
1,1,1,"In the beginning God created the heaven and the earth."
2011-01-30 09:43:07 +00:00
or
"Genesis",1,2,"And the earth was without form, and void; and...."
2011-01-27 03:13:43 +00:00
All CSV files are expected to use a comma (',') as the delimeter and double
quotes ('"') as the quote symbol.
"""
import logging
import chardet
import csv
2009-01-20 19:50:37 +00:00
2010-05-25 15:04:37 +00:00
from PyQt4 import QtCore
2010-12-06 19:30:04 +00:00
from openlp.core.lib import Receiver, translate
2011-01-27 16:45:23 +00:00
from openlp.plugins.bibles.lib.db import BibleDB, Testament
2008-11-17 21:22:14 +00:00
log = logging.getLogger(__name__)
class CSVBible(BibleDB):
"""
This class provides a specialisation for importing of CSV Bibles.
"""
def __init__(self, parent, **kwargs):
2008-11-17 21:22:14 +00:00
"""
2011-01-27 03:13:43 +00:00
Loads a Bible from a set of CVS files.
This class assumes the files contain all the information and
2008-11-17 21:22:14 +00:00
a clean bible is being loaded.
"""
log.info(self.__class__.__name__)
2010-12-06 19:30:04 +00:00
BibleDB.__init__(self, parent, **kwargs)
2011-01-27 03:13:43 +00:00
try:
self.testamentsfile = kwargs[u'testamentsfile']
except KeyError:
self.testamentsfile = None
self.booksfile = kwargs[u'booksfile']
2010-09-03 05:33:03 +00:00
self.versesfile = kwargs[u'versefile']
2010-02-06 15:33:23 +00:00
QtCore.QObject.connect(Receiver.get_receiver(),
2011-01-13 17:55:29 +00:00
QtCore.SIGNAL(u'openlp_stop_wizard'), self.stop_import)
2011-01-27 03:13:43 +00:00
def setup_testaments(self):
"""
Overrides parent method so we can handle importing a testament file.
"""
if self.testamentsfile:
self.wizard.progressBar.setMinimum(0)
self.wizard.progressBar.setMaximum(2)
self.wizard.progressBar.setValue(0)
testaments_file = None
try:
details = get_file_encoding(self.testamentsfile)
testaments_file = open(self.testamentsfile, 'rb')
testaments_reader = csv.reader(testaments_file, delimiter=',',
quotechar='"')
for line in testaments_reader:
if self.stop_import_flag:
break
self.wizard.incrementProgressBar(unicode(
translate('BibleDB.Wizard',
'Importing testaments... %s')) %
unicode(line[1], details['encoding']), 0)
self.save_object(Testament.populate(
name=unicode(line[1], details['encoding'])))
Receiver.send_message(u'openlp_process_events')
except (IOError, IndexError):
log.exception(u'Loading testaments from file failed')
finally:
if testaments_file:
testaments_file.close()
self.wizard.incrementProgressBar(unicode(translate(
'BibleDB.Wizard', 'Importing testaments... done.')), 2)
else:
BibleDB.setup_testaments(self)
def do_import(self):
2011-01-27 03:13:43 +00:00
"""
Import the bible books and verses.
"""
self.wizard.progressBar.setValue(0)
self.wizard.progressBar.setMinimum(0)
2011-01-27 16:55:49 +00:00
self.wizard.progressBar.setMaximum(66)
success = True
books_file = None
2011-01-27 03:13:43 +00:00
book_list = {}
2010-12-06 19:30:04 +00:00
# Populate the Tables
2009-11-07 00:00:36 +00:00
try:
2011-01-27 03:13:43 +00:00
details = get_file_encoding(self.booksfile)
books_file = open(self.booksfile, 'r')
2011-01-27 03:13:43 +00:00
books_reader = csv.reader(books_file, delimiter=',', quotechar='"')
for line in books_reader:
2010-02-06 15:33:23 +00:00
if self.stop_import_flag:
2009-11-07 00:00:36 +00:00
break
2011-01-27 03:13:43 +00:00
self.wizard.incrementProgressBar(unicode(
translate('BibleDB.Wizard', 'Importing books... %s')) %
unicode(line[2], details['encoding']))
self.create_book(unicode(line[2], details['encoding']),
unicode(line[3], details['encoding']), int(line[1]))
book_list[int(line[0])] = unicode(line[2], details['encoding'])
Receiver.send_message(u'openlp_process_events')
2011-01-13 02:28:03 +00:00
except (IOError, IndexError):
2009-11-07 00:00:36 +00:00
log.exception(u'Loading books from file failed')
success = False
2009-11-07 00:00:36 +00:00
finally:
if books_file:
books_file.close()
2011-01-27 03:13:43 +00:00
if self.stop_import_flag or not success:
return False
2011-01-27 03:13:43 +00:00
self.wizard.progressBar.setValue(0)
2011-01-27 16:55:49 +00:00
self.wizard.progressBar.setMaximum(67)
2011-01-27 03:13:43 +00:00
verse_file = None
2009-11-07 00:00:36 +00:00
try:
2011-01-27 03:13:43 +00:00
book_ptr = None
details = get_file_encoding(self.versesfile)
verse_file = open(self.versesfile, 'rb')
verse_reader = csv.reader(verse_file, delimiter=',', quotechar='"')
for line in verse_reader:
if self.stop_import_flag:
2009-11-07 00:00:36 +00:00
break
2011-01-30 09:32:09 +00:00
try:
line_book = book_list[int(line[0])]
except ValueError:
line_book = unicode(line[0], details['encoding'])
if book_ptr != line_book:
book = self.get_book(line_book)
2009-11-07 00:00:36 +00:00
book_ptr = book.name
self.wizard.incrementProgressBar(unicode(translate(
2011-01-27 03:13:43 +00:00
'BibleDB.Wizard', 'Importing verses from %s...',
'Importing verses from <book name>...')) % book.name)
2010-06-15 00:44:06 +00:00
self.session.commit()
self.create_verse(book.id, line[1], line[2],
2010-12-06 19:30:04 +00:00
unicode(line[3], details['encoding']))
2011-01-27 16:55:49 +00:00
self.wizard.incrementProgressBar(translate('BibleDB.Wizard',
'Importing verses... done.'))
2011-01-27 03:13:43 +00:00
Receiver.send_message(u'openlp_process_events')
2010-06-15 00:44:06 +00:00
self.session.commit()
2010-05-29 19:50:50 +00:00
except IOError:
2009-11-07 00:00:36 +00:00
log.exception(u'Loading verses from file failed')
success = False
2009-11-07 00:00:36 +00:00
finally:
if verse_file:
verse_file.close()
2010-02-06 15:33:23 +00:00
if self.stop_import_flag:
return False
else:
return success
2011-01-27 03:13:43 +00:00
def get_file_encoding(filename):
"""
Utility function to get the file encoding.
"""
detect_file = None
try:
detect_file = open(filename, 'r')
details = chardet.detect(detect_file.read(1024))
except IOError:
log.exception(u'Error detecting file encoding')
finally:
if detect_file:
detect_file.close()
return details