Merge branch 'master' into 192-hidden-platforms-and-tokens
This commit is contained in:
commit
f4c0bd8950
|
@ -20,3 +20,4 @@ crash.log
|
|||
config/newrelic.yml
|
||||
config/deploy/*.rb
|
||||
config/deploy.rb
|
||||
*.swo
|
||||
|
|
|
@ -50,7 +50,7 @@ function initLogWrapper() {
|
|||
var hScroll = l.scrollLeft;
|
||||
var onBottom = Math.abs((l.clientHeight + vScroll - l.scrollHeight)) < getLineHeight(l);
|
||||
|
||||
$logCont.text(data.log);
|
||||
CodeMirror.runMode(data.log.replace(/&/gi, '&'), "text/x-sh", document.getElementById("output"));
|
||||
|
||||
$logCont.scrollLeft(hScroll);
|
||||
$logCont.scrollTop((onBottom || first_open) ? l.scrollHeight - l.clientHeight : vScroll);
|
||||
|
@ -95,7 +95,7 @@ function initLogWrapper() {
|
|||
$autoload.on('change', reloadChange);
|
||||
|
||||
$('#word_wrap').on('change', function() {
|
||||
$logCont.attr({'wrap': ($(this).is(':checked')) ? 'soft' : 'off'});
|
||||
$logCont.css('white-space', ($(this).is(':checked')) ? 'normal' : 'pre');
|
||||
});
|
||||
|
||||
$('#reload_interval').on('change', function() {
|
||||
|
@ -104,8 +104,5 @@ function initLogWrapper() {
|
|||
t = setInterval($(this).val());
|
||||
}
|
||||
});
|
||||
$('#load_lines').on('change', function() {
|
||||
$logCont.data('load_lines', $(this).val());
|
||||
}).trigger('change');
|
||||
loadLog();
|
||||
}
|
|
@ -1461,18 +1461,18 @@ div.reloader {
|
|||
float: right;
|
||||
}
|
||||
|
||||
div.log-wrapper {
|
||||
.log-wrapper {
|
||||
width: 565px;
|
||||
|
||||
div.log-header {
|
||||
.log-header {
|
||||
margin-bottom: 10px;
|
||||
|
||||
div.text-wrap, span {
|
||||
.text-wrap, span {
|
||||
height: 16px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
div.text-wrap {
|
||||
.text-wrap {
|
||||
width: 545px;
|
||||
float: left;
|
||||
|
||||
|
@ -1496,13 +1496,13 @@ div.log-wrapper {
|
|||
}
|
||||
}
|
||||
|
||||
div.log-body {
|
||||
.log-body {
|
||||
|
||||
div.reloader, .log {
|
||||
.reloader, .log {
|
||||
border: 1px solid #ccc;
|
||||
}
|
||||
|
||||
div.reloader {
|
||||
.reloader {
|
||||
padding: 5px 10px;
|
||||
border-bottom: none;
|
||||
border-radius: 5px 5px 0 0;
|
||||
|
@ -1527,15 +1527,6 @@ div.log-wrapper {
|
|||
border-left: 1px solid #ccc;
|
||||
}
|
||||
}
|
||||
|
||||
tr.bottom {
|
||||
td {
|
||||
padding-left: 5px;
|
||||
}
|
||||
td.first {
|
||||
border-top: 1px solid #ccc;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1543,6 +1534,7 @@ div.log-wrapper {
|
|||
background: #f8f8f8;
|
||||
height: 500px;
|
||||
width: 543px;
|
||||
white-space: pre;
|
||||
min-width: 543px;
|
||||
border-radius: 5px 0 5px 5px;
|
||||
overflow: auto;
|
||||
|
|
|
@ -3,11 +3,11 @@ class Api::V1::IssuesController < Api::V1::BaseController
|
|||
respond_to :json
|
||||
|
||||
before_filter :authenticate_user!
|
||||
skip_before_filter :authenticate_user!, :only => [:show] if APP_CONFIG['anonymous_access']
|
||||
skip_before_filter :authenticate_user!, :only => [:index, :group_index, :show] if APP_CONFIG['anonymous_access']
|
||||
|
||||
load_resource :group, :only => :group_index, :find_by => :id, :parent => false
|
||||
load_resource :project
|
||||
load_and_authorize_resource :issue, :through => :project, :find_by => :serial_id, :only => [:show, :update, :destroy, :create, :index]
|
||||
load_and_authorize_resource :group, :only => :group_index, :find_by => :id, :parent => false
|
||||
load_and_authorize_resource :project
|
||||
load_and_authorize_resource :issue, :through => :project, :find_by => :serial_id, :only => [:show, :update, :create, :index]
|
||||
|
||||
def index
|
||||
@issues = @project.issues
|
||||
|
@ -15,20 +15,20 @@ class Api::V1::IssuesController < Api::V1::BaseController
|
|||
end
|
||||
|
||||
def all_index
|
||||
project_ids = get_all_project_ids Project.accessible_by(current_ability, :membered).uniq.pluck(:id)
|
||||
project_ids = get_all_project_ids Project.accessible_by(current_ability, :membered).pluck(:id)
|
||||
@issues = Issue.where('issues.project_id IN (?)', project_ids)
|
||||
render_issues_list
|
||||
end
|
||||
|
||||
def user_index
|
||||
project_ids = get_all_project_ids current_user.projects.select('distinct projects.id').pluck(:id)
|
||||
project_ids = get_all_project_ids current_user.projects.pluck(:id)
|
||||
@issues = Issue.where('issues.project_id IN (?)', project_ids)
|
||||
render_issues_list
|
||||
end
|
||||
|
||||
def group_index
|
||||
project_ids = @group.projects.select('distinct projects.id').pluck(:id)
|
||||
project_ids = Project.accessible_by(current_ability, :membered).where(:id => project_ids).uniq.pluck(:id)
|
||||
project_ids = @group.projects.pluck(:id)
|
||||
project_ids = Project.accessible_by(current_ability, :membered).where(:id => project_ids).pluck(:id)
|
||||
@issues = Issue.where(:project_id => project_ids)
|
||||
render_issues_list
|
||||
end
|
||||
|
@ -42,11 +42,18 @@ class Api::V1::IssuesController < Api::V1::BaseController
|
|||
end
|
||||
|
||||
def update
|
||||
update_subject @issue
|
||||
unless can?(:write, @project)
|
||||
params.delete :update_labels
|
||||
[:assignee_id, :labelings, :labelings_attributes].each do |k|
|
||||
params[:issue].delete k
|
||||
end if params[:issue]
|
||||
end
|
||||
|
||||
def destroy
|
||||
destroy_subject @issue
|
||||
@issue.labelings.destroy_all if params[:update_labels]
|
||||
if params[:issue] && status = params[:issue].delete(:status)
|
||||
@issue.set_close(current_user) if status == 'closed'
|
||||
@issue.set_open if status == 'open'
|
||||
end
|
||||
update_subject @issue
|
||||
end
|
||||
|
||||
private
|
||||
|
@ -92,7 +99,7 @@ class Api::V1::IssuesController < Api::V1::BaseController
|
|||
@issues = @issues.order("#{sort} #{direction}")
|
||||
|
||||
@issues = @issues.where('issues.created_at >= to_timestamp(?)', params[:since]) if params[:since] =~ /\A\d+\z/
|
||||
@issues.paginate(paginate_params)
|
||||
@issues = @issues.paginate(paginate_params)
|
||||
render :index
|
||||
end
|
||||
|
||||
|
@ -101,7 +108,7 @@ class Api::V1::IssuesController < Api::V1::BaseController
|
|||
if ['created', 'all'].include? params[:filter]
|
||||
# add own issues
|
||||
project_ids = Project.accessible_by(current_ability, :show).joins(:issues).
|
||||
where(:issues => {:user_id => current_user.id}).uniq.pluck('projects.id')
|
||||
where(:issues => {:user_id => current_user.id}).pluck('projects.id')
|
||||
end
|
||||
project_ids |= default_project_ids
|
||||
end
|
||||
|
|
|
@ -0,0 +1,168 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
class Api::V1::PullRequestsController < Api::V1::BaseController
|
||||
respond_to :json
|
||||
|
||||
before_filter :authenticate_user!
|
||||
skip_before_filter :authenticate_user!, :only => [:show, :index, :group_index, :commits, :files] if APP_CONFIG['anonymous_access']
|
||||
|
||||
load_resource :group, :only => :group_index, :find_by => :id, :parent => false
|
||||
load_resource :project
|
||||
load_resource :issue, :through => :project, :find_by => :serial_id, :parent => false, :only => [:show, :index, :commits, :files, :merge, :update]
|
||||
load_and_authorize_resource :instance_name => :pull, :through => :issue, :singleton => true, :only => [:show, :index, :commits, :files, :merge, :update]
|
||||
|
||||
def index
|
||||
@pulls = @project.pull_requests
|
||||
@pulls_url = api_v1_project_pull_requests_path(@project, :format => :json)
|
||||
render_pulls_list
|
||||
end
|
||||
|
||||
def all_index
|
||||
project_ids = get_all_project_ids Project.accessible_by(current_ability, :membered).pluck(:id)
|
||||
@pulls = PullRequest.where('pull_requests.to_project_id IN (?)', project_ids)
|
||||
@pulls_url = api_v1_pull_requests_path :format => :json
|
||||
render_pulls_list
|
||||
end
|
||||
|
||||
def user_index
|
||||
project_ids = get_all_project_ids current_user.projects.pluck(:id)
|
||||
@pulls = PullRequest.where('pull_requests.to_project_id IN (?)', project_ids)
|
||||
@pulls_url = pull_requests_api_v1_user_path :format => :json
|
||||
render_pulls_list
|
||||
end
|
||||
|
||||
def group_index
|
||||
project_ids = @group.projects.pluck(:id)
|
||||
project_ids = Project.accessible_by(current_ability, :membered).where(:id => project_ids).pluck(:id)
|
||||
@pulls = PullRequest.where(:to_project_id => project_ids)
|
||||
@pulls_url = pull_requests_api_v1_group_path
|
||||
render_pulls_list
|
||||
end
|
||||
|
||||
def show
|
||||
end
|
||||
|
||||
def create
|
||||
from_project = Project.find(pull_params[:from_project]) if pull_params.try('[]', :from_project).present?
|
||||
from_project ||= @project
|
||||
authorize! :read, from_project
|
||||
|
||||
@pull = @project.pull_requests.new
|
||||
@pull.build_issue :title => pull_params[:title], :body => pull_params[:body]
|
||||
@pull.from_project = @project
|
||||
@pull.to_ref, @pull.from_ref = pull_params[:to_ref], pull_params[:from_ref]
|
||||
@pull.issue.assignee_id = pull_params[:assignee_id] if can?(:write, @project)
|
||||
@pull.issue.user, @pull.issue.project = current_user, @project
|
||||
render_validation_error(@pull, "#{@pull.class.name} has not been created") && return unless @pull.valid?
|
||||
|
||||
@pull.save # set pull id
|
||||
@pull.check(false) # don't make event transaction
|
||||
if @pull.already?
|
||||
@pull.destroy
|
||||
error_message = I18n.t('projects.pull_requests.up_to_date', :to_ref => @pull.to_ref, :from_ref => @pull.from_ref)
|
||||
render_json_response(@pull, error_message, 422)
|
||||
else
|
||||
@pull.send(@pull.status == 'blocked' ? 'block' : @pull.status)
|
||||
render_json_response @pull, "#{@pull.class.name} has been created successfully"
|
||||
end
|
||||
end
|
||||
|
||||
def update
|
||||
@pull = @project.pull_requests.includes(:issue).where(:issues => {:serial_id => params[:id]}).first
|
||||
authorize! :update, @pull
|
||||
|
||||
if pull_params.present?
|
||||
attrs = pull_params.slice(:title, :body)
|
||||
attrs.merge(:assignee_id => pull_params[:assignee_id]) if can?(:write, @project)
|
||||
|
||||
if (action = pull_params[:status]) && %w(close reopen).include?(pull_params[:status])
|
||||
if @pull.send("can_#{action}?")
|
||||
@pull.set_user_and_time current_user
|
||||
need_check = true if action == 'reopen'
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
class_name = @pull.class.name
|
||||
if @pull.issue.update_attributes(attrs)
|
||||
@pull.send(action) if action.present?
|
||||
@pull.check if need_check
|
||||
render_json_response @pull, "#{class_name} has been updated successfully"
|
||||
else
|
||||
render_validation_error @pull, "#{class_name} has not been updated"
|
||||
end
|
||||
end
|
||||
|
||||
def commits
|
||||
@commits = @pull.repo.commits_between(@pull.to_commit, @pull.from_commit).paginate(paginate_params)
|
||||
end
|
||||
|
||||
def files
|
||||
@stats = @pull.diff_stats.zip(@pull.diff).paginate(paginate_params)
|
||||
end
|
||||
|
||||
def merge
|
||||
class_name = @pull.class.name
|
||||
if @pull.merge!(current_user)
|
||||
render_json_response @pull, "#{class_name} has been merged successfully"
|
||||
else
|
||||
render_validation_error @pull, "#{class_name} has not been merged"
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def render_pulls_list
|
||||
@pulls = @pulls.includes(:issue => [:user, :assignee])
|
||||
if params[:status] == 'closed'
|
||||
@pulls = @pulls.closed_or_merged
|
||||
else
|
||||
@pulls = @pulls.not_closed_or_merged
|
||||
end
|
||||
|
||||
if action_name == 'index' && params[:assignee].present?
|
||||
case params[:assignee]
|
||||
when 'none'
|
||||
@pulls = @pulls.where('issues.assigned_id IS NULL')
|
||||
when '*'
|
||||
@pulls = @pulls.where('issues.assigned_id IS NOT NULL')
|
||||
else
|
||||
@pulls = @pulls.where('assignees_issues.uname = ?', params[:assignee])
|
||||
end
|
||||
end
|
||||
|
||||
if %w[all_index user_index group_index].include?(action_name)
|
||||
case params[:filter]
|
||||
when 'created'
|
||||
@pulls = @pulls.where('issues.user_id = ?', current_user.id)
|
||||
when 'all'
|
||||
else
|
||||
@pulls = @pulls.where('issues.assignee_id = ?', current_user.id)
|
||||
end
|
||||
else
|
||||
@pulls.where('users.uname = ?', params[:creator]) if params[:creator].present?
|
||||
end
|
||||
|
||||
sort = params[:sort] == 'updated' ? 'issues.updated_at' : 'issues.created_at'
|
||||
direction = params[:direction] == 'asc' ? 'ASC' : 'DESC'
|
||||
@pulls = @pulls.order("#{sort} #{direction}")
|
||||
|
||||
@pulls = @pulls.where('issues.created_at >= to_timestamp(?)', params[:since]) if params[:since] =~ /\A\d+\z/
|
||||
@pulls = @pulls.paginate(paginate_params)
|
||||
render :index
|
||||
end
|
||||
|
||||
def get_all_project_ids default_project_ids
|
||||
project_ids = []
|
||||
if ['created', 'all'].include? params[:filter]
|
||||
# add own pulls
|
||||
project_ids = Project.accessible_by(current_ability, :show).joins(:issues).
|
||||
where(:issues => {:user_id => current_user.id}).pluck('projects.id')
|
||||
end
|
||||
project_ids |= default_project_ids
|
||||
end
|
||||
|
||||
|
||||
def pull_params
|
||||
@pull_params ||= params[:pull_request] || {}
|
||||
end
|
||||
end
|
|
@ -78,7 +78,6 @@ class Projects::PullRequestsController < Projects::BaseController
|
|||
end
|
||||
|
||||
def merge
|
||||
@pull.check
|
||||
unless @pull.merge!(current_user)
|
||||
flash.now[:error] = t('flash.pull_request.save_error')
|
||||
flash.now[:warning] = @pull.errors.full_messages.join('. ')
|
||||
|
|
|
@ -17,7 +17,7 @@ class Ability
|
|||
can :get_id, Project, :visibility => 'open' # api
|
||||
can :archive, Project, :visibility => 'open'
|
||||
can :read, Issue, :project => {:visibility => 'open'}
|
||||
can :read, PullRequest, :to_project => {:visibility => 'open'}
|
||||
can [:read, :commits, :files], PullRequest, :to_project => {:visibility => 'open'}
|
||||
can :search, BuildList
|
||||
can [:read, :log, :everything], BuildList, :project => {:visibility => 'open'}
|
||||
can [:read, :log], ProductBuildList#, :product => {:platform => {:visibility => 'open'}} # double nested hash don't work
|
||||
|
@ -142,11 +142,12 @@ class Ability
|
|||
can(:update, Issue) {|issue| issue.user_id == user.id or local_admin?(issue.project)}
|
||||
cannot :manage, Issue, :project => {:has_issues => false} # switch off issues
|
||||
|
||||
can :read, PullRequest, :to_project => {:owner_type => 'User', :owner_id => user.id}
|
||||
can :read, PullRequest, :to_project => {:owner_type => 'Group', :owner_id => user.group_ids}
|
||||
can(:read, PullRequest, read_relations_for('pull_requests', 'to_projects')) {|pull| can? :read, pull.to_project rescue nil}
|
||||
can [:read, :commits, :files], PullRequest, :to_project => {:owner_type => 'User', :owner_id => user.id}
|
||||
can [:read, :commits, :files], PullRequest, :to_project => {:owner_type => 'Group', :owner_id => user.group_ids}
|
||||
can([:read, :commits, :files], PullRequest, read_relations_for('pull_requests', 'to_projects')) {|pull| can? :read, pull.to_project}
|
||||
can :create, PullRequest
|
||||
can([:update, :merge], PullRequest) {|pull| pull.user_id == user.id or local_admin?(pull.to_project)}
|
||||
can(:update, PullRequest) {|pull| pull.user_id == user.id or local_admin?(pull.to_project)}
|
||||
can(:merge, PullRequest) {|pull| local_admin?(pull.to_project)}
|
||||
|
||||
can([:create, :new_line], Comment) {|comment| can? :read, comment.project}
|
||||
can([:update, :destroy], Comment) {|comment| comment.user == user or comment.project.owner == user or local_admin?(comment.project)}
|
||||
|
|
|
@ -1,185 +0,0 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
class ActivityFeedObserver < ActiveRecord::Observer
|
||||
observe :issue, :comment, :user, :build_list
|
||||
|
||||
BUILD_LIST_STATUSES = [
|
||||
BuildList::BUILD_PUBLISHED,
|
||||
BuildList::SUCCESS,
|
||||
BuildList::BUILD_ERROR,
|
||||
BuildList::FAILED_PUBLISH,
|
||||
BuildList::TESTS_FAILED
|
||||
].freeze
|
||||
|
||||
def after_create(record)
|
||||
case record.class.to_s
|
||||
when 'User'
|
||||
ActivityFeed.create(
|
||||
:user => record,
|
||||
:kind => 'new_user_notification',
|
||||
:data => {:user_name => record.user_appeal, :user_email => record.email}
|
||||
)
|
||||
|
||||
when 'Issue'
|
||||
record.collect_recipients.each do |recipient|
|
||||
next if record.user_id == recipient.id
|
||||
UserMailer.new_issue_notification(record, recipient).deliver if recipient.notifier.can_notify && recipient.notifier.new_issue
|
||||
ActivityFeed.create(
|
||||
:user => recipient,
|
||||
:kind => 'new_issue_notification',
|
||||
:data => {:user_name => record.user.name, :user_email => record.user.email, :user_id => record.user_id,:issue_serial_id => record.serial_id,
|
||||
:issue_title => record.title, :project_id => record.project.id, :project_name => record.project.name, :project_owner => record.project.owner.uname}
|
||||
)
|
||||
end
|
||||
|
||||
if record.assignee_id_changed?
|
||||
UserMailer.new_issue_notification(record, record.assignee).deliver if record.assignee.notifier.issue_assign && record.assignee.notifier.can_notify
|
||||
ActivityFeed.create(
|
||||
:user => record.user,
|
||||
:kind => 'issue_assign_notification',
|
||||
:data => {:user_name => record.user.name, :user_email => record.user.email, :user_id => record.user_id, :issue_serial_id => record.serial_id,
|
||||
:project_id => record.project.id, :issue_title => record.title, :project_name => record.project.name, :project_owner => record.project.owner.uname}
|
||||
)
|
||||
end
|
||||
record.project.hooks.each{ |h| h.receive_issues(record, :create) }
|
||||
Comment.create_link_on_issues_from_item(record)
|
||||
when 'Comment'
|
||||
return if record.automatic
|
||||
if record.issue_comment?
|
||||
subscribes = record.commentable.subscribes
|
||||
subscribes.each do |subscribe|
|
||||
if record.user_id != subscribe.user_id
|
||||
UserMailer.new_comment_notification(record, subscribe.user).deliver if record.can_notify_on_new_comment?(subscribe)
|
||||
ActivityFeed.create(
|
||||
:user => subscribe.user,
|
||||
:kind => 'new_comment_notification',
|
||||
:data => {:user_name => record.user.name, :user_email => record.user.email, :user_id => record.user_id, :comment_body => record.body,
|
||||
:issue_title => record.commentable.title, :issue_serial_id => record.commentable.serial_id, :project_id => record.commentable.project.id,
|
||||
:comment_id => record.id, :project_name => record.project.name, :project_owner => record.project.owner.uname}
|
||||
)
|
||||
end
|
||||
end
|
||||
elsif record.commit_comment?
|
||||
subscribes = Subscribe.comment_subscribes(record).where(:status => true)
|
||||
subscribes.each do |subscribe|
|
||||
next if record.own_comment?(subscribe.user)
|
||||
if subscribe.user.notifier.can_notify and
|
||||
( (subscribe.project.owner?(subscribe.user) && subscribe.user.notifier.new_comment_commit_repo_owner) or
|
||||
(subscribe.user.commentor?(record.commentable) && subscribe.user.notifier.new_comment_commit_commentor) or
|
||||
(subscribe.user.committer?(record.commentable) && subscribe.user.notifier.new_comment_commit_owner) )
|
||||
UserMailer.new_comment_notification(record, subscribe.user).deliver
|
||||
end
|
||||
ActivityFeed.create(
|
||||
:user => subscribe.user,
|
||||
:kind => 'new_comment_commit_notification',
|
||||
:data => {:user_name => record.user.name, :user_email => record.user.email, :user_id => record.user_id, :comment_body => record.body,
|
||||
:commit_message => record.commentable.message, :commit_id => record.commentable.id,
|
||||
:project_id => record.project.id, :comment_id => record.id, :project_name => record.project.name, :project_owner => record.project.owner.uname}
|
||||
)
|
||||
end
|
||||
end
|
||||
Comment.create_link_on_issues_from_item(record)
|
||||
|
||||
when 'GitHook'
|
||||
return unless record.project
|
||||
PullRequest.where("from_project_id = ? OR to_project_id = ?", record.project, record.project).needed_checking.each {|pull| pull.check}
|
||||
record.project.hooks.each{ |h| h.receive_push(record) }
|
||||
|
||||
change_type = record.change_type
|
||||
branch_name = record.refname.split('/').last
|
||||
|
||||
if change_type == 'delete'
|
||||
kind = 'git_delete_branch_notification'
|
||||
options = {:project_id => record.project.id, :project_name => record.project.name, :branch_name => branch_name,
|
||||
:change_type => change_type, :project_owner => record.project.owner.uname}
|
||||
else
|
||||
if record.message # online update
|
||||
last_commits, commits = [[record.newrev, record.message]], []
|
||||
all_commits = last_commits
|
||||
else
|
||||
commits = record.project.repo.commits_between(record.oldrev, record.newrev)
|
||||
all_commits = commits.collect { |commit| [commit.sha, commit.message] }
|
||||
last_commits = all_commits.last(3).reverse
|
||||
end
|
||||
|
||||
kind = 'git_new_push_notification'
|
||||
options = {:project_id => record.project.id, :project_name => record.project.name, :last_commits => last_commits,
|
||||
:branch_name => branch_name, :change_type => change_type, :project_owner => record.project.owner.uname}
|
||||
if commits.count > 3
|
||||
commits = commits[0...-3]
|
||||
options.merge!({:other_commits_count => commits.count, :other_commits => "#{commits[0].sha[0..9]}...#{commits[-1].sha[0..9]}"})
|
||||
end
|
||||
Comment.create_link_on_issues_from_item(record, all_commits) if all_commits.count > 0
|
||||
end
|
||||
options.merge!({:user_id => record.user.id, :user_name => record.user.name, :user_email => record.user.email}) if record.user
|
||||
|
||||
record.project.admins.each do |recipient|
|
||||
next if record.user && record.user.id == recipient.id
|
||||
ActivityFeed.create!(
|
||||
:user => recipient,
|
||||
:kind => kind,
|
||||
:data => options
|
||||
)
|
||||
end
|
||||
|
||||
when 'Hash' # 'Gollum::Committer'
|
||||
actor = User.find_by_uname! record[:actor_name]
|
||||
project = Project.find record[:project_id]
|
||||
|
||||
project.admins.each do |recipient|
|
||||
ActivityFeed.create!(
|
||||
:user => recipient,
|
||||
:kind => 'wiki_new_commit_notification',
|
||||
:data => {:user_id => actor.id, :user_name => actor.name, :user_email => actor.email, :project_id => project.id,
|
||||
:project_name => project.name, :commit_sha => record[:commit_sha], :project_owner => project.owner.uname}
|
||||
)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def after_update(record)
|
||||
case record.class.to_s
|
||||
when 'Issue'
|
||||
if record.assignee_id && record.assignee_id_changed?
|
||||
UserMailer.issue_assign_notification(record, record.assignee).deliver if record.assignee.notifier.issue_assign && record.assignee.notifier.can_notify
|
||||
ActivityFeed.create(
|
||||
:user => record.assignee,
|
||||
:kind => 'issue_assign_notification',
|
||||
:data => {:user_name => record.assignee.name, :user_email => record.assignee.email, :issue_serial_id => record.serial_id, :issue_title => record.title,
|
||||
:project_id => record.project.id, :project_name => record.project.name, :project_owner => record.project.owner.uname}
|
||||
)
|
||||
end
|
||||
record.project.hooks.each{ |h| h.receive_issues(record, :update) } if record.status_changed?
|
||||
# dont remove outdated issues link
|
||||
Comment.create_link_on_issues_from_item(record)
|
||||
when 'BuildList'
|
||||
if record.mass_build.blank? && ( # Do not show mass build activity in activity feeds
|
||||
record.status_changed? && BUILD_LIST_STATUSES.include?(record.status) ||
|
||||
record.status == BuildList::BUILD_PENDING && record.bs_id_changed?
|
||||
)
|
||||
|
||||
record.project.admins.each do |recipient|
|
||||
user = record.publisher || record.user
|
||||
ActivityFeed.create(
|
||||
:user => recipient,
|
||||
:kind => 'build_list_notification',
|
||||
:data => {
|
||||
:task_num => record.bs_id,
|
||||
:build_list_id => record.id,
|
||||
:status => record.status,
|
||||
:updated_at => record.updated_at,
|
||||
:project_id => record.project_id,
|
||||
:project_name => record.project.name,
|
||||
:project_owner => record.project.owner.uname,
|
||||
:user_name => user.name,
|
||||
:user_email => user.email,
|
||||
:user_id => user.id
|
||||
}
|
||||
)
|
||||
end
|
||||
end
|
||||
when 'Comment'
|
||||
# dont remove outdated issues link
|
||||
Comment.create_link_on_issues_from_item(record)
|
||||
end
|
||||
end
|
||||
end
|
|
@ -3,6 +3,7 @@ class BuildList < ActiveRecord::Base
|
|||
include Modules::Models::CommitAndVersion
|
||||
include Modules::Models::FileStoreClean
|
||||
include AbfWorker::ModelHelper
|
||||
include Modules::Observers::ActivityFeed::BuildList
|
||||
|
||||
belongs_to :project
|
||||
belongs_to :arch
|
||||
|
|
|
@ -1,5 +1,7 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
class Comment < ActiveRecord::Base
|
||||
include Modules::Observers::ActivityFeed::Comment
|
||||
|
||||
# regexp take from http://code.google.com/p/concerto-platform/source/browse/v3/cms/lib/CodeMirror/mode/gfm/gfm.js?spec=svn861&r=861#71
|
||||
# User/Project#Num
|
||||
# User#Num
|
||||
|
@ -54,10 +56,6 @@ class Comment < ActiveRecord::Base
|
|||
user_id == user.id
|
||||
end
|
||||
|
||||
def can_notify_on_new_comment?(subscribe)
|
||||
User.find(subscribe.user).notifier.new_comment && User.find(subscribe.user).notifier.can_notify
|
||||
end
|
||||
|
||||
def actual_inline_comment?(diff = nil, force = false)
|
||||
unless force
|
||||
raise "This is not inline comment!" if data.blank? # for debug
|
||||
|
|
|
@ -65,7 +65,7 @@ class GitHook
|
|||
end
|
||||
|
||||
def self.process(*args)
|
||||
ActivityFeedObserver.instance.after_create(args.size > 1 ? GitHook.new(*args) : args.first)
|
||||
Modules::Observers::ActivityFeed::Git.create_notifications(args.size > 1 ? GitHook.new(*args) : args.first)
|
||||
end
|
||||
|
||||
def find_user(user)
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
class Issue < ActiveRecord::Base
|
||||
include Modules::Observers::ActivityFeed::Issue
|
||||
STATUSES = ['open', 'closed']
|
||||
|
||||
belongs_to :project
|
||||
|
|
|
@ -20,6 +20,8 @@ class PullRequest < ActiveRecord::Base
|
|||
attr_accessible :issue_attributes, :to_ref, :from_ref
|
||||
|
||||
scope :needed_checking, includes(:issue).where(:issues => {:status => ['open', 'blocked', 'ready']})
|
||||
scope :not_closed_or_merged, needed_checking
|
||||
scope :closed_or_merged, where(:issues => {:status => ['closed', 'merged']})
|
||||
|
||||
state_machine :status, :initial => :open do
|
||||
event :ready do
|
||||
|
|
|
@ -60,6 +60,7 @@ class User < Avatar
|
|||
|
||||
include Modules::Models::PersonalRepository
|
||||
include Modules::Models::ActsLikeMember
|
||||
include Modules::Observers::ActivityFeed::User
|
||||
|
||||
def admin?
|
||||
role == 'admin'
|
||||
|
|
|
@ -0,0 +1,24 @@
|
|||
json.number pull.serial_id
|
||||
json.(pull, :title, :status)
|
||||
json.to_ref do |json_ref|
|
||||
json_ref.ref pull.to_ref
|
||||
json_ref.sha pull.to_commit.id
|
||||
json_ref.project do |json_project|
|
||||
json_project.partial! 'api/v1/projects/project', :project => pull.to_project, :json => json
|
||||
end
|
||||
end
|
||||
json.from_ref do |json_ref|
|
||||
json_ref.ref pull.from_ref
|
||||
json_ref.sha pull.from_commit.id
|
||||
json_ref.project do |json_project|
|
||||
json_project.partial! 'api/v1/projects/project', :project => pull.from_project, :json => json
|
||||
end
|
||||
end
|
||||
json.partial! 'api/v1/shared/owner', :owner => pull.user
|
||||
json.assignee do |json_assignee|
|
||||
json.partial! 'api/v1/shared/member', :member => pull.issue.assignee, :tag => json_assignee
|
||||
end if pull.issue.assignee
|
||||
json.mergeable pull.can_merging?
|
||||
json.merged_at pull.issue.closed_at.to_i if pull.merged?
|
||||
|
||||
json.url api_v1_project_pull_request_path(pull.to_project.id, pull.id, :format => :json)
|
|
@ -0,0 +1,25 @@
|
|||
json.commits @commits do |json_commit, commit|
|
||||
json_commit.sha commit.id
|
||||
json_commit.https_url commit_path(@project, commit.id)
|
||||
json.author do |json_author|
|
||||
json_author.name commit.author.name
|
||||
json_author.email commit.author.email
|
||||
json_author.date commit.authored_date.to_i
|
||||
end
|
||||
json.committer do |json_committer|
|
||||
json_committer.name commit.committer.name
|
||||
json_committer.email commit.committer.email
|
||||
json_committer.date commit.committed_date.to_i
|
||||
end
|
||||
json.message commit.message
|
||||
json.tree do |json_tree|
|
||||
json_tree.sha commit.id
|
||||
json_tree.https_url commit_path(@project, commit.id)
|
||||
end
|
||||
json.parents commit.parents do |json, parent|
|
||||
json.sha parent.id
|
||||
json.https_url commit_path(@project, parent.id)
|
||||
end
|
||||
end
|
||||
|
||||
json.url commits_api_v1_project_pull_request_path(:format => :json)
|
|
@ -0,0 +1,24 @@
|
|||
json.files @stats do |json_stat, stat|
|
||||
fstat, diff = stat
|
||||
commit_id = diff.deleted_file ? @pull.to_commit.id : @pull.from_commit.id
|
||||
json_stat.sha commit_id
|
||||
json_stat.filename diff.b_path
|
||||
status = case
|
||||
when diff.new_file
|
||||
'added'
|
||||
when diff.deleted_file
|
||||
'deleted'
|
||||
when diff.renamed_file
|
||||
'renamed'
|
||||
else
|
||||
'modified'
|
||||
end
|
||||
json_stat.status status
|
||||
json_stat.additions fstat.additions
|
||||
json_stat.deletions fstat.deletions
|
||||
json_stat.changes fstat.additions + fstat.deletions
|
||||
json_stat.blob_https_url blob_path(@project, commit_id, diff.b_path)
|
||||
json_stat.raw_https_url raw_path(@project, commit_id, diff.b_path)
|
||||
end
|
||||
|
||||
json.url files_api_v1_project_pull_request_path(:format => :json)
|
|
@ -0,0 +1,5 @@
|
|||
json.pull_requests @pulls do |json, pull|
|
||||
json.partial! 'pull', :pull => pull, :json => json
|
||||
end
|
||||
|
||||
json.url @pulls_url
|
|
@ -0,0 +1,13 @@
|
|||
json.issue do |json|
|
||||
json.partial! 'pull', :pull => @pull, :json => json
|
||||
json.body @pull.body
|
||||
json.closed_at pull.issue.closed_at.to_i if @pull.merged? || @pull.closed?
|
||||
json.closed_by do |json_user|
|
||||
json.partial! 'api/v1/shared/member', :member => @pull.issue.closer, :tag => json_user
|
||||
end if @pull.issue.closer
|
||||
json.merged_by do |json_user|
|
||||
json.partial! 'api/v1/shared/member', :member => @pull.issue.closer, :tag => json_user
|
||||
end if @pull.merged?
|
||||
json.created_at @pull.issue.created_at.to_i
|
||||
json.updated_at @pull.issue.updated_at.to_i
|
||||
end
|
|
@ -2,8 +2,7 @@
|
|||
.file
|
||||
%a{:name => "diff-#{pull_diff_counter}"}
|
||||
.top
|
||||
.l= h(pull_diff.renamed_file ? "#{pull_diff.a_path.rtruncate 60} -> #{pull_diff.b_path.rtruncate 60}" : pull_diff.a_path.rtruncate(120))
|
||||
- if pull_diff.b_path.present?
|
||||
.l= h(pull_diff.renamed_file ? "#{pull_diff.a_path.rtruncate 60}=>#{pull_diff.b_path.rtruncate 60}" : pull_diff.b_path.rtruncate(120))
|
||||
.r= link_to "view file @ #{short_hash_id(commit_id)}", blob_path(@project, commit_id, pull_diff.b_path)
|
||||
.clear
|
||||
-if pull_diff.diff.present? && !(@pull.repo.tree(commit_id) / pull_diff.b_path).binary?
|
||||
|
|
|
@ -28,18 +28,14 @@
|
|||
= check_box_tag :autoreload, true, build_started
|
||||
= t("layout.build_lists.log.autoreload")
|
||||
= select_tag :reload_interval, log_reload_time_options
|
||||
%tr.bottom
|
||||
%td.first
|
||||
- if download_log_url
|
||||
= link_to t("layout.build_lists.log.download"), download_log_url, :id => :log_url
|
||||
%td.last{ :class => build_started ? nil : :hidden }
|
||||
= label_tag :load_lines do
|
||||
= raw t("layout.build_lists.log.load_lines", :count => select_tag(:load_lines, log_reload_lines_options))
|
||||
|
||||
.both
|
||||
%textarea.log{ :readonly => :readonly, :wrap => 'off',
|
||||
#output.cm-s-default.log{ :readonly => :readonly,
|
||||
:data => { :url => get_log_path, :log_type => :build } }
|
||||
= t("layout.build_lists.log.not_available")
|
||||
%pre#code
|
||||
.both
|
||||
.both
|
||||
|
||||
|
||||
:javascript
|
||||
$(function() {
|
||||
|
|
|
@ -34,7 +34,7 @@ module Rosa
|
|||
# config.plugins = [ :exception_notification, :ssl_requirement, :all ]
|
||||
|
||||
# Activate observers that should always be running.
|
||||
config.active_record.observers = :event_log_observer, :activity_feed_observer, :build_list_observer
|
||||
config.active_record.observers = :event_log_observer, :build_list_observer
|
||||
|
||||
# Set Time.zone default to the specified zone and make Active Record auto-convert to this zone.
|
||||
# Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC.
|
||||
|
|
|
@ -2,3 +2,5 @@
|
|||
APP_CONFIG = YAML.load_file("#{Rails.root}/config/application.yml")[Rails.env]
|
||||
# Remove '/' from the end of url
|
||||
APP_CONFIG.keys.select {|key| key =~ /_url\Z/}.each {|key| APP_CONFIG[key] = APP_CONFIG[key].chomp('/') if APP_CONFIG[key].respond_to?(:chomp)}
|
||||
# Paginates a static array
|
||||
require 'will_paginate/array'
|
||||
|
|
|
@ -60,6 +60,13 @@ Rosa::Application.routes.draw do
|
|||
}
|
||||
resources :build_lists, :only => :index
|
||||
resources :issues, :only => [:index, :create, :show, :update]
|
||||
resources :pull_requests, :only => [:index, :create, :show, :update] do
|
||||
member {
|
||||
get :commits
|
||||
get :files
|
||||
put :merge
|
||||
}
|
||||
end
|
||||
end
|
||||
resources :users, :only => [:show]
|
||||
get 'user' => 'users#show_current_user'
|
||||
|
@ -68,6 +75,7 @@ Rosa::Application.routes.draw do
|
|||
get :notifiers
|
||||
put :notifiers
|
||||
get '/issues' => 'issues#user_index'
|
||||
get '/pull_requests' => 'pull_requests#user_index'
|
||||
}
|
||||
end
|
||||
resources :groups, :only => [:index, :show, :update, :create, :destroy] do
|
||||
|
@ -77,6 +85,7 @@ Rosa::Application.routes.draw do
|
|||
delete :remove_member
|
||||
put :update_member
|
||||
get '/issues' => 'issues#group_index'
|
||||
get '/pull_requests' => 'pull_requests#group_index'
|
||||
}
|
||||
end
|
||||
resources :products, :only => [:show, :update, :create, :destroy] do
|
||||
|
@ -87,6 +96,7 @@ Rosa::Application.routes.draw do
|
|||
end
|
||||
#resources :ssh_keys, :only => [:index, :create, :destroy]
|
||||
get 'issues' => 'issues#all_index'
|
||||
get 'pull_requests' => 'pull_requests#all_index'
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -278,7 +288,7 @@ Rosa::Application.routes.draw do
|
|||
match 'compare/:versions' => 'wiki#compare', :versions => /([a-f0-9\^]{6,40})(\.\.\.[a-f0-9\^]{6,40})/, :as => :compare_versions, :via => :get
|
||||
end
|
||||
end
|
||||
resources :issues, :except => :edit do
|
||||
resources :issues, :except => [:destroy, :edit] do
|
||||
resources :comments, :only => [:edit, :create, :update, :destroy]
|
||||
resources :subscribes, :only => [:create, :destroy]
|
||||
collection do
|
||||
|
|
|
@ -150,7 +150,7 @@ module Modules
|
|||
module ClassMethods
|
||||
def process_hook(owner_uname, repo, newrev, oldrev, ref, newrev_type, user = nil, message = nil)
|
||||
rec = GitHook.new(owner_uname, repo, newrev, oldrev, ref, newrev_type, user, message)
|
||||
ActivityFeedObserver.instance.after_create rec
|
||||
Modules::Observers::ActivityFeed::Git.create_notifications rec
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -0,0 +1,46 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
module Modules::Observers::ActivityFeed::BuildList
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
after_update :build_list_notifications
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def build_list_notifications
|
||||
if mass_build.blank? && ( # Do not show mass build activity in activity feeds
|
||||
status_changed? && [
|
||||
BuildList::BUILD_PUBLISHED,
|
||||
BuildList::SUCCESS,
|
||||
BuildList::BUILD_ERROR,
|
||||
BuildList::FAILED_PUBLISH,
|
||||
BuildList::TESTS_FAILED
|
||||
].include?(status) ||
|
||||
status == BuildList::BUILD_PENDING && bs_id_changed?
|
||||
)
|
||||
|
||||
updater = publisher || user
|
||||
project.admins.each do |recipient|
|
||||
ActivityFeed.create(
|
||||
:user => recipient,
|
||||
:kind => 'build_list_notification',
|
||||
:data => {
|
||||
:task_num => bs_id,
|
||||
:build_list_id => id,
|
||||
:status => status,
|
||||
:updated_at => updated_at,
|
||||
:project_id => project_id,
|
||||
:project_name => project.name,
|
||||
:project_owner => project.owner.uname,
|
||||
:user_name => updater.name,
|
||||
:user_email => updater.email,
|
||||
:user_id => updater.id
|
||||
}
|
||||
)
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,70 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
module Modules::Observers::ActivityFeed::Comment
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
after_commit :new_comment_notifications, :on => :create
|
||||
# dont remove outdated issues link
|
||||
after_update -> { Comment.create_link_on_issues_from_item(self) }
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def new_comment_notifications
|
||||
return if automatic?
|
||||
if issue_comment?
|
||||
commentable.subscribes.each do |subscribe|
|
||||
if user_id != subscribe.user_id
|
||||
UserMailer.new_comment_notification(self, subscribe.user).deliver if can_notify_on_new_comment?(subscribe)
|
||||
ActivityFeed.create(
|
||||
:user => subscribe.user,
|
||||
:kind => 'new_comment_notification',
|
||||
:data => {
|
||||
:user_name => user.name,
|
||||
:user_email => user.email,
|
||||
:user_id => user_id,
|
||||
:comment_body => body,
|
||||
:issue_title => commentable.title,
|
||||
:issue_serial_id => commentable.serial_id,
|
||||
:project_id => commentable.project.id,
|
||||
:comment_id => id,
|
||||
:project_name => project.name,
|
||||
:project_owner => project.owner.uname
|
||||
}
|
||||
)
|
||||
end
|
||||
end
|
||||
elsif commit_comment?
|
||||
Subscribe.comment_subscribes(self).where(:status => true).each do |subscribe|
|
||||
next if own_comment?(subscribe.user)
|
||||
if subscribe.user.notifier.can_notify and
|
||||
( (subscribe.project.owner?(subscribe.user) && subscribe.user.notifier.new_comment_commit_repo_owner) or
|
||||
(subscribe.user.commentor?(self.commentable) && subscribe.user.notifier.new_comment_commit_commentor) or
|
||||
(subscribe.user.committer?(self.commentable) && subscribe.user.notifier.new_comment_commit_owner) )
|
||||
UserMailer.new_comment_notification(self, subscribe.user).deliver
|
||||
end
|
||||
ActivityFeed.create(
|
||||
:user => subscribe.user,
|
||||
:kind => 'new_comment_commit_notification',
|
||||
:data => {
|
||||
:user_name => user.name,
|
||||
:user_email => user.email,
|
||||
:user_id => user_id,
|
||||
:comment_body => body,
|
||||
:commit_message => commentable.message,
|
||||
:commit_id => commentable.id,
|
||||
:project_id => project.id,
|
||||
:comment_id => id,
|
||||
:project_name => project.name,
|
||||
:project_owner => project.owner.uname}
|
||||
)
|
||||
end
|
||||
end
|
||||
Comment.create_link_on_issues_from_item(self)
|
||||
end
|
||||
|
||||
def can_notify_on_new_comment?(subscribe)
|
||||
User.find(subscribe.user).notifier.new_comment && User.find(subscribe.user).notifier.can_notify
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,65 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
module Modules::Observers::ActivityFeed::Git
|
||||
|
||||
def self.create_notifications(record)
|
||||
|
||||
case record.class.to_s
|
||||
when 'GitHook'
|
||||
return unless record.project
|
||||
PullRequest.where("from_project_id = ? OR to_project_id = ?", record.project, record.project).needed_checking.each {|pull| pull.check}
|
||||
record.project.hooks.each{ |h| h.receive_push(record) }
|
||||
|
||||
change_type = record.change_type
|
||||
branch_name = record.refname.split('/').last
|
||||
|
||||
if change_type == 'delete'
|
||||
kind = 'git_delete_branch_notification'
|
||||
options = {:project_id => record.project.id, :project_name => record.project.name, :branch_name => branch_name,
|
||||
:change_type => change_type, :project_owner => record.project.owner.uname}
|
||||
else
|
||||
if record.message # online update
|
||||
last_commits, commits = [[record.newrev, record.message]], []
|
||||
all_commits = last_commits
|
||||
else
|
||||
commits = record.project.repo.commits_between(record.oldrev, record.newrev)
|
||||
all_commits = commits.collect { |commit| [commit.sha, commit.message] }
|
||||
last_commits = all_commits.last(3).reverse
|
||||
end
|
||||
|
||||
kind = 'git_new_push_notification'
|
||||
options = {:project_id => record.project.id, :project_name => record.project.name, :last_commits => last_commits,
|
||||
:branch_name => branch_name, :change_type => change_type, :project_owner => record.project.owner.uname}
|
||||
if commits.count > 3
|
||||
commits = commits[0...-3]
|
||||
options.merge!({:other_commits_count => commits.count, :other_commits => "#{commits[0].sha[0..9]}...#{commits[-1].sha[0..9]}"})
|
||||
end
|
||||
Comment.create_link_on_issues_from_item(record, all_commits) if all_commits.count > 0
|
||||
end
|
||||
options.merge!({:user_id => record.user.id, :user_name => record.user.name, :user_email => record.user.email}) if record.user
|
||||
|
||||
record.project.admins.each do |recipient|
|
||||
next if record.user && record.user.id == recipient.id
|
||||
ActivityFeed.create!(
|
||||
:user => recipient,
|
||||
:kind => kind,
|
||||
:data => options
|
||||
)
|
||||
end
|
||||
|
||||
when 'Hash' # 'Gollum::Committer'
|
||||
actor = User.find_by_uname! record[:actor_name]
|
||||
project = Project.find record[:project_id]
|
||||
|
||||
project.admins.each do |recipient|
|
||||
ActivityFeed.create!(
|
||||
:user => recipient,
|
||||
:kind => 'wiki_new_commit_notification',
|
||||
:data => {:user_id => actor.id, :user_name => actor.name, :user_email => actor.email, :project_id => project.id,
|
||||
:project_name => project.name, :commit_sha => record[:commit_sha], :project_owner => project.owner.uname}
|
||||
)
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,59 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
module Modules::Observers::ActivityFeed::Issue
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
after_commit :new_issue_notifications, :on => :create
|
||||
after_update -> { send_assign_notifications(:update) }
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def new_issue_notifications
|
||||
collect_recipients.each do |recipient|
|
||||
next if user_id == recipient.id
|
||||
UserMailer.new_issue_notification(self, recipient).deliver if recipient.notifier.can_notify && recipient.notifier.new_issue
|
||||
ActivityFeed.create(
|
||||
:user => recipient,
|
||||
:kind => 'new_issue_notification',
|
||||
:data => {
|
||||
:user_name => user.name,
|
||||
:user_email => user.email,
|
||||
:user_id => user_id,
|
||||
:issue_serial_id => serial_id,
|
||||
:issue_title => title,
|
||||
:project_id => project.id,
|
||||
:project_name => project.name,
|
||||
:project_owner => project.owner.uname
|
||||
}
|
||||
)
|
||||
end
|
||||
send_assign_notifications
|
||||
end
|
||||
|
||||
def send_assign_notifications(action = :create)
|
||||
if assignee_id && assignee_id_changed?
|
||||
if assignee.notifier.issue_assign && assignee.notifier.can_notify
|
||||
user_mailer_action = action == :create ? :new_issue_notification : :issue_assign_notification
|
||||
UserMailer.send(user_mailer_action, self, assignee).deliver
|
||||
end
|
||||
ActivityFeed.create(
|
||||
:user => assignee,
|
||||
:kind => 'issue_assign_notification',
|
||||
:data => {
|
||||
:user_name => assignee.name,
|
||||
:user_email => assignee.email,
|
||||
:issue_serial_id => serial_id,
|
||||
:issue_title => title,
|
||||
:project_id => project.id,
|
||||
:project_name => project.name,
|
||||
:project_owner => project.owner.uname
|
||||
}
|
||||
)
|
||||
end
|
||||
project.hooks.each{ |h| h.receive_issues(self, action) } if action == :create || status_changed?
|
||||
# dont remove outdated issues link
|
||||
Comment.create_link_on_issues_from_item(self)
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,19 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
module Modules::Observers::ActivityFeed::User
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
after_commit :new_user_notification, :on => :create
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def new_user_notification
|
||||
ActivityFeed.create(
|
||||
:user => self,
|
||||
:kind => 'new_user_notification',
|
||||
:data => {:user_name => self.user_appeal, :user_email => self.email}
|
||||
)
|
||||
end
|
||||
|
||||
end
|
|
@ -73,10 +73,9 @@ describe Api::V1::IssuesController do
|
|||
end
|
||||
|
||||
it 'should return only assigned issue' do
|
||||
http_login(@issue.user)
|
||||
get :user_index, :format => :json
|
||||
assigns[:issues].should include(@own_hidden_issue)
|
||||
assigns[:issues].count.should == 1
|
||||
assigns[:issues].should have(1).item
|
||||
end
|
||||
|
||||
it 'should render right template for user index action' do
|
||||
|
@ -107,42 +106,32 @@ describe Api::V1::IssuesController do
|
|||
context 'for user' do
|
||||
before(:each) do
|
||||
http_login(@issue.user)
|
||||
@count = Issue.count
|
||||
end
|
||||
|
||||
it 'can create issue in own project' do
|
||||
post :create, @create_params
|
||||
Issue.count.should == @count+1
|
||||
lambda { post :create, @create_params}.should change{ Issue.count }.by(1)
|
||||
end
|
||||
|
||||
it 'can create issue in own hidden project' do
|
||||
post :create, @create_params.merge(:project_id => @own_hidden_project.id)
|
||||
Issue.count.should == @count+1
|
||||
lambda { post :create, @create_params.merge(:project_id => @own_hidden_project.id)}.should change{ Issue.count }.by(1)
|
||||
end
|
||||
|
||||
it 'can create issue in open project' do
|
||||
post :create, @create_params.merge(:project_id => @open_project.id)
|
||||
Issue.count.should == @count+1
|
||||
lambda { post :create, @create_params.merge(:project_id => @open_project.id)}.should change{ Issue.count }.by(1)
|
||||
end
|
||||
|
||||
it 'cant create issue in hidden project' do
|
||||
post :create, @create_params.merge(:project_id => @hidden_project.id)
|
||||
Issue.count.should == @count
|
||||
lambda { post :create, @create_params.merge(:project_id => @hidden_project.id)}.should change{ Issue.count }.by(0)
|
||||
end
|
||||
end
|
||||
|
||||
context 'for anonymous user' do
|
||||
before(:each) do
|
||||
@count = Issue.count
|
||||
end
|
||||
it 'cant create issue in project', :anonymous_access => true do
|
||||
post :create, @create_params
|
||||
Issue.count.should == @count
|
||||
lambda { post :create, @create_params}.should change{ Issue.count }.by(0)
|
||||
end
|
||||
|
||||
it 'cant create issue in hidden project', :anonymous_access => true do
|
||||
post :create, @create_params.merge(:project_id => @hidden_project.id)
|
||||
Issue.count.should == @count
|
||||
lambda { post :create, @create_params.merge(:project_id => @hidden_project.id)}.should change{ Issue.count }.by(0)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
@ -189,6 +178,7 @@ describe Api::V1::IssuesController do
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
after(:all) do
|
||||
User.destroy_all
|
||||
Platform.destroy_all
|
||||
|
|
|
@ -0,0 +1,269 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
require 'spec_helper'
|
||||
|
||||
def create_pull to_ref, from_ref, owner, project = @project
|
||||
pull = project.pull_requests.new :issue_attributes => {:title => 'test', :body => 'testing'}
|
||||
pull.issue.user, pull.issue.project = owner, pull.to_project
|
||||
pull.to_ref, pull.from_ref, pull.from_project = to_ref, from_ref, project
|
||||
pull.save; pull.check
|
||||
pull
|
||||
end
|
||||
|
||||
describe Api::V1::PullRequestsController do
|
||||
before(:all) do
|
||||
stub_symlink_methods
|
||||
stub_redis
|
||||
@project = FactoryGirl.create(:project_with_commit)
|
||||
@pull = create_pull 'master', 'non_conflicts', @project.owner
|
||||
|
||||
@another_project = FactoryGirl.create(:project_with_commit)
|
||||
@another_pull = create_pull 'master', 'non_conflicts', @another_project.owner, @another_project
|
||||
|
||||
@hidden_project = FactoryGirl.create(:project_with_commit)
|
||||
@hidden_project.update_column :visibility, 'hidden'
|
||||
@hidden_pull = create_pull 'master', 'non_conflicts', @hidden_project.owner, @hidden_project
|
||||
|
||||
@own_hidden_project = FactoryGirl.create(:project_with_commit, :owner => @project.owner)
|
||||
@own_hidden_project.update_column :visibility, 'hidden'
|
||||
@own_hidden_pull = create_pull 'master', 'non_conflicts', @own_hidden_project.owner, @own_hidden_project
|
||||
@own_hidden_pull.issue.update_column :assignee_id, @project.owner.id
|
||||
|
||||
@membered_project = FactoryGirl.create(:project_with_commit)
|
||||
@membered_pull = create_pull 'master', 'non_conflicts', @membered_project.owner, @membered_project
|
||||
@membered_project.relations.create(:role => 'reader', :actor => @pull.user)
|
||||
|
||||
@create_params = {:pull_request => {:title => 'title', :body => 'body',
|
||||
:from_ref => 'conflicts', :to_ref => 'master'},
|
||||
:project_id => @project.id, :format => :json}
|
||||
|
||||
@update_params = {:pull_request => {:title => 'new title'},
|
||||
:project_id => @project.id, :id => @pull.serial_id, :format => :json}
|
||||
end
|
||||
|
||||
context 'read and accessible abilities' do
|
||||
context 'for user' do
|
||||
before(:each) do
|
||||
http_login(@project.owner)
|
||||
end
|
||||
|
||||
it 'can show pull request in own project' do
|
||||
get :show, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'should render right template for show action' do
|
||||
get :show, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should render_template('api/v1/pull_requests/show')
|
||||
end
|
||||
|
||||
it 'can show pull request in open project' do
|
||||
get :show, :project_id => @another_project.id, :id => @another_pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'can show pull request in own hidden project' do
|
||||
get :show, :project_id => @own_hidden_project.id, :id => @own_hidden_pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'cant show pull request in hidden project' do
|
||||
get :show, :project_id => @hidden_project.id, :id => @hidden_pull.serial_id, :format => :json
|
||||
response.status.should == 403
|
||||
end
|
||||
|
||||
it 'should return three pull requests' do
|
||||
get :all_index, :filter => 'all', :format => :json
|
||||
assigns[:pulls].should include(@pull)
|
||||
assigns[:pulls].should include(@own_hidden_pull)
|
||||
assigns[:pulls].should include(@membered_pull)
|
||||
end
|
||||
|
||||
it 'should render right template for all index action' do
|
||||
get :all_index, :format => :json
|
||||
response.should render_template('api/v1/pull_requests/index')
|
||||
end
|
||||
|
||||
it 'should return only assigned pull request' do
|
||||
get :user_index, :format => :json
|
||||
assigns[:pulls].should include(@own_hidden_pull)
|
||||
assigns[:pulls].should have(1).item
|
||||
end
|
||||
|
||||
it 'should render right template for user index action' do
|
||||
get :user_index, :format => :json
|
||||
response.should render_template('api/v1/pull_requests/index')
|
||||
end
|
||||
|
||||
%w(commits files).each do |action|
|
||||
it "can show pull request #{action} in own project" do
|
||||
get action, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it "should render right template for commits action" do
|
||||
get action, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should render_template("api/v1/pull_requests/#{action}")
|
||||
end
|
||||
|
||||
it "can't show pull request #{action} in hidden project" do
|
||||
get action, :project_id => @hidden_project.id, :id => @hidden_pull.serial_id, :format => :json
|
||||
response.should_not be_success
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context 'for anonymous user' do
|
||||
it 'can show pull request in open project', :anonymous_access => true do
|
||||
get :show, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'cant show pull request in hidden project', :anonymous_access => true do
|
||||
@project.update_column :visibility, 'hidden'
|
||||
get :show, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.status.should == 403
|
||||
end
|
||||
|
||||
it 'should not return any pull requests' do
|
||||
get :all_index, :filter => 'all', :format => :json
|
||||
response.status.should == 401
|
||||
end
|
||||
|
||||
%w(commits files).each do |action|
|
||||
it "can show pull request #{action} in project" do
|
||||
get action, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it "should render right template for commits action" do
|
||||
get action, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
response.should render_template("api/v1/pull_requests/#{action}")
|
||||
end
|
||||
|
||||
it "can't show pull request #{action} in hidden project" do
|
||||
get action, :project_id => @hidden_project.id, :id => @hidden_pull.serial_id, :format => :json
|
||||
response.should_not be_success
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context 'create accessibility' do
|
||||
context 'for user' do
|
||||
before(:each) do
|
||||
http_login(@pull.user)
|
||||
end
|
||||
|
||||
it 'can create pull request in own project' do
|
||||
lambda { post :create, @create_params }.should change{ PullRequest.count }.by(1)
|
||||
end
|
||||
|
||||
it 'can create pull request in own hidden project' do
|
||||
lambda { post :create, @create_params.merge(:project_id => @own_hidden_project.id) }.should
|
||||
change{ PullRequest.count }.by(1)
|
||||
end
|
||||
|
||||
it 'can create pull request in open project' do
|
||||
lambda { post :create, @create_params.merge(:project_id => @another_project.id) }.should
|
||||
change{ PullRequest.count }.by(1)
|
||||
end
|
||||
|
||||
it 'cant create pull request in hidden project' do
|
||||
lambda { post :create, @create_params.merge(:project_id => @hidden_project.id) }.should
|
||||
change{ PullRequest.count }.by(0)
|
||||
end
|
||||
end
|
||||
|
||||
context 'for anonymous user' do
|
||||
it 'cant create pull request in project', :anonymous_access => true do
|
||||
lambda { post :create, @create_params }.should change{ PullRequest.count }.by(0)
|
||||
end
|
||||
|
||||
it 'cant create pull request in hidden project', :anonymous_access => true do
|
||||
lambda { post :create, @create_params.merge(:project_id => @hidden_project.id) }.should
|
||||
change{ PullRequest.count }.by(0)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context 'update accessibility' do
|
||||
context 'for user' do
|
||||
before(:each) do
|
||||
http_login(@project.owner)
|
||||
end
|
||||
|
||||
it 'can update pull request in own project' do
|
||||
put :update, @update_params
|
||||
@pull.reload.title.should == 'new title'
|
||||
end
|
||||
|
||||
it 'can update pull request in own hidden project' do
|
||||
put :update, @update_params.merge(:project_id => @own_hidden_project.id, :id => @own_hidden_pull.serial_id)
|
||||
@own_hidden_pull.reload.title.should == 'new title'
|
||||
end
|
||||
|
||||
it 'cant update pull request in open project' do
|
||||
put :update, @update_params.merge(:project_id => @another_project.id, :id => @another_pull.serial_id)
|
||||
@another_pull.reload.title.should_not == 'new title'
|
||||
end
|
||||
|
||||
it 'cant update pull request in hidden project' do
|
||||
put :update, @update_params.merge(:project_id => @hidden_project.id, :id => @hidden_pull.serial_id)
|
||||
@hidden_pull.reload.title.should_not == 'title'
|
||||
end
|
||||
|
||||
it 'can merge pull request in own project' do
|
||||
put :merge, :project_id => @project.id, :id => @pull.serial_id, :format => :json
|
||||
@pull.reload.status.should == 'merged'
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'can merge pull request in own hidden project' do
|
||||
put :merge, :project_id => @own_hidden_project.id, :id => @own_hidden_pull.serial_id, :format => :json
|
||||
@own_hidden_pull.reload.status.should == 'merged'
|
||||
response.should be_success
|
||||
end
|
||||
|
||||
it 'cant merge pull request in open project' do
|
||||
put :merge, :project_id => @another_project.id, :id => @another_pull.serial_id, :format => :json
|
||||
@another_pull.reload.status.should == 'ready'
|
||||
response.status.should == 403
|
||||
end
|
||||
|
||||
it 'cant merge pull request in hidden project' do
|
||||
put :merge, :project_id => @hidden_project.id, :id => @hidden_pull.serial_id, :format => :json
|
||||
@hidden_pull.reload.status.should == 'ready'
|
||||
response.status.should == 403
|
||||
end
|
||||
end
|
||||
|
||||
context 'for anonymous user' do
|
||||
it 'cant update pull request in project', :anonymous_access => true do
|
||||
put :update, @update_params
|
||||
response.status.should == 401
|
||||
end
|
||||
|
||||
it 'cant update pull request in hidden project', :anonymous_access => true do
|
||||
put :update, @update_params.merge(:project_id => @hidden_project.id, :id => @hidden_pull.serial_id)
|
||||
response.status.should == 401
|
||||
end
|
||||
|
||||
it 'cant merge pull request in open project' do
|
||||
put :merge, :project_id => @another_project.id, :id => @another_pull.serial_id, :format => :json
|
||||
@another_pull.reload.status.should == 'ready'
|
||||
response.status.should == 401
|
||||
end
|
||||
|
||||
it 'cant merge pull request in hidden project' do
|
||||
put :merge, :project_id => @hidden_project.id, :id => @hidden_pull.serial_id, :format => :json
|
||||
@hidden_pull.reload.status.should == 'ready'
|
||||
response.status.should == 401
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
after(:all) do
|
||||
User.destroy_all
|
||||
Platform.destroy_all
|
||||
end
|
||||
end
|
|
@ -4,5 +4,7 @@ FactoryGirl.define do
|
|||
body { FactoryGirl.generate(:string) }
|
||||
association :user, :factory => :user
|
||||
association :commentable, :factory => :issue
|
||||
project { |c| c.commentable.project }
|
||||
after(:create) { |c| c.send(:new_comment_notifications) }
|
||||
end
|
||||
end
|
||||
|
|
|
@ -7,5 +7,6 @@ FactoryGirl.define do
|
|||
association :user, :factory => :user
|
||||
association :assignee, :factory => :user
|
||||
status "open"
|
||||
after(:create) { |i| i.send(:new_issue_notifications) }
|
||||
end
|
||||
end
|
||||
|
|
|
@ -7,6 +7,7 @@ FactoryGirl.define do
|
|||
password '123456'
|
||||
password_confirmation {|u| u.password}
|
||||
confirmed_at { Time.now.utc }
|
||||
after(:create) { |u| u.send(:new_user_notification) }
|
||||
end
|
||||
|
||||
factory :admin, :parent => :user do
|
||||
|
|
|
@ -2,7 +2,6 @@
|
|||
require "spec_helper"
|
||||
|
||||
describe UserMailer do
|
||||
pending "add some examples to (or delete) #{__FILE__}"
|
||||
|
||||
context 'On Issue create' do
|
||||
before(:each) do
|
||||
|
|
|
@ -1,6 +0,0 @@
|
|||
# -*- encoding : utf-8 -*-
|
||||
require 'spec_helper'
|
||||
|
||||
describe ActivityFeedObserver do
|
||||
pending "add some examples to (or delete) #{__FILE__}"
|
||||
end
|
Loading…
Reference in New Issue