# Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2000-2007 Donald N. Allingham # Copyright (C) 2008-2009 Brian G. Matherly # Copyright (C) 2009 Rob G. Healey # # 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: libholiday.py 13878 2009-12-21 10:30:00Z robhealey1 $ # #------------------------------------------------------------------------ # # python modules # #------------------------------------------------------------------------ from gen.ggettext import gettext as _ from xml.parsers import expat import datetime import math import const import os #------------------------------------------------------------------------ # # GRAMPS modules # #------------------------------------------------------------------------ #------------------------------------------------------------------------ # # Support functions # #------------------------------------------------------------------------ def g2iso(dow): """ Converst GRAMPS day of week to ISO day of week """ # GRAMPS: SUN = 1 # ISO: MON = 1 return (dow + 5) % 7 + 1 def easter(year): """ Computes the year/month/day of easter. Based on work by J.-M. Oudin (1940) and is reprinted in the "Explanatory Supplement to the Astronomical Almanac", ed. P. K. Seidelmann (1992). Note: Ash Wednesday is 46 days before Easter Sunday. """ c = year / 100 n = year - 19 * (year / 19) k = (c - 17) / 25 i = c - c / 4 - (c - k) / 3 + 19 * n + 15 i = i - 30 * (i / 30) i = i - (i / 28) * (1 - (i / 28) * (29 / (i + 1)) * ((21 - n) / 11)) j = year + year / 4 + i + 2 - c + c / 4 j = j - 7 * (j / 7) l = i - j month = 3 + (l + 40) / 44 day = l + 28 - 31 * (month / 4) return "%d/%d/%d" % (year, month, day) def dst(year, area="us"): """ Return Daylight Saving Time start/stop in a given area ("us", "eu"). US calculation valid 1976-2099; EU 1996-2099 """ if area == "us": if year > 2006: start = "%d/%d/%d" % (year, 3, 14 - (math.floor(1 + year * 5 / 4) % 7)) # March stop = "%d/%d/%d" % (year, 11, 7 - (math.floor(1 + year * 5 / 4) % 7)) # November else: start = "%d/%d/%d" % (year, 4, (2 + 6 * year - math.floor(year / 4)) % 7 + 1) # April stop = "%d/%d/%d" % (year, 10, (31 - (math.floor(year * 5 / 4) + 1) % 7)) # October elif area == "eu": start = "%d/%d/%d" % (year, 3, (31 - (math.floor(year * 5 / 4) + 4) % 7)) # March stop = "%d/%d/%d" % (year, 10, (31 - (math.floor(year * 5 / 4) + 1) % 7)) # Oct return (start, stop) #------------------------------------------------------------------------ # # HolidayTable # #------------------------------------------------------------------------ class HolidayTable(object): """ HolidayTable is a class which provides holidays for various countries and years. """ __holiday_files = [] __countries = [] def __init__(self): """ Find the holiday files and load the countries if it has not already been done. """ if( not HolidayTable.__holiday_files ): self.__find_holiday_files() if( not HolidayTable.__countries ): self.__build_country_list() # Initialize the holiday table to be empty self.__holidays = {} self.__init_table() def __find_holiday_files(self): """ Looks in multiple places for holidays.xml files It will search for the file in user;s plugin directories first, then it will search in program's plugins directories. """ holiday_file = 'holidays.xml' # Look for holiday files in the user plugins directory and all # subdirectories. for (dirpath, dirnames, filenames) in os.walk(const.USER_PLUGINS): holiday_full_path = os.path.join(dirpath, holiday_file) if os.path.exists(holiday_full_path): HolidayTable.__holiday_files.append(holiday_full_path) # Look for holiday files in the installation plugins directory and all # subdirectories. for (dirpath, dirnames, filenames) in os.walk(const.PLUGINS_DIR): holiday_full_path = os.path.join(dirpath, holiday_file) if os.path.exists(holiday_full_path): HolidayTable.__holiday_files.append(holiday_full_path) def __build_country_list(self): """ Generate the list of countries that have holiday information. """ for holiday_file_path in HolidayTable.__holiday_files: parser = _Xml2Obj() root_element = parser.parse(holiday_file_path) for country_element in root_element.get_children(): if country_element.get_name() == "country": country_name = country_element.get_attribute("name") if country_name not in HolidayTable.__countries: HolidayTable.__countries.append(country_name) def __init_table(self): """ Initialize the holiday table structure. """ for month in range(1, 13): self.__holidays[month] = {} for day in range(1, 32): self.__holidays[month][day] = [] def get_countries(self): """ Get all the country names that holidays are available for. @return: nothing """ return HolidayTable.__countries def load_holidays(self, year, country): """ Load the holiday table for the specified year and country. This must be called before get_holidays(). @param year: The year for which the holidays should be loaded. Example: 2010 @type year: int @param country: The country for which the holidays should be loaded. Example: "United States" @type country: str @return: nothing """ self.__init_table() for holiday_file_path in HolidayTable.__holiday_files: parser = _Xml2Obj() element = parser.parse(holiday_file_path) calendar = _Holidays(element, country) date = datetime.date(year, 1, 1) while date.year == year: holidays = calendar.check_date(date) for text in holidays: self.__holidays[date.month][date.day].append(text) date = date.fromordinal(date.toordinal() + 1) def get_holidays(self, month, day): """ Get the holidays for the given day of the year. @param month: The month for the requested holidays. Example: 1 @type month: int @param month: The day for the requested holidays. Example: 1 @type month: int @return: An array of strings with holiday names. @return type: [str] """ return self.__holidays[month][day] #------------------------------------------------------------------------ # # _Element # #------------------------------------------------------------------------ class _Element: """ A parsed XML element """ def __init__(self, name, attributes): 'Element constructor' # The element's tag name self.__name = name # The element's attribute dictionary self.__attributes = attributes # The element's child element list (sequence) self.__children = [] def add_child(self, element): 'Add a reference to a child element' self.__children.append(element) def get_attribute(self, key): 'Get an attribute value' return self.__attributes.get(key) def get_attributes(self): 'Get all the attributes' return self.__attributes def get_name(self): """ Get the name of this element. """ return self.__name def get_children(self): """ Get the children elements for this element. """ return self.__children #------------------------------------------------------------------------ # # _Xml2Obj # #------------------------------------------------------------------------ class _Xml2Obj: """ XML to Object """ def __init__(self): self.root = None self.nodeStack = [] def start_element(self, name, attributes): 'SAX start element even handler' # Instantiate an Element object element = _Element(name.encode(), attributes) # Push element onto the stack and make it a child of parent if len(self.nodeStack) > 0: parent = self.nodeStack[-1] parent.add_child(element) else: self.root = element self.nodeStack.append(element) def end_element(self, name): 'SAX end element event handler' self.nodeStack = self.nodeStack[:-1] def parse(self, filename): 'Create a SAX parser and parse filename ' parser = expat.ParserCreate() # SAX event handlers parser.StartElementHandler = self.start_element parser.EndElementHandler = self.end_element # Parse the XML File parser.Parse(open(filename, 'r').read(), 1) return self.root #------------------------------------------------------------------------ # # _Holidays # #------------------------------------------------------------------------ class _Holidays: """ Class used to read XML holidays to add to calendar. """ def __init__(self, elements, country="US"): self.debug = 0 self.elements = elements self.country = country self.dates = [] self.initialize() def set_country(self, country): """ Set the contry of holidays to read """ self.country = country self.dates = [] self.initialize() def initialize(self): """ Parse the holiday date XML items """ for country_set in self.elements.get_children(): if country_set.get_name() == "country" and \ country_set.get_attribute("name") == self.country: for date in country_set.get_children(): if date.get_name() == "date": data = {"value" : "", "name" : "", "offset": "", "type": "", "if": "", } # defaults for attr in date.get_attributes(): data[attr] = date.get_attribute(attr) self.dates.append(data) def get_daynames(self, year, month, dayname): """ Get the items for a particular year/month and day of week """ if self.debug: print "%s's in %d %d..." % (dayname, month, year) retval = [0] dow = ['mon', 'tue', 'wed', 'thu', 'fri', 'sat', 'sun'].index(dayname) for day in range(1, 32): try: date = datetime.date(year, month, day) except ValueError: continue if date.weekday() == dow: retval.append(day) if self.debug: print "dow=", dow, "days=", retval return retval def check_date(self, date): """ Return items that match rules """ retval = [] for rule in self.dates: if self.debug: print "Checking ", rule["name"], "..." offset = 0 if rule["offset"] != "": if rule["offset"].isdigit(): offset = int(rule["offset"]) elif rule["offset"][0] in ["-", "+"] and \ rule["offset"][1:].isdigit(): offset = int(rule["offset"]) else: # must be a dayname offset = rule["offset"] if rule["value"].startswith('>'): # eval exp -> year/num[/day[/month]] y, m, d = date.year, date.month, date.day rule["value"] = eval(rule["value"][1:]) if self.debug: print "rule['value']:", rule["value"] if rule["value"].count("/") == 3: # year/num/day/month, "3rd wednesday in april" y, num, dayname, mon = rule["value"].split("/") if y == "*": y = date.year else: y = int(y) if mon.isdigit(): m = int(mon) elif mon == "*": m = date.month else: m = ['jan', 'feb', 'mar', 'apr', 'may', 'jun', 'jul', 'aug', 'sep', 'oct', 'nov', 'dec'].index(mon) + 1 dates_of_dayname = self.get_daynames(y, m, dayname) if self.debug: print "num =", num d = dates_of_dayname[int(num)] elif rule["value"].count("/") == 2: # year/month/day y, m, d = rule["value"].split("/") if y == "*": y = date.year else: y = int(y) if m == "*": m = date.month else: m = int(m) if d == "*": d = date.day else: d = int(d) ndate = datetime.date(y, m, d) if self.debug: print ndate, offset, type(offset) if isinstance(offset, int): if offset != 0: ndate = ndate.fromordinal(ndate.toordinal() + offset) elif isinstance(offset, basestring): direction = 1 if offset[0] == "-": direction = -1 offset = offset[1:] if offset in ['mon', 'tue', 'wed', 'thu', 'fri', 'sat', 'sun']: # next tuesday you come to, including this one dow = ['mon', 'tue', 'wed', 'thu', 'fri', 'sat', 'sun'].index(offset) ordinal = ndate.toordinal() while ndate.fromordinal(ordinal).weekday() != dow: ordinal += direction ndate = ndate.fromordinal(ordinal) if self.debug: print "ndate:", ndate, "date:", date if ndate == date: if rule["if"] != "": if not eval(rule["if"]): continue retval.append(rule["name"]) return retval