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

Refactor FXIOS-10205 [Swiftlint] Enable implicitly_unwrapped_optional rule but keep it disabled for test files using nested Swiftlint configurations #24031

Conversation

tonell-m
Copy link
Contributor

@tonell-m tonell-m commented Jan 8, 2025

📜 Tickets

Jira ticket
Github issue

💡 Description

Now that all implicitly_unwrapped_optional violations in source files are resolved (still waiting on #23809 and #23880 to be merged), enable the rule in the Swiftlint configuration.

Since most of the test files use implicitly unwrapped optional properties that are initialized in setUp and reset in tearDown, we would like not to enable this rule for test files. An approach is to use Swiftlint nested configurations by adding additional .swiftlint.yml files that can alter the global configuration for subfolders (in this case, disable implicitly_unwrapped_optional for Tests folders).

Nested configurations cannot be used together with the --config option to point to the Swiftlint config file, so the workaround is to navigate to the directory containing the config before running swiftlint in the build phase.

📝 Checklist

You have to check all boxes before merging

  • Filled in the above information (tickets numbers and description of your work)
  • Updated the PR name to follow our PR naming guidelines
  • Wrote unit tests and/or ensured the tests suite is passing
  • When working on UI, I checked and implemented accessibility (minimum Dynamic Text and VoiceOver)
  • If needed, I updated documentation / comments for complex code and public methods
  • If needed, added a backport comment (example @Mergifyio backport release/v120)

@tonell-m tonell-m requested a review from a team as a code owner January 8, 2025 07:37
@tonell-m tonell-m requested a review from mdotb-moz January 8, 2025 07:37
Copy link
Collaborator

@FilippoZazzeroni FilippoZazzeroni left a comment

Choose a reason for hiding this comment

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

It looks great to me this is a really nice catch 🎉 🎉 . tagging also @OrlaM and @adudenamedruby to get extra eye on this modification since it impact a bit the project structure.
If this modification lands in i'll add it to our wiki.

@@ -15861,7 +15861,7 @@
);
runOnlyForDeploymentPostprocessing = 0;
shellPath = /bin/sh;
shellScript = "if [[ \"$(uname -m)\" == arm64 ]]; then\n export PATH=\"/opt/homebrew/bin:$PATH\"\nfi\n\nMODIFIED_FILES=$(git diff --name-only --diff-filter=ACM | grep -e '\\.swift$')\n\nif which swiftlint > /dev/null; then\n for file in $MODIFIED_FILES; do\n swiftlint lint \"${SRCROOT}/../${file}\" --config \"${SRCROOT}/../.swiftlint.yml\"\n done\nelse\n echo \"warning: SwiftLint not installed, download from https://github.com/realm/SwiftLint\" \nfi\n";
shellScript = "if [[ \"$(uname -m)\" == arm64 ]]; then\n export PATH=\"/opt/homebrew/bin:$PATH\"\nfi\n\nMODIFIED_FILES=$(git diff --name-only --diff-filter=ACM | grep -e '\\.swift$')\n\n# Location of the folder where the root swiftlint config is located.\n# Swiftlint commands will be ran from this location, allowing us to make use\n# of swiftlint's nested configurations to tweak rules for certain source\n# sets (e.g. test files).\n# Nested configuration can not work together with --config option, so we need\n# to run the swiftlint command from the directory containing the root\n# configuration file.\n# https://github.com/realm/SwiftLint#nested-configurations\nSWIFTLINT_ROOT=\"${SRCROOT}/../\"\n\nif which swiftlint > /dev/null; then\n for file in $MODIFIED_FILES; do\n ( cd ${SWIFTLINT_ROOT} && swiftlint lint \"${SRCROOT}/../${file}\" )\n done\nelse\n echo \"warning: SwiftLint not installed, download from https://github.com/realm/SwiftLint\" \nfi\n";
Copy link
Collaborator

