4114: Would like to use stdin and stdout for command line import/export; Xml import
svn: r16765
This commit is contained in:
parent
6d02067406
commit
b10d52a3bf
@ -84,6 +84,21 @@ class GedcomError(Exception):
|
|||||||
"Return string representation"
|
"Return string representation"
|
||||||
return self.value
|
return self.value
|
||||||
|
|
||||||
|
class GrampsImportError(Exception):
|
||||||
|
"""Error used to report mistakes during import of files into Gramps"""
|
||||||
|
def __init__(self, value, value2=""):
|
||||||
|
Exception.__init__(self)
|
||||||
|
self.value = value
|
||||||
|
self.value2 = value2
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
"Return string representation"
|
||||||
|
return self.value
|
||||||
|
|
||||||
|
def messages(self):
|
||||||
|
"Return the messages"
|
||||||
|
return (self.value, self.value2)
|
||||||
|
|
||||||
class PluginError(Exception):
|
class PluginError(Exception):
|
||||||
"""Error used to report plugin errors"""
|
"""Error used to report plugin errors"""
|
||||||
def __init__(self, value):
|
def __init__(self, value):
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
import time
|
||||||
from xml.parsers.expat import ExpatError, ParserCreate
|
from xml.parsers.expat import ExpatError, ParserCreate
|
||||||
from gen.ggettext import gettext as _
|
from gen.ggettext import gettext as _
|
||||||
import re
|
import re
|
||||||
@ -44,6 +45,7 @@ from QuestionDialog import ErrorDialog, WarningDialog
|
|||||||
import gen.mime
|
import gen.mime
|
||||||
import gen.lib
|
import gen.lib
|
||||||
from gen.db import DbTxn
|
from gen.db import DbTxn
|
||||||
|
from Errors import GrampsImportError
|
||||||
import Utils
|
import Utils
|
||||||
import DateHandler
|
import DateHandler
|
||||||
from gen.display.name import displayer as name_displayer
|
from gen.display.name import displayer as name_displayer
|
||||||
@ -99,21 +101,20 @@ def importData(database, filename, callback=None):
|
|||||||
database.smap = {}
|
database.smap = {}
|
||||||
database.pmap = {}
|
database.pmap = {}
|
||||||
database.fmap = {}
|
database.fmap = {}
|
||||||
|
line_cnt = 0
|
||||||
|
person_cnt = 0
|
||||||
|
|
||||||
xml_file = open_file(filename)
|
with ImportOpenFileContextManager(filename) as xml_file:
|
||||||
versionparser = VersionParser(xml_file)
|
if xml_file is None:
|
||||||
|
|
||||||
if xml_file is None or \
|
|
||||||
version_is_valid(versionparser) is False:
|
|
||||||
return
|
return
|
||||||
|
|
||||||
version_string = versionparser.get_xmlns_version()
|
if filename == '-':
|
||||||
#reset file to the start
|
change = time.time()
|
||||||
xml_file.seek(0)
|
else:
|
||||||
|
|
||||||
change = os.path.getmtime(filename)
|
change = os.path.getmtime(filename)
|
||||||
parser = GrampsParser(database, callback, change, version_string)
|
parser = GrampsParser(database, callback, change)
|
||||||
|
|
||||||
|
if filename != '-':
|
||||||
linecounter = LineParser(filename)
|
linecounter = LineParser(filename)
|
||||||
line_cnt = linecounter.get_count()
|
line_cnt = linecounter.get_count()
|
||||||
person_cnt = linecounter.get_person_count()
|
person_cnt = linecounter.get_person_count()
|
||||||
@ -123,6 +124,9 @@ def importData(database, filename, callback=None):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
info = parser.parse(xml_file, line_cnt, person_cnt)
|
info = parser.parse(xml_file, line_cnt, person_cnt)
|
||||||
|
except GrampsImportError, err: # version error
|
||||||
|
ErrorDialog(*err.messages())
|
||||||
|
return
|
||||||
except IOError, msg:
|
except IOError, msg:
|
||||||
ErrorDialog(_("Error reading %s") % filename, str(msg))
|
ErrorDialog(_("Error reading %s") % filename, str(msg))
|
||||||
import traceback
|
import traceback
|
||||||
@ -130,13 +134,11 @@ def importData(database, filename, callback=None):
|
|||||||
return
|
return
|
||||||
except ExpatError, msg:
|
except ExpatError, msg:
|
||||||
ErrorDialog(_("Error reading %s") % filename,
|
ErrorDialog(_("Error reading %s") % filename,
|
||||||
_("The file is probably either corrupt or not a valid Gramps database."))
|
_("The file is probably either corrupt or not a "
|
||||||
|
"valid Gramps database."))
|
||||||
return
|
return
|
||||||
|
|
||||||
xml_file.close()
|
|
||||||
|
|
||||||
database.readonly = read_only
|
database.readonly = read_only
|
||||||
|
|
||||||
return info
|
return info
|
||||||
|
|
||||||
## TODO - WITH MEDIA PATH, IS THIS STILL NEEDED?
|
## TODO - WITH MEDIA PATH, IS THIS STILL NEEDED?
|
||||||
@ -333,6 +335,64 @@ class LineParser(object):
|
|||||||
def get_person_count(self):
|
def get_person_count(self):
|
||||||
return self.person_count
|
return self.person_count
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
# ImportOpenFileContextManager
|
||||||
|
#
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
class ImportOpenFileContextManager:
|
||||||
|
"""
|
||||||
|
Context manager to open a file or stdin for reading.
|
||||||
|
"""
|
||||||
|
def __init__(self, filename):
|
||||||
|
self.filename = filename
|
||||||
|
self.filehandle = None
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
if self.filename == '-':
|
||||||
|
self.filehandle = sys.stdin
|
||||||
|
else:
|
||||||
|
self.filehandle = self.open_file(self.filename)
|
||||||
|
return self.filehandle
|
||||||
|
|
||||||
|
def __exit__(self, exc_type, exc_value, traceback):
|
||||||
|
if self.filename != '-':
|
||||||
|
self.filehandle.close()
|
||||||
|
return False
|
||||||
|
|
||||||
|
def open_file(self, filename):
|
||||||
|
"""
|
||||||
|
Open the xml file.
|
||||||
|
Return a valid file handle if the file opened sucessfully.
|
||||||
|
Return None if the file was not able to be opened.
|
||||||
|
"""
|
||||||
|
if GZIP_OK:
|
||||||
|
use_gzip = True
|
||||||
|
try:
|
||||||
|
ofile = gzip.open(filename, "r")
|
||||||
|
ofile.read(1)
|
||||||
|
ofile.close()
|
||||||
|
except IOError, msg:
|
||||||
|
use_gzip = False
|
||||||
|
except ValueError, msg:
|
||||||
|
use_gzip = True
|
||||||
|
else:
|
||||||
|
use_gzip = False
|
||||||
|
|
||||||
|
try:
|
||||||
|
if use_gzip:
|
||||||
|
xml_file = gzip.open(filename, "rb")
|
||||||
|
else:
|
||||||
|
xml_file = open(filename, "r")
|
||||||
|
except IOError, msg:
|
||||||
|
ErrorDialog(_("%s could not be opened") % filename, str(msg))
|
||||||
|
xml_file = None
|
||||||
|
except:
|
||||||
|
ErrorDialog(_("%s could not be opened") % filename)
|
||||||
|
xml_file = None
|
||||||
|
|
||||||
|
return xml_file
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
#
|
#
|
||||||
# Gramps database parsing class. Derived from SAX XML parser
|
# Gramps database parsing class. Derived from SAX XML parser
|
||||||
@ -340,10 +400,10 @@ class LineParser(object):
|
|||||||
#-------------------------------------------------------------------------
|
#-------------------------------------------------------------------------
|
||||||
class GrampsParser(UpdateCallback):
|
class GrampsParser(UpdateCallback):
|
||||||
|
|
||||||
def __init__(self, database, callback, change, version_string):
|
def __init__(self, database, callback, change):
|
||||||
UpdateCallback.__init__(self, callback)
|
UpdateCallback.__init__(self, callback)
|
||||||
#version of the xml file
|
self.__gramps_version = 'unknown'
|
||||||
self.version_string = version_string
|
self.__xml_version = '1.0.0'
|
||||||
self.stext_list = []
|
self.stext_list = []
|
||||||
self.scomments_list = []
|
self.scomments_list = []
|
||||||
self.note_list = []
|
self.note_list = []
|
||||||
@ -490,7 +550,7 @@ class GrampsParser(UpdateCallback):
|
|||||||
"region": (self.start_region, None),
|
"region": (self.start_region, None),
|
||||||
"father": (self.start_father, None),
|
"father": (self.start_father, None),
|
||||||
"gender": (None, self.stop_gender),
|
"gender": (None, self.stop_gender),
|
||||||
"header": (None, None),
|
"header": (None, self.stop_header),
|
||||||
"map": (self.start_namemap, None),
|
"map": (self.start_namemap, None),
|
||||||
"mediapath": (None, self.stop_mediapath),
|
"mediapath": (None, self.stop_mediapath),
|
||||||
"mother": (self.start_mother, None),
|
"mother": (self.start_mother, None),
|
||||||
@ -804,6 +864,93 @@ class GrampsParser(UpdateCallback):
|
|||||||
self.db.request_rebuild()
|
self.db.request_rebuild()
|
||||||
return self.info
|
return self.info
|
||||||
|
|
||||||
|
def start_database(self, attrs):
|
||||||
|
"""
|
||||||
|
Get the xml version of the file.
|
||||||
|
"""
|
||||||
|
if 'xmlns' in attrs:
|
||||||
|
xmlns = attrs.get('xmlns').split('/')
|
||||||
|
if len(xmlns)>= 2 and not xmlns[2] == 'gramps-project.org':
|
||||||
|
self.__xml_version = '0.0.0'
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
self.__xml_version = xmlns[4]
|
||||||
|
except:
|
||||||
|
#leave version at 1.0.0 although it could be 0.0.0 ??
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
#1.0 or before xml, no dtd schema yet on
|
||||||
|
# http://www.gramps-project.org/xml/
|
||||||
|
self.__xml_version = '0.0.0'
|
||||||
|
|
||||||
|
def start_created(self, attrs):
|
||||||
|
"""
|
||||||
|
Get the Gramps version that produced the file.
|
||||||
|
"""
|
||||||
|
if 'sources' in attrs:
|
||||||
|
self.num_srcs = int(attrs['sources'])
|
||||||
|
else:
|
||||||
|
self.num_srcs = 0
|
||||||
|
if 'places' in attrs:
|
||||||
|
self.num_places = int(attrs['places'])
|
||||||
|
else:
|
||||||
|
self.num_places = 0
|
||||||
|
if 'version' in attrs:
|
||||||
|
self.__gramps_version = attrs.get('version')
|
||||||
|
|
||||||
|
def stop_header(self, *dummy):
|
||||||
|
"""
|
||||||
|
Check the version of Gramps and XML.
|
||||||
|
"""
|
||||||
|
if self.__gramps_version == 'unknown':
|
||||||
|
msg = _("The .gramps file you are importing does not contain the "
|
||||||
|
"version of Gramps with which it was produced.\n\n"
|
||||||
|
"The file will not be imported.")
|
||||||
|
raise GrampsImportError(_('Import file misses Gramps version'), msg)
|
||||||
|
if not re.match("\d+\.\d+\.\d+", self.__xml_version):
|
||||||
|
msg = _("The .gramps file you are importing does not contain a "
|
||||||
|
"valid xml-namespace number.\n\n"
|
||||||
|
"The file will not be imported.")
|
||||||
|
raise GrampsImportError(_('Import file contains unacceptable XML '
|
||||||
|
'namespace version'), msg)
|
||||||
|
if self.__xml_version > libgrampsxml.GRAMPS_XML_VERSION:
|
||||||
|
msg = _("The .gramps file you are importing was made by "
|
||||||
|
"version %(newer)s of "
|
||||||
|
"Gramps, while you are running an older version %(older)s. "
|
||||||
|
"The file will not be imported. Please upgrade to the "
|
||||||
|
"latest version of Gramps and try again." ) % {
|
||||||
|
'newer' : self.__gramps_version, 'older' : const.VERSION }
|
||||||
|
raise GrampsImportError('', msg)
|
||||||
|
if self.__xml_version < '1.0.0':
|
||||||
|
msg = _("The .gramps file you are importing was made by version "
|
||||||
|
"%(oldgramps)s of Gramps, while you are running a more "
|
||||||
|
"recent version %(newgramps)s.\n\n"
|
||||||
|
"The file will not be imported. Please use an older version"
|
||||||
|
" of Gramps that supports version %(xmlversion)s of the "
|
||||||
|
"xml.\nSee\n "
|
||||||
|
"http://gramps-project.org/wiki/index.php?title=GRAMPS_XML"
|
||||||
|
"\n for more info."
|
||||||
|
) % {'oldgramps': self.__gramps_version,
|
||||||
|
'newgramps': const.VERSION,
|
||||||
|
'xmlversion': self.__xml_version,
|
||||||
|
}
|
||||||
|
raise GrampsImportError(_('The file will not be imported'), msg)
|
||||||
|
elif self.__xml_version < '1.1.0':
|
||||||
|
msg = _("The .gramps file you are importing was made by version "
|
||||||
|
"%(oldgramps)s of Gramps, while you are running a much "
|
||||||
|
"more recent version %(newgramps)s.\n\n"
|
||||||
|
"Ensure after import everything is imported correctly. In "
|
||||||
|
"the event of problems, please submit a bug and use an "
|
||||||
|
"older version of Gramps in the meantime to import this "
|
||||||
|
"file, which is version %(xmlversion)s of the xml.\nSee\n "
|
||||||
|
"http://gramps-project.org/wiki/index.php?title=GRAMPS_XML"
|
||||||
|
"\nfor more info."
|
||||||
|
) % {'oldgramps': self.__gramps_version,
|
||||||
|
'newgramps': const.VERSION,
|
||||||
|
'xmlversion': self.__xml_version,
|
||||||
|
}
|
||||||
|
WarningDialog(_('Old xml file'), msg)
|
||||||
|
|
||||||
def start_lds_ord(self, attrs):
|
def start_lds_ord(self, attrs):
|
||||||
self.ord = gen.lib.LdsOrd()
|
self.ord = gen.lib.LdsOrd()
|
||||||
self.ord.set_type_from_xml(attrs['type'])
|
self.ord.set_type_from_xml(attrs['type'])
|
||||||
@ -1293,7 +1440,7 @@ class GrampsParser(UpdateCallback):
|
|||||||
self.name = gen.lib.Name()
|
self.name = gen.lib.Name()
|
||||||
name_type = attrs.get('type', "Birth Name")
|
name_type = attrs.get('type', "Birth Name")
|
||||||
# Mapping "Other Name" from gramps 2.0.x to Unknown
|
# Mapping "Other Name" from gramps 2.0.x to Unknown
|
||||||
if (self.version_string=='1.0.0') and (name_type=='Other Name'):
|
if (self.__xml_version == '1.0.0') and (name_type == 'Other Name'):
|
||||||
self.name.set_type(gen.lib.NameType.UNKNOWN)
|
self.name.set_type(gen.lib.NameType.UNKNOWN)
|
||||||
else:
|
else:
|
||||||
self.name.type.set_from_xml_str(name_type)
|
self.name.type.set_from_xml_str(name_type)
|
||||||
@ -1955,20 +2102,6 @@ class GrampsParser(UpdateCallback):
|
|||||||
|
|
||||||
date_value.set_as_text(attrs['val'])
|
date_value.set_as_text(attrs['val'])
|
||||||
|
|
||||||
def start_created(self, attrs):
|
|
||||||
if 'sources' in attrs:
|
|
||||||
self.num_srcs = int(attrs['sources'])
|
|
||||||
else:
|
|
||||||
self.num_srcs = 0
|
|
||||||
if 'places' in attrs:
|
|
||||||
self.num_places = int(attrs['places'])
|
|
||||||
else:
|
|
||||||
self.num_places = 0
|
|
||||||
|
|
||||||
def start_database(self, attrs):
|
|
||||||
# we already parsed xml once in VersionParser to obtain version
|
|
||||||
pass
|
|
||||||
|
|
||||||
def start_pos(self, attrs):
|
def start_pos(self, attrs):
|
||||||
self.person.position = (int(attrs["x"]), int(attrs["y"]))
|
self.person.position = (int(attrs["x"]), int(attrs["y"]))
|
||||||
|
|
||||||
@ -2557,129 +2690,3 @@ def build_place_title(loc):
|
|||||||
value = append_value(value, loc.country)
|
value = append_value(value, loc.country)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
#-------------------------------------------------------------------------
|
|
||||||
#
|
|
||||||
# VersionParser
|
|
||||||
#
|
|
||||||
#-------------------------------------------------------------------------
|
|
||||||
class VersionParser(object):
|
|
||||||
"""
|
|
||||||
Utility class to quickly get the versions from an XML file.
|
|
||||||
"""
|
|
||||||
def __init__(self, xml_file):
|
|
||||||
"""
|
|
||||||
xml_file must be a file object that is already open.
|
|
||||||
"""
|
|
||||||
self.__p = ParserCreate()
|
|
||||||
self.__p.StartElementHandler = self.__element_handler
|
|
||||||
self.__gramps_version = 'unknown'
|
|
||||||
self.__xml_version = '1.0.0'
|
|
||||||
xml_file.seek(0)
|
|
||||||
self.__p.ParseFile(xml_file)
|
|
||||||
|
|
||||||
def __element_handler(self, tag, attrs):
|
|
||||||
" Handle XML elements "
|
|
||||||
if tag == "database" and 'xmlns' in attrs:
|
|
||||||
xmlns = attrs.get('xmlns').split('/')
|
|
||||||
if len(xmlns)>= 2 and not xmlns[2] == 'gramps-project.org':
|
|
||||||
self.__xml_version = '0.0.0'
|
|
||||||
else:
|
|
||||||
try:
|
|
||||||
self.__xml_version = xmlns[4]
|
|
||||||
except:
|
|
||||||
#leave version at 1.0.0 although it could be 0.0.0 ??
|
|
||||||
pass
|
|
||||||
elif tag == "database" and not 'xmlns' in attrs:
|
|
||||||
#1.0 or before xml, no dtd schema yet on
|
|
||||||
# http://www.gramps-project.org/xml/
|
|
||||||
self.__xml_version = '0.0.0'
|
|
||||||
|
|
||||||
elif tag == "created" and 'version' in attrs:
|
|
||||||
self.__gramps_version = attrs.get('version')
|
|
||||||
|
|
||||||
def get_xmlns_version(self):
|
|
||||||
" Get the namespace version of the file "
|
|
||||||
return self.__xml_version
|
|
||||||
|
|
||||||
def get_gramps_version(self):
|
|
||||||
" Get the version of Gramps that created the file "
|
|
||||||
return self.__gramps_version
|
|
||||||
|
|
||||||
def open_file(filename):
|
|
||||||
"""
|
|
||||||
Open the xml file.
|
|
||||||
Return a valid file handle if the file opened sucessfully.
|
|
||||||
Return None if the file was not able to be opened.
|
|
||||||
"""
|
|
||||||
if GZIP_OK:
|
|
||||||
use_gzip = True
|
|
||||||
try:
|
|
||||||
ofile = gzip.open(filename, "r")
|
|
||||||
ofile.read(1)
|
|
||||||
ofile.close()
|
|
||||||
except IOError, msg:
|
|
||||||
use_gzip = False
|
|
||||||
except ValueError, msg:
|
|
||||||
use_gzip = True
|
|
||||||
else:
|
|
||||||
use_gzip = False
|
|
||||||
|
|
||||||
try:
|
|
||||||
if use_gzip:
|
|
||||||
xml_file = gzip.open(filename, "rb")
|
|
||||||
else:
|
|
||||||
xml_file = open(filename, "r")
|
|
||||||
except IOError, msg:
|
|
||||||
ErrorDialog(_("%s could not be opened") % filename, str(msg))
|
|
||||||
xml_file = None
|
|
||||||
except:
|
|
||||||
ErrorDialog(_("%s could not be opened") % filename)
|
|
||||||
xml_file = None
|
|
||||||
|
|
||||||
return xml_file
|
|
||||||
|
|
||||||
def version_is_valid(versionparser):
|
|
||||||
"""
|
|
||||||
Validate the xml version.
|
|
||||||
:param versionparser: A VersionParser object to work with
|
|
||||||
"""
|
|
||||||
|
|
||||||
if versionparser.get_xmlns_version() > libgrampsxml.GRAMPS_XML_VERSION:
|
|
||||||
msg = _("The .gramps file you are importing was made by version %(newer)s of "
|
|
||||||
"Gramps, while you are running an older version %(older)s. "
|
|
||||||
"The file will not be imported. Please upgrade to the latest "
|
|
||||||
"version of Gramps and try again." ) % {
|
|
||||||
'newer' : versionparser.get_gramps_version(), 'older' : const.VERSION }
|
|
||||||
ErrorDialog(msg)
|
|
||||||
return False
|
|
||||||
if versionparser.get_xmlns_version() < '1.0.0':
|
|
||||||
msg = _("The .gramps file you are importing was made by version "
|
|
||||||
"%(oldgramps)s of Gramps, while you are running a more "
|
|
||||||
"recent version %(newgramps)s.\n\n"
|
|
||||||
"The file will not be imported. Please use an older version of"
|
|
||||||
" Gramps that supports version %(xmlversion)s of the xml.\nSee"
|
|
||||||
"\n http://gramps-project.org/wiki/index.php?title=GRAMPS_XML\n "
|
|
||||||
"for more info."
|
|
||||||
) % {'oldgramps': versionparser.get_gramps_version(),
|
|
||||||
'newgramps': const.VERSION,
|
|
||||||
'xmlversion': versionparser.get_xmlns_version(),
|
|
||||||
}
|
|
||||||
ErrorDialog(_('The file will not be imported'), msg)
|
|
||||||
return False
|
|
||||||
elif versionparser.get_xmlns_version() < '1.1.0':
|
|
||||||
msg = _("The .gramps file you are importing was made by version "
|
|
||||||
"%(oldgramps)s of Gramps, while you are running a much "
|
|
||||||
"more recent version %(newgramps)s.\n\n"
|
|
||||||
"Ensure after import everything is imported correctly. In the "
|
|
||||||
"event of problems, please submit a bug and use an older "
|
|
||||||
"version of Gramps in the meantime to import this file, which "
|
|
||||||
"is version %(xmlversion)s of the xml.\nSee"
|
|
||||||
"\n http://gramps-project.org/wiki/index.php?title=GRAMPS_XML\n"
|
|
||||||
"for more info."
|
|
||||||
) % {'oldgramps': versionparser.get_gramps_version(),
|
|
||||||
'newgramps': const.VERSION,
|
|
||||||
'xmlversion': versionparser.get_xmlns_version(),
|
|
||||||
}
|
|
||||||
WarningDialog(_('Old xml file'), msg)
|
|
||||||
return True
|
|
||||||
return True
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user