entities.rb 43.3 KB
Newer Older
1
module API
Nihad Abbasov's avatar
Nihad Abbasov committed
2
  module Entities
3 4 5 6 7 8 9 10 11 12
    class WikiPageBasic < Grape::Entity
      expose :format
      expose :slug
      expose :title
    end

    class WikiPage < WikiPageBasic
      expose :content
    end

13
    class UserSafe < Grape::Entity
14
      expose :id, :name, :username
15
    end
16

17
    class UserBasic < UserSafe
18
      expose :state
19

20 21 22
      expose :avatar_url do |user, options|
        user.avatar_url(only_path: false)
      end
Douwe Maan's avatar
Douwe Maan committed
23

24
      expose :avatar_path, if: ->(user, options) { options.fetch(:only_path, false) && user.avatar_path }
25
      expose :custom_attributes, using: 'API::Entities::CustomAttribute', if: :with_custom_attributes
26

Douwe Maan's avatar
Douwe Maan committed
27
      expose :web_url do |user, options|
28
        Gitlab::Routing.url_helpers.user_url(user)
Douwe Maan's avatar
Douwe Maan committed
29
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
30
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
31

32
    class User < UserBasic
33
      expose :created_at, if: ->(user, opts) { Ability.allowed?(opts[:current_user], :read_user_profile, user) }
34
      expose :bio, :location, :skype, :linkedin, :twitter, :website_url, :organization
35 36
    end

37 38
    class UserActivity < Grape::Entity
      expose :username
39 40
      expose :last_activity_on
      expose :last_activity_on, as: :last_activity_at # Back-compat
41 42
    end

43 44 45 46
    class Identity < Grape::Entity
      expose :provider, :extern_uid
    end

47
    class UserPublic < User
48 49
      expose :last_sign_in_at
      expose :confirmed_at
50
      expose :last_activity_on
51
      expose :email
52
      expose :theme_id, :color_scheme_id, :projects_limit, :current_sign_in_at
53
      expose :identities, using: Entities::Identity
54 55
      expose :can_create_group?, as: :can_create_group
      expose :can_create_project?, as: :can_create_project
56
      expose :two_factor_enabled?, as: :two_factor_enabled
57
      expose :external
58
      expose :private_profile
59 60
    end

61
    class UserWithAdmin < UserPublic
62
      expose :admin?, as: :is_admin
63 64
    end

65 66 67 68
    class Email < Grape::Entity
      expose :id, :email
    end

miks's avatar
miks committed
69
    class Hook < Grape::Entity
70
      expose :id, :url, :created_at, :push_events, :tag_push_events, :merge_requests_events, :repository_update_events
71
      expose :enable_ssl_verification
miks's avatar
miks committed
72 73
    end

74
    class ProjectHook < Hook
75
      expose :project_id, :issues_events, :confidential_issues_events
76
      expose :note_events, :confidential_note_events, :pipeline_events, :wiki_page_events
77
      expose :job_events
78 79
    end

80 81 82 83 84 85 86 87
    class SharedGroup < Grape::Entity
      expose :group_id
      expose :group_name do |group_link, options|
        group_link.group.name
      end
      expose :group_access, as: :group_access_level
    end

88 89
    class ProjectIdentity < Grape::Entity
      expose :id, :description
90 91
      expose :name, :name_with_namespace
      expose :path, :path_with_namespace
92 93 94
      expose :created_at
    end

95 96 97 98
    class ProjectExportStatus < ProjectIdentity
      include ::API::Helpers::RelatedResourcesHelpers

      expose :export_status
99
      expose :_links, if: lambda { |project, _options| project.export_status == :finished } do
100 101 102 103 104 105 106 107 108 109
        expose :api_url do |project|
          expose_url(api_v4_projects_export_download_path(id: project.id))
        end

        expose :web_url do |project|
          Gitlab::Routing.url_helpers.download_export_project_url(project)
        end
      end
    end

James Lopez's avatar
James Lopez committed
110 111
    class ProjectImportStatus < ProjectIdentity
      expose :import_status
James Lopez's avatar
James Lopez committed
112 113 114

      # TODO: Use `expose_nil` once we upgrade the grape-entity gem
      expose :import_error, if: lambda { |status, _ops| status.import_error }
James Lopez's avatar
James Lopez committed
115 116
    end

117
    class BasicProjectDetails < ProjectIdentity
Francisco Lopez's avatar
Francisco Lopez committed
118 119 120 121 122 123 124 125 126 127 128
      include ::API::ProjectsRelationBuilder

      expose :default_branch
      # Avoids an N+1 query: https://github.com/mbleigh/acts-as-taggable-on/issues/91#issuecomment-168273770
      expose :tag_list do |project|
        # project.tags.order(:name).pluck(:name) is the most suitable option
        # to avoid loading all the ActiveRecord objects but, if we use it here
        # it override the preloaded associations and makes a query
        # (fixed in https://github.com/rails/rails/pull/25976).
        project.tags.map(&:name).sort
      end
129
      expose :ssh_url_to_repo, :http_url_to_repo, :web_url, :readme_url
130 131 132
      expose :avatar_url do |project, options|
        project.avatar_url(only_path: false)
      end
133
      expose :star_count, :forks_count
Francisco Lopez's avatar
Francisco Lopez committed
134
      expose :last_activity_at
135

136
      expose :namespace, using: 'API::Entities::NamespaceBasic'
137 138
      expose :custom_attributes, using: 'API::Entities::CustomAttribute', if: :with_custom_attributes

139
      def self.preload_relation(projects_relation, options =  {})
140 141 142 143
        # Preloading tags, should be done with using only `:tags`,
        # as `:tags` are defined as: `has_many :tags, through: :taggings`
        # N+1 is solved then by using `subject.tags.map(&:name)`
        # MR describing the solution: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/20555
144
        projects_relation.preload(:project_feature, :route)
145 146
                         .preload(:import_state, :tags)
                         .preload(namespace: [:route, :owner])
147
      end
148 149
    end

150
    class Project < BasicProjectDetails
151 152 153 154 155 156 157
      include ::API::Helpers::RelatedResourcesHelpers

      expose :_links do
        expose :self do |project|
          expose_url(api_v4_projects_path(id: project.id))
        end

158
        expose :issues, if: -> (project, options) { issues_available?(project, options) } do |project|
159 160 161
          expose_url(api_v4_projects_issues_path(id: project.id))
        end

162
        expose :merge_requests, if: -> (project, options) { mrs_available?(project, options) } do |project|
163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182
          expose_url(api_v4_projects_merge_requests_path(id: project.id))
        end

        expose :repo_branches do |project|
          expose_url(api_v4_projects_repository_branches_path(id: project.id))
        end

        expose :labels do |project|
          expose_url(api_v4_projects_labels_path(id: project.id))
        end

        expose :events do |project|
          expose_url(api_v4_projects_events_path(id: project.id))
        end

        expose :members do |project|
          expose_url(api_v4_projects_members_path(id: project.id))
        end
      end

183
      expose :archived?, as: :archived
184
      expose :visibility
185
      expose :owner, using: Entities::UserBasic, unless: ->(project, options) { project.group }
186
      expose :resolve_outdated_diff_discussions
187 188 189
      expose :container_registry_enabled

      # Expose old field names with the new permissions methods to keep API compatible
190 191 192
      expose(:issues_enabled) { |project, options| project.feature_available?(:issues, options[:current_user]) }
      expose(:merge_requests_enabled) { |project, options| project.feature_available?(:merge_requests, options[:current_user]) }
      expose(:wiki_enabled) { |project, options| project.feature_available?(:wiki, options[:current_user]) }
193
      expose(:jobs_enabled) { |project, options| project.feature_available?(:builds, options[:current_user]) }
194
      expose(:snippets_enabled) { |project, options| project.feature_available?(:snippets, options[:current_user]) }
195

196 197
      expose :shared_runners_enabled
      expose :lfs_enabled?, as: :lfs_enabled
198
      expose :creator_id
199
      expose :forked_from_project, using: Entities::BasicProjectDetails, if: lambda { |project, options| project.forked? }
200 201
      expose :import_status
      expose :import_error, if: lambda { |_project, options| options[:user_can_admin_project] }
202

203
      expose :open_issues_count, if: lambda { |project, options| project.feature_available?(:issues, options[:current_user]) }
204
      expose :runners_token, if: lambda { |_project, options| options[:user_can_admin_project] }
205
      expose :public_builds, as: :public_jobs
206
      expose :ci_config_path
207
      expose :shared_with_groups do |project, options|
208
        SharedGroup.represent(project.project_group_links, options)
209
      end
210
      expose :only_allow_merge_if_pipeline_succeeds
211
      expose :request_access_enabled
212
      expose :only_allow_merge_if_all_discussions_are_resolved
213
      expose :printing_merge_request_link_enabled
214
      expose :merge_method
215 216

      expose :statistics, using: 'API::Entities::ProjectStatistics', if: :statistics
217 218

      def self.preload_relation(projects_relation, options =  {})
219 220 221 222
        # Preloading tags, should be done with using only `:tags`,
        # as `:tags` are defined as: `has_many :tags, through: :taggings`
        # N+1 is solved then by using `subject.tags.map(&:name)`
        # MR describing the solution: https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/20555
223 224 225 226
        super(projects_relation).preload(:group)
                                .preload(project_group_links: :group,
                                         fork_network: :root_project,
                                         forked_project_link: :forked_from_project,
227
                                         forked_from_project: [:route, :forks, :tags, namespace: :route])
228 229 230
      end

      def self.forks_counting_projects(projects_relation)
231
        projects_relation + projects_relation.map(&:forked_from_project).compact
232
      end
233 234 235 236 237 238 239
    end

    class ProjectStatistics < Grape::Entity
      expose :commit_count
      expose :storage_size
      expose :repository_size
      expose :lfs_objects_size
240
      expose :build_artifacts_size, as: :job_artifacts_size
Nihad Abbasov's avatar
Nihad Abbasov committed
241 242
    end

243 244 245 246
    class Member < Grape::Entity
      expose :user, merge: true, using: UserBasic
      expose :access_level
      expose :expires_at
247 248
    end

249 250 251
    class AccessRequester < Grape::Entity
      expose :user, merge: true, using: UserBasic
      expose :requested_at
miks's avatar
miks committed
252 253
    end

254 255 256 257 258 259 260 261
    class BasicGroupDetails < Grape::Entity
      expose :id
      expose :web_url
      expose :name
    end

    class Group < BasicGroupDetails
      expose :path, :description, :visibility
262
      expose :lfs_enabled?, as: :lfs_enabled
263 264
      expose :avatar_url do |group, options|
        group.avatar_url(only_path: false)
265
      end
266
      expose :request_access_enabled
267
      expose :full_name, :full_path
268 269 270 271

      if ::Group.supports_nested_groups?
        expose :parent_id
      end
272

273 274
      expose :custom_attributes, using: 'API::Entities::CustomAttribute', if: :with_custom_attributes

275 276 277 278 279
      expose :statistics, if: :statistics do
        with_options format_with: -> (value) { value.to_i } do
          expose :storage_size
          expose :repository_size
          expose :lfs_objects_size
280
          expose :build_artifacts_size, as: :job_artifacts_size
281 282
        end
      end
283
    end
Andrew8xx8's avatar
Andrew8xx8 committed
284

285
    class GroupDetail < Group
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300
      expose :projects, using: Entities::Project do |group, options|
        GroupProjectsFinder.new(
          group: group,
          current_user: options[:current_user],
          options: { only_owned: true }
        ).execute
      end

      expose :shared_projects, using: Entities::Project do |group, options|
        GroupProjectsFinder.new(
          group: group,
          current_user: options[:current_user],
          options: { only_shared: true }
        ).execute
      end
301 302
    end

303 304 305 306
    class DiffRefs < Grape::Entity
      expose :base_sha, :head_sha, :start_sha
    end

307
    class Commit < Grape::Entity
308 309 310 311 312 313 314
      expose :id, :short_id, :title, :created_at
      expose :parent_ids
      expose :safe_message, as: :message
      expose :author_name, :author_email, :authored_date
      expose :committer_name, :committer_email, :committed_date
    end

315
    class CommitStats < Grape::Entity
316 317 318
      expose :additions, :deletions, :total
    end

319 320 321 322
    class CommitWithStats < Commit
      expose :stats, using: Entities::CommitStats
    end

323
    class CommitDetail < Commit
324
      expose :stats, using: Entities::CommitStats, if: :stats
325
      expose :status
326
      expose :last_pipeline, using: 'API::Entities::PipelineBasic'
327
      expose :project_id
328 329
    end

330
    class BasicRef < Grape::Entity
331
      expose :type, :name
332 333
    end

Robert Schilling's avatar
Robert Schilling committed
334
    class Branch < Grape::Entity
335 336
      expose :name

337
      expose :commit, using: Entities::Commit do |repo_branch, options|
338
        options[:project].repository.commit(repo_branch.dereferenced_target)
339 340
      end

341
      expose :merged do |repo_branch, options|
342 343 344 345 346
        if options[:merged_branch_names]
          options[:merged_branch_names].include?(repo_branch.name)
        else
          options[:project].repository.merged_to_root_ref?(repo_branch)
        end
347 348
      end

349
      expose :protected do |repo_branch, options|
350
        ::ProtectedBranch.protected?(options[:project], repo_branch.name)
351 352
      end

353
      expose :developers_can_push do |repo_branch, options|
354
        options[:project].protected_branches.developers_can?(:push, repo_branch.name)
355
      end
356

357
      expose :developers_can_merge do |repo_branch, options|
358
        options[:project].protected_branches.developers_can?(:merge, repo_branch.name)
359 360 361 362
      end

      expose :can_push do |repo_branch, options|
        Gitlab::UserAccess.new(options[:current_user], project: options[:project]).can_push_to_branch?(repo_branch.name)
363
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
364
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
365

366
    class TreeObject < Grape::Entity
367
      expose :id, :name, :type, :path
368 369

      expose :mode do |obj, options|
370
        filemode = obj.mode
371 372 373 374 375
        filemode = "0" + filemode if filemode.length < 6
        filemode
      end
    end

Jarka Kadlecová's avatar
Jarka Kadlecová committed
376
    class Snippet < Grape::Entity
377
      expose :id, :title, :file_name, :description, :visibility
378
      expose :author, using: Entities::UserBasic
379
      expose :updated_at, :created_at
Jarka Kadlecová's avatar
Jarka Kadlecová committed
380 381
      expose :project_id
      expose :web_url do |snippet|
382 383
        Gitlab::UrlBuilder.build(snippet)
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
384
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
385

Jarka Kadlecová's avatar
Jarka Kadlecová committed
386 387
    class ProjectSnippet < Snippet
    end
388

Jarka Kadlecová's avatar
Jarka Kadlecová committed
389
    class PersonalSnippet < Snippet
390 391 392 393 394
      expose :raw_url do |snippet|
        Gitlab::UrlBuilder.build(snippet) + "/raw"
      end
    end

395 396
    class ProjectEntity < Grape::Entity
      expose :id, :iid
Felipe Artur's avatar
Felipe Artur committed
397
      expose(:project_id) { |entity| entity&.project.try(:id) }
398 399
      expose :title, :description
      expose :state, :created_at, :updated_at
400 401
    end

402
    class Diff < Grape::Entity
403
      expose :old_path, :new_path, :a_mode, :b_mode
404 405 406
      expose :new_file?, as: :new_file
      expose :renamed_file?, as: :renamed_file
      expose :deleted_file?, as: :deleted_file
407
      expose :json_safe_diff, as: :diff
408 409
    end

410 411 412 413 414 415 416 417 418 419 420 421 422
    class ProtectedRefAccess < Grape::Entity
      expose :access_level
      expose :access_level_description do |protected_ref_access|
        protected_ref_access.humanize
      end
    end

    class ProtectedBranch < Grape::Entity
      expose :name
      expose :push_access_levels, using: Entities::ProtectedRefAccess
      expose :merge_access_levels, using: Entities::ProtectedRefAccess
    end

Felipe Artur's avatar
Felipe Artur committed
423 424
    class Milestone < Grape::Entity
      expose :id, :iid
425 426
      expose :project_id, if: -> (entity, options) { entity&.project_id }
      expose :group_id, if: -> (entity, options) { entity&.group_id }
Felipe Artur's avatar
Felipe Artur committed
427 428
      expose :title, :description
      expose :state, :created_at, :updated_at
429
      expose :due_date
430
      expose :start_date
431 432 433 434

      expose :web_url do |milestone, _options|
        Gitlab::UrlBuilder.build(milestone)
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
435 436
    end

437
    class IssueBasic < ProjectEntity
438
      expose :closed_at
haseeb's avatar
haseeb committed
439
      expose :closed_by, using: Entities::UserBasic
440 441 442 443
      expose :labels do |issue, options|
        # Avoids an N+1 query since labels are preloaded
        issue.labels.map(&:title).sort
      end
444
      expose :milestone, using: Entities::Milestone
445 446 447 448 449
      expose :assignees, :author, using: Entities::UserBasic

      expose :assignee, using: ::API::Entities::UserBasic do |issue, options|
        issue.assignees.first
      end
450

Z.J. van de Weg's avatar
Z.J. van de Weg committed
451
      expose :user_notes_count
452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467
      expose :upvotes do |issue, options|
        if options[:issuable_metadata]
          # Avoids an N+1 query when metadata is included
          options[:issuable_metadata][issue.id].upvotes
        else
          issue.upvotes
        end
      end
      expose :downvotes do |issue, options|
        if options[:issuable_metadata]
          # Avoids an N+1 query when metadata is included
          options[:issuable_metadata][issue.id].downvotes
        else
          issue.downvotes
        end
      end
468
      expose :due_date
469
      expose :confidential
470
      expose :discussion_locked
471 472 473 474

      expose :web_url do |issue, options|
        Gitlab::UrlBuilder.build(issue)
      end
475 476 477 478

      expose :time_stats, using: 'API::Entities::IssuableTimeStats' do |issue|
        issue
      end
Nihad Abbasov's avatar
Nihad Abbasov committed
479
    end
Alex Denisov's avatar
Alex Denisov committed
480

481
    class Issue < IssueBasic
482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501
      include ::API::Helpers::RelatedResourcesHelpers

      expose :_links do
        expose :self do |issue|
          expose_url(api_v4_project_issue_path(id: issue.project_id, issue_iid: issue.iid))
        end

        expose :notes do |issue|
          expose_url(api_v4_projects_issues_notes_path(id: issue.project_id, noteable_id: issue.iid))
        end

        expose :award_emoji do |issue|
          expose_url(api_v4_projects_issues_award_emoji_path(id: issue.project_id, issue_iid: issue.iid))
        end

        expose :project do |issue|
          expose_url(api_v4_projects_path(id: issue.project_id))
        end
      end

502 503 504 505 506
      expose :subscribed do |issue, options|
        issue.subscribed?(options[:current_user], options[:project] || issue.project)
      end
    end

507
    class IssuableTimeStats < Grape::Entity
508 509 510 511
      format_with(:time_tracking_formatter) do |time_spent|
        Gitlab::TimeTrackingFormatter.output(time_spent)
      end

512 513 514
      expose :time_estimate
      expose :total_time_spent
      expose :human_time_estimate
515 516 517 518 519 520 521 522 523

      with_options(format_with: :time_tracking_formatter) do
        expose :total_time_spent, as: :human_total_time_spent
      end

      def total_time_spent
        # Avoids an N+1 query since timelogs are preloaded
        object.timelogs.map(&:time_spent).sum
      end
524 525
    end

526 527 528 529 530
    class ExternalIssue < Grape::Entity
      expose :title
      expose :id
    end

531 532
    class PipelineBasic < Grape::Entity
      expose :id, :sha, :ref, :status
533 534 535 536

      expose :web_url do |pipeline, _options|
        Gitlab::Routing.url_helpers.project_pipeline_url(pipeline.project, pipeline)
      end
537 538
    end

539 540 541 542 543 544 545
    class MergeRequestSimple < ProjectEntity
      expose :title
      expose :web_url do |merge_request, options|
        Gitlab::UrlBuilder.build(merge_request)
      end
    end

546
    class MergeRequestBasic < ProjectEntity
547
      expose :title_html, if: -> (_, options) { options[:render_html] } do |entity|
Phil Hughes's avatar
Phil Hughes committed
548
        MarkupHelper.markdown_field(entity, :title)
549 550
      end
      expose :description_html, if: -> (_, options) { options[:render_html] } do |entity|
Phil Hughes's avatar
Phil Hughes committed
551
        MarkupHelper.markdown_field(entity, :description)
552
      end
Valery Sizov's avatar
Valery Sizov committed
553
      expose :target_branch, :source_branch
554 555 556 557 558 559 560 561 562 563 564 565 566 567
      expose :upvotes do |merge_request, options|
        if options[:issuable_metadata]
          options[:issuable_metadata][merge_request.id].upvotes
        else
          merge_request.upvotes
        end
      end
      expose :downvotes do |merge_request, options|
        if options[:issuable_metadata]
          options[:issuable_metadata][merge_request.id].downvotes
        else
          merge_request.downvotes
        end
      end
568 569
      expose :author, :assignee, using: Entities::UserBasic
      expose :source_project_id, :target_project_id
570 571 572 573
      expose :labels do |merge_request, options|
        # Avoids an N+1 query since labels are preloaded
        merge_request.labels.map(&:title).sort
      end
574
      expose :work_in_progress?, as: :work_in_progress
575
      expose :milestone, using: Entities::Milestone
576
      expose :merge_when_pipeline_succeeds
577 578 579 580 581 582

      # Ideally we should deprecate `MergeRequest#merge_status` exposure and
      # use `MergeRequest#mergeable?` instead (boolean).
      # See https://gitlab.com/gitlab-org/gitlab-ce/issues/42344 for more
      # information.
      expose :merge_status do |merge_request|
583 584
        merge_request.check_if_can_be_merged
        merge_request.merge_status
585
      end
586 587
      expose :diff_head_sha, as: :sha
      expose :merge_commit_sha
Z.J. van de Weg's avatar
Z.J. van de Weg committed
588
      expose :user_notes_count
589
      expose :discussion_locked
590 591
      expose :should_remove_source_branch?, as: :should_remove_source_branch
      expose :force_remove_source_branch?, as: :force_remove_source_branch
592 593 594
      expose :allow_collaboration, if: -> (merge_request, _) { merge_request.for_fork? }
      # Deprecated
      expose :allow_collaboration, as: :allow_maintainer_to_push, if: -> (merge_request, _) { merge_request.for_fork? }
595 596 597 598

      expose :web_url do |merge_request, options|
        Gitlab::UrlBuilder.build(merge_request)
      end
599 600 601 602

      expose :time_stats, using: 'API::Entities::IssuableTimeStats' do |merge_request|
        merge_request
      end
603 604

      expose :squash
Alex Denisov's avatar
Alex Denisov committed
605
    end
606

607 608 609 610
    class MergeRequest < MergeRequestBasic
      expose :subscribed do |merge_request, options|
        merge_request.subscribed?(options[:current_user], options[:project])
      end
611 612 613 614

      expose :changes_count do |merge_request, _options|
        merge_request.merge_request_diff.real_size
      end
615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647

      expose :merged_by, using: Entities::UserBasic do |merge_request, _options|
        merge_request.metrics&.merged_by
      end

      expose :merged_at do |merge_request, _options|
        merge_request.metrics&.merged_at
      end

      expose :closed_by, using: Entities::UserBasic do |merge_request, _options|
        merge_request.metrics&.latest_closed_by
      end

      expose :closed_at do |merge_request, _options|
        merge_request.metrics&.latest_closed_at
      end

      expose :latest_build_started_at, if: -> (_, options) { build_available?(options) } do |merge_request, _options|
        merge_request.metrics&.latest_build_started_at
      end

      expose :latest_build_finished_at, if: -> (_, options) { build_available?(options) } do |merge_request, _options|
        merge_request.metrics&.latest_build_finished_at
      end

      expose :first_deployed_to_production_at, if: -> (_, options) { build_available?(options) } do |merge_request, _options|
        merge_request.metrics&.first_deployed_to_production_at
      end

      expose :pipeline, using: Entities::PipelineBasic, if: -> (_, options) { build_available?(options) } do |merge_request, _options|
        merge_request.metrics&.pipeline
      end

648 649
      expose :diff_refs, using: Entities::DiffRefs

650 651 652
      def build_available?(options)
        options[:project]&.feature_available?(:builds, options[:current_user])
      end
653 654
    end

655
    class MergeRequestChanges < MergeRequest
656
      expose :diffs, as: :changes, using: Entities::Diff do |compare, _|
Douwe Maan's avatar
Douwe Maan committed
657
        compare.raw_diffs(limits: false).to_a
658 659 660
      end
    end

661 662 663
    class MergeRequestDiff < Grape::Entity
      expose :id, :head_commit_sha, :base_commit_sha, :start_commit_sha,
        :created_at, :merge_request_id, :state, :real_size
664
    end
665

666
    class MergeRequestDiffFull < MergeRequestDiff
667
      expose :commits, using: Entities::Commit
668

669
      expose :diffs, using: Entities::Diff do |compare, _|
