158 lines
5.2 KiB
Ruby
158 lines
5.2 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Repositories
|
|
# A service class for generating a changelog section.
|
|
class ChangelogService
|
|
DEFAULT_TRAILER = 'Changelog'
|
|
DEFAULT_FILE = 'CHANGELOG.md'
|
|
|
|
# The maximum number of commits allowed to fetch in `from` and `to` range.
|
|
#
|
|
# This value is arbitrarily chosen. Increasing it means more Gitaly calls
|
|
# and more presure on Gitaly services.
|
|
#
|
|
# This number is 3x of the average number of commits per GitLab releases.
|
|
# Some examples for GitLab's own releases:
|
|
#
|
|
# * 13.6.0: 4636 commits
|
|
# * 13.5.0: 5912 commits
|
|
# * 13.4.0: 5541 commits
|
|
COMMITS_LIMIT = 15_000
|
|
|
|
# The `project` specifies the `Project` to generate the changelog section
|
|
# for.
|
|
#
|
|
# The `user` argument specifies a `User` to use for committing the changes
|
|
# to the Git repository.
|
|
#
|
|
# The `version` arguments must be a version `String` using semantic
|
|
# versioning as the format.
|
|
#
|
|
# The arguments `from` and `to` must specify a Git ref or SHA to use for
|
|
# fetching the commits to include in the changelog. The SHA/ref set in the
|
|
# `from` argument isn't included in the list.
|
|
#
|
|
# The `date` argument specifies the date of the release, and defaults to the
|
|
# current time/date.
|
|
#
|
|
# The `branch` argument specifies the branch to commit the changes to. The
|
|
# branch must already exist.
|
|
#
|
|
# The `trailer` argument is the Git trailer to use for determining what
|
|
# commits to include in the changelog.
|
|
#
|
|
# The `config_file` arguments specifies the path to the configuration file as
|
|
# stored in the project's Git repository.
|
|
#
|
|
# The `file` arguments specifies the name/path of the file to commit the
|
|
# changes to. If the file doesn't exist, it's created automatically.
|
|
#
|
|
# The `message` argument specifies the commit message to use when committing
|
|
# the changelog changes.
|
|
#
|
|
# rubocop: disable Metrics/ParameterLists
|
|
def initialize(
|
|
project,
|
|
user,
|
|
version:,
|
|
branch: project.default_branch_or_main,
|
|
from: nil,
|
|
to: branch,
|
|
date: DateTime.now,
|
|
trailer: DEFAULT_TRAILER,
|
|
config_file: Gitlab::Changelog::Config::DEFAULT_FILE_PATH,
|
|
file: DEFAULT_FILE,
|
|
message: "Add changelog for version #{version}"
|
|
)
|
|
@project = project
|
|
@user = user
|
|
@version = version
|
|
@from = from
|
|
@to = to
|
|
@date = date
|
|
@branch = branch
|
|
@trailer = trailer
|
|
@config_file = config_file
|
|
@file = file
|
|
@message = message
|
|
end
|
|
# rubocop: enable Metrics/ParameterLists
|
|
|
|
def execute(commit_to_changelog: true)
|
|
config = Gitlab::Changelog::Config.from_git(@project, @user, @config_file)
|
|
from = start_of_commit_range(config)
|
|
|
|
# For every entry we want to only include the merge request that
|
|
# originally introduced the commit, which is the oldest merge request that
|
|
# contains the commit. We fetch there merge requests in batches, reducing
|
|
# the number of SQL queries needed to get this data.
|
|
mrs_finder = MergeRequests::OldestPerCommitFinder.new(@project)
|
|
release = Gitlab::Changelog::Release
|
|
.new(version: @version, date: @date, config: config)
|
|
|
|
commits =
|
|
ChangelogCommitsFinder.new(project: @project, from: from, to: @to)
|
|
|
|
verify_commit_range!(from, @to)
|
|
|
|
commits.each_page(@trailer) do |page|
|
|
mrs = mrs_finder.execute(page)
|
|
|
|
# Preload the authors. This ensures we only need a single SQL query per
|
|
# batch of commits, instead of needing a query for every commit.
|
|
page.each(&:lazy_author)
|
|
|
|
# Preload author permissions
|
|
@project.team.max_member_access_for_user_ids(page.map(&:author).compact.map(&:id))
|
|
|
|
page.each do |commit|
|
|
release.add_entry(
|
|
title: commit.title,
|
|
commit: commit,
|
|
category: commit.trailers.fetch(@trailer),
|
|
author: commit.author,
|
|
merge_request: mrs[commit.id]
|
|
)
|
|
end
|
|
end
|
|
|
|
if commit_to_changelog
|
|
Gitlab::Changelog::Committer
|
|
.new(@project, @user)
|
|
.commit(release: release, file: @file, branch: @branch, message: @message)
|
|
else
|
|
Gitlab::Changelog::Generator.new.add(release)
|
|
end
|
|
end
|
|
|
|
def start_of_commit_range(config)
|
|
return @from if @from
|
|
|
|
finder = ChangelogTagFinder.new(@project, regex: config.tag_regex)
|
|
|
|
if (prev_tag = finder.execute(@version))
|
|
return prev_tag.target_commit.id
|
|
end
|
|
|
|
raise(
|
|
Gitlab::Changelog::Error,
|
|
'The commit start range is unspecified, and no previous tag ' \
|
|
'could be found to use instead'
|
|
)
|
|
end
|
|
|
|
def verify_commit_range!(from, to)
|
|
return unless Feature.enabled?(:changelog_commits_limitation, @project)
|
|
|
|
commits = @project.repository.commits_by(oids: [from, to])
|
|
|
|
raise Gitlab::Changelog::Error, "Invalid or not found commit value in the given range" unless commits.count == 2
|
|
|
|
_, commits_count = @project.repository.diverging_commit_count(from, to)
|
|
|
|
if commits_count > COMMITS_LIMIT
|
|
raise Gitlab::Changelog::Error, "The commits range exceeds #{COMMITS_LIMIT} elements."
|
|
end
|
|
end
|
|
end
|
|
end
|