diff options
| author | Louise Crow <louise.crow@gmail.com> | 2012-09-20 15:06:32 +0100 | 
|---|---|---|
| committer | Louise Crow <louise.crow@gmail.com> | 2012-09-20 15:06:32 +0100 | 
| commit | add084ee6047d3cae72e3e445135b122d8ba2cc8 (patch) | |
| tree | f9763b73753c44562d6572035ab71b4bae2957a0 | |
| parent | 69103f5e83d538cd3aa9a3ea39cc5736aaec21e8 (diff) | |
| parent | aebcdc082f277b3569d1300ebeda43c503d8aec5 (diff) | |
Merge branch 'release/0.6.6'0.6.6
Conflicts:
  locale/cs/app.po
  locale/es/app.po
69 files changed, 899 insertions, 228 deletions
| @@ -8,6 +8,8 @@  #     echo "rvm use 1.8.7" > .rvmrc  environment_id="ruby-1.8.7-p370" +rubygems_version="1.6.2" +  # Uncomment the following lines if you want to verify rvm version per project  # rvmrc_rvm_version="1.14.10 (stable)" # 1.10.1 seams as a safe start  # eval "$(echo ${rvm_version}.${rvmrc_rvm_version} | awk -F. '{print "[[ "$1*65536+$2*256+$3" -ge "$4*65536+$5*256+$6" ]]"}' )" || { @@ -37,6 +39,10 @@ else    }  fi +if [[ "$(gem --version)" != "${rubygems_version}" ]] ; then +  rvm rubygems ${rubygems_version} +fi +  # If you use bundler, this might be useful to you:  # if [[ -s Gemfile ]] && {  #   ! builtin command -v bundle >/dev/null || diff --git a/Capfile b/Capfile new file mode 100644 index 000000000..6a798eb2b --- /dev/null +++ b/Capfile @@ -0,0 +1,4 @@ +load 'deploy' +# Uncomment if you are using Rails' asset pipeline +    # load 'deploy/assets' +load 'config/deploy' # remove this line to skip loading any of the default tasks
\ No newline at end of file @@ -1,5 +1,5 @@  # Work around bug in Debian Squeeze - see https://github.com/mysociety/alaveteli/pull/297#issuecomment-4101012 -if File.exist? "/etc/debian_version" and File.open("/etc/debian_version").read.strip =~ /^(squeeze|6\.0\.[45])$/ +if File.exist? "/etc/debian_version" and File.open("/etc/debian_version").read.strip =~ /^(squeeze.*|6\.0\.[45])$/      if File.exist? "/lib/libuuid.so.1"          require 'dl'          DL::dlopen('/lib/libuuid.so.1') @@ -11,6 +11,7 @@ gem 'rails', '2.3.14'  gem 'pg'  gem 'fast_gettext', '>= 0.6.0' +gem 'fastercsv', '>=1.5.5'  gem 'gettext_i18n_rails', '>= 0.6.0', :git => "git://github.com/sebbacon/gettext_i18n_rails.git"  gem 'gettext', '>= 1.9.3'  gem 'json', '~> 1.5.1' @@ -25,6 +26,7 @@ gem 'recaptcha', '~> 0.3.1', :require => 'recaptcha/rails'  # :require avoids "already initialized constant" warnings  gem 'rmagick', :require => 'RMagick'  gem 'routing-filter', '~> 0.2.4' +gem 'rake', '~> 0.9.2.2'  gem 'rspec', '~> 1.3.2'  gem 'rspec-rails', '~> 1.3.4'  gem 'ruby-msg', '~> 1.5.0' @@ -35,11 +37,11 @@ gem 'will_paginate', '~> 2.3.11'  gem 'xapian-full-alaveteli', '~> 1.2.9.5'  gem 'xml-simple'  gem 'zip' +gem 'capistrano'  group :test do    gem 'fakeweb'    gem 'rspec-rails', '~> 1.3.4' -  gem 'rake'  end  group :develop do diff --git a/Gemfile.lock b/Gemfile.lock index 0d9d5cc1d..f243ffaad 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -19,11 +19,19 @@ GEM        activesupport (= 2.3.14)      activesupport (2.3.14)      annotate (2.4.0) +    capistrano (2.13.3) +      highline +      net-scp (>= 1.0.0) +      net-sftp (>= 2.0.0) +      net-ssh (>= 2.0.14) +      net-ssh-gateway (>= 1.1.0)      columnize (0.3.6)      fakeweb (1.3.0)      fast_gettext (0.6.8) +    fastercsv (1.5.5)      gettext (2.2.1)        locale +    highline (1.6.13)      json (1.5.4)      linecache (0.46)        rbx-require-relative (> 0.0.4) @@ -32,6 +40,13 @@ GEM      memcache-client (1.8.5)      net-http-local (0.1.2)      net-purge (0.1.0) +    net-scp (1.0.4) +      net-ssh (>= 1.99.1) +    net-sftp (2.0.5) +      net-ssh (>= 2.0.9) +    net-ssh (2.5.2) +    net-ssh-gateway (1.1.0) +      net-ssh (>= 1.99.1)      pg (0.13.2)      rack (1.1.3)      rails (2.3.14) @@ -72,8 +87,10 @@ PLATFORMS  DEPENDENCIES    annotate +  capistrano    fakeweb    fast_gettext (>= 0.6.0) +  fastercsv (>= 1.5.5)    gettext (>= 1.9.3)    gettext_i18n_rails (>= 0.6.0)!    json (~> 1.5.1) @@ -85,7 +102,7 @@ DEPENDENCIES    pg    rack (~> 1.1.0)    rails (= 2.3.14) -  rake +  rake (~> 0.9.2.2)    rdoc (~> 2.4.3)    recaptcha (~> 0.3.1)    rmagick diff --git a/app/controllers/admin_general_controller.rb b/app/controllers/admin_general_controller.rb index 2c961dfc5..7e8498d8a 100644 --- a/app/controllers/admin_general_controller.rb +++ b/app/controllers/admin_general_controller.rb @@ -61,16 +61,59 @@ class AdminGeneralController < AdminController              @events_title = "Events, all time"              date_back_to = Time.now - 1000.years          end -        @events = InfoRequestEvent.find(:all, :order => "created_at desc, id desc", -                :conditions => ["created_at > ? ", date_back_to.getutc]) -        @public_body_history = PublicBody.versioned_class.find(:all, :order => "updated_at desc, id desc", -                :conditions => ["updated_at > ? ", date_back_to.getutc]) -        for pbh in @public_body_history -            pbh.created_at = pbh.updated_at + +        # Get an array of event attributes within the timespan in the format +        # [id, type_of_model, event_timestamp] +        # Note that the relevent date for InfoRequestEvents is creation, but +        # for PublicBodyVersions is update thoughout +        connection = InfoRequestEvent.connection +        timestamps = connection.select_rows("SELECT id,'InfoRequestEvent', +                                                    created_at AS timestamp +                                             FROM info_request_events +                                             WHERE created_at > '#{date_back_to.getutc}' +                                             UNION +                                             SELECT id, 'PublicBodyVersion', +                                                  updated_at AS timestamp +                                             FROM #{PublicBody.versioned_class.table_name} +                                             WHERE updated_at > '#{date_back_to.getutc}' +                                             ORDER by timestamp desc") +        @events = WillPaginate::Collection.create((params[:page] or 1), 100) do |pager| +            # create a hash for each model type being returned +            info_request_event_ids = {} +            public_body_version_ids = {} +            # get the relevant slice from the paginator +            timestamps.slice(pager.offset, pager.per_page).each_with_index do |event, index| +                # for each event in the slice, add an item to the hash for the model type +                # whose key is the model id, and value is the position in the slice +                if event[1] == 'InfoRequestEvent' +                    info_request_event_ids[event[0].to_i] = index +                else +                    public_body_version_ids[event[0].to_i] = index +                end +            end +            # get all the models in the slice, eagerly loading the associations we use in the view +            public_body_versions = PublicBody.versioned_class.find(:all, +                                          :conditions => ['id in (?)', public_body_version_ids.keys], +                                          :include => [ { :public_body => :translations }]) +            info_request_events = InfoRequestEvent.find(:all, +                                          :conditions => ['id in (?)', info_request_event_ids.keys], +                                          :include => [:info_request]) +            @events = [] +            # drop the models into a combined array, ordered by their position in the timestamp slice +            public_body_versions.each do |version| +                @events[public_body_version_ids[version.id]] = [version, version.updated_at] +            end +            info_request_events.each do |event| +                @events[info_request_event_ids[event.id]] = [event, event.created_at] +            end + +            # inject the result array into the paginated collection: +            pager.replace(@events) + +            # set the total entries for the page to the overall number of results +            pager.total_entries = timestamps.size          end -        @events += @public_body_history -        @events.sort! { |a,b| b.created_at <=> a.created_at }      end      def stats diff --git a/app/controllers/admin_request_controller.rb b/app/controllers/admin_request_controller.rb index c5abf8769..7cf23e61e 100644 --- a/app/controllers/admin_request_controller.rb +++ b/app/controllers/admin_request_controller.rb @@ -16,12 +16,25 @@ class AdminRequestController < AdminController      def list          @query = params[:query] -        @info_requests = InfoRequest.paginate :order => "created_at desc", :page => params[:page], :per_page => 100, +        @info_requests = InfoRequest.paginate :order => "created_at desc", +                                              :page => params[:page], +                                              :per_page => 100,              :conditions =>  @query.nil? ? nil : ["lower(title) like lower('%'||?||'%')", @query]      end      def list_old_unclassified -        @info_requests = InfoRequest.find_old_unclassified(:conditions => ["prominence = 'normal'"]) +        @info_requests = WillPaginate::Collection.create((params[:page] or 1), 50) do |pager| +            info_requests = InfoRequest.find_old_unclassified(:conditions => ["prominence = 'normal'"], +                                                              :limit => pager.per_page, +                                                              :offset => pager.offset) +             # inject the result array into the paginated collection: +             pager.replace(info_requests) + +             unless pager.total_entries +               # the pager didn't manage to guess the total count, do it manually +               pager.total_entries = InfoRequest.count_old_unclassified(:conditions => ["prominence = 'normal'"]) +             end +         end      end      def show diff --git a/app/controllers/api_controller.rb b/app/controllers/api_controller.rb index 6c98ebeba..409a432eb 100644 --- a/app/controllers/api_controller.rb +++ b/app/controllers/api_controller.rb @@ -167,17 +167,41 @@ class ApiController < ApplicationController          feed_type = params[:feed_type]          raise PermissionDenied.new("#{@public_body.id} != #{params[:id]}") if @public_body.id != params[:id].to_i -        @events = InfoRequestEvent.find_by_sql([ -          %(select info_request_events.* -            from info_requests -            join info_request_events on info_requests.id = info_request_events.info_request_id -            where info_requests.public_body_id = ? -            and info_request_events.event_type in ( -              'sent', 'followup_sent', 'resent', 'followup_resent' -            ) -            order by info_request_events.created_at desc -          ), @public_body.id -        ]) +        since_date_str = params[:since_date] +        if since_date_str.nil? +            @events = InfoRequestEvent.find_by_sql([ +              %(select info_request_events.* +                from info_requests +                join info_request_events on info_requests.id = info_request_events.info_request_id +                where info_requests.public_body_id = ? +                and info_request_events.event_type in ( +                  'sent', 'followup_sent', 'resent', 'followup_resent' +                ) +                order by info_request_events.created_at desc +              ), @public_body.id +            ]) +        else +            begin +              since_date = Date.strptime(since_date_str, "%Y-%m-%d") +            rescue ArgumentError +              render :json => {"errors" => [ +                      "Parameter since_date must be in format yyyy-mm-dd (not '#{since_date_str}')" ] }, +                  :status => 500 +              return +            end +            @events = InfoRequestEvent.find_by_sql([ +              %(select info_request_events.* +                from info_requests +                join info_request_events on info_requests.id = info_request_events.info_request_id +                where info_requests.public_body_id = ? +                and info_request_events.event_type in ( +                  'sent', 'followup_sent', 'resent', 'followup_resent' +                ) +                and info_request_events.created_at >= ? +                order by info_request_events.created_at desc +              ), @public_body.id, since_date +            ]) +        end          if feed_type == "atom"              render :template => "api/request_events.atom", :layout => false          elsif feed_type == "json" diff --git a/app/controllers/public_body_controller.rb b/app/controllers/public_body_controller.rb index 95d936e54..b8ea82a66 100644 --- a/app/controllers/public_body_controller.rb +++ b/app/controllers/public_body_controller.rb @@ -7,7 +7,7 @@  #  # $Id: public_body_controller.rb,v 1.8 2009-09-14 13:27:00 francis Exp $ -require 'csv' +require 'fastercsv'  class PublicBodyController < ApplicationController      # XXX tidy this up with better error messages, and a more standard infrastructure for the redirect to canonical URL @@ -148,10 +148,10 @@ class PublicBodyController < ApplicationController      end      def list_all_csv -        public_bodies = PublicBody.find(:all, :order => 'url_name') -        report = StringIO.new -        CSV::Writer.generate(report, ',') do |title| -            title << [ +        public_bodies = PublicBody.find(:all, :order => 'url_name', +                                              :include => [:translations, :tags]) +        report = FasterCSV.generate() do |csv| +            csv << [                      'Name',                      'Short name',                      # deliberately not including 'Request email' @@ -164,7 +164,7 @@ class PublicBodyController < ApplicationController                      'Version',              ]              public_bodies.each do |public_body| -                title << [ +                csv << [                      public_body.name,                      public_body.short_name,                      # DO NOT include request_email (we don't want to make it @@ -179,8 +179,7 @@ class PublicBodyController < ApplicationController                  ]              end          end -        report.rewind -        send_data(report.read, :type=> 'text/csv; charset=utf-8; header=present', +        send_data(report, :type=> 'text/csv; charset=utf-8; header=present',                    :filename => 'all-authorities.csv',                    :disposition =>'attachment', :encoding => 'utf8')      end diff --git a/app/controllers/request_controller.rb b/app/controllers/request_controller.rb index 6e983a014..268ecc73a 100644 --- a/app/controllers/request_controller.rb +++ b/app/controllers/request_controller.rb @@ -422,19 +422,19 @@ class RequestController < ApplicationController          old_described_state = @info_request.described_state          @info_request.set_described_state(params[:incoming_message][:described_state]) -        # If you're not the *actual* requester owner. e.g. you are playing the +        # If you're not the *actual* requester. e.g. you are playing the          # classification game, or you're doing this just because you are an          # admin user (not because you also own the request).          if !@info_request.is_actual_owning_user?(authenticated_user) -            # Log what you did, for classification game score purposes. We -            # don't log if you were the requester XXX This is presumably so you -            # don't score for classifying your own requests. Could instead -            # always log and filter at display time. -            @info_request.log_event("status_update", +            # Log the status change by someone other than the requester +            event = @info_request.log_event("status_update",                  { :user_id => authenticated_user.id,                    :old_described_state => old_described_state,                    :described_state => @info_request.described_state,                  }) +            # Create a classification event for league tables +            RequestClassification.create!(:user_id => authenticated_user.id, +                                          :info_request_event_id => event.id)              # Don't give advice on what to do next, as it isn't their request              RequestMailer.deliver_old_unclassified_updated(@info_request) if !@info_request.is_external? diff --git a/app/controllers/request_game_controller.rb b/app/controllers/request_game_controller.rb index 904c44759..f22652dd1 100644 --- a/app/controllers/request_game_controller.rb +++ b/app/controllers/request_game_controller.rb @@ -11,13 +11,12 @@ class RequestGameController < ApplicationController      def play          session[:request_game] = Time.now -        old = InfoRequest.find_old_unclassified(:conditions => ["prominence = 'normal'"]) -        @missing = old.size +        @missing = InfoRequest.count_old_unclassified(:conditions => ["prominence = 'normal'"])          @total = InfoRequest.count          @done = @total - @missing          @percentage = (@done.to_f / @total.to_f * 10000).round / 100.0 -        @requests = old.sort_by{ rand }.slice(0..2) +        @requests = InfoRequest.get_random_old_unclassified(3)          if @missing == 0              flash[:notice] = _('<p>All done! Thank you very much for your help.</p><p>There are <a href="{{helpus_url}}">more things you can do</a> to help {{site_name}}.</p>', @@ -25,12 +24,8 @@ class RequestGameController < ApplicationController                  :site_name => site_name)          end -        @league_table_28_days = InfoRequestEvent.make_league_table( -            [ "event_type = 'status_update' and created_at >= ?", Time.now() - 28.days ] -        )[0..10] -        @league_table_all_time = InfoRequestEvent.make_league_table( -            [ "event_type = 'status_update'"] -        )[0..10] +        @league_table_28_days = RequestClassification.league_table(10, [ "created_at >= ?", Time.now() - 28.days ]) +        @league_table_all_time = RequestClassification.league_table(10)          @play_urls = true      end diff --git a/app/models/info_request.rb b/app/models/info_request.rb index 6f472c290..2e16d0f58 100644 --- a/app/models/info_request.rb +++ b/app/models/info_request.rb @@ -35,7 +35,7 @@ class InfoRequest < ActiveRecord::Base      belongs_to :user      validate :must_be_internal_or_external -    belongs_to :public_body +    belongs_to :public_body, :counter_cache => true      validates_presence_of :public_body_id      has_many :outgoing_messages, :order => 'created_at' @@ -223,7 +223,7 @@ class InfoRequest < ActiveRecord::Base              incoming_message.clear_in_database_caches!          end      end -     +      # For debugging      def InfoRequest.profile_search(query)          t = Time.now.usec @@ -939,26 +939,54 @@ public      # Used to find when event last changed      def InfoRequest.last_event_time_clause(event_type=nil)          event_type_clause = '' -        event_type_clause = " and info_request_events.event_type = '#{event_type}'" if event_type -        "(select created_at from info_request_events where info_request_events.info_request_id = info_requests.id#{event_type_clause} order by created_at desc limit 1)" +        event_type_clause = " AND info_request_events.event_type = '#{event_type}'" if event_type +        "(SELECT created_at +          FROM info_request_events +          WHERE info_request_events.info_request_id = info_requests.id +          #{event_type_clause} +          ORDER BY created_at desc +          LIMIT 1)"      end -    def InfoRequest.find_old_unclassified(extra_params={}) +    def InfoRequest.old_unclassified_params(extra_params, include_last_response_time=false)          last_response_created_at = last_event_time_clause('response')          age = extra_params[:age_in_days] ? extra_params[:age_in_days].days : OLD_AGE_IN_DAYS -        params = {:select => "*, #{last_response_created_at} as last_response_time", -                  :conditions => ["awaiting_description = ? and #{last_response_created_at} < ? and url_title != 'holding_pen' and user_id is not null", -                                 true, Time.now() - age], -                                 :order => "last_response_time"} -        params[:limit] = extra_params[:limit] if extra_params[:limit] -        params[:include] = extra_params[:include] if extra_params[:include] +        params = { :conditions => ["awaiting_description = ? +                                    AND #{last_response_created_at} < ? +                                    AND url_title != 'holding_pen' +                                    AND user_id IS NOT NULL", +                                    true, Time.now() - age] } +        if include_last_response_time +            params[:select] = "*, #{last_response_created_at} AS last_response_time" +            params[:order] = 'last_response_time' +        end +        return params +    end + +    def InfoRequest.count_old_unclassified(extra_params={}) +        params = old_unclassified_params(extra_params) +        count(:all, params) +    end + +    def InfoRequest.get_random_old_unclassified(limit) +        params = old_unclassified_params({}) +        params[:limit] = limit +        params[:order] = "random()" +        find(:all, params) +    end + +    def InfoRequest.find_old_unclassified(extra_params={}) +        params = old_unclassified_params(extra_params, include_last_response_time=true) +        [:limit, :include, :offset].each do |extra| +            params[extra] = extra_params[extra] if extra_params[extra] +        end          if extra_params[:order]              params[:order] = extra_params[:order]              params.delete(:select)          end          if extra_params[:conditions]              condition_string = extra_params[:conditions].shift -            params[:conditions][0] += " and #{condition_string}" +            params[:conditions][0] += " AND #{condition_string}"              params[:conditions] += extra_params[:conditions]          end          find(:all, params) diff --git a/app/models/info_request_event.rb b/app/models/info_request_event.rb index a827d19a4..54d2f5ef7 100644 --- a/app/models/info_request_event.rb +++ b/app/models/info_request_event.rb @@ -43,7 +43,7 @@ class InfoRequestEvent < ActiveRecord::Base           'resent',           'followup_sent',           'followup_resent', -          +           'edit', # title etc. edited (in admin interface)           'edit_outgoing', # outgoing message edited (in admin interface)           'edit_comment', # comment edited (in admin interface) @@ -53,7 +53,7 @@ class InfoRequestEvent < ActiveRecord::Base           'move_request', # changed user or public body (in admin interface)           'hide', # hid a request (in admin interface)           'manual', # you did something in the db by hand -          +           'response',           'comment',           'status_update' @@ -389,24 +389,6 @@ class InfoRequestEvent < ActiveRecord::Base          return TMail::Address.parse(prev_addr).address == TMail::Address.parse(curr_addr).address      end -    # Given a find condition clause, creates a league table of users who made those events. -    # XXX this isn't very generic yet, it is just used for the categorisation game tables. -    def InfoRequestEvent.make_league_table(conditions) -        status_update_events = InfoRequestEvent.find(:all, :conditions => conditions) -        table = Hash.new { |h,k| h[k] = 0 } -        for event in status_update_events -            user_id = event.params[:user_id] -            table[user_id] += 1 -        end -        league_table = [] -        for user_id, count in table -            user = User.find(user_id) -            league_table.push([user, count]) -        end -        league_table.sort! { |a,b| b[1] <=> a[1] } -        return league_table -    end -      def json_for_api(deep, snippet_highlight_proc = nil)          ret = {              :id => self.id, diff --git a/app/models/public_body.rb b/app/models/public_body.rb index 60ecb2781..77da81d4c 100644 --- a/app/models/public_body.rb +++ b/app/models/public_body.rb @@ -45,6 +45,8 @@ class PublicBody < ActiveRecord::Base      has_many :censor_rules, :order => 'created_at desc'      has_tag_string +    before_save :set_api_key, :set_default_publication_scheme +      translates :name, :short_name, :request_email, :url_name, :notes, :first_letter, :publication_scheme @@ -89,13 +91,13 @@ class PublicBody < ActiveRecord::Base          end      end -    def after_initialize +    def set_default_publication_scheme        # Make sure publication_scheme gets the correct default value.        # (This would work automatically, were publication_scheme not a translated attribute)        self.publication_scheme = "" if self.publication_scheme.nil?      end -    def before_save +    def set_api_key        self.api_key = SecureRandom.base64(33) if self.api_key.nil?      end @@ -104,7 +106,7 @@ class PublicBody < ActiveRecord::Base          locale = self.locale || I18n.locale          PublicBody.with_locale(locale) do              found = PublicBody.find(:all, -                                    :conditions => ["public_body_translations.url_name='#{name}'"], +                                    :conditions => ["public_body_translations.url_name=?", name],                                      :joins => :translations,                                      :readonly => false)              # If many bodies are found (usually because the url_name is the same across @@ -184,7 +186,7 @@ class PublicBody < ActiveRecord::Base      end      acts_as_versioned -    self.non_versioned_columns << 'created_at' << 'updated_at' << 'first_letter' << 'api_key' +    self.non_versioned_columns << 'created_at' << 'updated_at' << 'first_letter' << 'api_key' << 'info_requests_count'      class Version          attr_accessor :created_at @@ -549,9 +551,10 @@ class PublicBody < ActiveRecord::Base      def notes_as_html          self.notes      end +      def notes_without_html          # assume notes are reasonably behaved HTML, so just use simple regexp on this -        self.notes.nil? ? '' : self.notes.gsub(/<\/?[^>]*>/, "") +        @notes_without_html ||= (self.notes.nil? ? '' : self.notes.gsub(/<\/?[^>]*>/, ""))      end      def json_for_api diff --git a/app/models/request_classification.rb b/app/models/request_classification.rb new file mode 100644 index 000000000..678b6cd16 --- /dev/null +++ b/app/models/request_classification.rb @@ -0,0 +1,16 @@ +class RequestClassification < ActiveRecord::Base +    belongs_to :user + +    # return classification instances representing the top n +    # users, with a 'cnt' attribute representing the number +    # of classifications the user has made. +    def RequestClassification.league_table(size, conditions=[]) +        find(:all, :select => 'user_id, count(*) as cnt', +                                         :conditions => conditions, +                                         :group => 'user_id', +                                         :order => 'cnt desc', +                                         :limit => size, +                                         :include => :user) +    end + +end
\ No newline at end of file diff --git a/app/views/admin_general/timeline.rhtml b/app/views/admin_general/timeline.rhtml index eecab4823..e84539970 100644 --- a/app/views/admin_general/timeline.rhtml +++ b/app/views/admin_general/timeline.rhtml @@ -11,25 +11,25 @@  | <a href="?all=1">All time</a></p>  <% last_date = nil %> -<% for event in @events %> -    <% if last_date != event.created_at.to_date %> +<% for event, event_at in @events %> +    <% if last_date != event_at.to_date %>          <% if last_date.nil? %>              <p>          <% end %> -        <h3><%= simple_date(event.created_at) %></h3> +        <h3><%= simple_date(event_at) %></h3>          <p>      <% else %>          <br>      <% end %> -    <% last_date = event.created_at.to_date %> -     -    <%= simple_time(event.created_at) %> +    <% last_date = event_at.to_date %> -    <% if event.class.to_s == 'InfoRequestEvent' %> +    <%= simple_time(event_at) %> + +    <% if event.is_a? InfoRequestEvent %>          <%= request_both_links(event.info_request) %>          <% if event.event_type == 'edit' %>              was edited by administrator <strong><%=h event.params[:editor] %></strong>. -            <% for p in ['title', 'prominence', 'described_state', 'awaiting_description']  +            <% for p in ['title', 'prominence', 'described_state', 'awaiting_description']                      if event.params[p.to_sym] != event.params[('old_'+p).to_sym]                          %> Changed <%=p%> from '<%=h event.params[('old_'+p).to_sym]%>' to '<%=h event.params[p.to_sym] %>'. <%                      end @@ -39,7 +39,7 @@              <% outgoing_message = OutgoingMessage.find(event.params[:outgoing_message_id].to_i) %>              had outgoing message edited by administrator <strong><%=h event.params[:editor] %></strong>.              <% if outgoing_message %> -                <% for p in ['body']  +                <% for p in ['body']                          if event.params[p.to_sym] != event.params[('old_'+p).to_sym]                              %> Changed <%=p%> from '<%=h event.params[('old_'+p).to_sym]%>' to '<%=h event.params[p.to_sym] %>'. <%                          end @@ -52,7 +52,7 @@              <% comment = Comment.find(event.params[:comment_id].to_i) %>              had annotation edited by administrator <strong><%=h event.params[:editor] %></strong>.              <% if comment %> -                <% for p in ['body']  +                <% for p in ['body']                          if event.params[p.to_sym] != event.params[('old_'+p).to_sym]                              %> Changed <%=p%> from '<%=h event.params[('old_'+p).to_sym]%>' to '<%=h event.params[p.to_sym] %>'. <%                          end @@ -71,7 +71,7 @@              had incoming message redelivered to another request by administrator <strong><%=h event.params[:editor] %></strong>.           <% elsif event.event_type == 'response' %>              <% incoming_message = event.incoming_message %> -            received  +            received              <%= link_to 'a response', main_url(incoming_message_url(incoming_message)) %>              from <%=h event.info_request.public_body.name %>.          <% elsif event.event_type == 'sent' %> @@ -95,5 +95,5 @@  <% if not @events.empty? %>  </p>  <% end %> - +<%= will_paginate(@events) %> diff --git a/app/views/admin_request/list_old_unclassified.rhtml b/app/views/admin_request/list_old_unclassified.rhtml index f42ed0d43..2e75c2174 100644 --- a/app/views/admin_request/list_old_unclassified.rhtml +++ b/app/views/admin_request/list_old_unclassified.rhtml @@ -6,10 +6,11 @@  <ul>  <% for @request in @info_requests %>      <li> -        <%= request_both_links(@request) %>  +        <%= request_both_links(@request) %>          – <%=simple_date(@request.get_last_response_event.created_at)%>      </li>  <% end %>  </ul> +<%= will_paginate(@info_requests) %> diff --git a/app/views/api/request_events.atom.builder b/app/views/api/request_events.atom.builder index 4f0133051..44759ae7e 100644 --- a/app/views/api/request_events.atom.builder +++ b/app/views/api/request_events.atom.builder @@ -6,7 +6,7 @@ atom_feed("xmlns:alaveteli" => "http://www.alaveteli.org/API/v2/RequestEvents/At          feed.entry(event) do |entry|              request = event.info_request -            entry.published(event.created_at) +            entry.updated(event.created_at.utc.iso8601)              entry.tag!("alaveteli:event_type", event.event_type)              entry.tag!("alaveteli:request_url", main_url(request_url(request)))              entry.title(request.title) diff --git a/app/views/request_game/play.rhtml b/app/views/request_game/play.rhtml index acf6cce87..eedf19ca2 100644 --- a/app/views/request_game/play.rhtml +++ b/app/views/request_game/play.rhtml @@ -7,22 +7,22 @@      </p>      <h2>Top recent players</h2>      <table> -    <% c = 0; for user, count in @league_table_28_days %> +    <% c = 0; for classifications in @league_table_28_days %>          <tr>              <td> <%= c += 1 %>. <td> -            <td> <%= user_link(user) %> </td> -            <td> <%=pluralize(count, 'request').gsub(" ", " ")%> </td> +            <td> <%= user_link(classifications.user) %> </td> +            <td> <%=pluralize(classifications.cnt, 'request').gsub(" ", " ")%> </td>          </tr>      <% end %>      </table>      <h2>All time best players</h2>      <table> -    <% c = 0; for user, count in @league_table_all_time %> +    <% c = 0; for classifications in @league_table_all_time %>          <tr>              <td> <%= c += 1 %>. <td> -            <td> <%= user_link(user) %> </td> -            <td> <%=pluralize(count, 'request').gsub(" ", " ")%> </td> +            <td> <%= user_link(classifications.user) %> </td> +            <td> <%= pluralize(classifications.cnt, 'request').gsub(" ", " ")%> </td>          </tr>      <% end %>      </table> diff --git a/config/.gitignore b/config/.gitignore index 28efb03ec..78d586ea8 100644 --- a/config/.gitignore +++ b/config/.gitignore @@ -1,8 +1,9 @@ -/config.tmp -/general -/general.yml -/database.yml -/rails_env.rb -/logrotate -/memcached.yml -/*.deployed +config.tmp +general +general.yml +database.yml +rails_env.rb +logrotate +memcached.yml +*.deployed +deploy.yml diff --git a/config/deploy.rb b/config/deploy.rb new file mode 100644 index 000000000..888710f83 --- /dev/null +++ b/config/deploy.rb @@ -0,0 +1,69 @@ +require 'bundler/capistrano' + +set :stage, 'staging' unless exists? :stage + +configuration = YAML.load_file('config/deploy.yml')[stage] + +set :application, 'alaveteli' +set :scm, :git +set :deploy_via, :remote_cache +set :repository, configuration['repository'] +set :branch, configuration['branch'] +set :git_enable_submodules, true +set :deploy_to, configuration['deploy_to'] +set :user, configuration['user'] +set :use_sudo, false + +server configuration['server'], :app, :web, :db, :primary => true + +namespace :rake do +  namespace :themes do +    task :install do +      run "cd #{release_path} && bundle exec rake themes:install RAILS_ENV=#{rails_env}" +    end +  end +end + +# Not in the rake namespace because we're also specifying app-specific arguments here +namespace :xapian do +  desc 'Rebuilds the Xapian index as per the ./scripts/rebuild-xapian-index script' +  task :rebuild_index do +    run "cd #{current_path} && bundle exec rake xapian:rebuild_index models='PublicBody User InfoRequestEvent' RAILS_ENV=#{rails_env}" +  end +end + +namespace :deploy do +  desc "Restarting mod_rails with restart.txt" +  task :restart, :roles => :app, :except => { :no_release => true } do +    run "touch #{current_path}/tmp/restart.txt" +  end + +  [:start, :stop].each do |t| +    desc "#{t} task is a no-op with mod_rails" +    task t, :roles => :app do ; end +  end + +  desc 'Link configuration after a code update' +  task :symlink_configuration do +    links = { +      "#{release_path}/config/database.yml" => "#{shared_path}/database.yml", +      "#{release_path}/config/general.yml" => "#{shared_path}/general.yml", +      "#{release_path}/files" => "#{shared_path}/files", +      "#{release_path}/cache" => "#{shared_path}/cache", +      "#{release_path}/vendor/plugins/acts_as_xapian/xapiandbs" => "#{shared_path}/xapiandbs", +      "#{release_path}/public/download" => "#{release_path}/cache/zips/download" +    } + +    # "ln -sf <a> <b>" creates a symbolic link but deletes <b> if it already exists +    run links.map {|a| "ln -sf #{a.last} #{a.first}"}.join(";") +  end + +  after 'deploy:setup' do +    run "mkdir -p #{shared_path}/files" +    run "mkdir -p #{shared_path}/cache" +    run "mkdir -p #{shared_path}/xapiandbs" +  end +end + +after 'deploy:update_code', 'deploy:symlink_configuration' +after 'deploy:update_code', 'rake:themes:install' diff --git a/config/deploy.yml.example b/config/deploy.yml.example new file mode 100644 index 000000000..aea045dff --- /dev/null +++ b/config/deploy.yml.example @@ -0,0 +1,13 @@ +# Site-specific deployment configuration lives in this file +production: +  repository: git://github.com:mysociety/alaveteli.git +  branch: master +  server: www.example.com +  user: deploy +  deploy_to: /srv/www/alaveteli_production +staging: +  repository: git://github.com:mysociety/alaveteli.git +  branch: develop +  server: test.example.com +  user: deploy +  deploy_to: /srv/www/alaveteli_staging diff --git a/config/environment.rb b/config/environment.rb index 6234ae5c1..250d3eed0 100644 --- a/config/environment.rb +++ b/config/environment.rb @@ -32,7 +32,7 @@ require File.join(File.dirname(__FILE__), '../lib/old_rubygems_patch')  # Application version -ALAVETELI_VERSION = '0.6.5.1' +ALAVETELI_VERSION = '0.6.6'  Rails::Initializer.run do |config|    # Load intial mySociety config diff --git a/config/packages b/config/packages index d059d2906..3cffcb0f9 100644 --- a/config/packages +++ b/config/packages @@ -9,7 +9,7 @@ irb  wv  poppler-utils  pdftk (>> 1.41+dfsg-1) | pdftk (<< 1.41+dfsg-1) # that version has a non-functionining uncompress option -gs-gpl +ghostscript  catdoc  links  elinks diff --git a/db/migrate/20120910153022_create_request_classifications.rb b/db/migrate/20120910153022_create_request_classifications.rb new file mode 100644 index 000000000..7c6270c9e --- /dev/null +++ b/db/migrate/20120910153022_create_request_classifications.rb @@ -0,0 +1,14 @@ +class CreateRequestClassifications < ActiveRecord::Migration +  def self.up +    create_table :request_classifications do |t| +      t.integer :user_id +      t.integer :info_request_event_id +      t.timestamps +    end +    add_index :request_classifications, :user_id +  end + +  def self.down +    drop_table :request_classifications +  end +end diff --git a/db/migrate/20120912111713_add_raw_email_index_to_incoming_messages.rb b/db/migrate/20120912111713_add_raw_email_index_to_incoming_messages.rb new file mode 100644 index 000000000..14174935e --- /dev/null +++ b/db/migrate/20120912111713_add_raw_email_index_to_incoming_messages.rb @@ -0,0 +1,9 @@ +class AddRawEmailIndexToIncomingMessages < ActiveRecord::Migration +  def self.up +      add_index :incoming_messages, :raw_email_id +  end + +  def self.down +      remove_index :incoming_messages, :raw_email_id +  end +end diff --git a/db/migrate/20120912112036_add_info_request_id_index_to_exim_logs.rb b/db/migrate/20120912112036_add_info_request_id_index_to_exim_logs.rb new file mode 100644 index 000000000..81e2a7946 --- /dev/null +++ b/db/migrate/20120912112036_add_info_request_id_index_to_exim_logs.rb @@ -0,0 +1,9 @@ +class AddInfoRequestIdIndexToEximLogs < ActiveRecord::Migration +  def self.up +      add_index :exim_logs, :info_request_id +  end + +  def self.down +      remove_index :exim_logs, :info_request_id +  end +end diff --git a/db/migrate/20120912112312_add_info_request_id_index_to_incoming_and_outgoing_messages.rb b/db/migrate/20120912112312_add_info_request_id_index_to_incoming_and_outgoing_messages.rb new file mode 100644 index 000000000..814fa7540 --- /dev/null +++ b/db/migrate/20120912112312_add_info_request_id_index_to_incoming_and_outgoing_messages.rb @@ -0,0 +1,11 @@ +class AddInfoRequestIdIndexToIncomingAndOutgoingMessages < ActiveRecord::Migration +  def self.up +      add_index :incoming_messages, :info_request_id +      add_index :outgoing_messages, :info_request_id +  end + +  def self.down +      remove_index :incoming_messages, :info_request_id +      remove_index :outgoing_messages, :info_request_id +  end +end diff --git a/db/migrate/20120912112655_add_incoming_message_id_index_to_foi_attachments.rb b/db/migrate/20120912112655_add_incoming_message_id_index_to_foi_attachments.rb new file mode 100644 index 000000000..be0bf76c3 --- /dev/null +++ b/db/migrate/20120912112655_add_incoming_message_id_index_to_foi_attachments.rb @@ -0,0 +1,9 @@ +class AddIncomingMessageIdIndexToFoiAttachments < ActiveRecord::Migration +  def self.up +      add_index :foi_attachments, :incoming_message_id +  end + +  def self.down +      remove_index :foi_attachments, :incoming_message_id +  end +end diff --git a/db/migrate/20120912113004_add_indexes_to_info_request_events.rb b/db/migrate/20120912113004_add_indexes_to_info_request_events.rb new file mode 100644 index 000000000..b3780322f --- /dev/null +++ b/db/migrate/20120912113004_add_indexes_to_info_request_events.rb @@ -0,0 +1,13 @@ +class AddIndexesToInfoRequestEvents < ActiveRecord::Migration +  def self.up +      add_index :info_request_events, :incoming_message_id +      add_index :info_request_events, :outgoing_message_id +      add_index :info_request_events, :comment_id +  end + +  def self.down +      remove_index :info_request_events, :incoming_message_id +      remove_index :info_request_events, :outgoing_message_id +      remove_index :info_request_events, :comment_id +  end +end diff --git a/db/migrate/20120912113720_add_public_body_index_to_info_requests.rb b/db/migrate/20120912113720_add_public_body_index_to_info_requests.rb new file mode 100644 index 000000000..a56cad1f9 --- /dev/null +++ b/db/migrate/20120912113720_add_public_body_index_to_info_requests.rb @@ -0,0 +1,9 @@ +class AddPublicBodyIndexToInfoRequests < ActiveRecord::Migration +  def self.up +      add_index :info_requests, :public_body_id +  end + +  def self.down +      remove_index :info_requests, :public_body_id +  end +end diff --git a/db/migrate/20120912114022_add_user_index_to_info_requests.rb b/db/migrate/20120912114022_add_user_index_to_info_requests.rb new file mode 100644 index 000000000..8be51c0c8 --- /dev/null +++ b/db/migrate/20120912114022_add_user_index_to_info_requests.rb @@ -0,0 +1,9 @@ +class AddUserIndexToInfoRequests < ActiveRecord::Migration +  def self.up +      add_index :info_requests, :user_id +  end + +  def self.down +      remove_index :info_requests, :user_id +  end +end diff --git a/db/migrate/20120912170035_add_info_requests_count_to_public_bodies.rb b/db/migrate/20120912170035_add_info_requests_count_to_public_bodies.rb new file mode 100644 index 000000000..d187dcfa5 --- /dev/null +++ b/db/migrate/20120912170035_add_info_requests_count_to_public_bodies.rb @@ -0,0 +1,16 @@ +class AddInfoRequestsCountToPublicBodies < ActiveRecord::Migration +  def self.up +      add_column :public_bodies, :info_requests_count, :integer, :null => false, :default => 0 + +      PublicBody.connection.execute("UPDATE public_bodies +                                     SET info_requests_count = (SELECT COUNT(*) FROM info_requests +                                                                WHERE public_body_id = public_bodies.id);") + + +  end + +  def self.down +      remove_column :public_bodies, :info_requests_count +  end + +end diff --git a/db/migrate/20120913074940_add_incoming_message_index_to_outgoing_messages.rb b/db/migrate/20120913074940_add_incoming_message_index_to_outgoing_messages.rb new file mode 100644 index 000000000..893395f41 --- /dev/null +++ b/db/migrate/20120913074940_add_incoming_message_index_to_outgoing_messages.rb @@ -0,0 +1,9 @@ +class AddIncomingMessageIndexToOutgoingMessages < ActiveRecord::Migration +  def self.up +      add_index :outgoing_messages, :incoming_message_followup_id +  end + +  def self.down +      remove_index :outgoing_messages, :incoming_message_followup_id +  end +end diff --git a/db/migrate/20120913080807_add_info_request_event_index_to_track_things_sent_emails.rb b/db/migrate/20120913080807_add_info_request_event_index_to_track_things_sent_emails.rb new file mode 100644 index 000000000..d119f55b3 --- /dev/null +++ b/db/migrate/20120913080807_add_info_request_event_index_to_track_things_sent_emails.rb @@ -0,0 +1,9 @@ +class AddInfoRequestEventIndexToTrackThingsSentEmails < ActiveRecord::Migration +  def self.up +      add_index :track_things_sent_emails, :info_request_event_id +  end + +  def self.down +      remove_index :track_things_sent_emails, :info_request_event_id +  end +end diff --git a/db/migrate/20120913081136_add_info_request_event_index_to_user_info_request_sent_alerts.rb b/db/migrate/20120913081136_add_info_request_event_index_to_user_info_request_sent_alerts.rb new file mode 100644 index 000000000..aa9f404f7 --- /dev/null +++ b/db/migrate/20120913081136_add_info_request_event_index_to_user_info_request_sent_alerts.rb @@ -0,0 +1,9 @@ +class AddInfoRequestEventIndexToUserInfoRequestSentAlerts < ActiveRecord::Migration +  def self.up +      add_index :user_info_request_sent_alerts, :info_request_event_id +  end + +  def self.down +      remove_index :user_info_request_sent_alerts, :info_request_event_id +  end +end diff --git a/db/migrate/20120913135745_add_updated_at_index_to_public_body_versions.rb b/db/migrate/20120913135745_add_updated_at_index_to_public_body_versions.rb new file mode 100644 index 000000000..6ae58c884 --- /dev/null +++ b/db/migrate/20120913135745_add_updated_at_index_to_public_body_versions.rb @@ -0,0 +1,9 @@ +class AddUpdatedAtIndexToPublicBodyVersions < ActiveRecord::Migration +  def self.up +      add_index :public_body_versions, :updated_at +  end + +  def self.down +      remove_index :public_body_versions, :updated_at +  end +end diff --git a/doc/CHANGES.md b/doc/CHANGES.md index ae9418ef4..15df1dce5 100644 --- a/doc/CHANGES.md +++ b/doc/CHANGES.md @@ -1,3 +1,15 @@ +# Version 0.6.6 +## Highlighted features +* Adds deployment via Capistrano - see DEPLOY.md for details +* Speeds up several admin pages that were slow in large installs + +* [List of issues on github](https://github.com/mysociety/alaveteli/issues?milestone=22&state=closed) + +## Upgrade notes + +* Check out this version and run `rails-post-deploy` as usual.  +* Run `rake temp:populate_request_classifications` to populate the new request_classifications table which is used in generating the request categorisation game league tables and progress widget.   +  # Version 0.6.5  * This is a minor release, to update all documentation and example files to reflect the move of the official repository to http://github.com/mysociety/alaveteli and the alavetelitheme and adminbootstraptheme themes to http://github.com/mysociety/alavetelitheme and http://github.com/mysociety/adminbootstraptheme respectively.   * Some basic versioning has been added for themes. An ALAVETELI_VERSION constant has been added in config/environment.rb. When loading themes, `rails-post-deploy` now looks for a tag on the theme repository in the form 'use-with-alaveteli-0.6.5' that matches the ALAVETELI_VERSION being deployed - if it finds such a tag, the theme will be checked out from that tag, rather than from the HEAD of the theme repository. If no such tag is found, HEAD is used, as before [issue #573](https://github.com/mysociety/alaveteli/issues/573).  diff --git a/doc/DEPLOY.md b/doc/DEPLOY.md new file mode 100644 index 000000000..adeb0e113 --- /dev/null +++ b/doc/DEPLOY.md @@ -0,0 +1,41 @@ +# Deployment + +mySociety uses a custom deployment and buildout system however Capistrano is included as part of Alaveteli as a standard deployment system. + +## Capistrano + +### Set up + +First you need to customise your deployment settings, e.g. the name of the server you're deploying to. This is done by copying the example file `config/deploy.yml.example` to `config/deploy.yml` and editing the settings to suit you. + +TODO: The following instructions could be greatly improved + +These are the general steps required to get your staging server up and running: + +* Install packages from `config/packages` +* Install Postgres and configure a user +* Create a directory to deploy to and make sure your deployment user can write to it +* Run `cap deploy:setup` to create directories, etc. +* Run `cap deploy:update_code` so that we've got a copy of the example config on the server. This process will take a long time installing gems, etc. it will also fail on `rake:themes:install` but that's OK +* SSH to the server, change to the `deploy_to` directory +* `cp releases/[SOME_DATE]/config/general.yml-example shared/general.yml` +* `cp releases/[SOME_DATE]/config/general.yml-example shared/general.yml` +* Edit those files to match your required settings +* Back on your machine run `cap deploy` and it should successfully deploy +* Run the DB migrations `cap deploy:migrate` +* Build the Xapian DB `cap xapian:rebuild_index` +* Configure Apache/Passenger with a DocumentRoot of `your_deploy_to/current/public` +* Phew. Time to admire your work by browsing to the server! + +### Usage + +Ensure you've got a `config/deploy.yml` file with the correct settings for your site. You'll need to share this with everyone in your team that deploys so it might be a good idea to keep the latest version in a [Gist](http://gist.github.com/). + +To deploy to staging just run `cap deploy` but if you want to deploy to production you need to run `cap -S stage=production deploy`. + +For additional usage instructions, see the [Capistrano wiki](https://github.com/capistrano/capistrano/wiki/). + +### TODO + +* Get `cap deploy:setup` to do most of the work described above in the *Set up* section +* Use [Whenever](https://github.com/javan/whenever) to set up cronjobs diff --git a/lib/tasks/temp.rake b/lib/tasks/temp.rake index 9decc13db..e49a84ecb 100644 --- a/lib/tasks/temp.rake +++ b/lib/tasks/temp.rake @@ -1,5 +1,14 @@  namespace :temp do +    desc 'Populate the request_classifications table from info_request_events' +    task :populate_request_classifications => :environment do +        InfoRequestEvent.find_each(:conditions => ["event_type = 'status_update'"]) do |classification| +            RequestClassification.create!(:created_at => classification.created_at, +                                          :user_id => classification.params[:user_id], +                                          :info_request_event_id => classification.id) +        end +    end +      desc "Remove plaintext passwords from post_redirect params"      task :remove_post_redirect_passwords => :environment do          PostRedirect.find_each(:conditions => ['post_params_yaml is not null']) do |post_redirect| diff --git a/lib/tasks/themes.rake b/lib/tasks/themes.rake index 2b1dbb3a9..6eb64b4b0 100644 --- a/lib/tasks/themes.rake +++ b/lib/tasks/themes.rake @@ -9,33 +9,43 @@ namespace :themes do          File.join(plugin_dir, theme_name)      end +    def checkout_tag(version) +        checkout_command = "git checkout #{usage_tag(version)}" +        success = system(checkout_command) +        puts "Using tag #{usage_tag(version)}" if verbose && success +        success +    end + +    def usage_tag(version) +        "use-with-alaveteli-#{version}" +    end +      def install_theme_using_git(name, uri, verbose=false, options={}) -        mkdir_p(install_path = theme_dir(name)) -        Dir.chdir install_path do -            init_cmd = "git init" -            init_cmd += " -q" if options[:quiet] and not verbose -            puts init_cmd if verbose -            system(init_cmd) -            base_cmd = "git pull --depth 1 #{uri}" -            # Is there a tag for this version of Alaveteli? -            usage_tag = "use-with-alaveteli-#{ALAVETELI_VERSION}" -            # Query the remote repository passing flags for tags -            version_tag = `git ls-remote --tags #{uri} #{usage_tag}` -            if /^[a-z0-9]+\s+refs\/tags\/#{Regexp.escape(usage_tag)}$/.match(version_tag) -                # If we got a tag, pull that instead of HEAD -                puts "Using tag #{usage_tag}" if verbose -                base_cmd += " refs/tags/#{usage_tag}" -            else -                puts "No specific tag for this version: using HEAD" if verbose -            end -            base_cmd += " -q" if options[:quiet] and not verbose -            puts base_cmd if verbose -            if system(base_cmd) -                puts "removing: .git .gitignore" if verbose -                rm_rf %w(.git .gitignore) +        install_path = theme_dir(name) +        Dir.chdir(plugin_dir) do +            clone_command = "git clone #{uri} #{name}" +            if system(clone_command) +                Dir.chdir install_path do +                    # try to checkout a tag exactly matching ALAVETELI VERSION +                    tag_checked_out = checkout_tag(ALAVETELI_VERSION) +                    if ! tag_checked_out +                        # if we're on a hotfix release (four sequence elements or more), +                        # look for a usage tag matching the minor release (three sequence elements) +                        # and check that out if found +                        if hotfix_version = /^(\d+\.\d+\.\d+)(\.\d+)+/.match(ALAVETELI_VERSION) +                            base_version = hotfix_version[1] +                            tag_checked_out = checkout_tag(base_version) +                        end +                    end +                    if ! tag_checked_out +                        puts "No specific tag for this version: using HEAD" if verbose +                    end +                    puts "removing: .git .gitignore" if verbose +                    rm_rf %w(.git .gitignore) +                end              else                  rm_rf install_path -                raise "#{base_cmd} failed! Stopping." +                raise "#{clone_command} failed! Stopping."              end          end      end diff --git a/locale/aln/app.po b/locale/aln/app.po index a1b3bf9d6..0b07e2754 100644 --- a/locale/aln/app.po +++ b/locale/aln/app.po @@ -8,8 +8,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 08:21+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1968,6 +1968,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3453,6 +3456,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/app.pot b/locale/app.pot index 20433ad97..d927a0cb7 100644 --- a/locale/app.pot +++ b/locale/app.pot @@ -7,7 +7,7 @@  msgid ""  msgstr ""  "Project-Id-Version: version 0.0.1\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n"  "PO-Revision-Date: 2011-10-09 01:10+0200\n"  "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"  "Language-Team: LANGUAGE <LL@li.org>\n" @@ -1779,6 +1779,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3137,6 +3140,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/bs/app.po b/locale/bs/app.po index c91473e8f..50f53128b 100644 --- a/locale/bs/app.po +++ b/locale/bs/app.po @@ -10,8 +10,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:33+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1974,6 +1974,9 @@ msgstr "Javno tijelo|Početno slovo"  msgid "PublicBody|Home page"  msgstr "Javno tijelo|Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "Javno tijelo|Zadnji uređeni komentar" @@ -3465,6 +3468,9 @@ msgstr "svi zahtjevi"  msgid "also called {{public_body_short_name}}"  msgstr "takođe poznat/a kao {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "i" diff --git a/locale/ca/app.po b/locale/ca/app.po index 1621bec9b..570e8ae50 100644 --- a/locale/ca/app.po +++ b/locale/ca/app.po @@ -10,8 +10,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:31+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1970,6 +1970,9 @@ msgstr "Primera letra"  msgid "PublicBody|Home page"  msgstr "Sitio web" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3455,6 +3458,9 @@ msgstr "todas las solicitudes"  msgid "also called {{public_body_short_name}}"  msgstr "también conocido como {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "y" diff --git a/locale/cs/app.po b/locale/cs/app.po index 316b12a6b..2bd0e61ad 100644 --- a/locale/cs/app.po +++ b/locale/cs/app.po @@ -1,7 +1,7 @@  # SOME DESCRIPTIVE TITLE.  # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER  # This file is distributed under the same license as the PACKAGE package. -#  +#  # Translators:  # Hana Huntova <>, 2012.  # Jana Kneschke <>, 2012. @@ -11,9 +11,9 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-09-04 16:59+0000\n" -"Last-Translator: janakneschke <jana.kneschke@gmail.com>\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n" +"Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n"  "Content-Type: text/plain; charset=UTF-8\n" @@ -1975,6 +1975,9 @@ msgstr "PublicBody | První dopis"  msgid "PublicBody|Home page"  msgstr "PublicBody | Domovská stránka" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody | Naposled aktualizovaný komentář" @@ -3466,6 +3469,9 @@ msgstr "všechny dotazy"  msgid "also called {{public_body_short_name}}"  msgstr "také se nazývá {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "a" diff --git a/locale/cy/app.po b/locale/cy/app.po index 5bda3b075..f129d3685 100644 --- a/locale/cy/app.po +++ b/locale/cy/app.po @@ -3,13 +3,13 @@  # This file is distributed under the same license as the PACKAGE package.  #   # Translators: -#   <alex@alexskene.com>, 2011, 2012. +#   <alex@alexskene.com>, 2011-2012.  msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 08:21+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1976,6 +1976,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3473,6 +3476,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/de/app.po b/locale/de/app.po index 7f1867ab6..a890a48a1 100644 --- a/locale/de/app.po +++ b/locale/de/app.po @@ -11,8 +11,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:31+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1971,6 +1971,9 @@ msgstr "PublicBody|First letter"  msgid "PublicBody|Home page"  msgstr "PublicBody|Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3456,6 +3459,9 @@ msgstr "alle Anfragen"  msgid "also called {{public_body_short_name}}"  msgstr "auch genannt: {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "und" diff --git a/locale/en/app.po b/locale/en/app.po index 8fab8256f..2129ccb8c 100644 --- a/locale/en/app.po +++ b/locale/en/app.po @@ -6,7 +6,7 @@  msgid ""  msgstr ""  "Project-Id-Version: version 0.0.1\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n"  "PO-Revision-Date: 2011-02-24 07:11-0000\n"  "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"  "Language-Team: LANGUAGE <LL@li.org>\n" @@ -1778,6 +1778,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3136,6 +3139,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/en_IE/app.po b/locale/en_IE/app.po index 25909c3ff..dfc576c0b 100644 --- a/locale/en_IE/app.po +++ b/locale/en_IE/app.po @@ -8,8 +8,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 08:21+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1968,6 +1968,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3453,6 +3456,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/es/app.po b/locale/es/app.po index c1df2eff2..aa1e5c650 100644 --- a/locale/es/app.po +++ b/locale/es/app.po @@ -1,9 +1,9 @@  # SOME DESCRIPTIVE TITLE.  # Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER  # This file is distributed under the same license as the PACKAGE package. -#  +#  # Translators: -# David Cabo <david.cabo@gmail.com>, 2011, 2012. +# David Cabo <david.cabo@gmail.com>, 2011-2012.  #   <fabrizio.scrollini@gmail.com>, 2012.  #   <gabelula@gmail.com>, 2012.  # skenaja <alex@alexskene.com>, 2011. @@ -12,9 +12,9 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-09-14 16:51+0000\n" -"Last-Translator: fabrizioscrollini <fabrizio.scrollini@gmail.com>\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n" +"Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n"  "Content-Type: text/plain; charset=UTF-8\n" @@ -1972,6 +1972,9 @@ msgstr "Primera letra"  msgid "PublicBody|Home page"  msgstr "Sitio web" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3457,6 +3460,9 @@ msgstr "todas las solicitudes"  msgid "also called {{public_body_short_name}}"  msgstr "también conocido como {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "y" diff --git a/locale/eu/app.po b/locale/eu/app.po index 63d76124f..6bceee0df 100644 --- a/locale/eu/app.po +++ b/locale/eu/app.po @@ -9,8 +9,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:33+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1969,6 +1969,9 @@ msgstr "Primera letra"  msgid "PublicBody|Home page"  msgstr "Sitio web" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3454,6 +3457,9 @@ msgstr "eskabide guztiak"  msgid "also called {{public_body_short_name}}"  msgstr "{{public_body_short_name}} izenez ere ezaguna." +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "eta" diff --git a/locale/fr/app.po b/locale/fr/app.po index a328641a3..2234e2f4b 100644 --- a/locale/fr/app.po +++ b/locale/fr/app.po @@ -13,8 +13,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 08:21+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1973,6 +1973,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3458,6 +3461,9 @@ msgstr "toutes les demandes"  msgid "also called {{public_body_short_name}}"  msgstr "aussi appelé {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "et" diff --git a/locale/gl/app.po b/locale/gl/app.po index 2ab7d5d82..269505180 100644 --- a/locale/gl/app.po +++ b/locale/gl/app.po @@ -8,8 +8,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:34+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1968,6 +1968,9 @@ msgstr "Primera letra"  msgid "PublicBody|Home page"  msgstr "Sitio web" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3453,6 +3456,9 @@ msgstr "todas las solicitudes"  msgid "also called {{public_body_short_name}}"  msgstr "también conocido como {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "y" diff --git a/locale/hu_HU/app.po b/locale/hu_HU/app.po index aa264991e..286237b84 100644 --- a/locale/hu_HU/app.po +++ b/locale/hu_HU/app.po @@ -8,8 +8,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:32+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1968,6 +1968,9 @@ msgstr "PublicBody|First letter"  msgid "PublicBody|Home page"  msgstr "PublicBody|Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3453,6 +3456,9 @@ msgstr " minden igénylés "  msgid "also called {{public_body_short_name}}"  msgstr "más néven {{public_body_short_name}} " +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "és" diff --git a/locale/id/app.po b/locale/id/app.po index 8e3995eee..dd56d7be6 100644 --- a/locale/id/app.po +++ b/locale/id/app.po @@ -10,8 +10,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:34+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1966,6 +1966,9 @@ msgstr "PublicBody|First letter"  msgid "PublicBody|Home page"  msgstr "PublicBody|Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody|Last edit comment" @@ -3445,6 +3448,9 @@ msgstr "semua permintaan"  msgid "also called {{public_body_short_name}}"  msgstr "juga disebut {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "dan" diff --git a/locale/model_attributes.rb b/locale/model_attributes.rb index bf4cd68da..9944bd11d 100644 --- a/locale/model_attributes.rb +++ b/locale/model_attributes.rb @@ -77,6 +77,7 @@ _('Public body')  _('PublicBody|Api key')  _('PublicBody|First letter')  _('PublicBody|Home page') +_('PublicBody|Info requests count')  _('PublicBody|Last edit comment')  _('PublicBody|Last edit editor')  _('PublicBody|Name') diff --git a/locale/nb_NO/app.po b/locale/nb_NO/app.po index 32c00610b..352e04bfe 100644 --- a/locale/nb_NO/app.po +++ b/locale/nb_NO/app.po @@ -7,7 +7,7 @@ msgid ""  msgstr ""  "Project-Id-Version: version 0.0.1\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n"  "PO-Revision-Date: 2011-03-09 17:48+0000\n"  "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"  "Language-Team: LANGUAGE <LL@li.org>\n" @@ -1779,6 +1779,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3137,6 +3140,9 @@ msgstr ""  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "" diff --git a/locale/pt_BR/app.po b/locale/pt_BR/app.po index 04780b017..82b3a0447 100644 --- a/locale/pt_BR/app.po +++ b/locale/pt_BR/app.po @@ -15,14 +15,14 @@  #   <luis.leao@gmail.com>, 2011.  #   <Nitaibezerra@gmail.com>, 2012.  #   <patriciacornils@gmail.com>, 2011. -#   <pedro@esfera.mobi>, 2011, 2012. +#   <pedro@esfera.mobi>, 2011-2012.  #   <rafael.moretti@gmail.com>, 2012.  msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:30+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1981,6 +1981,9 @@ msgstr "PublicBody | Primeira letra"  msgid "PublicBody|Home page"  msgstr "PublicBody | Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody | Última edição" @@ -3466,6 +3469,9 @@ msgstr "todos os pedidos"  msgid "also called {{public_body_short_name}}"  msgstr "também conhecido como {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "e" diff --git a/locale/sq/app.po b/locale/sq/app.po index 10ece753d..c9174ccdd 100644 --- a/locale/sq/app.po +++ b/locale/sq/app.po @@ -6,15 +6,15 @@  #   <bresta@gmail.com>, 2011.  # driton <dritoni.h@gmail.com>, 2011.  # Hana Huntova <>, 2012. -# Valon  <vbrestovci@gmail.com>, 2011, 2012. +# Valon  <vbrestovci@gmail.com>, 2011-2012.  #   <vbrestovci@gmail.com>, 2011.  # vbrestovci <vbrestovci@gmail.com>, 2011.  msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:30+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1973,6 +1973,9 @@ msgstr "PublicBody |Germa e parë"  msgid "PublicBody|Home page"  msgstr "PublicBody |Ballina" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "PublicBody | Editimi i fundit i komentit" @@ -3458,6 +3461,9 @@ msgstr "të gjitha kërkesat"  msgid "also called {{public_body_short_name}}"  msgstr "i quajtur edhe {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "dhe" diff --git a/locale/sr@latin/app.po b/locale/sr@latin/app.po index a79cadc9d..88291fcbf 100644 --- a/locale/sr@latin/app.po +++ b/locale/sr@latin/app.po @@ -9,8 +9,8 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-24 10:31+0000\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n"  "Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n" @@ -1973,6 +1973,9 @@ msgstr "Javno tijelo|Početno slovo"  msgid "PublicBody|Home page"  msgstr "Javno tijelo|Home page" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "Javno tijelo|Zadnji uređeni komentar" @@ -3464,6 +3467,9 @@ msgstr "svi zahtevi"  msgid "also called {{public_body_short_name}}"  msgstr "takođe poznat/a kao {{public_body_short_name}}" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "i" diff --git a/locale/uk/app.po b/locale/uk/app.po index caac936f5..98490f9ef 100644 --- a/locale/uk/app.po +++ b/locale/uk/app.po @@ -9,9 +9,9 @@ msgid ""  msgstr ""  "Project-Id-Version: alaveteli\n"  "Report-Msgid-Bugs-To: http://github.com/sebbacon/alaveteli/issues\n" -"POT-Creation-Date: 2012-08-24 09:20+0100\n" -"PO-Revision-Date: 2012-08-28 10:57+0000\n" -"Last-Translator: hiiri <murahoid@gmail.com>\n" +"POT-Creation-Date: 2012-09-19 09:37+0100\n" +"PO-Revision-Date: 2012-09-19 08:48+0000\n" +"Last-Translator: louisecrow <louise@mysociety.org>\n"  "Language-Team: LANGUAGE <LL@li.org>\n"  "MIME-Version: 1.0\n"  "Content-Type: text/plain; charset=UTF-8\n" @@ -1973,6 +1973,9 @@ msgstr ""  msgid "PublicBody|Home page"  msgstr "" +msgid "PublicBody|Info requests count" +msgstr "" +  msgid "PublicBody|Last edit comment"  msgstr "" @@ -3464,6 +3467,9 @@ msgstr "усі запити"  msgid "also called {{public_body_short_name}}"  msgstr "" +msgid "an anonymous user" +msgstr "" +  msgid "and"  msgstr "і" diff --git a/spec/controllers/admin_general_controller_spec.rb b/spec/controllers/admin_general_controller_spec.rb index 820d1e7f3..dc1eb0d97 100644 --- a/spec/controllers/admin_general_controller_spec.rb +++ b/spec/controllers/admin_general_controller_spec.rb @@ -1,18 +1,39 @@  require File.expand_path(File.dirname(__FILE__) + '/../spec_helper') -describe AdminGeneralController, "when viewing front page of admin interface" do -    integrate_views -    before { basic_auth_login @request } -   -    it "should render the front page" do -        get :index, :suppress_redirect => 1 -        response.should render_template('index') -    end +describe AdminGeneralController do + +    describe "when viewing front page of admin interface" do + +        integrate_views +        before { basic_auth_login @request } + +        it "should render the front page" do +            get :index, :suppress_redirect => 1 +            response.should render_template('index') +        end + +        it "should redirect to include trailing slash" do +            get :index +            response.should redirect_to(:controller => 'admin_general', +                                        :action => 'index') +        end -    it "should redirect to include trailing slash" do -        get :index -        response.should redirect_to(:controller => 'admin_general', -                                    :action => 'index')      end +    describe 'when viewing the timeline' do + +        it 'should assign an array of events in order of descending date to the view' do +            get :timeline, :all => 1 +            previous_event = nil +            previous_event_at = nil +            assigns[:events].each do |event, event_at| +                if previous_event +                    (event_at <= previous_event_at).should be_true +                end +                previous_event = event +                previous_event_at = event_at +            end +        end + +    end  end diff --git a/spec/controllers/api_controller_spec.rb b/spec/controllers/api_controller_spec.rb index 925b7adb4..ded9a040a 100644 --- a/spec/controllers/api_controller_spec.rb +++ b/spec/controllers/api_controller_spec.rb @@ -320,6 +320,21 @@ describe ApiController, "when using the API" do          assigns[:event_data].should == [first_event]      end +    it "should honour the since_date parameter for the Atom feed" do +        get :body_request_events, +            :id => public_bodies(:humpadink_public_body).id, +            :k => public_bodies(:humpadink_public_body).api_key, +            :since_date => "2010-01-01", +            :feed_type => "atom" +         +        response.should be_success +        response.should render_template("api/request_events.atom") +        assigns[:events].size.should > 0 +        assigns[:events].each do |event| +            event.created_at.should >= Date.new(2010, 1, 1) +        end +    end +          it "should return a JSON 404 error for non-existent requests" do          request_id = 123459876 # Let's hope this doesn't exist!          sent_at = "2012-05-28T12:35:39+01:00" diff --git a/spec/controllers/request_controller_spec.rb b/spec/controllers/request_controller_spec.rb index b8425613c..95737a250 100644 --- a/spec/controllers/request_controller_spec.rb +++ b/spec/controllers/request_controller_spec.rb @@ -1275,7 +1275,8 @@ describe RequestController, "when classifying an information request" do                      expected_params = {:user_id => users(:silly_name_user).id,                                         :old_described_state => 'waiting_response',                                         :described_state => 'rejected'} -                    @dog_request.should_receive(:log_event).with("status_update", expected_params) +                    event = mock_model(InfoRequestEvent) +                    @dog_request.should_receive(:log_event).with("status_update", expected_params).and_return(event)                      post_status('rejected')                  end @@ -1314,10 +1315,19 @@ describe RequestController, "when classifying an information request" do              end              it 'should log a status update event' do +                event = mock_model(InfoRequestEvent)                  expected_params = {:user_id => @admin_user.id,                                     :old_described_state => 'waiting_response',                                     :described_state => 'rejected'} -                @dog_request.should_receive(:log_event).with("status_update", expected_params) +                @dog_request.should_receive(:log_event).with("status_update", expected_params).and_return(event) +                post_status('rejected') +            end + +            it 'should record a classification' do +                event = mock_model(InfoRequestEvent) +                @dog_request.stub!(:log_event).with("status_update", anything()).and_return(event) +                RequestClassification.should_receive(:create!).with(:user_id => @admin_user.id, +                                                                    :info_request_event_id => event.id)                  post_status('rejected')              end diff --git a/spec/fixtures/public_bodies.yml b/spec/fixtures/public_bodies.yml index 367e0fc50..615c4bcb6 100644 --- a/spec/fixtures/public_bodies.yml +++ b/spec/fixtures/public_bodies.yml @@ -1,4 +1,4 @@ -geraldine_public_body:  +geraldine_public_body:    name: The Geraldine Quango    first_letter: T    updated_at: 2007-10-24 10:51:01.161639 @@ -11,7 +11,8 @@ geraldine_public_body:    url_name: tgq    created_at: 2007-10-24 10:51:01.161639    api_key: 1 -humpadink_public_body:  +  info_requests_count: 4 +humpadink_public_body:    name: "Department for Humpadinking"    first_letter: D    updated_at: 2007-10-25 10:51:01.161639 @@ -25,6 +26,7 @@ humpadink_public_body:    created_at: 2007-10-25 10:51:01.161639    notes: An albatross told me!!!    api_key: 2 +  info_requests_count: 2  forlorn_public_body:    name: "Department of Loneliness"    first_letter: D @@ -39,7 +41,8 @@ forlorn_public_body:    created_at: 2011-01-26 14:11:02.12345    notes: A very lonely public body that no one has corresponded with    api_key: 3 -silly_walks_public_body:  +  info_requests_count: 0 +silly_walks_public_body:    id: 5    version: 1    name: "Ministry of Silly Walks" @@ -53,7 +56,8 @@ silly_walks_public_body:    created_at: 2007-10-25 10:51:01.161639    notes: You know the one.    api_key: 4 -sensible_walks_public_body:  +  info_requests_count: 2 +sensible_walks_public_body:    id: 6    version: 1    name: "Ministry of Sensible Walks" @@ -67,4 +71,4 @@ sensible_walks_public_body:    last_edit_editor: robin    created_at: 2008-10-25 10:51:01.161639    api_key: 5 - +  info_requests_count: 1 diff --git a/spec/fixtures/public_body_translations.yml b/spec/fixtures/public_body_translations.yml index cbb55bb0c..f3453e853 100644 --- a/spec/fixtures/public_body_translations.yml +++ b/spec/fixtures/public_body_translations.yml @@ -1,4 +1,4 @@ -geraldine_es_public_body_translation:  +geraldine_es_public_body_translation:    name: El A Geraldine Quango    first_letter: E    request_email: geraldine-requests@localhost @@ -8,8 +8,9 @@ geraldine_es_public_body_translation:    url_name: etgq    locale: es    notes: "" +  publication_scheme: "" -geraldine_en_public_body_translation:  +geraldine_en_public_body_translation:    name: Geraldine Quango    first_letter: G    request_email: geraldine-requests@localhost @@ -19,8 +20,9 @@ geraldine_en_public_body_translation:    url_name: tgq    locale: en    notes: "" +  publication_scheme: "" -humpadink_es_public_body_translation:  +humpadink_es_public_body_translation:    name: "El Department for Humpadinking"    first_letter: E    request_email: humpadink-requests@localhost @@ -30,8 +32,9 @@ humpadink_es_public_body_translation:    url_name: edfh    locale: es    notes: Baguette +  publication_scheme: "" -humpadink_en_public_body_translation:  +humpadink_en_public_body_translation:    name: "Department for Humpadinking"    first_letter: D    request_email: humpadink-requests@localhost @@ -41,8 +44,9 @@ humpadink_en_public_body_translation:    url_name: dfh    locale: en    notes: An albatross told me!!! +  publication_scheme: "" -forlorn_en_public_body_translation:  +forlorn_en_public_body_translation:    name: "Department of Loneliness"    first_letter: D    request_email: forlorn-requests@localhost @@ -52,6 +56,7 @@ forlorn_en_public_body_translation:    url_name: lonely    locale: en    notes: A very lonely public body that no one has corresponded with +  publication_scheme: ""  silly_walks_en_public_body_translation:    id: 6 @@ -63,6 +68,7 @@ silly_walks_en_public_body_translation:    short_name: MSW    url_name: msw    notes: You know the one. +  publication_scheme: ""  sensible_walks_en_public_body_translation:    id: 7 @@ -74,3 +80,4 @@ sensible_walks_en_public_body_translation:    short_name: SenseWalk    url_name: sensible_walks    notes: I bet you’ve never heard of it. +  publication_scheme: "" diff --git a/spec/models/info_request_spec.rb b/spec/models/info_request_spec.rb index c55127992..204c600d9 100644 --- a/spec/models/info_request_spec.rb +++ b/spec/models/info_request_spec.rb @@ -341,6 +341,14 @@ describe InfoRequest do              InfoRequest.find_old_unclassified(:limit => 5)          end +        it 'should ask for requests using any offset param supplied' do +            InfoRequest.should_receive(:find).with(:all, {:select => anything, +                                                          :order => anything, +                                                          :conditions=> anything, +                                                          :offset => 100}) +            InfoRequest.find_old_unclassified(:offset => 100) +        end +          it 'should not limit the number of requests returned by default' do              InfoRequest.should_not_receive(:find).with(:all, {:select => anything,                                                                :order => anything, @@ -350,20 +358,49 @@ describe InfoRequest do          end          it 'should add extra conditions if supplied' do -            InfoRequest.should_receive(:find).with(:all, -                  {:select=> anything, -                   :order=> anything, -                   :conditions=>["awaiting_description = ? and (select created_at from info_request_events where info_request_events.info_request_id = info_requests.id and info_request_events.event_type = 'response' order by created_at desc limit 1) < ? and url_title != 'holding_pen' and user_id is not null and prominence != 'backpage'", -                    true, Time.now - 21.days]}) +            expected_conditions = ["awaiting_description = ? +                                    AND (SELECT created_at +                                         FROM info_request_events +                                         WHERE info_request_events.info_request_id = info_requests.id +                                         AND info_request_events.event_type = 'response' +                                         ORDER BY created_at desc LIMIT 1) < ? +                                    AND url_title != 'holding_pen' +                                    AND user_id IS NOT NULL +                                    AND prominence != 'backpage'".split(' ').join(' '), +                                    true, Time.now - 21.days] +            # compare conditions ignoring whitespace differences +            InfoRequest.should_receive(:find) do |all, query_params| +                query_string = query_params[:conditions][0] +                query_params[:conditions][0] = query_string.split(' ').join(' ') +                query_params[:conditions].should == expected_conditions +            end              InfoRequest.find_old_unclassified({:conditions => ["prominence != 'backpage'"]})          end -        it 'should ask the database for requests that are awaiting description, have a last response older than 21 days old, are not the holding pen and are not backpaged' do -            InfoRequest.should_receive(:find).with(:all, -                  {:select=>"*, (select created_at from info_request_events where info_request_events.info_request_id = info_requests.id and info_request_events.event_type = 'response' order by created_at desc limit 1) as last_response_time", -                   :order=>"last_response_time", -                   :conditions=>["awaiting_description = ? and (select created_at from info_request_events where info_request_events.info_request_id = info_requests.id and info_request_events.event_type = 'response' order by created_at desc limit 1) < ? and url_title != 'holding_pen' and user_id is not null", -                    true, Time.now - 21.days]}) +        it 'should ask the database for requests that are awaiting description, have a last response older +        than 21 days old, have a user, are not the holding pen and are not backpaged' do +            expected_conditions = ["awaiting_description = ? +                                    AND (SELECT created_at +                                         FROM info_request_events +                                         WHERE info_request_events.info_request_id = info_requests.id +                                         AND info_request_events.event_type = 'response' +                                         ORDER BY created_at desc LIMIT 1) < ? +                                    AND url_title != 'holding_pen' +                                    AND user_id IS NOT NULL".split(' ').join(' '), +                                    true, Time.now - 21.days] +            expected_select = "*, (SELECT created_at +                                   FROM info_request_events +                                   WHERE info_request_events.info_request_id = info_requests.id +                                   AND info_request_events.event_type = 'response' +                                   ORDER BY created_at desc LIMIT 1) +                                   AS last_response_time".split(' ').join(' ') +            InfoRequest.should_receive(:find) do |all, query_params| +                query_string = query_params[:conditions][0] +                query_params[:conditions][0] = query_string.split(' ').join(' ') +                query_params[:conditions].should == expected_conditions +                query_params[:select].split(' ').join(' ').should == expected_select +                query_params[:order].should == "last_response_time" +            end              InfoRequest.find_old_unclassified          end diff --git a/spec/models/public_body_spec.rb b/spec/models/public_body_spec.rb index 8ff6afde3..011824190 100644 --- a/spec/models/public_body_spec.rb +++ b/spec/models/public_body_spec.rb @@ -222,6 +222,10 @@ describe PublicBody, "when searching" do              body.name.should == "El A Geraldine Quango"          end      end + +    it 'should not raise an error on a name with a single quote in it' do +        body = PublicBody.find_by_url_name_with_historic("belfast city council'") +    end  end  describe PublicBody, " when dealing public body locales" do @@ -403,6 +407,7 @@ describe PublicBody, " when loading CSV files" do  end  describe PublicBody do +    describe "calculated home page" do      it "should return the home page verbatim if it's present" do        public_body = PublicBody.new @@ -434,4 +439,17 @@ describe PublicBody do        public_body.calculated_home_page.should == "https://example.com"      end    end + +    describe 'when asked for notes without html' do + +        before do +            @public_body = PublicBody.new(:notes => 'some <a href="/notes">notes</a>') +        end + +        it 'should remove simple tags from notes' do +            @public_body.notes_without_html.should == 'some notes' +        end + +    end +  end diff --git a/spec/models/request_mailer_spec.rb b/spec/models/request_mailer_spec.rb index ea75ec765..98681a9e9 100644 --- a/spec/models/request_mailer_spec.rb +++ b/spec/models/request_mailer_spec.rb @@ -29,7 +29,7 @@ describe RequestMailer, " when receiving incoming mail" do          InfoRequest.holding_pen_request.incoming_messages.size.should == 1          last_event = InfoRequest.holding_pen_request.incoming_messages[0].info_request.get_last_event          last_event.params[:rejected_reason].should == "Could not identify the request from the email address" -         +          deliveries = ActionMailer::Base.deliveries          deliveries.size.should == 1          mail = deliveries[0] @@ -49,7 +49,7 @@ describe RequestMailer, " when receiving incoming mail" do          InfoRequest.holding_pen_request.incoming_messages.size.should == 1          last_event = InfoRequest.holding_pen_request.incoming_messages[0].info_request.get_last_event          last_event.params[:rejected_reason].should =~ /there is no "From" address/ -         +          deliveries = ActionMailer::Base.deliveries          deliveries.size.should == 1          mail = deliveries[0] @@ -69,7 +69,7 @@ describe RequestMailer, " when receiving incoming mail" do          InfoRequest.holding_pen_request.incoming_messages.size.should == 1          last_event = InfoRequest.holding_pen_request.incoming_messages[0].info_request.get_last_event          last_event.params[:rejected_reason].should =~ /Only the authority can reply/ -         +          deliveries = ActionMailer::Base.deliveries          deliveries.size.should == 1          mail = deliveries[0] @@ -222,12 +222,27 @@ describe RequestMailer, "when sending reminders to requesters to classify a resp          RequestMailer.alert_new_response_reminders_internal(7, 'new_response_reminder_1')      end -    it 'should ask for all requests that are awaiting description and whose latest response is older than the number of days given and that are not the holding pen' do -        expected_params = {:conditions => [ "awaiting_description = ? and (select created_at from info_request_events where info_request_events.info_request_id = info_requests.id and info_request_events.event_type = 'response' order by created_at desc limit 1) < ? and url_title != 'holding_pen' and user_id is not null", -                           true, Time.now() - 7.days ], -                           :include => [ :user ], -                           :order => "info_requests.id"} -        InfoRequest.should_receive(:find).with(:all, expected_params).and_return([]) +    it 'should ask for all requests that are awaiting description and whose latest response is older +        than the number of days given and that are not the holding pen' do +        expected_conditions = [ "awaiting_description = ? +                                 AND (SELECT created_at +                                      FROM info_request_events +                                      WHERE info_request_events.info_request_id = info_requests.id +                                      AND info_request_events.event_type = 'response' +                                      ORDER BY created_at desc LIMIT 1) < ? +                                 AND url_title != 'holding_pen' +                                 AND user_id IS NOT NULL".split(' ').join(' '), +                                 true, Time.now() - 7.days ] + +        # compare the query string ignoring any spacing differences +        InfoRequest.should_receive(:find) do |all, query_params| +            query_string = query_params[:conditions][0] +            query_params[:conditions][0] = query_string.split(' ').join(' ') +            query_params[:conditions].should == expected_conditions +            query_params[:include].should == [ :user ] +            query_params[:order].should == 'info_requests.id' +        end +          send_alerts      end | 
