865 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			865 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | #!/usr/bin/env perl | ||
|  | #=============================================================================== | ||
|  | # | ||
|  | #         FILE: reserve_cnews | ||
|  | # | ||
|  | #        USAGE: ./reserve_cnews [-from[=DATE]] [-count=COUNT] [-[no]dry-run] | ||
|  | #                       [-[no]silent] [-config=FILE] [-help] [-debug=N] | ||
|  | # | ||
|  | #  DESCRIPTION: Reserve a series of slots from a given date for the Community | ||
|  | #               News shows by computing the dates for the reservations and | ||
|  | #               then working out the show numbers from there. | ||
|  | # | ||
|  | #      OPTIONS: --- | ||
|  | # REQUIREMENTS: --- | ||
|  | #         BUGS: --- | ||
|  | #        NOTES: --- | ||
|  | #       AUTHOR: Dave Morriss (djm), Dave.Morriss@gmail.com | ||
|  | #      VERSION: 0.0.14 | ||
|  | #      CREATED: 2014-04-29 22:16:00 | ||
|  | #     REVISION: 2023-04-10 16:05:36 | ||
|  | # | ||
|  | #=============================================================================== | ||
|  | 
 | ||
|  | use 5.010; | ||
|  | use strict; | ||
|  | use warnings; | ||
|  | use utf8; | ||
|  | 
 | ||
|  | use Getopt::Long; | ||
|  | use Pod::Usage; | ||
|  | 
 | ||
|  | use Config::General; | ||
|  | 
 | ||
|  | use Date::Parse; | ||
|  | use Date::Calc qw{:all}; | ||
|  | 
 | ||
|  | use DBI; | ||
|  | 
 | ||
|  | use Data::Dumper; | ||
|  | 
 | ||
|  | # | ||
|  | # Version number (manually incremented) | ||
|  | # | ||
|  | our $VERSION = '0.0.14'; | ||
|  | 
 | ||
|  | # | ||
|  | # Script name | ||
|  | # | ||
|  | ( my $PROG = $0 ) =~ s|.*/||mx; | ||
|  | ( my $DIR  = $0 ) =~ s|/?[^/]*$||mx; | ||
|  | $DIR = '.' unless $DIR; | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Declarations | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # | ||
|  | # Constants and other declarations | ||
|  | # | ||
|  | my $basedir    = "$ENV{HOME}/HPR/Community_News"; | ||
|  | my $configfile = "$basedir/.hpr_db.cfg"; | ||
|  | 
 | ||
|  | my $hostname   = 'HPR Volunteers'; | ||
|  | my $seriesname = 'HPR Community News'; | ||
|  | my $tags       = 'Community News'; | ||
|  | 
 | ||
|  | my $titlefmt   = 'HPR Community News for %s %d'; | ||
|  | my $summaryfmt = 'HPR Volunteers talk about shows released and comments ' | ||
|  |     . 'posted in %s %d'; | ||
|  | 
 | ||
|  | my ( $dbh, $sth1, $sth2, $sth3, $h1, $h2, $rv ); | ||
|  | my (@startdate, @rdate,  @lastmonth, $show, | ||
|  |     $hostid,    $series, $title,     $summary | ||
|  | ); | ||
|  | 
 | ||
|  | # | ||
|  | # Enable Unicode mode | ||
|  | # | ||
|  | binmode STDOUT, ":encoding(UTF-8)"; | ||
|  | binmode STDERR, ":encoding(UTF-8)"; | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Options and arguments | ||
|  | #------------------------------------------------------------------------------- | ||
|  | my $DEFDEBUG  = 0; | ||
|  | my $DEF_COUNT = 12; | ||
|  | 
 | ||
|  | # | ||
|  | # Process options | ||
|  | # | ||
|  | my %options; | ||
|  | Options( \%options ); | ||
|  | 
 | ||
|  | # | ||
|  | # Default help | ||
|  | # | ||
|  | pod2usage( -msg => "$PROG version $VERSION\n", -exitval => 1 ) | ||
|  |     if ( $options{'help'} ); | ||
|  | 
 | ||
|  | # | ||
|  | # Collect options | ||
|  | # | ||
|  | my $DEBUG = ( $options{'debug'} ? $options{'debug'} : $DEFDEBUG ); | ||
|  | my $cfgfile | ||
|  |     = ( defined( $options{config} ) ? $options{config} : $configfile ); | ||
|  | my $dry_run = ( defined( $options{'dry-run'} ) ? $options{'dry-run'} : 0 ); | ||
|  | my $silent  = ( defined( $options{silent} )    ? $options{silent}    : 0 ); | ||
|  | my $count = ( defined( $options{count} ) ? $options{count} : $DEF_COUNT ); | ||
|  | my $from = $options{from}; | ||
|  | 
 | ||
|  | _debug( $DEBUG >= 1, 'Host name: ' . $hostname ); | ||
|  | _debug( $DEBUG >= 1, 'Series name: ' . $seriesname ); | ||
|  | _debug( $DEBUG >= 1, 'Tags: ' . $tags ); | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Configuration file - load data | ||
|  | #------------------------------------------------------------------------------- | ||
|  | my $conf = new Config::General( | ||
|  |     -ConfigFile      => $cfgfile, | ||
|  |     -InterPolateVars => 1, | ||
|  |     -ExtendedAccess  => 1 | ||
|  | ); | ||
|  | my %config = $conf->getall(); | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Connect to the database | ||
|  | #------------------------------------------------------------------------------- | ||
|  | my $dbhost = $config{database}->{host} // '127.0.0.1'; | ||
|  | my $dbport = $config{database}->{port} // 3306; | ||
|  | my $dbname = $config{database}->{name}; | ||
|  | my $dbuser = $config{database}->{user}; | ||
|  | my $dbpwd  = $config{database}->{password}; | ||
|  | $dbh = DBI->connect( "dbi:mysql:host=$dbhost;port=$dbport;database=$dbname", | ||
|  |     $dbuser, $dbpwd, { AutoCommit => 1 } ) | ||
|  |     or die $DBI::errstr; | ||
|  | 
 | ||
|  | # | ||
|  | # Enable client-side UTF8 | ||
|  | # | ||
|  | $dbh->{mysql_enable_utf8} = 1; | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Find the latest show for reference purposes | ||
|  | #------------------------------------------------------------------------------- | ||
|  | $sth1 = $dbh->prepare( | ||
|  | #    q{SELECT id, date FROM eps | ||
|  | #        WHERE DATEDIFF(date,CURDATE()) <= 0 AND DATEDIFF(date,CURDATE()) >= -2 | ||
|  | #        ORDER BY date DESC LIMIT 1} | ||
|  |     q{SELECT id, date FROM eps | ||
|  |         WHERE DATEDIFF(date,CURDATE()) BETWEEN -2 AND 0 | ||
|  |         ORDER BY date DESC LIMIT 1} | ||
|  | ); | ||
|  | $sth1->execute; | ||
|  | if ( $dbh->err ) { | ||
|  |     warn $dbh->errstr; | ||
|  | } | ||
|  | 
 | ||
|  | $h1 = $sth1->fetchrow_hashref; | ||
|  | 
 | ||
|  | my $ref_date = $h1->{date}; | ||
|  | my $ref_show = $h1->{id}; | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Find the required hostid | ||
|  | #------------------------------------------------------------------------------- | ||
|  | $sth1 = $dbh->prepare(q{SELECT hostid FROM hosts WHERE host = ?}); | ||
|  | $sth1->execute($hostname); | ||
|  | if ( $dbh->err ) { | ||
|  |     warn $dbh->errstr; | ||
|  | } | ||
|  | 
 | ||
|  | unless ( $h1 = $sth1->fetchrow_hashref ) { | ||
|  |     warn "Unable to find host '$hostname' - cannot continue\n"; | ||
|  |     exit 1; | ||
|  | } | ||
|  | 
 | ||
|  | $hostid = $h1->{hostid}; | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Find the required series | ||
|  | #------------------------------------------------------------------------------- | ||
|  | $sth1 = $dbh->prepare(q{SELECT id FROM miniseries WHERE name = ?}); | ||
|  | $sth1->execute($seriesname); | ||
|  | if ( $dbh->err ) { | ||
|  |     warn $dbh->errstr; | ||
|  | } | ||
|  | 
 | ||
|  | unless ( $h1 = $sth1->fetchrow_hashref ) { | ||
|  |     warn "Unable to find series '$seriesname' - cannot continue\n"; | ||
|  |     exit 1; | ||
|  | } | ||
|  | 
 | ||
|  | $series = $h1->{id}; | ||
|  | 
 | ||
|  | _debug( $DEBUG >= 2, 'Reference date: ' . $ref_date ); | ||
|  | _debug( $DEBUG >= 2, 'Reference show: ' . $ref_show ); | ||
|  | _debug( $DEBUG >= 2, 'Host id: ' . $hostid ); | ||
|  | _debug( $DEBUG >= 2, 'Series id: ' . $series ); | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # The start date comes from the -from=DATE option, the database or is defaulted | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # | ||
|  | # Use the date provided or the default | ||
|  | # | ||
|  | if ( ! defined( $from ) ) { | ||
|  |     # | ||
|  |     # Compute the first of the current month | ||
|  |     # | ||
|  |     _debug($DEBUG >= 3, "From date: Default"); | ||
|  |     @startdate = ( ( Today() )[ 0 .. 1 ], 1 ); | ||
|  | } | ||
|  | elsif ( $from =~ /^$/ ) { | ||
|  |     _debug($DEBUG >= 3, "From date: Database"); | ||
|  |     @startdate = get_next_date( $dbh, $series ); | ||
|  | } | ||
|  | else { | ||
|  |     # | ||
|  |     # Parse the date, convert to start of month | ||
|  |     # | ||
|  |     _debug($DEBUG >= 3, "From date: Explicit"); | ||
|  |     @startdate = convert_date( $from, 0 ); | ||
|  | } | ||
|  | _debug($DEBUG >= 3,"Start date: " . ISO8601_Date(@startdate)); | ||
|  | 
 | ||
