From 1baafd72dcafbd88004a878b3e42be0fe4d33a7a Mon Sep 17 00:00:00 2001 From: Avielle Wolfe <awolfe@gitlab.com> Date: Wed, 28 Jul 2021 14:46:54 +0000 Subject: [PATCH] Add CI minutes usage data to GraphQL --- doc/api/graphql/reference/index.md | 75 +++++++++++++++++++ ee/app/graphql/ee/types/query_type.rb | 8 ++ .../minutes/namespace_monthly_usage_type.rb | 31 ++++++++ .../ci/minutes/project_monthly_usage_type.rb | 24 ++++++ .../ci/minutes/namespace_monthly_usage.rb | 1 + .../ci/minutes/project_monthly_usage.rb | 7 ++ .../namespace_monthly_usage_type_spec.rb | 9 +++ .../project_monthly_usage_type_spec.rb | 9 +++ ee/spec/graphql/types/query_type_spec.rb | 11 +-- .../minutes/namespace_monthly_usage_spec.rb | 11 +++ .../ci/minutes/project_monthly_usage_spec.rb | 15 ++++ .../api/graphql/ci/minutes/usage_spec.rb | 75 +++++++++++++++++++ 12 files changed, 271 insertions(+), 5 deletions(-) create mode 100644 ee/app/graphql/types/ci/minutes/namespace_monthly_usage_type.rb create mode 100644 ee/app/graphql/types/ci/minutes/project_monthly_usage_type.rb create mode 100644 ee/spec/graphql/types/ci/minutes/namespace_monthly_usage_type_spec.rb create mode 100644 ee/spec/graphql/types/ci/minutes/project_monthly_usage_type_spec.rb create mode 100644 ee/spec/requests/api/graphql/ci/minutes/usage_spec.rb diff --git a/doc/api/graphql/reference/index.md b/doc/api/graphql/reference/index.md index e137c69e2c20..7b5598de2715 100644 --- a/doc/api/graphql/reference/index.md +++ b/doc/api/graphql/reference/index.md @@ -54,6 +54,16 @@ Returns [`CiConfig`](#ciconfig). | <a id="queryciconfigprojectpath"></a>`projectPath` | [`ID!`](#id) | The project of the CI config. | | <a id="queryciconfigsha"></a>`sha` | [`String`](#string) | Sha for the pipeline. | +### `Query.ciMinutesUsage` + +The monthly CI minutes usage data for the current user. + +Returns [`CiMinutesNamespaceMonthlyUsageConnection`](#ciminutesnamespacemonthlyusageconnection). + +This field returns a [connection](#connections). It accepts the +four standard [pagination arguments](#connection-pagination-arguments): +`before: String`, `after: String`, `first: Int`, `last: Int`. + ### `Query.containerRepository` Find a container repository. @@ -4854,6 +4864,52 @@ The edge type for [`CiJob`](#cijob). | <a id="cijobedgecursor"></a>`cursor` | [`String!`](#string) | A cursor for use in pagination. | | <a id="cijobedgenode"></a>`node` | [`CiJob`](#cijob) | The item at the end of the edge. | +#### `CiMinutesNamespaceMonthlyUsageConnection` + +The connection type for [`CiMinutesNamespaceMonthlyUsage`](#ciminutesnamespacemonthlyusage). + +##### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesnamespacemonthlyusageconnectionedges"></a>`edges` | [`[CiMinutesNamespaceMonthlyUsageEdge]`](#ciminutesnamespacemonthlyusageedge) | A list of edges. | +| <a id="ciminutesnamespacemonthlyusageconnectionnodes"></a>`nodes` | [`[CiMinutesNamespaceMonthlyUsage]`](#ciminutesnamespacemonthlyusage) | A list of nodes. | +| <a id="ciminutesnamespacemonthlyusageconnectionpageinfo"></a>`pageInfo` | [`PageInfo!`](#pageinfo) | Information to aid in pagination. | + +#### `CiMinutesNamespaceMonthlyUsageEdge` + +The edge type for [`CiMinutesNamespaceMonthlyUsage`](#ciminutesnamespacemonthlyusage). + +##### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesnamespacemonthlyusageedgecursor"></a>`cursor` | [`String!`](#string) | A cursor for use in pagination. | +| <a id="ciminutesnamespacemonthlyusageedgenode"></a>`node` | [`CiMinutesNamespaceMonthlyUsage`](#ciminutesnamespacemonthlyusage) | The item at the end of the edge. | + +#### `CiMinutesProjectMonthlyUsageConnection` + +The connection type for [`CiMinutesProjectMonthlyUsage`](#ciminutesprojectmonthlyusage). + +##### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesprojectmonthlyusageconnectionedges"></a>`edges` | [`[CiMinutesProjectMonthlyUsageEdge]`](#ciminutesprojectmonthlyusageedge) | A list of edges. | +| <a id="ciminutesprojectmonthlyusageconnectionnodes"></a>`nodes` | [`[CiMinutesProjectMonthlyUsage]`](#ciminutesprojectmonthlyusage) | A list of nodes. | +| <a id="ciminutesprojectmonthlyusageconnectionpageinfo"></a>`pageInfo` | [`PageInfo!`](#pageinfo) | Information to aid in pagination. | + +#### `CiMinutesProjectMonthlyUsageEdge` + +The edge type for [`CiMinutesProjectMonthlyUsage`](#ciminutesprojectmonthlyusage). + +##### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesprojectmonthlyusageedgecursor"></a>`cursor` | [`String!`](#string) | A cursor for use in pagination. | +| <a id="ciminutesprojectmonthlyusageedgenode"></a>`node` | [`CiMinutesProjectMonthlyUsage`](#ciminutesprojectmonthlyusage) | The item at the end of the edge. | + #### `CiRunnerConnection` The connection type for [`CiRunner`](#cirunner). @@ -7836,6 +7892,25 @@ Represents the total number of issues and their weights for a particular day. | ---- | ---- | ----------- | | <a id="cijobtokenscopetypeprojects"></a>`projects` | [`ProjectConnection!`](#projectconnection) | Allow list of projects that can be accessed by CI Job tokens created by this project. (see [Connections](#connections)) | +### `CiMinutesNamespaceMonthlyUsage` + +#### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesnamespacemonthlyusageminutes"></a>`minutes` | [`Int`](#int) | The total number of minutes used by all projects in the namespace. | +| <a id="ciminutesnamespacemonthlyusagemonth"></a>`month` | [`String`](#string) | The month related to the usage data. | +| <a id="ciminutesnamespacemonthlyusageprojects"></a>`projects` | [`CiMinutesProjectMonthlyUsageConnection`](#ciminutesprojectmonthlyusageconnection) | CI minutes usage data for projects in the namespace. (see [Connections](#connections)) | + +### `CiMinutesProjectMonthlyUsage` + +#### Fields + +| Name | Type | Description | +| ---- | ---- | ----------- | +| <a id="ciminutesprojectmonthlyusageminutes"></a>`minutes` | [`Int`](#int) | The number of CI minutes used by the project in the month. | +| <a id="ciminutesprojectmonthlyusagename"></a>`name` | [`String`](#string) | The name of the project. | + ### `CiRunner` #### Fields diff --git a/ee/app/graphql/ee/types/query_type.rb b/ee/app/graphql/ee/types/query_type.rb index 1b1c9a4b7431..c67315000078 100644 --- a/ee/app/graphql/ee/types/query_type.rb +++ b/ee/app/graphql/ee/types/query_type.rb @@ -61,6 +61,10 @@ module QueryType null: true, resolver: ::Resolvers::Admin::CloudLicenses::LicenseHistoryEntriesResolver, description: 'Fields related to entries in the license history.' + + field :ci_minutes_usage, ::Types::Ci::Minutes::NamespaceMonthlyUsageType.connection_type, + null: true, + description: 'The monthly CI minutes usage data for the current user.' end def vulnerability(id:) @@ -76,6 +80,10 @@ def iteration(id:) id = ::Types::GlobalIDType[Iteration].coerce_isolated_input(id) ::GitlabSchema.find_by_gid(id) end + + def ci_minutes_usage + ::Ci::Minutes::NamespaceMonthlyUsage.for_namespace(current_user.namespace) + end end end end diff --git a/ee/app/graphql/types/ci/minutes/namespace_monthly_usage_type.rb b/ee/app/graphql/types/ci/minutes/namespace_monthly_usage_type.rb new file mode 100644 index 000000000000..7b9f7dfdba09 --- /dev/null +++ b/ee/app/graphql/types/ci/minutes/namespace_monthly_usage_type.rb @@ -0,0 +1,31 @@ +# frozen_string_literal: true + +module Types + module Ci + module Minutes + # rubocop: disable Graphql/AuthorizeTypes + # this type only exposes data related to the current user + class NamespaceMonthlyUsageType < BaseObject + graphql_name 'CiMinutesNamespaceMonthlyUsage' + + field :month, ::GraphQL::STRING_TYPE, null: true, + description: 'The month related to the usage data.' + + field :minutes, ::GraphQL::INT_TYPE, null: true, + method: :amount_used, + description: 'The total number of minutes used by all projects in the namespace.' + + field :projects, ::Types::Ci::Minutes::ProjectMonthlyUsageType.connection_type, null: true, + description: 'CI minutes usage data for projects in the namespace.' + + def month + object.date.strftime('%B') + end + + def projects + ::Ci::Minutes::ProjectMonthlyUsage.for_namespace_monthly_usage(object) + end + end + end + end +end diff --git a/ee/app/graphql/types/ci/minutes/project_monthly_usage_type.rb b/ee/app/graphql/types/ci/minutes/project_monthly_usage_type.rb new file mode 100644 index 000000000000..331149bff34c --- /dev/null +++ b/ee/app/graphql/types/ci/minutes/project_monthly_usage_type.rb @@ -0,0 +1,24 @@ +# frozen_string_literal: true + +module Types + module Ci + module Minutes + # rubocop: disable Graphql/AuthorizeTypes + # this type only exposes data related to the current user + class ProjectMonthlyUsageType < BaseObject + graphql_name 'CiMinutesProjectMonthlyUsage' + + field :minutes, ::GraphQL::INT_TYPE, null: true, + method: :amount_used, + description: 'The number of CI minutes used by the project in the month.' + + field :name, ::GraphQL::STRING_TYPE, null: true, + description: 'The name of the project.' + + def name + object.project.name + end + end + end + end +end diff --git a/ee/app/models/ci/minutes/namespace_monthly_usage.rb b/ee/app/models/ci/minutes/namespace_monthly_usage.rb index b27731539a50..064355a4f032 100644 --- a/ee/app/models/ci/minutes/namespace_monthly_usage.rb +++ b/ee/app/models/ci/minutes/namespace_monthly_usage.rb @@ -10,6 +10,7 @@ class NamespaceMonthlyUsage < ApplicationRecord belongs_to :namespace scope :current_month, -> { where(date: beginning_of_month) } + scope :for_namespace, -> (namespace) { where(namespace: namespace) } def self.beginning_of_month(time = Time.current) time.utc.beginning_of_month diff --git a/ee/app/models/ci/minutes/project_monthly_usage.rb b/ee/app/models/ci/minutes/project_monthly_usage.rb index 1c4afab0f788..ad134c36aa4e 100644 --- a/ee/app/models/ci/minutes/project_monthly_usage.rb +++ b/ee/app/models/ci/minutes/project_monthly_usage.rb @@ -11,6 +11,13 @@ class ProjectMonthlyUsage < ApplicationRecord scope :current_month, -> { where(date: beginning_of_month) } + scope :for_namespace_monthly_usage, -> (namespace_monthly_usage) do + where( + date: namespace_monthly_usage.date, + project: namespace_monthly_usage.namespace.projects + ) + end + def self.beginning_of_month(time = Time.current) time.utc.beginning_of_month end diff --git a/ee/spec/graphql/types/ci/minutes/namespace_monthly_usage_type_spec.rb b/ee/spec/graphql/types/ci/minutes/namespace_monthly_usage_type_spec.rb new file mode 100644 index 000000000000..0acf99b79673 --- /dev/null +++ b/ee/spec/graphql/types/ci/minutes/namespace_monthly_usage_type_spec.rb @@ -0,0 +1,9 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe GitlabSchema.types['CiMinutesNamespaceMonthlyUsage'] do + it do + expect(described_class).to have_graphql_fields(:minutes, :month, :projects) + end +end diff --git a/ee/spec/graphql/types/ci/minutes/project_monthly_usage_type_spec.rb b/ee/spec/graphql/types/ci/minutes/project_monthly_usage_type_spec.rb new file mode 100644 index 000000000000..8e0450e0e0d0 --- /dev/null +++ b/ee/spec/graphql/types/ci/minutes/project_monthly_usage_type_spec.rb @@ -0,0 +1,9 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe GitlabSchema.types['CiMinutesProjectMonthlyUsage'] do + it do + expect(described_class).to have_graphql_fields(:minutes, :name) + end +end diff --git a/ee/spec/graphql/types/query_type_spec.rb b/ee/spec/graphql/types/query_type_spec.rb index 8a8716cb8995..54b88f262b80 100644 --- a/ee/spec/graphql/types/query_type_spec.rb +++ b/ee/spec/graphql/types/query_type_spec.rb @@ -5,14 +5,15 @@ RSpec.describe GitlabSchema.types['Query'] do specify do expect(described_class).to have_graphql_fields( - :iteration, + :ci_minutes_usage, + :current_license, :geo_node, - :vulnerabilities, - :vulnerability, :instance_security_dashboard, + :iteration, + :license_history_entries, + :vulnerabilities, :vulnerabilities_count_by_day, - :current_license, - :license_history_entries + :vulnerability ).at_least end end diff --git a/ee/spec/models/ci/minutes/namespace_monthly_usage_spec.rb b/ee/spec/models/ci/minutes/namespace_monthly_usage_spec.rb index e64c6e456daf..ccfcb12f9f19 100644 --- a/ee/spec/models/ci/minutes/namespace_monthly_usage_spec.rb +++ b/ee/spec/models/ci/minutes/namespace_monthly_usage_spec.rb @@ -90,4 +90,15 @@ end end end + + describe '.for_namespace' do + it 'returns usages for the namespace' do + matching_usage = create(:ci_namespace_monthly_usage, namespace: namespace) + create(:ci_namespace_monthly_usage, namespace: create(:namespace)) + + usages = described_class.for_namespace(namespace) + + expect(usages).to contain_exactly(matching_usage) + end + end end diff --git a/ee/spec/models/ci/minutes/project_monthly_usage_spec.rb b/ee/spec/models/ci/minutes/project_monthly_usage_spec.rb index 5bbc0a2a0a99..6004f2dece7d 100644 --- a/ee/spec/models/ci/minutes/project_monthly_usage_spec.rb +++ b/ee/spec/models/ci/minutes/project_monthly_usage_spec.rb @@ -90,4 +90,19 @@ end end end + + describe '.for_namespace_monthly_usage' do + it "fetches project monthly usages matching the namespace monthly usage's date and namespace" do + date_for_usage = Date.new(2021, 5, 1) + date_not_for_usage = date_for_usage + 1.month + namespace_usage = create(:ci_namespace_monthly_usage, namespace: project.namespace, amount_used: 50, date: date_for_usage) + matching_project_usage = create(:ci_project_monthly_usage, project: project, amount_used: 50, date: date_for_usage) + create(:ci_project_monthly_usage, project: project, amount_used: 50, date: date_not_for_usage) + create(:ci_project_monthly_usage, project: create(:project), amount_used: 50, date: date_for_usage) + + project_usages = described_class.for_namespace_monthly_usage(namespace_usage) + + expect(project_usages).to contain_exactly(matching_project_usage) + end + end end diff --git a/ee/spec/requests/api/graphql/ci/minutes/usage_spec.rb b/ee/spec/requests/api/graphql/ci/minutes/usage_spec.rb new file mode 100644 index 000000000000..1d174898a11b --- /dev/null +++ b/ee/spec/requests/api/graphql/ci/minutes/usage_spec.rb @@ -0,0 +1,75 @@ +# frozen_string_literal: true + +require 'spec_helper' + +RSpec.describe 'Query.ciMinutesUsage' do + include GraphqlHelpers + + let_it_be(:user) { create(:user) } + let_it_be(:project) { create(:project, name: 'Project 1', namespace: user.namespace) } + + before(:all) do + create(:ci_namespace_monthly_usage, namespace: user.namespace, amount_used: 50, date: Date.new(2021, 5, 1)) + create(:ci_project_monthly_usage, project: project, amount_used: 50, date: Date.new(2021, 5, 1)) + end + + it 'returns usage data by month for the current user' do + query = <<-QUERY + { + ciMinutesUsage { + nodes { + minutes + month + projects { + nodes { + name + minutes + } + } + } + } + } + QUERY + + post_graphql(query, current_user: user) + + monthly_usage = graphql_data_at(:ci_minutes_usage, :nodes) + expect(monthly_usage).to contain_exactly({ + 'month' => 'May', + 'minutes' => 50, + 'projects' => { 'nodes' => [{ + 'name' => 'Project 1', + 'minutes' => 50 + }] } + }) + end + + it 'does not create N+1 queries' do + query = <<-QUERY + { + ciMinutesUsage { + nodes { + projects { + nodes { + name + } + } + } + } + } + QUERY + + control_count = ActiveRecord::QueryRecorder.new do + post_graphql(query, current_user: user) + end + expect(graphql_errors).to be_nil + + project_2 = create(:project, name: 'Project 2', namespace: user.namespace) + create(:ci_project_monthly_usage, project: project_2, amount_used: 50, date: Date.new(2021, 5, 1)) + + expect do + post_graphql(query, current_user: user) + end.not_to exceed_query_limit(control_count) + expect(graphql_errors).to be_nil + end +end -- GitLab