@FilippoZazzeroni FilippoZazzeroni Jan 8, 2025

Choose a reason for hiding this comment

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

Just a small change the rest looks great 🎉 . i'd have:

// move out of the for loop
cd $SWIFTLINT_ROOT
for loop {}

I'd also reduce the comment a bit and add it directly to the cd $SWIFTLINT_ROOT saying that this instruction is needed here in order to have nested swiftlint configurations.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Good point, done!

@FilippoZazzeroni FilippoZazzeroni requested review from OrlaM and adudenamedruby and removed request for mdotb-moz January 8, 2025 10:56
# nested configuration takes over.
# https://github.com/realm/SwiftLint#nested-configurations.

disabled_rules:
Copy link
Contributor

Choose a reason for hiding this comment

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

This looks great! Thanks for the work on this issue 🙌 I'll let my colleagues do the final review!

Copy link
Collaborator

@FilippoZazzeroni FilippoZazzeroni left a comment

Choose a reason for hiding this comment

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

It looks great to me. Thanks again for your contribution 🎉 🎉

@tonell-m
Copy link
Contributor Author

tonell-m commented Jan 9, 2025

Hi @FilippoZazzeroni, from the looks of the Bitrise output there are still some violations in SampleBrowser and SampleComponentLibraryApp, I'll open some PRs for those.

However, it seems that the Bitrise Swiftlint phase script still uses the --config option so it also lints the test files - I'm not sure I can fix this myself

@adudenamedruby adudenamedruby removed their request for review January 9, 2025 14:32
@adudenamedruby
Copy link
Contributor

LGTM :). I'll let @FilippoZazzeroni push this through

@FilippoZazzeroni
Copy link
Collaborator

Hi @tonell-m sorry for the late response, i've been going deeper into your solution and i see some nits that i'd like to understand better.
Right now we have two main ways of running swiftlint and those are via the run script and via CLI in this case via Bitrise.
The problem i see right now is that by running swiftlint even with the --config via the run script on firefox, it can create warnings of files out of the firefox project, like for example in SampleBrowser.
Also i noticed that only for firefox-ios we have a root yml file meanwhile for the other projects like Focus and SampleComponentAppLibrary we have a project level config yml.
I need to explore a bit more this solution and check with the team if we can move the firefox ios yml to its project folder so we keep it isolated like the other projects and we can then run directly swiftlint in the project folder.

I'll look better into it tomorrow and let you know as soon as i have news.
In the meantime if you need any help on anything we can chat directly on Element chat

@tonell-m
Copy link
Contributor Author

Hi @FilippoZazzeroni! I do agree with the points you're raising, personally (with the knowledge I have atm) I would suggest the following:

  • When we run Swiftlint from the build phase of either firefox-ios or focus-ios we should add a mask to the files that will be linted. Currently the linted files are obtained via git diff, but we could imagine adding folders to that command like so
    git diff --name-only --diff-filter=ACM ${BROWSER_KIT_ROOT} ${SRC_ROOT} | grep -e '\.swift$'
    
    That would limit the linted files to the ones tied to the current project + BrowserKit since it is shared between both apps
  • Same goes for linting in Bitrise, we could pass only the sources involved in the current project like swiftlint BrowserKit/**/*.swift firefox-ios/**/*.swift or swiftlint BrowserKit/**/*.swift focus-ios/**/*.swift
  • As for the configuration, it would make sense to me to have a shared swiftlint config at root level to unify linter rules accross firefox-ios and focus-ios but with nested configs we could also have specific configurations for both projects to enable/disable specific rules if needed

Let me know what you think when you'll have discussed it with your team ✌️

@FilippoZazzeroni
Copy link
Collaborator

Thanks very much @tonell-m i like this idea, i'll come up with a proposal and let you know how we can proceed.

@FilippoZazzeroni
Copy link
Collaborator

