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

pkg/golinters: add noreplace #1812

Closed
wants to merge 1 commit into from

Conversation

carnott-snap
Copy link

@carnott-snap carnott-snap commented Mar 3, 2021

Add support for enforcing that go.mod should not contain replace clauses.

https://git.sr.ht/~urandom/noreplace

@boring-cyborg
Copy link

boring-cyborg bot commented Mar 3, 2021

Hey, thank you for opening your first Pull Request !

@CLAassistant
Copy link

CLAassistant commented Mar 3, 2021

CLA assistant check
All committers have signed the CLA.

@carnott-snap carnott-snap force-pushed the noreplace branch 3 times, most recently from 596a91f to d49a358 Compare March 3, 2021 19:44
@ldez
Copy link
Member

ldez commented Mar 3, 2021

Hello,

I don't think that calling the go CLI is the right way to do that. (https://git.sr.ht/~urandom/noreplace/tree/master/item/noreplace.go#L22)

The following implementation seems better: https://github.com/gostaticanalysis/noreplace

@carnott-snap
Copy link
Author

carnott-snap commented Mar 3, 2021

oh, it is super gross, but I did not know about your noreplace, so I followed what gomodguard does. (a merged linter.)

I would rather not support this one off package, let me see if I can drop in the gostaticanalysis package.

EDIT: also, heads up they do the same thing just one level deeper. this is a fundamental limitation of the stdlib, since this is not exported at runtime and hidden behind cmd/go/internal/modload.Init.

@ldez
Copy link
Member

ldez commented Mar 3, 2021

ok but it seems weird, I will try to find how the do that without the go command.

FYI, I was not a maintainer when gomodguard was merged.

@ldez
Copy link
Member

ldez commented Mar 3, 2021

Another idea, maybe this can be added to gomodguard 🤔

Add support for enforcing that go.mod should not contain replace
clauses.
@carnott-snap
Copy link
Author

carnott-snap commented Mar 3, 2021

I looked at that first, but gomodguard is about enforcing dependencies and is a pretty complex by comparison. If the author is interested we can, but it seems like pulling in noreplace is easy enough.

Also, the implementation of noreplace is not optimal. Notice that my package actually returns token.Position information so that we can populate your result.Issue, specifically the Replacement LineRange and Pos fields; though I do like memoising the call to modfile.Parse. I think that this will require substantial changes to the analysis.Pass interface, see golang/go#44753.

cc @ryancurrah

@ldez ldez added blocked Need's direct action from maintainer linter: new Support new linter labels Mar 3, 2021
@ldez
Copy link
Member

ldez commented Mar 3, 2021

Yes in fact the noreplace implementation seems worse than the gomodguard implementation.


The code inside the Go command to find the module path:
https://github.com/golang/go/blob/6db80d74200675e20c562684c0bcc6d12a5631eb/src/cmd/go/internal/modload/init.go#L727

@ldez
Copy link
Member

ldez commented Mar 3, 2021

