Compare commits
71 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
54eecf54ff | ||
|
|
628e63eefe | ||
|
|
83b356484a | ||
|
|
5210b58226 | ||
|
|
a0932173a5 | ||
|
|
e889fe9c8a | ||
|
|
16b1fcb079 | ||
|
|
3b7793561a | ||
|
|
2c3c1d2a02 | ||
|
|
49cb3aaca6 | ||
|
|
8a8df72937 | ||
|
|
ad238abd0c | ||
|
|
d69cae859f | ||
|
|
59d4d210a7 | ||
|
|
633b86236c | ||
|
|
91514dd682 | ||
|
|
917127f179 | ||
|
|
49295481de | ||
|
|
8cc56cce66 | ||
|
|
8d5fadc773 | ||
|
|
eb3e79b70d | ||
|
|
5d3bdb66cd | ||
|
|
1974123562 | ||
|
|
bcef197b41 | ||
|
|
231145160f | ||
|
|
0e1ae572c7 | ||
|
|
b3a86771ce | ||
|
|
2516d4f408 | ||
|
|
03d6c4668e | ||
|
|
a97d6baddd | ||
|
|
f3a8494cfb | ||
|
|
14dcefaa97 | ||
|
|
f4a7d6ca8c | ||
|
|
aedd5c5e35 | ||
|
|
554d6bcba2 | ||
|
|
4fcec69a40 | ||
|
|
7e6cc2f583 | ||
|
|
16383ed386 | ||
|
|
27127e1a3b | ||
|
|
a92ef237d3 | ||
|
|
a82a16378f | ||
|
|
886daba641 | ||
|
|
ab8dd5aa6e | ||
|
|
dbc02e66dc | ||
|
|
75e350f4e6 | ||
|
|
81553d396e | ||
|
|
77e03b508b | ||
|
|
1b0e717902 | ||
|
|
53b8de5e39 | ||
|
|
34df573ec2 | ||
|
|
e2f00792a8 | ||
|
|
9ed998af9b | ||
|
|
1881706df4 | ||
|
|
a3bf12ab9f | ||
|
|
668981b8cd | ||
|
|
b303dddbe3 | ||
|
|
933e96116e | ||
|
|
10711fda6d | ||
|
|
b3b829c025 | ||
|
|
8cd0baf773 | ||
|
|
033aa68427 | ||
|
|
1eb20d42bd | ||
|
|
a9dbecd5a5 | ||
|
|
a13218e9f4 | ||
|
|
57b0faf128 | ||
|
|
ba3828b49f | ||
|
|
fe1a152e02 | ||
|
|
69709a2513 | ||
|
|
7b7bca0b59 | ||
|
|
64b0cb336d | ||
|
|
79b7b32980 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -18,6 +18,7 @@
|
||||
/public/plugin_assets
|
||||
/tmp/*
|
||||
/tmp/cache/*
|
||||
/tmp/pdf/*
|
||||
/tmp/sessions/*
|
||||
/tmp/sockets/*
|
||||
/tmp/test/*
|
||||
|
||||
@@ -20,6 +20,7 @@ public/dispatch.*
|
||||
public/plugin_assets
|
||||
tmp/*
|
||||
tmp/cache/*
|
||||
tmp/pdf/*
|
||||
tmp/sessions/*
|
||||
tmp/sockets/*
|
||||
tmp/test/*
|
||||
|
||||
4
Gemfile
4
Gemfile
@@ -71,8 +71,10 @@ end
|
||||
|
||||
group :test do
|
||||
gem "shoulda", "~> 2.10.3"
|
||||
# Shoulda does not work nice on Ruby 1.9.3 and seems to need test-unit explicitely.
|
||||
gem "test-unit", :platforms => [:mri_19]
|
||||
gem "edavis10-object_daddy", :require => "object_daddy"
|
||||
gem "mocha"
|
||||
gem "mocha", "0.12.3"
|
||||
end
|
||||
|
||||
local_gemfile = File.join(File.dirname(__FILE__), "Gemfile.local")
|
||||
|
||||
@@ -56,7 +56,7 @@ class ApplicationController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
before_filter :user_setup, :check_if_login_required, :set_localization
|
||||
before_filter :session_expiration, :user_setup, :check_if_login_required, :set_localization
|
||||
filter_parameter_logging :password
|
||||
|
||||
rescue_from ActionController::InvalidAuthenticityToken, :with => :invalid_authenticity_token
|
||||
@@ -70,6 +70,38 @@ class ApplicationController < ActionController::Base
|
||||
require_dependency "repository/#{scm.underscore}"
|
||||
end
|
||||
|
||||
def session_expiration
|
||||
if session[:user_id]
|
||||
if session_expired? && !try_to_autologin
|
||||
reset_session
|
||||
flash[:error] = l(:error_session_expired)
|
||||
redirect_to signin_url
|
||||
else
|
||||
session[:atime] = Time.now.utc.to_i
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def session_expired?
|
||||
if Setting.session_lifetime?
|
||||
unless session[:ctime] && (Time.now.utc.to_i - session[:ctime].to_i <= Setting.session_lifetime.to_i * 60)
|
||||
return true
|
||||
end
|
||||
end
|
||||
if Setting.session_timeout?
|
||||
unless session[:atime] && (Time.now.utc.to_i - session[:atime].to_i <= Setting.session_timeout.to_i * 60)
|
||||
return true
|
||||
end
|
||||
end
|
||||
false
|
||||
end
|
||||
|
||||
def start_user_session(user)
|
||||
session[:user_id] = user.id
|
||||
session[:ctime] = Time.now.utc.to_i
|
||||
session[:atime] = Time.now.utc.to_i
|
||||
end
|
||||
|
||||
def user_setup
|
||||
# Check the settings cache for each request
|
||||
Setting.check_cache
|
||||
@@ -83,10 +115,7 @@ class ApplicationController < ActionController::Base
|
||||
if session[:user_id]
|
||||
# existing session
|
||||
(User.active.find(session[:user_id]) rescue nil)
|
||||
elsif cookies[:autologin] && Setting.autologin?
|
||||
# auto-login feature starts a new session
|
||||
user = User.try_to_autologin(cookies[:autologin])
|
||||
session[:user_id] = user.id if user
|
||||
elsif user = try_to_autologin
|
||||
user
|
||||
elsif params[:format] == 'atom' && params[:key] && request.get? && accept_rss_auth?
|
||||
# RSS key authentication does not start a session
|
||||
@@ -104,12 +133,24 @@ class ApplicationController < ActionController::Base
|
||||
end
|
||||
end
|
||||
|
||||
def try_to_autologin
|
||||
if cookies[:autologin] && Setting.autologin?
|
||||
# auto-login feature starts a new session
|
||||
user = User.try_to_autologin(cookies[:autologin])
|
||||
if user
|
||||
reset_session
|
||||
start_user_session(user)
|
||||
end
|
||||
user
|
||||
end
|
||||
end
|
||||
|
||||
# Sets the logged in user
|
||||
def logged_user=(user)
|
||||
reset_session
|
||||
if user && user.is_a?(User)
|
||||
User.current = user
|
||||
session[:user_id] = user.id
|
||||
start_user_session(user)
|
||||
else
|
||||
User.current = User.anonymous
|
||||
end
|
||||
@@ -458,9 +499,9 @@ class ApplicationController < ActionController::Base
|
||||
# Returns the API key present in the request
|
||||
def api_key_from_request
|
||||
if params[:key].present?
|
||||
params[:key]
|
||||
params[:key].to_s
|
||||
elsif request.headers["X-Redmine-API-Key"].present?
|
||||
request.headers["X-Redmine-API-Key"]
|
||||
request.headers["X-Redmine-API-Key"].to_s
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -39,7 +39,7 @@ class TimelogController < ApplicationController
|
||||
|
||||
def index
|
||||
sort_init 'spent_on', 'desc'
|
||||
sort_update 'spent_on' => 'spent_on',
|
||||
sort_update 'spent_on' => ['spent_on', "#{TimeEntry.table_name}.created_on"],
|
||||
'user' => 'user_id',
|
||||
'activity' => 'activity_id',
|
||||
'project' => "#{Project.table_name}.name",
|
||||
|
||||
@@ -799,7 +799,7 @@ module ApplicationHelper
|
||||
end
|
||||
end
|
||||
|
||||
HEADING_RE = /(<h(1|2|3|4)( [^>]+)?>(.+?)<\/h(1|2|3|4)>)/i unless const_defined?(:HEADING_RE)
|
||||
HEADING_RE = /(<h(\d)( [^>]+)?>(.+?)<\/h(\d)>)/i unless const_defined?(:HEADING_RE)
|
||||
|
||||
def parse_sections(text, project, obj, attr, only_path, options)
|
||||
return unless options[:edit_section_links]
|
||||
@@ -870,6 +870,8 @@ module ApplicationHelper
|
||||
# Renders the TOC with given headings
|
||||
def replace_toc(text, headings)
|
||||
text.gsub!(TOC_RE) do
|
||||
# Keep only the 4 first levels
|
||||
headings = headings.select{|level, anchor, item| level <= 4}
|
||||
if headings.empty?
|
||||
''
|
||||
else
|
||||
|
||||
@@ -24,6 +24,7 @@ class Attachment < ActiveRecord::Base
|
||||
validates_presence_of :filename, :author
|
||||
validates_length_of :filename, :maximum => 255
|
||||
validates_length_of :disk_filename, :maximum => 255
|
||||
validates_length_of :description, :maximum => 255, :allow_blank => true
|
||||
validate :validate_max_file_size
|
||||
|
||||
acts_as_event :title => :filename,
|
||||
|
||||
@@ -42,7 +42,9 @@ class Document < ActiveRecord::Base
|
||||
def initialize(attributes=nil, *args)
|
||||
super
|
||||
if new_record?
|
||||
self.category ||= DocumentCategory.default
|
||||
# Rails3 use this instead
|
||||
# self.category ||= DocumentCategory.default
|
||||
self.category_id = DocumentCategory.default.id if self.category_id == 0
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -31,4 +31,12 @@ class DocumentCategory < Enumeration
|
||||
def transfer_relations(to)
|
||||
documents.update_all("category_id = #{to.id}")
|
||||
end
|
||||
|
||||
def self.default
|
||||
d = super
|
||||
if d.nil?
|
||||
d = find(:first)
|
||||
end
|
||||
d
|
||||
end
|
||||
end
|
||||
|
||||
@@ -123,6 +123,28 @@ class Issue < ActiveRecord::Base
|
||||
end
|
||||
end
|
||||
|
||||
# AR#Base#destroy would raise and StaleObjectError exception
|
||||
# if the issue was already deleted or updated (non matching lock_version).
|
||||
# This is a problem when bulk deleting issues or deleting a project
|
||||
# (because an issue may already be deleted if its parent was deleted
|
||||
# first).
|
||||
# The issue is reloaded by the nested_set before being deleted so
|
||||
# the lock_version condition should not be an issue but we handle it.
|
||||
def destroy
|
||||
super
|
||||
rescue ActiveRecord::StaleObjectError
|
||||
# Stale or already deleted
|
||||
begin
|
||||
reload
|
||||
rescue ActiveRecord::RecordNotFound
|
||||
# The issue was actually already deleted
|
||||
@destroyed = true
|
||||
return freeze
|
||||
end
|
||||
# The issue was stale, retry to destroy
|
||||
super
|
||||
end
|
||||
|
||||
# Overrides Redmine::Acts::Customizable::InstanceMethods#available_custom_fields
|
||||
def available_custom_fields
|
||||
(project && tracker) ? (project.all_issue_custom_fields & tracker.custom_fields.all) : []
|
||||
|
||||
@@ -44,8 +44,8 @@ class MailHandler < ActionMailer::Base
|
||||
|
||||
cattr_accessor :ignored_emails_headers
|
||||
@@ignored_emails_headers = {
|
||||
'X-Auto-Response-Suppress' => 'OOF',
|
||||
'Auto-Submitted' => 'auto-replied'
|
||||
'X-Auto-Response-Suppress' => 'oof',
|
||||
'Auto-Submitted' => /^auto-/
|
||||
}
|
||||
|
||||
# Processes incoming emails
|
||||
@@ -63,11 +63,14 @@ class MailHandler < ActionMailer::Base
|
||||
# Ignore auto generated emails
|
||||
self.class.ignored_emails_headers.each do |key, ignored_value|
|
||||
value = email.header_string(key)
|
||||
if value && value.to_s.downcase == ignored_value.downcase
|
||||
if logger && logger.info
|
||||
logger.info "MailHandler: ignoring email with #{key}:#{value} header"
|
||||
if value
|
||||
value = value.to_s.downcase
|
||||
if (ignored_value.is_a?(Regexp) && value.match(ignored_value)) || value == ignored_value
|
||||
if logger && logger.info
|
||||
logger.info "MailHandler: ignoring email with #{key}:#{value} header"
|
||||
end
|
||||
return false
|
||||
end
|
||||
return false
|
||||
end
|
||||
end
|
||||
@user = User.find_by_mail(sender_email) if sender_email.present?
|
||||
|
||||
@@ -174,9 +174,9 @@ class Query < ActiveRecord::Base
|
||||
if values_for(field)
|
||||
case type_for(field)
|
||||
when :integer
|
||||
add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^\d+$/) }
|
||||
add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^[+-]?\d+$/) }
|
||||
when :float
|
||||
add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^\d+(\.\d*)?$/) }
|
||||
add_filter_error(field, :invalid) if values_for(field).detect {|v| v.present? && !v.match(/^[+-]?\d+(\.\d*)?$/) }
|
||||
when :date, :date_past
|
||||
case operator_for(field)
|
||||
when "=", ">=", "<=", "><"
|
||||
|
||||
@@ -37,8 +37,8 @@ class Repository < ActiveRecord::Base
|
||||
validates_presence_of :identifier, :unless => Proc.new { |r| r.is_default? || r.set_as_default? }
|
||||
validates_uniqueness_of :identifier, :scope => :project_id, :allow_blank => true
|
||||
validates_exclusion_of :identifier, :in => %w(show entry raw changes annotate diff show stats graph)
|
||||
# donwcase letters, digits, dashes but not digits only
|
||||
validates_format_of :identifier, :with => /^(?!\d+$)[a-z0-9\-]*$/, :allow_blank => true
|
||||
# donwcase letters, digits, dashes, underscores but not digits only
|
||||
validates_format_of :identifier, :with => /^(?!\d+$)[a-z0-9\-_]*$/, :allow_blank => true
|
||||
# Checks if the SCM is enabled when creating a repository
|
||||
validate :repo_create_validation, :on => :create
|
||||
|
||||
|
||||
@@ -66,7 +66,7 @@ class TimeEntry < ActiveRecord::Base
|
||||
end
|
||||
}
|
||||
|
||||
safe_attributes 'hours', 'comments', 'issue_id', 'activity_id', 'spent_on', 'custom_field_values'
|
||||
safe_attributes 'hours', 'comments', 'issue_id', 'activity_id', 'spent_on', 'custom_field_values', 'custom_fields'
|
||||
|
||||
def initialize(attributes=nil, *args)
|
||||
super
|
||||
|
||||
@@ -130,8 +130,11 @@ class User < Principal
|
||||
|
||||
# Returns the user that matches provided login and password, or nil
|
||||
def self.try_to_login(login, password)
|
||||
login = login.to_s
|
||||
password = password.to_s
|
||||
|
||||
# Make sure no one can sign in with an empty password
|
||||
return nil if password.to_s.empty?
|
||||
return nil if password.empty?
|
||||
user = find_by_login(login)
|
||||
if user
|
||||
# user is already in local database
|
||||
@@ -164,7 +167,7 @@ class User < Principal
|
||||
|
||||
# Returns the user who matches the given autologin +key+ or nil
|
||||
def self.try_to_autologin(key)
|
||||
tokens = Token.find_all_by_action_and_value('autologin', key)
|
||||
tokens = Token.find_all_by_action_and_value('autologin', key.to_s)
|
||||
# Make sure there's only 1 token that matches the key
|
||||
if tokens.size == 1
|
||||
token = tokens.first
|
||||
@@ -338,12 +341,12 @@ class User < Principal
|
||||
end
|
||||
|
||||
def self.find_by_rss_key(key)
|
||||
token = Token.find_by_value(key)
|
||||
token = Token.find_by_action_and_value('feeds', key.to_s)
|
||||
token && token.user.active? ? token.user : nil
|
||||
end
|
||||
|
||||
def self.find_by_api_key(key)
|
||||
token = Token.find_by_action_and_value('api', key)
|
||||
token = Token.find_by_action_and_value('api', key.to_s)
|
||||
token && token.user.active? ? token.user : nil
|
||||
end
|
||||
|
||||
|
||||
@@ -10,7 +10,7 @@
|
||||
<span>
|
||||
<%= file_field_tag 'attachments[1][file]', :size => 30, :id => nil, :class => 'file',
|
||||
:onchange => "checkFileSize(this, #{Setting.attachment_max_size.to_i.kilobytes}, '#{escape_javascript(l(:error_attachment_too_big, :max_size => number_to_human_size(Setting.attachment_max_size.to_i.kilobytes)))}');" -%>
|
||||
<%= text_field_tag 'attachments[1][description]', '', :id => nil, :class => 'description', :placeholder => l(:label_optional_description) %>
|
||||
<%= text_field_tag 'attachments[1][description]', '', :id => nil, :class => 'description', :maxlength => 255, :placeholder => l(:label_optional_description) %>
|
||||
<%= link_to_function(image_tag('delete.png'), 'removeFileField(this)', :title => (l(:button_delete))) %>
|
||||
</span>
|
||||
</span>
|
||||
|
||||
@@ -1,8 +1,9 @@
|
||||
<h2><%= l(:label_revision) %> <%= @diff_format_revisions %> <%=h @path %></h2>
|
||||
|
||||
<!-- Choose view type -->
|
||||
<% form_tag({:path => to_path_param(@path)}, :method => 'get') do %>
|
||||
<%= hidden_field_tag('rev', params[:rev]) if params[:rev] %>
|
||||
<% form_tag({:action => 'diff', :id => @project,
|
||||
:repository_id => @repository.identifier_param,
|
||||
:path => to_path_param(@path), :rev => @rev}, :method => 'get') do %>
|
||||
<%= hidden_field_tag('rev_to', params[:rev_to]) if params[:rev_to] %>
|
||||
<p>
|
||||
<label><%= l(:label_view_diff) %></label>
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
<div class="contextual">
|
||||
<% form_tag(
|
||||
{:action => 'revision', :id => @project,
|
||||
:repository_id => @repository.identifier_param}
|
||||
:repository_id => @repository.identifier_param},
|
||||
:method => :get
|
||||
) do %>
|
||||
<%= l(:label_revision) %>: <%= text_field_tag 'rev', @rev, :size => 8 %>
|
||||
<%= submit_tag 'OK' %>
|
||||
|
||||
@@ -21,5 +21,16 @@
|
||||
<p><%= setting_check_box :rest_api_enabled %></p>
|
||||
</div>
|
||||
|
||||
<fieldset class="box">
|
||||
<legend><%= l(:label_session_expiration) %></legend>
|
||||
|
||||
<div class="tabular settings">
|
||||
<p><%= setting_select :session_lifetime, [[l(:label_disabled), 0]] + [1, 7, 30, 60, 365].collect{|days| [l('datetime.distance_in_words.x_days', :count => days), (days * 60 * 24).to_s]} %></p>
|
||||
<p><%= setting_select :session_timeout, [[l(:label_disabled), 0]] + [1, 2, 4, 8, 12, 24, 48].collect{|hours| [l('datetime.distance_in_words.x_hours', :count => hours), (hours * 60).to_s]} %></p>
|
||||
</div>
|
||||
|
||||
<p><em class="info"><%= l(:text_session_expiration_settings) %></em></p>
|
||||
</fieldset>
|
||||
|
||||
<%= submit_tag l(:button_save) %>
|
||||
<% end %>
|
||||
|
||||
@@ -8,6 +8,7 @@ api.array :versions, api_meta(:total_count => @versions.size) do
|
||||
api.description version.description
|
||||
api.status version.status
|
||||
api.due_date version.effective_date
|
||||
api.sharing version.sharing
|
||||
|
||||
render_api_custom_values version.custom_field_values, api
|
||||
|
||||
|
||||
@@ -6,6 +6,7 @@ api.version do
|
||||
api.description @version.description
|
||||
api.status @version.status
|
||||
api.due_date @version.effective_date
|
||||
api.sharing @version.sharing
|
||||
|
||||
render_api_custom_values @version.custom_field_values, api
|
||||
|
||||
|
||||
@@ -88,22 +88,25 @@ module ActionController
|
||||
|
||||
# CVE-2012-2660
|
||||
# https://groups.google.com/group/rubyonrails-security/browse_thread/thread/f1203e3376acec0f
|
||||
# CVE-2012-2694
|
||||
# https://groups.google.com/group/rubyonrails-security/browse_thread/thread/8c82d9df8b401c5e
|
||||
class Request
|
||||
protected
|
||||
|
||||
# Remove nils from the params hash
|
||||
def deep_munge(hash)
|
||||
keys = hash.keys.find_all { |k| hash[k] == [nil] }
|
||||
keys.each { |k| hash[k] = nil }
|
||||
|
||||
hash.each_value do |v|
|
||||
case v
|
||||
when Array
|
||||
v.grep(Hash) { |x| deep_munge(x) }
|
||||
v.compact!
|
||||
when Hash
|
||||
deep_munge(v)
|
||||
end
|
||||
end
|
||||
|
||||
keys = hash.keys.find_all { |k| hash[k] == [nil] }
|
||||
keys.each { |k| hash[k] = nil }
|
||||
hash
|
||||
end
|
||||
|
||||
@@ -112,3 +115,40 @@ module ActionController
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# CVE-2012-2695
|
||||
# https://groups.google.com/group/rubyonrails-security/browse_thread/thread/9782f44c4540cf59
|
||||
module ActiveRecord
|
||||
class Base
|
||||
class << self
|
||||
def sanitize_sql_hash_for_conditions(attrs, default_table_name = quoted_table_name, top_level = true)
|
||||
attrs = expand_hash_conditions_for_aggregates(attrs)
|
||||
|
||||
conditions = attrs.map do |attr, value|
|
||||
table_name = default_table_name
|
||||
|
||||
if not value.is_a?(Hash)
|
||||
attr = attr.to_s
|
||||
|
||||
# Extract table name from qualified attribute names.
|
||||
if attr.include?('.') and top_level
|
||||
attr_table_name, attr = attr.split('.', 2)
|
||||
attr_table_name = connection.quote_table_name(attr_table_name)
|
||||
else
|
||||
attr_table_name = table_name
|
||||
end
|
||||
|
||||
attribute_condition("#{attr_table_name}.#{connection.quote_column_name(attr)}", value)
|
||||
elsif top_level
|
||||
sanitize_sql_hash_for_conditions(value, connection.quote_table_name(attr.to_s), false)
|
||||
else
|
||||
raise ActiveRecord::StatementInvalid
|
||||
end
|
||||
end.join(' AND ')
|
||||
|
||||
replace_bind_variables(conditions, expand_range_bind_variables(attrs.values))
|
||||
end
|
||||
alias_method :sanitize_sql_hash, :sanitize_sql_hash_for_conditions
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -49,6 +49,9 @@ ar:
|
||||
about_x_hours:
|
||||
one: "حوالي ساعة"
|
||||
other: "ساعات %{count}حوالي "
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "يوم"
|
||||
other: "%{count} أيام"
|
||||
@@ -1031,3 +1034,8 @@ ar:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -50,6 +50,9 @@ bg:
|
||||
about_x_hours:
|
||||
one: "около 1 час"
|
||||
other: "около %{count} часа"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 ден"
|
||||
other: "%{count} дена"
|
||||
@@ -195,6 +198,7 @@ bg:
|
||||
error_unable_delete_issue_status: Невъзможност за изтриване на състояние на задача
|
||||
error_unable_to_connect: Невъзможност за свързване с (%{value})
|
||||
error_attachment_too_big: Този файл не може да бъде качен, понеже надхвърля максималната възможна големина (%{max_size})
|
||||
error_session_expired: Вашата сесия е изтекла. Моля влезете в Redmine отново.
|
||||
warning_attachments_not_saved: "%{count} файла не бяха записани."
|
||||
|
||||
mail_subject_lost_password: "Вашата парола (%{value})"
|
||||
@@ -386,6 +390,8 @@ bg:
|
||||
setting_default_issue_start_date_to_creation_date: Начална дата на новите задачи по подразбиране да бъде днешната дата
|
||||
setting_commit_cross_project_ref: Отбелязване и приключване на задачи от други проекти, несвързани с конкретното хранилище
|
||||
setting_unsubscribe: Потребителите могат да премахват профилите си
|
||||
setting_session_lifetime: Максимален живот на сесиите
|
||||
setting_session_timeout: Таймаут за неактивност преди прекратяване на сесиите
|
||||
|
||||
permission_add_project: Създаване на проект
|
||||
permission_add_subprojects: Създаване на подпроекти
|
||||
@@ -848,6 +854,7 @@ bg:
|
||||
label_item_position: "%{position}/%{count}"
|
||||
label_completed_versions: Завършени версии
|
||||
label_search_for_watchers: Търсене на потребители за наблюдатели
|
||||
label_session_expiration: Изтичане на сесиите
|
||||
|
||||
button_login: Вход
|
||||
button_submit: Прикачване
|
||||
@@ -983,6 +990,7 @@ bg:
|
||||
text_issue_conflict_resolution_add_notes: Добавяне на моите коментари и отхвърляне на другите мои промени
|
||||
text_issue_conflict_resolution_cancel: Отхвърляне на всички мои промени и презареждане на %{link}
|
||||
text_account_destroy_confirmation: "Сигурен/на ли сте, че желаете да продължите?\nВашият профил ще бъде премахнат без възможност за възстановяване."
|
||||
text_session_expiration_settings: "Внимание: промяната на тези установяваноя може да прекрати всички активни сесии, включително и вашата."
|
||||
|
||||
default_role_manager: Мениджър
|
||||
default_role_developer: Разработчик
|
||||
|
||||
@@ -48,6 +48,9 @@ bs:
|
||||
about_x_hours:
|
||||
one: "oko 1 sahat"
|
||||
other: "oko %{count} sahata"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dan"
|
||||
other: "%{count} dana"
|
||||
@@ -1045,3 +1048,8 @@ bs:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -52,6 +52,9 @@ ca:
|
||||
about_x_hours:
|
||||
one: "aproximadament 1 hora"
|
||||
other: "aproximadament %{count} hores"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dia"
|
||||
other: "%{count} dies"
|
||||
@@ -1033,3 +1036,8 @@ ca:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -53,6 +53,9 @@ cs:
|
||||
about_x_hours:
|
||||
one: "asi 1 hodina"
|
||||
other: "asi %{count} hodin"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 den"
|
||||
other: "%{count} dnů"
|
||||
@@ -1034,3 +1037,8 @@ cs:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -51,6 +51,9 @@ da:
|
||||
about_x_hours:
|
||||
one: "cirka en time"
|
||||
other: "cirka %{count} timer"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "en dag"
|
||||
other: "%{count} dage"
|
||||
@@ -1048,3 +1051,8 @@ da:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -52,6 +52,9 @@ de:
|
||||
about_x_hours:
|
||||
one: 'etwa 1 Stunde'
|
||||
other: 'etwa %{count} Stunden'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: '1 Tag'
|
||||
other: '%{count} Tagen'
|
||||
@@ -1016,8 +1019,8 @@ de:
|
||||
description_date_from: Startdatum eintragen
|
||||
description_date_to: Enddatum eintragen
|
||||
|
||||
label_parent_revision: Parent
|
||||
label_child_revision: Child
|
||||
label_parent_revision: Vorgänger
|
||||
label_child_revision: Nachfolger
|
||||
error_scm_annotate_big_text_file: Der Eintrag kann nicht umgesetzt werden, da er die maximale Textlänge überschreitet.
|
||||
setting_default_issue_start_date_to_creation_date: Aktuelles Datum als Beginn für neue Tickets verwenden
|
||||
button_edit_section: Diesen Bereich bearbeiten
|
||||
@@ -1036,7 +1039,7 @@ de:
|
||||
label_copy_attachments: Anhänge Kopieren
|
||||
label_item_position: "%{position}/%{count}"
|
||||
label_completed_versions: Abgeschlossene Versionen
|
||||
field_multiple: Mehrer Werte
|
||||
field_multiple: Mehrere Werte
|
||||
setting_commit_cross_project_ref: Erlauben auf Tickets aller anderen Projekte zu referenzieren
|
||||
text_issue_conflict_resolution_add_notes: Meine Änderungen übernehmen und alle anderen Änderungen verwerfen
|
||||
text_issue_conflict_resolution_overwrite: Meine Änderungen trotzdem übernehmen (vorherige Notizen bleiben erhalten aber manche können überschrieben werden)
|
||||
@@ -1049,3 +1052,8 @@ de:
|
||||
setting_unsubscribe: Erlaubt Benutzern das eigene Benutzerkonto zu löschen
|
||||
button_delete_my_account: Mein Benutzerkonto löschen
|
||||
text_account_destroy_confirmation: Möchten Sie wirklich fortfahren?\nIhr Benutzerkonto wird für immer gelöscht und kann nicht wiederhergestellt werden.
|
||||
error_session_expired: Ihre Sitzung ist abgelaufen. Bitte melden Sie sich erneut an.
|
||||
text_session_expiration_settings: "Achtung: Änderungen können aktuelle Sitzungen beenden, Ihre eingeschlossen!"
|
||||
setting_session_lifetime: Längste Dauer einer Sitzung
|
||||
setting_session_timeout: Zeitüberschreitung bei Inaktivität
|
||||
label_session_expiration: Ende einer Sitzung
|
||||
|
||||
@@ -51,6 +51,9 @@ el:
|
||||
about_x_hours:
|
||||
one: "περίπου 1 ώρα"
|
||||
other: "περίπου %{count} ώρες"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 ημέρα"
|
||||
other: "%{count} ημέρες"
|
||||
@@ -1031,3 +1034,8 @@ el:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ en-GB:
|
||||
about_x_hours:
|
||||
one: "about 1 hour"
|
||||
other: "about %{count} hours"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 day"
|
||||
other: "%{count} days"
|
||||
@@ -1033,3 +1036,8 @@ en-GB:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -49,6 +49,9 @@ en:
|
||||
about_x_hours:
|
||||
one: "about 1 hour"
|
||||
other: "about %{count} hours"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 day"
|
||||
other: "%{count} days"
|
||||
@@ -194,6 +197,7 @@ en:
|
||||
error_unable_delete_issue_status: 'Unable to delete issue status'
|
||||
error_unable_to_connect: "Unable to connect (%{value})"
|
||||
error_attachment_too_big: "This file cannot be uploaded because it exceeds the maximum allowed file size (%{max_size})"
|
||||
error_session_expired: "Your session has expired. Please login again."
|
||||
warning_attachments_not_saved: "%{count} file(s) could not be saved."
|
||||
|
||||
mail_subject_lost_password: "Your %{value} password"
|
||||
@@ -385,6 +389,8 @@ en:
|
||||
setting_default_issue_start_date_to_creation_date: Use current date as start date for new issues
|
||||
setting_commit_cross_project_ref: Allow issues of all the other projects to be referenced and fixed
|
||||
setting_unsubscribe: Allow users to delete their own account
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
|
||||
permission_add_project: Create project
|
||||
permission_add_subprojects: Create subprojects
|
||||
@@ -847,6 +853,7 @@ en:
|
||||
label_item_position: "%{position} of %{count}"
|
||||
label_completed_versions: Completed versions
|
||||
label_search_for_watchers: Search for watchers to add
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
button_login: Login
|
||||
button_submit: Submit
|
||||
@@ -982,6 +989,7 @@ en:
|
||||
text_issue_conflict_resolution_add_notes: "Add my notes and discard my other changes"
|
||||
text_issue_conflict_resolution_cancel: "Discard all my changes and redisplay %{link}"
|
||||
text_account_destroy_confirmation: "Are you sure you want to proceed?\nYour account will be permanently deleted, with no way to reactivate it."
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
|
||||
default_role_manager: Manager
|
||||
default_role_developer: Developer
|
||||
|
||||
@@ -79,6 +79,9 @@ es:
|
||||
about_x_hours:
|
||||
one: "alrededor de 1 hora"
|
||||
other: "alrededor de %{count} horas"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 día"
|
||||
other: "%{count} días"
|
||||
@@ -1068,3 +1071,8 @@ es:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -67,6 +67,9 @@ et:
|
||||
about_x_hours:
|
||||
one: "umbes 1 tund"
|
||||
other: "umbes %{count} tundi"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 päev"
|
||||
other: "%{count} päeva"
|
||||
@@ -1047,3 +1050,8 @@ et:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -52,6 +52,9 @@ eu:
|
||||
about_x_hours:
|
||||
one: "ordu 1 inguru"
|
||||
other: "%{count} ordu inguru"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "egun 1"
|
||||
other: "%{count} egun"
|
||||
@@ -1034,3 +1037,8 @@ eu:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -49,6 +49,9 @@ fa:
|
||||
about_x_hours:
|
||||
one: "نزدیک 1 ساعت"
|
||||
other: "نزدیک %{count} ساعت"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 روز"
|
||||
other: "%{count} روز"
|
||||
@@ -1033,3 +1036,8 @@ fa:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -94,6 +94,9 @@ fi:
|
||||
about_x_hours:
|
||||
one: "noin tunti"
|
||||
other: "noin %{count} tuntia"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "päivä"
|
||||
other: "%{count} päivää"
|
||||
@@ -1052,3 +1055,8 @@ fi:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -53,6 +53,9 @@ fr:
|
||||
about_x_hours:
|
||||
one: "environ une heure"
|
||||
other: "environ %{count} heures"
|
||||
x_hours:
|
||||
one: "une heure"
|
||||
other: "%{count} heures"
|
||||
x_days:
|
||||
one: "un jour"
|
||||
other: "%{count} jours"
|
||||
@@ -201,6 +204,7 @@ fr:
|
||||
error_workflow_copy_target: 'Veuillez sélectionner les trackers et rôles cibles'
|
||||
error_issue_done_ratios_not_updated: L'avancement des demandes n'a pas pu être mis à jour.
|
||||
error_attachment_too_big: Ce fichier ne peut pas être attaché car il excède la taille maximale autorisée (%{max_size})
|
||||
error_session_expired: "Votre session a expiré. Veuillez vous reconnecter."
|
||||
|
||||
warning_attachments_not_saved: "%{count} fichier(s) n'ont pas pu être sauvegardés."
|
||||
|
||||
@@ -381,6 +385,8 @@ fr:
|
||||
setting_default_issue_start_date_to_creation_date: Donner à la date de début d'une nouvelle demande la valeur de la date du jour
|
||||
setting_commit_cross_project_ref: Permettre le référencement et la résolution des demandes de tous les autres projets
|
||||
setting_unsubscribe: Permettre aux utilisateurs de supprimer leur propre compte
|
||||
setting_session_lifetime: Durée de vie maximale des sessions
|
||||
setting_session_timeout: Durée maximale d'inactivité
|
||||
|
||||
permission_add_project: Créer un projet
|
||||
permission_add_subprojects: Créer des sous-projets
|
||||
@@ -822,6 +828,7 @@ fr:
|
||||
label_copy_attachments: Copier les fichiers
|
||||
label_item_position: "%{position} sur %{count}"
|
||||
label_completed_versions: Versions passées
|
||||
label_session_expiration: Expiration des sessions
|
||||
|
||||
button_login: Connexion
|
||||
button_submit: Soumettre
|
||||
@@ -938,6 +945,7 @@ fr:
|
||||
text_issue_conflict_resolution_add_notes: "Ajouter mes notes et ignorer mes autres changements"
|
||||
text_issue_conflict_resolution_cancel: "Annuler ma mise à jour et réafficher %{link}"
|
||||
text_account_destroy_confirmation: "Êtes-vous sûr de vouloir continuer ?\nVotre compte sera définitivement supprimé, sans aucune possibilité de le réactiver."
|
||||
text_session_expiration_settings: "Attention : le changement de ces paramètres peut entrainer l'expiration des sessions utilisateurs en cours, y compris la vôtre."
|
||||
|
||||
default_role_manager: "Manager "
|
||||
default_role_developer: "Développeur "
|
||||
|
||||
@@ -90,6 +90,9 @@ gl:
|
||||
about_x_hours:
|
||||
one: 'aproximadamente unha hora'
|
||||
other: '%{count} horas'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: '1 día'
|
||||
other: '%{count} días'
|
||||
@@ -1042,3 +1045,8 @@ gl:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -55,6 +55,9 @@ he:
|
||||
about_x_hours:
|
||||
one: 'בערך שעה אחת'
|
||||
other: 'בערך %{count} שעות'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: 'יום אחד'
|
||||
other: '%{count} ימים'
|
||||
@@ -1036,3 +1039,8 @@ he:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ hr:
|
||||
about_x_hours:
|
||||
one: "oko sat vremena"
|
||||
other: "oko %{count} sati"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dan"
|
||||
other: "%{count} dana"
|
||||
@@ -1034,3 +1037,8 @@ hr:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -50,6 +50,9 @@
|
||||
about_x_hours:
|
||||
one: 'csaknem 1 órája'
|
||||
other: 'csaknem %{count} órája'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: '1 napja'
|
||||
other: '%{count} napja'
|
||||
@@ -1050,3 +1053,8 @@
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -46,6 +46,9 @@ id:
|
||||
about_x_hours:
|
||||
one: "sekitar sejam"
|
||||
other: "sekitar %{count} jam"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "sehari"
|
||||
other: "%{count} hari"
|
||||
@@ -1037,3 +1040,8 @@ id:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -54,6 +54,9 @@ it:
|
||||
about_x_hours:
|
||||
one: "circa un'ora"
|
||||
other: "circa %{count} ore"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 giorno"
|
||||
other: "%{count} giorni"
|
||||
@@ -1032,3 +1035,8 @@ it:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -52,6 +52,9 @@ ja:
|
||||
about_x_hours:
|
||||
one: "約1時間"
|
||||
other: "約%{count}時間"
|
||||
x_hours:
|
||||
one: "1時間"
|
||||
other: "%{count}時間"
|
||||
x_days:
|
||||
one: "1日"
|
||||
other: "%{count}日"
|
||||
@@ -758,7 +761,7 @@ ja:
|
||||
label_week: 週
|
||||
label_date_from: "日付指定: "
|
||||
label_date_to: から
|
||||
label_language_based: 既定の言語の設定に従う
|
||||
label_language_based: ユーザの言語の設定に従う
|
||||
label_sort_by: "並び替え %{value}"
|
||||
label_send_test_email: テストメールを送信
|
||||
label_feeds_access_key: RSSアクセスキー
|
||||
@@ -1061,3 +1064,8 @@ ja:
|
||||
text_account_destroy_confirmation: |-
|
||||
本当にアカウントを削除しますか?
|
||||
アカウントは恒久的に削除されます。削除後に再度アカウントを有効にする手段はありません。
|
||||
error_session_expired: セッションが失効しました。ログインし直してください。
|
||||
text_session_expiration_settings: "警告: この設定を変更すると現在有効なセッションが失効する可能性があります。"
|
||||
setting_session_lifetime: 有効期間の最大値
|
||||
setting_session_timeout: 無操作タイムアウト
|
||||
label_session_expiration: セッション有効期間
|
||||
|
||||
@@ -49,6 +49,9 @@ ko:
|
||||
about_x_hours:
|
||||
one: "약 한시간"
|
||||
other: "약 %{count}시간"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "하루"
|
||||
other: "%{count}일"
|
||||
@@ -1081,3 +1084,8 @@ ko:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -58,6 +58,9 @@ lt:
|
||||
about_x_hours:
|
||||
one: "apie 1 valanda"
|
||||
other: "apie %{count} valandų"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 diena"
|
||||
other: "%{count} dienų"
|
||||
@@ -1091,3 +1094,8 @@ lt:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -45,6 +45,9 @@ lv:
|
||||
about_x_hours:
|
||||
one: "aptuveni 1 stunda"
|
||||
other: "aptuveni %{count} stundas"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 diena"
|
||||
other: "%{count} dienas"
|
||||
@@ -1025,3 +1028,8 @@ lv:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -49,6 +49,9 @@ mk:
|
||||
about_x_hours:
|
||||
one: "околу 1 час"
|
||||
other: "околу %{count} часа"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 ден"
|
||||
other: "%{count} дена"
|
||||
@@ -1031,3 +1034,8 @@ mk:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ mn:
|
||||
about_x_hours:
|
||||
one: "1 цаг орчим"
|
||||
other: "ойролцоогоор %{count} цаг"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 өдөр"
|
||||
other: "%{count} өдөр"
|
||||
@@ -1031,3 +1034,8 @@ mn:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ nl:
|
||||
about_x_hours:
|
||||
one: "ongeveer 1 uur"
|
||||
other: "ongeveer %{count} uren"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dag"
|
||||
other: "%{count} dagen"
|
||||
@@ -1013,3 +1016,8 @@ nl:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -43,6 +43,9 @@
|
||||
about_x_hours:
|
||||
one: "rundt 1 time"
|
||||
other: "rundt %{count} timer"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dag"
|
||||
other: "%{count} dager"
|
||||
@@ -1021,3 +1024,8 @@
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -81,6 +81,9 @@ pl:
|
||||
about_x_hours:
|
||||
one: "około godziny"
|
||||
other: "około %{count} godzin"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dzień"
|
||||
other: "%{count} dni"
|
||||
@@ -1048,3 +1051,8 @@ pl:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -18,10 +18,10 @@ pt-BR:
|
||||
|
||||
time:
|
||||
formats:
|
||||
default: "%A, %d de %B de %Y, %H:%M hs"
|
||||
time: "%H:%M hs"
|
||||
short: "%d/%m, %H:%M hs"
|
||||
long: "%A, %d de %B de %Y, %H:%M hs"
|
||||
default: "%A, %d de %B de %Y, %H:%M h"
|
||||
time: "%H:%M h"
|
||||
short: "%d/%m, %H:%M h"
|
||||
long: "%A, %d de %B de %Y, %H:%M h"
|
||||
only_second: "%S"
|
||||
datetime:
|
||||
formats:
|
||||
@@ -52,6 +52,9 @@ pt-BR:
|
||||
about_x_hours:
|
||||
one: 'aproximadamente 1 hora'
|
||||
other: 'aproximadamente %{count} horas'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
|
||||
x_days:
|
||||
one: '1 dia'
|
||||
@@ -1053,3 +1056,8 @@ pt-BR:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -50,6 +50,9 @@ pt:
|
||||
about_x_hours:
|
||||
one: "aproximadamente 1 hora"
|
||||
other: "aproximadamente %{count} horas"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dia"
|
||||
other: "%{count} dias"
|
||||
@@ -1036,3 +1039,8 @@ pt:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -46,6 +46,9 @@ ro:
|
||||
about_x_hours:
|
||||
one: "aproximativ o oră"
|
||||
other: "aproximativ %{count} ore"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "o zi"
|
||||
other: "%{count} zile"
|
||||
@@ -1028,3 +1031,8 @@ ro:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -116,6 +116,9 @@ ru:
|
||||
few: "около %{count} часов"
|
||||
many: "около %{count} часов"
|
||||
other: "около %{count} часа"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "%{count} день"
|
||||
few: "%{count} дня"
|
||||
@@ -1089,11 +1092,11 @@ ru:
|
||||
notice_issue_successful_create: Задача %{id} создана.
|
||||
label_between: между
|
||||
setting_issue_group_assignment: Разрешить назначение задач группам пользователей
|
||||
label_diff: diff
|
||||
label_diff: Разница(diff)
|
||||
text_git_repository_note: Repository is bare and local (e.g. /gitrepo, c:\gitrepo)
|
||||
description_query_sort_criteria_direction: Sort direction
|
||||
description_query_sort_criteria_direction: Порядок сортировки
|
||||
description_project_scope: Search scope
|
||||
description_filter: Filter
|
||||
description_filter: Фильтр
|
||||
description_user_mail_notification: Mail notification settings
|
||||
description_date_from: Enter start date
|
||||
description_message_content: Message content
|
||||
@@ -1101,23 +1104,23 @@ ru:
|
||||
description_date_range_interval: Choose range by selecting start and end date
|
||||
description_issue_category_reassign: Choose issue category
|
||||
description_search: Searchfield
|
||||
description_notes: Notes
|
||||
description_date_range_list: Choose range from list
|
||||
description_choose_project: Projects
|
||||
description_date_to: Enter end date
|
||||
description_query_sort_criteria_attribute: Sort attribute
|
||||
description_wiki_subpages_reassign: Choose new parent page
|
||||
description_selected_columns: Selected Columns
|
||||
label_parent_revision: Parent
|
||||
label_child_revision: Child
|
||||
error_scm_annotate_big_text_file: The entry cannot be annotated, as it exceeds the maximum text file size.
|
||||
setting_default_issue_start_date_to_creation_date: Use current date as start date for new issues
|
||||
button_edit_section: Edit this section
|
||||
setting_repositories_encodings: Attachments and repositories encodings
|
||||
description_all_columns: All Columns
|
||||
button_export: Export
|
||||
label_export_options: "%{export_format} export options"
|
||||
error_attachment_too_big: This file cannot be uploaded because it exceeds the maximum allowed file size (%{max_size})
|
||||
description_notes: Примечания
|
||||
description_date_range_list: Выберите диапазон из списка
|
||||
description_choose_project: Проекты
|
||||
description_date_to: Введите дату выполнения
|
||||
description_query_sort_criteria_attribute: Критерий сортировки
|
||||
description_wiki_subpages_reassign: Выбрать новую родительскую страницу
|
||||
description_selected_columns: Выбранные столбцы
|
||||
label_parent_revision: Родительский
|
||||
label_child_revision: Дочерний
|
||||
error_scm_annotate_big_text_file: Комментарий невозможен из-за превышения максимального размера текстового файла.
|
||||
setting_default_issue_start_date_to_creation_date: Использовать текущую дату в качестве даты начала для новых задач
|
||||
button_edit_section: Редактировать эту секцию
|
||||
setting_repositories_encodings: Кодировка вложений и хранилищ
|
||||
description_all_columns: Все столбцы
|
||||
button_export: Экспорт
|
||||
label_export_options: "%{export_format} параметры экспорта"
|
||||
error_attachment_too_big: Этот файл нельзя загрузить из-за превышения максимального размера файла (%{max_size})
|
||||
notice_failed_to_save_time_entries: "Failed to save %{count} time entrie(s) on %{total} selected: %{ids}."
|
||||
label_x_issues:
|
||||
zero: 0 Задач
|
||||
@@ -1125,22 +1128,27 @@ ru:
|
||||
few: "%{count} Задач"
|
||||
many: "%{count} Задач"
|
||||
other: "%{count} Задач"
|
||||
label_repository_new: New repository
|
||||
field_repository_is_default: Main repository
|
||||
label_copy_attachments: Copy attachments
|
||||
label_repository_new: Новое хранилище
|
||||
field_repository_is_default: Хранилище по умолчанию
|
||||
label_copy_attachments: Копировать вложения
|
||||
label_item_position: "%{position}/%{count}"
|
||||
label_completed_versions: Completed versions
|
||||
text_project_identifier_info: Only lower case letters (a-z), numbers, dashes and underscores are allowed.<br />Once saved, the identifier cannot be changed.
|
||||
field_multiple: Multiple values
|
||||
setting_commit_cross_project_ref: Allow issues of all the other projects to be referenced and fixed
|
||||
text_issue_conflict_resolution_add_notes: Add my notes and discard my other changes
|
||||
text_issue_conflict_resolution_overwrite: Apply my changes anyway (previous notes will be kept but some changes may be overwritten)
|
||||
notice_issue_update_conflict: The issue has been updated by an other user while you were editing it.
|
||||
text_issue_conflict_resolution_cancel: Discard all my changes and redisplay %{link}
|
||||
permission_manage_related_issues: Manage related issues
|
||||
field_ldap_filter: LDAP filter
|
||||
label_search_for_watchers: Search for watchers to add
|
||||
label_completed_versions: Завершенные версии
|
||||
text_project_identifier_info: Допускаются только строчные латинские буквы (a-z), цифры, тире и подчеркивания.<br />После сохранения идентификатор изменить нельзя.
|
||||
field_multiple: Множественные значения
|
||||
setting_commit_cross_project_ref: Разрешить ссылаться и исправлять задачи во всех остальных проектах
|
||||
text_issue_conflict_resolution_add_notes: Добавить мои примечания и отказаться от моих изменений
|
||||
text_issue_conflict_resolution_overwrite: Применить мои изменения (все предыдущие замечания будут сохранены, но некоторые изменения могут быть перезаписаны)
|
||||
notice_issue_update_conflict: Кто-то изменил задачу, пока вы ее редактировали.
|
||||
text_issue_conflict_resolution_cancel: Отменить мои изменения и показать задачу заново %{link}
|
||||
permission_manage_related_issues: Управление связанными задачами
|
||||
field_ldap_filter: Фильтр LDAP
|
||||
label_search_for_watchers: Найти наблюдателей
|
||||
notice_account_deleted: "Ваша учетная запись полностью удалена"
|
||||
setting_unsubscribe: "Разрешить пользователям удалять свои учетные записи"
|
||||
button_delete_my_account: "Удалить мою учетную запись"
|
||||
text_account_destroy_confirmation: "Ваша учетная запись будет полностью удалена без возможности восстановления.\nВы уверены, что хотите продолжить?"
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ sk:
|
||||
about_x_hours:
|
||||
one: "okolo 1 hodiny"
|
||||
other: "okolo %{count} hodín"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 deň"
|
||||
other: "%{count} dní"
|
||||
@@ -1031,3 +1034,8 @@ sk:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -49,6 +49,9 @@ sl:
|
||||
about_x_hours:
|
||||
one: "okrog 1. ure"
|
||||
other: "okrog %{count} ur"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 dan"
|
||||
other: "%{count} dni"
|
||||
@@ -1031,3 +1034,8 @@ sl:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -49,6 +49,9 @@ sq:
|
||||
about_x_hours:
|
||||
one: "about 1 hour"
|
||||
other: "about %{count} hours"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 day"
|
||||
other: "%{count} days"
|
||||
@@ -1027,3 +1030,8 @@ sq:
|
||||
description_date_range_interval: Choose range by selecting start and end date
|
||||
description_date_from: Enter start date
|
||||
description_date_to: Enter end date
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -50,6 +50,9 @@ sr-YU:
|
||||
about_x_hours:
|
||||
one: "približno jedan sat"
|
||||
other: "približno %{count} sati"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "jedan dan"
|
||||
other: "%{count} dana"
|
||||
@@ -1031,3 +1034,8 @@ sr-YU:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -50,6 +50,9 @@ sr:
|
||||
about_x_hours:
|
||||
one: "приближно један сат"
|
||||
other: "приближно %{count} сати"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "један дан"
|
||||
other: "%{count} дана"
|
||||
@@ -1032,3 +1035,8 @@ sr:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -78,6 +78,9 @@ sv:
|
||||
about_x_hours:
|
||||
one: "ungefär en timme"
|
||||
other: "ungefär %{count} timmar"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "en dag"
|
||||
other: "%{count} dagar"
|
||||
@@ -1069,3 +1072,8 @@ sv:
|
||||
description_date_range_interval: Ange intervall genom att välja start- och slutdatum
|
||||
description_date_from: Ange startdatum
|
||||
description_date_to: Ange slutdatum
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ th:
|
||||
about_x_hours:
|
||||
one: "about 1 hour"
|
||||
other: "about %{count} hours"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 day"
|
||||
other: "%{count} days"
|
||||
@@ -1028,3 +1031,8 @@ th:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -55,6 +55,9 @@ tr:
|
||||
about_x_hours:
|
||||
one: 'yaklaşık 1 saat'
|
||||
other: 'yaklaşık %{count} saat'
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: '1 gün'
|
||||
other: '%{count} gün'
|
||||
@@ -1050,3 +1053,8 @@ tr:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -48,6 +48,9 @@ uk:
|
||||
about_x_hours:
|
||||
one: "about 1 hour"
|
||||
other: "about %{count} hours"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 day"
|
||||
other: "%{count} days"
|
||||
@@ -1026,3 +1029,8 @@ uk:
|
||||
setting_unsubscribe: "Дозволити користувачам видаляти свої облікові записи"
|
||||
button_delete_my_account: "Видалити мій обліковий запис"
|
||||
text_account_destroy_confirmation: "Ваш обліковий запис буде повністю видалений без можливості відновлення.\nВи певні, что бажаете продовжити?"
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -79,6 +79,9 @@ vi:
|
||||
about_x_hours:
|
||||
one: "khoảng 1 giờ"
|
||||
other: "khoảng %{count} giờ"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 ngày"
|
||||
other: "%{count} ngày"
|
||||
@@ -1082,3 +1085,8 @@ vi:
|
||||
text_account_destroy_confirmation: |-
|
||||
Are you sure you want to proceed?
|
||||
Your account will be permanently deleted, with no way to reactivate it.
|
||||
error_session_expired: Your session has expired. Please login again.
|
||||
text_session_expiration_settings: "Warning: changing these settings may expire the current sessions including yours."
|
||||
setting_session_lifetime: Session maximum lifetime
|
||||
setting_session_timeout: Session inactivity timeout
|
||||
label_session_expiration: Session expiration
|
||||
|
||||
@@ -120,6 +120,9 @@
|
||||
about_x_hours:
|
||||
one: "約 1 小時"
|
||||
other: "約 %{count} 小時"
|
||||
x_hours:
|
||||
one: "1 hour"
|
||||
other: "%{count} hours"
|
||||
x_days:
|
||||
one: "1 天"
|
||||
other: "%{count} 天"
|
||||
@@ -276,6 +279,7 @@
|
||||
error_unable_delete_issue_status: '無法刪除問題狀態'
|
||||
error_unable_to_connect: "無法連線至(%{value})"
|
||||
error_attachment_too_big: "這個檔案無法被上傳,因為它已經超過最大的檔案大小 (%{max_size})"
|
||||
error_session_expired: "您的工作階段已經過期。請重新登入。"
|
||||
warning_attachments_not_saved: "%{count} 個附加檔案無法被儲存。"
|
||||
|
||||
mail_subject_lost_password: 您的 Redmine 網站密碼
|
||||
@@ -467,6 +471,8 @@
|
||||
setting_default_issue_start_date_to_creation_date: 設定新問題的起始日期為今天的日期
|
||||
setting_commit_cross_project_ref: 允許關聯並修正其他專案的問題
|
||||
setting_unsubscribe: 允許用戶取消註冊(刪除帳戶)
|
||||
setting_session_lifetime: 工作階段存留時間最大值
|
||||
setting_session_timeout: 工作階段無活動逾時時間
|
||||
|
||||
permission_add_project: 建立專案
|
||||
permission_add_subprojects: 建立子專案
|
||||
@@ -929,6 +935,7 @@
|
||||
label_item_position: "%{position} / %{count}"
|
||||
label_completed_versions: 已完成版本
|
||||
label_search_for_watchers: 搜尋可供加入的監看者
|
||||
label_session_expiration: 工作階段逾期
|
||||
|
||||
button_login: 登入
|
||||
button_submit: 送出
|
||||
@@ -1066,6 +1073,7 @@
|
||||
text_account_destroy_confirmation: |-
|
||||
您確定要繼續這個動作嗎?
|
||||
您的帳戶將會被永久刪除,且無法被重新啟用。
|
||||
text_session_expiration_settings: "警告:變更這些設定將會導致包含您在內的所有工作階段過期。"
|
||||
|
||||
default_role_manager: 管理人員
|
||||
default_role_developer: 開發人員
|
||||
|
||||
@@ -51,6 +51,9 @@ zh:
|
||||
about_x_hours:
|
||||
one: "大约一小时"
|
||||
other: "大约 %{count} 小时"
|
||||
x_hours:
|
||||
one: "1 小时"
|
||||
other: "%{count} 小时"
|
||||
x_days:
|
||||
one: "一天"
|
||||
other: "%{count} 天"
|
||||
@@ -1033,3 +1036,8 @@ zh:
|
||||
text_account_destroy_confirmation: |-
|
||||
确定继续处理?
|
||||
您的账号一旦删除,将无法再次激活使用。
|
||||
error_session_expired: 您的会话已过期。请重新登陆。
|
||||
text_session_expiration_settings: "警告: 更改这些设置将会使包括你在内的当前会话失效。"
|
||||
setting_session_lifetime: 会话最大有效时间
|
||||
setting_session_timeout: 会话闲置超时
|
||||
label_session_expiration: 会话过期
|
||||
|
||||
@@ -36,6 +36,14 @@ unsubscribe:
|
||||
password_min_length:
|
||||
format: int
|
||||
default: 4
|
||||
# Maximum lifetime of user sessions in minutes
|
||||
session_lifetime:
|
||||
format: int
|
||||
default: 0
|
||||
# User session timeout in minutes
|
||||
session_timeout:
|
||||
format: int
|
||||
default: 0
|
||||
attachment_max_size:
|
||||
format: int
|
||||
default: 5120
|
||||
|
||||
@@ -4,6 +4,40 @@ Redmine - project management software
|
||||
Copyright (C) 2006-2012 Jean-Philippe Lang
|
||||
http://www.redmine.org/
|
||||
|
||||
== 2012-11-17 v1.4.5
|
||||
|
||||
* Defect #10818: Running rake in test environment causes exception
|
||||
* Defect #11192: Make repository identifier accept underscores
|
||||
* Defect #11298: Issue API may not work on Ruby 1.9 in Redmine 1.4
|
||||
* Defect #11307: Can't filter for negative numeric custom fields
|
||||
* Defect #11365: Attachment description length is not validated
|
||||
* Defect #11541: Version sharing is missing in the REST API
|
||||
* Defect #11665: New document category is not saved properly
|
||||
* Defect #11789: Edit section links broken with h5/h6 headings
|
||||
* Defect #11966: 404 Error when switching mode in view revision differences in non-main repo
|
||||
* Defect #12189: No tmp/pdf directory
|
||||
* Defect #12196: "Page not found" on OK button in SCM "View all revisions" page
|
||||
* Feature #11338: Exclude emails with auto-submitted => auto-generated
|
||||
* Patch #9732: German translations
|
||||
* Patch #11261: Simplified Chinese translation for configurable session lifetime and timeout
|
||||
* Patch #11328: Fix Japanese mistranslation for 'label_language_based'
|
||||
* Patch #11406: German translation for configurable session lifetime and timeout
|
||||
* Patch #11448: Russian translation for 1.4-stable and 2.0-stable
|
||||
* Patch #11600: Fix plural form of the abbreviation for hours in Brazilian Portuguese
|
||||
|
||||
== 2012-06-18 v1.4.4
|
||||
|
||||
* Defect #10688: PDF export from Wiki - Problems with tables
|
||||
* Defect #11061: Cannot choose commit versions to view differences in Git/Mercurial repository view
|
||||
* Defect #11112: REST API - custom fields in POST/PUT ignored for time_entries
|
||||
* Defect #11133: Wiki-page section edit link can point to incorrect section
|
||||
* Defect #11160: SQL Error on time report if a custom field has multiple values for an entry
|
||||
* Defect #11178: Spent time sorted by date-descending order lists same-date entries in physical order
|
||||
* Defect #11185: Redmine fails to delete a project with parent/child task
|
||||
* Feature #6597: Configurable session lifetime and timeout
|
||||
* Patch #11113: Small glitch in German localization
|
||||
* Fix for Rails vulnerabilities CVE-2012-2694 and CVE-2012-2695
|
||||
|
||||
== 2012-06-05 v1.4.3
|
||||
|
||||
* Defect #11038: "Create and continue" should preserve project, issue and activity when logging time
|
||||
|
||||
@@ -138,21 +138,21 @@ module Redmine
|
||||
# Add list and boolean custom fields as available criteria
|
||||
custom_fields = (@project.nil? ? IssueCustomField.for_all : @project.all_issue_custom_fields)
|
||||
custom_fields.select {|cf| %w(list bool).include? cf.field_format }.each do |cf|
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'Issue' AND c.customized_id = #{Issue.table_name}.id)",
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'Issue' AND c.customized_id = #{Issue.table_name}.id ORDER BY c.value LIMIT 1)",
|
||||
:format => cf.field_format,
|
||||
:label => cf.name}
|
||||
end if @project
|
||||
|
||||
# Add list and boolean time entry custom fields
|
||||
TimeEntryCustomField.find(:all).select {|cf| %w(list bool).include? cf.field_format }.each do |cf|
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'TimeEntry' AND c.customized_id = #{TimeEntry.table_name}.id)",
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'TimeEntry' AND c.customized_id = #{TimeEntry.table_name}.id ORDER BY c.value LIMIT 1)",
|
||||
:format => cf.field_format,
|
||||
:label => cf.name}
|
||||
end
|
||||
|
||||
# Add list and boolean time entry activity custom fields
|
||||
TimeEntryActivityCustomField.find(:all).select {|cf| %w(list bool).include? cf.field_format }.each do |cf|
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'Enumeration' AND c.customized_id = #{TimeEntry.table_name}.activity_id)",
|
||||
@available_criteria["cf_#{cf.id}"] = {:sql => "(SELECT c.value FROM #{CustomValue.table_name} c WHERE c.custom_field_id = #{cf.id} AND c.customized_type = 'Enumeration' AND c.customized_id = #{TimeEntry.table_name}.activity_id ORDER BY c.value LIMIT 1)",
|
||||
:format => cf.field_format,
|
||||
:label => cf.name}
|
||||
end
|
||||
|
||||
@@ -4,7 +4,7 @@ module Redmine
|
||||
module VERSION #:nodoc:
|
||||
MAJOR = 1
|
||||
MINOR = 4
|
||||
TINY = 3
|
||||
TINY = 5
|
||||
|
||||
# Branch values:
|
||||
# * official release: nil
|
||||
|
||||
@@ -29,7 +29,7 @@ module Redmine
|
||||
end
|
||||
|
||||
def method_missing(sym, *args, &block)
|
||||
if args.size == 1 && args.first.is_a?(Time)
|
||||
if args.size == 1 && args.first.is_a?(::Time)
|
||||
__send__ sym, args.first.xmlschema, &block
|
||||
else
|
||||
super
|
||||
|
||||
@@ -69,7 +69,7 @@ module Redmine
|
||||
l = 1
|
||||
started = false
|
||||
ended = false
|
||||
text.scan(/(((?:.*?)(\A|\r?\n\r?\n))(h(\d+)(#{A}#{C})\.(?::(\S+))? (.*?)$)|.*)/m).each do |all, content, lf, heading, level|
|
||||
text.scan(/(((?:.*?)(\A|\r?\n\s*\r?\n))(h(\d+)(#{A}#{C})\.(?::(\S+))? (.*?)$)|.*)/m).each do |all, content, lf, heading, level|
|
||||
if heading.nil?
|
||||
if ended
|
||||
after << all
|
||||
|
||||
@@ -4,7 +4,7 @@ table.revision-info td {
|
||||
padding: 0px;
|
||||
}
|
||||
|
||||
div.revision-graph { position: absolute; }
|
||||
div.revision-graph { position: absolute; min-width: 1px; }
|
||||
|
||||
div.changeset-changes ul { margin: 0; padding: 0; }
|
||||
div.changeset-changes ul > ul { margin-left: 18px; padding: 0; }
|
||||
|
||||
4
test/fixtures/enumerations.yml
vendored
4
test/fixtures/enumerations.yml
vendored
@@ -4,16 +4,19 @@ enumerations_001:
|
||||
id: 1
|
||||
type: DocumentCategory
|
||||
active: true
|
||||
position: 1
|
||||
enumerations_002:
|
||||
name: User documentation
|
||||
id: 2
|
||||
type: DocumentCategory
|
||||
active: true
|
||||
position: 2
|
||||
enumerations_003:
|
||||
name: Technical documentation
|
||||
id: 3
|
||||
type: DocumentCategory
|
||||
active: true
|
||||
position: 3
|
||||
enumerations_004:
|
||||
name: Low
|
||||
id: 4
|
||||
@@ -92,3 +95,4 @@ enumerations_016:
|
||||
id: 16
|
||||
type: DocumentCategory
|
||||
active: false
|
||||
position: 4
|
||||
|
||||
@@ -96,8 +96,13 @@ class AttachmentsControllerTest < ActionController::TestCase
|
||||
end
|
||||
|
||||
def test_save_diff_type
|
||||
@request.session[:user_id] = 1 # admin
|
||||
user1 = User.find(1)
|
||||
user1.pref[:diff_type] = nil
|
||||
user1.preference.save
|
||||
user = User.find(1)
|
||||
assert_nil user.pref[:diff_type]
|
||||
|
||||
@request.session[:user_id] = 1 # admin
|
||||
get :show, :id => 5
|
||||
assert_response :success
|
||||
assert_template 'diff'
|
||||
|
||||
@@ -18,7 +18,9 @@
|
||||
require File.expand_path('../../test_helper', __FILE__)
|
||||
|
||||
class DocumentsControllerTest < ActionController::TestCase
|
||||
fixtures :projects, :users, :roles, :members, :member_roles, :enabled_modules, :documents, :enumerations
|
||||
fixtures :projects, :users, :roles, :members, :member_roles,
|
||||
:enabled_modules, :documents, :enumerations,
|
||||
:groups_users, :attachments
|
||||
|
||||
def setup
|
||||
User.current = nil
|
||||
@@ -95,16 +97,16 @@ LOREM
|
||||
|
||||
def test_create_with_one_attachment
|
||||
ActionMailer::Base.deliveries.clear
|
||||
Setting.notified_events << 'document_added'
|
||||
@request.session[:user_id] = 2
|
||||
set_tmp_attachments_directory
|
||||
|
||||
post :create, :project_id => 'ecookbook',
|
||||
with_settings :notified_events => %w(document_added) do
|
||||
post :create, :project_id => 'ecookbook',
|
||||
:document => { :title => 'DocumentsControllerTest#test_post_new',
|
||||
:description => 'This is a new document',
|
||||
:category_id => 2},
|
||||
:attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain')}}
|
||||
|
||||
end
|
||||
assert_redirected_to '/projects/ecookbook/documents'
|
||||
|
||||
document = Document.find_by_title('DocumentsControllerTest#test_post_new')
|
||||
@@ -124,6 +126,23 @@ LOREM
|
||||
assert_template 'new'
|
||||
end
|
||||
|
||||
def test_create_non_default_category
|
||||
@request.session[:user_id] = 2
|
||||
category2 = Enumeration.find_by_name('User documentation')
|
||||
category2.update_attributes(:is_default => true)
|
||||
category1 = Enumeration.find_by_name('Uncategorized')
|
||||
post :create,
|
||||
:project_id => 'ecookbook',
|
||||
:document => { :title => 'no default',
|
||||
:description => 'This is a new document',
|
||||
:category_id => category1.id }
|
||||
assert_redirected_to '/projects/ecookbook/documents'
|
||||
doc = Document.find_by_title('no default')
|
||||
assert_not_nil doc
|
||||
assert_equal category1.id, doc.category_id
|
||||
assert_equal category1, doc.category
|
||||
end
|
||||
|
||||
def test_edit
|
||||
@request.session[:user_id] = 2
|
||||
get :edit, :id => 1
|
||||
|
||||
@@ -58,13 +58,14 @@ class FilesControllerTest < ActionController::TestCase
|
||||
def test_create_file
|
||||
set_tmp_attachments_directory
|
||||
@request.session[:user_id] = 2
|
||||
Setting.notified_events = ['file_added']
|
||||
ActionMailer::Base.deliveries.clear
|
||||
|
||||
assert_difference 'Attachment.count' do
|
||||
post :create, :project_id => 1, :version_id => '',
|
||||
:attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain')}}
|
||||
assert_response :redirect
|
||||
with_settings :notified_events => %w(file_added) do
|
||||
assert_difference 'Attachment.count' do
|
||||
post :create, :project_id => 1, :version_id => '',
|
||||
:attachments => {'1' => {'file' => uploaded_test_file('testfile.txt', 'text/plain')}}
|
||||
assert_response :redirect
|
||||
end
|
||||
end
|
||||
assert_redirected_to '/projects/ecookbook/files'
|
||||
a = Attachment.find(:first, :order => 'created_on DESC')
|
||||
@@ -80,7 +81,6 @@ class FilesControllerTest < ActionController::TestCase
|
||||
def test_create_version_file
|
||||
set_tmp_attachments_directory
|
||||
@request.session[:user_id] = 2
|
||||
Setting.notified_events = ['file_added']
|
||||
|
||||
assert_difference 'Attachment.count' do
|
||||
post :create, :project_id => 1, :version_id => '2',
|
||||
|
||||
@@ -1305,6 +1305,7 @@ class IssuesControllerTest < ActionController::TestCase
|
||||
:tag => 'input',
|
||||
:attributes => {:type => 'file', :name => 'attachments[1][file]'}
|
||||
}
|
||||
assert_select 'input[name=?][maxlength=255]', 'attachments[1][description]'
|
||||
end
|
||||
|
||||
def test_get_new_should_prefill_the_form_from_params
|
||||
|
||||
@@ -91,11 +91,12 @@ class MessagesControllerTest < ActionController::TestCase
|
||||
def test_post_new
|
||||
@request.session[:user_id] = 2
|
||||
ActionMailer::Base.deliveries.clear
|
||||
Setting.notified_events = ['message_posted']
|
||||
|
||||
post :new, :board_id => 1,
|
||||
with_settings :notified_events => %w(message_posted) do
|
||||
post :new, :board_id => 1,
|
||||
:message => { :subject => 'Test created message',
|
||||
:content => 'Message body'}
|
||||
end
|
||||
message = Message.find_by_subject('Test created message')
|
||||
assert_not_nil message
|
||||
assert_redirected_to "/boards/1/topics/#{message.to_param}"
|
||||
|
||||
@@ -82,12 +82,13 @@ class NewsControllerTest < ActionController::TestCase
|
||||
|
||||
def test_post_create
|
||||
ActionMailer::Base.deliveries.clear
|
||||
Setting.notified_events << 'news_added'
|
||||
|
||||
@request.session[:user_id] = 2
|
||||
post :create, :project_id => 1, :news => { :title => 'NewsControllerTest',
|
||||
|
||||
with_settings :notified_events => %w(news_added) do
|
||||
post :create, :project_id => 1, :news => { :title => 'NewsControllerTest',
|
||||
:description => 'This is the description',
|
||||
:summary => '' }
|
||||
end
|
||||
assert_redirected_to '/projects/ecookbook/news'
|
||||
|
||||
news = News.find_by_title('NewsControllerTest')
|
||||
|
||||
@@ -53,11 +53,48 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
if @char_1.respond_to?(:force_encoding)
|
||||
@char_1.force_encoding('UTF-8')
|
||||
end
|
||||
end
|
||||
|
||||
Setting.default_language = 'en'
|
||||
def test_create_and_update
|
||||
@request.session[:user_id] = 1
|
||||
assert_difference 'Repository.count' do
|
||||
post :create, :project_id => 'subproject1',
|
||||
:repository_scm => 'Git',
|
||||
:repository => {
|
||||
:url => '/test',
|
||||
:is_default => '0',
|
||||
:identifier => 'test-create',
|
||||
:extra_report_last_commit => '1',
|
||||
}
|
||||
end
|
||||
assert_response 302
|
||||
repository = Repository.first(:order => 'id DESC')
|
||||
assert_kind_of Repository::Git, repository
|
||||
assert_equal '/test', repository.url
|
||||
assert_equal true, repository.extra_report_last_commit
|
||||
|
||||
put :update, :id => repository.id,
|
||||
:repository => {
|
||||
:extra_report_last_commit => '0',
|
||||
:identifier => 'test-update',
|
||||
}
|
||||
assert_response 302
|
||||
repo2 = Repository.find(repository.id)
|
||||
assert_equal 'test-update', repo2.identifier
|
||||
assert_equal false, repo2.extra_report_last_commit
|
||||
end
|
||||
|
||||
if File.directory?(REPOSITORY_PATH)
|
||||
## Ruby uses ANSI api to fork a process on Windows.
|
||||
## Japanese Shift_JIS and Traditional Chinese Big5 have 0x5c(backslash) problem
|
||||
## and these are incompatible with ASCII.
|
||||
## Git for Windows (msysGit) changed internal API from ANSI to Unicode in 1.7.10
|
||||
## http://code.google.com/p/msysgit/issues/detail?id=80
|
||||
## So, Latin-1 path tests fail on Japanese Windows
|
||||
WINDOWS_PASS = (Redmine::Platform.mswin? &&
|
||||
Redmine::Scm::Adapters::GitAdapter.client_version_above?([1, 7, 10]))
|
||||
WINDOWS_SKIP_STR = "TODO: This test fails in Git for Windows above 1.7.10"
|
||||
|
||||
def test_get_new
|
||||
@request.session[:user_id] = 1
|
||||
@project.repository.destroy
|
||||
@@ -185,6 +222,8 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
def test_entry_show_latin_1
|
||||
if @ruby19_non_utf8_pass
|
||||
puts_ruby19_non_utf8_pass()
|
||||
elsif WINDOWS_PASS
|
||||
puts WINDOWS_SKIP_STR
|
||||
elsif JRUBY_SKIP
|
||||
puts JRUBY_SKIP_STR
|
||||
else
|
||||
@@ -224,6 +263,8 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
end
|
||||
|
||||
def test_diff
|
||||
assert_equal true, @repository.is_default
|
||||
assert_nil @repository.identifier
|
||||
assert_equal 0, @repository.changesets.count
|
||||
@repository.fetch_changesets
|
||||
@project.reload
|
||||
@@ -246,26 +287,55 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
def test_diff_with_rev_and_path
|
||||
assert_equal 0, @repository.changesets.count
|
||||
@repository.fetch_changesets
|
||||
@project.reload
|
||||
assert_equal NUM_REV, @repository.changesets.count
|
||||
with_settings :diff_max_lines_displayed => 1000 do
|
||||
# Full diff of changeset 2f9c0091
|
||||
['inline', 'sbs'].each do |dt|
|
||||
get :diff,
|
||||
:id => PRJ_ID,
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7',
|
||||
:path => repository_path_hash(['sources', 'watchers_controller.rb'])[:param],
|
||||
:type => dt
|
||||
assert_response :success
|
||||
assert_template 'diff'
|
||||
# Line 22 removed
|
||||
assert_tag :tag => 'th',
|
||||
:content => '22',
|
||||
:sibling => { :tag => 'td',
|
||||
:attributes => { :class => /diff_out/ },
|
||||
:content => /def remove/ }
|
||||
assert_tag :tag => 'h2', :content => /2f9c0091/
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def test_diff_truncated
|
||||
assert_equal 0, @repository.changesets.count
|
||||
@repository.fetch_changesets
|
||||
@project.reload
|
||||
assert_equal NUM_REV, @repository.changesets.count
|
||||
Setting.diff_max_lines_displayed = 5
|
||||
|
||||
# Truncated diff of changeset 2f9c0091
|
||||
with_cache do
|
||||
get :diff, :id => PRJ_ID, :type => 'inline',
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
assert_response :success
|
||||
assert @response.body.include?("... This diff was truncated")
|
||||
|
||||
Setting.default_language = 'fr'
|
||||
get :diff, :id => PRJ_ID, :type => 'inline',
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
assert_response :success
|
||||
assert ! @response.body.include?("... This diff was truncated")
|
||||
assert @response.body.include?("... Ce diff")
|
||||
with_settings :diff_max_lines_displayed => 5 do
|
||||
# Truncated diff of changeset 2f9c0091
|
||||
with_cache do
|
||||
with_settings :default_language => 'en' do
|
||||
get :diff, :id => PRJ_ID, :type => 'inline',
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
assert_response :success
|
||||
assert @response.body.include?("... This diff was truncated")
|
||||
end
|
||||
with_settings :default_language => 'fr' do
|
||||
get :diff, :id => PRJ_ID, :type => 'inline',
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
assert_response :success
|
||||
assert ! @response.body.include?("... This diff was truncated")
|
||||
assert @response.body.include?("... Ce diff")
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -285,9 +355,55 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
diff = assigns(:diff)
|
||||
assert_not_nil diff
|
||||
assert_tag :tag => 'h2', :content => /2f9c0091:61b685fb/
|
||||
assert_tag :tag => "form",
|
||||
:attributes => {
|
||||
:action => "/projects/subproject1/repository/revisions/" +
|
||||
"61b685fbe55ab05b5ac68402d5720c1a6ac973d1/diff"
|
||||
}
|
||||
assert_tag :tag => 'input',
|
||||
:attributes => {
|
||||
:id => "rev_to",
|
||||
:name => "rev_to",
|
||||
:type => "hidden",
|
||||
:value => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
def test_diff_path_in_subrepo
|
||||
repo = Repository::Git.create(
|
||||
:project => @project,
|
||||
:url => REPOSITORY_PATH,
|
||||
:identifier => 'test-diff-path',
|
||||
:path_encoding => 'ISO-8859-1'
|
||||
);
|
||||
assert repo
|
||||
assert_equal false, repo.is_default
|
||||
assert_equal 'test-diff-path', repo.identifier
|
||||
get :diff,
|
||||
:id => PRJ_ID,
|
||||
:repository_id => 'test-diff-path',
|
||||
:rev => '61b685fbe55ab05b',
|
||||
:rev_to => '2f9c0091c754a91a',
|
||||
:type => 'inline'
|
||||
assert_response :success
|
||||
assert_template 'diff'
|
||||
diff = assigns(:diff)
|
||||
assert_not_nil diff
|
||||
assert_tag :tag => "form",
|
||||
:attributes => {
|
||||
:action => "/projects/subproject1/repository/test-diff-path/" +
|
||||
"revisions/61b685fbe55ab05b/diff"
|
||||
}
|
||||
assert_tag :tag => 'input',
|
||||
:attributes => {
|
||||
:id => "rev_to",
|
||||
:name => "rev_to",
|
||||
:type => "hidden",
|
||||
:value => '2f9c0091c754a91a'
|
||||
}
|
||||
end
|
||||
|
||||
def test_diff_latin_1
|
||||
if @ruby19_non_utf8_pass
|
||||
puts_ruby19_non_utf8_pass()
|
||||
@@ -319,8 +435,13 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
end
|
||||
|
||||
def test_save_diff_type
|
||||
@request.session[:user_id] = 1 # admin
|
||||
user1 = User.find(1)
|
||||
user1.pref[:diff_type] = nil
|
||||
user1.preference.save
|
||||
user = User.find(1)
|
||||
assert_nil user.pref[:diff_type]
|
||||
|
||||
@request.session[:user_id] = 1 # admin
|
||||
get :diff,
|
||||
:id => PRJ_ID,
|
||||
:rev => '2f9c0091c754a91af7a9c478e36556b4bde8dcf7'
|
||||
@@ -406,6 +527,8 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
def test_annotate_latin_1
|
||||
if @ruby19_non_utf8_pass
|
||||
puts_ruby19_non_utf8_pass()
|
||||
elsif WINDOWS_PASS
|
||||
puts WINDOWS_SKIP_STR
|
||||
elsif JRUBY_SKIP
|
||||
puts JRUBY_SKIP_STR
|
||||
else
|
||||
@@ -424,6 +547,21 @@ class RepositoriesGitControllerTest < ActionController::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
def test_revisions
|
||||
assert_equal 0, @repository.changesets.count
|
||||
@repository.fetch_changesets
|
||||
@project.reload
|
||||
assert_equal NUM_REV, @repository.changesets.count
|
||||
get :revisions, :id => PRJ_ID
|
||||
assert_response :success
|
||||
assert_template 'revisions'
|
||||
assert_tag :tag => 'form',
|
||||
:attributes => {
|
||||
:method => 'get',
|
||||
:action => '/projects/subproject1/repository/revision'
|
||||
}
|
||||
end
|
||||
|
||||
def test_revision
|
||||
assert_equal 0, @repository.changesets.count
|
||||
@repository.fetch_changesets
|
||||
|
||||
113
test/functional/sessions_test.rb
Normal file
113
test/functional/sessions_test.rb
Normal file
@@ -0,0 +1,113 @@
|
||||
# Redmine - project management software
|
||||
# Copyright (C) 2006-2012 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.expand_path('../../test_helper', __FILE__)
|
||||
|
||||
class SessionStartTest < ActionController::TestCase
|
||||
tests AccountController
|
||||
|
||||
def test_login_should_set_session_timestamps
|
||||
post :login, :username => 'jsmith', :password => 'jsmith'
|
||||
assert_response 302
|
||||
assert_equal 2, session[:user_id]
|
||||
assert_not_nil session[:ctime]
|
||||
assert_not_nil session[:atime]
|
||||
end
|
||||
end
|
||||
|
||||
class SessionsTest < ActionController::TestCase
|
||||
tests WelcomeController
|
||||
|
||||
def test_atime_from_user_session_should_be_updated
|
||||
created = 2.hours.ago.utc.to_i
|
||||
get :index, {}, {:user_id => 2, :ctime => created, :atime => created}
|
||||
assert_response :success
|
||||
assert_equal created, session[:ctime]
|
||||
assert_not_equal created, session[:atime]
|
||||
assert session[:atime] > created
|
||||
end
|
||||
|
||||
def test_user_session_should_not_be_reset_if_lifetime_and_timeout_disabled
|
||||
with_settings :session_lifetime => '0', :session_timeout => '0' do
|
||||
get :index, {}, {:user_id => 2}
|
||||
assert_response :success
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_without_ctime_should_be_reset_if_lifetime_enabled
|
||||
with_settings :session_lifetime => '720' do
|
||||
get :index, {}, {:user_id => 2}
|
||||
assert_redirected_to '/login'
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_with_expired_ctime_should_be_reset_if_lifetime_enabled
|
||||
with_settings :session_timeout => '720' do
|
||||
get :index, {}, {:user_id => 2, :atime => 2.days.ago.utc.to_i}
|
||||
assert_redirected_to '/login'
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_with_valid_ctime_should_not_be_reset_if_lifetime_enabled
|
||||
with_settings :session_timeout => '720' do
|
||||
get :index, {}, {:user_id => 2, :atime => 3.hours.ago.utc.to_i}
|
||||
assert_response :success
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_without_atime_should_be_reset_if_timeout_enabled
|
||||
with_settings :session_timeout => '60' do
|
||||
get :index, {}, {:user_id => 2}
|
||||
assert_redirected_to '/login'
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_with_expired_atime_should_be_reset_if_timeout_enabled
|
||||
with_settings :session_timeout => '60' do
|
||||
get :index, {}, {:user_id => 2, :atime => 4.hours.ago.utc.to_i}
|
||||
assert_redirected_to '/login'
|
||||
end
|
||||
end
|
||||
|
||||
def test_user_session_with_valid_atime_should_not_be_reset_if_timeout_enabled
|
||||
with_settings :session_timeout => '60' do
|
||||
get :index, {}, {:user_id => 2, :atime => 10.minutes.ago.utc.to_i}
|
||||
assert_response :success
|
||||
end
|
||||
end
|
||||
|
||||
def test_expired_user_session_should_be_restarted_if_autologin
|
||||
with_settings :session_lifetime => '720', :session_timeout => '60', :autologin => 7 do
|
||||
token = Token.create!(:user_id => 2, :action => 'autologin', :created_on => 1.day.ago)
|
||||
@request.cookies['autologin'] = token.value
|
||||
created = 2.hours.ago.utc.to_i
|
||||
|
||||
get :index, {}, {:user_id => 2, :ctime => created, :atime => created}
|
||||
assert_equal 2, session[:user_id]
|
||||
assert_response :success
|
||||
assert_not_equal created, session[:ctime]
|
||||
assert session[:ctime] >= created
|
||||
end
|
||||
end
|
||||
|
||||
def test_anonymous_session_should_not_be_reset
|
||||
with_settings :session_lifetime => '720', :session_timeout => '60' do
|
||||
get :index
|
||||
assert_response :success
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -57,6 +57,7 @@ class SettingsControllerTest < ActionController::TestCase
|
||||
assert !Setting.bcc_recipients?
|
||||
assert_equal %w(issue_added issue_updated news_added), Setting.notified_events
|
||||
assert_equal 'Test footer', Setting.emails_footer
|
||||
Setting.clear_cache
|
||||
end
|
||||
|
||||
def test_get_plugin_settings
|
||||
|
||||
@@ -33,6 +33,10 @@ class SysControllerTest < ActionController::TestCase
|
||||
Setting.enabled_scm = %w(Subversion Git)
|
||||
end
|
||||
|
||||
def teardown
|
||||
Setting.clear_cache
|
||||
end
|
||||
|
||||
def test_projects_with_repository_enabled
|
||||
get :projects
|
||||
assert_response :success
|
||||
|
||||
@@ -80,6 +80,16 @@ class TimeEntryReportsControllerTest < ActionController::TestCase
|
||||
assert_equal "162.90", "%.2f" % assigns(:report).total_hours
|
||||
end
|
||||
|
||||
def test_report_custom_field_criteria_with_multiple_values
|
||||
field = TimeEntryCustomField.create!(:name => 'multi', :field_format => 'list', :possible_values => ['value1', 'value2'])
|
||||
entry = TimeEntry.create!(:project => Project.find(1), :hours => 1, :activity_id => 10, :user => User.find(2), :spent_on => Date.today)
|
||||
CustomValue.create!(:customized => entry, :custom_field => field, :value => 'value1')
|
||||
CustomValue.create!(:customized => entry, :custom_field => field, :value => 'value2')
|
||||
|
||||
get :report, :project_id => 1, :columns => 'day', :criteria => ["cf_#{field.id}"]
|
||||
assert_response :success
|
||||
end
|
||||
|
||||
def test_report_one_day
|
||||
get :report, :project_id => 1, :columns => 'day', :from => "2007-03-23", :to => "2007-03-23", :criteria => ["member", "activity"]
|
||||
assert_response :success
|
||||
|
||||
@@ -550,6 +550,20 @@ class TimelogControllerTest < ActionController::TestCase
|
||||
:attributes => {:action => "/projects/ecookbook/issues/1/time_entries", :id => 'query_form'}
|
||||
end
|
||||
|
||||
def test_index_should_sort_by_spent_on_and_created_on
|
||||
t1 = TimeEntry.create!(:user => User.find(1), :project => Project.find(1), :hours => 1, :spent_on => '2012-06-16', :created_on => '2012-06-16 20:00:00', :activity_id => 10)
|
||||
t2 = TimeEntry.create!(:user => User.find(1), :project => Project.find(1), :hours => 1, :spent_on => '2012-06-16', :created_on => '2012-06-16 20:05:00', :activity_id => 10)
|
||||
t3 = TimeEntry.create!(:user => User.find(1), :project => Project.find(1), :hours => 1, :spent_on => '2012-06-15', :created_on => '2012-06-16 20:10:00', :activity_id => 10)
|
||||
|
||||
get :index, :project_id => 1, :from => '2012-06-15', :to => '2012-06-16'
|
||||
assert_response :success
|
||||
assert_equal [t2, t1, t3], assigns(:entries)
|
||||
|
||||
get :index, :project_id => 1, :from => '2012-06-15', :to => '2012-06-16', :sort => 'spent_on'
|
||||
assert_response :success
|
||||
assert_equal [t3, t1, t2], assigns(:entries)
|
||||
end
|
||||
|
||||
def test_index_atom_feed
|
||||
get :index, :project_id => 1, :format => 'atom'
|
||||
assert_response :success
|
||||
|
||||
@@ -25,6 +25,7 @@ class ApiTest::IssuesTest < ActionController::IntegrationTest
|
||||
:member_roles,
|
||||
:issues,
|
||||
:issue_statuses,
|
||||
:issue_relations,
|
||||
:versions,
|
||||
:trackers,
|
||||
:projects_trackers,
|
||||
|
||||
@@ -79,6 +79,21 @@ class ApiTest::TimeEntriesTest < ActionController::IntegrationTest
|
||||
assert_equal 3.5, entry.hours
|
||||
assert_equal TimeEntryActivity.find(11), entry.activity
|
||||
end
|
||||
|
||||
should "accept custom fields" do
|
||||
field = TimeEntryCustomField.create!(:name => 'Test', :field_format => 'string')
|
||||
|
||||
assert_difference 'TimeEntry.count' do
|
||||
post '/time_entries.xml', {:time_entry => {
|
||||
:issue_id => '1', :spent_on => '2010-12-02', :hours => '3.5', :activity_id => '11', :custom_fields => [{:id => field.id.to_s, :value => 'accepted'}]
|
||||
}}, credentials('jsmith')
|
||||
end
|
||||
assert_response :created
|
||||
assert_equal 'application/xml', @response.content_type
|
||||
|
||||
entry = TimeEntry.first(:order => 'id DESC')
|
||||
assert_equal 'accepted', entry.custom_field_value(field)
|
||||
end
|
||||
end
|
||||
|
||||
context "with project_id" do
|
||||
|
||||
@@ -103,15 +103,11 @@ class ApiTest::VersionsTest < ActionController::IntegrationTest
|
||||
|
||||
assert_response :success
|
||||
assert_equal 'application/xml', @response.content_type
|
||||
assert_tag 'version',
|
||||
:child => {
|
||||
:tag => 'id',
|
||||
:content => '2',
|
||||
:sibling => {
|
||||
:tag => 'name',
|
||||
:content => '1.0'
|
||||
}
|
||||
}
|
||||
assert_select 'version' do
|
||||
assert_select 'id', :text => '2'
|
||||
assert_select 'name', :text => '1.0'
|
||||
assert_select 'sharing', :text => 'none'
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -75,6 +75,12 @@ class AttachmentTest < ActiveSupport::TestCase
|
||||
end
|
||||
end
|
||||
|
||||
def test_description_length_should_be_validated
|
||||
a = Attachment.new(:description => 'a' * 300)
|
||||
assert !a.save
|
||||
assert_not_nil a.errors[:description]
|
||||
end
|
||||
|
||||
def test_destroy
|
||||
a = Attachment.new(:container => Issue.find(1),
|
||||
:file => uploaded_test_file("testfile.txt", "text/plain"),
|
||||
|
||||
@@ -179,7 +179,8 @@ class ChangesetTest < ActiveSupport::TestCase
|
||||
end
|
||||
|
||||
def test_commit_closing_a_subproject_issue
|
||||
with_settings :commit_fix_status_id => 5, :commit_fix_keywords => 'closes' do
|
||||
with_settings :commit_fix_status_id => 5, :commit_fix_keywords => 'closes',
|
||||
:default_language => 'en' do
|
||||
issue = Issue.find(5)
|
||||
assert !issue.closed?
|
||||
assert_difference 'Journal.count' do
|
||||
|
||||
@@ -33,11 +33,12 @@ class CommentTest < ActiveSupport::TestCase
|
||||
end
|
||||
|
||||
def test_create_should_send_notification
|
||||
Setting.notified_events << 'news_comment_added'
|
||||
Watcher.create!(:watchable => @news, :user => @jsmith)
|
||||
|
||||
assert_difference 'ActionMailer::Base.deliveries.size' do
|
||||
Comment.create!(:commented => @news, :author => @jsmith, :comments => "my comment")
|
||||
with_settings :notified_events => %w(news_comment_added) do
|
||||
assert_difference 'ActionMailer::Base.deliveries.size' do
|
||||
Comment.create!(:commented => @news, :author => @jsmith, :comments => "my comment")
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -32,4 +32,16 @@ class DocumentCategoryTest < ActiveSupport::TestCase
|
||||
def test_option_name
|
||||
assert_equal :enumeration_doc_categories, DocumentCategory.new.option_name
|
||||
end
|
||||
|
||||
def test_default
|
||||
assert_nil DocumentCategory.find(:first, :conditions => { :is_default => true })
|
||||
e = Enumeration.find_by_name('Technical documentation')
|
||||
e.update_attributes(:is_default => true)
|
||||
assert_equal 3, DocumentCategory.default.id
|
||||
end
|
||||
|
||||
def test_force_default
|
||||
assert_nil DocumentCategory.find(:first, :conditions => { :is_default => true })
|
||||
assert_equal 1, DocumentCategory.default.id
|
||||
end
|
||||
end
|
||||
|
||||
@@ -18,7 +18,10 @@
|
||||
require File.expand_path('../../test_helper', __FILE__)
|
||||
|
||||
class DocumentTest < ActiveSupport::TestCase
|
||||
fixtures :projects, :enumerations, :documents, :attachments
|
||||
fixtures :projects, :enumerations, :documents, :attachments,
|
||||
:enabled_modules,
|
||||
:users, :members, :member_roles, :roles,
|
||||
:groups_users
|
||||
|
||||
def test_create
|
||||
doc = Document.new(:project => Project.find(1), :title => 'New document', :category => Enumeration.find_by_name('User documentation'))
|
||||
@@ -27,10 +30,11 @@ class DocumentTest < ActiveSupport::TestCase
|
||||
|
||||
def test_create_should_send_email_notification
|
||||
ActionMailer::Base.deliveries.clear
|
||||
Setting.notified_events << 'document_added'
|
||||
doc = Document.new(:project => Project.find(1), :title => 'New document', :category => Enumeration.find_by_name('User documentation'))
|
||||
|
||||
assert doc.save
|
||||
|
||||
with_settings :notified_events => %w(document_added) do
|
||||
doc = Document.new(:project => Project.find(1), :title => 'New document', :category => Enumeration.find_by_name('User documentation'))
|
||||
assert doc.save
|
||||
end
|
||||
assert_equal 1, ActionMailer::Base.deliveries.size
|
||||
end
|
||||
|
||||
|
||||
@@ -107,25 +107,53 @@ class IssuesHelperTest < ActionView::TestCase
|
||||
|
||||
context "with a start_date attribute" do
|
||||
should "format the current date" do
|
||||
@detail = JournalDetail.new(:property => 'attr', :old_value => '2010-01-01', :value => '2010-01-31', :prop_key => 'start_date')
|
||||
assert_match "01/31/2010", show_detail(@detail, true)
|
||||
@detail = JournalDetail.new(
|
||||
:property => 'attr',
|
||||
:old_value => '2010-01-01',
|
||||
:value => '2010-01-31',
|
||||
:prop_key => 'start_date'
|
||||
)
|
||||
with_settings :date_format => '%m/%d/%Y' do
|
||||
assert_match "01/31/2010", show_detail(@detail, true)
|
||||
end
|
||||
end
|
||||
|
||||
should "format the old date" do
|
||||
@detail = JournalDetail.new(:property => 'attr', :old_value => '2010-01-01', :value => '2010-01-31', :prop_key => 'start_date')
|
||||
assert_match "01/01/2010", show_detail(@detail, true)
|
||||
@detail = JournalDetail.new(
|
||||
:property => 'attr',
|
||||
:old_value => '2010-01-01',
|
||||
:value => '2010-01-31',
|
||||
:prop_key => 'start_date'
|
||||
)
|
||||
with_settings :date_format => '%m/%d/%Y' do
|
||||
assert_match "01/01/2010", show_detail(@detail, true)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "with a due_date attribute" do
|
||||
should "format the current date" do
|
||||
@detail = JournalDetail.new(:property => 'attr', :old_value => '2010-01-01', :value => '2010-01-31', :prop_key => 'due_date')
|
||||
assert_match "01/31/2010", show_detail(@detail, true)
|
||||
@detail = JournalDetail.new(
|
||||
:property => 'attr',
|
||||
:old_value => '2010-01-01',
|
||||
:value => '2010-01-31',
|
||||
:prop_key => 'due_date'
|
||||
)
|
||||
with_settings :date_format => '%m/%d/%Y' do
|
||||
assert_match "01/31/2010", show_detail(@detail, true)
|
||||
end
|
||||
end
|
||||
|
||||
should "format the old date" do
|
||||
@detail = JournalDetail.new(:property => 'attr', :old_value => '2010-01-01', :value => '2010-01-31', :prop_key => 'due_date')
|
||||
assert_match "01/01/2010", show_detail(@detail, true)
|
||||
@detail = JournalDetail.new(
|
||||
:property => 'attr',
|
||||
:old_value => '2010-01-01',
|
||||
:value => '2010-01-31',
|
||||
:prop_key => 'due_date'
|
||||
)
|
||||
with_settings :date_format => '%m/%d/%Y' do
|
||||
assert_match "01/01/2010", show_detail(@detail, true)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -18,7 +18,17 @@
|
||||
require File.expand_path('../../test_helper', __FILE__)
|
||||
|
||||
class IssueRelationTest < ActiveSupport::TestCase
|
||||
fixtures :issue_relations, :issues
|
||||
fixtures :projects,
|
||||
:users,
|
||||
:roles,
|
||||
:members,
|
||||
:member_roles,
|
||||
:issues,
|
||||
:issue_statuses,
|
||||
:issue_relations,
|
||||
:enabled_modules,
|
||||
:enumerations,
|
||||
:trackers
|
||||
|
||||
def test_create
|
||||
from = Issue.find(1)
|
||||
|
||||
@@ -748,6 +748,30 @@ class IssueTest < ActiveSupport::TestCase
|
||||
assert_nil TimeEntry.find_by_issue_id(1)
|
||||
end
|
||||
|
||||
def test_destroying_a_deleted_issue_should_not_raise_an_error
|
||||
issue = Issue.find(1)
|
||||
Issue.find(1).destroy
|
||||
|
||||
assert_nothing_raised do
|
||||
assert_no_difference 'Issue.count' do
|
||||
issue.destroy
|
||||
end
|
||||
assert issue.destroyed?
|
||||
end
|
||||
end
|
||||
|
||||
def test_destroying_a_stale_issue_should_not_raise_an_error
|
||||
issue = Issue.find(1)
|
||||
Issue.find(1).update_attribute :subject, "Updated"
|
||||
|
||||
assert_nothing_raised do
|
||||
assert_difference 'Issue.count', -1 do
|
||||
issue.destroy
|
||||
end
|
||||
assert issue.destroyed?
|
||||
end
|
||||
end
|
||||
|
||||
def test_blocked
|
||||
blocked_issue = Issue.find(9)
|
||||
blocking_issue = Issue.find(10)
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user