Hi @tonell-m i've created this task with the results of the proposal. Let me know if you'd like to take on this. thanks again for all the good info, let me know if the proposal makes sense to you.
We should try to merge the task before this PR so we can keep adding nested config files.
As a side note i didn't go much into the problem of files linted out of the repo since it is something that can happen also with the current state of the repo so i wouldn't worry about that.

@tonell-m
Copy link
Contributor Author

tonell-m commented Jan 13, 2025

Hi @FilippoZazzeroni! The task looks good, can you assign the issue to me please? I will look into it sometime this week 🙂

Only thing I think might be missing is updating the Bitrise configuration. Currently it seems to be using --config to look for the root config file, so it doesn't take nested configs into account. Until we also update this the pipelines will still be failing.

What bugs me is that looking at bitriste.yml only the linting path is provided so in theory (if I look at the source code for this step) it should not use --config, but still when looking at the pipeline's output, there are some violations raised from test files that should be ignored thanks to nested configs. So I'm guessing I'm missing something there, maybe you have an idea?

@FilippoZazzeroni
Copy link
Collaborator

Hi @tonell-m yes of course. Can you please comment on github issue so i can assign you directly otherwise GitHub doesn't show your name on the assignee list. Anyway i agree with you for Bitrise i also read the code and theoretically it shouldn't use --config. let see when the PR is up what bitrise does. Anyway thanks so much 🎉 .

@tonell-m tonell-m force-pushed the fxios-10205/swiftlint-nested-config branch from 7feedec to cdbff29 Compare January 22, 2025 09:20
let view: EngineView

init(engine: Engine = WKEngine.factory(),
sessionDependencies: EngineSessionDependencies? = nil) {
init?(engine: Engine = WKEngine.factory(),
Copy link
Collaborator

Choose a reason for hiding this comment

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

Not sure if this is intended behavior, LGTM just tagging @lmarceau that worked on this implementation

Copy link
Contributor Author

Choose a reason for hiding this comment

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

wow, fast review! As I've mentioned below:

SampleBrowser doesn't compile anymore because RootViewController is missing a method from AddressToolbarDelegate

So I couldn't really test this fix, but it was the simplest workaround to get rid of the implicitly unwrapped property. Since engineSession was implicitly unwrapped in BrowserViewController it would have crashed anyway if engineProvider.session was nil

Copy link
Contributor

Choose a reason for hiding this comment

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

The change works for me 👍

@tonell-m
Copy link
Contributor Author

Hi @FilippoZazzeroni @lmarceau, now that #24219 has been merged I've updated this PR 😄

  • implicitly_unwrapped_optional rule is enabled in the root config but disabled using nested config in:
    • focus-ios/.swiftlint.yml
    • firefox-ios/firefox-ios-tests/Tests/.swiftlint.yml
    • BrowserKit/Tests/.swiftlint.yml
  • There was 3 remaining violations in SampleComponentLibraryApp and SampleBrowser that I've resolved (btw I've noted that SampleBrowser doesn't compile anymore because RootViewController is missing a method from AddressToolbarDelegate but that's not for this PR)

@FilippoZazzeroni
Copy link
Collaborator

Thanks @tonell-m 🎉 , regarding Sample browser we can create a separate PR to fix it, let me know if you'd like to do it, i can create a ticket for you

@mobiletest-ci-bot
Copy link

Messages
📖 Edited 7 files
📖 Created 2 files

Generated by 🚫 Danger Swift against cdbff29

@tonell-m
Copy link
Contributor Author

Thanks @tonell-m 🎉 , regarding Sample browser we can create a separate PR to fix it, let me know if you'd like to do it, i can create a ticket for you

@FilippoZazzeroni sure I'm down for it!

@FilippoZazzeroni FilippoZazzeroni merged commit 55b930a into mozilla-mobile:main Jan 23, 2025
12 of 13 checks passed
@FilippoZazzeroni
Copy link
Collaborator

Thanks a lot @tonell-m here is the issue

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants