|
|
|
#!/usr/bin/env python3
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# summarypage.py
|
|
|
|
#
|
|
|
|
# Copyright 2015 GOLDERWEB – Jonathan Golder <jonathan@golderweb.de>
|
|
|
|
#
|
|
|
|
# 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., 51 Franklin Street, Fifth Floor, Boston,
|
|
|
|
# MA 02110-1301, USA.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
"""
|
|
|
|
Provides classes for handling Charts summary page
|
|
|
|
"""
|
|
|
|
|
|
|
|
import locale
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
|
|
import pywikibot
|
|
|
|
import mwparserfromhell as mwparser
|
|
|
|
|
|
|
|
from countrylist import CountryList, CountryListError
|
|
|
|
|
|
|
|
class SummaryPage():
|
|
|
|
"""
|
|
|
|
Handles summary page related actions
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__( self, text ):
|
|
|
|
"""
|
|
|
|
Create Instance
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Parse Text with mwparser
|
|
|
|
self.wikicode = mwparser.parse( text )
|
|
|
|
|
|
|
|
def treat( self ):
|
|
|
|
"""
|
|
|
|
Handles parsing/editing of text
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Get mwparser.template objects for Template "/Eintrag"
|
|
|
|
for entry in self.wikicode.filter_templates( matches="/Eintrag" ) :
|
|
|
|
|
|
|
|
# Instantiate SummaryPageEntry-object
|
|
|
|
summarypageentry = SummaryPageEntry( entry )
|
|
|
|
# Treat SummaryPageEntry-object
|
|
|
|
summarypageentry.treat()
|
|
|
|
|
|
|
|
# Get result
|
|
|
|
# We need to replace origninal entry since objectid changes due to
|
|
|
|
# recreation of template object and reassignment won't be reflected
|
|
|
|
self.wikicode.replace( entry, summarypageentry.new_entry.template )
|
|
|
|
|
|
|
|
def get_new_text( self ):
|
|
|
|
"""
|
|
|
|
If writing page is needed, return new text, otherwise false
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Get information wether writing is needed from class attribute
|
|
|
|
if SummaryPageEntry.write_needed:
|
|
|
|
|
|
|
|
# Convert wikicode back to string and return
|
|
|
|
return str( self.wikicode )
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class SummaryPageEntry():
|
|
|
|
"""
|
|
|
|
Provides a generic wrapper for summary page entry template
|
|
|
|
"""
|
|
|
|
|
|
|
|
write_needed = False
|
|
|
|
|
|
|
|
def __init__( self, entry ):
|
|
|
|
"""
|
|
|
|
Constructor
|
|
|
|
"""
|
|
|
|
self.old_entry = SummaryPageEntryTemplate( entry )
|
|
|
|
self.new_entry = SummaryPageEntryTemplate( )
|
|
|
|
|
|
|
|
def treat( self ):
|
|
|
|
"""
|
|
|
|
Controls parsing/update-sequence of entry
|
|
|
|
"""
|
|
|
|
self.parse()
|
|
|
|
|
|
|
|
self.correct_chartein()
|
|
|
|
|
|
|
|
self.update_params()
|
|
|
|
|
|
|
|
self.is_write_needed()
|
|
|
|
|
|
|
|
def parse( self ):
|
|
|
|
"""
|
|
|
|
Handles parsing process of entry template
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Get wikilink to related countrylist
|
|
|
|
self.get_countrylist_wikilink()
|
|
|
|
|
|
|
|
# Get saved revision of related countrylist
|
|
|
|
self.get_countrylist_saved_revid()
|
|
|
|
|
|
|
|
|
|
|
|
# Get current year
|
|
|
|
current_year = datetime.now().year;
|
|
|
|
|
|
|
|
# Store old link.title
|
|
|
|
link_title = self.countrylist_wikilink.title
|
|
|
|
|
|
|
|
# If list is from last year, replace year
|
|
|
|
if (current_year - 1) in link_title:
|
|
|
|
self.countrylist_wikilink.title.replace( (current_year - 1), current_year )
|
|
|
|
|
|
|
|
# Try to get current years list
|
|
|
|
try:
|
|
|
|
self.countrylist = CountryList( self.countrylist_wikilink )
|
|
|
|
|
|
|
|
if self.countrylist:
|
|
|
|
self.countrylist.parse()
|
|
|
|
|
|
|
|
# Maybe fallback to last years list
|
|
|
|
except CountryListError:
|
|
|
|
|
|
|
|
self.countrylist_wikilink.title = link_title
|
|
|
|
self.countrylist = CountryList( self.countrylist_wikilink )
|
|
|
|
|
|
|
|
if self.countrylist:
|
|
|
|
self.countrylist.parse()
|
|
|
|
else:
|
|
|
|
raise SummaryPageEntryError( "CountryList does not exists!" )
|
|
|
|
|
|
|
|
def get_countrylist_wikilink( self ):
|
|
|
|
"""
|
|
|
|
Load wikilink to related countrylist
|
|
|
|
"""
|
|
|
|
if self.old_entry.Liste:
|
|
|
|
try:
|
|
|
|
self.countrylist_wikilink = next( self.old_entry.Liste.ifilter_wikilinks() )
|
|
|
|
except StopIteration:
|
|
|
|
raise SummaryPageEntryError( "Parameter Liste does not contain valid wikilink!")
|
|
|
|
else:
|
|
|
|
raise SummaryPageEntryError( "Parameter Liste is not present!")
|
|
|
|
|
|
|
|
def get_countrylist_saved_revid( self ):
|
|
|
|
"""
|
|
|
|
Load saved revid of related countrylist if Param is present
|
|
|
|
"""
|
|
|
|
if self.old_entry.Liste_Revision:
|
|
|
|
self.countrylist_revid = int( self.old_entry.Liste_Revision.strip())
|
|
|
|
else:
|
|
|
|
self.countrylist_revid = 0
|
|
|
|
|
|
|
|
def update_params( self ):
|
|
|
|
"""
|
|
|
|
Updates values of Parameters of template
|
|
|
|
"""
|
|
|
|
|
|
|
|
self.new_entry.Liste = self.countrylist_wikilink
|
|
|
|
self.new_entry.Liste_Revision = self.countrylist.page.latest_revision_id
|
|
|
|
self.new_entry.Interpret = self.countrylist.interpret
|
|
|
|
self.new_entry.Titel = self.countrylist.titel
|
|
|
|
self.new_entry.Chartein = self._corrected_chartein
|
|
|
|
|
|
|
|
if self.old_entry.Korrektur:
|
|
|
|
self.new_entry.Korrektur = self.old_entry.Korrektur
|
|
|
|
else:
|
|
|
|
self.new_entry.Korrektur = ""
|
|
|
|
|
|
|
|
if self.old_entry.Hervor:
|
|
|
|
self.new_entry.Hervor = self.old_entry.Hervor
|
|
|
|
else:
|
|
|
|
self.new_entry.Hervor = ""
|
|
|
|
|
|
|
|
def correct_chartein( self ):
|
|
|
|
"""
|
|
|
|
Calulates the correct value of chartein, based on the chartein value
|
|
|
|
from countrylist entry and param Korrektur of summarypage entry
|
|
|
|
"""
|
|
|
|
# If param Korrektur is present extract the value
|
|
|
|
if self.old_entry.Korrektur:
|
|
|
|
# If Korrektur is (after striping) castable to int use it
|
|
|
|
try:
|
|
|
|
days = int( str( self.old_entry.Korrektur ).strip() )
|
|
|
|
# Otherwise, if casting fails, ignore it
|
|
|
|
except ValueError:
|
|
|
|
days = 0
|
|
|
|
else:
|
|
|
|
days = 0
|
|
|
|
|
|
|
|
corrected = self.countrylist.chartein + timedelta( days=days )
|
|
|
|
self._corrected_chartein = corrected.strftime( "%d. %B" ).lstrip( "0" )
|
|
|
|
|
|
|
|
def is_write_needed( self ):
|
|
|
|
"""
|
|
|
|
Detects wether writing of entry is needed and stores information in
|
|
|
|
Class-Attribute
|
|
|
|
"""
|
|
|
|
type( self ).write_needed = ( ( self.old_entry != self.new_entry ) or \
|
|
|
|
type( self ).write_needed )
|
|
|
|
|
|
|
|
|
|
|
|
class SummaryPageEntryTemplate():
|
|
|
|
"""
|
|
|
|
Interface class for mwparser.template to simply use template params as
|
|
|
|
Properties
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Classatribute
|
|
|
|
params = ( "Liste", "Liste_Revision", "Interpret", "Titel", "Chartein",
|
|
|
|
"Korrektur", "Hervor" )
|
|
|
|
|
|
|
|
def __init__( self, template_obj=None ):
|
|
|
|
"""
|
|
|
|
Creates Instance of Class for given mwparser.template object of
|
|
|
|
SummmaryPageEntry Template. If no object was given create empty one.
|
|
|
|
|
|
|
|
@param template_obj mw.parser.template Object of
|
|
|
|
SummmaryPageEntry Template
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Check if object was given
|
|
|
|
if( template_obj ):
|
|
|
|
|
|
|
|
# Check if object has correct type
|
|
|
|
if isinstance( template_obj,
|
|
|
|
mwparser.nodes.template.Template ):
|
|
|
|
|
|
|
|
self.template = template_obj;
|
|
|
|
self.__initial = False;
|
|
|
|
|
|
|
|
# Otherwise raise error
|
|
|
|
else:
|
|
|
|
raise SummaryPageEntryTemplateError( "Wrong type given" );
|
|
|
|
|
|
|
|
# Otherwise initialise template
|
|
|
|
else:
|
|
|
|
self.__initial_template()
|
|
|
|
self.__initial = True;
|
|
|
|
|
|
|
|
def __initial_template( self ):
|
|
|
|
"""
|
|
|
|
Builds the initial template
|
|
|
|
"""
|
|
|
|
|
|
|
|
self.template = next( mwparser.parse(
|
|
|
|
"{{Portal:Charts und Popmusik/Aktuelle Nummer-eins-Hits/Eintrag|Liste=|Liste_Revision=|Interpret=|Titel=NN\
|
|
|
|
|Chartein=|Korrektur=|Hervor=}}" ).ifilter_templates() )
|
|
|
|
|
|
|
|
def __getattr__( self, name ):
|
|
|
|
"""
|
|
|
|
Special getter for template params
|
|
|
|
"""
|
|
|
|
if name in type(self).params:
|
|
|
|
|
|
|
|
if( self.template.has( name ) ):
|
|
|
|
return self.template.get( name ).value
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
else:
|
|
|
|
raise AttributeError
|
|
|
|
|
|
|
|
def __setattr__( self, name, value ):
|
|
|
|
"""
|
|
|
|
Special setter for template params
|
|
|
|
"""
|
|
|
|
if name in type(self).params:
|
|
|
|
|
|
|
|
self.__dict__[ 'template' ].add( name, value )
|
|
|
|
|
|
|
|
else:
|
|
|
|
object.__setattr__(self, name, value)
|
|
|
|
|
|
|
|
def __ne__( self, other ):
|
|
|
|
"""
|
|
|
|
Checks wether all Template param values except for Liste_Revision are
|
|
|
|
equal
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Detect which of the two was initialised (without)
|
|
|
|
# If none raise error
|
|
|
|
if( self.__initial ):
|
|
|
|
initial = self
|
|
|
|
cmpto = other
|
|
|
|
elif( other.__initial ):
|
|
|
|
initial = other
|
|
|
|
cmpto = self
|
|
|
|
else:
|
|
|
|
raise SummaryPageEntryTemplateError(
|
|
|
|
"One of the compared instances must have been initial!" )
|
|
|
|
|
|
|
|
# Iterate over each param
|
|
|
|
for param in initial.template.params:
|
|
|
|
|
|
|
|
# Slice out only Param.name
|
|
|
|
param = param[:param.find("=")].strip()
|
|
|
|
|
|
|
|
# If param is missing, writing is needed
|
|
|
|
if not cmpto.template.has( param ):
|
|
|
|
return True
|
|
|
|
|
|
|
|
# Do not compare List Revisions (not just write about Revids)
|
|
|
|
if param == "Liste_Revision":
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Compare other param values, if one unequal write is needed
|
|
|
|
if initial.template.get( param ).value.strip() != \
|
|
|
|
cmpto.template.get( param ).value.strip():
|
|
|
|
return True
|
|
|
|
|
|
|
|
# If not returned True until now
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
class SummaryPageError( Exception ):
|
|
|
|
"""
|
|
|
|
Handles errors occuring in class SummaryPage
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
class SummaryPageEntryError( SummaryPageError ):
|
|
|
|
"""
|
|
|
|
Handles errors occuring in class SummaryPageEntry
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
class SummaryPageEntryTemplateError( SummaryPageError ):
|
|
|
|
"""
|
|
|
|
Handles errors occuring in class SummaryPageEntryTemplate
|
|
|
|
"""
|
|
|
|
pass
|