mirror of
https://github.com/wavetermdev/waveterm.git
synced 2024-12-21 16:38:23 +01:00
206 lines
7.0 KiB
TypeScript
206 lines
7.0 KiB
TypeScript
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import { fireAndForget } from "@/util/util";
|
|
import { app, dialog, ipcMain, shell } from "electron";
|
|
import envPaths from "env-paths";
|
|
import { existsSync, mkdirSync } from "fs";
|
|
import os from "os";
|
|
import path from "path";
|
|
import { WaveDevVarName, WaveDevViteVarName } from "../frontend/util/isdev";
|
|
import * as keyutil from "../frontend/util/keyutil";
|
|
|
|
// This is a little trick to ensure that Electron puts all its runtime data into a subdirectory to avoid conflicts with our own data.
|
|
// On macOS, it will store to ~/Library/Application \Support/waveterm/electron
|
|
// On Linux, it will store to ~/.config/waveterm/electron
|
|
// On Windows, it will store to %LOCALAPPDATA%/waveterm/electron
|
|
app.setName("waveterm/electron");
|
|
|
|
const isDev = !app.isPackaged;
|
|
const isDevVite = isDev && process.env.ELECTRON_RENDERER_URL;
|
|
console.log(`Running in ${isDev ? "development" : "production"} mode`);
|
|
if (isDev) {
|
|
process.env[WaveDevVarName] = "1";
|
|
}
|
|
if (isDevVite) {
|
|
process.env[WaveDevViteVarName] = "1";
|
|
}
|
|
|
|
const waveDirNamePrefix = "waveterm";
|
|
const waveDirNameSuffix = isDev ? "dev" : "";
|
|
const waveDirName = `${waveDirNamePrefix}${waveDirNameSuffix ? `-${waveDirNameSuffix}` : ""}`;
|
|
|
|
const paths = envPaths("waveterm", { suffix: waveDirNameSuffix });
|
|
|
|
app.setName(isDev ? "Wave (Dev)" : "Wave");
|
|
const unamePlatform = process.platform;
|
|
const unameArch: string = process.arch;
|
|
keyutil.setKeyUtilPlatform(unamePlatform);
|
|
|
|
const WaveConfigHomeVarName = "WAVETERM_CONFIG_HOME";
|
|
const WaveDataHomeVarName = "WAVETERM_DATA_HOME";
|
|
const WaveHomeVarName = "WAVETERM_HOME";
|
|
|
|
export function checkIfRunningUnderARM64Translation(fullConfig: FullConfigType) {
|
|
if (!fullConfig.settings["app:dismissarchitecturewarning"] && app.runningUnderARM64Translation) {
|
|
console.log("Running under ARM64 translation, alerting user");
|
|
const dialogOpts: Electron.MessageBoxOptions = {
|
|
type: "warning",
|
|
buttons: ["See documentation", "Dismiss"],
|
|
title: "Wave has detected a performance issue",
|
|
message: `Wave has detected that it is running in ARM64 translation mode.\n\nThis may cause performance issues.\n\nPlease download the native version of Wave for your architecture (${unameArch})`,
|
|
};
|
|
|
|
const choice = dialog.showMessageBoxSync(null, dialogOpts);
|
|
if (choice === 0) {
|
|
// Open the documentation URL
|
|
console.log("Opening documentation URL");
|
|
fireAndForget(() => shell.openExternal("https://docs.waveterm.dev"));
|
|
} else {
|
|
console.log("User dismissed the dialog");
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Gets the path to the old Wave home directory (defaults to `~/.waveterm`).
|
|
* @returns The path to the directory if it exists and contains valid data for the current app, otherwise null.
|
|
*/
|
|
function getWaveHomeDir(): string {
|
|
let home = process.env[WaveHomeVarName];
|
|
if (!home) {
|
|
const homeDir = app.getPath("home");
|
|
if (homeDir) {
|
|
home = path.join(homeDir, `.${waveDirName}`);
|
|
}
|
|
}
|
|
// If home exists and it has `wave.lock` in it, we know it has valid data from Wave >=v0.8. Otherwise, it could be for WaveLegacy (<v0.8)
|
|
if (home && existsSync(home) && existsSync(path.join(home, "wave.lock"))) {
|
|
return home;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Ensure the given path exists, creating it recursively if it doesn't.
|
|
* @param path The path to ensure.
|
|
* @returns The same path, for chaining.
|
|
*/
|
|
function ensurePathExists(path: string): string {
|
|
if (!existsSync(path)) {
|
|
mkdirSync(path, { recursive: true });
|
|
}
|
|
return path;
|
|
}
|
|
|
|
/**
|
|
* Gets the path to the directory where Wave configurations are stored. Creates the directory if it does not exist.
|
|
* Handles backwards compatibility with the old Wave Home directory model, where configurations and data were stored together.
|
|
* @returns The path where configurations should be stored.
|
|
*/
|
|
function getWaveConfigDir(): string {
|
|
// If wave home dir exists, use it for backwards compatibility
|
|
const waveHomeDir = getWaveHomeDir();
|
|
if (waveHomeDir) {
|
|
return path.join(waveHomeDir, "config");
|
|
}
|
|
|
|
const override = process.env[WaveConfigHomeVarName];
|
|
const xdgConfigHome = process.env.XDG_CONFIG_HOME;
|
|
let retVal: string;
|
|
if (override) {
|
|
retVal = override;
|
|
} else if (xdgConfigHome) {
|
|
retVal = path.join(xdgConfigHome, waveDirName);
|
|
} else {
|
|
retVal = path.join(app.getPath("home"), ".config", waveDirName);
|
|
}
|
|
return ensurePathExists(retVal);
|
|
}
|
|
|
|
/**
|
|
* Gets the path to the directory where Wave data is stored. Creates the directory if it does not exist.
|
|
* Handles backwards compatibility with the old Wave Home directory model, where configurations and data were stored together.
|
|
* @returns The path where data should be stored.
|
|
*/
|
|
function getWaveDataDir(): string {
|
|
// If wave home dir exists, use it for backwards compatibility
|
|
const waveHomeDir = getWaveHomeDir();
|
|
if (waveHomeDir) {
|
|
return waveHomeDir;
|
|
}
|
|
|
|
const override = process.env[WaveDataHomeVarName];
|
|
const xdgDataHome = process.env.XDG_DATA_HOME;
|
|
let retVal: string;
|
|
if (override) {
|
|
retVal = override;
|
|
} else if (xdgDataHome) {
|
|
retVal = path.join(xdgDataHome, waveDirName);
|
|
} else {
|
|
retVal = paths.data;
|
|
}
|
|
return ensurePathExists(retVal);
|
|
}
|
|
|
|
function getElectronAppBasePath(): string {
|
|
return path.dirname(import.meta.dirname);
|
|
}
|
|
|
|
function getElectronAppUnpackedBasePath(): string {
|
|
return getElectronAppBasePath().replace("app.asar", "app.asar.unpacked");
|
|
}
|
|
|
|
const wavesrvBinName = `wavesrv.${unameArch}`;
|
|
|
|
function getWaveSrvPath(): string {
|
|
if (process.platform === "win32") {
|
|
const winBinName = `${wavesrvBinName}.exe`;
|
|
const appPath = path.join(getElectronAppUnpackedBasePath(), "bin", winBinName);
|
|
return `${appPath}`;
|
|
}
|
|
return path.join(getElectronAppUnpackedBasePath(), "bin", wavesrvBinName);
|
|
}
|
|
|
|
function getWaveSrvCwd(): string {
|
|
return getWaveDataDir();
|
|
}
|
|
|
|
ipcMain.on("get-is-dev", (event) => {
|
|
event.returnValue = isDev;
|
|
});
|
|
ipcMain.on("get-platform", (event, url) => {
|
|
event.returnValue = unamePlatform;
|
|
});
|
|
ipcMain.on("get-user-name", (event) => {
|
|
const userInfo = os.userInfo();
|
|
event.returnValue = userInfo.username;
|
|
});
|
|
ipcMain.on("get-host-name", (event) => {
|
|
event.returnValue = os.hostname();
|
|
});
|
|
ipcMain.on("get-webview-preload", (event) => {
|
|
event.returnValue = path.join(getElectronAppBasePath(), "preload", "preload-webview.cjs");
|
|
});
|
|
ipcMain.on("get-data-dir", (event) => {
|
|
event.returnValue = getWaveDataDir();
|
|
});
|
|
ipcMain.on("get-config-dir", (event) => {
|
|
event.returnValue = getWaveConfigDir();
|
|
});
|
|
|
|
export {
|
|
getElectronAppBasePath,
|
|
getElectronAppUnpackedBasePath,
|
|
getWaveConfigDir,
|
|
getWaveDataDir,
|
|
getWaveSrvCwd,
|
|
getWaveSrvPath,
|
|
isDev,
|
|
isDevVite,
|
|
unameArch,
|
|
unamePlatform,
|
|
WaveConfigHomeVarName,
|
|
WaveDataHomeVarName,
|
|
};
|