Douwe Maan's avatar
Douwe Maan committed
670
        compare.raw_diffs(limits: false).to_a
671 672 673
      end
    end

674
    class SSHKey < Grape::Entity
675
      expose :id, :title, :key, :created_at
676
    end
677

678
    class SSHKeyWithUser < SSHKey
679
      expose :user, using: Entities::UserPublic
680 681
    end

682 683 684 685 686
    class DeployKeysProject < Grape::Entity
      expose :deploy_key, merge: true, using: Entities::SSHKey
      expose :can_push
    end

687 688 689 690
    class GPGKey < Grape::Entity
      expose :id, :key, :created_at
    end

691 692 693 694 695
    class DiffPosition < Grape::Entity
      expose :base_sha, :start_sha, :head_sha, :old_path, :new_path,
        :position_type
    end

696
    class Note < Grape::Entity
sue445's avatar
sue445 committed
697 698 699
      # Only Issue and MergeRequest have iid
      NOTEABLE_TYPES_WITH_IID = %w(Issue MergeRequest).freeze

700
      expose :id
Jan Provaznik's avatar
Jan Provaznik committed
701
      expose :type
702
      expose :note, as: :body
703
      expose :attachment_identifier, as: :attachment
704
      expose :author, using: Entities::UserBasic
705
      expose :created_at, :updated_at
706
      expose :system?, as: :system
707
      expose :noteable_id, :noteable_type
sue445's avatar
sue445 committed
708

709
      expose :position, if: ->(note, options) { note.is_a?(DiffNote) } do |note|
710 711 712 713 714 715 716
        note.position.to_h
      end

      expose :resolvable?, as: :resolvable
      expose :resolved?, as: :resolved, if: ->(note, options) { note.resolvable? }
      expose :resolved_by, using: Entities::UserBasic, if: ->(note, options) { note.resolvable? }

sue445's avatar
sue445 committed
717 718
      # Avoid N+1 queries as much as possible
      expose(:noteable_iid) { |note| note.noteable.iid if NOTEABLE_TYPES_WITH_IID.include?(note.noteable_type) }
719
    end
720

Jan Provaznik's avatar
Jan Provaznik committed
721 722 723 724 725 726
    class Discussion < Grape::Entity
      expose :id
      expose :individual_note?, as: :individual_note
      expose :notes, using: Entities::Note
    end

Imre's avatar
Imre committed
727 728 729 730 731 732
    class Avatar < Grape::Entity
      expose :avatar_url do |avatarable, options|
        avatarable.avatar_url(only_path: false, size: options[:size])
      end
    end

733 734 735 736 737 738 739 740
    class AwardEmoji < Grape::Entity
      expose :id
      expose :name
      expose :user, using: Entities::UserBasic
      expose :created_at, :updated_at
      expose :awardable_id, :awardable_type
    end

741 742 743 744
    class MRNote < Grape::Entity
      expose :note
      expose :author, using: Entities::UserBasic
    end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
745

746 747
    class CommitNote < Grape::Entity
      expose :note
748 749 750
      expose(:path) { |note| note.diff_file.try(:file_path) if note.diff_note? }
      expose(:line) { |note| note.diff_line.try(:new_line) if note.diff_note? }
      expose(:line_type) { |note| note.diff_line.try(:type) if note.diff_note? }
751
      expose :author, using: Entities::UserBasic
752
      expose :created_at
753 754
    end

755 756
    class CommitStatus < Grape::Entity
      expose :id, :sha, :ref, :status, :name, :target_url, :description,
757
             :created_at, :started_at, :finished_at, :allow_failure, :coverage
Kamil Trzcinski's avatar
Kamil Trzcinski committed
758
      expose :author, using: Entities::UserBasic
759 760
    end

761 762 763 764 765
    class PushEventPayload < Grape::Entity
      expose :commit_count, :action, :ref_type, :commit_from, :commit_to
      expose :ref, :commit_title
    end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
766
    class Event < Grape::Entity
767
      expose :project_id, :action_name
sue445's avatar
sue445 committed
768
      expose :target_id, :target_iid, :target_type, :author_id
769
      expose :target_title
770
      expose :created_at
771 772
      expose :note, using: Entities::Note, if: ->(event, options) { event.note? }
      expose :author, using: Entities::UserBasic, if: ->(event, options) { event.author }
773

774 775 776 777 778
      expose :push_event_payload,
        as: :push_data,
        using: PushEventPayload,
        if: -> (event, _) { event.push? }

779
      expose :author_username do |event, options|
780
        event.author&.username
781
      end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
782
    end
783

784
    class ProjectGroupLink < Grape::Entity
785
      expose :id, :project_id, :group_id, :group_access, :expires_at
786 787
    end

Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
788 789
    class Todo < Grape::Entity
      expose :id
790
      expose :project, using: Entities::BasicProjectDetails
Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
791
      expose :author, using: Entities::UserBasic
Robert Schilling's avatar
Robert Schilling committed
792
      expose :action_name
Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
793
      expose :target_type
794 795

      expose :target do |todo, options|
796
        Entities.const_get(todo.target_type).represent(todo.target, options)
Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
797 798 799 800
      end

      expose :target_url do |todo, options|
        target_type   = todo.target_type.underscore
801
        target_url    = "namespace_project_#{target_type}_url"
802
        target_anchor = "note_#{todo.note_id}" if todo.note_id?
Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
803

804 805
        Gitlab::Routing
          .url_helpers
806
          .public_send(target_url, todo.project.namespace, todo.project, todo.target, anchor: target_anchor) # rubocop:disable GitlabSecurity/PublicSend
Douglas Barbosa Alexandre's avatar
Douglas Barbosa Alexandre committed
807 808 809 810 811 812 813
      end

      expose :body
      expose :state
      expose :created_at
    end

814
    class NamespaceBasic < Grape::Entity
815
      expose :id, :name, :path, :kind, :full_path, :parent_id
816
    end
817

818
    class Namespace < NamespaceBasic
819 820 821 822 823 824
      expose :members_count_with_descendants, if: -> (namespace, opts) { expose_members_count_with_descendants?(namespace, opts) } do |namespace, _|
        namespace.users_with_descendants.count
      end

      def expose_members_count_with_descendants?(namespace, opts)
        namespace.kind == 'group' && Ability.allowed?(opts[:current_user], :admin_group, namespace)
825
      end
826
    end
827

828
    class MemberAccess < Grape::Entity
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
829
      expose :access_level