offtopic: @carnott-snap how it is possible to contribute to SourceHut repo? (please don't say "with patch by mail" 😸 )

Edit: I have my answer: it's with patch by mail 😹 (git send-email)

@ldez
Copy link
Member

ldez commented Mar 3, 2021

Are you sure to want to use SourceHut instead of Github/Gitlab/Bitbucket for this kind of project?

SourceHut uses an old way for community contributions, for me it's more a way to limit contributions because the contribution cost is high.

@carnott-snap
Copy link
Author

carnott-snap commented Mar 3, 2021

it is my default, so yes; plus the code should be pretty stable, with the exception of needing to pass more data to golangci-lint, or an eventual modanalyser.Analysis symbol or what ever golang/go#44753 comes up with.

if you would like noreplace.Check to return a different interface (more, less, or different data) let us sort that out now, since even on github it will be harder to make changes after we merge this.

@ldez
Copy link
Member

ldez commented Mar 3, 2021

So maybe you can split your code into two parts:

  • one to get the mod path
  • one to analyze the mod file

Like that it will be possible to get the mod path only one time.

Edit: in fact, I'm not sure what is the best way 🤔

@carnott-snap
Copy link
Author

carnott-snap commented Mar 3, 2021

can you elaborate? (suggestions are welcome) I am not familiar with the inner workings of the golangci-lint framework. this code is copypasta from gomodguard.

I am just unclear why noreplace.Check would be running more than once?

@ldez
Copy link
Member

ldez commented Mar 3, 2021

My suggestion ldez@11f852d

I tested it on golangci-lint with:

go run ./cmd/golangci-lint/ run --no-config --disable-all --enable=noreplace

@ldez
Copy link
Member

ldez commented Mar 3, 2021

More I watch your linter more I think there is a problem with the current implementation of gomodguard inside golangci-lint

@carnott-snap
Copy link
Author

carnott-snap commented Mar 3, 2021

ldez@11f852d

left some comments inline; though I was suggesting you use the ```suggestion interface (ctrl+g), as it would keep everything in the pr.

More I watch your linter more I think there is a problem with the current implementation of gomodguard inside golangci-lint

yeah, I think that anything doing go.mod lints may not really be playing nice with analysis.Analyzer, though gomodguard appears to do both module and package lints.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I know "github suggestions" but my suggestions are too large.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I tested the PR on a sandbox project and in fact it doesn't work.

@ryancurrah
Copy link
Member

We could add noreplace check to gomodguard. Not sure what issues your seeing.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I think that gomodguard also doesn't work as expected

Edit: it works as expected

@ldez
Copy link
Member

ldez commented Mar 4, 2021

@carnott-snap Have you tried your PR (not your noreplace binary but a golangci-lint binary) on a real project?

@carnott-snap
Copy link
Author

We could add noreplace check to gomodguard. Not sure what issues your seeing.

If everybody prefers that, I have some sample code I was messing around with, but it was not working. Probably speaks to @ldez's comment. I will wait for confirmation before opening that pr.

@ryancurrah
Copy link
Member

ryancurrah commented Mar 4, 2021

I think that gomodguard is also doesn't work as expected

I have been using it where I work for a while, no issues so far. What were you expecting it to do? Maybe I can help clear any confusion up.

If everybody prefers that, I have some sample code I was messing around with, but it was not working. Probably speaks to @ldez's comment. I will wait for confirmation before opening that pr.

It should be pretty easy to identify replace directives with the golang.org/x/mod package.

As per the module documentation.

If the path on the right side of the arrow is an absolute or relative path (beginning with ./ or ../), it is interpreted as the local file path to the replacement module root directory, which must contain a go.mod file. The replacement version must be omitted in this case. - https://golang.org/ref/mod#go-mod-file-replace

So we look for replace directives matching the above conditions and set them as blocked. If this is for local replace directives only.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I have been using it where I work for a while, no issues so far. What were you expecting it to do? Maybe I can help clear any confusion up.

I edited my message, it works.


noreplace is really close to the blocked.local_replace_directives option of gomodguard

gomodguard works with go.mod, noreplace doesn't work but currently I don't know why.


I see 2 use-cases for a replace directive linter:

  • the need to ban all replace directives
  • the need to allow only local replace directives (for dev) and ban all other replace directives.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I can see a third case:

  • allow only some replace directive because of old dead transitive dependencies or dependencies on docker/moby weird dependencies.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

So there are 3 options:

  • make this linter works as expected
  • merge this linter with gomodguard
  • create a dedicated gomodreplace linter

@ryancurrah
Copy link
Member

Yeah, those are definitely valid cases. The gomodreplace linter sounds interesting.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

I agree, the gomodreplace linter is my favorite option.
I am also interested in working on this topic but maybe @carnott-snap wants to handle that.

@ldez
Copy link
Member

ldez commented Mar 4, 2021

@carnott-snap I'm not sure we understand each other, I was talking about creating a dedicated linter for this.

@ldez
Copy link
Member

ldez commented Mar 5, 2021

I created the gomodreplace linter https://github.com/ldez/gomodreplace

EDIT: I created the gomoddirectives linter https://github.com/ldez/gomoddirectives

@ldez
Copy link
Member

ldez commented Mar 5, 2021

I will close this PR in favor of #1817

@ldez ldez closed this Mar 5, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
blocked Need's direct action from maintainer linter: new Support new linter
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants