2018-05-17 16:58:30 +02:00
|
|
|
import * as fs from "fs";
|
|
|
|
import * as path from "path";
|
|
|
|
|
2022-03-03 18:24:41 +01:00
|
|
|
import * as inquirer from "inquirer";
|
2022-03-03 15:32:49 +01:00
|
|
|
import * as JSZip from "jszip";
|
|
|
|
|
2024-10-08 16:14:39 +02:00
|
|
|
import { CollectionView } from "@bitwarden/admin-console/common";
|
[AC-1011] Admin Console / Billing code ownership (#4973)
* refactor: move SCIM component to admin-console, refs EC-1011
* refactor: move scimProviderType to admin-console, refs EC-1011
* refactor: move scim-config.api to admin-console, refs EC-1011
* refactor: create models folder and nest existing api contents, refs EC-1011
* refactor: move scim-config to admin-console models, refs EC-1011
* refactor: move billing.component to billing, refs EC-1011
* refactor: remove nested app folder from new billing structure, refs EC-1011
* refactor: move organizations/billing to billing, refs EC-1011
* refactor: move add-credit and adjust-payment to billing/settings, refs EC-1011
* refactor: billing history/sync to billing, refs EC-1011
* refactor: move org plans, payment/method to billing/settings, refs EC-1011
* fix: update legacy file paths for payment-method and tax-info, refs EC-1011
* fix: update imports for scim component, refs EC-1011
* refactor: move subscription and tax-info into billing, refs EC-1011
* refactor: move user-subscription to billing, refs EC-1011
* refactor: move images/cards to billing and update base path, refs EC-1011
* refactor: move payment-method, plan subscription, and plan to billing, refs EC-1011
* refactor: move transaction-type to billing, refs EC-1011
* refactor: move billing-sync-config to billing, refs EC-1011
* refactor: move billing-sync and bit-pay-invoice request to billing, refs EC-1011
* refactor: move org subscription and tax info update requests to billing, refs EC-1011
* fix: broken paths to billing, refs EC-1011
* refactor: move payment request to billing, refs EC-1011
* fix: update remaining imports for payment-request, refs EC-1011
* refactor: move tax-info-update to billing, refs EC-1011
* refactor: move billing-payment, billing-history, and billing responses to billing, refs EC-1011
* refactor: move organization-subscription-responset to billing, refs EC-1011
* refactor: move payment and plan responses to billing, refs EC-1011
* refactor: move subscription response to billing ,refs EC-1011
* refactor: move tax info and rate responses to billing, refs EC-1011
* fix: update remaining path to base response for tax-rate response, refs EC-1011
* refactor: (browser) move organization-service to admin-console, refs EC-1011
* refactor: (browser) move organizaiton-service to admin-console, refs EC-1011
* refactor: (cli) move share command to admin-console, refs EC-1011
* refactor: move organization-collect request model to admin-console, refs EC-1011
* refactor: (web) move organization, collection/user responses to admin-console, refs EC-1011
* refactor: (cli) move selection-read-only to admin-console, refs EC-1011
* refactor: (desktop) move organization-filter to admin-console, refs EC-1011
* refactor: (web) move organization-switcher to admin-console, refs EC-1011
* refactor: (web) move access-selector to admin-console, refs EC-1011
* refactor: (web) move create folder to admin-console, refs EC-1011
* refactor: (web) move org guards folder to admin-console, refs EC-1011
* refactor: (web) move org layout to admin-console, refs EC-1011
* refactor: move manage collections to admin console, refs EC-1011
* refactor: (web) move collection-dialog to admin-console, refs EC-1011
* refactor: (web) move entity users/events and events component to admin-console, refs EC-1011
* refactor: (web) move groups/group-add-edit to admin-console, refs EC-1011
* refactor: (web) move manage, org-manage module, and user-confirm to admin-console, refs EC-1011
* refactor: (web) move people to admin-console, refs EC-1011
* refactor: (web) move reset-password to admin-console, refs EC-1011
* refactor: (web) move organization-routing and module to admin-console, refs EC-1011
* refactor: move admin-console and billing within app scope, refs EC-1011
* fix: update leftover merge conflicts, refs EC-1011
* refactor: (web) member-dialog to admin-console, refs EC-1011
* refactor: (web) move policies to admin-console, refs EC-1011
* refactor: (web) move reporting to admin-console, refs EC-1011
* refactor: (web) move settings to admin-console, refs EC-1011
* refactor: (web) move sponsorships to admin-console, refs EC-1011
* refactor: (web) move tools to admin-console, refs EC-1011
* refactor: (web) move users to admin-console, refs EC-1011
* refactor: (web) move collections to admin-console, refs EC-1011
* refactor: (web) move create-organization to admin-console, refs EC-1011
* refactor: (web) move licensed components to admin-console, refs EC-1011
* refactor: (web) move bit organization modules to admin-console, refs EC-1011
* fix: update leftover import statements for organizations.module, refs EC-1011
* refactor: (web) move personal vault and max timeout to admin-console, refs EC-1011
* refactor: (web) move providers to admin-console, refs EC-1011
* refactor: (libs) move organization service to admin-console, refs EC-1011
* refactor: (libs) move profile org/provider responses and other misc org responses to admin-console, refs EC-1011
* refactor: (libs) move provider request and selectionion-read-only request to admin-console, refs EC-1011
* fix: update missed import path for provider-user-update request, refs EC-1011
* refactor: (libs) move abstractions to admin-console, refs EC-1011
* refactor: (libs) move org/provider enums to admin-console, refs EC-1011
* fix: update downstream import statements from libs changes, refs EC-1011
* refactor: (libs) move data files to admin-console, refs EC-1011
* refactor: (libs) move domain to admin-console, refs EC-1011
* refactor: (libs) move request objects to admin-console, refs EC-1011
* fix: update downstream import changes from libs, refs EC-1011
* refactor: move leftover provider files to admin-console, refs EC-1011
* refactor: (browser) move group policy environment to admin-console, refs EC-1011
* fix: (browser) update downstream import statements, refs EC-1011
* fix: (desktop) update downstream libs moves, refs EC-1011
* fix: (cli) update downstream import changes from libs, refs EC-1011
* refactor: move org-auth related files to admin-console, refs EC-1011
* refactor: (libs) move request objects to admin-console, refs EC-1011
* refactor: move persmissions to admin-console, refs EC-1011
* refactor: move sponsored families to admin-console and fix libs changes, refs EC-1011
* refactor: move collections to admin-console, refs EC-1011
* refactor: move spec file back to spec scope, refs EC-1011
* fix: update downstream imports due to libs changes, refs EC-1011
* fix: udpate downstream import changes due to libs, refs EC-1011
* fix: update downstream imports due to libs changes, refs EC-1011
* fix: update downstream imports from libs changes, refs EC-1011
* fix: update path malformation in jslib-services.module, refs EC-1011
* fix: lint errors from improper casing, refs AC-1011
* fix: update downstream filename changes, refs AC-1011
* fix: (cli) update downstream filename changes, refs AC-1011
* fix: (desktop) update downstream filename changes, refs AC-1011
* fix: (browser) update downstream filename changes, refs AC-1011
* fix: lint errors, refs AC-1011
* fix: prettier, refs AC-1011
* fix: lint fixes for import order, refs AC-1011
* fix: update import path for provider user type, refs AC-1011
* fix: update new codes import paths for admin console structure, refs AC-1011
* fix: lint/prettier, refs AC-1011
* fix: update layout stories path, refs AC-1011
* fix: update comoponents card icons base variable in styles, refs AC-1011
* fix: update provider service path in permissions guard spec, refs AC-1011
* fix: update provider permission guard path, refs AC-1011
* fix: remove unecessary TODO for shared index export statement, refs AC-1011
* refactor: move browser-organization service and cli organization-user response out of admin-console, refs AC-1011
* refactor: move web/browser/desktop collections component to vault domain, refs AC-1011
* refactor: move organization.module out of admin-console scope, refs AC-1011
* fix: prettier, refs AC-1011
* refactor: move organizations-api-key.request out of admin-console scope, refs AC-1011
2023-03-22 16:03:50 +01:00
|
|
|
import { Organization } from "@bitwarden/common/admin-console/models/domain/organization";
|
2023-06-06 22:34:53 +02:00
|
|
|
import { LogService } from "@bitwarden/common/platform/abstractions/log.service";
|
|
|
|
import { Utils } from "@bitwarden/common/platform/misc/utils";
|
2023-01-31 22:08:37 +01:00
|
|
|
import { FolderView } from "@bitwarden/common/vault/models/view/folder.view";
|
2023-11-27 21:59:44 +01:00
|
|
|
import { NodeUtils } from "@bitwarden/node/node-utils";
|
2022-11-18 13:20:19 +01:00
|
|
|
|
|
|
|
import { Response } from "./models/response";
|
|
|
|
import { MessageResponse } from "./models/response/message.response";
|
2018-05-16 17:17:40 +02:00
|
|
|
|
|
|
|
export class CliUtils {
|
2022-03-03 18:24:41 +01:00
|
|
|
static writeLn(s: string, finalLine = false, error = false) {
|
2019-06-05 03:03:26 +02:00
|
|
|
const stream = error ? process.stderr : process.stdout;
|
2019-09-09 18:07:15 +02:00
|
|
|
if (finalLine && (process.platform === "win32" || !stream.isTTY)) {
|
2018-08-06 15:38:17 +02:00
|
|
|
stream.write(s);
|
|
|
|
} else {
|
2018-08-06 16:38:32 +02:00
|
|
|
stream.write(s + "\n");
|
2018-08-06 15:38:17 +02:00
|
|
|
}
|
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
|
|
|
|
2023-08-16 16:17:03 +02:00
|
|
|
static extractZipContent(input: string, filepath: string): Promise<string> {
|
2022-03-08 13:50:36 +01:00
|
|
|
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;
|
2022-03-03 15:32:49 +01:00
|
|
|
}
|
2022-03-08 13:50:36 +01:00
|
|
|
} else {
|
|
|
|
reject("You must specify a file path.");
|
|
|
|
}
|
|
|
|
fs.readFile(p, function (err, data) {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
}
|
|
|
|
JSZip.loadAsync(data).then(
|
|
|
|
(zip) => {
|
2023-08-16 16:17:03 +02:00
|
|
|
resolve(zip.file(filepath).async("string"));
|
2022-03-08 13:50:36 +01:00
|
|
|
},
|
|
|
|
(reason) => {
|
|
|
|
reject(reason);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2022-03-03 15:32:49 +01:00
|
|
|
}
|
2023-08-16 16:17:03 +02:00
|
|
|
|
2020-08-25 20:24:43 +02: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;
|
2020-08-25 20:24:43 +02:00
|
|
|
if (osOutput.endsWith(path.sep)) {
|
|
|
|
p = path.join(osOutput, defaultFileName);
|
|
|
|
} else {
|
2018-05-17 19:28:22 +02:00
|
|
|
p = osOutput;
|
2020-08-25 20:24:43 +02:00
|
|
|
}
|
|
|
|
}
|
2018-05-17 04:40:48 +02:00
|
|
|
} 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
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-08-25 20:24:43 +02: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
|
|
|
*
|
2020-08-25 20:24:43 +02: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.
|
2020-08-25 20:24:43 +02:00
|
|
|
* @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
|
|
|
|
2020-08-25 20:24:43 +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
|
|
|
}
|
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
static readStdin(): Promise<string> {
|
|
|
|
return new Promise((resolve, reject) => {
|
2022-03-03 18:24:41 +01:00
|
|
|
let input = "";
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
if (process.stdin.isTTY) {
|
|
|
|
resolve(input);
|
2021-12-20 18:04:00 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
process.stdin.setEncoding("utf8");
|
|
|
|
process.stdin.on("readable", () => {
|
2022-03-03 18:24:41 +01:00
|
|
|
// eslint-disable-next-line
|
2018-05-17 04:40:48 +02:00
|
|
|
while (true) {
|
|
|
|
const chunk = process.stdin.read();
|
|
|
|
if (chunk == null) {
|
2021-12-20 18:04:00 +01:00
|
|
|
break;
|
|
|
|
}
|
2018-05-17 04:40:48 +02:00
|
|
|
input += chunk;
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-05-17 04:40:48 +02: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();
|
2021-02-04 05:51:59 +01:00
|
|
|
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();
|
2021-02-04 05:51:59 +01:00
|
|
|
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();
|
2021-02-04 05:51:59 +01:00
|
|
|
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
|
|
|
|
2022-02-23 22:47:32 +01:00
|
|
|
/**
|
|
|
|
* Gets a password from all available sources. In order of priority these are:
|
|
|
|
* * passwordfile
|
|
|
|
* * passwordenv
|
|
|
|
* * user interaction
|
|
|
|
*
|
|
|
|
* Returns password string if successful, Response if not.
|
|
|
|
*/
|
|
|
|
static async getPassword(
|
|
|
|
password: string,
|
|
|
|
options: { passwordFile?: string; passwordEnv?: string },
|
|
|
|
logService?: LogService,
|
|
|
|
): Promise<string | Response> {
|
|
|
|
if (Utils.isNullOrEmpty(password)) {
|
|
|
|
if (options?.passwordFile) {
|
|
|
|
password = await NodeUtils.readFirstLine(options.passwordFile);
|
|
|
|
} else if (options?.passwordEnv) {
|
|
|
|
if (process.env[options.passwordEnv]) {
|
|
|
|
password = process.env[options.passwordEnv];
|
|
|
|
} else if (logService) {
|
|
|
|
logService.warning(`Warning: Provided passwordenv ${options.passwordEnv} is not set`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Utils.isNullOrEmpty(password)) {
|
|
|
|
if (process.env.BW_NOINTERACTION !== "true") {
|
|
|
|
const answer: inquirer.Answers = await inquirer.createPromptModule({
|
|
|
|
output: process.stderr,
|
|
|
|
})({
|
|
|
|
type: "password",
|
|
|
|
name: "password",
|
|
|
|
message: "Master password:",
|
|
|
|
});
|
|
|
|
|
|
|
|
password = answer.password;
|
|
|
|
} else {
|
|
|
|
return Response.badRequest(
|
|
|
|
"Master password is required. Try again in interactive mode or provide a password file or environment variable.",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return password;
|
|
|
|
}
|
|
|
|
|
2022-01-19 16:45:14 +01:00
|
|
|
static convertBooleanOption(optionValue: any) {
|
|
|
|
return optionValue || optionValue === "" ? true : false;
|
|
|
|
}
|
2023-09-04 22:01:16 +02:00
|
|
|
|
|
|
|
static convertNumberOption(optionValue: any, defaultValue: number) {
|
|
|
|
try {
|
|
|
|
if (optionValue != null) {
|
|
|
|
const numVal = parseInt(optionValue);
|
|
|
|
return !Number.isNaN(numVal) ? numVal : defaultValue;
|
|
|
|
}
|
|
|
|
return defaultValue;
|
|
|
|
} catch {
|
|
|
|
return defaultValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static convertStringOption(optionValue: any, defaultValue: string) {
|
|
|
|
return optionValue != null ? String(optionValue) : defaultValue;
|
|
|
|
}
|
2018-05-16 17:17:40 +02:00
|
|
|
}
|