require 'rubygems'
require 'sinatra'
require 'haml'
require './models'
require 'rack-flash'
set :root, File.dirname(__FILE__)
enable :sessions
use Rack::Flash
class String
def pluralize(num)
if num == 1
return self
end
case self[-1]
when 'y'
self[0..-2] + 'ies'
when 's'
self + "es"
else
self + "s"
end
end
end
helpers do
# Format a number with commas for every ^3
def commify(num)
num.to_s.reverse.gsub(/(\d\d\d)(?=\d)(?!\d*\.)/,'\1,').reverse
end
# From http://snippets.dzone.com/posts/show/593
def to_ordinal(num)
num = num.to_i
if (10...20) === num
"#{num}th"
else
g = %w{ th st nd rd th th th th th th }
a = num.to_s
c = a[-1..-1].to_i
a + g[c]
end
end
def format_percent(num)
sprintf("%.0f%%", num)
end
def short_date(d)
d.strftime("%e %b %Y")
end
def long_date(d)
d.strftime("%e %B %Y")
end
# Exception for Labour/Co-operative candidacies
def party_name(labcoop, party_name)
labcoop ? "Labour and Co-operative Party" : party_name
end
end
get '/' do
@election = Election.get(9) # FIXME magic number
@election_title = "#{@election.body.name} #{@election.kind} #{long_date(@election.d)}"
if params[:postcode]
if @p = Postcode.get(params[:postcode].strip.upcase)
# Postcode is valid and in LB Sutton
if @election.body.district_name == 'constituency'
@district = District.get(@p.constituency_id)
else
@district = District.get(@p.ward_id)
end
flash[:notice] = "Postcode #{@postcode} is in #{@district.name} #{@election.body.district_name}"
if @p.polling_station
@ps_postcode = Postcode.get(@p.polling_station.postcode)
@polling_station = "Your polling station is \
%s, %s, %s" \
% [ @ps_postcode.lat, @ps_postcode.lng, @p.polling_station.name, \
@p.polling_station.address, @p.polling_station.postcode]
end
redirect "/bodies/#{@election.body.slug}/elections/#{@election.d}/#{@election.body.districts_name}/#{@district.slug}"
else
flash.now[:error] = "#{@postcode} is not a postcode in Sutton"
end
end
# Display a random postcode as default search term
@random_pc = repository(:default).adapter.select("
SELECT postcode
FROM postcodes
ORDER BY RANDOM()
LIMIT 1
")
@default_pc = @random_pc[0]
@future_elections = Election.future
@past_elections = Election.past
haml :index
end
get '/bodies/:body/elections/:date' do
@body = Body.first(:slug => params[:body])
@election = Election.first(:body => @body, :d => params[:date])
@elections_for_this_body = Election.all(:body => @body, :order => [:d])
@total_seats = Candidacy.sum(:seats, :election => @election)
@total_votes = Candidacy.sum(:votes, :election => @election)
# There's got to be a better way to do this, either with SQL or Datamapper
@total_districts = repository(:default).adapter.select("
SELECT district_id
FROM candidacies
WHERE election_id = ?
GROUP BY district_id
ORDER BY district_id
", @election.id).count
@results_by_party = repository(:default).adapter.select("
SELECT
p.colour,
p.name,
SUM(c.votes) AS votez,
SUM(c.seats) AS seatz,
COUNT(*) AS cands
FROM candidacies c
LEFT JOIN parties p ON p.id = c.party_id
WHERE c.election_id = ?
GROUP BY c.party_id, p.colour, p.name
ORDER BY seatz DESC, votez DESC
", @election.id)
@results_by_district = repository(:default).adapter.select("
SELECT
d.name,
d.slug AS district_slug,
SUM(c.seats) AS seats,
SUM(c.votes) AS votez,
COUNT(c.id) AS num_candidates
FROM districts d, candidacies c
WHERE
c.district_id = d.id
AND c.election_id = ?
GROUP BY c.district_id, d.name, d.slug
ORDER BY d.name
", @election.id)
# For elections that haven't yet been held
@districts_in_this_election = repository(:default).adapter.select("
SELECT DISTINCT d.name, d.slug
FROM candidacies c
LEFT JOIN districts d
ON c.district_id = d.id
WHERE c.election_id = ?
ORDER BY d.name
", @election.id)
haml :electionsummary
end
# get '/bodies/:body/elections/:date/parties/:party' do
# Not written yet. Show how this party did at this election.
# end
get '/bodies/:body/?' do
@body = Body.first(:slug => params[:body])
@districts = District.all(:body => @body, :order => [:name])
@elections = repository(:default).adapter.select("
SELECT
e.id,
e.kind,
e.d,
SUM(p.ballot_papers_issued)::float / SUM(p.electorate) * 100 AS turnout_percent
FROM elections e
LEFT JOIN polls p
ON e.id = p.election_id
WHERE e.body_id = ?
GROUP BY p.election_id, e.id
ORDER BY e.d DESC
", @body.id)
haml :body
end
# get '/wards/:slug/postcode/:postcode/?' do
# @ward = Ward.first(:slug => params[:slug])
# @postcode = params[:postcode]
# haml :wards
# end
get '/candidates/:id/?' do
if @deleted_candidate = DeletedCandidate.get(params[:id])
redirect "/candidates/#{@deleted_candidate.candidate_id}", 302 # HTTP 302 Moved Temporarily
end
if @candidate = Candidate.get(params[:id])
@candidacies = repository(:default).adapter.select("
SELECT
e.d,
c.*,
p.name AS party_name,
p.colour AS party_colour,
b.name AS body_name,
b.slug AS body_slug,
b.districts_name AS districts_name,
d.name AS district_name,
d.slug AS district_slug
FROM candidacies c
INNER JOIN elections e
ON c.election_id = e.id
INNER JOIN parties p
ON c.party_id = p.id
INNER JOIN bodies b
ON e.body_id = b.id
INNER JOIN districts d
ON c.district_id = d.id
WHERE c.candidate_id = ?
ORDER BY d
", @candidate.id)
haml :candidate
else
404
end
end
get '/candidates/?' do
@candidates = Candidate.all(:order => [ :surname, :forenames ])
haml :candidates
end
get '/bodies/:body/elections/:date/:districts_name/:district' do
@district = District.first(:slug => params[:district])
@body = Body.first(:slug => params[:body])
@election = Election.first(:body => @body, :d => params[:date])
@candidacies = Candidacy.all(:district => @district, :election => @election, :order => [:votes.desc])
@total_votes = Candidacy.sum(:votes, :district => @district, :election => @election)
@total_candidates = Candidacy.count(:district => @district, :election => @election)
@total_seats = Candidacy.sum(:seats, :district => @district, :election => @election)
@districts_in_this_election = @election.candidacies.districts
@poll = Poll.get(@district.id, @election.id)
if @total_seats == 1
@share_denominator = @total_votes
elsif @poll && @poll.valid_ballot_papers
@share_denominator = @poll.valid_ballot_papers
else
@share_denominator = @total_votes / @total_seats
@share_message = "The vote share percentages have been estimated as we don't have data for the number of valid ballot papers in this poll."
end
# Postgres: All the columns selected when using GROUP BY must either be aggregate functions or appear in the GROUP BY clause
@results_by_party = repository(:default).adapter.select("
SELECT
p.name AS party_name,
p.colour AS party_colour,
COUNT(c.id) AS num_candidates,
SUM(c.seats) AS num_seats,
SUM(c.votes) AS total_votes
FROM candidacies c
LEFT JOIN parties p
ON c.party_id = p.id
WHERE c.district_id = ?
AND c.election_id = ?
GROUP BY p.name, p.colour
ORDER BY total_votes DESC
", @district.id, @election.id)
haml :resultsdistrict
end
get '/bodies/:body/:districts_name/:district' do
@district = District.first(:slug => params[:district])
@body = Body.first(:slug => params[:body])
haml :district
end
get '/how-the-council-election-works' do
haml :election
end
get '/how-the-parliament-election-works' do
haml :parliament
end
get '/error' do
haml :error
end
get '/about' do
haml :about
end
not_found do
haml :not_found
end