summaryrefslogtreecommitdiff
path: root/spec/features/static_site_editor_spec.rb
blob: a47579582e2a42731a3a9aca68f9f17b53d20b6d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
# frozen_string_literal: true

require 'spec_helper'

RSpec.describe 'Static Site Editor' do
  let_it_be(:user) { create(:user) }
  let_it_be(:project) { create(:project, :public, :repository) }

  let(:sse_path) { project_show_sse_path(project, 'master/README.md') }

  before_all do
    project.add_developer(user)
  end

  before do
    sign_in(user)
  end

  context "when no config file is present" do
    before do
      visit sse_path
    end

    it 'renders SSE page with all generated config values and default config file values' do
      node = page.find('#static-site-editor')

      # assert generated config values are present
      expect(node['data-base-url']).to eq("/#{project.full_path}/-/sse/master%2FREADME.md")
      expect(node['data-branch']).to eq('master')
      expect(node['data-commit-id']).to match(/\A[0-9a-f]{40}\z/)
      expect(node['data-is-supported-content']).to eq('true')
      expect(node['data-merge-requests-illustration-path'])
        .to match(%r{/assets/illustrations/merge_requests-.*\.svg})
      expect(node['data-namespace']).to eq(project.namespace.full_path)
      expect(node['data-project']).to eq(project.path)
      expect(node['data-project-id']).to eq(project.id.to_s)

      # assert default config file values are present
      expect(node['data-image-upload-path']).to eq('source/images')
      expect(node['data-mounts']).to eq('[{"source":"source","target":""}]')
      expect(node['data-static-site-generator']).to eq('middleman')
    end
  end

  context "when a config file is present" do
    let(:config_file_yml) do
      <<~YAML
        image_upload_path: custom-image-upload-path
        mounts:
          - source: source1
            target: ""
          - source: source2
            target: target2
        static_site_generator: middleman
      YAML
    end

    before do
      allow_next_instance_of(Repository) do |repository|
        allow(repository).to receive(:blob_data_at).and_return(config_file_yml)
      end

      visit sse_path
    end

    it 'renders Static Site Editor page values read from config file' do
      node = page.find('#static-site-editor')

      # assert user-specified config file values are present
      expected_mounts = '[{"source":"source1","target":""},{"source":"source2","target":"target2"}]'
      expect(node['data-image-upload-path']).to eq('custom-image-upload-path')
      expect(node['data-mounts']).to eq(expected_mounts)
      expect(node['data-static-site-generator']).to eq('middleman')
    end
  end

  describe 'Static Site Editor Content Security Policy' do
    subject { response_headers['Content-Security-Policy'] }

    context 'when no global CSP config exists' do
      before do
        expect_next_instance_of(Projects::StaticSiteEditorController) do |controller|
          expect(controller).to receive(:current_content_security_policy)
            .and_return(ActionDispatch::ContentSecurityPolicy.new)
        end
      end

      it 'does not add CSP directives' do
        visit sse_path

        is_expected.to be_blank
      end
    end

    context 'when a global CSP config exists' do
      let_it_be(:cdn_url) { 'https://some-cdn.test' }
      let_it_be(:youtube_url) { 'https://www.youtube.com' }

      before do
        csp = ActionDispatch::ContentSecurityPolicy.new do |p|
          p.frame_src :self, cdn_url
        end

        expect_next_instance_of(Projects::StaticSiteEditorController) do |controller|
          expect(controller).to receive(:current_content_security_policy).and_return(csp)
        end
      end

      it 'appends youtube to the CSP frame-src policy' do
        visit sse_path

        is_expected.to eql("frame-src 'self' #{cdn_url} #{youtube_url}")
      end
    end
  end
end