154 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
<script>
 | 
						|
import { GlSkeletonLoader } from '@gitlab/ui';
 | 
						|
import { flatten, isEqual, keyBy } from 'lodash';
 | 
						|
import createFlash from '~/flash';
 | 
						|
import { sprintf, s__ } from '~/locale';
 | 
						|
import { METRICS_POPOVER_CONTENT } from '../constants';
 | 
						|
import { removeFlash, prepareTimeMetricsData } from '../utils';
 | 
						|
import MetricTile from './metric_tile.vue';
 | 
						|
 | 
						|
const requestData = ({ request, endpoint, path, params, name }) => {
 | 
						|
  return request({ endpoint, params, requestPath: path })
 | 
						|
    .then(({ data }) => data)
 | 
						|
    .catch(() => {
 | 
						|
      const message = sprintf(
 | 
						|
        s__(
 | 
						|
          'ValueStreamAnalytics|There was an error while fetching value stream analytics %{requestTypeName} data.',
 | 
						|
        ),
 | 
						|
        { requestTypeName: name },
 | 
						|
      );
 | 
						|
      createFlash({ message });
 | 
						|
    });
 | 
						|
};
 | 
						|
 | 
						|
const fetchMetricsData = (reqs = [], path, params) => {
 | 
						|
  const promises = reqs.map((r) => requestData({ ...r, path, params }));
 | 
						|
  return Promise.all(promises).then((responses) =>
 | 
						|
    prepareTimeMetricsData(flatten(responses), METRICS_POPOVER_CONTENT),
 | 
						|
  );
 | 
						|
};
 | 
						|
 | 
						|
const extractMetricsGroupData = (keyList = [], data = []) => {
 | 
						|
  if (!keyList.length || !data.length) return [];
 | 
						|
  const kv = keyBy(data, 'identifier');
 | 
						|
  return keyList.map((id) => kv[id] || null).filter((obj) => Boolean(obj));
 | 
						|
};
 | 
						|
 | 
						|
const groupRawMetrics = (groups = [], rawData = []) => {
 | 
						|
  return groups.map((curr) => {
 | 
						|
    const { keys, ...rest } = curr;
 | 
						|
    return {
 | 
						|
      data: extractMetricsGroupData(keys, rawData),
 | 
						|
      keys,
 | 
						|
      ...rest,
 | 
						|
    };
 | 
						|
  });
 | 
						|
};
 | 
						|
 | 
						|
export default {
 | 
						|
  name: 'ValueStreamMetrics',
 | 
						|
  components: {
 | 
						|
    GlSkeletonLoader,
 | 
						|
    MetricTile,
 | 
						|
  },
 | 
						|
  props: {
 | 
						|
    requestPath: {
 | 
						|
      type: String,
 | 
						|
      required: true,
 | 
						|
    },
 | 
						|
    requestParams: {
 | 
						|
      type: Object,
 | 
						|
      required: true,
 | 
						|
    },
 | 
						|
    requests: {
 | 
						|
      type: Array,
 | 
						|
      required: true,
 | 
						|
    },
 | 
						|
    filterFn: {
 | 
						|
      type: Function,
 | 
						|
      required: false,
 | 
						|
      default: null,
 | 
						|
    },
 | 
						|
    groupBy: {
 | 
						|
      type: Array,
 | 
						|
      required: false,
 | 
						|
      default: () => [],
 | 
						|
    },
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      metrics: [],
 | 
						|
      groupedMetrics: [],
 | 
						|
      isLoading: false,
 | 
						|
    };
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    hasGroupedMetrics() {
 | 
						|
      return Boolean(this.groupBy.length);
 | 
						|
    },
 | 
						|
  },
 | 
						|
  watch: {
 | 
						|
    requestParams(newVal, oldVal) {
 | 
						|
      if (!isEqual(newVal, oldVal)) {
 | 
						|
        this.fetchData();
 | 
						|
      }
 | 
						|
    },
 | 
						|
  },
 | 
						|
  mounted() {
 | 
						|
    this.fetchData();
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    fetchData() {
 | 
						|
      removeFlash();
 | 
						|
      this.isLoading = true;
 | 
						|
      return fetchMetricsData(this.requests, this.requestPath, this.requestParams)
 | 
						|
        .then((data) => {
 | 
						|
          this.metrics = this.filterFn ? this.filterFn(data) : data;
 | 
						|
 | 
						|
          if (this.hasGroupedMetrics) {
 | 
						|
            this.groupedMetrics = groupRawMetrics(this.groupBy, this.metrics);
 | 
						|
          }
 | 
						|
 | 
						|
          this.isLoading = false;
 | 
						|
        })
 | 
						|
        .catch(() => {
 | 
						|
          this.isLoading = false;
 | 
						|
        });
 | 
						|
    },
 | 
						|
  },
 | 
						|
};
 | 
						|
</script>
 | 
						|
<template>
 | 
						|
  <div class="gl-display-flex" data-testid="vsa-metrics" :class="isLoading ? 'gl-my-6' : 'gl-mt-6'">
 | 
						|
    <gl-skeleton-loader v-if="isLoading" />
 | 
						|
    <template v-else>
 | 
						|
      <div v-if="hasGroupedMetrics" class="gl-flex-direction-column">
 | 
						|
        <div
 | 
						|
          v-for="group in groupedMetrics"
 | 
						|
          :key="group.key"
 | 
						|
          class="gl-mb-7"
 | 
						|
          data-testid="vsa-metrics-group"
 | 
						|
        >
 | 
						|
          <h4 class="gl-my-0">{{ group.title }}</h4>
 | 
						|
          <div class="gl-display-flex gl-flex-wrap">
 | 
						|
            <metric-tile
 | 
						|
              v-for="metric in group.data"
 | 
						|
              :key="metric.identifier"
 | 
						|
              :metric="metric"
 | 
						|
              class="gl-mt-5 gl-pr-10"
 | 
						|
            />
 | 
						|
          </div>
 | 
						|
        </div>
 | 
						|
      </div>
 | 
						|
      <div v-else class="gl-display-flex gl-flex-wrap gl-mb-7">
 | 
						|
        <metric-tile
 | 
						|
          v-for="metric in metrics"
 | 
						|
          :key="metric.identifier"
 | 
						|
          :metric="metric"
 | 
						|
          class="gl-mt-5 gl-pr-10"
 | 
						|
        />
 | 
						|
      </div>
 | 
						|
    </template>
 | 
						|
  </div>
 | 
						|
</template>
 |