2023-10-09 19:40:36 +00:00
|
|
|
#!/usr/bin/env node
|
|
|
|
|
2024-01-12 00:51:40 +00:00
|
|
|
const HELP = `
|
|
|
|
USAGE
|
|
|
|
zed-local [options] [zed args]
|
|
|
|
|
2024-01-17 20:45:18 +00:00
|
|
|
SUMMARY
|
2024-01-18 20:18:12 +00:00
|
|
|
Runs 1-6 instances of Zed using a locally-running collaboration server.
|
2024-01-17 20:45:18 +00:00
|
|
|
Each instance of Zed will be signed in as a different user specified in
|
|
|
|
either \`.admins.json\` or \`.admins.default.json\`.
|
|
|
|
|
2024-01-12 00:51:40 +00:00
|
|
|
OPTIONS
|
2024-01-18 20:18:12 +00:00
|
|
|
--help Print this help message
|
|
|
|
--release Build Zed in release mode
|
|
|
|
-2, -3, -4, ... Spawn multiple Zed instances, with their windows tiled.
|
|
|
|
--top Arrange the Zed windows so they take up the top half of the screen.
|
2024-04-04 13:26:41 +00:00
|
|
|
--stable Use stable Zed release installed on local machine for all instances (except for the first one).
|
2024-01-12 00:51:40 +00:00
|
|
|
`.trim();
|
|
|
|
|
2024-06-12 16:27:40 +00:00
|
|
|
const { spawn, execSync, execFileSync } = require("child_process");
|
2024-01-17 20:45:18 +00:00
|
|
|
const assert = require("assert");
|
|
|
|
|
2024-04-23 12:41:31 +00:00
|
|
|
let users;
|
|
|
|
if (process.env.SEED_PATH) {
|
|
|
|
users = require(process.env.SEED_PATH).admins;
|
|
|
|
} else {
|
|
|
|
users = require("../crates/collab/seed.default.json").admins;
|
|
|
|
try {
|
|
|
|
const defaultUsers = users;
|
|
|
|
const customUsers = require("../crates/collab/seed.json").admins;
|
|
|
|
assert(customUsers.length > 0);
|
|
|
|
users = customUsers.concat(
|
|
|
|
defaultUsers.filter((user) => !customUsers.includes(user)),
|
|
|
|
);
|
|
|
|
} catch (_) {}
|
|
|
|
}
|
2023-10-09 19:40:36 +00:00
|
|
|
|
2023-12-15 21:27:58 +00:00
|
|
|
const RESOLUTION_REGEX = /(\d+) x (\d+)/;
|
|
|
|
const DIGIT_FLAG_REGEX = /^--?(\d+)$/;
|
2023-10-09 19:40:36 +00:00
|
|
|
|
2023-12-15 21:27:58 +00:00
|
|
|
let instanceCount = 1;
|
2024-01-10 21:04:26 +00:00
|
|
|
let isReleaseMode = false;
|
|
|
|
let isTop = false;
|
2024-04-04 13:26:41 +00:00
|
|
|
let othersOnStable = false;
|
2024-04-23 21:33:09 +00:00
|
|
|
let isStateful = false;
|
2024-01-10 21:04:26 +00:00
|
|
|
|
|
|
|
const args = process.argv.slice(2);
|
2024-01-12 00:51:40 +00:00
|
|
|
while (args.length > 0) {
|
|
|
|
const arg = args[0];
|
|
|
|
|
2024-01-10 21:04:26 +00:00
|
|
|
const digitMatch = arg.match(DIGIT_FLAG_REGEX);
|
|
|
|
if (digitMatch) {
|
|
|
|
instanceCount = parseInt(digitMatch[1]);
|
2024-01-12 00:51:40 +00:00
|
|
|
} else if (arg === "--release") {
|
2024-01-10 21:04:26 +00:00
|
|
|
isReleaseMode = true;
|
2024-04-23 21:33:09 +00:00
|
|
|
} else if (arg == "--stateful") {
|
|
|
|
isStateful = true;
|
2024-01-12 00:51:40 +00:00
|
|
|
} else if (arg === "--top") {
|
2024-01-10 21:04:26 +00:00
|
|
|
isTop = true;
|
2024-01-12 00:51:40 +00:00
|
|
|
} else if (arg === "--help") {
|
|
|
|
console.log(HELP);
|
|
|
|
process.exit(0);
|
2024-04-04 13:26:41 +00:00
|
|
|
} else if (arg === "--stable") {
|
|
|
|
othersOnStable = true;
|
2024-01-12 00:51:40 +00:00
|
|
|
} else {
|
|
|
|
break;
|
2024-01-10 21:04:26 +00:00
|
|
|
}
|
2024-01-12 00:51:40 +00:00
|
|
|
|
|
|
|
args.shift();
|
2023-10-09 19:40:36 +00:00
|
|
|
}
|
2024-06-12 16:27:40 +00:00
|
|
|
const os = require("os");
|
|
|
|
const platform = os.platform();
|
|
|
|
|
|
|
|
let screenWidth, screenHeight;
|
2024-06-13 17:16:10 +00:00
|
|
|
const titleBarHeight = 24;
|
2024-06-12 16:27:40 +00:00
|
|
|
|
|
|
|
if (platform === "darwin") {
|
|
|
|
// macOS
|
|
|
|
const displayInfo = JSON.parse(
|
|
|
|
execFileSync("system_profiler", ["SPDisplaysDataType", "-json"], {
|
|
|
|
encoding: "utf8",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
const mainDisplayResolution = displayInfo?.SPDisplaysDataType?.flatMap(
|
|
|
|
(display) => display?.spdisplays_ndrvs,
|
|
|
|
)
|
|
|
|
?.find((entry) => entry?.spdisplays_main === "spdisplays_yes")
|
|
|
|
?._spdisplays_resolution?.match(RESOLUTION_REGEX);
|
|
|
|
if (!mainDisplayResolution) {
|
|
|
|
throw new Error("Could not parse screen resolution");
|
|
|
|
}
|
|
|
|
screenWidth = parseInt(mainDisplayResolution[1]);
|
|
|
|
screenHeight = parseInt(mainDisplayResolution[2]) - titleBarHeight;
|
|
|
|
} else if (platform === "linux") {
|
|
|
|
// Linux
|
|
|
|
try {
|
|
|
|
const xrandrOutput = execSync('xrandr | grep "\\*" | cut -d" " -f4', {
|
|
|
|
encoding: "utf8",
|
|
|
|
}).trim();
|
|
|
|
[screenWidth, screenHeight] = xrandrOutput.split("x").map(Number);
|
|
|
|
} catch (err) {
|
|
|
|
console.log(err);
|
|
|
|
throw new Error("Could not get screen resolution");
|
|
|
|
}
|
2023-10-09 19:40:36 +00:00
|
|
|
}
|
2024-06-12 16:27:40 +00:00
|
|
|
|
|
|
|
screenHeight -= titleBarHeight;
|
2024-01-10 21:04:26 +00:00
|
|
|
|
|
|
|
if (isTop) {
|
|
|
|
screenHeight = Math.floor(screenHeight / 2);
|
|
|
|
}
|
2023-10-09 19:40:36 +00:00
|
|
|
|
|
|
|
// Determine the window size for each instance
|
2024-01-18 20:18:12 +00:00
|
|
|
let rows;
|
|
|
|
let columns;
|
|
|
|
switch (instanceCount) {
|
|
|
|
case 1:
|
|
|
|
[rows, columns] = [1, 1];
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
[rows, columns] = [1, 2];
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
[rows, columns] = [2, 2];
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
case 6:
|
|
|
|
[rows, columns] = [2, 3];
|
|
|
|
break;
|
2023-10-09 19:40:36 +00:00
|
|
|
}
|
|
|
|
|
2024-01-18 20:18:12 +00:00
|
|
|
const instanceWidth = Math.floor(screenWidth / columns);
|
|
|
|
const instanceHeight = Math.floor(screenHeight / rows);
|
|
|
|
|
2023-10-09 19:40:36 +00:00
|
|
|
// If a user is specified, make sure it's first in the list
|
2023-12-15 21:27:58 +00:00
|
|
|
const user = process.env.ZED_IMPERSONATE;
|
2023-10-09 19:40:36 +00:00
|
|
|
if (user) {
|
2023-12-15 21:27:58 +00:00
|
|
|
users = [user].concat(users.filter((u) => u !== user));
|
2023-10-09 19:40:36 +00:00
|
|
|
}
|
|
|
|
|
2024-01-17 20:45:18 +00:00
|
|
|
let buildArgs = ["build"];
|
2024-06-04 22:31:01 +00:00
|
|
|
let zedBinary = "target/debug/zed";
|
2024-01-17 20:45:18 +00:00
|
|
|
if (isReleaseMode) {
|
|
|
|
buildArgs.push("--release");
|
2024-06-04 22:31:01 +00:00
|
|
|
zedBinary = "target/release/zed";
|
2024-01-17 20:45:18 +00:00
|
|
|
}
|
2023-12-15 21:28:23 +00:00
|
|
|
|
2024-01-18 21:42:11 +00:00
|
|
|
try {
|
2024-06-12 16:27:40 +00:00
|
|
|
execFileSync("cargo", buildArgs, {
|
|
|
|
stdio: "inherit",
|
|
|
|
});
|
2024-01-18 21:42:11 +00:00
|
|
|
} catch (e) {
|
|
|
|
process.exit(0);
|
|
|
|
}
|
|
|
|
|
2023-10-09 19:40:36 +00:00
|
|
|
setTimeout(() => {
|
|
|
|
for (let i = 0; i < instanceCount; i++) {
|
2024-01-18 20:18:12 +00:00
|
|
|
const row = Math.floor(i / columns);
|
|
|
|
const column = i % columns;
|
2024-01-18 21:42:11 +00:00
|
|
|
const position = [
|
|
|
|
column * instanceWidth,
|
|
|
|
row * instanceHeight + titleBarHeight,
|
|
|
|
].join(",");
|
2024-01-18 20:18:12 +00:00
|
|
|
const size = [instanceWidth, instanceHeight].join(",");
|
2024-04-04 13:26:41 +00:00
|
|
|
let binaryPath = zedBinary;
|
|
|
|
if (i != 0 && othersOnStable) {
|
|
|
|
binaryPath = "/Applications/Zed.app/Contents/MacOS/zed";
|
|
|
|
}
|
|
|
|
spawn(binaryPath, i == 0 ? args : [], {
|
2023-12-15 21:27:58 +00:00
|
|
|
stdio: "inherit",
|
2024-04-23 23:23:26 +00:00
|
|
|
env: Object.assign({}, process.env, {
|
2023-10-09 19:40:36 +00:00
|
|
|
ZED_IMPERSONATE: users[i],
|
2024-01-18 20:18:12 +00:00
|
|
|
ZED_WINDOW_POSITION: position,
|
2024-06-18 00:58:47 +00:00
|
|
|
ZED_STATELESS: isStateful && i == 0 ? "" : "1",
|
2023-12-15 21:27:58 +00:00
|
|
|
ZED_ALWAYS_ACTIVE: "1",
|
2024-01-18 06:02:36 +00:00
|
|
|
ZED_SERVER_URL: "http://localhost:3000",
|
|
|
|
ZED_RPC_URL: "http://localhost:8080/rpc",
|
2023-12-15 21:27:58 +00:00
|
|
|
ZED_ADMIN_API_TOKEN: "secret",
|
2024-01-18 20:18:12 +00:00
|
|
|
ZED_WINDOW_SIZE: size,
|
2024-02-03 02:24:49 +00:00
|
|
|
ZED_CLIENT_CHECKSUM_SEED: "development-checksum-seed",
|
2024-01-17 20:45:18 +00:00
|
|
|
RUST_LOG: process.env.RUST_LOG || "info",
|
2024-04-23 23:23:26 +00:00
|
|
|
}),
|
2023-12-15 21:27:58 +00:00
|
|
|
});
|
2023-10-09 19:40:36 +00:00
|
|
|
}
|
2023-12-15 21:27:58 +00:00
|
|
|
}, 0.1);
|