|  | #------------------------------------------------------------------------------- | ||
|  | # Set up for date manipulation | ||
|  | #------------------------------------------------------------------------------- | ||
|  | my @cdate  = @startdate; | ||
|  | my $monday = 1;            # Day of week number 1-7, Monday-Sunday | ||
|  | print "Start date: ", ISO8601_Date(@startdate), "\n" unless ($silent); | ||
|  | 
 | ||
|  | # | ||
|  | # The reference show, taken from the database | ||
|  | # | ||
|  | my @ref_date = split( /-/, $ref_date ); | ||
|  | print "Reference show: hpr$ref_show on ", ISO8601_Date(@ref_date), "\n\n" | ||
|  |     unless ($silent); | ||
|  | 
 | ||
|  | # | ||
|  | # Prepare some SQL (Note stopgap fix for the INSERT statement associated with $sth3) | ||
|  | # | ||
|  | $sth1 = $dbh->prepare(q{SELECT id FROM eps where id = ?}); | ||
|  | $sth2 = $dbh->prepare(q{SELECT id, date FROM eps where title = ?}); | ||
|  | $sth3 = $dbh->prepare( | ||
|  |     q{ | ||
|  |     INSERT INTO eps (id,date,hostid,title,summary,series,tags, | ||
|  |         duration,notes,downloads) | ||
|  |     VALUES(?,?,?,?,?,?,?,0,'',0) | ||
|  | } | ||
|  | ); | ||
|  | 
 | ||
|  | # | ||
|  | # Compute a series of dates from the start date | ||
|  | # | ||
|  | for my $i ( 1 .. $count ) { | ||
|  |     # | ||
|  |     # Determine the next first Monday of the month and the show number that | ||
|  |     # goes with it | ||
|  |     # | ||
|  |     @rdate = make_date( \@cdate, $monday, 1, 0 ); | ||
|  |     $show = $ref_show + Delta_Business_Days( @ref_date, @rdate ); | ||
|  |     _debug($DEBUG >= 3,"Date: " . ISO8601_Date(@rdate) . " Show: $show"); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Make the text strings for this month | ||
|  |     # | ||
|  |     @lastmonth = Add_Delta_YM( @rdate, 0, -1 ); | ||
|  |     $title | ||
|  |         = sprintf( $titlefmt, Month_to_Text( $lastmonth[1] ), $lastmonth[0] ); | ||
|  |     $summary | ||
|  |         = sprintf( $summaryfmt, Month_to_Text( $lastmonth[1] ), | ||
|  |         $lastmonth[0] ); | ||
|  |     _debug($DEBUG >= 3,"Title: $title"); | ||
|  |     _debug($DEBUG >= 3,"Summary: $summary"); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Do we already have a show with this title? | ||
|  |     # | ||
|  |     $rv = $sth2->execute($title); | ||
|  |     if ( $dbh->err ) { | ||
|  |         warn $dbh->errstr; | ||
|  |     } | ||
|  |     if ( $rv > 0 ) { | ||
|  |         $h2 = $sth2->fetchrow_hashref; | ||
|  |         unless ($silent) { | ||
|  |             printf | ||
|  |                 "Skipping; an episode already exists with title '%s' (hpr%s, %s)\n", | ||
|  |                 $title, $h2->{id}, $h2->{date}; | ||
|  |         } | ||
|  |         @cdate = Add_Delta_YM( @cdate, 0, 1 ); | ||
|  |         next; | ||
|  |     } | ||
|  | 
 | ||
|  |     # | ||
|  |     # Is this show number taken? | ||
|  |     # | ||
|  |     $rv = $sth1->execute($show); | ||
|  |     if ( $dbh->err ) { | ||
|  |         warn $dbh->errstr; | ||
|  |     } | ||
|  |     if ( $rv > 0 ) { | ||
|  |         # | ||
|  |         # Find a free slot | ||
|  |         # | ||
|  |         print "Slot $show for '$title' is allocated. " unless ($silent); | ||
|  |         until ( $rv == 0 && ( Day_of_Week(@rdate) < 6 ) ) { | ||
|  |             $show++ if ( Day_of_Week(@rdate) < 6 ); | ||
|  |             @rdate = Add_Delta_Days( @rdate, 1 ); | ||
|  |             $rv = $sth1->execute($show); | ||
|  |             if ( $dbh->err ) { | ||
|  |                 warn $dbh->errstr; | ||
|  |             } | ||
|  |         } | ||
|  |         print "Next free slot is $show\n" unless ($silent); | ||
|  |     } | ||
|  | 
 | ||
|  |     # | ||
|  |     # Reserve the slot or pretend to | ||
|  |     # | ||
|  |     unless ($dry_run) { | ||
|  |         $rv = $sth3->execute( $show, ISO8601_Date(@rdate), $hostid, | ||
|  |             $title, $summary, $series, $tags ); | ||
|  |         if ( $dbh->err ) { | ||
|  |             warn $dbh->errstr; | ||
|  |         } | ||
|  |         if ( $rv > 0 ) { | ||
|  |             printf "Reserved show hpr%d on %s for '%s'\n", | ||
|  |                 $show, ISO8601_Date(@rdate), $title | ||
|  |                 unless ($silent); | ||
|  |         } | ||
|  |         else { | ||
|  |             print "Error reserving slot for '$title'\n" unless ($silent); | ||
|  |         } | ||
|  |     } | ||
|  |     else { | ||
|  |         printf "Show hpr%d on %s for '%s' not reserved - dry run\n", | ||
|  |             $show, ISO8601_Date(@rdate), $title | ||
|  |             unless ($silent); | ||
|  |     } | ||
|  | 
 | ||
|  |     @cdate = Add_Delta_YM( @cdate, 0, 1 ); | ||
|  | } | ||
|  | 
 | ||
|  | for my $sth ( $sth1, $sth2, $sth3 ) { | ||
|  |     $sth->finish; | ||
|  | } | ||
|  | 
 | ||
|  | $dbh->disconnect; | ||
|  | 
 | ||
|  | exit; | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: convert_date | ||
|  | #      PURPOSE: Convert a textual date (ideally YYYY-MM-DD) to a Date::Calc | ||
|  | #               date for the start of the given month. | ||
|  | #   PARAMETERS: $textdate       date in text form | ||
|  | #               $force          Boolean defining whether to skip validating | ||
|  | #                               the date | ||
|  | #      RETURNS: The start of the month in the textual date in Date::Calc | ||
|  | #               format | ||
|  | #  DESCRIPTION: Parses the date string and makes a Date::Calc date from the | ||
|  | #               result where the day part is 1. Optionally checks that the | ||
|  | #               date isn't in the past, though $force = 1 ignores this check. | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: Requires Date::Calc and Date::Parse | ||
|  | #               Note the validation 'die' has a non-generic message | ||
|  | #     SEE ALSO: N/A | ||
|  | #=============================================================================== | ||
|  | sub convert_date { | ||
|  |     my ( $textdate, $force ) = @_; | ||
|  | 
 | ||
|  |     my ( @today, @parsed, @startdate ); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Reference date | ||
|  |     # | ||
|  |     @today = Today(); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Parse and perform rudimentary validation on the $textdate date. Function | ||
|  |     # 'strptime' returns "($ss,$mm,$hh,$day,$month,$year,$zone,$century)". | ||
|  |     # | ||
|  |     # The Date::Calc date $startdate[0] gets the returned year or the current | ||
|  |     # year if no year was parsed, $startdate[1] gets the parsed month or the | ||
|  |     # current month if no month was parsed, and $startdate[2] gets a day of 1. | ||
|  |     # | ||
|  |     @parsed    = strptime($textdate); | ||
|  |     die "Unable to parse date '$textdate'\n" unless @parsed; | ||
|  | 
 | ||
|  |     @startdate = ( | ||
|  |         ( defined( $parsed[5] ) ? $parsed[5] + 1900 : $today[0] ),    # year | ||
|  |         ( defined( $parsed[4] ) ? $parsed[4] + 1 : $today[1] ), 1 | ||
|  |     ); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Unless we've overridden the check there should be a positive or zero | ||
|  |     # difference in days between the target date and today's date to prevent | ||
|  |     # going backwards in time. | ||
|  |     # | ||
|  |     unless ($force) { | ||
|  |         unless ( Delta_Days( @today[ 0, 1 ], 1, @startdate ) ge 0 ) { | ||
|  |             warn "Invalid date $textdate (in the past)\n"; | ||
|  |             die "Use -force to create a back-dated calendar\n"; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     return @startdate; | ||
|  | 
 | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: get_next_date | ||
|  | #      PURPOSE: Find the next unused date from the database | ||
|  | #   PARAMETERS: $dbh            Database handle | ||
|  | #               $series         The id of the Community News series (from | ||
|  | #                               a previous query) | ||
|  | #      RETURNS: The start of the month of the next free date in Date::Calc | ||
|  | #               format | ||
|  | #  DESCRIPTION: Finds the latest reservation in the database. Uses the date | ||
|  | #               associated with this reservation, converts to Date::Calc | ||
|  | #               format, adds a month to it and ensures it's the first Monday | ||
|  | #               of that month (in case a non-standard reservation had been | ||
|  | #               made) | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: TODO: do we need the show number of the latest reservation? | ||
|  | #     SEE ALSO: N/A | ||
|  | #=============================================================================== | ||
|  | sub get_next_date { | ||
|  |     my ( $dbh, $series ) = @_; | ||
|  | 
 | ||
|  |     my ( $sth, $h ); | ||
|  |     my ( $id, $lastdate, @startdate ); | ||
|  | 
 | ||
|  |     # | ||
|  |     # Find the last reservation in the database | ||
|  |     # | ||
|  |     $sth = $dbh->prepare( q{ | ||
|  |         SELECT id, date | ||
|  |         FROM eps WHERE series = ? | ||
|  |         ORDER BY id DESC LIMIT 1; | ||
|  |     } | ||
|  |     ); | ||
|  |     $sth->execute($series); | ||
|  |     if ( $dbh->err ) { | ||
|  |         warn $dbh->errstr; | ||
|  |     } | ||
|  | 
 | ||
|  |     # | ||
|  |     # Get the values returned | ||
|  |     # | ||
|  |     $h        = $sth->fetchrow_hashref; | ||
|  |     $id       = $h->{id}; | ||
|  |     $lastdate = $h->{date}; | ||
|  | 
 | ||
|  |     # | ||
|  |     # Convert the date to Date::Calc format, increment by a month and ensure | ||
|  |     # it's the first Monday of the month (in case the last reservation is not | ||
|  |     # on the right day for some reason - such as the day being reserved by | ||
|  |     # some other mechanism) | ||
|  |     # | ||
|  |     @startdate = convert_date( $lastdate, 0 ); | ||
|  |     @startdate = Add_Delta_YM( @startdate, 0, 1 ); | ||
|  |     @startdate = make_date( \@startdate, 1, 1, 0 ); | ||
|  | 
 | ||
|  |     return @startdate; | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: make_date | ||
|  | #      PURPOSE: Make the event date for recurrence | ||
|  | #   PARAMETERS: $refdate | ||
|  | #                       An arrayref to the reference date array (usually | ||
|  | #                       today's date) | ||
|  | #               $dow    Day of week for the event date (1-7, 1=Monday) | ||
|  | #               $n      The nth day of the week in the given month required | ||
|  | #                       for the event date ($dow=1, $n=1 means first Monday) | ||
|  | #               $offset Number of days to offset the computed date | ||
|  | #      RETURNS: The resulting date as a list for Date::Calc | ||
|  | #  DESCRIPTION: We want to compute a simple date with an offset, such as | ||
|  | #               "the Saturday before the first Monday of the month". We do | ||
|  | #               this by computing a pre-offset date (first Monday of month) | ||
|  | #               then apply the offset (Saturday before). | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: TODO Needs more testing to be considered truly universal | ||
|  | #     SEE ALSO: | ||
|  | #=============================================================================== | ||
|  | sub make_date { | ||
|  |     my ( $refdate, $dow, $n, $offset ) = @_; | ||
|  | 
 | ||
|  |     # | ||
|  |     # Compute the required date: the "$n"th day of week "$dow" in the year and | ||
|  |     # month in @$refdate. This could be a date in the past. | ||
|  |     # | ||
|  |     my @date = Nth_Weekday_of_Month_Year( @$refdate[ 0, 1 ], $dow, $n ); | ||
|  | 
 | ||
|  |     # | ||
|  |     # If the computed date plus the offset is before the base date advance | ||
|  |     # a month | ||
|  |     # | ||
|  |     if ( Day_of_Year(@date) + $offset < Day_of_Year(@$refdate) ) { | ||
|  |         # | ||
|  |         # Add a month and recompute | ||
|  |         # | ||
|  |         @date = Add_Delta_YM( @date, 0, 1 ); | ||
|  |         @date = Nth_Weekday_of_Month_Year( @date[ 0, 1 ], $dow, $n ); | ||
|  |     } | ||
|  | 
 | ||
|  |     # | ||
|  |     # Apply the day offset | ||
|  |     # | ||
|  |     @date = Add_Delta_Days( @date, $offset ) if $offset; | ||
|  | 
 | ||
|  |     # | ||
|  |     # Return a list | ||
|  |     # | ||
|  |     return (@date); | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: Delta_Business_Days | ||
|  | #      PURPOSE: Computes the number of weekdays between two dates | ||
|  | #   PARAMETERS: @date1 - first date in Date::Calc format | ||
|  | #               @date2 - second date in Date::Calc format | ||
|  | #      RETURNS: The business day offset | ||
|  | #  DESCRIPTION: This is a direct copy of the routine of the same name on the | ||
|  | #               Date::Calc manpage. | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: Lifted from the manpage for Date::Calc | ||
|  | #     SEE ALSO: N/A | ||
|  | #=============================================================================== | ||
|  | sub Delta_Business_Days { | ||
|  |     my (@date1) = (@_)[ 0, 1, 2 ]; | ||
|  |     my (@date2) = (@_)[ 3, 4, 5 ]; | ||
|  |     my ( $minus, $result, $dow1, $dow2, $diff, $temp ); | ||
|  | 
 | ||
|  |     $minus = 0; | ||
|  |     $result = Delta_Days( @date1, @date2 ); | ||
|  |     if ( $result != 0 ) { | ||
|  |         if ( $result < 0 ) { | ||
|  |             $minus  = 1; | ||
|  |             $result = -$result; | ||
|  |             $dow1   = Day_of_Week(@date2); | ||
|  |             $dow2   = Day_of_Week(@date1); | ||
|  |         } | ||
|  |         else { | ||
|  |             $dow1 = Day_of_Week(@date1); | ||
|  |             $dow2 = Day_of_Week(@date2); | ||
|  |         } | ||
|  |         $diff = $dow2 - $dow1; | ||
|  |         $temp = $result; | ||
|  |         if ( $diff != 0 ) { | ||
|  |             if ( $diff < 0 ) { | ||
|  |                 $diff += 7; | ||
|  |             } | ||
|  |             $temp -= $diff; | ||
|  |             $dow1 += $diff; | ||
|  |             if ( $dow1 > 6 ) { | ||
|  |                 $result--; | ||
|  |                 if ( $dow1 > 7 ) { | ||
|  |                     $result--; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |         if ( $temp != 0 ) { | ||
|  |             $temp /= 7; | ||
|  |             $result -= ( $temp << 1 ); | ||
|  |         } | ||
|  |     } | ||
|  |     if   ($minus) { return -$result; } | ||
|  |     else          { return $result; } | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: ISO8601_Date | ||
|  | #      PURPOSE: Format a Date::Calc date in ISO8601 format | ||
|  | #   PARAMETERS: @date   - a date in the Date::Calc format | ||
|  | #      RETURNS: Text string containing a YYYY-MM-DD date | ||
|  | #  DESCRIPTION: Just a convenience to allow a simple call like | ||
|  | #               $str = ISO8601_Date(@date) | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: None | ||
|  | #     SEE ALSO: N/A | ||
|  | #=============================================================================== | ||
|  | sub ISO8601_Date { | ||
|  |     my (@date) = (@_)[ 0, 1, 2 ]; | ||
|  | 
 | ||
|  |     if ( check_date(@date) ) { | ||
|  |         return sprintf( "%04d-%02d-%02d", @date ); | ||
|  |     } | ||
|  |     else { | ||
|  |         return "*Invalid Date*"; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: _debug | ||
|  | #      PURPOSE: Prints debug reports | ||
|  | #   PARAMETERS: $active         Boolean: 1 for print, 0 for no print | ||
|  | #               $message        Message to print | ||
|  | #      RETURNS: Nothing | ||
|  | #  DESCRIPTION: Outputs a message if $active is true. It removes any trailing | ||
|  | #               newline and then adds one in the 'print' to the caller doesn't | ||
|  | #               have to bother. Prepends the message with 'D> ' to show it's | ||
|  | #               a debug message. | ||
|  | #       THROWS: No exceptions | ||
|  | #     COMMENTS: None | ||
|  | #     SEE ALSO: N/A | ||
|  | #=============================================================================== | ||
|  | sub _debug { | ||
|  |     my ( $active, $message ) = @_; | ||
|  | 
 | ||
|  |     chomp($message); | ||
|  |     print "D> $message\n" if $active; | ||
|  | } | ||
|  | 
 | ||
|  | #===  FUNCTION  ================================================================ | ||
|  | #         NAME: Options | ||
|  | #      PURPOSE: Processes command-line options | ||
|  | #   PARAMETERS: $optref     Hash reference to hold the options | ||
|  | #      RETURNS: Undef | ||
|  | #  DESCRIPTION: | ||
|  | #       THROWS: no exceptions | ||
|  | #     COMMENTS: none | ||
|  | #     SEE ALSO: n/a | ||
|  | #=============================================================================== | ||
|  | sub Options { | ||
|  |     my ($optref) = @_; | ||
|  | 
 | ||
|  |     my @options = ( | ||
|  |         "help",    "debug=i",  "config=s", "from:s", | ||
|  |         "count=i", "dry-run!", "silent!", | ||
|  |     ); | ||
|  | 
 | ||
|  |     if ( !GetOptions( $optref, @options ) ) { | ||
|  |         pod2usage( -msg => "$PROG version $VERSION\n", -exitval => 1 ); | ||
|  |     } | ||
|  | 
 | ||
|  |     return; | ||
|  | } | ||
|  | 
 | ||
|  | __END__ | ||
|  | 
 | ||
|  | #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
|  | #  Application Documentation | ||
|  | #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | ||
|  | #{{{ | ||
|  | 
 | ||
|  | =head1 NAME | ||
|  | 
 | ||
|  | reserve_cnews - reserve Community News shows in the HPR database | ||
|  | 
 | ||
|  | =head1 VERSION | ||
|  | 
 | ||
|  | This documentation refers to B<reserve_cnews> version 0.0.14 | ||
|  | 
 | ||
|  | =head1 USAGE | ||
|  | 
 | ||
|  |     ./reserve_cnews [-help] [-from[=DATE]] [-count=COUNT] | ||
|  |         [-[no]dry-run] [-[no]silent] [-config=FILE] [-debug=N] | ||
|  | 
 | ||
|  |     Examples: | ||
|  | 
 | ||
|  |         ./reserve_cnews -help | ||
|  |         ./reserve_cnews | ||
|  |         ./reserve_cnews -from=1-June-2014 -dry-run | ||
|  |         ./reserve_cnews -from=15-Aug-2015 -count=6 | ||
|  |         ./reserve_cnews -from=2015-12-06 -count=1 -silent | ||
|  |         ./reserve_cnews -from -count=1 | ||
|  |         ./reserve_cnews -from -count=2 -debug=4 | ||
|  |         ./reserve_cnews -config=.hpr_livedb.cfg -from=1-March-2019 -dry-run | ||
|  | 
 | ||
|  | =head1 OPTIONS | ||
|  | 
 | ||
|  | =over 8 | ||
|  | 
 | ||
|  | =item B<-help> | ||
|  | 
 | ||
|  | Prints a brief help message describing the usage of the program, and then exits. | ||
|  | 
 | ||
|  | =item B<-from=DATE> or B<-from> | ||
|  | 
 | ||
|  | This option defines the starting date from which reservations are to be | ||
|  | created. The program ignores the day part, though it must be provided, and | ||
|  | replaces it with the first day of the month. | ||
|  | 
 | ||
|  | The date format should be B<DD-Mon-YYYY> (e.g. 12-Jun-2014), B<DD-MM-YYYY> | ||
|  | (e.g. 12-06-2014) or B<YYYY-MM-DD> (e.g. 2014-06-12). | ||
|  | 
 | ||
|  | If this option is omitted the current date is used. | ||
|  | 
 | ||
|  | If the B<DATE> part is omitted the script will search the database for the | ||
|  | reservation with the latest date and will use it as the starting point to | ||
|  | generate B<-count=COUNT> (or the default 12) reservations. | ||
|  | 
 | ||
|  | =item B<-count=COUNT> | ||
|  | 
 | ||
|  | This option defines the number of slots to reserve. | ||
|  | 
 | ||
|  | If this option is omitted then 12 slots are reserved. | ||
|  | 
 | ||
|  | =item B<-[no]dry-run> | ||
|  | 
 | ||
|  | This option in the form B<-dry-run> causes the program omit the step of adding | ||
|  | reservations to the database. In the form B<-nodry-run> or if omitted, the | ||
|  | program will perform the update(s). | ||
|  | 
 | ||
|  | =item B<-[no]silent> | ||
|  | 
 | ||
|  | This option in the form B<-silent> causes the program omit the reporting of | ||
|  | what it has done. In the form B<-nosilent> or if omitted, the program will | ||
|  | report what it is doing. | ||
|  | 
 | ||
|  | =item B<-config=FILE> | ||
|  | 
 | ||
|  | This option defines a configuration file other than the default | ||
|  | I<.hpr_db.cfg>. The file must be formatted as described below in the section | ||
|  | I<CONFIGURATION AND ENVIRONMENT>. | ||
|  | 
 | ||
|  | =item B<-debug=N> | ||
|  | 
 | ||
|  | Sets the level of debugging. The default is 0: no debugging. | ||
|  | 
 | ||
|  | Values are: | ||
|  | 
 | ||
|  | =over 4 | ||
|  | 
 | ||
|  | =item 1 | ||
|  | 
 | ||
|  | Produces details of some of the built-in values used. | ||
|  | 
 | ||
|  | =item 2 | ||
|  | 
 | ||
|  | Produces any output defined for lower levels as well as details of the values | ||
|  | taken from the database for use when reserving the show(s). | ||
|  | 
 | ||
|  | =item 3 | ||
|  | 
 | ||
|  | Produces any output defined for lower levels as well as: | ||
|  | 
 | ||
|  | =over 4 | ||
|  | 
 | ||
|  | =item . | ||
|  | 
 | ||
|  | Details of how the `-from` date is being interpreted: default, computed from | ||
|  | the database or explicit. The actual date being used is reported. | ||
|  | 
 | ||
|  | =item . | ||
|  | 
 | ||
|  | Details of all dates chosen and their associated sho numbers using the | ||
|  | algorithm "first Monday of the month". | ||
|  | 
 | ||
|  | =item . | ||
|  | 
 | ||
|  | The show title chosen for each reservation is displayed as well as the summary. | ||
|  | 
 | ||
|  | =back | ||
|  | 
 | ||
|  | =back | ||
|  | 
 | ||
|  | =back | ||
|  | 
 | ||
|  | =head1 DESCRIPTION | ||
|  | 
 | ||
|  | Hacker Public Radio produces a Community News show every month. The show is | ||
|  | recorded on the Saturday before the first Monday of the month, and should be | ||
|  | released as soon as possible afterwards. | ||
|  | 
 | ||
|  | This program reserves future slots in the database for upcoming shows. It | ||
|  | computes the date of the first Monday of all of the months in the requested | ||
|  | sequence then determines which show number matches that date. It writes rows | ||
|  | into the I<reservations> table containing the episode number, the host | ||
|  | identifier ('HPR Admins') and the reason for the reservation. | ||
|  | 
 | ||
|  | It is possible that an HPR host has already requested the slot that this | ||
|  | program determines it should reserve. When this happens the program increments | ||
|  | the episode number and checks again, and repeats this process until a free | ||
|  | slot is discovered. | ||
|  | 
 | ||
|  | It is also possible that a reservation has previously been made in the | ||
|  | I<reservations> table. When this case occurs the program ignores this | ||
|  | particular reservation. | ||
|  | 
 | ||
|  | =head1 DIAGNOSTICS | ||
|  | 
 | ||
|  | =over 8 | ||
|  | 
 | ||
|  | =item B<Invalid date ...> | ||
|  | 
 | ||
|  | The date element of the B<-from=DATE> option is not valid. See the description | ||
|  | of this option for details of what formats are acceptable. | ||
|  | 
 | ||
|  | =item B<Various database messages> | ||
|  | 
 | ||
|  | The program can generate warning messages from the database. | ||
|  | 
 | ||
|  | =item B<Unable to find host '...' - cannot continue> | ||
|  | 
 | ||
|  | The script needs to find the id number relating to the host that will be used | ||
|  | for Community News episodes. It does this by looking in the hosts table for | ||
|  | the name "HPR Volunteers". If this cannot be found, perhaps because it has | ||
|  | been changed, then the script cannot continue. The remedy is to change the | ||
|  | variable $hostname to match the new name. | ||
|  | 
 | ||
|  | =item B<Unable to find series '...' - cannot continue> | ||
|  | 
 | ||
|  | The script needs to find the id number relating to the series that will be | ||
|  | used for Community News episodes. It does this by looking in the miniseries | ||
|  | table for the name "HPR Community News". If this cannot be found, perhaps | ||
|  | because it has been changed, then the script cannot continue. The remedy is to | ||
|  | change the variable $seriesname to match the new name. | ||
|  | 
 | ||
|  | =back | ||
|  | 
 | ||
|  | 
 | ||
|  | =head1 CONFIGURATION AND ENVIRONMENT | ||
|  | 
 | ||
|  | The program obtains the credentials it requires for connecting to the HPR | ||
|  | database by loading them from a configuration file. The file is called | ||
|  | B<.hpr_db.cfg> and should contain the following data: | ||
|  | 
 | ||
|  |     <database> | ||
|  |         host = 127.0.0.1 | ||
|  |         port = PORT | ||
|  |         name = DBNAME | ||
|  |         user = USER | ||
|  |         password = PASSWORD | ||
|  |     </database> | ||
|  | 
 | ||
|  | =head1 DEPENDENCIES | ||
|  | 
 | ||
|  |     Config::General | ||
|  |     Data::Dumper | ||
|  |     Date::Calc | ||
|  |     Date::Parse | ||
|  |     DBI | ||
|  |     Getopt::Long | ||
|  |     Pod::Usage | ||
|  | 
 | ||
|  | =head1 BUGS AND LIMITATIONS | ||
|  | 
 | ||
|  | There are no known bugs in this module. | ||
|  | Please report problems to Dave Morriss (Dave.Morriss@gmail.com) | ||
|  | Patches are welcome. | ||
|  | 
 | ||
|  | =head1 AUTHOR | ||
|  | 
 | ||
|  | Dave Morriss (Dave.Morriss@gmail.com) | ||
|  | 
 | ||
|  | =head1 LICENCE AND COPYRIGHT | ||
|  | 
 | ||
|  | Copyright (c) 2014 - 2023 Dave Morriss (Dave.Morriss@gmail.com). All | ||
|  | rights reserved. | ||
|  | 
 | ||
|  | This module is free software; you can redistribute it and/or | ||
|  | modify it under the same terms as Perl itself. See perldoc perlartistic. | ||
|  | 
 | ||
|  | =cut | ||
|  | 
 | ||
|  | #}}} | ||
|  | 
 | ||
|  | # [zo to open fold, zc to close] | ||
|  | 
 | ||
|  | # vim: syntax=perl:ts=8:sw=4:et:ai:tw=78:fo=tcrqn21:fdm=marker | ||
|  | 
 |