Skip to content

Commit

Permalink
bazel: upgrade to rules_nodejs 5.4.2
Browse files Browse the repository at this point in the history
Upgrading to Bazel's rules_nodejs 5.x exposed a flaw in our previous
Bazel integration: because rules_nodejs explicitly doesn't support
yarn's "workspaces" feature [1] (in which common dependencies are
hoisted to the lowest common parent directory), any NPM dependencies
with different major versions between db-console and cluster-ui would
get flattened to a single version. This left one of those packages using
an unsupported (and un-requested) version of a dependency. Removing
the yarn workspace layout and using separate Bazel repositories for
each JS project ensured each project received the correct dependencies,
but revealed incompatibilities with the requested versions. Upgrade
rules_nodejs to the latest released version, remove yarn workspaces from
the pkg/ui/ tree, and fix all revealed compatibility issues.

[1] bazel-contrib/rules_nodejs#266
  • Loading branch information
sjbarag committed May 25, 2022
1 parent 41ed791 commit a7b73ae
Show file tree
Hide file tree
Showing 26 changed files with 37,156 additions and 18,799 deletions.
89 changes: 73 additions & 16 deletions WORKSPACE
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,10 @@
workspace(
name = "com_github_cockroachdb_cockroach",
managed_directories = {
"@npm": ["pkg/ui/node_modules"],
"@yarn_vendor": ["pkg/ui/yarn-vendor"],
"@npm_protos": ["pkg/ui/workspaces/db-console/src/js/node_modules"],
"@npm_cluster_ui": ["pkg/ui/workspaces/cluster_ui/node_modules"],
"@npm_db_console": ["pkg/ui/workspaces/db-console/node_modules"],
},
)

