Commit 2e7f0316 authored by GitLab Bot's avatar GitLab Bot

Automatic merge of gitlab-org/gitlab-ce master

parents 5376e281 d8bbaae6
...@@ -6,4 +6,12 @@ class ApplicationRecord < ActiveRecord::Base ...@@ -6,4 +6,12 @@ class ApplicationRecord < ActiveRecord::Base
def self.id_in(ids) def self.id_in(ids)
where(id: ids) where(id: ids)
end end
def self.safe_find_or_create_by(*args)
transaction(requires_new: true) do
find_or_create_by(*args)
end
rescue ActiveRecord::RecordNotUnique
retry
end
end end
...@@ -256,32 +256,12 @@ violation, for example. ...@@ -256,32 +256,12 @@ violation, for example.
Using transactions does not solve this problem. Using transactions does not solve this problem.
The following pattern should be used to avoid the problem: To solve this we've added the `ApplicationRecord.safe_find_or_create_by`.
```ruby This method can be used just as you would the normal
Project.transaction do `find_or_create_by` but it wraps the call in a *new* transaction and
begin retries if it were to fail because of an
User.find_or_create_by(username: "foo") `ActiveRecord::RecordNotUnique` error.
rescue ActiveRecord::RecordNotUnique
retry
end
end
```
If the above block is run inside a transaction and hits the race
condition, the transaction is aborted and we cannot simply retry (any
further queries inside the aborted transaction are going to fail). We
can employ [nested transactions](http://api.rubyonrails.org/classes/ActiveRecord/Transactions/ClassMethods.html#module-ActiveRecord::Transactions::ClassMethods-label-Nested+transactions)
here to only rollback the "inner transaction". Note that `requires_new: true` is required here.
```ruby To be able to use this method, make sure the model you want to use
Project.transaction do this on inherits from `ApplicationRecord`.
begin
User.transaction(requires_new: true) do
User.find_or_create_by(username: "foo")
end
rescue ActiveRecord::RecordNotUnique
retry
end
end
```
...@@ -31,12 +31,26 @@ The optional settings, custom domain, DNS records, and SSL/TLS certificates, are ...@@ -31,12 +31,26 @@ The optional settings, custom domain, DNS records, and SSL/TLS certificates, are
## Project ## Project
Your GitLab Pages project is a regular project created the Your GitLab Pages project is a regular project created the
same way you do for the other ones. To get started with GitLab Pages, you have two ways: same way you do for the other ones. To get started with GitLab Pages, you have three ways:
- Use one of the popular templates already in the app,
- Fork one of the templates from Page Examples, or - Fork one of the templates from Page Examples, or
- Create a new project from scratch - Create a new project from scratch
Let's go over both options. Let's go over each option.
### Use one of the popular Pages templates bundled with GitLab
> [Introduced](https://gitlab.com/gitlab-org/gitlab-ce/issues/47857)
in GitLab 11.8.
The simplest way to create a GitLab Pages site is to use one of the most
popular templates, which come already bundled and ready to go. To use one
of these templates:
1. From the top navigation, click the **+** button and select **New project**
1. Select **Create from Template**
1. Choose one of the templates starting with **Pages**
### Fork a project to get started from ### Fork a project to get started from
......
...@@ -91,8 +91,8 @@ site under the HTTPS protocol. ...@@ -91,8 +91,8 @@ site under the HTTPS protocol.
## Getting started ## Getting started
To get started with GitLab Pages, you can either [create a project from scratch](getting_started_part_two.md#create-a-project-from-scratch) To get started with GitLab Pages, you can either [create a project from scratch](getting_started_part_two.md#create-a-project-from-scratch),
or quickly start from copying an existing example project, as follows: use a [bundled template](getting_started_part_two.md#use-one-of-the-popular-pages-templates-bundled-with-gitlab), or copy any of our existing example projects:
1. Choose an [example project](https://gitlab.com/pages) to [fork](../../../gitlab-basics/fork-project.md#how-to-fork-a-project): 1. Choose an [example project](https://gitlab.com/pages) to [fork](../../../gitlab-basics/fork-project.md#how-to-fork-a-project):
by forking a project, you create a copy of the codebase you're forking from to start from a template instead of starting from scratch. by forking a project, you create a copy of the codebase you're forking from to start from a template instead of starting from scratch.
......
...@@ -10,4 +10,14 @@ describe ApplicationRecord do ...@@ -10,4 +10,14 @@ describe ApplicationRecord do
expect(User.id_in(records.last(2).map(&:id))).to eq(records.last(2)) expect(User.id_in(records.last(2).map(&:id))).to eq(records.last(2))
end end
end end
describe '#safe_find_or_create_by' do
it 'creates the user avoiding race conditions' do
expect(Suggestion).to receive(:find_or_create_by).and_raise(ActiveRecord::RecordNotUnique)
allow(Suggestion).to receive(:find_or_create_by).and_call_original
expect { Suggestion.safe_find_or_create_by(build(:suggestion).attributes) }
.to change { Suggestion.count }.by(1)
end
end
end end
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