summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJean-Philippe Lang <jp_lang@yahoo.fr>2007-03-17 15:18:50 +0000
committerJean-Philippe Lang <jp_lang@yahoo.fr>2007-03-17 15:18:50 +0000
commit10cf1ccc1abe1c6ffdf21259b54292f9068b569c (patch)
treed43c2fd622b2126f72672d177f482e4934106e3d
parent22b42dc90a86b16904224bf638054fffd062b53f (diff)
downloadredmine-10cf1ccc1abe1c6ffdf21259b54292f9068b569c.tar.gz
redmine-10cf1ccc1abe1c6ffdf21259b54292f9068b569c.zip
added rss/atom feeds at project levels for:
* news * new issues reported * details of issue changes issue cutom queries can be used as feeds git-svn-id: http://redmine.rubyforge.org/svn/trunk@339 e93f8b46-1217-0410-a6f0-8f06a7374b81
-rw-r--r--app/controllers/feeds_controller.rb81
-rw-r--r--app/controllers/projects_controller.rb5
-rw-r--r--app/models/journal.rb4
-rw-r--r--app/models/token.rb2
-rw-r--r--app/models/user.rb10
-rw-r--r--app/views/feeds/history.rxml28
-rw-r--r--app/views/feeds/history_atom.rxml28
-rw-r--r--app/views/feeds/issues.rxml20
-rw-r--r--app/views/feeds/issues_atom.rxml22
-rw-r--r--app/views/feeds/news.rxml38
-rw-r--r--app/views/feeds/news_atom.rxml22
-rw-r--r--app/views/projects/feeds.rhtml33
-rw-r--r--app/views/projects/show.rhtml4
-rw-r--r--db/migrate/030_add_projects_feeds_permissions.rb9
-rw-r--r--lang/de.yml2
-rw-r--r--lang/en.yml2
-rw-r--r--lang/es.yml2
-rw-r--r--lang/fr.yml2
-rw-r--r--lang/it.yml2
-rw-r--r--lang/ja.yml2
-rw-r--r--public/stylesheets/application.css2
-rw-r--r--test/fixtures/members.yml7
-rw-r--r--test/functional/feeds_controller_test.rb66
-rw-r--r--test/unit/user_test.rb13
24 files changed, 383 insertions, 23 deletions
diff --git a/app/controllers/feeds_controller.rb b/app/controllers/feeds_controller.rb
index 7a454d80a..46e8ae05c 100644
--- a/app/controllers/feeds_controller.rb
+++ b/app/controllers/feeds_controller.rb
@@ -1,5 +1,5 @@
# redMine - project management software
-# Copyright (C) 2006 Jean-Philippe Lang
+# Copyright (C) 2006-2007 Jean-Philippe Lang
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
@@ -16,10 +16,85 @@
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
class FeedsController < ApplicationController
+ before_filter :find_scope
session :off
-
+
+ helper :issues
+ include IssuesHelper
+ helper :custom_fields
+ include CustomFieldsHelper
+
+ # news feeds
def news
- @news = News.find :all, :order => "#{News.table_name}.created_on DESC", :limit => 10, :include => [ :author, :project ]
+ News.with_scope(:find => @find_options) do
+ @news = News.find :all, :order => "#{News.table_name}.created_on DESC", :limit => 10, :include => [ :author, :project ]
+ end
headers["Content-Type"] = "application/rss+xml"
+ render :action => 'news_atom' if 'atom' == params[:format]
+ end
+
+ # issue feeds
+ def issues
+ conditions = nil
+
+ if params[:query_id]
+ query = Query.find(params[:query_id])
+ # ignore query if it's not valid
+ query = nil unless query.valid?
+ conditions = query.statement if query
+ end
+
+ Issue.with_scope(:find => @find_options) do
+ @issues = Issue.find :all, :include => [:project, :author, :tracker, :status],
+ :order => "#{Issue.table_name}.created_on DESC",
+ :conditions => conditions
+ end
+ @title = (@project ? @project.name : Setting.app_title) + ": " + (query ? query.name : l(:label_reported_issues))
+ headers["Content-Type"] = "application/rss+xml"
+ render :action => 'issues_atom' if 'atom' == params[:format]
+ end
+
+ # issue changes feeds
+ def history
+ conditions = nil
+
+ if params[:query_id]
+ query = Query.find(params[:query_id])
+ # ignore query if it's not valid
+ query = nil unless query.valid?
+ conditions = query.statement if query
+ end
+
+ Journal.with_scope(:find => @find_options) do
+ @journals = Journal.find :all, :include => [ :details, :user, {:issue => [:project, :author, :tracker, :status]} ],
+ :order => "#{Journal.table_name}.created_on DESC",
+ :conditions => conditions
+ end
+
+ @title = (@project ? @project.name : Setting.app_title) + ": " + (query ? query.name : l(:label_reported_issues))
+ headers["Content-Type"] = "application/rss+xml"
+ render :action => 'history_atom' if 'atom' == params[:format]
+ end
+
+private
+ # override for feeds specific authentication
+ def check_if_login_required
+ @user = User.find_by_rss_key(params[:key])
+ render(:nothing => true, :status => 403) and return false if !@user && Setting.login_required?
+ end
+
+ def find_scope
+ if params[:project_id]
+ # project feed
+ # check if project is public or if the user is a member
+ @project = Project.find(params[:project_id])
+ render(:nothing => true, :status => 403) and return false unless @project.is_public? || (@user && @user.role_for_project(@project.id))
+ scope = ["#{Project.table_name}.id=?", params[:project_id].to_i]
+ else
+ # global feed
+ scope = ["#{Project.table_name}.is_public=?", true]
+ end
+ @find_options = {:conditions => scope, :limit => 10}
+ return true
end
end
diff --git a/app/controllers/projects_controller.rb b/app/controllers/projects_controller.rb
index fe4836811..61b772194 100644
--- a/app/controllers/projects_controller.rb
+++ b/app/controllers/projects_controller.rb
@@ -575,6 +575,11 @@ class ProjectsController < ApplicationController
end
end
+ def feeds
+ @queries = @project.queries.find :all, :conditions => ["is_public=? or user_id=?", true, (logged_in_user ? logged_in_user.id : 0)]
+ @key = logged_in_user.get_or_create_rss_key.value if logged_in_user
+ end
+
private
# Find project of id params[:id]
# if not found, redirect to project list
diff --git a/app/models/journal.rb b/app/models/journal.rb
index 18a6ec083..f70a69863 100644
--- a/app/models/journal.rb
+++ b/app/models/journal.rb
@@ -17,6 +17,10 @@
class Journal < ActiveRecord::Base
belongs_to :journalized, :polymorphic => true
+ # added as a quick fix to allow eager loading of the polymorphic association
+ # since always associated to an issue, for now
+ belongs_to :issue, :foreign_key => :journalized_id
+
belongs_to :user
has_many :details, :class_name => "JournalDetail", :dependent => :delete_all
end
diff --git a/app/models/token.rb b/app/models/token.rb
index 98745d29e..0e8c2c3e2 100644
--- a/app/models/token.rb
+++ b/app/models/token.rb
@@ -31,7 +31,7 @@ class Token < ActiveRecord::Base
# Delete all expired tokens
def self.destroy_expired
- Token.delete_all ["created_on < ?", Time.now - @@validity_time]
+ Token.delete_all ["action <> 'feeds' AND created_on < ?", Time.now - @@validity_time]
end
private
diff --git a/app/models/user.rb b/app/models/user.rb
index 3a315239c..869be920e 100644
--- a/app/models/user.rb
+++ b/app/models/user.rb
@@ -22,6 +22,7 @@ class User < ActiveRecord::Base
has_many :projects, :through => :memberships
has_many :custom_values, :dependent => :delete_all, :as => :customized
has_one :preference, :dependent => :destroy, :class_name => 'UserPreference'
+ has_one :rss_key, :dependent => :destroy, :class_name => 'Token', :conditions => "action='feeds'"
belongs_to :auth_source
attr_accessor :password, :password_confirmation
@@ -133,6 +134,15 @@ class User < ActiveRecord::Base
def pref
self.preference ||= UserPreference.new(:user => self)
end
+
+ def get_or_create_rss_key
+ self.rss_key || Token.create(:user => self, :action => 'feeds')
+ end
+
+ def self.find_by_rss_key(key)
+ token = Token.find_by_value(key)
+ token && token.user.active? ? token.user : nil
+ end
private
# Return password digest
diff --git a/app/views/feeds/history.rxml b/app/views/feeds/history.rxml
new file mode 100644
index 000000000..b7e5a3509
--- /dev/null
+++ b/app/views/feeds/history.rxml
@@ -0,0 +1,28 @@
+xml.instruct!
+xml.rss "version" => "2.0", "xmlns:dc" => "http://purl.org/dc/elements/1.1/" do
+ xml.channel do
+ xml.title @title
+ xml.link url_for(:controller => 'welcome', :only_path => false)
+ xml.pubDate CGI.rfc1123_date(@journals.first ? @journals.first.created_on : Time.now)
+ xml.description l(:label_reported_issues)
+ @journals.each do |journal|
+ issue = journal.issue
+ xml.item do
+ xml.title "#{issue.project.name} - #{issue.tracker.name} ##{issue.id}: #{issue.subject}"
+ url = url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.link url
+ xml.description do
+ xml.text! h(journal.notes)
+ xml.text! "<ul>"
+ journal.details.each do |detail|
+ xml.text! "<li>" + show_detail(detail, false) + "</li>"
+ end
+ xml.text! "</ul>"
+ end
+ xml.pubDate CGI.rfc1123_date(journal.created_on)
+ xml.guid url
+ xml.author h(journal.user.name)
+ end
+ end
+ end
+end \ No newline at end of file
diff --git a/app/views/feeds/history_atom.rxml b/app/views/feeds/history_atom.rxml
new file mode 100644
index 000000000..9d82ef606
--- /dev/null
+++ b/app/views/feeds/history_atom.rxml
@@ -0,0 +1,28 @@
+xml.instruct!
+xml.feed "xmlns" => "http://www.w3.org/2005/Atom" do
+ xml.title @title
+ xml.link "rel" => "self", "href" => url_for(:controller => 'feeds', :action => 'history', :format => 'atom', :only_path => false)
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'welcome', :only_path => false)
+ xml.id url_for(:controller => 'welcome', :only_path => false)
+ xml.updated CGI.rfc1123_date(@journals.first.created_on) if @journals.any?
+ xml.author { xml.name "#{Setting.app_title}" }
+ @journals.each do |journal|
+ issue = journal.issue
+ xml.entry do
+ xml.title "#{issue.project.name} - #{issue.tracker.name} ##{issue.id}: #{issue.subject}"
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.id url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.updated CGI.rfc1123_date(journal.created_on)
+ xml.author { xml.name journal.user.name }
+ xml.summary journal.notes
+ xml.content "type" => "html" do
+ xml.text! journal.notes
+ xml.text! "<ul>"
+ journal.details.each do |detail|
+ xml.text! "<li>" + show_detail(detail, false) + "</li>"
+ end
+ xml.text! "</ul>"
+ end
+ end
+ end
+end \ No newline at end of file
diff --git a/app/views/feeds/issues.rxml b/app/views/feeds/issues.rxml
new file mode 100644
index 000000000..fb120b7cb
--- /dev/null
+++ b/app/views/feeds/issues.rxml
@@ -0,0 +1,20 @@
+xml.instruct!
+xml.rss "version" => "2.0", "xmlns:dc" => "http://purl.org/dc/elements/1.1/" do
+ xml.channel do
+ xml.title @title
+ xml.link url_for(:controller => 'welcome', :only_path => false)
+ xml.pubDate CGI.rfc1123_date(@issues.first ? @issues.first.created_on : Time.now)
+ xml.description l(:label_reported_issues)
+ @issues.each do |issue|
+ xml.item do
+ xml.title "#{issue.project.name} - #{issue.tracker.name} ##{issue.id}: #{issue.subject}"
+ url = url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.link url
+ xml.description h(issue.description)
+ xml.pubDate CGI.rfc1123_date(issue.created_on)
+ xml.guid url
+ xml.author h(issue.author.name)
+ end
+ end
+ end
+end \ No newline at end of file
diff --git a/app/views/feeds/issues_atom.rxml b/app/views/feeds/issues_atom.rxml
new file mode 100644
index 000000000..bb15fc493
--- /dev/null
+++ b/app/views/feeds/issues_atom.rxml
@@ -0,0 +1,22 @@
+xml.instruct!
+xml.feed "xmlns" => "http://www.w3.org/2005/Atom" do
+ xml.title @title
+ xml.link "rel" => "self", "href" => url_for(:controller => 'feeds', :action => 'issues', :format => 'atom', :only_path => false)
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'welcome', :only_path => false)
+ xml.id url_for(:controller => 'welcome', :only_path => false)
+ xml.updated CGI.rfc1123_date(@issues.first.updated_on) if @issues.any?
+ xml.author { xml.name "#{Setting.app_title}" }
+ @issues.each do |issue|
+ xml.entry do
+ xml.title "#{issue.project.name} - #{issue.tracker.name} ##{issue.id}: #{issue.subject}"
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.id url_for(:controller => 'issues' , :action => 'show', :id => issue, :only_path => false)
+ xml.updated CGI.rfc1123_date(issue.updated_on)
+ xml.author { xml.name issue.author.name }
+ xml.summary issue.description
+ xml.content "type" => "html" do
+ xml.text! issue.description
+ end
+ end
+ end
+end \ No newline at end of file
diff --git a/app/views/feeds/news.rxml b/app/views/feeds/news.rxml
index a85d86715..d7248b6cb 100644
--- a/app/views/feeds/news.rxml
+++ b/app/views/feeds/news.rxml
@@ -1,20 +1,20 @@
-xml.instruct!
-xml.rss "version" => "2.0", "xmlns:dc" => "http://purl.org/dc/elements/1.1/" do
- xml.channel do
- xml.title "#{Setting.app_title}: #{l(:label_news_latest)}"
- xml.link url_for(:controller => 'welcome', :only_path => false)
- xml.pubDate CGI.rfc1123_date(@news.first ? @news.first.created_on : Time.now)
- xml.description l(:label_news_latest)
- @news.each do |news|
- xml.item do
- xml.title "#{news.project.name}: #{news.title}"
- news_url = url_for(:controller => 'news' , :action => 'show', :id => news, :only_path => false)
- xml.link news_url
- xml.description h(news.summary)
- xml.pubDate CGI.rfc1123_date(news.created_on)
- xml.guid news_url
- xml.author h(news.author.name)
- end
- end
- end
+xml.instruct!
+xml.rss "version" => "2.0", "xmlns:dc" => "http://purl.org/dc/elements/1.1/" do
+ xml.channel do
+ xml.title "#{Setting.app_title}: #{l(:label_news_latest)}"
+ xml.link url_for(:controller => 'welcome', :only_path => false)
+ xml.pubDate CGI.rfc1123_date(@news.first ? @news.first.created_on : Time.now)
+ xml.description l(:label_news_latest)
+ @news.each do |news|
+ xml.item do
+ xml.title "#{news.project.name}: #{news.title}"
+ news_url = url_for(:controller => 'news' , :action => 'show', :id => news, :only_path => false)
+ xml.link news_url
+ xml.description h(news.summary)
+ xml.pubDate CGI.rfc1123_date(news.created_on)
+ xml.guid news_url
+ xml.author h(news.author.name)
+ end
+ end
+ end
end \ No newline at end of file
diff --git a/app/views/feeds/news_atom.rxml b/app/views/feeds/news_atom.rxml
new file mode 100644
index 000000000..2550341c8
--- /dev/null
+++ b/app/views/feeds/news_atom.rxml
@@ -0,0 +1,22 @@
+xml.instruct!
+xml.feed "xmlns" => "http://www.w3.org/2005/Atom" do
+ xml.title "#{Setting.app_title}: #{l(:label_news_latest)}"
+ xml.link "rel" => "self", "href" => url_for(:controller => 'feeds', :action => 'news', :format => 'atom', :only_path => false)
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'welcome', :only_path => false)
+ xml.id url_for(:controller => 'welcome', :only_path => false)
+ xml.updated CGI.rfc1123_date(@news.first.created_on) if @news.any?
+ xml.author { xml.name "#{Setting.app_title}" }
+ @news.each do |news|
+ xml.entry do
+ xml.title news.title
+ xml.link "rel" => "alternate", "href" => url_for(:controller => 'news' , :action => 'show', :id => news, :only_path => false)
+ xml.id url_for(:controller => 'news' , :action => 'show', :id => news, :only_path => false)
+ xml.updated CGI.rfc1123_date(news.created_on)
+ xml.author { xml.name news.author.name }
+ xml.summary h(news.summary)
+ xml.content "type" => "html" do
+ xml.text! news.description
+ end
+ end
+ end
+end \ No newline at end of file
diff --git a/app/views/projects/feeds.rhtml b/app/views/projects/feeds.rhtml
new file mode 100644
index 000000000..037469a8a
--- /dev/null
+++ b/app/views/projects/feeds.rhtml
@@ -0,0 +1,33 @@
+<h2><%= l(:label_feed_plural) %> (<%=h @project.name %>)</h2>
+
+<table>
+
+<tr><td colspan="3"><h3><%= l(:label_issue_plural) %></h3></td></tr>
+<tr><td><%= l(:label_reported_issues) %></td>
+<td><%= link_to 'RSS', {:controller => 'feeds', :action => 'issues', :project_id => @project, :key => @key}, :class => 'icon icon-feed' %></td>
+<td><%= link_to 'Atom', {:controller => 'feeds', :action => 'issues', :project_id => @project, :key => @key, :format => 'atom'}, :class => 'icon icon-feed' %></td>
+</tr>
+<tr><td><%= l(:label_changes_details) %></td>
+<td><%= link_to 'RSS', {:controller => 'feeds', :action => 'history', :project_id => @project, :key => @key}, :class => 'icon icon-feed' %></td>
+<td><%= link_to 'Atom', {:controller => 'feeds', :action => 'history', :project_id => @project, :key => @key, :format => 'atom'}, :class => 'icon icon-feed' %></td>
+</tr>
+
+<% @queries.each do |query| %>
+<tr><td colspan="3"><h4><%=h query.name %></h4></td></tr>
+<tr><td><%= l(:label_reported_issues) %></td>
+<td><%= link_to 'RSS', {:controller => 'feeds', :action => 'issues', :project_id => @project, :query_id => query, :key => @key}, :class => 'icon icon-feed' %></td>
+<td><%= link_to 'Atom', {:controller => 'feeds', :action => 'issues', :project_id => @project, :query_id => query, :key => @key, :format => 'atom'}, :class => 'icon icon-feed' %></td>
+</tr>
+<tr><td><%= l(:label_changes_details) %></td>
+<td><%= link_to 'RSS', {:controller => 'feeds', :action => 'history', :project_id => @project, :query_id => query, :key => @key}, :class => 'icon icon-feed' %></td>
+<td><%= link_to 'Atom', {:controller => 'feeds', :action => 'history', :project_id => @project, :query_id => query, :key => @key, :format => 'atom'}, :class => 'icon icon-feed' %></td>
+</tr>
+<% end %>
+
+<tr><td colspan="3">&nbsp;<h3><%= l(:label_news_plural) %></h3></td></tr>
+<tr><td><%= l(:label_news_latest) %></td>
+<td><%= link_to 'RSS', {:controller => 'feeds', :action => 'news', :project_id => @project, :key => @key}, :class => 'icon icon-feed' %></td>
+<td><%= link_to 'Atom', {:controller => 'feeds', :action => 'news', :project_id => @project, :key => @key, :format => 'atom'}, :class => 'icon icon-feed' %></td>
+</tr>
+
+</table> \ No newline at end of file
diff --git a/app/views/projects/show.rhtml b/app/views/projects/show.rhtml
index b38eceaac..9ecbbb663 100644
--- a/app/views/projects/show.rhtml
+++ b/app/views/projects/show.rhtml
@@ -1,3 +1,7 @@
+<div class="contextual">
+<%= link_to l(:label_feed_plural), {:action => 'feeds', :id => @project}, :class => 'icon icon-feed' %>
+</div>
+
<h2><%=l(:label_overview)%></h2>
<div class="splitcontentleft">
diff --git a/db/migrate/030_add_projects_feeds_permissions.rb b/db/migrate/030_add_projects_feeds_permissions.rb
new file mode 100644
index 000000000..63131001e
--- /dev/null
+++ b/db/migrate/030_add_projects_feeds_permissions.rb
@@ -0,0 +1,9 @@
+class AddProjectsFeedsPermissions < ActiveRecord::Migration
+ def self.up
+ Permission.create :controller => "projects", :action => "feeds", :description => "label_feed_plural", :sort => 132, :is_public => true, :mail_option => 0, :mail_enabled => 0
+ end
+
+ def self.down
+ Permission.find_by_controller_and_action('projects', 'feeds').destroy
+ end
+end
diff --git a/lang/de.yml b/lang/de.yml
index 1e301456e..9d4015d3d 100644
--- a/lang/de.yml
+++ b/lang/de.yml
@@ -328,6 +328,8 @@ label_wiki: Wiki
label_page_index: Index
label_current_version: Gegenwärtige Version
label_preview: Vorbetrachtung
+label_feed_plural: Feeds
+label_changes_details: Details of all changes
button_login: Einloggen
button_submit: Einreichen
diff --git a/lang/en.yml b/lang/en.yml
index f369afbd8..637c36f5c 100644
--- a/lang/en.yml
+++ b/lang/en.yml
@@ -328,6 +328,8 @@ label_wiki: Wiki
label_page_index: Index
label_current_version: Current version
label_preview: Preview
+label_feed_plural: Feeds
+label_changes_details: Details of all changes
button_login: Login
button_submit: Submit
diff --git a/lang/es.yml b/lang/es.yml
index 0162b6bdd..604a9e1cc 100644
--- a/lang/es.yml
+++ b/lang/es.yml
@@ -328,6 +328,8 @@ label_wiki: Wiki
label_page_index: Índice
label_current_version: Versión actual
label_preview: Previo
+label_feed_plural: Feeds
+label_changes_details: Detalles de todos los cambios
button_login: Conexión
button_submit: Someter
diff --git a/lang/fr.yml b/lang/fr.yml
index 8f629f7bb..a41f61f3f 100644
--- a/lang/fr.yml
+++ b/lang/fr.yml
@@ -328,6 +328,8 @@ label_wiki: Wiki
label_page_index: Index
label_current_version: Version actuelle
label_preview: Prévisualisation
+label_feed_plural: Flux RSS
+label_changes_details: Détails de tous les changements
button_login: Connexion
button_submit: Soumettre
diff --git a/lang/it.yml b/lang/it.yml
index eae7c882c..4380651f2 100644
--- a/lang/it.yml
+++ b/lang/it.yml
@@ -328,6 +328,8 @@ label_wiki: Wiki
label_page_index: Indice
label_current_version: Versione corrente
label_preview: Previsione
+label_feed_plural: Feeds
+label_changes_details: Particolari di tutti i cambiamenti
button_login: Login
button_submit: Invia
diff --git a/lang/ja.yml b/lang/ja.yml
index be45f58a6..0c7c00ad8 100644
--- a/lang/ja.yml
+++ b/lang/ja.yml
@@ -329,6 +329,8 @@ label_wiki: Wiki
label_page_index: 索引
label_current_version: 最近版
label_preview: 下検分
+label_feed_plural: Feeds
+label_changes_details: Details of all changes
button_login: ログイン
button_submit: 変更
diff --git a/public/stylesheets/application.css b/public/stylesheets/application.css
index 3fe0b1d68..7fda16cd3 100644
--- a/public/stylesheets/application.css
+++ b/public/stylesheets/application.css
@@ -154,6 +154,7 @@ vertical-align: middle;
.icon-attachment { background-image: url(../images/attachment.png); }
.icon-index { background-image: url(../images/index.png); }
.icon-history { background-image: url(../images/history.png); }
+.icon-feed { background-image: url(../images/feed.png); }
.icon22-projects { background-image: url(../images/22x22/projects.png); }
.icon22-users { background-image: url(../images/22x22/users.png); }
@@ -247,6 +248,7 @@ legend {color: #505050;}
.even {background-color: #fff;}
hr { border:0; border-top: dotted 1px #fff; border-bottom: dotted 1px #c0c0c0; }
table p {margin:0; padding:0;}
+table td {padding-right: 1em;}
.highlight { background-color: #FCFD8D;}
diff --git a/test/fixtures/members.yml b/test/fixtures/members.yml
index 0626bdb18..392225e52 100644
--- a/test/fixtures/members.yml
+++ b/test/fixtures/members.yml
@@ -11,3 +11,10 @@ members_002:
role_id: 2
id: 2
user_id: 3
+members_003:
+ created_on: 2006-07-19 19:35:36 +02:00
+ project_id: 2
+ role_id: 2
+ id: 3
+ user_id: 2
+ \ No newline at end of file
diff --git a/test/functional/feeds_controller_test.rb b/test/functional/feeds_controller_test.rb
new file mode 100644
index 000000000..279b2c1a7
--- /dev/null
+++ b/test/functional/feeds_controller_test.rb
@@ -0,0 +1,66 @@
+# redMine - project management software
+# Copyright (C) 2006-2007 Jean-Philippe Lang
+#
+# This program is free software; you can redistribute it and/or
+# modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation; either version 2
+# of the License, or (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+
+require File.dirname(__FILE__) + '/../test_helper'
+require 'feeds_controller'
+
+# Re-raise errors caught by the controller.
+class FeedsController; def rescue_action(e) raise e end; end
+
+class FeedsControllerTest < Test::Unit::TestCase
+ fixtures :projects, :users, :members, :roles
+
+ def setup
+ @controller = FeedsController.new
+ @request = ActionController::TestRequest.new
+ @response = ActionController::TestResponse.new
+ end
+
+ def test_news
+ get :news
+ assert_response :success
+ assert_template 'news'
+ assert_not_nil assigns(:news)
+ end
+
+ def test_issues
+ get :issues
+ assert_response :success
+ assert_template 'issues'
+ assert_not_nil assigns(:issues)
+ end
+
+ def test_history
+ get :history
+ assert_response :success
+ assert_template 'history'
+ assert_not_nil assigns(:journals)
+ end
+
+ def test_project_privacy
+ get :news, :project_id => 2
+ assert_response 403
+ end
+
+ def test_rss_key
+ user = User.find(2)
+ key = user.get_or_create_rss_key.value
+
+ get :news, :project_id => 2, :key => key
+ assert_response :success
+ end
+end
diff --git a/test/unit/user_test.rb b/test/unit/user_test.rb
index 211e6554c..10aafa58a 100644
--- a/test/unit/user_test.rb
+++ b/test/unit/user_test.rb
@@ -85,4 +85,17 @@ class UserTest < Test::Unit::TestCase
user = User.try_to_login("jsmith", "jsmith")
assert_equal nil, user
end
+
+ def test_rss_key
+ assert_nil @jsmith.rss_key
+ key = @jsmith.get_or_create_rss_key
+ assert_kind_of Token, key
+ assert_equal 40, key.value.length
+
+ @jsmith.reload
+ assert_equal key.value, @jsmith.get_or_create_rss_key.value
+
+ @jsmith.reload
+ assert_equal key.value, @jsmith.rss_key.value
+ end
end