Commit ee21564e authored by Rémy Coutable's avatar Rémy Coutable

Merge branch '5766-add-pod-dropdown-to-pod-logs-screen' into 'master'

Add `pod` dropdown to pod logs screen

Closes #5766

See merge request gitlab-org/gitlab-ee!6111
parents 6e8d50d9 4d9536a1
...@@ -232,3 +232,102 @@ ...@@ -232,3 +232,102 @@
word-break: break-word; word-break: break-word;
max-width: 100%; max-width: 100%;
} }
/*
* Mixin that handles the container for the job logs (CI/CD and kubernetes pod logs)
*/
@mixin build-trace {
background: $black;
color: $gray-darkest;
white-space: pre;
overflow-x: auto;
font-size: 12px;
border-radius: 0;
border: 0;
padding: $grid-size;
.bash {
display: block;
}
&.build-trace-rounded {
border-radius: $border-radius-base;
}
}
@mixin build-trace-top-bar($height, $top_position) {
height: $height;
min-height: $height;
background: $gray-light;
border: 1px solid $border-color;
color: $gl-text-color;
position: sticky;
position: -webkit-sticky;
top: $top_position;
padding: $grid-size;
}
/*
* Mixin that handles the position of the controls placed on the top bar
*/
@mixin build-controllers($control_font_size, $flex_direction, $with_grow, $flex_grow_size) {
display: flex;
font-size: $control_font_size;
justify-content: $flex_direction;
align-items: center;
align-self: baseline;
@if $with_grow {
flex-grow: $flex_grow_size;
}
svg {
width: 15px;
height: 15px;
display: block;
fill: $gl-text-color;
}
.controllers-buttons {
color: $gl-text-color;
margin: 0 $grid-size;
&:last-child {
margin-right: 0;
}
}
.btn-scroll.animate {
.first-triangle {
animation: blinking-scroll-button 1s ease infinite;
animation-delay: 0.3s;
}
.second-triangle {
animation: blinking-scroll-button 1s ease infinite;
animation-delay: 0.2s;
}
.third-triangle {
animation: blinking-scroll-button 1s ease infinite;
}
&:disabled {
opacity: 1;
}
}
.btn-scroll:disabled,
.btn-refresh:disabled {
opacity: 0.35;
cursor: not-allowed;
}
}
@mixin build-loader-animation() {
position: relative;
width: 6px;
height: 6px;
margin: auto auto 12px 2px;
border-radius: 50%;
animation: blinking-dots 1s linear infinite;
}
...@@ -264,6 +264,7 @@ $row-hover: $blue-50; ...@@ -264,6 +264,7 @@ $row-hover: $blue-50;
$row-hover-border: $blue-200; $row-hover-border: $blue-200;
$progress-color: #c0392b; $progress-color: #c0392b;
$header-height: 40px; $header-height: 40px;
$header-height-pod-logs: 75px;
$ide-statusbar-height: 25px; $ide-statusbar-height: 25px;
$fixed-layout-width: 1280px; $fixed-layout-width: 1280px;
$limited-layout-width: 990px; $limited-layout-width: 990px;
......
...@@ -55,34 +55,11 @@ ...@@ -55,34 +55,11 @@
} }
.build-trace { .build-trace {
background: $black; @include build-trace();
color: $gray-darkest;
white-space: pre;
overflow-x: auto;
font-size: 12px;
border-radius: 0;
border: 0;
padding: $grid-size;
.bash {
display: block;
}
&.build-trace-rounded {
border-radius: $border-radius-base;
}
} }
.top-bar { .top-bar {
height: 35px; @include build-trace-top-bar(35px, $header-height);
min-height: 35px;
background: $gray-light;
border: 1px solid $border-color;
color: $gl-text-color;
position: sticky;
position: -webkit-sticky;
top: $header-height;
padding: $grid-size;
&.affix { &.affix {
top: $header-height; top: $header-height;
...@@ -120,56 +97,7 @@ ...@@ -120,56 +97,7 @@
} }
.controllers { .controllers {
display: flex; @include build-controllers(15px, center, false, 0);
font-size: 15px;
justify-content: center;
align-items: center;
svg {
width: 15px;
height: 15px;
display: block;
fill: $gl-text-color;
}
.controllers-buttons {
color: $gl-text-color;
margin: 0 $grid-size;
&:last-child {
margin-right: 0;
}
}
.btn-scroll.animate {
.first-triangle {
animation: blinking-scroll-button 1s ease infinite;
animation-delay: 0.3s;
}
.second-triangle {
animation: blinking-scroll-button 1s ease infinite;
animation-delay: 0.2s;
}
.third-triangle {
animation: blinking-scroll-button 1s ease infinite;
}
&:disabled {
opacity: 1;
}
}
.btn-refresh {
border-radius: 4px;
}
.btn-scroll:disabled,
.btn-refresh:disabled {
opacity: 0.35;
cursor: not-allowed;
}
} }
} }
...@@ -188,12 +116,7 @@ ...@@ -188,12 +116,7 @@
} }
.build-loader-animation { .build-loader-animation {
position: relative; @include build-loader-animation();
width: 6px;
height: 6px;
margin: auto auto 12px 2px;
border-radius: 50%;
animation: blinking-dots 1s linear infinite;
} }
} }
...@@ -207,6 +130,66 @@ ...@@ -207,6 +130,66 @@
} }
} }
.build-page-pod-logs {
.build-trace-container {
position: relative;
}
.build-trace {
@include build-trace();
}
.top-bar {
@include build-trace-top-bar(48px, $header-height-pod-logs);
display: flex;
border-top-left-radius: $border-radius-default;
border-top-right-radius: $border-radius-default;
.truncated-info {
display: flex;
justify-content: center;
align-items: center;
}
.dropdown {
display: flex;
> .dropdown-menu-toggle {
display: flex;
align-content: center;
align-self: center;
width: 300px;
}
> .dropdown-menu {
width: 300px;
}
}
.controllers {
@include build-controllers(16px, flex-end, true, 2);
}
.refresh-control {
@include build-controllers(16px, flex-end, true, 0);
margin-left: 2px;
.controllers-buttons {
.btn-refresh {
border-radius: 4px;
width: 32px;
height: 32px;
vertical-align: middle;
}
}
}
}
.build-loader-animation {
@include build-loader-animation();
}
}
.build-header { .build-header {
.ci-header-container, .ci-header-container,
.header-action-buttons { .header-action-buttons {
......
class Environment < ActiveRecord::Base class Environment < ActiveRecord::Base
prepend EE::Environment
# Used to generate random suffixes for the slug # Used to generate random suffixes for the slug
LETTERS = 'a'..'z' LETTERS = 'a'..'z'
NUMBERS = '0'..'9' NUMBERS = '0'..'9'
......
...@@ -4,20 +4,21 @@ import { getParameterValues } from '~/lib/utils/url_utility'; ...@@ -4,20 +4,21 @@ import { getParameterValues } from '~/lib/utils/url_utility';
import { isScrolledToBottom, scrollDown, toggleDisableButton } from '~/lib/utils/scroll_utils'; import { isScrolledToBottom, scrollDown, toggleDisableButton } from '~/lib/utils/scroll_utils';
import LogOutputBehaviours from '~/lib/utils/logoutput_behaviours'; import LogOutputBehaviours from '~/lib/utils/logoutput_behaviours';
import createFlash from '~/flash'; import createFlash from '~/flash';
import { __, s__, sprintf } from '~/locale'; import { __, s__ } from '~/locale';
import _ from 'underscore'; import _ from 'underscore';
export default class KubernetesPodLogs extends LogOutputBehaviours { export default class KubernetesPodLogs extends LogOutputBehaviours {
constructor(container) { constructor(container) {
super(); super();
this.options = $(container).data(); this.options = $(container).data();
this.podNameContainer = $(container).find('.js-pod-name');
[this.podName] = getParameterValues('pod_name'); [this.podName] = getParameterValues('pod_name');
this.podName = _.escape(this.podName);
this.$buildOutputContainer = $(container).find('.js-build-output'); this.$buildOutputContainer = $(container).find('.js-build-output');
this.$window = $(window); this.$window = $(window);
this.$refreshLogBtn = $(container).find('.js-refresh-log'); this.$refreshLogBtn = $(container).find('.js-refresh-log');
this.$buildRefreshAnimation = $(container).find('.js-build-refresh'); this.$buildRefreshAnimation = $(container).find('.js-build-refresh');
this.isLogComplete = false; this.isLogComplete = false;
this.$podDropdown = $(container).find('.js-pod-dropdown');
this.scrollThrottled = _.throttle(this.toggleScroll.bind(this), 100); this.scrollThrottled = _.throttle(this.toggleScroll.bind(this), 100);
...@@ -26,16 +27,6 @@ export default class KubernetesPodLogs extends LogOutputBehaviours { ...@@ -26,16 +27,6 @@ export default class KubernetesPodLogs extends LogOutputBehaviours {
return; return;
} }
const podTitle = sprintf(
s__('Environments|Pod logs from %{podName}'),
{
podName: `<strong>${_.escape(this.podName)}</strong>`,
},
false,
);
this.podNameContainer.empty();
this.podNameContainer.append(podTitle);
this.$window.off('scroll').on('scroll', () => { this.$window.off('scroll').on('scroll', () => {
if (!isScrolledToBottom()) { if (!isScrolledToBottom()) {
this.toggleScrollAnimation(false); this.toggleScrollAnimation(false);
...@@ -70,6 +61,7 @@ export default class KubernetesPodLogs extends LogOutputBehaviours { ...@@ -70,6 +61,7 @@ export default class KubernetesPodLogs extends LogOutputBehaviours {
}) })
.then(res => { .then(res => {
const { logs } = res.data; const { logs } = res.data;
this.populateDropdown(res.data.pods);
const formattedLogs = logs.map(logEntry => `${_.escape(logEntry)} <br />`); const formattedLogs = logs.map(logEntry => `${_.escape(logEntry)} <br />`);
this.$buildOutputContainer.append(formattedLogs); this.$buildOutputContainer.append(formattedLogs);
scrollDown(); scrollDown();
...@@ -79,4 +71,34 @@ export default class KubernetesPodLogs extends LogOutputBehaviours { ...@@ -79,4 +71,34 @@ export default class KubernetesPodLogs extends LogOutputBehaviours {
}) })
.catch(() => createFlash(__('Something went wrong on our end'))); .catch(() => createFlash(__('Something went wrong on our end')));
} }
populateDropdown(pods) {
// set the selected element from the pod set on the url params
const $podDropdownMenu = this.$podDropdown.find('.dropdown-menu');
this.$podDropdown
.find('.dropdown-menu-toggle')
.html(`${this.podName}<i class="fa fa-chevron-down"></i>`);
$podDropdownMenu.off('click');
$podDropdownMenu.empty();
pods.forEach((pod) => {
$podDropdownMenu.append(`
<button class='dropdown-item'>
${_.escape(pod)}
</button>
`);
});
$podDropdownMenu.find('li').on('click', this.changePodLog.bind(this));
}
changePodLog(el) {
const selectedPodName = el.currentTarget.textContent.trim();
if (selectedPodName !== this.podName) {
this.podName = selectedPodName;
this.getPodLogs();
}
}
} }
...@@ -15,7 +15,8 @@ module EE ...@@ -15,7 +15,8 @@ module EE
::Gitlab::PollingInterval.set_header(response, interval: 3_000) ::Gitlab::PollingInterval.set_header(response, interval: 3_000)
render json: { render json: {
logs: pod_logs.strip.split("\n").as_json logs: pod_logs.strip.split("\n").as_json,
pods: environment.pod_names
} }
end end
end end
...@@ -28,7 +29,7 @@ module EE ...@@ -28,7 +29,7 @@ module EE
end end
def pod_logs def pod_logs
@pod_logs ||= environment.deployment_platform.read_pod_logs(params[:pod_name]) environment.deployment_platform.read_pod_logs(params[:pod_name])
end end
end end
end end
......
module EE
module Environment
def pod_names
return [] unless rollout_status
rollout_status.instances.map do |instance|
instance[:pod_name]
end
end
end
end
.js-kubernetes-logs{ data: { logs_path: logs_project_environment_path(@project, @environment, format: :json) } } .js-kubernetes-logs{ data: { logs_path: logs_project_environment_path(@project, @environment, format: :json) } }
.build-page .build-page-pod-logs
.build-trace-container.prepend-top-default .build-trace-container.prepend-top-default
.top-bar.js-top-bar .top-bar.js-top-bar
.truncated-info.hidden-xs.pull-left.js-pod-name .truncated-info.hidden-xs.pull-left
= s_('Environments|Pod logs from')
.dropdown.prepend-left-10.js-pod-dropdown
%button.dropdown-menu-toggle{ type: 'button', data: { toggle: 'dropdown' }, 'aria-expanded': false }
= icon('chevron-down')
.dropdown-menu.dropdown-menu-selectable.dropdown-menu-drop-up
.controllers.pull-right .controllers.pull-right
.has-tooltip.controllers-buttons{ title: _('Scroll to top'), data: { placement: 'top', container: 'body'} } .has-tooltip.controllers-buttons{ title: _('Scroll to top'), data: { placement: 'top', container: 'body'} }
%button.js-scroll-up.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true } %button.js-scroll-up.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true }
...@@ -11,6 +15,7 @@ ...@@ -11,6 +15,7 @@
.has-tooltip.controllers-buttons{ title: _('Scroll to bottom'), data: { placement: 'top', container: 'body'} } .has-tooltip.controllers-buttons{ title: _('Scroll to bottom'), data: { placement: 'top', container: 'body'} }
%button.js-scroll-down.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true } %button.js-scroll-down.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true }
= custom_icon('scroll_down') = custom_icon('scroll_down')
.refresh-control.pull-right
.has-tooltip.controllers-buttons{ title: _('Refresh'), data: { placement: 'top', container: 'body'} } .has-tooltip.controllers-buttons{ title: _('Refresh'), data: { placement: 'top', container: 'body'} }
%button.js-refresh-log.btn-default.btn-refresh{ type: 'button', disabled: true } %button.js-refresh-log.btn-default.btn-refresh{ type: 'button', disabled: true }
= sprite_icon('retry') = sprite_icon('retry')
......
---
title: Adds pod selection dropdown to pod logs screen
merge_request: 6111
author:
type: added
...@@ -75,16 +75,17 @@ describe Projects::EnvironmentsController do ...@@ -75,16 +75,17 @@ describe Projects::EnvironmentsController do
end end
describe 'GET logs' do describe 'GET logs' do
let(:logs) { "Log 1\nLog 2\nLog 3" } let(:pod_name) { "foo" }
let(:pod_name) { 'foo' }
before do before do
stub_licensed_features(pod_logs: true) stub_licensed_features(pod_logs: true)
create(:cluster, :provided_by_gcp, create(:cluster, :provided_by_gcp,
environment_scope: '*', projects: [project]) environment_scope: '*', projects: [project])
create(:deployment, environment: environment)
allow_any_instance_of(EE::KubernetesService).to receive(:read_pod_logs).with(pod_name).and_return(logs) allow_any_instance_of(EE::KubernetesService).to receive(:read_pod_logs).with(pod_name).and_return(kube_logs_body)
allow_any_instance_of(Gitlab::Kubernetes::RolloutStatus).to receive(:instances).and_return([{ pod_name: pod_name }])
end end
context 'when unlicensed' do context 'when unlicensed' do
...@@ -114,6 +115,7 @@ describe Projects::EnvironmentsController do ...@@ -114,6 +115,7 @@ describe Projects::EnvironmentsController do
expect(response).to be_ok expect(response).to be_ok
expect(json_response["logs"]).to match_array(["Log 1", "Log 2", "Log 3"]) expect(json_response["logs"]).to match_array(["Log 1", "Log 2", "Log 3"])
expect(json_response["pods"]).to match_array([pod_name])
end end
end end
end end
......
require 'spec_helper'
feature 'Environment > Pod Logs', :js do
include KubernetesHelpers
given(:pod_names) { %w(foo bar) }
given(:pod_name) { pod_names.first }
given(:project) { create(:project, :repository) }
given(:environment) { create(:environment, project: project) }
background do
stub_licensed_features(pod_logs: true)
create(:cluster, :provided_by_gcp, environment_scope: '*', projects: [project])
create(:deployment, environment: environment)
allow_any_instance_of(EE::KubernetesService).to receive(:read_pod_logs).with(pod_name).and_return(kube_logs_body)
allow_any_instance_of(EE::Environment).to receive(:pod_names).and_return(pod_names)
sign_in(project.owner)
end
context 'with logs' do
scenario "shows pod logs" do
visit logs_project_environment_path(environment.project, environment, pod_name: pod_name)
wait_for_requests
page.within('.js-pod-dropdown') do
find(".dropdown-menu-toggle").click
dropdown_items = find(".dropdown-menu").all(".dropdown-item")
expect(dropdown_items.size).to eq(2)
dropdown_items.each_with_index do |item, i|
expect(item.text).to eq(pod_names[i])
end
end
expect(page).to have_content("Log 1\nLog 2\nLog 3")
end
end
end
require 'spec_helper'
describe Environment do
let(:project) { create(:project, :stubbed_repository) }
let(:environment) { create(:environment, project: project) }
describe '#pod_names' do
context 'when environment does not have a rollout status' do
it 'returns an empty array' do
expect(environment.pod_names).to eq([])
end
end
context 'when environment has a rollout status' do
it 'returns the pod_names' do
pod_name = "pod_1"
create(:cluster, :provided_by_gcp, environment_scope: '*', projects: [project])
create(:deployment, environment: environment)
allow_any_instance_of(Gitlab::Kubernetes::RolloutStatus).to receive(:instances)
.and_return([{ pod_name: pod_name }])
expect(environment.pod_names).to eq([pod_name])
end
end
end
end
...@@ -104,4 +104,6 @@ export const logMockData = [ ...@@ -104,4 +104,6 @@ export const logMockData = [
'- -> /', '- -> /',
]; ];
export const podMockData = ['production-tanuki-1', 'production-tanuki-2'];
export default {}; export default {};
...@@ -2,15 +2,18 @@ ...@@ -2,15 +2,18 @@
.build-page .build-page
.build-trace-container.prepend-top-default .build-trace-container.prepend-top-default
.top-bar.js-top-bar .top-bar.js-top-bar
.truncated-info.hidden-xs.pull-left.js-pod-name .truncated-info.hidden-xs.pull-left
Pod logs from pod name .dropdown.prepend-left-10.js-pod-dropdown
%button.dropdown-menu-toggle{ type: 'button', data: { toggle: 'dropdown' }, 'aria-expanded': false }
%i.fa.fa-chevron-down
.dropdown-menu.dropdown-menu-selectable.dropdown-menu-drop-up
.controllers.pull-right .controllers.pull-right
.has-tooltip.controllers-buttons{ title: 'Scroll to top', data: { placement: 'top', container: 'body'} } .has-tooltip.controllers-buttons{ title: 'Scroll to top', data: { placement: 'top', container: 'body'} }
%button.js-scroll-up.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true } %button.js-scroll-up.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true }
.has-tooltip.controllers-buttons{ title: 'Scroll to bottom', data: { placement: 'top', container: 'body'} } .has-tooltip.controllers-buttons{ title: 'Scroll to bottom', data: { placement: 'top', container: 'body'} }
%button.js-scroll-down.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true } %button.js-scroll-down.btn-scroll.btn-transparent.btn-blank{ type: 'button', disabled: true }
.has-tooltip.controllers-buttons{ title: 'Refresh', data: { placement: 'top', container: 'body'} } .refresh-control.pull-right
.has-tooltip.controllers-buttons{ title: _('Refresh'), data: { placement: 'top', container: 'body'} }
%button.js-refresh-log.btn-default.btn-refresh{ type: 'button', disabled: true } %button.js-refresh-log.btn-default.btn-refresh{ type: 'button', disabled: true }
%pre.build-trace#build-trace %pre.build-trace#build-trace
......
import $ from 'jquery';
import KubernetesLogs from 'ee/kubernetes_logs'; import KubernetesLogs from 'ee/kubernetes_logs';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import { logMockData } from './ee/kubernetes_mock_data'; import { logMockData, podMockData } from './ee/kubernetes_mock_data';
describe('Kubernetes Logs', () => { describe('Kubernetes Logs', () => {
const fixtureTemplate = 'static/environments_logs.html.raw'; const fixtureTemplate = 'static/environments_logs.html.raw';
...@@ -20,7 +21,7 @@ describe('Kubernetes Logs', () => { ...@@ -20,7 +21,7 @@ describe('Kubernetes Logs', () => {
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
mock.onGet(logMockPath).reply(200, { logs: logMockData }); mock.onGet(logMockPath).reply(200, { logs: logMockData, pods: podMockData });
kubernetesLogContainer = document.querySelector('.js-kubernetes-logs'); kubernetesLogContainer = document.querySelector('.js-kubernetes-logs');
}); });
...@@ -29,11 +30,18 @@ describe('Kubernetes Logs', () => { ...@@ -29,11 +30,18 @@ describe('Kubernetes Logs', () => {
mock.restore(); mock.restore();
}); });
it('has the pod name placed on the top bar', () => { it('has the pod name placed on the dropdown', (done) => {
kubernetesLog = new KubernetesLogs(kubernetesLogContainer); kubernetesLog = new KubernetesLogs(kubernetesLogContainer);
const topBar = document.querySelector('.js-pod-name'); kubernetesLog.getPodLogs();
setTimeout(() => {
const podDropdown = document
.querySelector('.js-pod-dropdown')
.querySelector('.dropdown-menu-toggle');
expect(topBar.textContent).toContain(kubernetesLog.podName); expect(podDropdown.textContent).toContain(mockPodName);
done();
}, 0);
}); });
it('queries the pod log data and sets the dom elements', (done) => { it('queries the pod log data and sets the dom elements', (done) => {
...@@ -50,6 +58,58 @@ describe('Kubernetes Logs', () => { ...@@ -50,6 +58,58 @@ describe('Kubernetes Logs', () => {
done(); done();
}, 0); }, 0);
}); });
it('asks for the pod logs from another pod', (done) => {
const changePodLogSpy = spyOn(KubernetesLogs.prototype, 'getPodLogs').and.callThrough();
kubernetesLog = new KubernetesLogs(kubernetesLogContainer);
kubernetesLog.getPodLogs();
setTimeout(() => {
const podDropdown = document.querySelectorAll('.js-pod-dropdown .dropdown-menu button');
const anotherPod = podDropdown[podDropdown.length - 1];
anotherPod.click();
expect(changePodLogSpy).toHaveBeenCalled();
done();
}, 0);
});
it('clears the pod dropdown contents when pod logs are requested', (done) => {
const emptySpy = spyOn($.prototype, 'empty').and.callThrough();
kubernetesLog = new KubernetesLogs(kubernetesLogContainer);
kubernetesLog.getPodLogs();
setTimeout(() => {
// This is because it clears both the job log contents and the dropdown
expect(emptySpy.calls.count()).toEqual(2);
done();
});
});
});
describe('XSS Protection', () => {
const hackyPodName = '">&lt;img src=x onerror=alert(document.domain)&gt; production';
beforeEach(() => {
loadFixtures(fixtureTemplate);
spyOnDependency(KubernetesLogs, 'getParameterValues').and.callFake(() => [hackyPodName]);
mock = new MockAdapter(axios);
mock.onGet(logMockPath).reply(200, { logs: logMockData, pods: [hackyPodName] });
kubernetesLogContainer = document.querySelector('.js-kubernetes-logs');
});
afterEach(() => {
mock.restore();
});
it('escapes the pod name', () => {
kubernetesLog = new KubernetesLogs(kubernetesLogContainer);
expect(kubernetesLog.podName).toContain('&quot;&gt;&amp;lt;img src=x onerror=alert(document.domain)&amp;gt; production');
});
}); });
describe('When no pod name is available', () => { describe('When no pod name is available', () => {
......
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