Commit 0a0e82d1 authored by GitLab Bot's avatar GitLab Bot

Add latest changes from gitlab-org/gitlab@master

parent f82d5dca
...@@ -12,23 +12,24 @@ Set the title to: `Security Release: 12.2.X, 12.1.X, and 12.0.X` ...@@ -12,23 +12,24 @@ Set the title to: `Security Release: 12.2.X, 12.1.X, and 12.0.X`
## Version issues: ## Version issues:
* 12.2.X: {release task link} 12.2.X, 12.1.X, 12.0.X: {release task link}
* 12.1.X: {release task link}
* 12.0.X: {release task link}
## Issues in GitLab Security ## Issues in GitLab Security
* {https://gitlab.com/gitlab-org/security/gitlab/issues/ link} To include your issue and merge requests in this Security Release, please mark
your security issues as related to this release tracking issue. You can do this
in the "Linked issues" section below this issue description.
| Version | MR | :warning: If your security issues are not marked as related to this release
|---------|----| tracking issue, their merge requests may not be included in the security
| 12.2 | {https://gitlab.com/gitlab-org/security/gitlab/-/merge_requests link} | release.
| 12.1 | {https://gitlab.com/gitlab-org/security/gitlab/-/merge_requests link} |
| 12.0 | {https://gitlab.com/gitlab-org/security/gitlab/-/merge_requests link} |
| master | {https://gitlab.com/gitlab-org/security/gitlab/-/merge_requests link} |
## Issues in Omnibus-GitLab ## Issues in Omnibus-GitLab
Omnibus security fixes need to be added manually to this issue description
using and below the following template:
```markdown
* {https://gitlab.com/gitlab-org/security/gitlab/issues/ link} * {https://gitlab.com/gitlab-org/security/gitlab/issues/ link}
| Version | MR | | Version | MR |
...@@ -37,6 +38,7 @@ Set the title to: `Security Release: 12.2.X, 12.1.X, and 12.0.X` ...@@ -37,6 +38,7 @@ Set the title to: `Security Release: 12.2.X, 12.1.X, and 12.0.X`
| 12.1 | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} | | 12.1 | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} |
| 12.0 | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} | | 12.0 | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} |
| master | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} | | master | {https://dev.gitlab.org/gitlab/omnibus-gitlab/merge_requests/ link} |
```
## QA ## QA
{QA issue link} {QA issue link}
...@@ -49,5 +51,5 @@ GitLab.com: {https://gitlab.com/gitlab-com/www-gitlab-com/merge_requests/ link} ...@@ -49,5 +51,5 @@ GitLab.com: {https://gitlab.com/gitlab-com/www-gitlab-com/merge_requests/ link}
## Email notification ## Email notification
{https://gitlab.com/gitlab-com/marketing/general/issues/ link} {https://gitlab.com/gitlab-com/marketing/general/issues/ link}
/label ~security /label ~security ~"upcoming security release"
/confidential /confidential
...@@ -26,36 +26,7 @@ class ListIssue { ...@@ -26,36 +26,7 @@ class ListIssue {
} }
refreshData(obj, defaultAvatar) { refreshData(obj, defaultAvatar) {
this.id = obj.id; boardsStore.refreshIssueData(this, obj, defaultAvatar);
this.iid = obj.iid;
this.title = obj.title;
this.confidential = obj.confidential;
this.dueDate = obj.due_date;
this.sidebarInfoEndpoint = obj.issue_sidebar_endpoint;
this.referencePath = obj.reference_path;
this.path = obj.real_path;
this.toggleSubscriptionEndpoint = obj.toggle_subscription_endpoint;
this.project_id = obj.project_id;
this.timeEstimate = obj.time_estimate;
this.assignableLabelsEndpoint = obj.assignable_labels_endpoint;
this.blocked = obj.blocked;
if (obj.project) {
this.project = new IssueProject(obj.project);
}
if (obj.milestone) {
this.milestone = new ListMilestone(obj.milestone);
this.milestone_id = obj.milestone.id;
}
if (obj.labels) {
this.labels = obj.labels.map(label => new ListLabel(label));
}
if (obj.assignees) {
this.assignees = obj.assignees.map(a => new ListAssignee(a, defaultAvatar));
}
} }
addLabel(label) { addLabel(label) {
......
...@@ -12,6 +12,10 @@ import axios from '~/lib/utils/axios_utils'; ...@@ -12,6 +12,10 @@ import axios from '~/lib/utils/axios_utils';
import { mergeUrlParams } from '~/lib/utils/url_utility'; import { mergeUrlParams } from '~/lib/utils/url_utility';
import eventHub from '../eventhub'; import eventHub from '../eventhub';
import { ListType } from '../constants'; import { ListType } from '../constants';
import IssueProject from '../models/project';
import ListLabel from '../models/label';
import ListAssignee from '../models/assignee';
import ListMilestone from '../models/milestone';
const boardsStore = { const boardsStore = {
disabled: false, disabled: false,
...@@ -593,6 +597,38 @@ const boardsStore = { ...@@ -593,6 +597,38 @@ const boardsStore = {
clearMultiSelect() { clearMultiSelect() {
this.multiSelect.list = []; this.multiSelect.list = [];
}, },
refreshIssueData(issue, obj, defaultAvatar) {
issue.id = obj.id;
issue.iid = obj.iid;
issue.title = obj.title;
issue.confidential = obj.confidential;
issue.dueDate = obj.due_date;
issue.sidebarInfoEndpoint = obj.issue_sidebar_endpoint;
issue.referencePath = obj.reference_path;
issue.path = obj.real_path;
issue.toggleSubscriptionEndpoint = obj.toggle_subscription_endpoint;
issue.project_id = obj.project_id;
issue.timeEstimate = obj.time_estimate;
issue.assignableLabelsEndpoint = obj.assignable_labels_endpoint;
issue.blocked = obj.blocked;
if (obj.project) {
issue.project = new IssueProject(obj.project);
}
if (obj.milestone) {
issue.milestone = new ListMilestone(obj.milestone);
issue.milestone_id = obj.milestone.id;
}
if (obj.labels) {
issue.labels = obj.labels.map(label => new ListLabel(label));
}
if (obj.assignees) {
issue.assignees = obj.assignees.map(a => new ListAssignee(a, defaultAvatar));
}
},
}; };
BoardsStoreEE.initEESpecific(boardsStore); BoardsStoreEE.initEESpecific(boardsStore);
......
...@@ -522,7 +522,7 @@ export default { ...@@ -522,7 +522,7 @@ export default {
<div v-if="!showEmptyState"> <div v-if="!showEmptyState">
<graph-group <graph-group
v-for="(groupData, index) in dashboard.panel_groups" v-for="(groupData, index) in dashboard.panelGroups"
:key="`${groupData.group}.${groupData.priority}`" :key="`${groupData.group}.${groupData.priority}`"
:name="groupData.group" :name="groupData.group"
:show-panels="showPanels" :show-panels="showPanels"
......
...@@ -28,10 +28,10 @@ export default { ...@@ -28,10 +28,10 @@ export default {
...mapState('monitoringDashboard', ['dashboard']), ...mapState('monitoringDashboard', ['dashboard']),
...mapGetters('monitoringDashboard', ['metricsWithData']), ...mapGetters('monitoringDashboard', ['metricsWithData']),
charts() { charts() {
if (!this.dashboard || !this.dashboard.panel_groups) { if (!this.dashboard || !this.dashboard.panelGroups) {
return []; return [];
} }
const groupWithMetrics = this.dashboard.panel_groups.find(group => const groupWithMetrics = this.dashboard.panelGroups.find(group =>
group.panels.find(chart => this.chartHasData(chart)), group.panels.find(chart => this.chartHasData(chart)),
) || { panels: [] }; ) || { panels: [] };
......
...@@ -51,9 +51,11 @@ export const requestMetricsDashboard = ({ commit }) => { ...@@ -51,9 +51,11 @@ export const requestMetricsDashboard = ({ commit }) => {
commit(types.REQUEST_METRICS_DATA); commit(types.REQUEST_METRICS_DATA);
}; };
export const receiveMetricsDashboardSuccess = ({ commit, dispatch }, { response, params }) => { export const receiveMetricsDashboardSuccess = ({ commit, dispatch }, { response, params }) => {
commit(types.SET_ALL_DASHBOARDS, response.all_dashboards); const { all_dashboards, dashboard, metrics_data } = response;
commit(types.RECEIVE_METRICS_DATA_SUCCESS, response.dashboard);
commit(types.SET_ENDPOINTS, convertObjectPropsToCamelCase(response.metrics_data)); commit(types.SET_ALL_DASHBOARDS, all_dashboards);
commit(types.RECEIVE_METRICS_DATA_SUCCESS, dashboard);
commit(types.SET_ENDPOINTS, convertObjectPropsToCamelCase(metrics_data));
return dispatch('fetchPrometheusMetrics', params); return dispatch('fetchPrometheusMetrics', params);
}; };
...@@ -149,16 +151,16 @@ export const fetchPrometheusMetric = ({ commit }, { metric, params }) => { ...@@ -149,16 +151,16 @@ export const fetchPrometheusMetric = ({ commit }, { metric, params }) => {
step, step,
}; };
commit(types.REQUEST_METRIC_RESULT, { metricId: metric.metric_id }); commit(types.REQUEST_METRIC_RESULT, { metricId: metric.metricId });
return fetchPrometheusResult(metric.prometheus_endpoint_path, queryParams) return fetchPrometheusResult(metric.prometheusEndpointPath, queryParams)
.then(result => { .then(result => {
commit(types.RECEIVE_METRIC_RESULT_SUCCESS, { metricId: metric.metric_id, result }); commit(types.RECEIVE_METRIC_RESULT_SUCCESS, { metricId: metric.metricId, result });
}) })
.catch(error => { .catch(error => {
Sentry.captureException(error); Sentry.captureException(error);
commit(types.RECEIVE_METRIC_RESULT_FAILURE, { metricId: metric.metric_id, error }); commit(types.RECEIVE_METRIC_RESULT_FAILURE, { metricId: metric.metricId, error });
// Continue to throw error so the dashboard can notify using createFlash // Continue to throw error so the dashboard can notify using createFlash
throw error; throw error;
}); });
...@@ -168,7 +170,7 @@ export const fetchPrometheusMetrics = ({ state, commit, dispatch, getters }, par ...@@ -168,7 +170,7 @@ export const fetchPrometheusMetrics = ({ state, commit, dispatch, getters }, par
commit(types.REQUEST_METRICS_DATA); commit(types.REQUEST_METRICS_DATA);
const promises = []; const promises = [];
state.dashboard.panel_groups.forEach(group => { state.dashboard.panelGroups.forEach(group => {
group.panels.forEach(panel => { group.panels.forEach(panel => {
panel.metrics.forEach(metric => { panel.metrics.forEach(metric => {
promises.push(dispatch('fetchPrometheusMetric', { metric, params })); promises.push(dispatch('fetchPrometheusMetric', { metric, params }));
......
...@@ -11,7 +11,7 @@ const metricsIdsInPanel = panel => ...@@ -11,7 +11,7 @@ const metricsIdsInPanel = panel =>
* states in all the metric in the dashboard or group. * states in all the metric in the dashboard or group.
*/ */
export const getMetricStates = state => groupKey => { export const getMetricStates = state => groupKey => {
let groups = state.dashboard.panel_groups; let groups = state.dashboard.panelGroups;
if (groupKey) { if (groupKey) {
groups = groups.filter(group => group.key === groupKey); groups = groups.filter(group => group.key === groupKey);
} }
...@@ -43,7 +43,7 @@ export const getMetricStates = state => groupKey => { ...@@ -43,7 +43,7 @@ export const getMetricStates = state => groupKey => {
* filtered by group key. * filtered by group key.
*/ */
export const metricsWithData = state => groupKey => { export const metricsWithData = state => groupKey => {
let groups = state.dashboard.panel_groups; let groups = state.dashboard.panelGroups;
if (groupKey) { if (groupKey) {
groups = groups.filter(group => group.key === groupKey); groups = groups.filter(group => group.key === groupKey);
} }
......
import Vue from 'vue'; import Vue from 'vue';
import pick from 'lodash/pick'; import pick from 'lodash/pick';
import { slugify } from '~/lib/utils/text_utility';
import * as types from './mutation_types'; import * as types from './mutation_types';
import { normalizeMetric, normalizeQueryResult } from './utils'; import { mapToDashboardViewModel, normalizeQueryResult } from './utils';
import { BACKOFF_TIMEOUT } from '../../lib/utils/common_utils'; import { BACKOFF_TIMEOUT } from '../../lib/utils/common_utils';
import { metricStates } from '../constants'; import { metricStates } from '../constants';
import httpStatusCodes from '~/lib/utils/http_status'; import httpStatusCodes from '~/lib/utils/http_status';
const normalizePanelMetrics = (metrics, defaultLabel) =>
metrics.map(metric => ({
...normalizeMetric(metric),
label: metric.label || defaultLabel,
}));
/** /**
* Locate and return a metric in the dashboard by its id * Locate and return a metric in the dashboard by its id
* as generated by `uniqMetricsId()`. * as generated by `uniqMetricsId()`.
...@@ -21,10 +14,10 @@ const normalizePanelMetrics = (metrics, defaultLabel) => ...@@ -21,10 +14,10 @@ const normalizePanelMetrics = (metrics, defaultLabel) =>
*/ */
const findMetricInDashboard = (metricId, dashboard) => { const findMetricInDashboard = (metricId, dashboard) => {
let res = null; let res = null;
dashboard.panel_groups.forEach(group => { dashboard.panelGroups.forEach(group => {
group.panels.forEach(panel => { group.panels.forEach(panel => {
panel.metrics.forEach(metric => { panel.metrics.forEach(metric => {
if (metric.metric_id === metricId) { if (metric.metricId === metricId) {
res = metric; res = metric;
} }
}); });
...@@ -86,27 +79,9 @@ export default { ...@@ -86,27 +79,9 @@ export default {
state.showEmptyState = true; state.showEmptyState = true;
}, },
[types.RECEIVE_METRICS_DATA_SUCCESS](state, dashboard) { [types.RECEIVE_METRICS_DATA_SUCCESS](state, dashboard) {
state.dashboard = { state.dashboard = mapToDashboardViewModel(dashboard);
...dashboard,
panel_groups: dashboard.panel_groups.map((group, i) => {
const key = `${slugify(group.group || 'default')}-${i}`;
let { panels = [] } = group;
// each panel has metric information that needs to be normalized
panels = panels.map(panel => ({
...panel,
metrics: normalizePanelMetrics(panel.metrics, panel.y_label),
}));
return {
...group,
panels,
key,
};
}),
};
if (!state.dashboard.panel_groups.length) { if (!state.dashboard.panelGroups.length) {
state.emptyState = 'noData'; state.emptyState = 'noData';
} }
}, },
...@@ -206,7 +181,7 @@ export default { ...@@ -206,7 +181,7 @@ export default {
state.showErrorBanner = enabled; state.showErrorBanner = enabled;
}, },
[types.SET_PANEL_GROUP_METRICS](state, payload) { [types.SET_PANEL_GROUP_METRICS](state, payload) {
const panelGroup = state.dashboard.panel_groups.find(pg => payload.key === pg.key); const panelGroup = state.dashboard.panelGroups.find(pg => payload.key === pg.key);
panelGroup.panels = payload.panels; panelGroup.panels = payload.panels;
}, },
[types.SET_ENVIRONMENTS_FILTER](state, searchTerm) { [types.SET_ENVIRONMENTS_FILTER](state, searchTerm) {
......
...@@ -15,7 +15,7 @@ export default () => ({ ...@@ -15,7 +15,7 @@ export default () => ({
showEmptyState: true, showEmptyState: true,
showErrorBanner: true, showErrorBanner: true,
dashboard: { dashboard: {
panel_groups: [], panelGroups: [],
}, },
allDashboards: [], allDashboards: [],
......
import { omit } from 'lodash'; import { slugify } from '~/lib/utils/text_utility';
import createGqClient, { fetchPolicies } from '~/lib/graphql'; import createGqClient, { fetchPolicies } from '~/lib/graphql';
import { getIdFromGraphQLId } from '~/graphql_shared/utils'; import { getIdFromGraphQLId } from '~/graphql_shared/utils';
...@@ -9,6 +9,13 @@ export const gqClient = createGqClient( ...@@ -9,6 +9,13 @@ export const gqClient = createGqClient(
}, },
); );
/**
* Metrics loaded from project-defined dashboards do not have a metric_id.
* This method creates a unique ID combining metric_id and id, if either is present.
* This is hopefully a temporary solution until BE processes metrics before passing to fE
* @param {Object} metric - metric
* @returns {Object} - normalized metric with a uniqueID
*/
export const uniqMetricsId = metric => `${metric.metric_id}_${metric.id}`; export const uniqMetricsId = metric => `${metric.metric_id}_${metric.id}`;
/** /**
...@@ -41,22 +48,75 @@ export const parseEnvironmentsResponse = (response = [], projectPath) => ...@@ -41,22 +48,75 @@ export const parseEnvironmentsResponse = (response = [], projectPath) =>
}); });
/** /**
* Metrics loaded from project-defined dashboards do not have a metric_id. * Maps metrics to its view model
* This method creates a unique ID combining metric_id and id, if either is present. *
* This is hopefully a temporary solution until BE processes metrics before passing to fE * This function difers from other in that is maps all
* @param {Object} metric - metric * non-define properties as-is to the object. This is not
* @returns {Object} - normalized metric with a uniqueID * advisable as it could lead to unexpected side-effects.
*
* Related issue:
* https://gitlab.com/gitlab-org/gitlab/issues/207198
*
* @param {Array} metrics - Array of prometheus metrics
* @param {String} defaultLabel - Default label for metrics
* @returns {Object}
*/ */
const mapToMetricsViewModel = (metrics, defaultLabel) =>
metrics.map(({ label, id, metric_id, query_range, prometheus_endpoint_path, ...metric }) => ({
label: label || defaultLabel,
queryRange: query_range,
prometheusEndpointPath: prometheus_endpoint_path,
metricId: uniqMetricsId({ metric_id, id }),
export const normalizeMetric = (metric = {}) => // `metric_id` is used by embed.vue, keeping this duplicated.
omit( // https://gitlab.com/gitlab-org/gitlab/issues/37492
{ metric_id: uniqMetricsId({ metric_id, id }),
...metric, ...metric,
metric_id: uniqMetricsId(metric), }));
metricId: uniqMetricsId(metric),
}, /**
'id', * Maps a metrics panel to its view model
); *
* @param {Object} panel - Metrics panel
* @returns {Object}
*/
const mapToPanelViewModel = ({ title = '', type, y_label, metrics = [] }) => {
return {
title,
type,
y_label,
metrics: mapToMetricsViewModel(metrics, y_label),
};
};
/**
* Maps a metrics panel group to its view model
*
* @param {Object} panelGroup - Panel Group
* @returns {Object}
*/
const mapToPanelGroupViewModel = ({ group = '', panels = [] }, i) => {
return {
key: `${slugify(group || 'default')}-${i}`,
group,
panels: panels.map(mapToPanelViewModel),
};
};
/**
* Maps a dashboard json object to its view model
*
* @param {Object} dashboard - Dashboard object
* @param {String} dashboard.dashboard - Dashboard name object
* @param {Array} dashboard.panel_groups - Panel groups array
* @returns {Object}
*/
export const mapToDashboardViewModel = ({ dashboard = '', panel_groups = [] }) => {
return {
dashboard,
panelGroups: panel_groups.map(mapToPanelGroupViewModel),
};
};
export const normalizeQueryResult = timeSeries => { export const normalizeQueryResult = timeSeries => {
let normalizedResult = {}; let normalizedResult = {};
......
...@@ -7,7 +7,7 @@ import { ...@@ -7,7 +7,7 @@ import {
/** /**
* This method is used to validate if the graph data format for a chart component * This method is used to validate if the graph data format for a chart component
* that needs a time series as a response from a prometheus query (query_range) is * that needs a time series as a response from a prometheus query (queryRange) is
* of a valid format or not. * of a valid format or not.
* @param {Object} graphData the graph data response from a prometheus request * @param {Object} graphData the graph data response from a prometheus request
* @returns {boolean} whether the graphData format is correct * @returns {boolean} whether the graphData format is correct
......
...@@ -17,7 +17,7 @@ module Groups ...@@ -17,7 +17,7 @@ module Groups
serializer = ContainerRepositoriesSerializer serializer = ContainerRepositoriesSerializer
.new(current_user: current_user) .new(current_user: current_user)
if Feature.enabled?(:vue_container_registry_explorer) if Feature.enabled?(:vue_container_registry_explorer, group)
render json: serializer.with_pagination(request, response) render json: serializer.with_pagination(request, response)
.represent_read_only(@images) .represent_read_only(@images)
else else
......
...@@ -17,7 +17,7 @@ module Projects ...@@ -17,7 +17,7 @@ module Projects
serializer = ContainerRepositoriesSerializer serializer = ContainerRepositoriesSerializer
.new(project: project, current_user: current_user) .new(project: project, current_user: current_user)
if Feature.enabled?(:vue_container_registry_explorer) if Feature.enabled?(:vue_container_registry_explorer, project.group)
render json: serializer.with_pagination(request, response).represent(@images) render json: serializer.with_pagination(request, response).represent(@images)
else else
render json: serializer.represent(@images) render json: serializer.represent(@images)
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
%section %section
.row.registry-placeholder.prepend-bottom-10 .row.registry-placeholder.prepend-bottom-10
.col-12 .col-12
- if Feature.enabled?(:vue_container_registry_explorer, @project) - if Feature.enabled?(:vue_container_registry_explorer, @project.group)
#js-container-registry{ data: { endpoint: project_container_registry_index_path(@project), #js-container-registry{ data: { endpoint: project_container_registry_index_path(@project),
project_path: @project.full_path, project_path: @project.full_path,
"help_page_path" => help_page_path('user/packages/container_registry/index'), "help_page_path" => help_page_path('user/packages/container_registry/index'),
......
---
title: Add confidential attribute to notes table
merge_request:
author:
type: other
---
title: Moves refreshData from issue model to board store
merge_request: 21409
author: nuwe1
type: other
# frozen_string_literal: true
class AddConfidentialToNote < ActiveRecord::Migration[6.0]
include Gitlab::Database::MigrationHelpers
DOWNTIME = false
def change
with_lock_retries do
add_column :notes, :confidential, :boolean
end
end
end
...@@ -2806,6 +2806,7 @@ ActiveRecord::Schema.define(version: 2020_02_24_163804) do ...@@ -2806,6 +2806,7 @@ ActiveRecord::Schema.define(version: 2020_02_24_163804) do
t.text "change_position" t.text "change_position"
t.boolean "resolved_by_push" t.boolean "resolved_by_push"
t.bigint "review_id" t.bigint "review_id"
t.boolean "confidential"
t.index ["author_id"], name: "index_notes_on_author_id" t.index ["author_id"], name: "index_notes_on_author_id"
t.index ["commit_id"], name: "index_notes_on_commit_id" t.index ["commit_id"], name: "index_notes_on_commit_id"
t.index ["created_at"], name: "index_notes_on_created_at" t.index ["created_at"], name: "index_notes_on_created_at"
......
...@@ -20,7 +20,7 @@ alternative. ...@@ -20,7 +20,7 @@ alternative.
## Serialized Data Is Less Powerful ## Serialized Data Is Less Powerful
When using a relational database you have the ability to query individual When using a relational database you have the ability to query individual
fields, change the schema, index data and so forth. When you use serialized data fields, change the schema, index data, and so forth. When you use serialized data
all of that becomes either very difficult or downright impossible. While all of that becomes either very difficult or downright impossible. While
PostgreSQL does offer the ability to query JSON fields it is mostly meant for PostgreSQL does offer the ability to query JSON fields it is mostly meant for
very specialized use cases, and not for more general use. If you use YAML in very specialized use cases, and not for more general use. If you use YAML in
......
...@@ -215,4 +215,4 @@ When importing, GitLab would execute the following command, passing the `import_ ...@@ -215,4 +215,4 @@ When importing, GitLab would execute the following command, passing the `import_
git clone file://git:/tmp/lol git clone file://git:/tmp/lol
``` ```
Git would simply ignore the `git:` part, interpret the path as `file:///tmp/lol` and import the repository into the new project, in turn potentially giving the attacker access to any repository in the system, whether private or not. Git would simply ignore the `git:` part, interpret the path as `file:///tmp/lol`, and import the repository into the new project. This action could potentially give the attacker access to any repository in the system, whether private or not.
...@@ -671,7 +671,7 @@ The sharing dialog within Grafana provides the link, as highlighted below. ...@@ -671,7 +671,7 @@ The sharing dialog within Grafana provides the link, as highlighted below.
![Grafana Direct Linked Rendered Image](img/grafana_live_embed.png) ![Grafana Direct Linked Rendered Image](img/grafana_live_embed.png)
NOTE: **Note:** NOTE: **Note:**
For this embed to display correctly, the Grafana instance must be available to the target user, either as a public dashboard or on the same network. For this embed to display correctly, the Grafana instance must be available to the target user, either as a public dashboard, or on the same network.
Copy the link and add an image tag as [inline HTML](../../markdown.md#inline-html) in your Markdown. You may tweak the query parameters as required. For instance, removing the `&from=` and `&to=` parameters will give you a live chart. Here is example markup for a live chart from GitLab's public dashboard: Copy the link and add an image tag as [inline HTML](../../markdown.md#inline-html) in your Markdown. You may tweak the query parameters as required. For instance, removing the `&from=` and `&to=` parameters will give you a live chart. Here is example markup for a live chart from GitLab's public dashboard:
......
...@@ -147,7 +147,7 @@ Create lists for each of your team members and quickly drag-and-drop issues onto ...@@ -147,7 +147,7 @@ Create lists for each of your team members and quickly drag-and-drop issues onto
## Permissions ## Permissions
[Reporters and up](../permissions.md) can use all the functionality of the [Reporters and up](../permissions.md) can use all the functionality of the
Issue Board, that is, create or delete lists and drag issues from one list to another. Issue Board to create or delete lists, and drag issues from one list to another.
## GitLab Enterprise features for Issue Boards ## GitLab Enterprise features for Issue Boards
...@@ -373,7 +373,7 @@ window where you can see all the issues that do not belong to any list. ...@@ -373,7 +373,7 @@ window where you can see all the issues that do not belong to any list.
Select one or more issues by clicking on the cards and then click **Add issues** Select one or more issues by clicking on the cards and then click **Add issues**
to add them to the selected list. You can limit the issues you want to add to to add them to the selected list. You can limit the issues you want to add to
the list by filtering by author, assignee, milestone and label. the list by filtering by author, assignee, milestone, and label.
![Bulk adding issues to lists](img/issue_boards_add_issues_modal.png) ![Bulk adding issues to lists](img/issue_boards_add_issues_modal.png)
...@@ -419,7 +419,7 @@ You should be able to use the filters on top of your Issue Board to show only ...@@ -419,7 +419,7 @@ You should be able to use the filters on top of your Issue Board to show only
the results you want. This is similar to the filtering used in the issue tracker the results you want. This is similar to the filtering used in the issue tracker
since the metadata from the issues and labels are re-used in the Issue Board. since the metadata from the issues and labels are re-used in the Issue Board.
You can filter by author, assignee, milestone and label. You can filter by author, assignee, milestone, and label.
### Creating workflows ### Creating workflows
......
...@@ -6,7 +6,7 @@ Issues are the fundamental medium for collaborating on ideas and planning work i ...@@ -6,7 +6,7 @@ Issues are the fundamental medium for collaborating on ideas and planning work i
The GitLab issue tracker is an advanced tool for collaboratively developing ideas, solving problems, and planning work. The GitLab issue tracker is an advanced tool for collaboratively developing ideas, solving problems, and planning work.
Issues can allow you, your team, and your collaborators to share and discuss proposals Issues can allow you and your team to share and discuss proposals
before, and during, their implementation. However, they can be used for a variety of before, and during, their implementation. However, they can be used for a variety of
other purposes, customized to your needs and workflow. other purposes, customized to your needs and workflow.
......
...@@ -48,7 +48,7 @@ about CI/CD pipelines, when present), followed by the discussion threads of the ...@@ -48,7 +48,7 @@ about CI/CD pipelines, when present), followed by the discussion threads of the
collaborating with that MR. collaborating with that MR.
MRs also contain navigation tabs from which you can see the discussion happening on the thread, MRs also contain navigation tabs from which you can see the discussion happening on the thread,
the list of commits, the list of pipelines and jobs, the code changes and inline code reviews. the list of commits, the list of pipelines and jobs, the code changes, and inline code reviews.
To get started, read the [introduction to merge requests](getting_started.md). To get started, read the [introduction to merge requests](getting_started.md).
......
...@@ -28,7 +28,7 @@ For an overview, check the video demonstration on [Mapping work versus time with ...@@ -28,7 +28,7 @@ For an overview, check the video demonstration on [Mapping work versus time with
## Use cases ## Use cases
Burndown Charts, in general, are used for tracking and analyzing the completion of Burndown Charts are generally used for tracking and analyzing the completion of
a milestone. Therefore, their use cases are tied to the a milestone. Therefore, their use cases are tied to the
[use you are assigning your milestone to](index.md). [use you are assigning your milestone to](index.md).
......
...@@ -27,7 +27,7 @@ they are static, hence we are not dealing with server-side scripts ...@@ -27,7 +27,7 @@ they are static, hence we are not dealing with server-side scripts
nor credit card transactions, then why do we need secure connections? nor credit card transactions, then why do we need secure connections?
Back in the 1990s, where HTTPS came out, [SSL](https://en.wikipedia.org/wiki/Transport_Layer_Security#SSL_1.0.2C_2.0_and_3.0) was considered a "special" Back in the 1990s, where HTTPS came out, [SSL](https://en.wikipedia.org/wiki/Transport_Layer_Security#SSL_1.0.2C_2.0_and_3.0) was considered a "special"
security measure, necessary just for big companies, like banks and shoppings sites security measure, necessary just for big companies like banks and shopping sites
with financial transactions. with financial transactions.
Now we have a different picture. [According to Josh Aas](https://letsencrypt.org/2015/10/29/phishing-and-malware.html), Executive Director at [ISRG](https://en.wikipedia.org/wiki/Internet_Security_Research_Group): Now we have a different picture. [According to Josh Aas](https://letsencrypt.org/2015/10/29/phishing-and-malware.html), Executive Director at [ISRG](https://en.wikipedia.org/wiki/Internet_Security_Research_Group):
......
...@@ -16,7 +16,7 @@ To do so, follow the steps below. ...@@ -16,7 +16,7 @@ To do so, follow the steps below.
click **New project**, and name it according to the click **New project**, and name it according to the
[Pages domain names](../getting_started_part_one.md#gitlab-pages-default-domain-names). [Pages domain names](../getting_started_part_one.md#gitlab-pages-default-domain-names).
1. Clone it to your local computer, add your website 1. Clone it to your local computer, add your website
files to your project, add, commit and push to GitLab. files to your project, add, commit, and push to GitLab.
Alternatively, you can run `git init` in your local directory, Alternatively, you can run `git init` in your local directory,
add the remote URL: add the remote URL:
`git remote add origin git@gitlab.com:namespace/project-name.git`, `git remote add origin git@gitlab.com:namespace/project-name.git`,
......
...@@ -50,7 +50,7 @@ For more examples on artifacts, follow the [artifacts reference in ...@@ -50,7 +50,7 @@ For more examples on artifacts, follow the [artifacts reference in
## Browsing artifacts ## Browsing artifacts
> - From GitLab 9.2, PDFs, images, videos and other formats can be previewed directly in the job artifacts browser without the need to download them. > - From GitLab 9.2, PDFs, images, videos, and other formats can be previewed directly in the job artifacts browser without the need to download them.
> - Introduced in [GitLab 10.1][ce-14399], HTML files in a public project can be previewed directly in a new tab without the need to download them when [GitLab Pages](../../../administration/pages/index.md) is enabled. The same applies for textual formats (currently supported extensions: `.txt`, `.json`, and `.log`). > - Introduced in [GitLab 10.1][ce-14399], HTML files in a public project can be previewed directly in a new tab without the need to download them when [GitLab Pages](../../../administration/pages/index.md) is enabled. The same applies for textual formats (currently supported extensions: `.txt`, `.json`, and `.log`).
> - Introduced in [GitLab 12.4][gitlab-16675], artifacts in private projects can be previewed when [GitLab Pages access control](../../../administration/pages/index.md#access-control) is enabled. > - Introduced in [GitLab 12.4][gitlab-16675], artifacts in private projects can be previewed when [GitLab Pages access control](../../../administration/pages/index.md#access-control) is enabled.
......
...@@ -119,7 +119,7 @@ The next time a pipeline is scheduled, your credentials will be used. ...@@ -119,7 +119,7 @@ The next time a pipeline is scheduled, your credentials will be used.
NOTE: **Note:** NOTE: **Note:**
If the owner of a pipeline schedule doesn't have the ability to create pipelines If the owner of a pipeline schedule doesn't have the ability to create pipelines
on the target branch, the schedule will stop creating new pipelines. This can on the target branch, the schedule will stop creating new pipelines. This can
happen if, for example, the owner is blocked or removed from the project, or happen if the owner is blocked or removed from the project, or
the target branch or tag is protected. In this case, someone with sufficient the target branch or tag is protected. In this case, someone with sufficient
privileges must take ownership of the schedule. privileges must take ownership of the schedule.
......
...@@ -13,7 +13,7 @@ assets output by your CI system to use them, not just the raw source ...@@ -13,7 +13,7 @@ assets output by your CI system to use them, not just the raw source
code. code.
GitLab's **Releases** are a way to track deliverables in your project. Consider them GitLab's **Releases** are a way to track deliverables in your project. Consider them
a snapshot in time of the source, build output, and other metadata or artifacts a snapshot in time of the source, build output, artifacts, and other metadata
associated with a released version of your code. associated with a released version of your code.
There are several ways to create a Release: There are several ways to create a Release:
......
...@@ -104,7 +104,7 @@ Some things to note about precedence: ...@@ -104,7 +104,7 @@ Some things to note about precedence:
[Jupyter](https://jupyter.org/) Notebook (previously IPython Notebook) files are used for [Jupyter](https://jupyter.org/) Notebook (previously IPython Notebook) files are used for
interactive computing in many fields and contain a complete record of the interactive computing in many fields and contain a complete record of the
user's sessions and include code, narrative text, equations and rich output. user's sessions and include code, narrative text, equations, and rich output.
[Read how to use Jupyter notebooks with GitLab.](jupyter_notebooks/index.md) [Read how to use Jupyter notebooks with GitLab.](jupyter_notebooks/index.md)
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
[Jupyter](https://jupyter.org/) Notebook (previously IPython Notebook) files are used for [Jupyter](https://jupyter.org/) Notebook (previously IPython Notebook) files are used for
interactive computing in many fields and contain a complete record of the interactive computing in many fields and contain a complete record of the
user's sessions and include code, narrative text, equations and rich output. user's sessions and include code, narrative text, equations, and rich output.
When added to a repository, Jupyter Notebooks with a `.ipynb` extension will be When added to a repository, Jupyter Notebooks with a `.ipynb` extension will be
rendered to HTML when viewed. rendered to HTML when viewed.
......
...@@ -7,7 +7,7 @@ type: concepts, howto ...@@ -7,7 +7,7 @@ type: concepts, howto
[x509](https://en.wikipedia.org/wiki/X.509) is a standard format for public key [x509](https://en.wikipedia.org/wiki/X.509) is a standard format for public key
certificates issued by a public or private Public Key Infrastructure (PKI). certificates issued by a public or private Public Key Infrastructure (PKI).
Personal x509 certificates are used for authentication or signing purposes Personal x509 certificates are used for authentication or signing purposes
such as SMIME, but beside that, Git supports signing of commits and tags such as SMIME, but Git also supports signing of commits and tags
with x509 certificates in a similar way as with [GPG](../gpg_signed_commits/index.md). with x509 certificates in a similar way as with [GPG](../gpg_signed_commits/index.md).
The main difference is the trust anchor which is the PKI for x509 certificates The main difference is the trust anchor which is the PKI for x509 certificates
instead of a web of trust with GPG. instead of a web of trust with GPG.
......
...@@ -115,7 +115,7 @@ no longer actively maintained. Projects that have been archived can also be ...@@ -115,7 +115,7 @@ no longer actively maintained. Projects that have been archived can also be
unarchived. Only project Owners and Admin users have the unarchived. Only project Owners and Admin users have the
[permissions](../../permissions.md#project-members-permissions) to archive a project. [permissions](../../permissions.md#project-members-permissions) to archive a project.
When a project is archived, the repository, issues, merge requests and all When a project is archived, the repository, issues, merge requests, and all
other features are read-only. Archived projects are also hidden other features are read-only. Archived projects are also hidden
in project listings. in project listings.
......
...@@ -30,11 +30,11 @@ the Web IDE will make your direct editing even easier. ...@@ -30,11 +30,11 @@ the Web IDE will make your direct editing even easier.
The Web IDE currently provides: The Web IDE currently provides:
- Basic syntax colorization for a variety of programming, scripting and markup - Basic syntax colorization for a variety of programming, scripting and markup
languages such as XML, PHP, C#, C++, Markdown, Java, VB, Batch, Python, Ruby languages such as XML, PHP, C#, C++, Markdown, Java, VB, Batch, Python, Ruby,
and Objective-C. and Objective-C.
- IntelliSense and validation support (displaying errors and warnings, providing - IntelliSense and validation support (displaying errors and warnings, providing
smart completions, formatting, and outlining) for some languages. For example: smart completions, formatting, and outlining) for some languages. For example:
TypeScript, JavaScript, CSS, LESS, SCSS, JSON and HTML. TypeScript, JavaScript, CSS, LESS, SCSS, JSON, and HTML.
Because the Web IDE is based on the [Monaco Editor](https://microsoft.github.io/monaco-editor/), Because the Web IDE is based on the [Monaco Editor](https://microsoft.github.io/monaco-editor/),
you can find a more complete list of supported languages in the you can find a more complete list of supported languages in the
......
...@@ -38,7 +38,7 @@ automatically. For example, a title of `docs/my-page` will create a wiki ...@@ -38,7 +38,7 @@ automatically. For example, a title of `docs/my-page` will create a wiki
page with a path `/wikis/docs/my-page`. page with a path `/wikis/docs/my-page`.
Once you enter the page name, it's time to fill in its content. GitLab wikis Once you enter the page name, it's time to fill in its content. GitLab wikis
support Markdown, RDoc, AsciiDoc and Org. For Markdown based pages, all the support Markdown, RDoc, AsciiDoc, and Org. For Markdown based pages, all the
[Markdown features](../../markdown.md) are supported and for links there is [Markdown features](../../markdown.md) are supported and for links there is
some [wiki specific](../../markdown.md#wiki-specific-markdown) behavior. some [wiki specific](../../markdown.md#wiki-specific-markdown) behavior.
...@@ -121,7 +121,7 @@ The changes of a wiki page over time are recorded in the wiki's Git repository, ...@@ -121,7 +121,7 @@ The changes of a wiki page over time are recorded in the wiki's Git repository,
and you can view them by clicking the **Page history** button. and you can view them by clicking the **Page history** button.
From the history page you can see the revision of the page (Git commit SHA), its From the history page you can see the revision of the page (Git commit SHA), its
author, the commit message, when it was last updated and the page markup format. author, the commit message, when it was last updated, and the page markup format.
To see how a previous version of the page looked like, click on a revision To see how a previous version of the page looked like, click on a revision
number. number.
......
...@@ -119,7 +119,7 @@ Your [To-Do List](../todos.md#gitlab-to-do-list) can be searched by "to do" and ...@@ -119,7 +119,7 @@ Your [To-Do List](../todos.md#gitlab-to-do-list) can be searched by "to do" and
You can [filter](../todos.md#filtering-your-to-do-list) them per project, You can [filter](../todos.md#filtering-your-to-do-list) them per project,
author, type, and action. Also, you can sort them by author, type, and action. Also, you can sort them by
[**Label priority**](../../user/project/labels.md#label-priority), [**Label priority**](../../user/project/labels.md#label-priority),
**Last created** and **Oldest created**. **Last created**, and **Oldest created**.
## Projects ## Projects
......
...@@ -34,7 +34,7 @@ These shortcuts are available in most areas of GitLab ...@@ -34,7 +34,7 @@ These shortcuts are available in most areas of GitLab
| <kbd>p</kbd> + <kbd>b</kbd> | Show/hide the Performance Bar. | | <kbd>p</kbd> + <kbd>b</kbd> | Show/hide the Performance Bar. |
Additionally, the following shortcuts are available when editing text in text fields, Additionally, the following shortcuts are available when editing text in text fields,
for example comments, replies, or issue and merge request descriptions: for example comments, replies, issue descriptions, and merge request descriptions:
| Keyboard Shortcut | Description | | Keyboard Shortcut | Description |
| ---------------------------------------------------------------------- | ----------- | | ---------------------------------------------------------------------- | ----------- |
......
...@@ -93,7 +93,7 @@ describe Groups::Registry::RepositoriesController do ...@@ -93,7 +93,7 @@ describe Groups::Registry::RepositoriesController do
context 'with :vue_container_registry_explorer feature flag disabled' do context 'with :vue_container_registry_explorer feature flag disabled' do
before do before do
stub_feature_flags(vue_container_registry_explorer: false) stub_feature_flags(vue_container_registry_explorer: { enabled: false, thing: group })
end end
it 'has the correct response schema' do it 'has the correct response schema' do
......
...@@ -3,7 +3,7 @@ ...@@ -3,7 +3,7 @@
require 'spec_helper' require 'spec_helper'
describe Projects::Registry::RepositoriesController do describe Projects::Registry::RepositoriesController do
let(:user) { create(:user) } let(:user) { create(:user) }
let(:project) { create(:project, :private) } let(:project) { create(:project, :private) }
before do before do
...@@ -88,7 +88,7 @@ describe Projects::Registry::RepositoriesController do ...@@ -88,7 +88,7 @@ describe Projects::Registry::RepositoriesController do
context 'with :vue_container_registry_explorer feature flag disabled' do context 'with :vue_container_registry_explorer feature flag disabled' do
before do before do
stub_feature_flags(vue_container_registry_explorer: false) stub_feature_flags(vue_container_registry_explorer: { enabled: false, thing: project.group })
stub_container_registry_tags(repository: project.full_path, stub_container_registry_tags(repository: project.full_path,
tags: %w[rc1 latest]) tags: %w[rc1 latest])
end end
......
...@@ -19,7 +19,7 @@ describe 'Container Registry', :js do ...@@ -19,7 +19,7 @@ describe 'Container Registry', :js do
describe 'Registry explorer is off' do describe 'Registry explorer is off' do
before do before do
stub_feature_flags(vue_container_registry_explorer: false) stub_feature_flags(vue_container_registry_explorer: { enabled: false, thing: project.group })
end end
it 'has a page title set' do it 'has a page title set' do
......
...@@ -26,7 +26,6 @@ ...@@ -26,7 +26,6 @@
"stop_path": { "type": "string" }, "stop_path": { "type": "string" },
"cancel_auto_stop_path": { "type": "string" }, "cancel_auto_stop_path": { "type": "string" },
"folder_path": { "type": "string" }, "folder_path": { "type": "string" },
"project_path": { "type": "string" },
"created_at": { "type": "string", "format": "date-time" }, "created_at": { "type": "string", "format": "date-time" },
"updated_at": { "type": "string", "format": "date-time" }, "updated_at": { "type": "string", "format": "date-time" },
"auto_stop_at": { "type": "string", "format": "date-time" }, "auto_stop_at": { "type": "string", "format": "date-time" },
......
...@@ -8,14 +8,16 @@ function hasHiddenStyle(node) { ...@@ -8,14 +8,16 @@ function hasHiddenStyle(node) {
return false; return false;
} }
function createDefaultClientRect() { function createDefaultClientRect(node) {
const { outerWidth: width, outerHeight: height } = node;
return { return {
bottom: 0, bottom: height,
height: 0, height,
left: 0, left: 0,
right: 0, right: width,
top: 0, top: 0,
width: 0, width,
x: 0, x: 0,
y: 0, y: 0,
}; };
...@@ -46,5 +48,5 @@ window.Element.prototype.getClientRects = function getClientRects() { ...@@ -46,5 +48,5 @@ window.Element.prototype.getClientRects = function getClientRects() {
return []; return [];
} }
return [createDefaultClientRect()]; return [createDefaultClientRect(node)];
}; };
...@@ -2,3 +2,5 @@ import './element_scroll_into_view'; ...@@ -2,3 +2,5 @@ import './element_scroll_into_view';
import './get_client_rects'; import './get_client_rects';
import './inner_text'; import './inner_text';
import './window_scroll_to'; import './window_scroll_to';
import './scroll_by';
import './size_properties';
window.scrollX = 0;
window.scrollY = 0;
window.scrollBy = (x, y) => {
window.scrollX += x;
window.scrollY += y;
};
const convertFromStyle = style => {
if (style.match(/[0-9](px|rem)/g)) {
return Number(style.replace(/[^0-9]/g, ''));
}
return 0;
};
Object.defineProperty(global.HTMLElement.prototype, 'offsetWidth', {
get() {
return convertFromStyle(this.style.width || '0px');
},
});
Object.defineProperty(global.HTMLElement.prototype, 'offsetHeight', {
get() {
return convertFromStyle(this.style.height || '0px');
},
});
export const faviconDataUrl =
'';
export const overlayDataUrl =
'';
export const faviconWithOverlayDataUrl =
'';
...@@ -12,6 +12,7 @@ import { ...@@ -12,6 +12,7 @@ import {
deploymentData, deploymentData,
metricsDashboardPayload, metricsDashboardPayload,
mockedQueryResultPayload, mockedQueryResultPayload,
metricsDashboardViewModel,
mockProjectDir, mockProjectDir,
mockHost, mockHost,
} from '../../mock_data'; } from '../../mock_data';
...@@ -65,7 +66,7 @@ describe('Time series component', () => { ...@@ -65,7 +66,7 @@ describe('Time series component', () => {
); );
// Pick the second panel group and the first panel in it // Pick the second panel group and the first panel in it
[mockGraphData] = store.state.monitoringDashboard.dashboard.panel_groups[1].panels; [mockGraphData] = store.state.monitoringDashboard.dashboard.panelGroups[0].panels;
}); });
describe('general functions', () => { describe('general functions', () => {
...@@ -188,7 +189,7 @@ describe('Time series component', () => { ...@@ -188,7 +189,7 @@ describe('Time series component', () => {
}); });
it('formats tooltip content', () => { it('formats tooltip content', () => {
const name = 'Pod average'; const name = 'Total';
const value = '5.556'; const value = '5.556';
const dataIndex = 0; const dataIndex = 0;
const seriesLabel = timeSeriesChart.find(GlChartSeriesLabel); const seriesLabel = timeSeriesChart.find(GlChartSeriesLabel);
...@@ -439,7 +440,7 @@ describe('Time series component', () => { ...@@ -439,7 +440,7 @@ describe('Time series component', () => {
it('constructs a label for the chart y-axis', () => { it('constructs a label for the chart y-axis', () => {
const { yAxis } = getChartOptions(); const { yAxis } = getChartOptions();
expect(yAxis[0].name).toBe('Memory Used per Pod'); expect(yAxis[0].name).toBe('Total Memory Used');
}); });
}); });
}); });
...@@ -535,48 +536,24 @@ describe('Time series component', () => { ...@@ -535,48 +536,24 @@ describe('Time series component', () => {
}); });
describe('with multiple time series', () => { describe('with multiple time series', () => {
const mockedResultMultipleSeries = [];
const [, , panelData] = metricsDashboardPayload.panel_groups[1].panels;
for (let i = 0; i < panelData.metrics.length; i += 1) {
mockedResultMultipleSeries.push(cloneDeep(mockedQueryResultPayload));
mockedResultMultipleSeries[
i
].metricId = `${panelData.metrics[i].metric_id}_${panelData.metrics[i].id}`;
}
beforeEach(() => {
setTestTimeout(1000);
store = createStore();
store.commit(
`monitoringDashboard/${types.RECEIVE_METRICS_DATA_SUCCESS}`,
metricsDashboardPayload,
);
store.commit(`monitoringDashboard/${types.RECEIVE_DEPLOYMENTS_DATA_SUCCESS}`, deploymentData);
// Mock data contains the metric_id for a multiple time series panel
for (let i = 0; i < panelData.metrics.length; i += 1) {
store.commit(
`monitoringDashboard/${types.RECEIVE_METRIC_RESULT_SUCCESS}`,
mockedResultMultipleSeries[i],
);
}
// Pick the second panel group and the second panel in it
[, , mockGraphData] = store.state.monitoringDashboard.dashboard.panel_groups[1].panels;
});
describe('General functions', () => { describe('General functions', () => {
let timeSeriesChart; let timeSeriesChart;
beforeEach(done => { beforeEach(done => {
timeSeriesChart = makeTimeSeriesChart(mockGraphData, 'area-chart'); store = createStore();
const graphData = cloneDeep(metricsDashboardViewModel.panelGroups[0].panels[3]);
graphData.metrics.forEach(metric =>
Object.assign(metric, { result: mockedQueryResultPayload.result }),
);
timeSeriesChart = makeTimeSeriesChart(graphData, 'area-chart');
timeSeriesChart.vm.$nextTick(done); timeSeriesChart.vm.$nextTick(done);
}); });
afterEach(() => {
timeSeriesChart.destroy();
});
describe('computed', () => { describe('computed', () => {
let chartData; let chartData;
......
...@@ -17,12 +17,13 @@ import { setupComponentStore, propsData } from '../init_utils'; ...@@ -17,12 +17,13 @@ import { setupComponentStore, propsData } from '../init_utils';
import { import {
metricsDashboardPayload, metricsDashboardPayload,
mockedQueryResultPayload, mockedQueryResultPayload,
metricsDashboardViewModel,
environmentData, environmentData,
dashboardGitResponse, dashboardGitResponse,
} from '../mock_data'; } from '../mock_data';
const localVue = createLocalVue(); const localVue = createLocalVue();
const expectedPanelCount = 3; const expectedPanelCount = 4;
describe('Dashboard', () => { describe('Dashboard', () => {
let store; let store;
...@@ -366,7 +367,7 @@ describe('Dashboard', () => { ...@@ -366,7 +367,7 @@ describe('Dashboard', () => {
it('metrics can be swapped', () => { it('metrics can be swapped', () => {
const firstDraggable = findDraggables().at(0); const firstDraggable = findDraggables().at(0);
const mockMetrics = [...metricsDashboardPayload.panel_groups[1].panels]; const mockMetrics = [...metricsDashboardViewModel.panelGroups[0].panels];
const firstTitle = mockMetrics[0].title; const firstTitle = mockMetrics[0].title;
const secondTitle = mockMetrics[1].title; const secondTitle = mockMetrics[1].title;
...@@ -376,7 +377,7 @@ describe('Dashboard', () => { ...@@ -376,7 +377,7 @@ describe('Dashboard', () => {
firstDraggable.vm.$emit('input', mockMetrics); firstDraggable.vm.$emit('input', mockMetrics);
return wrapper.vm.$nextTick(() => { return wrapper.vm.$nextTick(() => {
const { panels } = wrapper.vm.dashboard.panel_groups[1]; const { panels } = wrapper.vm.dashboard.panelGroups[0];
expect(panels[1].title).toEqual(firstTitle); expect(panels[1].title).toEqual(firstTitle);
expect(panels[0].title).toEqual(secondTitle); expect(panels[0].title).toEqual(secondTitle);
......
...@@ -69,8 +69,8 @@ describe('Embed', () => { ...@@ -69,8 +69,8 @@ describe('Embed', () => {
describe('metrics are available', () => { describe('metrics are available', () => {
beforeEach(() => { beforeEach(() => {
store.state.monitoringDashboard.dashboard.panel_groups = groups; store.state.monitoringDashboard.dashboard.panelGroups = groups;
store.state.monitoringDashboard.dashboard.panel_groups[0].panels = metricsData; store.state.monitoringDashboard.dashboard.panelGroups[0].panels = metricsData;
metricsWithDataGetter.mockReturnValue(metricsWithData); metricsWithDataGetter.mockReturnValue(metricsWithData);
......
import { mapToDashboardViewModel } from '~/monitoring/stores/utils';
// This import path needs to be relative for now because this mock data is used in // This import path needs to be relative for now because this mock data is used in
// Karma specs too, where the helpers/test_constants alias can not be resolved // Karma specs too, where the helpers/test_constants alias can not be resolved
import { TEST_HOST } from '../helpers/test_constants'; import { TEST_HOST } from '../helpers/test_constants';
...@@ -246,7 +248,7 @@ export const mockedEmptyResult = { ...@@ -246,7 +248,7 @@ export const mockedEmptyResult = {
}; };
export const mockedQueryResultPayload = { export const mockedQueryResultPayload = {
metricId: '17_system_metrics_kubernetes_container_memory_average', metricId: '12_system_metrics_kubernetes_container_memory_total',
result: [ result: [
{ {
metric: {}, metric: {},
...@@ -378,122 +380,28 @@ export const environmentData = [ ...@@ -378,122 +380,28 @@ export const environmentData = [
}, },
].concat(extraEnvironmentData); ].concat(extraEnvironmentData);
export const metricsDashboardResponse = {
dashboard: {
dashboard: 'Environment metrics',
priority: 1,
panel_groups: [
{
group: 'System metrics (Kubernetes)',
priority: 5,
panels: [
{
title: 'Memory Usage (Total)',
type: 'area-chart',
y_label: 'Total Memory Used',
weight: 4,
metrics: [
{
id: 'system_metrics_kubernetes_container_memory_total',
query_range:
'avg(sum(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) by (job)) without (job) /1024/1024/1024',
label: 'Total',
unit: 'GB',
metric_id: 12,
prometheus_endpoint_path: 'http://test',
},
],
},
{
title: 'Core Usage (Total)',
type: 'area-chart',
y_label: 'Total Cores',
weight: 3,
metrics: [
{
id: 'system_metrics_kubernetes_container_cores_total',
query_range:
'avg(sum(rate(container_cpu_usage_seconds_total{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}[15m])) by (job)) without (job)',
label: 'Total',
unit: 'cores',
metric_id: 13,
},
],
},
{
title: 'Memory Usage (Pod average)',
type: 'line-chart',
y_label: 'Memory Used per Pod',
weight: 2,
metrics: [
{
id: 'system_metrics_kubernetes_container_memory_average',
query_range:
'avg(sum(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) by (job)) without (job) / count(avg(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) without (job)) /1024/1024',
label: 'Pod average',
unit: 'MB',
metric_id: 14,
},
],
},
],
},
],
},
status: 'success',
};
export const metricsDashboardPayload = { export const metricsDashboardPayload = {
dashboard: 'Environment metrics', dashboard: 'Environment metrics',
priority: 1,
panel_groups: [ panel_groups: [
{
group: 'Response metrics (NGINX Ingress VTS)',
priority: 10,
panels: [
{
metrics: [
{
id: 'response_metrics_nginx_ingress_throughput_status_code',
label: 'Status Code',
metric_id: 1,
prometheus_endpoint_path:
'/root/autodevops-deploy/environments/32/prometheus/api/v1/query_range?query=sum%28rate%28nginx_upstream_responses_total%7Bupstream%3D~%22%25%7Bkube_namespace%7D-%25%7Bci_environment_slug%7D-.%2A%22%7D%5B2m%5D%29%29+by+%28status_code%29',
query_range:
'sum(rate(nginx_upstream_responses_total{upstream=~"%{kube_namespace}-%{ci_environment_slug}-.*"}[2m])) by (status_code)',
unit: 'req / sec',
},
],
title: 'Throughput',
type: 'area-chart',
weight: 1,
y_label: 'Requests / Sec',
},
],
},
{ {
group: 'System metrics (Kubernetes)', group: 'System metrics (Kubernetes)',
priority: 5, priority: 5,
panels: [ panels: [
{ {
title: 'Memory Usage (Pod average)', title: 'Memory Usage (Total)',
type: 'area-chart', type: 'area-chart',
y_label: 'Memory Used per Pod', y_label: 'Total Memory Used',
weight: 2, weight: 4,
metrics: [ metrics: [
{ {
id: 'system_metrics_kubernetes_container_memory_average', id: 'system_metrics_kubernetes_container_memory_total',
query_range: query_range:
'avg(sum(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-([^c].*|c([^a]|a([^n]|n([^a]|a([^r]|r[^y])))).*|)-(.*)",namespace="%{kube_namespace}"}) by (job)) without (job) / count(avg(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-([^c].*|c([^a]|a([^n]|n([^a]|a([^r]|r[^y])))).*|)-(.*)",namespace="%{kube_namespace}"}) without (job)) /1024/1024', 'avg(sum(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) by (job)) without (job) /1024/1024/1024',
label: 'Pod average', label: 'Total',
unit: 'MB', unit: 'GB',
metric_id: 17, metric_id: 12,
prometheus_endpoint_path: prometheus_endpoint_path: 'http://test',
'/root/autodevops-deploy/environments/32/prometheus/api/v1/query_range?query=avg%28sum%28container_memory_usage_bytes%7Bcontainer_name%21%3D%22POD%22%2Cpod_name%3D~%22%5E%25%7Bci_environment_slug%7D-%28%5B%5Ec%5D.%2A%7Cc%28%5B%5Ea%5D%7Ca%28%5B%5En%5D%7Cn%28%5B%5Ea%5D%7Ca%28%5B%5Er%5D%7Cr%5B%5Ey%5D%29%29%29%29.%2A%7C%29-%28.%2A%29%22%2Cnamespace%3D%22%25%7Bkube_namespace%7D%22%7D%29+by+%28job%29%29+without+%28job%29+%2F+count%28avg%28container_memory_usage_bytes%7Bcontainer_name%21%3D%22POD%22%2Cpod_name%3D~%22%5E%25%7Bci_environment_slug%7D-%28%5B%5Ec%5D.%2A%7Cc%28%5B%5Ea%5D%7Ca%28%5B%5En%5D%7Cn%28%5B%5Ea%5D%7Ca%28%5B%5Er%5D%7Cr%5B%5Ey%5D%29%29%29%29.%2A%7C%29-%28.%2A%29%22%2Cnamespace%3D%22%25%7Bkube_namespace%7D%22%7D%29+without+%28job%29%29+%2F1024%2F1024',
appearance: {
line: {
width: 2,
},
},
}, },
], ],
}, },
...@@ -513,6 +421,22 @@ export const metricsDashboardPayload = { ...@@ -513,6 +421,22 @@ export const metricsDashboardPayload = {
}, },
], ],
}, },
{
title: 'Memory Usage (Pod average)',
type: 'line-chart',
y_label: 'Memory Used per Pod',
weight: 2,
metrics: [
{
id: 'system_metrics_kubernetes_container_memory_average',
query_range:
'avg(sum(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) by (job)) without (job) / count(avg(container_memory_usage_bytes{container_name!="POD",pod_name=~"^%{ci_environment_slug}-(.*)",namespace="%{kube_namespace}"}) without (job)) /1024/1024',
label: 'Pod average',
unit: 'MB',
metric_id: 14,
},
],
},
{ {
title: 'memories', title: 'memories',
type: 'area-chart', type: 'area-chart',
...@@ -557,9 +481,45 @@ export const metricsDashboardPayload = { ...@@ -557,9 +481,45 @@ export const metricsDashboardPayload = {
}, },
], ],
}, },
{
group: 'Response metrics (NGINX Ingress VTS)',
priority: 10,
panels: [
{
metrics: [
{
id: 'response_metrics_nginx_ingress_throughput_status_code',
label: 'Status Code',
metric_id: 1,
prometheus_endpoint_path:
'/root/autodevops-deploy/environments/32/prometheus/api/v1/query_range?query=sum%28rate%28nginx_upstream_responses_total%7Bupstream%3D~%22%25%7Bkube_namespace%7D-%25%7Bci_environment_slug%7D-.%2A%22%7D%5B2m%5D%29%29+by+%28status_code%29',
query_range:
'sum(rate(nginx_upstream_responses_total{upstream=~"%{kube_namespace}-%{ci_environment_slug}-.*"}[2m])) by (status_code)',
unit: 'req / sec',
},
],
title: 'Throughput',
type: 'area-chart',
weight: 1,
y_label: 'Requests / Sec',
},
],
},
], ],
}; };
/**
* Mock of response of metrics_dashboard.json
*/
export const metricsDashboardResponse = {
all_dashboards: [],
dashboard: metricsDashboardPayload,
metrics_data: {},
status: 'success',
};
export const metricsDashboardViewModel = mapToDashboardViewModel(metricsDashboardPayload);
const customDashboardsData = new Array(30).fill(null).map((_, idx) => ({ const customDashboardsData = new Array(30).fill(null).map((_, idx) => ({
default: false, default: false,
display_name: `Custom Dashboard ${idx}`, display_name: `Custom Dashboard ${idx}`,
......
...@@ -3,7 +3,7 @@ import testAction from 'helpers/vuex_action_helper'; ...@@ -3,7 +3,7 @@ import testAction from 'helpers/vuex_action_helper';
import Tracking from '~/tracking'; import Tracking from '~/tracking';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import statusCodes from '~/lib/utils/http_status'; import statusCodes from '~/lib/utils/http_status';
import { backOff } from '~/lib/utils/common_utils'; import * as commonUtils from '~/lib/utils/common_utils';
import createFlash from '~/flash'; import createFlash from '~/flash';
import store from '~/monitoring/stores'; import store from '~/monitoring/stores';
...@@ -28,11 +28,10 @@ import { ...@@ -28,11 +28,10 @@ import {
deploymentData, deploymentData,
environmentData, environmentData,
metricsDashboardResponse, metricsDashboardResponse,
metricsDashboardPayload, metricsDashboardViewModel,
dashboardGitResponse, dashboardGitResponse,
} from '../mock_data'; } from '../mock_data';
jest.mock('~/lib/utils/common_utils');
jest.mock('~/flash'); jest.mock('~/flash');
const resetStore = str => { const resetStore = str => {
...@@ -44,14 +43,17 @@ const resetStore = str => { ...@@ -44,14 +43,17 @@ const resetStore = str => {
}; };
describe('Monitoring store actions', () => { describe('Monitoring store actions', () => {
const { convertObjectPropsToCamelCase } = commonUtils;
let mock; let mock;
beforeEach(() => { beforeEach(() => {
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
// Mock `backOff` function to remove exponential algorithm delay. // Mock `backOff` function to remove exponential algorithm delay.
jest.useFakeTimers(); jest.useFakeTimers();
backOff.mockImplementation(callback => { jest.spyOn(commonUtils, 'backOff').mockImplementation(callback => {
const q = new Promise((resolve, reject) => { const q = new Promise((resolve, reject) => {
const stop = arg => (arg instanceof Error ? reject(arg) : resolve(arg)); const stop = arg => (arg instanceof Error ? reject(arg) : resolve(arg));
const next = () => callback(next, stop); const next = () => callback(next, stop);
...@@ -69,7 +71,7 @@ describe('Monitoring store actions', () => { ...@@ -69,7 +71,7 @@ describe('Monitoring store actions', () => {
resetStore(store); resetStore(store);
mock.reset(); mock.reset();
backOff.mockReset(); commonUtils.backOff.mockReset();
createFlash.mockReset(); createFlash.mockReset();
}); });
...@@ -115,7 +117,6 @@ describe('Monitoring store actions', () => { ...@@ -115,7 +117,6 @@ describe('Monitoring store actions', () => {
afterEach(() => { afterEach(() => {
resetStore(store); resetStore(store);
jest.restoreAllMocks();
}); });
it('setting SET_ENVIRONMENTS_FILTER should dispatch fetchEnvironmentsData', () => { it('setting SET_ENVIRONMENTS_FILTER should dispatch fetchEnvironmentsData', () => {
...@@ -365,6 +366,7 @@ describe('Monitoring store actions', () => { ...@@ -365,6 +366,7 @@ describe('Monitoring store actions', () => {
); );
expect(commit).toHaveBeenCalledWith( expect(commit).toHaveBeenCalledWith(
types.RECEIVE_METRICS_DATA_SUCCESS, types.RECEIVE_METRICS_DATA_SUCCESS,
metricsDashboardResponse.dashboard, metricsDashboardResponse.dashboard,
); );
expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetrics', params); expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetrics', params);
...@@ -443,8 +445,11 @@ describe('Monitoring store actions', () => { ...@@ -443,8 +445,11 @@ describe('Monitoring store actions', () => {
.catch(done.fail); .catch(done.fail);
}); });
it('dispatches fetchPrometheusMetric for each panel query', done => { it('dispatches fetchPrometheusMetric for each panel query', done => {
state.dashboard.panel_groups = metricsDashboardResponse.dashboard.panel_groups; state.dashboard.panelGroups = convertObjectPropsToCamelCase(
const [metric] = state.dashboard.panel_groups[0].panels[0].metrics; metricsDashboardResponse.dashboard.panel_groups,
);
const [metric] = state.dashboard.panelGroups[0].panels[0].metrics;
const getters = { const getters = {
metricsWithData: () => [metric.id], metricsWithData: () => [metric.id],
}; };
...@@ -473,16 +478,16 @@ describe('Monitoring store actions', () => { ...@@ -473,16 +478,16 @@ describe('Monitoring store actions', () => {
}); });
it('dispatches fetchPrometheusMetric for each panel query, handles an error', done => { it('dispatches fetchPrometheusMetric for each panel query, handles an error', done => {
state.dashboard.panel_groups = metricsDashboardResponse.dashboard.panel_groups; state.dashboard.panelGroups = metricsDashboardViewModel.panelGroups;
const metric = state.dashboard.panel_groups[0].panels[0].metrics[0]; const metric = state.dashboard.panelGroups[0].panels[0].metrics[0];
// Mock having one out of three metrics failing // Mock having one out of four metrics failing
dispatch.mockRejectedValueOnce(new Error('Error fetching this metric')); dispatch.mockRejectedValueOnce(new Error('Error fetching this metric'));
dispatch.mockResolvedValue(); dispatch.mockResolvedValue();
fetchPrometheusMetrics({ state, commit, dispatch }, params) fetchPrometheusMetrics({ state, commit, dispatch }, params)
.then(() => { .then(() => {
expect(dispatch).toHaveBeenCalledTimes(3); expect(dispatch).toHaveBeenCalledTimes(9); // one per metric
expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetric', { expect(dispatch).toHaveBeenCalledWith('fetchPrometheusMetric', {
metric, metric,
params, params,
...@@ -508,7 +513,12 @@ describe('Monitoring store actions', () => { ...@@ -508,7 +513,12 @@ describe('Monitoring store actions', () => {
beforeEach(() => { beforeEach(() => {
state = storeState(); state = storeState();
[metric] = metricsDashboardResponse.dashboard.panel_groups[0].panels[0].metrics; [metric] = metricsDashboardResponse.dashboard.panel_groups[0].panels[0].metrics;
[data] = metricsDashboardPayload.panel_groups[0].panels[0].metrics; metric = convertObjectPropsToCamelCase(metric, { deep: true });
data = {
metricId: metric.metricId,
result: [1582065167.353, 5, 1582065599.353],
};
}); });
it('commits result', done => { it('commits result', done => {
...@@ -522,13 +532,13 @@ describe('Monitoring store actions', () => { ...@@ -522,13 +532,13 @@ describe('Monitoring store actions', () => {
{ {
type: types.REQUEST_METRIC_RESULT, type: types.REQUEST_METRIC_RESULT,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
}, },
}, },
{ {
type: types.RECEIVE_METRIC_RESULT_SUCCESS, type: types.RECEIVE_METRIC_RESULT_SUCCESS,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
result: data.result, result: data.result,
}, },
}, },
...@@ -556,13 +566,13 @@ describe('Monitoring store actions', () => { ...@@ -556,13 +566,13 @@ describe('Monitoring store actions', () => {
{ {
type: types.REQUEST_METRIC_RESULT, type: types.REQUEST_METRIC_RESULT,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
}, },
}, },
{ {
type: types.RECEIVE_METRIC_RESULT_SUCCESS, type: types.RECEIVE_METRIC_RESULT_SUCCESS,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
result: data.result, result: data.result,
}, },
}, },
...@@ -592,13 +602,13 @@ describe('Monitoring store actions', () => { ...@@ -592,13 +602,13 @@ describe('Monitoring store actions', () => {
{ {
type: types.REQUEST_METRIC_RESULT, type: types.REQUEST_METRIC_RESULT,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
}, },
}, },
{ {
type: types.RECEIVE_METRIC_RESULT_FAILURE, type: types.RECEIVE_METRIC_RESULT_FAILURE,
payload: { payload: {
metricId: metric.metric_id, metricId: metric.metricId,
error, error,
}, },
}, },
......
...@@ -32,7 +32,7 @@ describe('Monitoring store Getters', () => { ...@@ -32,7 +32,7 @@ describe('Monitoring store Getters', () => {
it('when dashboard has no panel groups, returns empty', () => { it('when dashboard has no panel groups, returns empty', () => {
setupState({ setupState({
dashboard: { dashboard: {
panel_groups: [], panelGroups: [],
}, },
}); });
...@@ -43,10 +43,10 @@ describe('Monitoring store Getters', () => { ...@@ -43,10 +43,10 @@ describe('Monitoring store Getters', () => {
let groups; let groups;
beforeEach(() => { beforeEach(() => {
setupState({ setupState({
dashboard: { panel_groups: [] }, dashboard: { panelGroups: [] },
}); });
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](state, metricsDashboardPayload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](state, metricsDashboardPayload);
groups = state.dashboard.panel_groups; groups = state.dashboard.panelGroups;
}); });
it('no loaded metric returns empty', () => { it('no loaded metric returns empty', () => {
...@@ -84,8 +84,8 @@ describe('Monitoring store Getters', () => { ...@@ -84,8 +84,8 @@ describe('Monitoring store Getters', () => {
expect(getMetricStates()).toEqual([metricStates.OK]); expect(getMetricStates()).toEqual([metricStates.OK]);
// Filtered by groups // Filtered by groups
expect(getMetricStates(state.dashboard.panel_groups[0].key)).toEqual([]); expect(getMetricStates(state.dashboard.panelGroups[0].key)).toEqual([metricStates.OK]);
expect(getMetricStates(state.dashboard.panel_groups[1].key)).toEqual([metricStates.OK]); expect(getMetricStates(state.dashboard.panelGroups[1].key)).toEqual([]);
}); });
it('on multiple metrics errors', () => { it('on multiple metrics errors', () => {
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](state, metricsDashboardPayload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](state, metricsDashboardPayload);
...@@ -94,10 +94,10 @@ describe('Monitoring store Getters', () => { ...@@ -94,10 +94,10 @@ describe('Monitoring store Getters', () => {
metricId: groups[0].panels[0].metrics[0].metricId, metricId: groups[0].panels[0].metrics[0].metricId,
}); });
mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, { mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, {
metricId: groups[1].panels[0].metrics[0].metricId, metricId: groups[0].panels[0].metrics[0].metricId,
}); });
mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, { mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, {
metricId: groups[1].panels[1].metrics[0].metricId, metricId: groups[1].panels[0].metrics[0].metricId,
}); });
// Entire dashboard fails // Entire dashboard fails
...@@ -113,18 +113,18 @@ describe('Monitoring store Getters', () => { ...@@ -113,18 +113,18 @@ describe('Monitoring store Getters', () => {
mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayload); mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayload);
// An error in 2 groups // An error in 2 groups
mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, { mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, {
metricId: groups[0].panels[0].metrics[0].metricId, metricId: groups[0].panels[1].metrics[0].metricId,
}); });
mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, { mutations[types.RECEIVE_METRIC_RESULT_FAILURE](state, {
metricId: groups[1].panels[1].metrics[0].metricId, metricId: groups[1].panels[0].metrics[0].metricId,
}); });
expect(getMetricStates()).toEqual([metricStates.OK, metricStates.UNKNOWN_ERROR]); expect(getMetricStates()).toEqual([metricStates.OK, metricStates.UNKNOWN_ERROR]);
expect(getMetricStates(groups[0].key)).toEqual([metricStates.UNKNOWN_ERROR]); expect(getMetricStates(groups[0].key)).toEqual([
expect(getMetricStates(groups[1].key)).toEqual([
metricStates.OK, metricStates.OK,
metricStates.UNKNOWN_ERROR, metricStates.UNKNOWN_ERROR,
]); ]);
expect(getMetricStates(groups[1].key)).toEqual([metricStates.UNKNOWN_ERROR]);
}); });
}); });
}); });
...@@ -154,7 +154,7 @@ describe('Monitoring store Getters', () => { ...@@ -154,7 +154,7 @@ describe('Monitoring store Getters', () => {
it('when dashboard has no panel groups, returns empty', () => { it('when dashboard has no panel groups, returns empty', () => {
setupState({ setupState({
dashboard: { dashboard: {
panel_groups: [], panelGroups: [],
}, },
}); });
...@@ -164,7 +164,7 @@ describe('Monitoring store Getters', () => { ...@@ -164,7 +164,7 @@ describe('Monitoring store Getters', () => {
describe('when the dashboard is set', () => { describe('when the dashboard is set', () => {
beforeEach(() => { beforeEach(() => {
setupState({ setupState({
dashboard: { panel_groups: [] }, dashboard: { panelGroups: [] },
}); });
}); });
...@@ -204,14 +204,14 @@ describe('Monitoring store Getters', () => { ...@@ -204,14 +204,14 @@ describe('Monitoring store Getters', () => {
mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayload); mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayload);
mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayloadCoresTotal); mutations[types.RECEIVE_METRIC_RESULT_SUCCESS](state, mockedQueryResultPayloadCoresTotal);
// First group has no metrics // First group has metrics
expect(metricsWithData(state.dashboard.panel_groups[0].key)).toEqual([]); expect(metricsWithData(state.dashboard.panelGroups[0].key)).toEqual([
// Second group has metrics
expect(metricsWithData(state.dashboard.panel_groups[1].key)).toEqual([
mockedQueryResultPayload.metricId, mockedQueryResultPayload.metricId,
mockedQueryResultPayloadCoresTotal.metricId, mockedQueryResultPayloadCoresTotal.metricId,
]); ]);
// Second group has no metrics
expect(metricsWithData(state.dashboard.panelGroups[1].key)).toEqual([]);
}); });
}); });
}); });
......
...@@ -4,12 +4,8 @@ import mutations from '~/monitoring/stores/mutations'; ...@@ -4,12 +4,8 @@ import mutations from '~/monitoring/stores/mutations';
import * as types from '~/monitoring/stores/mutation_types'; import * as types from '~/monitoring/stores/mutation_types';
import state from '~/monitoring/stores/state'; import state from '~/monitoring/stores/state';
import { metricStates } from '~/monitoring/constants'; import { metricStates } from '~/monitoring/constants';
import {
metricsDashboardPayload, import { metricsDashboardPayload, deploymentData, dashboardGitResponse } from '../mock_data';
deploymentData,
metricsDashboardResponse,
dashboardGitResponse,
} from '../mock_data';
describe('Monitoring mutations', () => { describe('Monitoring mutations', () => {
let stateCopy; let stateCopy;
...@@ -17,27 +13,29 @@ describe('Monitoring mutations', () => { ...@@ -17,27 +13,29 @@ describe('Monitoring mutations', () => {
beforeEach(() => { beforeEach(() => {
stateCopy = state(); stateCopy = state();
}); });
describe('RECEIVE_METRICS_DATA_SUCCESS', () => { describe('RECEIVE_METRICS_DATA_SUCCESS', () => {
let payload; let payload;
const getGroups = () => stateCopy.dashboard.panel_groups; const getGroups = () => stateCopy.dashboard.panelGroups;
beforeEach(() => { beforeEach(() => {
stateCopy.dashboard.panel_groups = []; stateCopy.dashboard.panelGroups = [];
payload = metricsDashboardPayload; payload = metricsDashboardPayload;
}); });
it('adds a key to the group', () => { it('adds a key to the group', () => {
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload);
const groups = getGroups(); const groups = getGroups();
expect(groups[0].key).toBe('response-metrics-nginx-ingress-vts-0'); expect(groups[0].key).toBe('system-metrics-kubernetes-0');
expect(groups[1].key).toBe('system-metrics-kubernetes-1'); expect(groups[1].key).toBe('response-metrics-nginx-ingress-vts-1');
}); });
it('normalizes values', () => { it('normalizes values', () => {
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload);
const expectedLabel = 'Pod average'; const expectedLabel = 'Pod average';
const { label, query_range } = getGroups()[1].panels[0].metrics[0];
const { label, queryRange } = getGroups()[0].panels[2].metrics[0];
expect(label).toEqual(expectedLabel); expect(label).toEqual(expectedLabel);
expect(query_range.length).toBeGreaterThan(0); expect(queryRange.length).toBeGreaterThan(0);
}); });
it('contains two groups, with panels with a metric each', () => { it('contains two groups, with panels with a metric each', () => {
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload);
...@@ -47,13 +45,14 @@ describe('Monitoring mutations', () => { ...@@ -47,13 +45,14 @@ describe('Monitoring mutations', () => {
expect(groups).toBeDefined(); expect(groups).toBeDefined();
expect(groups).toHaveLength(2); expect(groups).toHaveLength(2);
expect(groups[0].panels).toHaveLength(1); expect(groups[0].panels).toHaveLength(4);
expect(groups[0].panels[0].metrics).toHaveLength(1); expect(groups[0].panels[0].metrics).toHaveLength(1);
expect(groups[0].panels[1].metrics).toHaveLength(1);
expect(groups[0].panels[2].metrics).toHaveLength(1);
expect(groups[0].panels[3].metrics).toHaveLength(5);
expect(groups[1].panels).toHaveLength(3); expect(groups[1].panels).toHaveLength(1);
expect(groups[1].panels[0].metrics).toHaveLength(1); expect(groups[1].panels[0].metrics).toHaveLength(1);
expect(groups[1].panels[1].metrics).toHaveLength(1);
expect(groups[1].panels[2].metrics).toHaveLength(5);
}); });
it('assigns metrics a metric id', () => { it('assigns metrics a metric id', () => {
mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload); mutations[types.RECEIVE_METRICS_DATA_SUCCESS](stateCopy, payload);
...@@ -61,10 +60,10 @@ describe('Monitoring mutations', () => { ...@@ -61,10 +60,10 @@ describe('Monitoring mutations', () => {
const groups = getGroups(); const groups = getGroups();
expect(groups[0].panels[0].metrics[0].metricId).toEqual( expect(groups[0].panels[0].metrics[0].metricId).toEqual(
'1_response_metrics_nginx_ingress_throughput_status_code', '12_system_metrics_kubernetes_container_memory_total',
); );
expect(groups[1].panels[0].metrics[0].metricId).toEqual( expect(groups[1].panels[0].metrics[0].metricId).toEqual(
'17_system_metrics_kubernetes_container_memory_average', '1_response_metrics_nginx_ingress_throughput_status_code',
); );
}); });
}); });
...@@ -130,8 +129,8 @@ describe('Monitoring mutations', () => { ...@@ -130,8 +129,8 @@ describe('Monitoring mutations', () => {
values: [[0, 1], [1, 1], [1, 3]], values: [[0, 1], [1, 1], [1, 3]],
}, },
]; ];
const { dashboard } = metricsDashboardResponse; const dashboard = metricsDashboardPayload;
const getMetric = () => stateCopy.dashboard.panel_groups[0].panels[0].metrics[0]; const getMetric = () => stateCopy.dashboard.panelGroups[0].panels[0].metrics[0];
describe('REQUEST_METRIC_RESULT', () => { describe('REQUEST_METRIC_RESULT', () => {
beforeEach(() => { beforeEach(() => {
......
import { import {
normalizeMetric,
uniqMetricsId, uniqMetricsId,
parseEnvironmentsResponse, parseEnvironmentsResponse,
removeLeadingSlash, removeLeadingSlash,
mapToDashboardViewModel,
} from '~/monitoring/stores/utils'; } from '~/monitoring/stores/utils';
const projectPath = 'gitlab-org/gitlab-test'; const projectPath = 'gitlab-org/gitlab-test';
describe('normalizeMetric', () => { describe('mapToDashboardViewModel', () => {
[ it('maps an empty dashboard', () => {
{ args: [], expected: 'undefined_undefined' }, expect(mapToDashboardViewModel({})).toEqual({
{ args: [undefined], expected: 'undefined_undefined' }, dashboard: '',
{ args: [{ id: 'something' }], expected: 'undefined_something' }, panelGroups: [],
{ args: [{ id: 45 }], expected: 'undefined_45' }, });
{ args: [{ metric_id: 5 }], expected: '5_undefined' }, });
{ args: [{ metric_id: 'something' }], expected: 'something_undefined' },
{ it('maps a simple dashboard', () => {
args: [{ metric_id: 5, id: 'system_metrics_kubernetes_container_memory_total' }], const response = {
expected: '5_system_metrics_kubernetes_container_memory_total', dashboard: 'Dashboard Name',
}, panel_groups: [
].forEach(({ args, expected }) => { {
it(`normalizes metric to "${expected}" with args=${JSON.stringify(args)}`, () => { group: 'Group 1',
expect(normalizeMetric(...args)).toEqual({ metric_id: expected, metricId: expected }); panels: [
{
title: 'Title A',
type: 'chart-type',
y_label: 'Y Label A',
metrics: [],
},
],
},
],
};
expect(mapToDashboardViewModel(response)).toEqual({
dashboard: 'Dashboard Name',
panelGroups: [
{
group: 'Group 1',
key: 'group-1-0',
panels: [
{
title: 'Title A',
type: 'chart-type',
y_label: 'Y Label A',
metrics: [],
},
],
},
],
});
});
describe('panel groups mapping', () => {
it('key', () => {
const response = {
dashboard: 'Dashboard Name',
panel_groups: [
{
group: 'Group A',
},
{
group: 'Group B',
},
{
group: '',
unsupported_property: 'This should be removed',
},
],
};
expect(mapToDashboardViewModel(response).panelGroups).toEqual([
{
group: 'Group A',
key: 'group-a-0',
panels: [],
},
{
group: 'Group B',
key: 'group-b-1',
panels: [],
},
{
group: '',
key: 'default-2',
panels: [],
},
]);
});
});
describe('metrics mapping', () => {
const defaultLabel = 'Panel Label';
const dashboardWithMetric = (metric, label = defaultLabel) => ({
panel_groups: [
{
panels: [
{
y_label: label,
metrics: [metric],
},
],
},
],
});
const getMappedMetric = dashboard => {
return mapToDashboardViewModel(dashboard).panelGroups[0].panels[0].metrics[0];
};
it('creates a metric', () => {
const dashboard = dashboardWithMetric({});
expect(getMappedMetric(dashboard)).toEqual({
label: expect.any(String),
metricId: expect.any(String),
metric_id: expect.any(String),
});
});
it('creates a metric with a correct ids', () => {
const dashboard = dashboardWithMetric({
id: 'http_responses',
metric_id: 1,
});
expect(getMappedMetric(dashboard)).toMatchObject({
metricId: '1_http_responses',
metric_id: '1_http_responses',
});
});
it('creates a metric with a default label', () => {
const dashboard = dashboardWithMetric({});
expect(getMappedMetric(dashboard)).toMatchObject({
label: defaultLabel,
});
});
it('creates a metric with an endpoint and query', () => {
const dashboard = dashboardWithMetric({
prometheus_endpoint_path: 'http://test',
query_range: 'http_responses',
});
expect(getMappedMetric(dashboard)).toMatchObject({
prometheusEndpointPath: 'http://test',
queryRange: 'http_responses',
});
});
it('creates a metric with an ad-hoc property', () => {
// This behavior is deprecated and should be removed
// https://gitlab.com/gitlab-org/gitlab/issues/207198
const dashboard = dashboardWithMetric({
x_label: 'Another label',
unkown_option: 'unkown_data',
});
expect(getMappedMetric(dashboard)).toMatchObject({
x_label: 'Another label',
unkown_option: 'unkown_data',
});
}); });
}); });
}); });
......
/**
* This file should only contain browser specific specs.
* If you need to add or update a spec, please see spec/frontend/lib/utils/*.js
* https://gitlab.com/gitlab-org/gitlab/issues/194242#note_292137135
* https://gitlab.com/groups/gitlab-org/-/epics/895#what-if-theres-a-karma-spec-which-is-simply-unmovable-to-jest-ie-it-is-dependent-on-a-running-browser-environment
*/
import MockAdapter from 'axios-mock-adapter';
import { GlBreakpointInstance as breakpointInstance } from '@gitlab/ui/dist/utils';
import axios from '~/lib/utils/axios_utils';
import * as commonUtils from '~/lib/utils/common_utils';
import { faviconDataUrl, overlayDataUrl, faviconWithOverlayDataUrl } from './mock_data';
const PIXEL_TOLERANCE = 0.2;
/**
* Loads a data URL as the src of an
* {@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLImageElement/Image|Image}
* and resolves to that Image once loaded.
*
* @param url
* @returns {Promise}
*/
const urlToImage = url =>
new Promise(resolve => {
const img = new Image();
img.onload = function() {
resolve(img);
};
img.src = url;
});
describe('common_utils browser specific specs', () => {
describe('contentTop', () => {
it('does not add height for fileTitle or compareVersionsHeader if screen is too small', () => {
spyOn(breakpointInstance, 'isDesktop').and.returnValue(false);
setFixtures(`
<div class="diff-file file-title-flex-parent">
blah blah blah
</div>
<div class="mr-version-controls">
more blah blah blah
</div>
`);
expect(commonUtils.contentTop()).toBe(0);
});
it('adds height for fileTitle and compareVersionsHeader screen is large enough', () => {
spyOn(breakpointInstance, 'isDesktop').and.returnValue(true);
setFixtures(`
<div class="diff-file file-title-flex-parent">
blah blah blah
</div>
<div class="mr-version-controls">
more blah blah blah
</div>
`);
expect(commonUtils.contentTop()).toBe(18);
});
});
describe('createOverlayIcon', () => {
it('should return the favicon with the overlay', done => {
commonUtils
.createOverlayIcon(faviconDataUrl, overlayDataUrl)
.then(url => Promise.all([urlToImage(url), urlToImage(faviconWithOverlayDataUrl)]))
.then(([actual, expected]) => {
expect(actual).toImageDiffEqual(expected, PIXEL_TOLERANCE);
done();
})
.catch(done.fail);
});
});
describe('setFaviconOverlay', () => {
beforeEach(() => {
const favicon = document.createElement('link');
favicon.setAttribute('id', 'favicon');
favicon.setAttribute('data-original-href', faviconDataUrl);
document.body.appendChild(favicon);
});
afterEach(() => {
document.body.removeChild(document.getElementById('favicon'));
});
it('should set page favicon to provided favicon overlay', done => {
commonUtils
.setFaviconOverlay(overlayDataUrl)
.then(() => document.getElementById('favicon').getAttribute('href'))
.then(url => Promise.all([urlToImage(url), urlToImage(faviconWithOverlayDataUrl)]))
.then(([actual, expected]) => {
expect(actual).toImageDiffEqual(expected, PIXEL_TOLERANCE);
done();
})
.catch(done.fail);
});
});
describe('setCiStatusFavicon', () => {
const BUILD_URL = `${gl.TEST_HOST}/frontend-fixtures/builds-project/-/jobs/1/status.json`;
let mock;
beforeEach(() => {
const favicon = document.createElement('link');
favicon.setAttribute('id', 'favicon');
favicon.setAttribute('href', 'null');
favicon.setAttribute('data-original-href', faviconDataUrl);
document.body.appendChild(favicon);
mock = new MockAdapter(axios);
});
afterEach(() => {
mock.restore();
document.body.removeChild(document.getElementById('favicon'));
});
it('should reset favicon in case of error', done => {
mock.onGet(BUILD_URL).replyOnce(500);
commonUtils.setCiStatusFavicon(BUILD_URL).catch(() => {
const favicon = document.getElementById('favicon');
expect(favicon.getAttribute('href')).toEqual(faviconDataUrl);
done();
});
});
it('should set page favicon to CI status favicon based on provided status', done => {
mock.onGet(BUILD_URL).reply(200, {
favicon: overlayDataUrl,
});
commonUtils
.setCiStatusFavicon(BUILD_URL)
.then(() => document.getElementById('favicon').getAttribute('href'))
.then(url => Promise.all([urlToImage(url), urlToImage(faviconWithOverlayDataUrl)]))
.then(([actual, expected]) => {
expect(actual).toImageDiffEqual(expected, PIXEL_TOLERANCE);
done();
})
.catch(done.fail);
});
});
describe('isInViewport', () => {
let el;
beforeEach(() => {
el = document.createElement('div');
});
afterEach(() => {
document.body.removeChild(el);
});
it('returns true when provided `el` is in viewport', () => {
el.setAttribute('style', `position: absolute; right: ${window.innerWidth + 0.2};`);
document.body.appendChild(el);
expect(commonUtils.isInViewport(el)).toBe(true);
});
it('returns false when provided `el` is not in viewport', () => {
el.setAttribute('style', 'position: absolute; top: -1000px; left: -1000px;');
document.body.appendChild(el);
expect(commonUtils.isInViewport(el)).toBe(false);
});
});
});
export const faviconDataUrl = export * from '../../../frontend/lib/utils/mock_data.js';
'';
export const overlayDataUrl =
'';
export const faviconWithOverlayDataUrl =
'';
...@@ -112,7 +112,7 @@ describe('Dashboard', () => { ...@@ -112,7 +112,7 @@ describe('Dashboard', () => {
setupComponentStore(component); setupComponentStore(component);
return Vue.nextTick().then(() => { return Vue.nextTick().then(() => {
[, promPanel] = component.$el.querySelectorAll('.prometheus-panel'); [promPanel] = component.$el.querySelectorAll('.prometheus-panel');
promGroup = promPanel.querySelector('.prometheus-graph-group'); promGroup = promPanel.querySelector('.prometheus-graph-group');
panelToggle = promPanel.querySelector('.js-graph-group-toggle'); panelToggle = promPanel.querySelector('.js-graph-group-toggle');
chart = promGroup.querySelector('.position-relative svg'); chart = promGroup.querySelector('.position-relative svg');
......
...@@ -72,6 +72,7 @@ Note: ...@@ -72,6 +72,7 @@ Note:
- resolved_by_push - resolved_by_push
- discussion_id - discussion_id
- original_discussion_id - original_discussion_id
- confidential
LabelLink: LabelLink:
- id - id
- target_type - target_type
......
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