830
      expose :notification_level do |member, options|
831 832 833
        if member.notification_setting
          ::NotificationSetting.levels[member.notification_setting.level]
        end
834
      end
835 836
    end

837
    class ProjectAccess < MemberAccess
838 839
    end

840
    class GroupAccess < MemberAccess
841 842
    end

843 844 845 846 847 848 849 850 851 852 853 854 855 856 857
    class NotificationSetting < Grape::Entity
      expose :level
      expose :events, if: ->(notification_setting, _) { notification_setting.custom? } do
        ::NotificationSetting::EMAIL_EVENTS.each do |event|
          expose event
        end
      end
    end

    class GlobalNotificationSetting < NotificationSetting
      expose :notification_email do |notification_setting, options|
        notification_setting.user.notification_email
      end
    end

858 859
    class ProjectService < Grape::Entity
      expose :id, :title, :created_at, :updated_at, :active
860 861
      expose :push_events, :issues_events, :confidential_issues_events
      expose :merge_requests_events, :tag_push_events, :note_events
862
      expose :confidential_note_events, :pipeline_events, :wiki_page_events
863
      expose :job_events
864 865
      # Expose serialized properties
      expose :properties do |service, options|
Stan Hu's avatar
Stan Hu committed
866
        service.properties.slice(*service.api_field_names)
867 868 869
      end
    end

870 871 872
    class ProjectWithAccess < Project
      expose :permissions do
        expose :project_access, using: Entities::ProjectAccess do |project, options|
873 874
          if options[:project_members]
            options[:project_members].find { |member| member.source_id == project.id }
875 876
          else
            project.project_member(options[:current_user])
877
          end
878 879 880
        end

        expose :group_access, using: Entities::GroupAccess do |project, options|
881
          if project.group
882 883
            if options[:group_members]
              options[:group_members].find { |member| member.source_id == project.namespace_id }
884 885
            else
              project.group.group_member(options[:current_user])
886
            end
887
          end
888 889
        end
      end
890 891 892 893

      def self.preload_relation(projects_relation, options = {})
        relation = super(projects_relation, options)

894 895 896 897 898 899 900
        # MySQL doesn't support LIMIT inside an IN subquery
        if Gitlab::Database.mysql?
          project_ids = relation.pluck('projects.id')
          namespace_ids = relation.pluck(:namespace_id)
        else
          project_ids = relation.select('projects.id')
          namespace_ids = relation.select(:namespace_id)
901 902
        end

903 904 905 906 907 908 909 910 911
        options[:project_members] = options[:current_user]
          .project_members
          .where(source_id: project_ids)
          .preload(:source, user: [notification_settings: :source])

        options[:group_members] = options[:current_user]
          .group_members
          .where(source_id: namespace_ids)
          .preload(:source, user: [notification_settings: :source])
912 913 914

        relation
      end
915
    end
916

917
    class LabelBasic < Grape::Entity
Rares Sfirlogea's avatar
Rares Sfirlogea committed
918
      expose :id, :name, :color, :description
919 920 921
    end

    class Label < LabelBasic
922
      expose :open_issues_count do |label, options|
Francesco Coda Zabetta's avatar
Francesco Coda Zabetta committed
923 924
        label.open_issues_count(options[:current_user])
      end
925

Francesco Coda Zabetta's avatar
Francesco Coda Zabetta committed
926 927 928
      expose :closed_issues_count do |label, options|
        label.closed_issues_count(options[:current_user])
      end
929

Francesco Coda Zabetta's avatar
Francesco Coda Zabetta committed
930 931
      expose :open_merge_requests_count do |label, options|
        label.open_merge_requests_count(options[:current_user])
932 933
      end

934 935 936
      expose :priority do |label, options|
        label.priority(options[:project])
      end
937 938

      expose :subscribed do |label, options|
939
        label.subscribed?(options[:current_user], options[:project])
940
      end
941
    end
942

943 944 945 946 947 948 949 950
    class List < Grape::Entity
      expose :id
      expose :label, using: Entities::LabelBasic
      expose :position
    end

    class Board < Grape::Entity
      expose :id
Felipe Artur's avatar
Felipe Artur committed
951 952
      expose :project, using: Entities::BasicProjectDetails

953 954 955 956 957
      expose :lists, using: Entities::List do |board|
        board.lists.destroyable
      end
    end

958
    class Compare < Grape::Entity
959 960
      expose :commit, using: Entities::Commit do |compare, options|
        ::Commit.decorate(compare.commits, nil).last
961
      end
962

963 964
      expose :commits, using: Entities::Commit do |compare, options|
        ::Commit.decorate(compare.commits, nil)
965
      end
966

967
      expose :diffs, using: Entities::Diff do |compare, options|
Douwe Maan's avatar
Douwe Maan committed
968
        compare.diffs(limits: false).to_a
969
      end
970 971

      expose :compare_timeout do |compare, options|
972
        compare.diffs.overflow?
973 974 975
      end

      expose :same, as: :compare_same_ref
976
    end
977 978 979 980

    class Contributor < Grape::Entity
      expose :name, :email, :commits, :additions, :deletions
    end
981 982 983 984

    class BroadcastMessage < Grape::Entity
      expose :message, :starts_at, :ends_at, :color, :font
    end
985 986

    class ApplicationSetting < Grape::Entity
987 988 989 990 991 992 993 994 995 996
      def self.exposed_attributes
        attributes = ::ApplicationSettingsHelper.visible_attributes
        attributes.delete(:performance_bar_allowed_group_path)
        attributes.delete(:performance_bar_enabled)

        attributes
      end

      expose :id, :performance_bar_allowed_group_id
      expose(*exposed_attributes)
997 998 999 1000 1001 1002
      expose(:restricted_visibility_levels) do |setting, _options|
        setting.restricted_visibility_levels.map { |level| Gitlab::VisibilityLevel.string_level(level) }
      end
      expose(:default_project_visibility) { |setting, _options| Gitlab::VisibilityLevel.string_level(setting.default_project_visibility) }
      expose(:default_snippet_visibility) { |setting, _options| Gitlab::VisibilityLevel.string_level(setting.default_snippet_visibility) }
      expose(:default_group_visibility) { |setting, _options| Gitlab::VisibilityLevel.string_level(setting.default_group_visibility) }
1003 1004 1005 1006

      # support legacy names, can be removed in v5
      expose :password_authentication_enabled_for_web, as: :password_authentication_enabled
      expose :password_authentication_enabled_for_web, as: :signin_enabled
1007
    end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
1008 1009

    class Release < Grape::Entity
1010 1011
      expose :tag, as: :tag_name
      expose :description
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
1012
    end
1013

Robert Schilling's avatar
Robert Schilling committed
1014
    class Tag < Grape::Entity
1015
      expose :name, :message, :target
1016

1017
      expose :commit, using: Entities::Commit do |repo_tag, options|
1018
        options[:project].repository.commit(repo_tag.dereferenced_target)
1019 1020
      end

1021 1022
      expose :release, using: Entities::Release do |repo_tag, options|
        options[:project].releases.find_by(tag: repo_tag.name)
1023 1024
      end
    end
Kamil Trzcinski's avatar
Kamil Trzcinski committed
1025

1026
    class Runner < Grape::Entity
1027 1028
      expose :id
      expose :description
1029
      expose :ip_address
1030
      expose :active
1031
      expose :instance_type?, as: :is_shared
1032
      expose :name
1033
      expose :online?, as: :online
1034
      expose :status
1035 1036
    end

1037 1038
    class RunnerDetails < Runner
      expose :tag_list
1039
      expose :run_untagged
1040
      expose :locked
1041
      expose :maximum_timeout
Shinya Maeda's avatar
Shinya Maeda committed
1042
      expose :access_level
1043
      expose :version, :revision, :platform, :architecture
1044
      expose :contacted_at
1045
      expose :token, if: lambda { |runner, options| options[:current_user].admin? || !runner.instance_type? }
1046
      expose :projects, with: Entities::BasicProjectDetails do |runner, options|
1047
        if options[:current_user].admin?
1048 1049
          runner.projects
        else
1050
          options[:current_user].authorized_projects.where(id: runner.projects)
1051 1052
        end
      end
1053 1054 1055 1056 1057 1058 1059
      expose :groups, with: Entities::BasicGroupDetails do |runner, options|
        if options[:current_user].admin?
          runner.groups
        else
          options[:current_user].authorized_groups.where(id: runner.groups)
        end
      end
1060 1061
    end

1062 1063 1064 1065
    class RunnerRegistrationDetails < Grape::Entity
      expose :id, :token
    end

1066
    class JobArtifactFile < Grape::Entity
1067 1068 1069
      expose :filename, :size
    end

Tomasz Maczukin's avatar
Tomasz Maczukin committed
1070
    class JobBasic < Grape::Entity
1071
      expose :id, :status, :stage, :name, :ref, :tag, :coverage
1072
      expose :created_at, :started_at, :finished_at
1073
      expose :duration
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1074
      expose :user, with: User
1075
      expose :commit, with: Commit
1076
      expose :pipeline, with: PipelineBasic
1077 1078 1079 1080

      expose :web_url do |job, _options|
        Gitlab::Routing.url_helpers.project_job_url(job.project, job)
      end
1081
    end
1082

Tomasz Maczukin's avatar
Tomasz Maczukin committed
1083 1084 1085
    class Job < JobBasic
      expose :artifacts_file, using: JobArtifactFile, if: -> (job, opts) { job.artifacts? }
      expose :runner, with: Runner
1086
      expose :artifacts_expire_at
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1087 1088 1089
    end

    class JobBasicWithProject < JobBasic
1090 1091 1092
      expose :project, with: ProjectIdentity
    end

1093
    class Trigger < Grape::Entity
1094
      expose :id
1095
      expose :token, :description
1096
      expose :created_at, :updated_at, :last_used
1097
      expose :owner, using: Entities::UserBasic
1098
    end
1099

1100
    class Variable < Grape::Entity
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1101
      expose :key, :value
Shinya Maeda's avatar
Shinya Maeda committed
1102
      expose :protected?, as: :protected, if: -> (entity, _) { entity.respond_to?(:protected?) }
1103
    end
1104

1105 1106
    class Pipeline < PipelineBasic
      expose :before_sha, :tag, :yaml_errors
1107 1108 1109 1110

      expose :user, with: Entities::UserBasic
      expose :created_at, :updated_at, :started_at, :finished_at, :committed_at
      expose :duration
1111
      expose :coverage
1112 1113
    end

1114 1115 1116
    class PipelineSchedule < Grape::Entity
      expose :id
      expose :description, :ref, :cron, :cron_timezone, :next_run_at, :active
1117
      expose :created_at, :updated_at
1118 1119 1120
      expose :owner, using: Entities::UserBasic
    end

Shinya Maeda's avatar
Shinya Maeda committed
1121 1122
    class PipelineScheduleDetails < PipelineSchedule
      expose :last_pipeline, using: Entities::PipelineBasic
1123
      expose :variables, using: Entities::Variable
Shinya Maeda's avatar
Shinya Maeda committed
1124 1125
    end

1126
    class EnvironmentBasic < Grape::Entity
Nick Thomas's avatar
Nick Thomas committed
1127
      expose :id, :name, :slug, :external_url
1128 1129
    end

1130
    class Environment < EnvironmentBasic
1131
      expose :project, using: Entities::BasicProjectDetails
Z.J. van de Weg's avatar
Z.J. van de Weg committed
1132 1133 1134 1135
    end

    class Deployment < Grape::Entity
      expose :id, :iid, :ref, :sha, :created_at
1136 1137
      expose :user,        using: Entities::UserBasic
      expose :environment, using: Entities::EnvironmentBasic
1138
      expose :deployable,  using: Entities::Job
1139 1140
    end

1141
    class License < Grape::Entity
1142 1143
      expose :key, :name, :nickname
      expose :featured, as: :popular
1144 1145 1146
      expose :url, as: :html_url
      expose(:source_url) { |license| license.meta['source'] }
      expose(:description) { |license| license.meta['description'] }
1147 1148 1149
      expose(:conditions) { |license| license.meta['conditions'] }
      expose(:permissions) { |license| license.meta['permissions'] }
      expose(:limitations) { |license| license.meta['limitations'] }
1150 1151
      expose :content
    end
1152

1153
    class TemplatesList < Grape::Entity
1154 1155 1156
      expose :name
    end

1157
    class Template < Grape::Entity
1158 1159
      expose :name, :content
    end
1160 1161 1162 1163 1164

    class BroadcastMessage < Grape::Entity
      expose :id, :message, :starts_at, :ends_at, :color, :font
      expose :active?, as: :active
    end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1165

1166
    class PersonalAccessToken < Grape::Entity
1167 1168 1169 1170 1171 1172 1173
      expose :id, :name, :revoked, :created_at, :scopes
      expose :active?, as: :active
      expose :expires_at do |personal_access_token|
        personal_access_token.expires_at ? personal_access_token.expires_at.strftime("%Y-%m-%d") : nil
      end
    end

1174
    class PersonalAccessTokenWithToken < PersonalAccessToken
1175 1176
      expose :token
    end
1177 1178 1179 1180

    class ImpersonationToken < PersonalAccessTokenWithToken
      expose :impersonation
    end
1181

1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203
    class FeatureGate < Grape::Entity
      expose :key
      expose :value
    end

    class Feature < Grape::Entity
      expose :name
      expose :state
      expose :gates, using: FeatureGate do |model|
        model.gates.map do |gate|
          value = model.gate_values[gate.key]

          # By default all gate values are populated. Only show relevant ones.
          if (value.is_a?(Integer) && value.zero?) || (value.is_a?(Set) && value.empty?)
            next
          end

          { key: gate.key, value: value }
        end.compact
      end
    end

1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219
    module JobRequest
      class JobInfo < Grape::Entity
        expose :name, :stage
        expose :project_id, :project_name
      end

      class GitInfo < Grape::Entity
        expose :repo_url, :ref, :sha, :before_sha
        expose :ref_type do |model|
          if model.tag
            'tag'
          else
            'branch'
          end
        end
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1220

1221
      class RunnerInfo < Grape::Entity
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1222
        expose :metadata_timeout, as: :timeout
1223
        expose :runner_session_url
1224
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1225

1226
      class Step < Grape::Entity
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1227
        expose :name, :script, :timeout, :when, :allow_failure
1228
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1229

1230
      class Image < Grape::Entity
1231 1232 1233
        expose :name, :entrypoint
      end

1234
      class Service < Image
1235
        expose :alias, :command
1236
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1237

1238
      class Artifacts < Grape::Entity
1239 1240 1241 1242 1243 1244 1245
        expose :name
        expose :untracked
        expose :paths
        expose :when
        expose :expire_in
        expose :artifact_type
        expose :artifact_format
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1246 1247
      end

1248
      class Cache < Grape::Entity
1249
        expose :key, :untracked, :paths, :policy
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1250 1251
      end

1252 1253 1254
      class Credentials < Grape::Entity
        expose :type, :url, :username, :password
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1255

1256
      class Dependency < Grape::Entity
1257
        expose :id, :name, :token
1258
        expose :artifacts_file, using: JobArtifactFile, if: ->(job, _) { job.artifacts? }
1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280
      end

      class Response < Grape::Entity
        expose :id
        expose :token
        expose :allow_git_fetch

        expose :job_info, using: JobInfo do |model|
          model
        end

        expose :git_info, using: GitInfo do |model|
          model
        end

        expose :runner_info, using: RunnerInfo do |model|
          model
        end

        expose :variables
        expose :steps, using: Step
        expose :image, using: Image
1281
        expose :services, using: Service
1282 1283 1284
        expose :artifacts, using: Artifacts
        expose :cache, using: Cache
        expose :credentials, using: Credentials
1285
        expose :dependencies, using: Dependency
1286
        expose :features
1287
      end
Tomasz Maczukin's avatar
Tomasz Maczukin committed
1288
    end
1289 1290 1291 1292

    class UserAgentDetail < Grape::Entity
      expose :user_agent
      expose :ip_address
1293
      expose :submitted, as: :akismet_submitted
1294
    end
1295 1296 1297 1298 1299 1300

    class RepositoryStorageHealth < Grape::Entity
      expose :storage_name
      expose :failing_on_hosts
      expose :total_failures
    end
1301 1302 1303 1304 1305

    class CustomAttribute < Grape::Entity
      expose :key
      expose :value
    end
1306

1307 1308 1309 1310 1311
    class PagesDomainCertificateExpiration < Grape::Entity
      expose :expired?, as: :expired
      expose :expiration
    end

1312 1313 1314 1315 1316 1317 1318
    class PagesDomainCertificate < Grape::Entity
      expose :subject
      expose :expired?, as: :expired
      expose :certificate
      expose :certificate_text
    end

1319 1320 1321
    class PagesDomainBasic < Grape::Entity
      expose :domain
      expose :url
1322
      expose :project_id
1323 1324 1325 1326
      expose :verified?, as: :verified
      expose :verification_code, as: :verification_code
      expose :enabled_until

1327 1328 1329 1330 1331 1332 1333 1334
      expose :certificate,
        as: :certificate_expiration,
        if: ->(pages_domain, _) { pages_domain.certificate? },
        using: PagesDomainCertificateExpiration do |pages_domain|
        pages_domain
      end
    end

1335 1336 1337
    class PagesDomain < Grape::Entity
      expose :domain
      expose :url
1338 1339 1340 1341
      expose :verified?, as: :verified
      expose :verification_code, as: :verification_code
      expose :enabled_until

1342
      expose :certificate,
1343 1344
        if: ->(pages_domain, _) { pages_domain.certificate? },
        using: PagesDomainCertificate do |pages_domain|
1345 1346 1347
        pages_domain
      end
    end
1348 1349 1350 1351 1352

    class Application < Grape::Entity
      expose :uid, as: :application_id
      expose :redirect_uri, as: :callback_url
    end
1353 1354 1355 1356 1357

    # Use with care, this exposes the secret
    class ApplicationWithSecret < Application
      expose :secret
    end
Jarka Kadlecová's avatar
Jarka Kadlecová committed
1358 1359 1360 1361 1362 1363 1364 1365

    class Blob < Grape::Entity
      expose :basename
      expose :data
      expose :filename
      expose :id
      expose :ref
      expose :startline
1366
      expose :project_id
Jarka Kadlecová's avatar
Jarka Kadlecová committed
1367
    end
1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385

    class BasicBadgeDetails < Grape::Entity
      expose :link_url
      expose :image_url
      expose :rendered_link_url do |badge, options|
        badge.rendered_link_url(options.fetch(:project, nil))
      end
      expose :rendered_image_url do |badge, options|
        badge.rendered_image_url(options.fetch(:project, nil))
      end
    end

    class Badge < BasicBadgeDetails
      expose :id
      expose :kind do |badge|
        badge.type == 'ProjectBadge' ? 'project' : 'group'
      end
    end
Nihad Abbasov's avatar
Nihad Abbasov committed
1386 1387
  end
end