1
0
mirror of https://github.com/bitwarden/browser.git synced 2024-11-10 09:59:48 +01:00
bitwarden-browser/src/utils.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

193 lines
6.0 KiB
TypeScript
Raw Normal View History

2018-05-17 16:58:30 +02:00
import * as fs from "fs";
import * as path from "path";
import { Response } from "jslib-node/cli/models/response";
import { MessageResponse } from "jslib-node/cli/models/response/messageResponse";
import { Organization } from "jslib-common/models/domain/organization";
import { CollectionView } from "jslib-common/models/view/collectionView";
import { FolderView } from "jslib-common/models/view/folderView";
2018-05-16 17:17:40 +02:00
import { NodeUtils } from "jslib-common/misc/nodeUtils";
import { FlagName, Flags } from "./flags";
2018-05-31 15:08:54 +02:00
2018-05-16 17:17:40 +02:00
export class CliUtils {
2019-06-05 03:03:26 +02:00
static writeLn(s: string, finalLine: boolean = false, error: boolean = false) {
const stream = error ? process.stderr : process.stdout;
if (finalLine && (process.platform === "win32" || !stream.isTTY)) {
stream.write(s);
} else {
2018-08-06 16:38:32 +02:00
stream.write(s + "\n");
}
2021-12-20 18:04:00 +01:00
}
2018-05-17 19:28:22 +02:00
static readFile(input: string): Promise<string> {
return new Promise<string>((resolve, reject) => {
let p: string = null;
if (input != null && input !== "") {
const osInput = path.join(input);
if (osInput.indexOf(path.sep) === -1) {
p = path.join(process.cwd(), osInput);
} else {
p = osInput;
}
} else {
reject("You must specify a file path.");
2021-12-20 18:04:00 +01:00
}
2018-05-17 19:28:22 +02:00
fs.readFile(p, "utf8", (err, data) => {
if (err != null) {
2018-08-28 05:12:53 +02:00
reject(err.message);
2018-05-17 19:28:22 +02:00
}
resolve(data);
2021-12-20 18:04:00 +01:00
});
2018-05-17 19:28:22 +02:00
});
}
2021-12-20 18:04:00 +01:00
/**
* Save the given data to a file and determine the target file if necessary.
* If output is non-empty, it is used as target filename. Otherwise the target filename is
* built from the current working directory and the given defaultFileName.
*
* @param data to be written to the file.
* @param output file to write to or empty to choose automatically.
* @param defaultFileName to use when no explicit output filename is given.
* @return the chosen output file.
*/
static saveFile(data: string | Buffer, output: string, defaultFileName: string) {
let p: string = null;
let mkdir = false;
if (output != null && output !== "") {
const osOutput = path.join(output);
if (osOutput.indexOf(path.sep) === -1) {
p = path.join(process.cwd(), osOutput);
2021-12-20 18:04:00 +01:00
} else {
2018-05-17 19:28:22 +02:00
mkdir = true;
if (osOutput.endsWith(path.sep)) {
p = path.join(osOutput, defaultFileName);
} else {
2018-05-17 19:28:22 +02:00
p = osOutput;
}
}
} else {
p = path.join(process.cwd(), defaultFileName);
}
2018-05-16 17:17:40 +02:00
p = path.resolve(p);
if (mkdir) {
const dir = p.substring(0, p.lastIndexOf(path.sep));
if (!fs.existsSync(dir)) {
NodeUtils.mkdirpSync(dir, "700");
}
}
return new Promise<string>((resolve, reject) => {
fs.writeFile(p, data, { encoding: "utf8", mode: 0o600 }, (err) => {
if (err != null) {
reject("Cannot save file to " + p);
2021-12-20 18:04:00 +01:00
}
2018-05-16 17:17:40 +02:00
resolve(p);
2021-12-20 18:04:00 +01:00
});
});
}
/**
* Process the given data and write it to a file if possible. If the user requested RAW output and
2018-05-16 17:17:40 +02:00
* no output name is given, the file is directly written to stdout. The resulting Response contains
* an otherwise empty message then to prevent writing other information to stdout.
2021-12-20 18:04:00 +01:00
*
* If an output is given or no RAW output is requested, the rules from [saveFile] apply.
2021-12-20 18:04:00 +01:00
*
2018-05-16 17:17:40 +02:00
* @param data to be written to the file or stdout.
* @param output file to write to or empty to choose automatically.
* @param defaultFileName to use when no explicit output filename is given.
2018-05-16 17:17:40 +02:00
* @return an empty [Response] if written to stdout or a [Response] with the chosen output file otherwise.
2021-12-20 18:04:00 +01:00
*/
2018-05-16 17:17:40 +02:00
static async saveResultToFile(data: string | Buffer, output: string, defaultFileName: string) {
if ((output == null || output === "") && process.env.BW_RAW === "true") {
// No output is given and the user expects raw output. Since the command result is about content,
// we directly return the command result to stdout (and suppress further messages).
process.stdout.write(data);
return Response.success();
}
2018-05-18 21:26:59 +02:00
const filePath = await this.saveFile(data, output, defaultFileName);
const res = new MessageResponse("Saved " + filePath, null);
res.raw = filePath;
return Response.success(res);
2021-12-20 18:04:00 +01:00
}
static readStdin(): Promise<string> {
return new Promise((resolve, reject) => {
let input: string = "";
2021-12-20 18:04:00 +01:00
if (process.stdin.isTTY) {
resolve(input);
2021-12-20 18:04:00 +01:00
return;
}
process.stdin.setEncoding("utf8");
process.stdin.on("readable", () => {
while (true) {
const chunk = process.stdin.read();
if (chunk == null) {
2021-12-20 18:04:00 +01:00
break;
}
input += chunk;
2021-12-20 18:04:00 +01:00
}
});
process.stdin.on("end", () => {
resolve(input);
2021-12-20 18:04:00 +01:00
});
});
}
2018-05-16 17:17:40 +02:00
static searchFolders(folders: FolderView[], search: string) {
search = search.toLowerCase();
return folders.filter((f) => {
2018-05-16 17:17:40 +02:00
if (f.name != null && f.name.toLowerCase().indexOf(search) > -1) {
return true;
2021-12-20 18:04:00 +01:00
}
2018-05-16 17:17:40 +02:00
return false;
2021-12-20 18:04:00 +01:00
});
}
2018-05-16 17:17:40 +02:00
static searchCollections(collections: CollectionView[], search: string) {
search = search.toLowerCase();
return collections.filter((c) => {
2018-05-16 17:17:40 +02:00
if (c.name != null && c.name.toLowerCase().indexOf(search) > -1) {
return true;
2021-12-20 18:04:00 +01:00
}
2018-05-16 17:17:40 +02:00
return false;
2021-12-20 18:04:00 +01:00
});
}
2018-05-18 21:26:59 +02:00
static searchOrganizations(organizations: Organization[], search: string) {
search = search.toLowerCase();
return organizations.filter((o) => {
2018-05-18 21:26:59 +02:00
if (o.name != null && o.name.toLowerCase().indexOf(search) > -1) {
return true;
}
return false;
});
}
2022-01-19 16:45:14 +01:00
static convertBooleanOption(optionValue: any) {
return optionValue || optionValue === "" ? true : false;
}
static flagEnabled(flag: FlagName) {
return this.flags[flag] == null || this.flags[flag];
}
private static get flags(): Flags {
const envFlags = process.env.FLAGS;
if (typeof envFlags === "string") {
return JSON.parse(envFlags) as Flags;
} else {
return envFlags as Flags;
}
}
2018-05-16 17:17:40 +02:00
}