Fix flakiness of //scouting:scouting_test

I ran the `//scouting:scouting_test` 10'000 and got 6 failures.
They all had messages like this:

    2022/03/13 23:41:36 Error calling ListenAndServe(): listen tcp :36144: bind: address already in use

The protractor framework finds a random port for the webserver to use.
After it finds an available port, it starts the webserver and tells it
about the port to use with the `--port` argument. The problem here is
that before the webserver has a chance to claim the port, another
process that grabs random ports can come in and claim the same port.

This patch fixes the race condition by using the fixed port of 6060.
This port is outside of the random port search window of the other
processes so it should not interfere.

After this patch, running the test 10'000 times produced 0 failures.

Signed-off-by: Philipp Schrader <philipp.schrader@gmail.com>
Change-Id: I5b415114d09d4af8862116a8df90f40e084f9218
diff --git a/WORKSPACE b/WORKSPACE
index f574965..eb250cf 100644
--- a/WORKSPACE
+++ b/WORKSPACE
@@ -767,6 +767,10 @@
     name = "npm",
     frozen_lockfile = True,
     package_json = "//:package.json",
+    patch_args = ["-p1"],
+    post_install_patches = [
+        "//third_party:npm/@bazel/protractor/bazel-protractor.patch",
+    ],
     symlink_node_modules = False,
     yarn_lock = "//:yarn.lock",
 )
diff --git a/third_party/npm/@bazel/protractor/bazel-protractor.patch b/third_party/npm/@bazel/protractor/bazel-protractor.patch
new file mode 100644
index 0000000..ee49a91
--- /dev/null
+++ b/third_party/npm/@bazel/protractor/bazel-protractor.patch
@@ -0,0 +1,34 @@
+diff --git a/node_modules/@bazel/protractor/protractor-utils.js b/node_modules/@bazel/protractor/protractor-utils.js
+index 2d577b82..c5c10c22 100755
+--- a/node_modules/@bazel/protractor/protractor-utils.js
++++ b/node_modules/@bazel/protractor/protractor-utils.js
+@@ -67,19 +67,9 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
+     }
+     exports.isTcpPortBound = isTcpPortBound;
+     function findFreeTcpPort() {
+-        return __awaiter(this, void 0, void 0, function* () {
+-            const range = {
+-                min: 32768,
+-                max: 60000,
+-            };
+-            for (let i = 0; i < 100; i++) {
+-                let port = Math.floor(Math.random() * (range.max - range.min) + range.min);
+-                if (yield isTcpPortFree(port)) {
+-                    return port;
+-                }
+-            }
+-            throw new Error('Unable to find a free port');
+-        });
++        // FRC971 hack. Since we're in linux-sandbox, we can use a fixed port.
++        // The original code has a race condition.
++        return 6060;
+     }
+     exports.findFreeTcpPort = findFreeTcpPort;
+     function waitForServer(port, timeout) {
+@@ -121,4 +111,4 @@ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, ge
+     }
+     exports.runServer = runServer;
+ });
+-//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"protractor-utils.js","sourceRoot":"","sources":["../../../../../packages/protractor/protractor-utils.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;;;;;;;;;;;;;;;;;;;;;;IAEH,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAE,CAAC,CAAC;IACrE,+CAA+C;IAC/C,2BAA2B;IAE3B,SAAgB,aAAa,CAAC,IAAY;QACxC,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,MAAM,GAAG,GAAG,CAAC,YAAY,EAAE,CAAC;YAClC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE;gBACvB,OAAO,CAAC,KAAK,CAAC,CAAC;YACjB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;gBACtB,OAAO,CAAC,IAAI,CAAC,CAAC;YAChB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,EAAE;gBACvB,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAbD,sCAaC;IAED,SAAgB,cAAc,CAAC,IAAY;QACzC,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;YAChC,MAAM,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,EAAE;gBAC1B,OAAO,CAAC,IAAI,CAAC,CAAC;YAChB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,EAAE;gBACzB,OAAO,CAAC,KAAK,CAAC,CAAC;YACjB,CAAC,CAAC,CAAC;YACH,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QACvB,CAAC,CAAC,CAAC;IACL,CAAC;IAXD,wCAWC;IAED,SAAsB,eAAe;;YACnC,MAAM,KAAK,GAAG;gBACZ,GAAG,EAAE,KAAK;gBACV,GAAG,EAAE,KAAK;aACX,CAAC;YACF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE;gBAC5B,IAAI,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,CAAC,GAAG,GAAG,KAAK,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC;gBAC3E,IAAI,MAAM,aAAa,CAAC,IAAI,CAAC,EAAE;oBAC7B,OAAO,IAAI,CAAC;iBACb;aACF;YACD,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;KAAA;IAZD,0CAYC;IAWD,SAAgB,aAAa,CAAC,IAAY,EAAE,OAAe;QACzD,OAAO,cAAc,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACzC,IAAI,CAAC,OAAO,EAAE;gBACZ,IAAI,OAAO,IAAI,CAAC,EAAE;oBAChB,MAAM,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAC;iBACxD;gBACD,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;gBACpC,OAAO,IAAI,OAAO,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,UAAU,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;qBAClD,IAAI,CAAC,GAAG,EAAE,CAAC,aAAa,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC;aACtD;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;IACL,CAAC;IAZD,sCAYC;IAQD;;;;;OAKG;IACH,SAAsB,SAAS,CAC3B,SAAiB,EAAE,YAAoB,EAAE,QAAgB,EAAE,UAAoB,EAC/E,OAAO,GAAG,IAAI;;YAChB,MAAM,UAAU,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,SAAS,IAAI,YAAY,EAAE,CAAC,CAAC;YACpE,MAAM,IAAI,GAAG,MAAM,eAAe,EAAE,CAAC;YAErC,6DAA6D;YAC7D,MAAM,aAAa,GAAG,aAAa,CAAC,KAAK,CACrC,UAAU,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,EAAE,EAAC,KAAK,EAAE,SAAS,EAAC,CAAC,CAAC;YAEpF,4EAA4E;YAC5E,aAAa,CAAC,EAAE,CAAC,MAAM,EAAE,QAAQ,CAAC,EAAE;gBAClC,IAAI,QAAQ,KAAK,CAAC,EAAE;oBAClB,MAAM,IAAI,KAAK,CAAC,kCAAkC,QAAQ,EAAE,CAAC,CAAC;iBAC/D;YACH,CAAC,CAAC,CAAC;YAEH,qDAAqD;YACrD,MAAM,aAAa,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;YAEnC,OAAO,EAAC,IAAI,EAAC,CAAC;QAChB,CAAC;KAAA;IArBD,8BAqBC","sourcesContent":["/**\n * @license\n * Copyright 2017 The Bazel Authors. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n *\n * You may obtain a copy of the License at\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst runfiles = require(process.env['BAZEL_NODE_RUNFILES_HELPER']!);\nimport * as child_process from 'child_process';\nimport * as net from 'net';\n\nexport function isTcpPortFree(port: number): Promise<boolean> {\n  return new Promise((resolve, reject) => {\n    const server = net.createServer();\n    server.on('error', (e) => {\n      resolve(false);\n    });\n    server.on('close', () => {\n      resolve(true);\n    });\n    server.listen(port, () => {\n      server.close();\n    });\n  });\n}\n\nexport function isTcpPortBound(port: number): Promise<boolean> {\n  return new Promise((resolve, reject) => {\n    const client = new net.Socket();\n    client.once('connect', () => {\n      resolve(true);\n    });\n    client.once('error', (e) => {\n      resolve(false);\n    });\n    client.connect(port);\n  });\n}\n\nexport async function findFreeTcpPort(): Promise<number> {\n  const range = {\n    min: 32768,\n    max: 60000,\n  };\n  for (let i = 0; i < 100; i++) {\n    let port = Math.floor(Math.random() * (range.max - range.min) + range.min);\n    if (await isTcpPortFree(port)) {\n      return port;\n    }\n  }\n  throw new Error('Unable to find a free port');\n}\n\n// Interface for config parameter of the protractor_web_test_suite onPrepare function\nexport interface OnPrepareConfig {\n  // The workspace name\n  workspace: string;\n\n  // The server binary to run\n  server: string;\n}\n\nexport function waitForServer(port: number, timeout: number): Promise<boolean> {\n  return isTcpPortBound(port).then(isBound => {\n    if (!isBound) {\n      if (timeout <= 0) {\n        throw new Error('Timeout waiting for server to start');\n      }\n      const wait = Math.min(timeout, 500);\n      return new Promise((res, rej) => setTimeout(res, wait))\n          .then(() => waitForServer(port, timeout - wait));\n    }\n    return true;\n  });\n}\n\n// Return type from runServer function\nexport interface ServerSpec {\n  // Port number that the server is running on\n  port: number;\n}\n\n/**\n * Runs the specified server binary from a given workspace and waits for the server\n * being ready. The server binary will be resolved from the Bazel runfiles. Note that\n * the server will be launched with a random free port in order to support test concurrency\n * with Bazel.\n */\nexport async function runServer(\n    workspace: string, serverTarget: string, portFlag: string, serverArgs: string[],\n    timeout = 5000): Promise<ServerSpec> {\n  const serverPath = runfiles.resolve(`${workspace}/${serverTarget}`);\n  const port = await findFreeTcpPort();\n\n  // Start the Bazel server binary with a random free TCP port.\n  const serverProcess = child_process.spawn(\n      serverPath, serverArgs.concat([portFlag, port.toString()]), {stdio: 'inherit'});\n\n  // In case the process exited with an error, we want to propagate the error.\n  serverProcess.on('exit', exitCode => {\n    if (exitCode !== 0) {\n      throw new Error(`Server exited with error code: ${exitCode}`);\n    }\n  });\n\n  // Wait for the server to be bound to the given port.\n  await waitForServer(port, timeout);\n\n  return {port};\n}\n"]}
+\ No newline at end of file
++//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"protractor-utils.js","sourceRoot":"","sources":["../../../../../packages/protractor/protractor-utils.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2017 The Bazel Authors. All rights reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n *\n * You may obtain a copy of the License at\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst runfiles = require(process.env['BAZEL_NODE_RUNFILES_HELPER']!);\nimport * as child_process from 'child_process';\nimport * as net from 'net';\n\nexport function isTcpPortFree(port: number): Promise<boolean> {\n  return new Promise((resolve, reject) => {\n    const server = net.createServer();\n    server.on('error', (e) => {\n      resolve(false);\n    });\n    server.on('close', () => {\n      resolve(true);\n    });\n    server.listen(port, () => {\n      server.close();\n    });\n  });\n}\n\nexport function isTcpPortBound(port: number): Promise<boolean> {\n  return new Promise((resolve, reject) => {\n    const client = new net.Socket();\n    client.once('connect', () => {\n      resolve(true);\n    });\n    client.once('error', (e) => {\n      resolve(false);\n    });\n    client.connect(port);\n  });\n}\n\nexport async function findFreeTcpPort(): Promise<number> {\n  const range = {\n    min: 32768,\n    max: 60000,\n  };\n  for (let i = 0; i < 100; i++) {\n    let port = Math.floor(Math.random() * (range.max - range.min) + range.min);\n    if (await isTcpPortFree(port)) {\n      return port;\n    }\n  }\n  throw new Error('Unable to find a free port');\n}\n\n// Interface for config parameter of the protractor_web_test_suite onPrepare function\nexport interface OnPrepareConfig {\n  // The workspace name\n  workspace: string;\n\n  // The server binary to run\n  server: string;\n}\n\nexport function waitForServer(port: number, timeout: number): Promise<boolean> {\n  return isTcpPortBound(port).then(isBound => {\n    if (!isBound) {\n      if (timeout <= 0) {\n        throw new Error('Timeout waiting for server to start');\n      }\n      const wait = Math.min(timeout, 500);\n      return new Promise((res, rej) => setTimeout(res, wait))\n          .then(() => waitForServer(port, timeout - wait));\n    }\n    return true;\n  });\n}\n\n// Return type from runServer function\nexport interface ServerSpec {\n  // Port number that the server is running on\n  port: number;\n}\n\n/**\n * Runs the specified server binary from a given workspace and waits for the server\n * being ready. The server binary will be resolved from the Bazel runfiles. Note that\n * the server will be launched with a random free port in order to support test concurrency\n * with Bazel.\n */\nexport async function runServer(\n    workspace: string, serverTarget: string, portFlag: string, serverArgs: string[],\n    timeout = 5000): Promise<ServerSpec> {\n  const serverPath = runfiles.resolve(`${workspace}/${serverTarget}`);\n  const port = await findFreeTcpPort();\n\n  // Start the Bazel server binary with a random free TCP port.\n  const serverProcess = child_process.spawn(\n      serverPath, serverArgs.concat([portFlag, port.toString()]), {stdio: 'inherit'});\n\n  // In case the process exited with an error, we want to propagate the error.\n  serverProcess.on('exit', exitCode => {\n    if (exitCode !== 0) {\n      throw new Error(`Server exited with error code: ${exitCode}`);\n    }\n  });\n\n  // Wait for the server to be bound to the given port.\n  await waitForServer(port, timeout);\n\n  return {port};\n}\n"]}