#!/usr/local/bin/perl # # The ticker (client & admin) loosely follows an MVC architecture. The model # is over in Ticker.pm (it does all the DB work). ticker.cgi is the view for # the client, while this is the view+controller for the admin. # use strict; use warnings; use lib '/home/slowtwitch/site/slowtwitch.com/cgi-bin/articles/admin'; use Links qw(:objects); use Ticker; use Error qw(:try); Links::init('/var/home/slowtwitch/site/slowtwitch.com/cgi-bin/articles/admin'); Links::init_admin(); use constant { STATE_NORMAL => 0, STATE_UPDATE => 1, STATE_REDIRECT => 2, ADMIN_URL => "https://www.slowtwitch.com/cgi-bin/articles/ticker/admin/admin.cgi" }; # prototypes. sub controller(); sub view($); # And call the controller. controller(); # # This is the controller. This is where work gets done. # sub controller() { my %viewstate = (state => STATE_NORMAL); # if we have an action parameter (that's non-empty), then do work: if(defined $IN->param('action') && length($IN->param('action')) > 0) { $viewstate{state} = STATE_REDIRECT; try { my $p = $IN->get_hash(); if($p->{action} =~ /create/) { Ticker::create_ticker($p->{msg}, $p->{link}); } elsif($p->{action} =~ /show_update/) { $viewstate{state} = STATE_UPDATE; $viewstate{id} = $p->{'id'}; } elsif($p->{action} =~ /do_update/) { Ticker::update_ticker($p->{'id'}, $p->{'msg'}, $p->{'link'}); } elsif($p->{action} =~ /delete/) { Ticker::delete_ticker($p->{'id'}); } } catch Ticker::TickerException with { # oops. something bad happened. $viewstate{error} = "Error: " . shift; # reset the viewstate so that we display the error message. $viewstate{state} = STATE_NORMAL; }; } return view(\%viewstate); } # # Build the view, which takes a single hashref describing how the view should # behave. # sub view($) { my ($state) = @_; my %s = %$state; # If the state is redirect, we're done. if($s{state} == STATE_REDIRECT) { print $IN->redirect(ADMIN_URL); return; } # Now let's actually build the view, depending on our current state: print $IN->header(); print qq{Ticker Admin}; # Try to load all the tickers: try { $s{data} = Ticker::read_tickers(); # Are we also trying to update a record? Yeah, I know - we're # hitting the db again. I wish that fetchall_hashref actually worked... if(defined $s{id}) { my $result = Ticker::read_ticker($s{id}); $s{msg} = $result->{ticker_text}; $s{link} = $result->{ticker_link}; } } catch Ticker::TickerException with { $s{error} .= " Could not read tickers from database!"; }; # Print an error message if we have one: if(defined $s{error}) { print "

" . $s{error} . "

"; } # What should the top form look like? if($s{state} == STATE_NORMAL) { $s{title} = 'Create a ticker:'; $s{submit} = 'Create ticker'; $s{action} = 'create'; $s{id} = ""; $s{msg} = ""; $s{link} = ""; } elsif($s{state} == STATE_UPDATE) { $s{title} = 'Update a ticker:'; $s{submit} = 'Update ticker'; $s{action} = 'do_update'; } # print the form, which is configured for the type of action we're # performing: print qq{

} . $s{title} . qq{

}; # provide a way to get back to the create interface: if($s{action} =~ /update/) { print qq{Create a ticker instead.}; } # Now print the entire list of all tickers. print qq{

Current tickers:

}; # If there are no tickers, say so: if(@{$s{data}} == 0) { print "There are no tickers."; } else { # Print a table showing the ID, delete/edit links, messages, and links: print ""; print ""; foreach my $k (@{$s{data}}) { my $id = $k->{ticker_id}; my $msg = $k->{ticker_text}; my $link = $k->{ticker_link}; print qq{"; } print "
IDMessageLink
$id Delete Update} . $msg . "" . qq{} . $link . "
"; } print qq{}; }