021b754939
svn: r12559
610 lines
23 KiB
Python
610 lines
23 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2006 Donald N. Allingham, A. Roitman
|
|
# Copyright (C) 2007-2008 B. Malengier
|
|
# Copyright (C) 2008 Lukasz Rymarczyk
|
|
# Copyright (C) 2008 Raphael Ackermann
|
|
# Copyright (C) 2008 Brian G. Matherly
|
|
#
|
|
# 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 2 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, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
|
|
# $Id$
|
|
|
|
"""
|
|
Module responsible for handling the command line arguments for GRAMPS.
|
|
"""
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Standard python modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import os
|
|
import sys
|
|
import getopt
|
|
from gettext import gettext as _
|
|
import logging
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# gramps modules
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
import const
|
|
import Config
|
|
import RecentFiles
|
|
import Utils
|
|
import gen
|
|
from DbManager import CLIDbManager, NAME_FILE, find_locker_name
|
|
|
|
from PluginUtils import Tool
|
|
from gen.plug import PluginManager
|
|
from ReportBase import CATEGORY_BOOK, CATEGORY_CODE, cl_report
|
|
|
|
# Note: Make sure to edit const.py POPT_TABLE too!
|
|
_help = """
|
|
Usage: gramps.py [OPTION...]
|
|
--load-modules=MODULE1,MODULE2,... Dynamic modules to load
|
|
|
|
Help options
|
|
-?, --help Show this help message
|
|
--usage Display brief usage message
|
|
|
|
Application options
|
|
-O, --open=FAMILY_TREE Open family tree
|
|
-i, --import=FILENAME Import file
|
|
-e, --export=FILENAME Export file
|
|
-f, --format=FORMAT Specify format
|
|
-a, --action=ACTION Specify action
|
|
-p, --options=OPTIONS_STRING Specify options
|
|
-d, --debug=LOGGER_NAME Enable debug logs
|
|
-l List Family Trees
|
|
-L List Family Tree Details
|
|
-u, --force-unlock Force unlock of family tree
|
|
"""
|
|
|
|
#-------------------------------------------------------------------------
|
|
# ArgHandler
|
|
#-------------------------------------------------------------------------
|
|
class ArgHandler(object):
|
|
"""
|
|
This class is responsible for handling command line arguments (if any)
|
|
given to gramps. The valid arguments are:
|
|
|
|
FAMTREE : family tree name or database dir to open.
|
|
All following arguments will be ignored.
|
|
-O, --open=FAMTREE : Family tree or family tree database dir to open.
|
|
-i, --import=FILE : filename to import.
|
|
-e, --export=FILE : filename to export.
|
|
-f, --format=FORMAT : format of the file preceding this option.
|
|
|
|
If the filename (no flags) is specified, the interactive session is
|
|
launched using data from filename.
|
|
In this mode (filename, no flags), the rest of the arguments is ignored.
|
|
This is a mode suitable by default for GUI launchers, mime type handlers,
|
|
and the like
|
|
|
|
If no filename or -i option is given, a new interactive session (empty
|
|
database) is launched, since no data is given anyway.
|
|
|
|
If -O or -i option is given, but no -e or -a options are given, an
|
|
interactive session is launched with the FILE (specified with -i).
|
|
|
|
If both input (-O or -i) and processing (-e or -a) options are given,
|
|
interactive session will not be launched.
|
|
"""
|
|
|
|
def __init__(self, state, vm, args):
|
|
self.state = state
|
|
self.vm = vm
|
|
self.args = args
|
|
|
|
self.open_gui = None
|
|
self.open = None
|
|
self.cl = 0
|
|
self.exports = []
|
|
self.actions = []
|
|
self.imports = []
|
|
self.imp_db_path = None
|
|
self.list = False
|
|
self.list_more = False
|
|
self.help = False
|
|
self.force_unlock = False
|
|
self.dbman = CLIDbManager(self.state)
|
|
|
|
self.parse_args()
|
|
|
|
#-------------------------------------------------------------------------
|
|
# Argument parser: sorts out given arguments
|
|
#-------------------------------------------------------------------------
|
|
def parse_args(self):
|
|
"""
|
|
Fill in lists with open, exports, imports, and actions options.
|
|
|
|
Any parsing errors lead to abort.
|
|
|
|
Possible:
|
|
1/ Just the family tree (name or database dir)
|
|
2/ -O, Open of a family tree
|
|
3/ -i, Import of any format understood by an importer, optionally provide
|
|
-f to indicate format
|
|
4/ -e, export a family tree in required format, optionally provide
|
|
-f to indicate format
|
|
5/ -a, --action: An action (possible: 'check', 'summary', 'report',
|
|
'tool')
|
|
6/ -u, --force-unlock: A locked database can be unlocked by given this
|
|
argument when opening it
|
|
|
|
"""
|
|
try:
|
|
options, leftargs = getopt.getopt(self.args[1:],
|
|
const.SHORTOPTS, const.LONGOPTS)
|
|
except getopt.GetoptError, msg:
|
|
print msg
|
|
# return without filling anything if we could not parse the args
|
|
print "Error parsing the arguments: %s " % self.args[1:]
|
|
print "Type gramps --help for an overview of commands, or ",
|
|
print "read manual pages."
|
|
sys.exit(0)
|
|
|
|
if leftargs:
|
|
# if there were an argument without option,
|
|
# use it as a file to open and return
|
|
self.open_gui = leftargs[0]
|
|
print "Trying to open: %s ..." % leftargs[0]
|
|
#see if force open is on
|
|
for opt_ix in range(len(options)):
|
|
option, value = options[opt_ix]
|
|
if option in ('-u', '--force-unlock'):
|
|
self.force_unlock = True
|
|
break
|
|
return
|
|
|
|
# Go over all given option and place them into appropriate lists
|
|
for opt_ix in range(len(options)):
|
|
option, value = options[opt_ix]
|
|
if option in ( '-O', '--open'):
|
|
self.__handle_open_option(value)
|
|
elif option in ( '-i', '--import'):
|
|
format = None
|
|
if opt_ix < len(options) - 1 \
|
|
and options[opt_ix + 1][0] in ( '-f', '--format'):
|
|
format = options[opt_ix + 1][1]
|
|
self.__handle_import_option(value, format)
|
|
elif option in ( '-e', '--export' ):
|
|
format = None
|
|
if opt_ix < len(options) - 1 \
|
|
and options[opt_ix + 1][0] in ( '-f', '--format'):
|
|
format = options[opt_ix + 1][1]
|
|
self.__handle_export_option(value, format)
|
|
elif option in ( '-a', '--action' ):
|
|
action = value
|
|
if action not in ( 'check', 'summary', 'report', 'tool' ):
|
|
print "Unknown action: %s. Ignoring." % action
|
|
continue
|
|
options_str = ""
|
|
if opt_ix < len(options)-1 \
|
|
and options[opt_ix+1][0] in ( '-p', '--options' ):
|
|
options_str = options[opt_ix+1][1]
|
|
self.actions.append((action, options_str))
|
|
elif option in ('-d', '--debug'):
|
|
logger = logging.getLogger(value)
|
|
logger.setLevel(logging.DEBUG)
|
|
elif option in ('-l',):
|
|
self.list = True
|
|
elif option in ('-L',):
|
|
self.list_more = True
|
|
elif option in ('-h', '-?', '--help'):
|
|
self.help = True
|
|
elif option in ('-u', '--force-unlock'):
|
|
self.force_unlock = True
|
|
|
|
def __handle_open_option(self, value):
|
|
"""
|
|
Handle the "-O" or "--open" option.
|
|
"""
|
|
db_path = self.__deduce_db_path(value)
|
|
|
|
if db_path:
|
|
# We have a potential database path.
|
|
# Check if it is good.
|
|
if not self.__check_db(db_path, self.force_unlock):
|
|
sys.exit(0)
|
|
self.open = db_path
|
|
else:
|
|
print _('Input family tree "%s" does not exist.') % value
|
|
print _("If gedcom, gramps-xml or grdb, use the -i option to "
|
|
"import into a family tree instead")
|
|
sys.exit(0)
|
|
|
|
def __handle_import_option(self, value, format):
|
|
"""
|
|
Handle the "-i" or "--import" option.
|
|
"""
|
|
fname = value
|
|
fullpath = os.path.abspath(os.path.expanduser(fname))
|
|
if not os.path.exists(fullpath):
|
|
print 'Import file not found.'
|
|
print "Ignoring import file: %s" % fname
|
|
return
|
|
|
|
if format is None:
|
|
# Guess the file format based on the file extension.
|
|
# This will get the lower case extension without a period,
|
|
# or an empty string.
|
|
format = os.path.splitext(fname)[-1][1:].lower()
|
|
|
|
pmgr = PluginManager.get_instance()
|
|
plugin_found = False
|
|
for plugin in pmgr.get_import_plugins():
|
|
if format == plugin.get_extension():
|
|
plugin_found = True
|
|
|
|
if plugin_found:
|
|
self.imports.append((fname, format))
|
|
else:
|
|
print 'Unrecognized type: "%s" for import file: %s' \
|
|
% (format, fname)
|
|
print "Ignoring import file: %s" % fname
|
|
|
|
def __handle_export_option(self, value, format):
|
|
"""
|
|
Handle the "-e" or "--export" option.
|
|
"""
|
|
fname = value
|
|
fullpath = os.path.abspath(os.path.expanduser(fname))
|
|
if os.path.exists(fullpath):
|
|
print "WARNING: Output file already exist!"
|
|
print "WARNING: It will be overwritten:\n %s" % fullpath
|
|
answer = None
|
|
while not answer:
|
|
answer = raw_input('OK to overwrite? (yes/no) ')
|
|
if answer.upper() in ('Y','YES'):
|
|
print "Will overwrite the existing file: %s" % fullpath
|
|
else:
|
|
print "Will skip the output file: %s" % fullpath
|
|
return
|
|
|
|
if format is None:
|
|
# Guess the file format based on the file extension.
|
|
# This will get the lower case extension without a period,
|
|
# or an empty string.
|
|
format = os.path.splitext(fname)[-1][1:].lower()
|
|
|
|
pmgr = PluginManager.get_instance()
|
|
plugin_found = False
|
|
for plugin in pmgr.get_export_plugins():
|
|
if format == plugin.get_extension():
|
|
plugin_found = True
|
|
|
|
if plugin_found:
|
|
self.exports.append((fullpath, format))
|
|
else:
|
|
print "Unrecognized format for export file %s" % fname
|
|
print "Ignoring export file: %s" % fname
|
|
|
|
def __deduce_db_path(self, db_name_or_path):
|
|
"""
|
|
Attempt to find a database path for the given parameter.
|
|
|
|
@return: The path to a Gramps DB
|
|
or None if a database can not be deduced.
|
|
"""
|
|
# First, check if this is the name of a family tree
|
|
db_path = self.dbman.get_family_tree_path(db_name_or_path)
|
|
|
|
if db_path is None:
|
|
# This is not a known database name.
|
|
# Check if the user provided a db path instead.
|
|
fullpath = os.path.abspath(os.path.expanduser(db_name_or_path))
|
|
if os.path.isdir(fullpath):
|
|
# The user provided a directory. Check if it is a valid tree.
|
|
name_file_path = os.path.join(fullpath, NAME_FILE)
|
|
if os.path.isfile(name_file_path):
|
|
db_path = fullpath
|
|
|
|
return db_path
|
|
|
|
#-------------------------------------------------------------------------
|
|
# Determine the need for GUI
|
|
#-------------------------------------------------------------------------
|
|
def need_gui(self):
|
|
"""
|
|
Determine whether we need a GUI session for the given tasks.
|
|
"""
|
|
if self.open_gui:
|
|
# No-option argument, definitely GUI
|
|
return True
|
|
|
|
# If we have data to work with:
|
|
if (self.open or self.imports):
|
|
if (self.exports or self.actions):
|
|
# have both data and what to do with it => no GUI
|
|
return False
|
|
else:
|
|
# data given, but no action/export => GUI
|
|
return True
|
|
|
|
# No data, can only do GUI here
|
|
return True
|
|
|
|
#-------------------------------------------------------------------------
|
|
# Overall argument handler:
|
|
# sorts out the sequence and details of operations
|
|
#-------------------------------------------------------------------------
|
|
def handle_args(self):
|
|
"""
|
|
Depending on the given arguments, import or open data, launch
|
|
session, write files, and/or perform actions.
|
|
"""
|
|
|
|
if self.list:
|
|
print 'List of known family trees in your database path\n'
|
|
for name, dirname in self.dbman.family_tree_list():
|
|
print dirname, ', with name ', name
|
|
sys.exit(0)
|
|
|
|
if self.list_more:
|
|
print 'GRAMPS Family Trees:'
|
|
summary_list = self.dbman.family_tree_summary()
|
|
for summary in summary_list:
|
|
print "Family Tree \"%s\":" % summary["Family tree"]
|
|
for item in summary:
|
|
if item != "Family tree":
|
|
print " %s: %s" % (item, summary[item])
|
|
sys.exit(0)
|
|
|
|
if self.help:
|
|
print _help
|
|
sys.exit(0)
|
|
|
|
if self.open_gui:
|
|
# First check if a Gramps database was provided
|
|
# (either a database path or a database name)
|
|
db_path = self.__deduce_db_path(self.open_gui)
|
|
|
|
if not db_path:
|
|
# Apparently it is not a database. See if it is a file that
|
|
# can be imported.
|
|
db_path, title = self.dbman.import_new_db(self.open_gui, None)
|
|
|
|
if db_path:
|
|
# Test if not locked or problematic
|
|
if not self.__check_db(db_path, self.force_unlock):
|
|
sys.exit(0)
|
|
# Add the file to the recent items
|
|
path = os.path.join(db_path, "name.txt")
|
|
try:
|
|
ifile = open(path)
|
|
title = ifile.readline().strip()
|
|
ifile.close()
|
|
except:
|
|
title = db_path
|
|
RecentFiles.recent_files(db_path, title)
|
|
else:
|
|
sys.exit(1)
|
|
return db_path
|
|
|
|
if self.open:
|
|
# Family Tree to open was given. Open it
|
|
# Then go on and process the rest of the command line arguments.
|
|
self.cl = bool(self.exports or self.actions)
|
|
|
|
filename = self.open
|
|
|
|
try:
|
|
self.vm.open_activate(filename)
|
|
print "Opened successfully!"
|
|
except:
|
|
print "Error opening the file."
|
|
print "Exiting..."
|
|
sys.exit(1)
|
|
|
|
if self.imports:
|
|
self.cl = bool(self.exports or self.actions or self.cl)
|
|
|
|
if not self.open:
|
|
# Create empty dir for imported database(s)
|
|
self.imp_db_path = Utils.get_empty_tempdir("import_dbdir")
|
|
|
|
newdb = gen.db.GrampsDBDir()
|
|
newdb.write_version(self.imp_db_path)
|
|
|
|
if not self.vm.db_loader.read_file(self.imp_db_path):
|
|
sys.exit(0)
|
|
|
|
for imp in self.imports:
|
|
print "Importing: file %s, format %s." % imp
|
|
self.cl_import(imp[0], imp[1])
|
|
|
|
elif len(self.args) > 1 and not self.open:
|
|
print "No data was given -- will launch interactive session."
|
|
print "To use in the command-line mode,", \
|
|
"supply at least one input file to process."
|
|
print "Launching interactive session..."
|
|
|
|
if self.cl:
|
|
for (action, options_str) in self.actions:
|
|
print "Performing action: %s." % action
|
|
if options_str:
|
|
print "Using options string: %s" % options_str
|
|
self.cl_action(action, options_str)
|
|
|
|
for expt in self.exports:
|
|
print "Exporting: file %s, format %s." % expt
|
|
self.cl_export(expt[0], expt[1])
|
|
|
|
print "Cleaning up."
|
|
# remove files in import db subdir after use
|
|
self.state.db.close()
|
|
if self.imp_db_path:
|
|
Utils.rm_tempdir(self.imp_db_path)
|
|
print "Exiting."
|
|
sys.exit(0)
|
|
|
|
elif Config.get(Config.RECENT_FILE) and Config.get(Config.AUTOLOAD):
|
|
filename = Config.get(Config.RECENT_FILE)
|
|
if os.path.isdir(filename) and \
|
|
os.path.isfile(os.path.join(filename, "name.txt")) and \
|
|
self.__check_db(filename):
|
|
self.vm.db_loader.read_file(filename)
|
|
return filename
|
|
|
|
def __check_db(self, dbpath, force_unlock = False):
|
|
# Test if not locked or problematic
|
|
if force_unlock:
|
|
self.dbman.break_lock(dbpath)
|
|
if self.dbman.is_locked(dbpath):
|
|
print _("Database is locked, cannot open it!")
|
|
print _(" Info: %s") % find_locker_name(dbpath)
|
|
return False
|
|
if self.dbman.needs_recovery(dbpath):
|
|
print _("Database needs recovery, cannot open it!")
|
|
return False
|
|
return True
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Import handler
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
def cl_import(self, filename, format):
|
|
"""
|
|
Command-line import routine. Try to import filename using the format.
|
|
Any errors will cause the sys.exit(1) call.
|
|
"""
|
|
pmgr = PluginManager.get_instance()
|
|
for plugin in pmgr.get_import_plugins():
|
|
if format == plugin.get_extension():
|
|
import_function = plugin.get_import_function()
|
|
import_function(self.state.db, filename, None)
|
|
|
|
if not self.cl:
|
|
if self.imp_db_path:
|
|
return self.vm.open_activate(self.imp_db_path)
|
|
else:
|
|
return self.vm.open_activate(self.open)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Export handler
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
def cl_export(self, filename, format):
|
|
"""
|
|
Command-line export routine.
|
|
Try to write into filename using the format.
|
|
Any errors will cause the sys.exit(1) call.
|
|
"""
|
|
pmgr = PluginManager.get_instance()
|
|
for plugin in pmgr.get_export_plugins():
|
|
if format == plugin.get_extension():
|
|
export_function = plugin.get_export_function()
|
|
export_function(self.state.db, filename)
|
|
|
|
#-------------------------------------------------------------------------
|
|
#
|
|
# Action handler
|
|
#
|
|
#-------------------------------------------------------------------------
|
|
def cl_action(self, action, options_str):
|
|
"""
|
|
Command-line action routine. Try to perform specified action.
|
|
"""
|
|
pmgr = PluginManager.get_instance()
|
|
if action == 'check':
|
|
import Check
|
|
checker = Check.CheckIntegrity(self.state.db, None, None)
|
|
checker.check_for_broken_family_links()
|
|
checker.cleanup_missing_photos(1)
|
|
checker.check_parent_relationships()
|
|
checker.cleanup_empty_families(0)
|
|
errs = checker.build_report(1)
|
|
if errs:
|
|
checker.report(1)
|
|
elif action == 'summary':
|
|
import Summary
|
|
text = Summary.build_report(self.state.db, None)
|
|
print text
|
|
elif action == "report":
|
|
try:
|
|
options_str_dict = dict( [ tuple(chunk.split('='))
|
|
for chunk in options_str.split(',') ] )
|
|
except:
|
|
options_str_dict = {}
|
|
print "Ignoring invalid options string."
|
|
|
|
name = options_str_dict.pop('name', None)
|
|
_cl_list = pmgr.get_cl_list()
|
|
if name:
|
|
for item in _cl_list:
|
|
if name == item[0]:
|
|
category = item[1]
|
|
report_class = item[2]
|
|
options_class = item[3]
|
|
if category in (CATEGORY_BOOK, CATEGORY_CODE):
|
|
options_class(self.state.db, name, category,
|
|
options_str_dict)
|
|
else:
|
|
cl_report(self.state.db, name, category, report_class,
|
|
options_class, options_str_dict)
|
|
return
|
|
# name exists, but is not in the list of valid report names
|
|
msg = "Unknown report name."
|
|
else:
|
|
msg = "Report name not given. Please use -p name=reportname."
|
|
|
|
print "%s\n Available names are:" % msg
|
|
for item in _cl_list:
|
|
# Print cli report name ([item[0]) and GUI report name (item[4])
|
|
if len(item[0]) <= 25:
|
|
print " %s%s- %s" % (item[0],
|
|
" " * (26 - len(item[0])), item[4])
|
|
else:
|
|
print " %s\t- %s" % (item[0], item[4])
|
|
|
|
elif action == "tool":
|
|
try:
|
|
options_str_dict = dict( [ tuple(chunk.split('=')) for
|
|
chunk in options_str.split(',') ] )
|
|
except:
|
|
options_str_dict = {}
|
|
print "Ignoring invalid options string."
|
|
|
|
name = options_str_dict.pop('name', None)
|
|
_cli_tool_list = pmgr.get_cl_tool_list()
|
|
if name:
|
|
for item in _cli_tool_list:
|
|
if name == item[0]:
|
|
category = item[1]
|
|
tool_class = item[2]
|
|
options_class = item[3]
|
|
Tool.cli_tool(self.state, name, category, tool_class,
|
|
options_class, options_str_dict)
|
|
return
|
|
msg = "Unknown tool name."
|
|
else:
|
|
msg = "Tool name not given. Please use -p name=toolname."
|
|
|
|
print "%s\n Available names are:" % msg
|
|
for item in _cli_tool_list:
|
|
print " %s" % item[0]
|
|
else:
|
|
print "Unknown action: %s." % action
|
|
sys.exit(1)
|
|
|