Expand All @@ -25,10 +28,8 @@ http_archive(
# Like the above, but for nodeJS.
http_archive(
name = "build_bazel_rules_nodejs",
sha256 = "cfc289523cf1594598215901154a6c2515e8bf3671fd708264a6f6aefe02bf39",
urls = [
"https://storage.googleapis.com/public-bazel-artifacts/bazel/rules_nodejs-4.4.6.tar.gz",
],
sha256 = "e328cb2c9401be495fa7d79c306f5ee3040e8a03b2ebb79b022e15ca03770096",
urls = ["https://github.com/bazelbuild/rules_nodejs/releases/download/5.4.2/rules_nodejs-5.4.2.tar.gz"],
)

# Load gazelle. This lets us auto-generate BUILD.bazel files throughout the
Expand Down Expand Up @@ -195,8 +196,13 @@ go_register_toolchains(nogo = "@com_github_cockroachdb_cockroach//:crdb_nogo")
# begin rules_nodejs dependencies #
###################################

# Install rules_nodejs dependencies
load("@build_bazel_rules_nodejs//:repositories.bzl", "build_bazel_rules_nodejs_dependencies")
build_bazel_rules_nodejs_dependencies()

# Configure nodeJS.
load("@build_bazel_rules_nodejs//:index.bzl", "node_repositories", "yarn_install")
load("@rules_nodejs//nodejs:yarn_repositories.bzl", "yarn_repositories")

node_repositories(
node_repositories = {
Expand All @@ -210,8 +216,11 @@ node_repositories(
"https://storage.googleapis.com/public-bazel-artifacts/js/node/v{version}/{filename}",
],
node_version = "16.13.0",
package_json = ["//pkg/ui:package.json"],
yarn_repositories = {
)

yarn_repositories(
name = "yarn",
yarn_releases = {
"1.22.11": ("yarn-v1.22.11.tar.gz", "yarn-v1.22.11", "2c320de14a6014f62d29c34fec78fdbb0bc71c9ccba48ed0668de452c1f5fe6c"),
},
yarn_urls = [
Expand All @@ -220,21 +229,69 @@ node_repositories(
yarn_version = "1.22.11",
)

# install external dependencies for pkg/ui package
load("//build/bazelutil:seed_yarn_cache.bzl", "seed_yarn_cache")
seed_yarn_cache(name = "yarn_cache")

# Install external dependencies for NPM packages in pkg/ui/ as separate bazel
# repositories, to avoid version conflicts between those packages.
# Unfortunately Bazel's rules_nodejs does not support yarn workspaces, so
# packages have isolated dependencies and must be installed as isolated
# Bazel repositories.
yarn_install(
name = "npm_protos",
args = [
"--offline",
"--ignore-optional",
],
data = [
"//pkg/ui:.yarnrc",
"@yarn_cache//:protos.seed",
],
package_path = "/",
package_json = "//pkg/ui/workspaces/db-console/src/js:package.json",
strict_visibility = False,
yarn_lock = "//pkg/ui/workspaces/db-console/src/js:yarn.lock",
)

yarn_install(
name = "npm",
name = "npm_db_console",
args = [
"--offline",
"--ignore-optional",
],
data = [
"//pkg/ui:.yarnrc",
"@yarn_cache//:db_console.seed",
],
package_json = "//pkg/ui/workspaces/db-console:package.json",
yarn_lock = "//pkg/ui/workspaces/db-console:yarn.lock",
strict_visibility = False,
patch_args = [ "-p0", "--remove-empty-files", "--silent" ],
post_install_patches = [
"//pkg/ui:patches/aria-query/remove-filenames-with-spaces.db-console.patch",
],
symlink_node_modules = True,
)

yarn_install(
name = "npm_cluster_ui",
args = [
"--verbose",
"--offline",
"--ignore-optional",
],
package_json = "//pkg/ui:package.json",
data = [
"//pkg/ui:.yarnrc",
"@yarn_cache//:cluster_ui.seed",
],
package_json = "//pkg/ui/workspaces/cluster-ui:package.json",
strict_visibility = False,
yarn_lock = "//pkg/ui:yarn.lock",
links = {
"@cockroachlabs/crdb-protobuf-client": "//pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client",
"@cockroachlabs/crdb-protobuf-client-ccl": "//pkg/ui/workspaces/db-console/ccl/src/js:crdb-protobuf-client-ccl",
"@cockroachlabs/cluster-ui": "//pkg/ui/workspaces/cluster-ui:cluster-ui",
},
yarn_lock = "//pkg/ui/workspaces/cluster-ui:yarn.lock",
patch_args = [ "-p0", "--remove-empty-files", "--silent" ],
post_install_patches = [
"//pkg/ui:patches/aria-query/remove-filenames-with-spaces.cluster-ui.patch",
],
symlink_node_modules = True,
)

#################################
Expand Down
16 changes: 16 additions & 0 deletions build/bazelutil/check_yarn_vendor_submodule.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
#!/usr/bin/env bash
set -eu

if [ "$#" -ne 1 ]; then
echo "Usage: $0 PATH"
echo "Example: check_yarn_vendor_submodule.sh /path/to/cockroachdb/cockroach/pkg/ui/yarn-vendor"
exit 1
fi

yarnVendorDir=$1
if [ -z "$(ls -A $yarnVendorDir)" ]; then
echo "No packages available from yarn-vendor submodule." >&2
echo "You may need to run 'git submodule update --init'." >&2
exit 2
fi

67 changes: 67 additions & 0 deletions build/bazelutil/seed_yarn_cache.bzl
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
load("@build_bazel_rules_nodejs//:index.bzl", "node_repositories", "yarn_install")

# Returns the root of the user workspace. No built-in way to get
# this but we can derive it from the path of the package.json file
# in the user workspace sources.
# From https://github.com/bazelbuild/rules_nodejs/blob/55a84d14604b121230a2013b2469d93e38f1f601/internal/npm_install/npm_install.bzl#L367
def _seed_yarn_cache_impl(rctx):
workspace_root = str(
rctx.path(
Label("//:WORKSPACE")
).dirname
)

rctx.report_progress("Checking for yarn-vendor submodule...")
res = rctx.execute(
[
rctx.path(Label("//build/bazelutil:check_yarn_vendor_submodule.sh")),
workspace_root + "/pkg/ui"
],
)
if res.return_code != 0:
fail("Unable to seed yarn cache: " + res.stderr)

paths = {
"protos": "pkg/ui/workspaces/db-console/src/js",
"cluster_ui": "pkg/ui/workspaces/cluster-ui",
"db_console": "pkg/ui/workspaces/db-console",
}
yarn_dir = str(rctx.path(Label("@yarn//:bin/yarn")).dirname)
for key, path in paths.items():
rctx.report_progress("Seeding yarn cache for {}...".format(path))
# Execute a script that uses the bazel-installed yarn (via $PATH) to install
# dependencies while running at the root of the *non-sandboxed* node
# package (argv[1]), but putting those dependencies in a repo-specific
# location (argv[2]) to avoid corrupting the local node_modules
# directories.
res = rctx.execute(
[
rctx.path(Label("//build/bazelutil:seed_yarn_cache.sh")),
workspace_root + "/" + path,
rctx.path("node_modules." + key),
],
environment = {
"PATH": "{}:{}".format(yarn_dir, rctx.os.environ["PATH"])
},
)

if res.return_code != 0:
fail("Unable to seed yarn cache: " + res.stderr)

rctx.file(
key + ".seed",
content = "",
executable = False,
)


rctx.file(
"BUILD.bazel",
content = """exports_files(["protos.seed", "cluster_ui.seed", "db_console.seed"])""",
executable = False,
)

seed_yarn_cache = repository_rule(
implementation = _seed_yarn_cache_impl,
local = True,
)
20 changes: 20 additions & 0 deletions build/bazelutil/seed_yarn_cache.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
#!/usr/bin/env bash
set -euo pipefail

if [ "$#" -ne 2 ]; then
echo "Usage: $0 PKG_PATH MODULES_DIR"
echo "Example: seed_yarn_cache.sh /path/to/cockroachdb/cockroach/pkg/ui/workspaces/cluster-ui /path/to/repo/node_modules.cluster_ui"
exit 1
fi

rootDir=$1
moduleDir=$2

yarn \
--cwd $rootDir \
--modules-folder $moduleDir \
--offline \
--ignore-optional \
--no-progress \
--mutex network \
install
13 changes: 1 addition & 12 deletions pkg/cmd/dev/build.go
Original file line number Diff line number Diff line change
Expand Up @@ -136,17 +136,6 @@ func (d *dev) build(cmd *cobra.Command, commandLine []string) error {
args = append(args, additionalBazelArgs...)

if cross == "" {
// run "yarn --check-files" before any bazel target that includes UI to ensure that node_modules dir is consistent
// see related issue: https://github.com/cockroachdb/cockroach/issues/70867
for _, arg := range args {
if arg == "--config=with_ui" {
logCommand("bazel", "run", "@nodejs//:yarn", "--", "--check-files", "--cwd", "pkg/ui", "--offline")
if err := d.exec.CommandContextInheritingStdStreams(ctx, "bazel", "run", "@nodejs//:yarn", "--", "--check-files", "--cwd", "pkg/ui", "--offline"); err != nil {
return err
}
break
}
}
logCommand("bazel", args...)
if err := d.exec.CommandContextInheritingStdStreams(ctx, "bazel", args...); err != nil {
return err
Expand Down Expand Up @@ -174,7 +163,7 @@ func (d *dev) crossBuild(
script.WriteString(fmt.Sprintf("bazel %s\n", shellescape.QuoteCommand(bazelArgs)))
for _, arg := range bazelArgs {
if arg == "--config=with_ui" {
script.WriteString("bazel run @nodejs//:yarn -- --check-files --cwd pkg/ui --offline\n")
script.WriteString("bazel run @yarn//:yarn -- --check-files --cwd pkg/ui --offline\n")
break
}
}
Expand Down
2 changes: 1 addition & 1 deletion pkg/cmd/dev/testdata/datadriven/dev-build
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ cp sandbox/pkg/cmd/cockroach-short/cockroach-short_/cockroach-short crdb-checkou
exec
dev build -- --verbose_failures --sandbox_debug
----
bazel run @nodejs//:yarn -- --check-files --cwd pkg/ui --offline
bazel run @yarn//:yarn -- --check-files --cwd pkg/ui --offline
bazel build //pkg/cmd/cockroach:cockroach --config=with_ui --verbose_failures --sandbox_debug
bazel info workspace --color=no
mkdir crdb-checkout/bin
Expand Down
24 changes: 12 additions & 12 deletions pkg/cmd/dev/testdata/datadriven/ui
Original file line number Diff line number Diff line change
Expand Up @@ -3,40 +3,40 @@ dev ui watch
----
bazel build //pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client //pkg/ui/workspaces/db-console/ccl/src/js:crdb-protobuf-client-ccl
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 3000
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 3000

exec
dev ui watch --oss
----
bazel build //pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=oss --env.target=http://localhost:8080 --port 3000
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=oss --env.target=http://localhost:8080 --port 3000

exec
dev ui watch --secure
----
bazel build //pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client //pkg/ui/workspaces/db-console/ccl/src/js:crdb-protobuf-client-ccl
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 3000 --https
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 3000 --https

exec
dev ui watch --db http://example.crdb.io:4848
----
bazel build //pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client //pkg/ui/workspaces/db-console/ccl/src/js:crdb-protobuf-client-ccl
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://example.crdb.io:4848 --port 3000
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://example.crdb.io:4848 --port 3000

exec
dev ui watch --port 12345
----
bazel build //pkg/ui/workspaces/db-console/src/js:crdb-protobuf-client //pkg/ui/workspaces/db-console/ccl/src/js:crdb-protobuf-client-ccl
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 12345
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui build:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console webpack-dev-server --config webpack.app.js --mode development --env.WEBPACK_SERVE --env.dist=ccl --env.target=http://localhost:8080 --port 12345

exec
dev ui lint
Expand Down Expand Up @@ -71,8 +71,8 @@ cp sandbox/pkg/ui/workspaces/db-console/ccl/src/js/protos.d.ts crdb-checkout/pkg
rm -rf crdb-checkout/pkg/ui/workspaces/cluster-ui/dist
cp -r sandbox/pkg/ui/workspaces/cluster-ui/dist crdb-checkout/pkg/ui/workspaces/cluster-ui/dist
bazel info workspace --color=no
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console karma:watch
bazel run @nodejs//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui jest --watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/db-console karma:watch
bazel run @yarn//:yarn -- --silent --cwd crdb-checkout/pkg/ui/workspaces/cluster-ui jest --watch

exec
dev ui clean
Expand Down
2 changes: 1 addition & 1 deletion pkg/cmd/dev/ui.go
Original file line number Diff line number Diff line change
Expand Up @@ -500,6 +500,6 @@ Replaces 'make ui-test' and 'make ui-test-watch'.`,
// d.exec.CommandContextWithEnv(ctx, env, "bazel", argv)
func buildBazelYarnArgv(argv ...string) []string {
return append([]string{
"run", "@nodejs//:yarn", "--",
"run", "@yarn//:yarn", "--",
}, argv...)
}
20 changes: 20 additions & 0 deletions pkg/ui/not-yarn-workspace.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
#!/usr/bin/env bash
set -euo pipefail

(
set -x
yarn --cwd workspaces/db-console/src/js install
yarn --cwd workspaces/cluster-ui install
yarn --cwd workspaces/db-console install
)

cat << "EOF"
=================================== WARNING ====================================
As-of May 2022, the pkg/ui/ tree is no longer a "yarn workspace", due to Bazel's
lack of support for them. When adding, updating, or otherwise modifying JS
dependencies, be sure to `cd` into the correct package within workspaces/.
As a convenience, running `yarn install` from pkg/ui/ still installs the
dependencies for all packages within workspaces/.
================================================================================
EOF
Loading

0 comments on commit a7b73ae

Please sign in to comment.