Commit 0935dfb5 authored by Pavel Shutsin's avatar Pavel Shutsin

Add endpoint for embeddable insights

Some of insights charts can be embedded to
handbook pages now.
parent 56e102a2
...@@ -3,6 +3,20 @@ ...@@ -3,6 +3,20 @@
module InsightsActions module InsightsActions
extend ActiveSupport::Concern extend ActiveSupport::Concern
module Embeddable
extend ActiveSupport::Concern
def embedded
response.set_header('X-Frame-Options', 'SAMEORIGIN')
return render_404 unless Feature.enabled?(:embed_analytics_report, insights_entity)
return render_404 unless can?(current_user, :embed_analytics_report, insights_entity)
render :embedded, layout: false
end
end
include Embeddable
included do included do
before_action :check_insights_available! before_action :check_insights_available!
before_action :validate_params, only: [:query] before_action :validate_params, only: [:query]
......
...@@ -251,6 +251,8 @@ module EE ...@@ -251,6 +251,8 @@ module EE
rule { can?(:maintainer_access) & push_rules_available }.enable :change_push_rules rule { can?(:maintainer_access) & push_rules_available }.enable :change_push_rules
rule { admin & is_gitlab_com }.enable :update_subscription_limit rule { admin & is_gitlab_com }.enable :update_subscription_limit
rule { public_group }.enable :embed_analytics_report
end end
override :lookup_access_level! override :lookup_access_level!
......
...@@ -419,6 +419,8 @@ module EE ...@@ -419,6 +419,8 @@ module EE
rule { status_page_available & can?(:owner_access) }.enable :mark_issue_for_publication rule { status_page_available & can?(:owner_access) }.enable :mark_issue_for_publication
rule { status_page_available & can?(:developer_access) }.enable :publish_status_page rule { status_page_available & can?(:developer_access) }.enable :publish_status_page
rule { public_project }.enable :embed_analytics_report
end end
override :lookup_access_level! override :lookup_access_level!
......
- @no_container = true
= render('shared/insights', endpoint: group_insights_path(@group, format: :json), query_endpoint: query_group_insights_path(@group))
- @no_container = true
= render('shared/insights', endpoint: namespace_project_insights_path(@project.namespace, @project, format: :json), query_endpoint: query_namespace_project_insights_path(@project.namespace, @project), notice: project_insights_config.notice_text)
...@@ -61,6 +61,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do ...@@ -61,6 +61,7 @@ constraints(::Constraints::GroupUrlConstrainer.new) do
resource :insights, only: [:show], trailing_slash: true do resource :insights, only: [:show], trailing_slash: true do
collection do collection do
post :query post :query
get :embedded
end end
end end
......
...@@ -123,6 +123,7 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do ...@@ -123,6 +123,7 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do
resource :insights, only: [:show], trailing_slash: true do resource :insights, only: [:show], trailing_slash: true do
collection do collection do
post :query post :query
get :embedded
end end
end end
# All new routes should go under /-/ scope. # All new routes should go under /-/ scope.
......
...@@ -145,5 +145,47 @@ RSpec.describe Groups::InsightsController do ...@@ -145,5 +145,47 @@ RSpec.describe Groups::InsightsController do
it_behaves_like '200 status' it_behaves_like '200 status'
end end
end end
describe 'GET #embedded' do
subject { get :embedded, params: params.merge(group_id: parent_group.to_param) }
shared_examples 'has iframe options set' do
it 'sets SAMEORIGIN frame option' do
subject
expect(response.headers['X-Frame-Options']).to eq 'SAMEORIGIN'
end
end
context 'when feature is disabled' do
before do
stub_feature_flags(embed_analytics_report: false)
end
it_behaves_like '404 status'
include_examples 'has iframe options set'
end
context 'when project is public' do
let_it_be(:parent_group) { create(:group, :public) }
let_it_be(:project) { create(:project, :public) }
it_behaves_like '200 status'
include_examples 'has iframe options set'
end
context 'when project is internal' do
let_it_be(:parent_group) { create(:group, :internal) }
let_it_be(:project) { create(:project, :internal) }
it_behaves_like '404 status'
include_examples 'has iframe options set'
end
context 'when project is private' do
it_behaves_like '404 status'
include_examples 'has iframe options set'
end
end
end end
end end
...@@ -1046,4 +1046,6 @@ RSpec.describe GroupPolicy do ...@@ -1046,4 +1046,6 @@ RSpec.describe GroupPolicy do
end end
it_behaves_like 'update namespace limit policy' it_behaves_like 'update namespace limit policy'
include_examples 'analytics report embedding'
end end
...@@ -1473,4 +1473,6 @@ RSpec.describe ProjectPolicy do ...@@ -1473,4 +1473,6 @@ RSpec.describe ProjectPolicy do
it { is_expected.to(allowed ? be_allowed(policy) : be_disallowed(policy)) } it { is_expected.to(allowed ? be_allowed(policy) : be_disallowed(policy)) }
end end
end end
include_examples 'analytics report embedding'
end end
# frozen_string_literal: true
RSpec.shared_examples 'analytics report embedding' do
let(:current_user) { nil }
context 'when subject is not public' do
before do
allow(subject.subject).to receive(:public?).and_return(false)
end
it { is_expected.to be_disallowed(:embed_analytics_report) }
end
context 'when subject is public' do
before do
allow(subject.subject).to receive(:public?).and_return(true)
end
it { is_expected.to be_allowed(:embed_analytics_report) }
end
end
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