Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Include thread-safety rubocop and refactor minimum rubocop.gemfile #703

Merged
merged 4 commits into from
Mar 21, 2025

Conversation

alexpark20
Copy link
Contributor

@alexpark20 alexpark20 commented Mar 6, 2025

This PR contains mainly these two changes:


Introduce thread_safety rubocop

Include thread_safety rubocop as part of Shopify style rubocop. This PR only enables the small subset of the rules as a starting point. In addition, not part of thread_safety rubocop, but also enabled Style/MutableConstant to prevent constant being mutated concurrently unintentionally.

Note that this is part of Falcon on SFR effort (vault project here).


Refactor minimum_rubocop.gemfile

We have updated minimum_rubocop.gemfile so that we could handle the version dependency in tests better .

@github-actions github-actions bot added the config change Changes the Rubocop config by enabling, disabling, or reconfiguring one or many cops label Mar 6, 2025
@alexpark20 alexpark20 force-pushed the thread-safety-rubocop branch 2 times, most recently from 3999c6e to 5ab601c Compare March 6, 2025 16:43
ThreadSafety/NewThread:
Description: Avoid starting new threads. Let a framework like Sidekiq handle the
threads.
Enabled: true
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should not be enabled

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right, we are not enabling it on SFR too. I will disable it 👍

Enabled: true
ThreadSafety/MutableClassInstanceVariable:
Description: Do not assign mutable objects to class instance variables.
Enabled: true
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This cop is disallowing:

class FooTest < ActiveSupport::TestCase
  setup do
    @something = { a: "A" }
  end
end

Which doesn't have any thread safety issue.

We need to fix that in order to enable it.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will not enable it yet. Good check, lets socialize how we can fix it 👍

@alexpark20 alexpark20 force-pushed the thread-safety-rubocop branch 3 times, most recently from 4c2b934 to 3cd22cb Compare March 7, 2025 15:29
@alexpark20 alexpark20 requested a review from rafaelfranca March 7, 2025 15:50
@alexpark20 alexpark20 marked this pull request as ready for review March 10, 2025 15:13
@alexpark20 alexpark20 requested a review from a team as a code owner March 10, 2025 15:13
@sambostock sambostock force-pushed the thread-safety-rubocop branch from c04fc7a to a69842b Compare March 13, 2025 03:10
@sambostock
Copy link
Contributor

The unknown cop issue is fixed, but now we have the issue that rubocop-thread_safety imposes a minimum version requirement for rubocop which is higher than our current minimum (which we test against, so it breaks CI because the dependencies are unsatisfiable).

@sambostock
Copy link
Contributor

Simply bumping it doesn't work either, as it breaks some assumptions made by the tooling that ensures version compatibility. We'll have to fix that too.

@viralpraxis
Copy link

The unknown cop issue is fixed, but now we have the issue that rubocop-thread_safety imposes a minimum version requirement for rubocop which is higher than our current minimum (which we test against, so it breaks CI because the dependencies are unsatisfiable).

Would backporting the patch to 0.6 help? rubocop-thread_safety-0.6 requires rubocop >= 1.48.1

@alexpark20
Copy link
Contributor Author

The unknown cop issue is fixed, but now we have the issue that rubocop-thread_safety imposes a minimum version requirement for rubocop which is higher than our current minimum (which we test against, so it breaks CI because the dependencies are unsatisfiable).

Would backporting the patch to 0.6 help? rubocop-thread_safety-0.6 requires rubocop >= 1.48.1

We want to have the fix in v0.7.2 😞

Simply bumping it doesn't work either, as it breaks some assumptions made by the tooling that ensures version compatibility. We'll have to fix that too.

What are those assumptions?

@viralpraxis
Copy link

We want to have the fix in v0.7.2 😞

Sorry, I didnt quite get it since I don't have all the context. The patch was released in v0.7.1 and is also present in v0.7.2 (which was released today to fix an unreleated issue).

@sambostock
Copy link
Contributor

sambostock commented Mar 13, 2025

Both options work. The issue in CI is that we have a check that is along the lines of

Extract the Rubocop version from the gemspec, and run CI against that version.

The check currently extracts ~> x.y and runs CI with Rubocop version x.y, but that breaks because rubocop-thread_safety depends on ~> 1.72, >= 1.72.1, so if we try to just depend on ~> 1.72, CI fails because = 1.72 is incompatible with >= 1.72.1.

The backport idea would be nice because it would mean we don't have to increase the dependency requirement, but realistically that's not a blocker. I just have to fix CI to be smart enough to handle this case.

@alexpark20 alexpark20 requested a review from sambostock March 14, 2025 13:50
@alexpark20
Copy link
Contributor Author

@sambostock, I updated minimum_rubocop.gemfile, LMK what you think :)

requirements = version_match.scan(/\d+\.\d+/)
minimum_version = requirements.max

gem "rubocop", "~> #{minimum_version}.0"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We don't want to use ~>, because that doesn't guarantee that we're using the exact minimum version. For example, this results in ~> 1.72.0 which would likely result in us installing 1.72.2, whereas the actual minimum version we should be testing against is 1.71.1.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've switched to an approach of extracting the rubocop requirements from the gemspec, ensuring they're all using ~> or >=, and then grabbing the max version and using that as the exact version.

The other alternative I thought of was to get all the published versions from Rubygems, and then Array#bsearch through them and use satisfied_by? to find the first one that satisfies our requirements (and maybe sub-dependency requirements too), and use that. However, that approach would be more complex, and the other approach is probably sufficient.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, extracting ~> and >= and grab the max version makes it more straightforward 👍

@sambostock sambostock force-pushed the thread-safety-rubocop branch from 0173b51 to 87b827b Compare March 20, 2025 19:13
@sambostock
Copy link
Contributor

We should rebase/squash before merging if we're okay with these rules and the change to minimum_rubocop.gemfile.

.load("rubocop-shopify.gemspec")
.dependencies.find { |d| d.name == "rubocop" }
.requirement.requirements
.partition { |type, _version| supported_requirement_types.include?(type) }
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Clever 👏

@alexpark20 alexpark20 force-pushed the thread-safety-rubocop branch from 87b827b to efd84b9 Compare March 20, 2025 20:07
@alexpark20 alexpark20 changed the title Include thread-safety rubocop Include thread-safety rubocop and refactor minimum rubocop.gemfile Mar 20, 2025
alexpark20 and others added 4 commits March 20, 2025 16:44
This allows us to combine `~>` and `>=` `rubocop` version requirements.
This plugin helps ensure thread safety. As it uses the new RuboCop
plugin API, we must bump the minimum RuboCop version, and as such we can
remove many of our version checks.
@sambostock sambostock force-pushed the thread-safety-rubocop branch from efd84b9 to a8dd1f0 Compare March 20, 2025 20:54
@alexpark20 alexpark20 enabled auto-merge March 21, 2025 15:38
@alexpark20 alexpark20 merged commit 175c309 into main Mar 21, 2025
20 checks passed
@alexpark20 alexpark20 deleted the thread-safety-rubocop branch March 21, 2025 15:47
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
config change Changes the Rubocop config by enabling, disabling, or reconfiguring one or many cops
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants