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

dev: reject builds when CRL JS dependencies are 'pnpm link'ed #107129

Conversation

sjbarag
Copy link
Contributor

@sjbarag sjbarag commented Jul 18, 2023

When working with first-party JS dependencies that aren't in this monorepo, the idiomatic development workflow uses pnpm link [1] to link multiple JS packages together. Specifically, running pnpm link /path/to/github.com/cockroachdb/ui/packages/foo from within pkg/ui/workspaces/* creates a symbolic link at
node_modules/@cockroachlabs/foo that points to
../../(…)/ui/packages/foo. This works quite smoothly for local development, as changes in the 'foo' package are automatically seen by a pnpm webpack --watch running in CRDB. The two packages act like they're maintained in the same repo, while allowing independent version history, CI processes, etc.

Unfortunately, rules_js currently offers no way to link outside of the Bazel workspace. Such a symlink is either ignored by rules_js (since it doesn't appear in pnpm-lock.yaml) or is rejected if it's manually added to the lockfile [2]. Allowing Bazel-based builds of CockroachDB when 'pnpm link'ed packages are present introduces dependency skew between Bazel and non-Bazel builds. To allow pnpm link to be used safely, pre-emptively reject builds of 'cockroach' and 'cockroach-oss' that are run through the 'dev' helper when linked @cockroachlabs/ packages are detected.

[1] https://pnpm.io/cli/link
[2] aspect-build/rules_js#1165

Release note: None
Epic: none


Example output:
image

Total duration is 16ms on my machine since we're checking so few files. We can drop these into a goroutine per first-party JS if we want, but this is certainly fast enough to be conversational.

@sjbarag sjbarag requested a review from a team as a code owner July 18, 2023 22:55
@cockroach-teamcity
Copy link
Member

This change is Reviewable

@sjbarag sjbarag force-pushed the reject-cockroach-builds-when-linked-npm-deps-detected branch from f3e8c9d to 3b1d54e Compare July 18, 2023 23:47
@sjbarag
Copy link
Contributor Author

sjbarag commented Jul 19, 2023

super nit: should those arrows point right instead of left? I copied the pnpm link output. While that's in the context of adding a dependency (and the arrow is "I added it from $directory!"), it feels reasonable to keep them consistent:

image

@sjbarag sjbarag requested a review from nathanstilwell July 19, 2023 15:53
Copy link
Collaborator

@rickystewart rickystewart left a comment

Choose a reason for hiding this comment

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

A couple questions, looking good though.

@@ -434,6 +462,7 @@ func (o *OS) ListSubdirectories(path string) ([]string, error) {
if entry.IsDir() {
ret = append(ret, entry.Name())
}
fmt.Printf("entry %q is a dir? %t\n", entry.Name(), entry.IsDir())
Copy link
Collaborator

Choose a reason for hiding this comment

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

Should delete?

depPath := filepath.Join(crlModulesPath, depName)
resolved, err := d.os.Readlink(depPath)
if err != nil {
return fmt.Errorf("could not evaluate symlink %s: %w", depPath, err)
Copy link
Collaborator

@rickystewart rickystewart Jul 19, 2023

Choose a reason for hiding this comment

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

Is this an error condition (e.g. the function must fail if this is not a symlink), or something we can safely ignore? My intuition tells me that if it's e.g. a normal directory, then it's not a symlink and so we have passed the check. But maybe there is something I'm not understanding. I also don't fully understand the differences in behavior if you are using pnpm directly vs. bazel.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The good news is that pnpm guarantees everything here will be a symlink into pkg/ui/node_modules/.pnpm/, so there's no chance of getting a non-symlink here. I can add an isDir() (or use filepath.EvalSymlinks(), which should be safe to use on non-links) if you'd prefer!

Copy link
Collaborator

Choose a reason for hiding this comment

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

No worries, I'm satisfied as long as this is pnpm's default behavior.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yep — doc link for the future: https://pnpm.io/symlinked-node-modules-structure

filepath.Join(crlModulesPath, resolved),
)
if err != nil {
return fmt.Errorf("could not relitivize path %s: %w", resolved, err)
Copy link
Collaborator

Choose a reason for hiding this comment

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

Nit: relativize

pkg/cmd/dev/ui.go Show resolved Hide resolved
When working with first-party JS dependencies that aren't in this
monorepo, the idiomatic development workflow uses pnpm link [1] to link
multiple JS packages together. Specifically, running
'pnpm link /path/to/github.com/cockroachdb/ui/packages/foo' from within
pkg/ui/workspaces/* creates a symbolic link at
node_modules/@cockroachlabs/foo that points to
../../(…)/ui/packages/foo. This works quite smoothly for local
development, as changes in the 'foo' package are automatically seen by a
'pnpm webpack --watch' running in CRDB. The two packages act like
they're maintained in the same repo, while allowing independent version
history, CI processes, etc.

Unfortunately, rules_js currently offers no way to link outside of the
Bazel workspace. Such a symlink is either ignored by rules_js (since it
doesn't appear in pnpm-lock.yaml) or is rejected if it's manually added
to the lockfile [2]. Allowing Bazel-based builds of CockroachDB when
'pnpm link'ed packages are present introduces dependency skew between
Bazel and non-Bazel builds. To allow 'pnpm link' to be used safely,
pre-emptively reject builds of 'cockroach' and 'cockroach-oss' that are
run through the 'dev' helper when linked @cockroachlabs/ packages are
detected.

[1] https://pnpm.io/cli/link
[2] aspect-build/rules_js#1165

Release note: None
Epic: none
@sjbarag sjbarag force-pushed the reject-cockroach-builds-when-linked-npm-deps-detected branch from 3b1d54e to 3ba9bcb Compare July 19, 2023 18:19
@sjbarag sjbarag requested a review from rickystewart July 19, 2023 18:19
@sjbarag sjbarag added the backport-23.1.x Flags PRs that need to be backported to 23.1 label Jul 19, 2023
@sjbarag
Copy link
Contributor Author

sjbarag commented Jul 19, 2023

bors r=rickystewart

@craig craig bot merged commit 0f31d50 into cockroachdb:master Jul 19, 2023
@craig
Copy link
Contributor

craig bot commented Jul 19, 2023

Build succeeded:

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
backport-23.1.x Flags PRs that need to be backported to 23.1
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants