Commit 25521def authored by GitLab Bot's avatar GitLab Bot

Add latest changes from gitlab-org/gitlab@master

parent 9a1c5456
<script> <script>
/* eslint-disable @gitlab/vue-i18n/no-bare-strings */
import { GlTooltipDirective, GlLink, GlButton, GlLoadingIcon } from '@gitlab/ui'; import { GlTooltipDirective, GlLink, GlButton, GlLoadingIcon } from '@gitlab/ui';
import { sprintf, s__ } from '~/locale'; import { sprintf, s__ } from '~/locale';
import Icon from '../../vue_shared/components/icon.vue'; import Icon from '../../vue_shared/components/icon.vue';
...@@ -113,7 +112,7 @@ export default { ...@@ -113,7 +112,7 @@ export default {
> >
{{ commit.author.name }} {{ commit.author.name }}
</gl-link> </gl-link>
authored {{ s__('LastCommit|authored') }}
<timeago-tooltip :time="commit.authoredDate" tooltip-placement="bottom" /> <timeago-tooltip :time="commit.authoredDate" tooltip-placement="bottom" />
</div> </div>
<pre <pre
...@@ -125,6 +124,7 @@ export default { ...@@ -125,6 +124,7 @@ export default {
</pre> </pre>
</div> </div>
<div class="commit-actions flex-row"> <div class="commit-actions flex-row">
<div v-if="commit.signatureHtml" v-html="commit.signatureHtml"></div>
<gl-link <gl-link
v-if="commit.latestPipeline" v-if="commit.latestPipeline"
v-gl-tooltip v-gl-tooltip
......
...@@ -13,6 +13,7 @@ query pathLastCommit($projectPath: ID!, $path: String, $ref: String!) { ...@@ -13,6 +13,7 @@ query pathLastCommit($projectPath: ID!, $path: String, $ref: String!) {
avatarUrl avatarUrl
webUrl webUrl
} }
signatureHtml
latestPipeline { latestPipeline {
detailedStatus { detailedStatus {
detailsPath detailsPath
......
# frozen_string_literal: true
module Resolvers
class LastCommitResolver < BaseResolver
type Types::CommitType, null: true
alias_method :tree, :object
def resolve(**args)
# Ensure merge commits can be returned by sending nil to Gitaly instead of '/'
path = tree.path == '/' ? nil : tree.path
commit = Gitlab::Git::Commit.last_for_path(tree.repository, tree.sha, path)
::Commit.new(commit, tree.repository.project) if commit
end
end
end
...@@ -15,6 +15,8 @@ module Types ...@@ -15,6 +15,8 @@ module Types
field :message, type: GraphQL::STRING_TYPE, null: true # rubocop:disable Graphql/Descriptions field :message, type: GraphQL::STRING_TYPE, null: true # rubocop:disable Graphql/Descriptions
field :authored_date, type: Types::TimeType, null: true # rubocop:disable Graphql/Descriptions field :authored_date, type: Types::TimeType, null: true # rubocop:disable Graphql/Descriptions
field :web_url, type: GraphQL::STRING_TYPE, null: false # rubocop:disable Graphql/Descriptions field :web_url, type: GraphQL::STRING_TYPE, null: false # rubocop:disable Graphql/Descriptions
field :signature_html, type: GraphQL::STRING_TYPE,
null: true, calls_gitaly: true, description: 'Rendered html for the commit signature'
# models/commit lazy loads the author by email # models/commit lazy loads the author by email
field :author, type: Types::UserType, null: true # rubocop:disable Graphql/Descriptions field :author, type: Types::UserType, null: true # rubocop:disable Graphql/Descriptions
......
...@@ -7,9 +7,9 @@ module Types ...@@ -7,9 +7,9 @@ module Types
graphql_name 'Tree' graphql_name 'Tree'
# Complexity 10 as it triggers a Gitaly call on each render # Complexity 10 as it triggers a Gitaly call on each render
field :last_commit, Types::CommitType, null: true, complexity: 10, calls_gitaly: true, resolve: -> (tree, args, ctx) do # rubocop:disable Graphql/Descriptions field :last_commit, Types::CommitType,
tree.repository.last_commit_for_path(tree.sha, tree.path) null: true, complexity: 10, calls_gitaly: true, resolver: Resolvers::LastCommitResolver,
end description: 'Last commit for the tree'
field :trees, Types::Tree::TreeEntryType.connection_type, null: false, resolve: -> (obj, args, ctx) do # rubocop:disable Graphql/Descriptions field :trees, Types::Tree::TreeEntryType.connection_type, null: false, resolve: -> (obj, args, ctx) do # rubocop:disable Graphql/Descriptions
Gitlab::Graphql::Representation::TreeEntry.decorate(obj.trees, obj.repository) Gitlab::Graphql::Representation::TreeEntry.decorate(obj.trees, obj.repository)
......
...@@ -20,4 +20,15 @@ class CommitPresenter < Gitlab::View::Presenter::Delegated ...@@ -20,4 +20,15 @@ class CommitPresenter < Gitlab::View::Presenter::Delegated
def web_url def web_url
Gitlab::UrlBuilder.new(commit).url Gitlab::UrlBuilder.new(commit).url
end end
def signature_html
return unless commit.has_signature?
ApplicationController.renderer.render(
'projects/commit/_signature',
locals: { signature: commit.signature },
layout: false,
formats: [:html]
)
end
end end
...@@ -34,8 +34,9 @@ ...@@ -34,8 +34,9 @@
= _('Merge request') = _('Merge request')
- if branch.name != @repository.root_ref - if branch.name != @repository.root_ref
= link_to project_compare_path(@project, @repository.root_ref, branch.name), = link_to project_compare_index_path(@project, from: @repository.root_ref, to: branch.name),
class: "btn btn-default js-onboarding-compare-branches #{'prepend-left-10' unless merge_project}", class: "btn btn-default js-onboarding-compare-branches #{'prepend-left-10' unless merge_project}",
method: :post,
title: s_('Branches|Compare') do title: s_('Branches|Compare') do
= s_('Branches|Compare') = s_('Branches|Compare')
......
---
title: Fix inline rendering of videos for uploads with uppercase file extensions
merge_request: 17924
author:
type: fixed
---
title: Allow users to compare Git revisions on a read-only instance
merge_request: 18038
author:
type: fixed
...@@ -57,7 +57,7 @@ Once you've met the requirements, to enable Let's Encrypt integration: ...@@ -57,7 +57,7 @@ Once you've met the requirements, to enable Let's Encrypt integration:
1. Click **Save changes**. 1. Click **Save changes**.
Once enabled, GitLab will obtain a LE certificate and add it to the Once enabled, GitLab will obtain a LE certificate and add it to the
associated Pages domain. It will be also renewed automatically by GitLab. associated Pages domain. It also will be renewed automatically by GitLab.
> **Notes:** > **Notes:**
> >
......
...@@ -8,8 +8,8 @@ module Banzai ...@@ -8,8 +8,8 @@ module Banzai
# a "Download" link in the case the video cannot be played. # a "Download" link in the case the video cannot be played.
class VideoLinkFilter < HTML::Pipeline::Filter class VideoLinkFilter < HTML::Pipeline::Filter
def call def call
doc.xpath(query).each do |el| doc.xpath('descendant-or-self::img[not(ancestor::a)]').each do |el|
el.replace(video_node(doc, el)) el.replace(video_node(doc, el)) if has_video_extension?(el)
end end
doc doc
...@@ -17,22 +17,13 @@ module Banzai ...@@ -17,22 +17,13 @@ module Banzai
private private
def query def has_video_extension?(element)
@query ||= begin src = element.attr('data-canonical-src').presence || element.attr('src')
src_query = UploaderHelper::SAFE_VIDEO_EXT.map do |ext|
"'.#{ext}' = substring(@src, string-length(@src) - #{ext.size})"
end
if context[:asset_proxy_enabled].present? return unless src.present?
src_query.concat(
UploaderHelper::SAFE_VIDEO_EXT.map do |ext|
"'.#{ext}' = substring(@data-canonical-src, string-length(@data-canonical-src) - #{ext.size})"
end
)
end
"descendant-or-self::img[not(ancestor::a) and (#{src_query.join(' or ')})]" src_ext = File.extname(src).sub('.', '').downcase
end Gitlab::FileTypeDetection::SAFE_VIDEO_EXT.include?(src_ext)
end end
def video_node(doc, element) def video_node(doc, element)
......
...@@ -20,6 +20,10 @@ module Gitlab ...@@ -20,6 +20,10 @@ module Gitlab
'projects/lfs_locks_api' => %w{verify create unlock} 'projects/lfs_locks_api' => %w{verify create unlock}
}.freeze }.freeze
WHITELISTED_GIT_REVISION_ROUTES = {
'projects/compare' => %w{create}
}.freeze
GRAPHQL_URL = '/api/graphql' GRAPHQL_URL = '/api/graphql'
def initialize(app, env) def initialize(app, env)
...@@ -81,7 +85,7 @@ module Gitlab ...@@ -81,7 +85,7 @@ module Gitlab
# Overridden in EE module # Overridden in EE module
def whitelisted_routes def whitelisted_routes
grack_route? || internal_route? || lfs_route? || sidekiq_route? || graphql_query? grack_route? || internal_route? || lfs_route? || compare_git_revisions_route? || sidekiq_route? || graphql_query?
end end
def grack_route? def grack_route?
...@@ -96,6 +100,13 @@ module Gitlab ...@@ -96,6 +100,13 @@ module Gitlab
ReadOnly.internal_routes.any? { |path| request.path.include?(path) } ReadOnly.internal_routes.any? { |path| request.path.include?(path) }
end end
def compare_git_revisions_route?
# Calling route_hash may be expensive. Only do it if we think there's a possible match
return false unless request.post? && request.path.end_with?('compare')
WHITELISTED_GIT_REVISION_ROUTES[route_hash[:controller]]&.include?(route_hash[:action])
end
def lfs_route? def lfs_route?
# Calling route_hash may be expensive. Only do it if we think there's a possible match # Calling route_hash may be expensive. Only do it if we think there's a possible match
unless request.path.end_with?('/info/lfs/objects/batch', unless request.path.end_with?('/info/lfs/objects/batch',
......
...@@ -9224,6 +9224,9 @@ msgstr "" ...@@ -9224,6 +9224,9 @@ msgstr ""
msgid "Last used on:" msgid "Last used on:"
msgstr "" msgstr ""
msgid "LastCommit|authored"
msgstr ""
msgid "LastPushEvent|You pushed to" msgid "LastPushEvent|You pushed to"
msgstr "" msgstr ""
......
...@@ -117,7 +117,7 @@ module RuboCop ...@@ -117,7 +117,7 @@ module RuboCop
end end
def block_start?(line) def block_start?(line)
line.match(/ (do|{)( \|.*?\|)?\s?$/) line.match(/ (do|{)( \|.*?\|)?\s?(#.+)?\z/)
end end
def end_line?(line) def end_line?(line)
......
...@@ -246,7 +246,6 @@ describe 'Branches' do ...@@ -246,7 +246,6 @@ describe 'Branches' do
end end
expect(page).to have_content 'Commits' expect(page).to have_content 'Commits'
expect(page).to have_link 'Create merge request'
end end
end end
......
...@@ -12,6 +12,23 @@ describe "Compare", :js do ...@@ -12,6 +12,23 @@ describe "Compare", :js do
end end
describe "branches" do describe "branches" do
shared_examples 'compares branches' do
it 'compares branches' do
visit project_compare_index_path(project, from: 'master', to: 'master')
select_using_dropdown 'from', 'feature'
expect(find('.js-compare-from-dropdown .dropdown-toggle-text')).to have_content('feature')
select_using_dropdown 'to', 'binary-encoding'
expect(find('.js-compare-to-dropdown .dropdown-toggle-text')).to have_content('binary-encoding')
click_button 'Compare'
expect(page).to have_content 'Commits'
expect(page).to have_link 'Create merge request'
end
end
it "pre-populates fields" do it "pre-populates fields" do
visit project_compare_index_path(project, from: "master", to: "master") visit project_compare_index_path(project, from: "master", to: "master")
...@@ -19,19 +36,14 @@ describe "Compare", :js do ...@@ -19,19 +36,14 @@ describe "Compare", :js do
expect(find(".js-compare-to-dropdown .dropdown-toggle-text")).to have_content("master") expect(find(".js-compare-to-dropdown .dropdown-toggle-text")).to have_content("master")
end end
it "compares branches" do it_behaves_like 'compares branches'
visit project_compare_index_path(project, from: "master", to: "master")
select_using_dropdown "from", "feature"
expect(find(".js-compare-from-dropdown .dropdown-toggle-text")).to have_content("feature")
select_using_dropdown "to", "binary-encoding" context 'on a read-only instance' do
expect(find(".js-compare-to-dropdown .dropdown-toggle-text")).to have_content("binary-encoding") before do
allow(Gitlab::Database).to receive(:read_only?).and_return(true)
click_button "Compare" end
expect(page).to have_content "Commits" it_behaves_like 'compares branches'
expect(page).to have_link 'Create merge request'
end end
it 'renders additions info when click unfold diff' do it 'renders additions info when click unfold diff' do
......
...@@ -60,6 +60,111 @@ exports[`Repository last commit component renders commit widget 1`] = ` ...@@ -60,6 +60,111 @@ exports[`Repository last commit component renders commit widget 1`] = `
<div <div
class="commit-actions flex-row" class="commit-actions flex-row"
> >
<!---->
<gllink-stub
class="js-commit-pipeline"
data-original-title="Commit: failed"
href="https://test.com/pipeline"
title=""
>
<ciicon-stub
aria-label="Commit: failed"
cssclasses=""
size="24"
status="[object Object]"
/>
</gllink-stub>
<div
class="commit-sha-group d-flex"
>
<div
class="label label-monospace monospace"
>
12345678
</div>
<clipboardbutton-stub
cssclass="btn-default"
text="123456789"
title="Copy commit SHA to clipboard"
tooltipplacement="bottom"
/>
</div>
</div>
</div>
</div>
`;
exports[`Repository last commit component renders the signature HTML as returned by the backend 1`] = `
<div
class="info-well d-none d-sm-flex project-last-commit commit p-3"
>
<useravatarlink-stub
class="avatar-cell"
imgalt=""
imgcssclasses=""
imgsize="40"
imgsrc="https://test.com"
linkhref="https://test.com/test"
tooltipplacement="top"
tooltiptext=""
username=""
/>
<div
class="commit-detail flex-list"
>
<div
class="commit-content qa-commit-content"
>
<gllink-stub
class="commit-row-message item-title"
href="https://test.com/commit/123"
>
Commit title
</gllink-stub>
<!---->
<div
class="committer"
>
<gllink-stub
class="commit-author-link js-user-link"
href="https://test.com/test"
>
Test
</gllink-stub>
authored
<timeagotooltip-stub
cssclass=""
time="2019-01-01"
tooltipplacement="bottom"
/>
</div>
<!---->
</div>
<div
class="commit-actions flex-row"
>
<div>
<button>
Verified
</button>
</div>
<gllink-stub <gllink-stub
class="js-commit-pipeline" class="js-commit-pipeline"
data-original-title="Commit: failed" data-original-title="Commit: failed"
......
...@@ -107,4 +107,10 @@ describe('Repository last commit component', () => { ...@@ -107,4 +107,10 @@ describe('Repository last commit component', () => {
expect(vm.find('.commit-row-description').isVisible()).toBe(true); expect(vm.find('.commit-row-description').isVisible()).toBe(true);
expect(vm.find('.text-expander').classes('open')).toBe(true); expect(vm.find('.text-expander').classes('open')).toBe(true);
}); });
it('renders the signature HTML as returned by the backend', () => {
factory(createCommitData({ signatureHtml: '<button>Verified</button>' }));
expect(vm.element).toMatchSnapshot();
});
}); });
# frozen_string_literal: true
require 'spec_helper'
describe Resolvers::LastCommitResolver do
include GraphqlHelpers
let(:repository) { create(:project, :repository).repository }
let(:tree) { repository.tree(ref, path) }
let(:commit) { resolve(described_class, obj: tree) }
describe '#resolve' do
context 'last commit is a merge commit' do
let(:ref) { 'master' }
let(:path) { '/' }
it 'resolves to the merge commit' do
expect(commit).to eq(repository.commits(ref, limit: 1).last)
end
end
context 'last commit for a different branch and path' do
let(:ref) { 'fix' }
let(:path) { 'files' }
it 'resolves commit' do
expect(commit).to eq(repository.commits(ref, path: path, limit: 1).last)
end
end
context 'last commit does not exist' do
let(:ref) { 'master' }
let(:path) { 'does-not-exist' }
it 'returns nil' do
expect(commit).to be_nil
end
end
end
end
...@@ -7,5 +7,10 @@ describe GitlabSchema.types['Commit'] do ...@@ -7,5 +7,10 @@ describe GitlabSchema.types['Commit'] do
it { expect(described_class).to require_graphql_authorizations(:download_code) } it { expect(described_class).to require_graphql_authorizations(:download_code) }
it { expect(described_class).to have_graphql_fields(:id, :sha, :title, :description, :message, :authored_date, :author, :web_url, :latest_pipeline) } it 'contains attributes related to commit' do
expect(described_class).to have_graphql_fields(
:id, :sha, :title, :description, :message, :authored_date,
:author, :web_url, :latest_pipeline, :signature_html
)
end
end end
...@@ -12,15 +12,18 @@ describe Banzai::Filter::VideoLinkFilter do ...@@ -12,15 +12,18 @@ describe Banzai::Filter::VideoLinkFilter do
end end
def link_to_image(path) def link_to_image(path)
%(<img src="#{path}" />) return '<img/>' if path.nil?
%(<img src="#{path}"/>)
end end
let(:project) { create(:project, :repository) } let(:project) { create(:project, :repository) }
context 'when the element src has a video extension' do shared_examples 'a video element' do
UploaderHelper::SAFE_VIDEO_EXT.each do |ext| let(:image) { link_to_image(src) }
it "replaces the image tag 'path/video.#{ext}' with a video tag" do
container = filter(link_to_image("/path/video.#{ext}")).children.first it 'replaces the image tag with a video tag' do
container = filter(image).children.first
expect(container.name).to eq 'div' expect(container.name).to eq 'div'
expect(container['class']).to eq 'video-container' expect(container['class']).to eq 'video-container'
...@@ -28,36 +31,79 @@ describe Banzai::Filter::VideoLinkFilter do ...@@ -28,36 +31,79 @@ describe Banzai::Filter::VideoLinkFilter do
video, paragraph = container.children video, paragraph = container.children
expect(video.name).to eq 'video' expect(video.name).to eq 'video'
expect(video['src']).to eq "/path/video.#{ext}" expect(video['src']).to eq src
expect(paragraph.name).to eq 'p' expect(paragraph.name).to eq 'p'
link = paragraph.children.first link = paragraph.children.first
expect(link.name).to eq 'a' expect(link.name).to eq 'a'
expect(link['href']).to eq "/path/video.#{ext}" expect(link['href']).to eq src
expect(link['target']).to eq '_blank' expect(link['target']).to eq '_blank'
end end
end end
end
context 'when the element src is an image' do shared_examples 'an unchanged element' do |ext|
it 'leaves the document unchanged' do it 'leaves the document unchanged' do
element = filter(link_to_image('/path/my_image.jpg')).children.first element = filter(link_to_image(src)).children.first
expect(element.name).to eq 'img' expect(element.name).to eq 'img'
expect(element['src']).to eq '/path/my_image.jpg' expect(element['src']).to eq src
end end
end end
context 'when asset proxy is enabled' do context 'when the element src has a video extension' do
it 'uses the correct src' do Gitlab::FileTypeDetection::SAFE_VIDEO_EXT.each do |ext|
stub_asset_proxy_setting(enabled: true) it_behaves_like 'a video element' do
let(:src) { "/path/video.#{ext}" }
end
it_behaves_like 'a video element' do
let(:src) { "/path/video.#{ext.upcase}" }
end
end
end
context 'when the element has no src attribute' do
let(:src) { nil }
it_behaves_like 'an unchanged element'
end
context 'when the element src is an image' do
let(:src) { '/path/my_image.jpg' }
it_behaves_like 'an unchanged element'
end
context 'when the element src has an invalid file extension' do
let(:src) { '/path/my_video.somemp4' }
it_behaves_like 'an unchanged element'
end
context 'when data-canonical-src is empty' do
let(:image) { %(<img src="#{src}" data-canonical-src=""/>) }
context 'and src is a video' do
let(:src) { '/path/video.mp4' }
it_behaves_like 'a video element'
end
context 'and src is an image' do
let(:src) { '/path/my_image.jpg' }
it_behaves_like 'an unchanged element'
end
end
context 'when data-canonical-src is set' do
it 'uses the correct src' do
proxy_src = 'https://assets.example.com/6d8b63' proxy_src = 'https://assets.example.com/6d8b63'
canonical_src = 'http://example.com/test.mp4' canonical_src = 'http://example.com/test.mp4'
image = %(<img src="#{proxy_src}" data-canonical-src="#{canonical_src}" />) image = %(<img src="#{proxy_src}" data-canonical-src="#{canonical_src}"/>)
container = filter(image, asset_proxy_enabled: true).children.first container = filter(image).children.first
expect(container['class']).to eq 'video-container' expect(container['class']).to eq 'video-container'
......
...@@ -55,4 +55,17 @@ describe CommitPresenter do ...@@ -55,4 +55,17 @@ describe CommitPresenter do
end end
end end
end end
describe '#signature_html' do
let(:signature) { 'signature' }
before do
expect(commit).to receive(:has_signature?).and_return(true)
allow(ApplicationController.renderer).to receive(:render).and_return(signature)
end
it 'renders html for displaying signature' do
expect(presenter.signature_html).to eq(signature)
end
end
end end
...@@ -132,6 +132,19 @@ describe RuboCop::Cop::LineBreakAroundConditionalBlock do ...@@ -132,6 +132,19 @@ describe RuboCop::Cop::LineBreakAroundConditionalBlock do
expect(cop.offenses).to be_empty expect(cop.offenses).to be_empty
end end
it "doesn't flag violation for #{conditional} preceded by a block definition with a comment" do
source = <<~RUBY
on_block(param_a) do |item| # a short comment
#{conditional} condition
do_something
end
end
RUBY
inspect_source(source)
expect(cop.offenses).to be_empty
end
it "doesn't flag violation for #{conditional} preceded by a block definition using brackets" do it "doesn't flag violation for #{conditional} preceded by a block definition using brackets" do
source = <<~RUBY source = <<~RUBY
on_block(param_a) { |item| on_block(param_a) { |item|
......
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