BigW Consortium Gitlab

Commit e6fbae25 by Phil Hughes

Merge branch 'master' into commit-limited-container-width

parents a3143f83 554c4e0d
class Admin::HooksController < Admin::ApplicationController class Admin::HooksController < Admin::ApplicationController
before_action :hook, only: :edit
def index def index
@hooks = SystemHook.all @hooks = SystemHook.all
@hook = SystemHook.new @hook = SystemHook.new
...@@ -15,15 +17,25 @@ class Admin::HooksController < Admin::ApplicationController ...@@ -15,15 +17,25 @@ class Admin::HooksController < Admin::ApplicationController
end end
end end
def edit
end
def update
if hook.update_attributes(hook_params)
flash[:notice] = 'System hook was successfully updated.'
redirect_to admin_hooks_path
else
render 'edit'
end
end
def destroy def destroy
@hook = SystemHook.find(params[:id]) hook.destroy
@hook.destroy
redirect_to admin_hooks_path redirect_to admin_hooks_path
end end
def test def test
@hook = SystemHook.find(params[:hook_id])
data = { data = {
event_name: "project_create", event_name: "project_create",
name: "Ruby", name: "Ruby",
...@@ -32,11 +44,17 @@ class Admin::HooksController < Admin::ApplicationController ...@@ -32,11 +44,17 @@ class Admin::HooksController < Admin::ApplicationController
owner_name: "Someone", owner_name: "Someone",
owner_email: "example@gitlabhq.com" owner_email: "example@gitlabhq.com"
} }
@hook.execute(data, 'system_hooks') hook.execute(data, 'system_hooks')
redirect_back_or_default redirect_back_or_default
end end
private
def hook
@hook ||= SystemHook.find(params[:id])
end
def hook_params def hook_params
params.require(:hook).permit( params.require(:hook).permit(
:enable_ssl_verification, :enable_ssl_verification,
......
class Projects::HooksController < Projects::ApplicationController class Projects::HooksController < Projects::ApplicationController
# Authorize # Authorize
before_action :authorize_admin_project! before_action :authorize_admin_project!
before_action :hook, only: :edit
respond_to :html respond_to :html
...@@ -17,6 +18,18 @@ class Projects::HooksController < Projects::ApplicationController ...@@ -17,6 +18,18 @@ class Projects::HooksController < Projects::ApplicationController
redirect_to namespace_project_settings_integrations_path(@project.namespace, @project) redirect_to namespace_project_settings_integrations_path(@project.namespace, @project)
end end
def edit
end
def update
if hook.update_attributes(hook_params)
flash[:notice] = 'Hook was successfully updated.'
redirect_to namespace_project_settings_integrations_path(@project.namespace, @project)
else
render 'edit'
end
end
def test def test
if !@project.empty_repo? if !@project.empty_repo?
status, message = TestHookService.new.execute(hook, current_user) status, message = TestHookService.new.execute(hook, current_user)
......
module MergeRequestsHelper module MergeRequestsHelper
def new_mr_path_from_push_event(event) def new_mr_path_from_push_event(event)
target_project = event.project.forked_from_project || event.project target_project = event.project.default_merge_request_target
new_namespace_project_merge_request_path( new_namespace_project_merge_request_path(
event.project.namespace, event.project.namespace,
event.project, event.project,
...@@ -127,6 +127,10 @@ module MergeRequestsHelper ...@@ -127,6 +127,10 @@ module MergeRequestsHelper
end end
end end
def target_projects(project)
[project, project.default_merge_request_target].uniq
end
def merge_request_button_visibility(merge_request, closed) def merge_request_button_visibility(merge_request, closed)
return 'hidden' if merge_request.closed? == closed || (merge_request.merged? == closed && !merge_request.closed?) || merge_request.closed_without_fork? return 'hidden' if merge_request.closed? == closed || (merge_request.merged? == closed && !merge_request.closed?) || merge_request.closed_without_fork?
end end
......
...@@ -100,6 +100,7 @@ class MergeRequest < ActiveRecord::Base ...@@ -100,6 +100,7 @@ class MergeRequest < ActiveRecord::Base
validates :merge_user, presence: true, if: :merge_when_pipeline_succeeds?, unless: :importing? validates :merge_user, presence: true, if: :merge_when_pipeline_succeeds?, unless: :importing?
validate :validate_branches, unless: [:allow_broken, :importing?, :closed_without_fork?] validate :validate_branches, unless: [:allow_broken, :importing?, :closed_without_fork?]
validate :validate_fork, unless: :closed_without_fork? validate :validate_fork, unless: :closed_without_fork?
validate :validate_target_project, on: :create
scope :by_source_or_target_branch, ->(branch_name) do scope :by_source_or_target_branch, ->(branch_name) do
where("source_branch = :branch OR target_branch = :branch", branch: branch_name) where("source_branch = :branch OR target_branch = :branch", branch: branch_name)
...@@ -330,6 +331,12 @@ class MergeRequest < ActiveRecord::Base ...@@ -330,6 +331,12 @@ class MergeRequest < ActiveRecord::Base
end end
end end
def validate_target_project
return true if target_project.merge_requests_enabled?
errors.add :base, 'Target project has disabled merge requests'
end
def validate_fork def validate_fork
return true unless target_project && source_project return true unless target_project && source_project
return true if target_project == source_project return true if target_project == source_project
......
...@@ -1314,6 +1314,14 @@ class Project < ActiveRecord::Base ...@@ -1314,6 +1314,14 @@ class Project < ActiveRecord::Base
namespace_id_changed? namespace_id_changed?
end end
def default_merge_request_target
if forked_from_project&.merge_requests_enabled?
forked_from_project
else
self
end
end
alias_method :name_with_namespace, :full_name alias_method :name_with_namespace, :full_name
alias_method :human_name, :full_name alias_method :human_name, :full_name
alias_method :path_with_namespace, :full_path alias_method :path_with_namespace, :full_path
......
...@@ -7,6 +7,9 @@ class StatusEntity < Grape::Entity ...@@ -7,6 +7,9 @@ class StatusEntity < Grape::Entity
expose :details_path expose :details_path
expose :favicon do |status| expose :favicon do |status|
ActionController::Base.helpers.image_path(File.join('ci_favicons', "#{status.favicon}.ico")) dir = 'ci_favicons'
dir = File.join(dir, 'dev') if Rails.env.development?
ActionController::Base.helpers.image_path(File.join(dir, "#{status.favicon}.ico"))
end end
end end
...@@ -28,7 +28,7 @@ module MergeRequests ...@@ -28,7 +28,7 @@ module MergeRequests
def find_target_project def find_target_project
return target_project if target_project.present? && can?(current_user, :read_project, target_project) return target_project if target_project.present? && can?(current_user, :read_project, target_project)
project.forked_from_project || project project.default_merge_request_target
end end
def find_target_branch def find_target_branch
......
...@@ -73,6 +73,12 @@ ...@@ -73,6 +73,12 @@
= container_reg = container_reg
%span.light.pull-right %span.light.pull-right
= boolean_to_icon Gitlab.config.registry.enabled = boolean_to_icon Gitlab.config.registry.enabled
- gitlab_pages = 'GitLab Pages'
- gitlab_pages_enabled = Gitlab.config.pages.enabled
%p{ "aria-label" => "#{gitlab_pages}: status " + (gitlab_pages_enabled ? "on" : "off") }
= gitlab_pages
%span.light.pull-right
= boolean_to_icon gitlab_pages_enabled
.col-md-4 .col-md-4
%h4 %h4
......
= form_errors(hook)
.form-group
= form.label :url, 'URL', class: 'control-label'
.col-sm-10
= form.text_field :url, class: 'form-control'
.form-group
= form.label :token, 'Secret Token', class: 'control-label'
.col-sm-10
= form.text_field :token, class: 'form-control'
%p.help-block
Use this token to validate received payloads
.form-group
= form.label :url, 'Trigger', class: 'control-label'
.col-sm-10.prepend-top-10
%div
System hook will be triggered on set of events like creating project
or adding ssh key. But you can also enable extra triggers like Push events.
.prepend-top-default
= form.check_box :push_events, class: 'pull-left'
.prepend-left-20
= form.label :push_events, class: 'list-label' do
%strong Push events
%p.light
This url will be triggered by a push to the repository
%div
= form.check_box :tag_push_events, class: 'pull-left'
.prepend-left-20
= form.label :tag_push_events, class: 'list-label' do
%strong Tag push events
%p.light
This url will be triggered when a new tag is pushed to the repository
.form-group
= form.label :enable_ssl_verification, 'SSL verification', class: 'control-label checkbox'
.col-sm-10
.checkbox
= form.label :enable_ssl_verification do
= form.check_box :enable_ssl_verification
%strong Enable SSL verification
- page_title 'Edit System Hook'
%h3.page-title
Edit System Hook
%p.light
#{link_to 'System hooks ', help_page_path('system_hooks/system_hooks'), class: 'vlink'} can be
used for binding events when GitLab creates a User or Project.
%hr
= form_for @hook, as: :hook, url: admin_hook_path, html: { class: 'form-horizontal' } do |f|
= render partial: 'form', locals: { form: f, hook: @hook }
.form-actions
= f.submit 'Save changes', class: 'btn btn-create'
- page_title "System Hooks" - page_title 'System Hooks'
%h3.page-title %h3.page-title
System hooks System hooks
%p.light %p.light
#{link_to "System hooks ", help_page_path("system_hooks/system_hooks"), class: "vlink"} can be #{link_to 'System hooks ', help_page_path('system_hooks/system_hooks'), class: 'vlink'} can be
used for binding events when GitLab creates a User or Project. used for binding events when GitLab creates a User or Project.
%hr %hr
= form_for @hook, as: :hook, url: admin_hooks_path, html: { class: 'form-horizontal' } do |f| = form_for @hook, as: :hook, url: admin_hooks_path, html: { class: 'form-horizontal' } do |f|
= form_errors(@hook) = render partial: 'form', locals: { form: f, hook: @hook }
.form-group
= f.label :url, 'URL', class: 'control-label'
.col-sm-10
= f.text_field :url, class: 'form-control'
.form-group
= f.label :token, 'Secret Token', class: 'control-label'
.col-sm-10
= f.text_field :token, class: 'form-control'
%p.help-block
Use this token to validate received payloads
.form-group
= f.label :url, "Trigger", class: 'control-label'
.col-sm-10.prepend-top-10
%div
System hook will be triggered on set of events like creating project
or adding ssh key. But you can also enable extra triggers like Push events.
.prepend-top-default
= f.check_box :push_events, class: 'pull-left'
.prepend-left-20
= f.label :push_events, class: 'list-label' do
%strong Push events
%p.light
This url will be triggered by a push to the repository
%div
= f.check_box :tag_push_events, class: 'pull-left'
.prepend-left-20
= f.label :tag_push_events, class: 'list-label' do
%strong Tag push events
%p.light
This url will be triggered when a new tag is pushed to the repository
.form-group
= f.label :enable_ssl_verification, "SSL verification", class: 'control-label checkbox'
.col-sm-10
.checkbox
= f.label :enable_ssl_verification do
= f.check_box :enable_ssl_verification
%strong Enable SSL verification
.form-actions .form-actions
= f.submit "Add system hook", class: "btn btn-create" = f.submit 'Add system hook', class: 'btn btn-create'
%hr %hr
- if @hooks.any? - if @hooks.any?
...@@ -62,11 +22,12 @@ ...@@ -62,11 +22,12 @@
- @hooks.each do |hook| - @hooks.each do |hook|
%li %li
.controls .controls
= link_to 'Test hook', admin_hook_test_path(hook), class: "btn btn-sm" = link_to 'Test hook', test_admin_hook_path(hook), class: 'btn btn-sm'
= link_to 'Remove', admin_hook_path(hook), data: { confirm: 'Are you sure?' }, method: :delete, class: "btn btn-remove btn-sm" = link_to 'Edit', edit_admin_hook_path(hook), class: 'btn btn-sm'
= link_to 'Remove', admin_hook_path(hook), data: { confirm: 'Are you sure?' }, method: :delete, class: 'btn btn-remove btn-sm'
.monospace= hook.url .monospace= hook.url
%div %div
- %w(push_events tag_push_events issues_events note_events merge_requests_events build_events).each do |trigger| - %w(push_events tag_push_events issues_events note_events merge_requests_events build_events).each do |trigger|
- if hook.send(trigger) - if hook.send(trigger)
%span.label.label-gray= trigger.titleize %span.label.label-gray= trigger.titleize
%span.label.label-gray SSL Verification: #{hook.enable_ssl_verification ? "enabled" : "disabled"} %span.label.label-gray SSL Verification: #{hook.enable_ssl_verification ? 'enabled' : 'disabled'}
...@@ -60,7 +60,7 @@ ...@@ -60,7 +60,7 @@
git init git init
git remote add origin #{ content_tag(:span, default_url_to_repo, class: 'clone')} git remote add origin #{ content_tag(:span, default_url_to_repo, class: 'clone')}
git add . git add .
git commit git commit -m "Initial commit"
git push -u origin master git push -u origin master
%fieldset %fieldset
......
= render 'shared/web_hooks/form', hook: @hook, hooks: @hooks, url_components: [@project.namespace.becomes(Namespace), @project] .row.prepend-top-default
.col-lg-3
%h4.prepend-top-0
= page_title
%p
#{link_to 'Webhooks', help_page_path('user/project/integrations/webhooks')} can be
used for binding events when something is happening within the project.
.col-lg-9.append-bottom-default
= form_for @hook, as: :hook, url: polymorphic_path([@project.namespace.becomes(Namespace), @project, :hooks]) do |f|
= render partial: 'shared/web_hooks/form', locals: { form: f, hook: @hook }
= f.submit 'Add webhook', class: 'btn btn-create'
%hr
%h5.prepend-top-default
Webhooks (#{@hooks.count})
- if @hooks.any?
%ul.well-list
- @hooks.each do |hook|
= render 'project_hook', hook: hook
- else
%p.settings-message.text-center.append-bottom-0
No webhooks found, add one in the form above.
= render 'projects/settings/head'
.row.prepend-top-default
.col-lg-3
%h4.prepend-top-0
= page_title
%p
#{link_to 'Webhooks', help_page_path('user/project/integrations/webhooks')} can be
used for binding events when something is happening within the project.
.col-lg-9.append-bottom-default
= form_for [@project.namespace.becomes(Namespace), @project, @hook], as: :hook, url: namespace_project_hook_path do |f|
= render partial: 'shared/web_hooks/form', locals: { form: f, hook: @hook }
= f.submit 'Save changes', class: 'btn btn-create'
...@@ -38,7 +38,7 @@ ...@@ -38,7 +38,7 @@
.panel-heading .panel-heading
Target branch Target branch
.panel-body.clearfix .panel-body.clearfix
- projects = @project.forked_from_project.nil? ? [@project] : [@project, @project.forked_from_project] - projects = target_projects(@project)
.merge-request-select.dropdown .merge-request-select.dropdown
= f.hidden_field :target_project_id = f.hidden_field :target_project_id
= dropdown_toggle f.object.target_project.path_with_namespace, { toggle: "dropdown", field_name: "#{f.object_name}[target_project_id]", disabled: @merge_request.persisted? }, { toggle_class: "js-compare-dropdown js-target-project" } = dropdown_toggle f.object.target_project.path_with_namespace, { toggle: "dropdown", field_name: "#{f.object_name}[target_project_id]", disabled: @merge_request.persisted? }, { toggle_class: "js-compare-dropdown js-target-project" }
......
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
%span %span
Members Members
- if can_edit - if can_edit
= nav_link(controller: [:integrations, :services]) do = nav_link(controller: [:integrations, :services, :hooks]) do
= link_to project_settings_integrations_path(@project), title: 'Integrations' do = link_to project_settings_integrations_path(@project), title: 'Integrations' do
%span %span
Integrations Integrations
......
...@@ -9,6 +9,7 @@ ...@@ -9,6 +9,7 @@
.col-md-4.col-lg-5.text-right-lg.prepend-top-5 .col-md-4.col-lg-5.text-right-lg.prepend-top-5
%span.append-right-10.inline %span.append-right-10.inline
SSL Verification: #{hook.enable_ssl_verification ? "enabled" : "disabled"} SSL Verification: #{hook.enable_ssl_verification ? "enabled" : "disabled"}
= link_to "Edit", edit_namespace_project_hook_path(@project.namespace, @project, hook), class: "btn btn-sm"
= link_to "Test", test_namespace_project_hook_path(@project.namespace, @project, hook), class: "btn btn-sm" = link_to "Test", test_namespace_project_hook_path(@project.namespace, @project, hook), class: "btn btn-sm"
= link_to namespace_project_hook_path(@project.namespace, @project, hook), data: { confirm: 'Are you sure?'}, method: :delete, class: "btn btn-transparent" do = link_to namespace_project_hook_path(@project.namespace, @project, hook), data: { confirm: 'Are you sure?'}, method: :delete, class: "btn btn-transparent" do
%span.sr-only Remove %span.sr-only Remove
......
.row.prepend-top-default = form_errors(hook)
.col-lg-3
%h4.prepend-top-0
= page_title
%p
#{link_to "Webhooks", help_page_path("user/project/integrations/webhooks")} can be
used for binding events when something is happening within the project.
.col-lg-9.append-bottom-default
= form_for hook, as: :hook, url: polymorphic_path(url_components + [:hooks]) do |f|
= form_errors(hook)
.form-group .form-group
= f.label :url, "URL", class: 'label-light' = form.label :url, 'URL', class: 'label-light'
= f.text_field :url, class: "form-control", placeholder: 'http://example.com/trigger-ci.json' = form.text_field :url, class: 'form-control', placeholder: 'http://example.com/trigger-ci.json'
.form-group .form-group
= f.label :token, "Secret Token", class: 'label-light' = form.label :token, 'Secret Token', class: 'label-light'
= f.text_field :token, class: "form-control", placeholder: '' = form.text_field :token, class: 'form-control', placeholder: ''
%p.help-block %p.help-block
Use this token to validate received payloads. It will be sent with the request in the X-Gitlab-Token HTTP header. Use this token to validate received payloads. It will be sent with the request in the X-Gitlab-Token HTTP header.
.form-group .form-group
= f.label :url, "Trigger", class: 'label-light' = form.label :url, 'Trigger', class: 'label-light'
%ul.list-unstyled %ul.list-unstyled
%li %li
= f.check_box :push_events, class: 'pull-left' = form.check_box :push_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :push_events, class: 'list-label' do = form.label :push_events, class: 'list-label' do
%strong Push events %strong Push events
%p.light %p.light
This URL will be triggered by a push to the repository This URL will be triggered by a push to the repository
%li %li
= f.check_box :tag_push_events, class: 'pull-left' = form.check_box :tag_push_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :tag_push_events, class: 'list-label' do = form.label :tag_push_events, class: 'list-label' do
%strong Tag push events %strong Tag push events
%p.light %p.light
This URL will be triggered when a new tag is pushed to the repository This URL will be triggered when a new tag is pushed to the repository
%li %li
= f.check_box :note_events, class: 'pull-left' = form.check_box :note_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :note_events, class: 'list-label' do = form.label :note_events, class: 'list-label' do
%strong Comments %strong Comments
%p.light %p.light
This URL will be triggered when someone adds a comment This URL will be triggered when someone adds a comment
%li %li
= f.check_box :issues_events, class: 'pull-left' = form.check_box :issues_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :issues_events, class: 'list-label' do = form.label :issues_events, class: 'list-label' do
%strong Issues events %strong Issues events
%p.light %p.light
This URL will be triggered when an issue is created/updated/merged This URL will be triggered when an issue is created/updated/merged
%li %li
= f.check_box :confidential_issues_events, class: 'pull-left' = form.check_box :confidential_issues_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :confidential_issues_events, class: 'list-label' do = form.label :confidential_issues_events, class: 'list-label' do
%strong Confidential Issues events %strong Confidential Issues events
%p.light %p.light
This URL will be triggered when a confidential issue is created/updated/merged This URL will be triggered when a confidential issue is created/updated/merged
%li %li
= f.check_box :merge_requests_events, class: 'pull-left' = form.check_box :merge_requests_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :merge_requests_events, class: 'list-label' do = form.label :merge_requests_events, class: 'list-label' do
%strong Merge Request events %strong Merge Request events
%p.light %p.light
This URL will be triggered when a merge request is created/updated/merged This URL will be triggered when a merge request is created/updated/merged
%li %li
= f.check_box :build_events, class: 'pull-left' = form.check_box :build_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :build_events, class: 'list-label' do = form.label :build_events, class: 'list-label' do
%strong Jobs events %strong Jobs events
%p.light %p.light
This URL will be triggered when the job status changes This URL will be triggered when the job status changes
%li %li
= f.check_box :pipeline_events, class: 'pull-left' = form.check_box :pipeline_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :pipeline_events, class: 'list-label' do = form.label :pipeline_events, class: 'list-label' do
%strong Pipeline events %strong Pipeline events
%p.light %p.light
This URL will be triggered when the pipeline status changes This URL will be triggered when the pipeline status changes
%li %li
= f.check_box :wiki_page_events, class: 'pull-left' = form.check_box :wiki_page_events, class: 'pull-left'
.prepend-left-20 .prepend-left-20
= f.label :wiki_page_events, class: 'list-label' do = form.label :wiki_page_events, class: 'list-label' do
%strong Wiki Page events %strong Wiki Page events
%p.light %p.light
This URL will be triggered when a wiki page is created/updated This URL will be triggered when a wiki page is created/updated
.form-group .form-group
= f.label :enable_ssl_verification, "SSL verification", class: 'label-light checkbox' = form.label :enable_ssl_verification, 'SSL verification', class: 'label-light checkbox'
.checkbox .checkbox
= f.label :enable_ssl_verification do = form.label :enable_ssl_verification do
= f.check_box :enable_ssl_verification = form.check_box :enable_ssl_verification
%strong Enable SSL verification %strong Enable SSL verification
= f.submit "Add webhook", class: "btn btn-create"
%hr
%h5.prepend-top-default
Webhooks (#{hooks.count})
- if hooks.any?
%ul.well-list
- hooks.each do |hook|
= render "project_hook", hook: hook
- else
%p.settings-message.text-center.append-bottom-0
No webhooks found, add one in the form above.
---
title: Implement ability to edit hooks
merge_request: 10816
author: Alexander Randa
---
title: Disallow merge requests from fork when source project have disabled merge requests
merge_request:
author: mhasbini
---
title: Display GitLab Pages status in Admin Dashboard
merge_request:
author:
---
title: Change Git commit command in Existing folder to git commit -m
merge_request: 10900
author: TM Lee
---
title: Updated CI status favicons to include the tanuki
merge_request: 10923
author:
...@@ -50,8 +50,10 @@ namespace :admin do ...@@ -50,8 +50,10 @@ namespace :admin do
resources :deploy_keys, only: [:index, :new, :create, :destroy] resources :deploy_keys, only: [:index, :new, :create, :destroy]
resources :hooks, only: [:index, :create, :destroy] do resources :hooks, only: [:index, :create, :edit, :update, :destroy] do
get :test member do
get :test
end
end end
resources :broadcast_messages, only: [:index, :edit, :create, :update, :destroy] do resources :broadcast_messages, only: [:index, :edit, :create, :update, :destroy] do
......
...@@ -185,7 +185,7 @@ constraints(ProjectUrlConstrainer.new) do ...@@ -185,7 +185,7 @@ constraints(ProjectUrlConstrainer.new) do
end end
end end
resources :hooks, only: [:index, :create, :destroy], constraints: { id: /\d+/ } do resources :hooks, only: [:index, :create, :edit, :update, :destroy], constraints: { id: /\d+/ } do
member do member do
get :test get :test
end end
......
...@@ -32,7 +32,7 @@ In brief: ...@@ -32,7 +32,7 @@ In brief:
As web terminals use WebSockets, every HTTP/HTTPS reverse proxy in front of As web terminals use WebSockets, every HTTP/HTTPS reverse proxy in front of
Workhorse needs to be configured to pass the `Connection` and `Upgrade` headers Workhorse needs to be configured to pass the `Connection` and `Upgrade` headers
through to the next one in the chain. If you installed Gitlab using Omnibus, or through to the next one in the chain. If you installed GitLab using Omnibus, or
from source, starting with GitLab 8.15, this should be done by the default from source, starting with GitLab 8.15, this should be done by the default
configuration, so there's no need for you to do anything. configuration, so there's no need for you to do anything.
...@@ -58,7 +58,7 @@ document for more details. ...@@ -58,7 +58,7 @@ document for more details.
If you'd like to disable web terminal support in GitLab, just stop passing If you'd like to disable web terminal support in GitLab, just stop passing
the `Connection` and `Upgrade` hop-by-hop headers in the *first* HTTP reverse the `Connection` and `Upgrade` hop-by-hop headers in the *first* HTTP reverse
proxy in the chain. For most users, this will be the NGINX server bundled with proxy in the chain. For most users, this will be the NGINX server bundled with
Omnibus Gitlab, in which case, you need to: Omnibus GitLab, in which case, you need to:
* Find the `nginx['proxy_set_headers']` section of your `gitlab.rb` file * Find the `nginx['proxy_set_headers']` section of your `gitlab.rb` file
* Ensure the whole block is uncommented, and then comment out or remove the * Ensure the whole block is uncommented, and then comment out or remove the
......
# How to create a project in GitLab # How to create a project in GitLab
There are two ways to create a new project in GitLab. 1. In your dashboard, click the green **New project** button or use the plus
icon in the upper right corner of the navigation bar.
1. While in your dashboard, you can create a new project using the **New project**
green button or you can use the cross icon in the upper right corner next to
your avatar which is always visible.
![Create a project](img/create_new_project_button.png) ![Create a project](img/create_new_project_button.png)
1. From there you can see several options. 1. This opens the **New project** page.
![Project information](img/create_new_project_info.png) ![Project information](img/create_new_project_info.png)
1. Fill out the information: 1. Provide the following information:
- Enter the name of your project in the **Project name** field. You can't use
1. "Project name" is the name of your project (you can't use special characters, special characters, but you can use spaces, hyphens, underscores or even
but you can use spaces, hyphens, underscores or even emojis). emoji.
1. The "Project description" is optional and will be shown in your project's - If you have a project in a different repository, you can [import it] by
dashboard so others can briefly understand what your project is about. clicking an **Import project from** button provided this is enabled in
1. Select a [visibility level](../public_access/public_access.md). your GitLab instance. Ask your administrator if not.
1. You can also [import your existing projects](../workflow/importing/README.md). - The **Project description (optional)** field enables you to enter a
description for your project's dashboard, which will help others
1. Finally, click **Create project**. understand what your project is about. Though it's not required, it's a good
idea to fill this in.
- Changing the **Visibility Level** modifies the project's
[viewing and access rights](../public_access/public_access.md) for users.
1. Click **Create project**.
[import it]: ../workflow/importing/README.md
...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc ...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc
### 4. Get latest code ### 4. Get latest code
```bash ```bash
cd /home/git/gitlab
sudo -u git -H git fetch --all sudo -u git -H git fetch --all
sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically
``` ```
......
...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc ...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc
### 4. Get latest code ### 4. Get latest code
```bash ```bash
cd /home/git/gitlab
sudo -u git -H git fetch --all sudo -u git -H git fetch --all
sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically
``` ```
......
...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc ...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc
### 4. Get latest code ### 4. Get latest code
```bash ```bash
cd /home/git/gitlab
sudo -u git -H git fetch --all sudo -u git -H git fetch --all
sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically
``` ```
......
...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc ...@@ -49,6 +49,8 @@ sudo gem install bundler --no-ri --no-rdoc
### 4. Get latest code ### 4. Get latest code
```bash ```bash
cd /home/git/gitlab
sudo -u git -H git fetch --all sudo -u git -H git fetch --all
sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically sudo -u git -H git checkout -- db/schema.rb # local changes will be restored automatically
``` ```
......
...@@ -20,6 +20,8 @@ module API ...@@ -20,6 +20,8 @@ module API
error!(errors[:validate_fork], 422) error!(errors[:validate_fork], 422)
elsif errors[:validate_branches].any? elsif errors[:validate_branches].any?
conflict!(errors[:validate_branches]) conflict!(errors[:validate_branches])
elsif errors[:base].any?
error!(errors[:base], 422)
end end
render_api_error!(errors, 400) render_api_error!(errors, 400)
......
...@@ -23,6 +23,8 @@ module API ...@@ -23,6 +23,8 @@ module API
error!(errors[:validate_fork], 422) error!(errors[:validate_fork], 422)
elsif errors[:validate_branches].any? elsif errors[:validate_branches].any?
conflict!(errors[:validate_branches]) conflict!(errors[:validate_branches])
elsif errors[:base].any?
error!(errors[:base], 422)
end end
render_api_error!(errors, 400) render_api_error!(errors, 400)
......
FactoryGirl.define do FactoryGirl.define do
factory :project_hook do factory :project_hook do
url { generate(:url) } url { generate(:url) }
enable_ssl_verification false
trait :token do trait :token do
token { SecureRandom.hex(10) } token { SecureRandom.hex(10) }
...@@ -11,6 +12,7 @@ FactoryGirl.define do ...@@ -11,6 +12,7 @@ FactoryGirl.define do
merge_requests_events true merge_requests_events true
tag_push_events true tag_push_events true
issues_events true issues_events true
confidential_issues_events true
note_events true note_events true
build_events true build_events true
pipeline_events true pipeline_events true
......
require 'spec_helper' require 'spec_helper'
describe "Admin::Hooks", feature: true do describe 'Admin::Hooks', feature: true do
before do before do
@project = create(:project) @project = create(:project)
login_as :admin login_as :admin
...@@ -8,24 +8,24 @@ describe "Admin::Hooks", feature: true do ...@@ -8,24 +8,24 @@ describe "Admin::Hooks", feature: true do
@system_hook = create(:system_hook) @system_hook = create(:system_hook)
end end
describe "GET /admin/hooks" do describe 'GET /admin/hooks' do
it "is ok" do it 'is ok' do
visit admin_root_path visit admin_root_path
page.within ".layout-nav" do page.within '.layout-nav' do
click_on "Hooks" click_on 'Hooks'
end end
expect(current_path).to eq(admin_hooks_path) expect(current_path).to eq(admin_hooks_path)
end end
it "has hooks list" do it 'has hooks list' do
visit admin_hooks_path visit admin_hooks_path
expect(page).to have_content(@system_hook.url) expect(page).to have_content(@system_hook.url)
end end
end end
describe "New Hook" do describe 'New Hook' do
let(:url) { generate(:url) } let(:url) { generate(:url) }
it 'adds new hook' do it 'adds new hook' do
...@@ -40,11 +40,36 @@ describe "Admin::Hooks", feature: true do ...@@ -40,11 +40,36 @@ describe "Admin::Hooks", feature: true do
end end
end end
describe "Test" do describe 'Update existing hook' do
let(:new_url) { generate(:url) }
it 'updates existing hook' do
visit admin_hooks_path
click_link 'Edit'
fill_in 'hook_url', with: new_url
check 'Enable SSL verification'
click_button 'Save changes'
expect(page).to have_content 'SSL Verification: enabled'
expect(current_path).to eq(admin_hooks_path)
expect(page).to have_content(new_url)
end
end
describe 'Remove existing hook' do
it 'remove existing hook' do
visit admin_hooks_path
expect { click_link 'Remove' }.to change(SystemHook, :count).by(-1)
end
end
describe 'Test' do
before do before do
WebMock.stub_request(:post, @system_hook.url) WebMock.stub_request(:post, @system_hook.url)
visit admin_hooks_path visit admin_hooks_path
click_link "Test hook" click_link 'Test hook'
end end
it { expect(current_path).to eq(admin_hooks_path) } it { expect(current_path).to eq(admin_hooks_path) }
......
...@@ -231,7 +231,7 @@ feature 'File blob', :js, feature: true do ...@@ -231,7 +231,7 @@ feature 'File blob', :js, feature: true do
branch_name: 'master', branch_name: 'master',
commit_message: "Add PDF", commit_message: "Add PDF",
file_path: 'files/test.pdf', file_path: 'files/test.pdf',
file_content: File.read(Rails.root.join('spec/javascripts/blob/pdf/test.pdf')) file_content: project.repository.blob_at('add-pdf-file', 'files/pdf/test.pdf').data
).execute ).execute
visit_blob('files/test.pdf') visit_blob('files/test.pdf')
......
require 'spec_helper'
feature 'Integration settings', feature: true do
let(:project) { create(:empty_project) }
let(:user) { create(:user) }
let(:role) { :developer }
let(:integrations_path) { namespace_project_settings_integrations_path(project.namespace, project) }
background do
login_as(user)
project.team << [user, role]
end
context 'for developer' do
given(:role) { :developer }
scenario 'to be disallowed to view' do
visit integrations_path
expect(page.status_code).to eq(404)
end
end
context 'for master' do
given(:role) { :master }
context 'Webhooks' do
let(:hook) { create(:project_hook, :all_events_enabled, enable_ssl_verification: true, project: project) }
let(:url) { generate(:url) }
scenario 'show list of webhooks' do
hook
visit integrations_path
expect(page.status_code).to eq(200)
expect(page).to have_content(hook.url)
expect(page).to have_content('SSL Verification: enabled')
expect(page).to have_content('Push Events')
expect(page).to have_content('Tag Push Events')
expect(page).to have_content('Issues Events')
expect(page).to have_content('Confidential Issues Events')
expect(page).to have_content('Note Events')
expect(page).to have_content('Merge Requests Events')
expect(page).to have_content('Pipeline Events')
expect(page).to have_content('Wiki Page Events')
end
scenario 'create webhook' do
visit integrations_path
fill_in 'hook_url', with: url
check 'Tag push events'
check 'Enable SSL verification'
click_button 'Add webhook'
expect(page).to have_content(url)
expect(page).to have_content('SSL Verification: enabled')
expect(page).to have_content('Push Events')
expect(page).to have_content('Tag Push Events')
end
scenario 'edit existing webhook' do
hook
visit integrations_path
click_link 'Edit'
fill_in 'hook_url', with: url
check 'Enable SSL verification'
click_button 'Save changes'
expect(page).to have_content 'SSL Verification: enabled'
expect(page).to have_content(url)
end
scenario 'test existing webhook' do
WebMock.stub_request(:post, hook.url)
visit integrations_path
click_link 'Test'
expect(current_path).to eq(integrations_path)
end
scenario 'remove existing webhook' do
hook
visit integrations_path
expect { click_link 'Remove' }.to change(ProjectHook, :count).by(-1)
end
end
end
end
...@@ -64,7 +64,7 @@ describe MergeRequestsHelper do ...@@ -64,7 +64,7 @@ describe MergeRequestsHelper do
it do it do
@project = project @project = project
is_expected.to eq("#1, #2, and #{other_project.namespace.path}/#{other_project.path}#3") is_expected.to eq("#1, #2, and #{other_project.namespace.path}/#{other_project.path}#3")
end end
end end
...@@ -149,6 +149,50 @@ describe MergeRequestsHelper do ...@@ -149,6 +149,50 @@ describe MergeRequestsHelper do
end end
end end
describe '#target_projects' do
let(:project) { create(:empty_project) }
let(:fork_project) { create(:empty_project, forked_from_project: project) }
context 'when target project has enabled merge requests' do
it 'returns the forked_from project' do
expect(target_projects(fork_project)).to contain_exactly(project, fork_project)
end
end
context 'when target project has disabled merge requests' do
it 'returns the forked project' do
project.project_feature.update(merge_requests_access_level: 0)
expect(target_projects(fork_project)).to contain_exactly(fork_project)
end
end
end
describe '#new_mr_path_from_push_event' do
subject(:url_params) { URI.decode_www_form(new_mr_path_from_push_event(event)).to_h }
let(:user) { create(:user) }
let(:project) { create(:empty_project, creator: user) }
let(:fork_project) { create(:project, forked_from_project: project, creator: user) }
let(:event) do
push_data = Gitlab::DataBuilder::Push.build_sample(fork_project, user)
create(:event, :pushed, project: fork_project, target: fork_project, author: user, data: push_data)
end
context 'when target project has enabled merge requests' do
it 'returns link to create merge request on source project' do
expect(url_params['merge_request[target_project_id]'].to_i).to eq(project.id)
end
end
context 'when target project has disabled merge requests' do
it 'returns link to create merge request on forked project' do
project.project_feature.update(merge_requests_access_level: 0)
expect(url_params['merge_request[target_project_id]'].to_i).to eq(fork_project.id)
end
end
end
describe '#mr_issues_mentioned_but_not_closing' do describe '#mr_issues_mentioned_but_not_closing' do
let(:user_1) { create(:user) } let(:user_1) { create(:user) }
let(:user_2) { create(:user) } let(:user_2) { create(:user) }
......
...@@ -434,6 +434,19 @@ describe API::MergeRequests do ...@@ -434,6 +434,19 @@ describe API::MergeRequests do
expect(json_response['title']).to eq('Test merge_request') expect(json_response['title']).to eq('Test merge_request')
end end
it 'returns 422 when target project has disabled merge requests' do
project.project_feature.update(merge_requests_access_level: 0)
post api("/projects/#{fork_project.id}/merge_requests", user2),
title: 'Test',
target_branch: 'master',
source_branch: 'markdown',
author: user2,
target_project_id: project.id
expect(response).to have_http_status(422)
end
it "returns 400 when source_branch is missing" do it "returns 400 when source_branch is missing" do
post api("/projects/#{fork_project.id}/merge_requests", user2), post api("/projects/#{fork_project.id}/merge_requests", user2),
title: 'Test merge_request', target_branch: "master", author: user2, target_project_id: project.id title: 'Test merge_request', target_branch: "master", author: user2, target_project_id: project.id
......
...@@ -338,6 +338,19 @@ describe API::MergeRequests do ...@@ -338,6 +338,19 @@ describe API::MergeRequests do
expect(json_response['title']).to eq('Test merge_request') expect(json_response['title']).to eq('Test merge_request')
end end
it "returns 422 when target project has disabled merge requests" do
project.project_feature.update(merge_requests_access_level: 0)
post v3_api("/projects/#{fork_project.id}/merge_requests", user2),
title: 'Test',
target_branch: "master",
source_branch: 'markdown',
author: user2,
target_project_id: project.id
expect(response).to have_http_status(422)
end
it "returns 400 when source_branch is missing" do it "returns 400 when source_branch is missing" do
post v3_api("/projects/#{fork_project.id}/merge_requests", user2), post v3_api("/projects/#{fork_project.id}/merge_requests", user2),
title: 'Test merge_request', target_branch: "master", author: user2, target_project_id: project.id title: 'Test merge_request', target_branch: "master", author: user2, target_project_id: project.id
......
...@@ -71,13 +71,15 @@ describe Admin::ProjectsController, "routing" do ...@@ -71,13 +71,15 @@ describe Admin::ProjectsController, "routing" do
end end
end end
# admin_hook_test GET /admin/hooks/:hook_id/test(.:format) admin/hooks#test # admin_hook_test GET /admin/hooks/:id/test(.:format) admin/hooks#test
# admin_hooks GET /admin/hooks(.:format) admin/hooks#index # admin_hooks GET /admin/hooks(.:format) admin/hooks#index
# POST /admin/hooks(.:format) admin/hooks#create # POST /admin/hooks(.:format) admin/hooks#create
# admin_hook DELETE /admin/hooks/:id(.:format) admin/hooks#destroy # admin_hook DELETE /admin/hooks/:id(.:format) admin/hooks#destroy
# PUT /admin/hooks/:id(.:format) admin/hooks#update
# edit_admin_hook GET /admin/hooks/:id(.:format) admin/hooks#edit
describe Admin::HooksController, "routing" do describe Admin::HooksController, "routing" do
it "to #test" do it "to #test" do
expect(get("/admin/hooks/1/test")).to route_to('admin/hooks#test', hook_id: '1') expect(get("/admin/hooks/1/test")).to route_to('admin/hooks#test', id: '1')
end end
it "to #index" do it "to #index" do
...@@ -88,6 +90,14 @@ describe Admin::HooksController, "routing" do ...@@ -88,6 +90,14 @@ describe Admin::HooksController, "routing" do
expect(post("/admin/hooks")).to route_to('admin/hooks#create') expect(post("/admin/hooks")).to route_to('admin/hooks#create')
end end
it "to #edit" do
expect(get("/admin/hooks/1/edit")).to route_to('admin/hooks#edit', id: '1')
end
it "to #update" do
expect(put("/admin/hooks/1")).to route_to('admin/hooks#update', id: '1')
end
it "to #destroy" do it "to #destroy" do
expect(delete("/admin/hooks/1")).to route_to('admin/hooks#destroy', id: '1') expect(delete("/admin/hooks/1")).to route_to('admin/hooks#destroy', id: '1')
end end
......
...@@ -340,14 +340,16 @@ describe 'project routing' do ...@@ -340,14 +340,16 @@ describe 'project routing' do
# test_project_hook GET /:project_id/hooks/:id/test(.:format) hooks#test # test_project_hook GET /:project_id/hooks/:id/test(.:format) hooks#test
# project_hooks GET /:project_id/hooks(.:format) hooks#index # project_hooks GET /:project_id/hooks(.:format) hooks#index
# POST /:project_id/hooks(.:format) hooks#create # POST /:project_id/hooks(.:format) hooks#create
# project_hook DELETE /:project_id/hooks/:id(.:format) hooks#destroy # edit_project_hook GET /:project_id/hooks/:id/edit(.:format) hooks#edit
# project_hook PUT /:project_id/hooks/:id(.:format) hooks#update
# DELETE /:project_id/hooks/:id(.:format) hooks#destroy
describe Projects::HooksController, 'routing' do describe Projects::HooksController, 'routing' do
it 'to #test' do it 'to #test' do
expect(get('/gitlab/gitlabhq/hooks/1/test')).to route_to('projects/hooks#test', namespace_id: 'gitlab', project_id: 'gitlabhq', id: '1') expect(get('/gitlab/gitlabhq/hooks/1/test')).to route_to('projects/hooks#test', namespace_id: 'gitlab', project_id: 'gitlabhq', id: '1')
end end
it_behaves_like 'RESTful project resources' do it_behaves_like 'RESTful project resources' do
let(:actions) { [:index, :create, :destroy] } let(:actions) { [:index, :create, :destroy, :edit, :update] }
let(:controller) { 'hooks' } let(:controller) { 'hooks' }
end end
end end
......
...@@ -18,6 +18,12 @@ describe StatusEntity do ...@@ -18,6 +18,12 @@ describe StatusEntity do
it 'contains status details' do it 'contains status details' do
expect(subject).to include :text, :icon, :favicon, :label, :group expect(subject).to include :text, :icon, :favicon, :label, :group
expect(subject).to include :has_details, :details_path expect(subject).to include :has_details, :details_path
expect(subject[:favicon]).to eq('/assets/ci_favicons/favicon_status_success.ico')
end
it 'contains a dev namespaced favicon if dev env' do
allow(Rails.env).to receive(:development?) { true }
expect(entity.as_json[:favicon]).to eq('/assets/ci_favicons/dev/favicon_status_success.ico')
end end
end end
end end
...@@ -261,6 +261,16 @@ describe MergeRequests::BuildService, services: true do ...@@ -261,6 +261,16 @@ describe MergeRequests::BuildService, services: true do
end end
end end
context 'upstream project has disabled merge requests' do
let(:upstream_project) { create(:empty_project, :merge_requests_disabled) }
let(:project) { create(:empty_project, forked_from_project: upstream_project) }
let(:commits) { Commit.decorate([commit_1], project) }
it 'sets target project correctly' do
expect(merge_request.target_project).to eq(project)
end
end
context 'target_project is set and accessible by current_user' do context 'target_project is set and accessible by current_user' do
let(:target_project) { create(:project, :public, :repository)} let(:target_project) { create(:project, :public, :repository)}
let(:commits) { Commit.decorate([commit_1], project) } let(:commits) { Commit.decorate([commit_1], project) }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment