1
0
mirror of https://github.com/bitwarden/browser.git synced 2024-10-30 08:10:34 +01:00
bitwarden-browser/apps/cli/src/commands/create.command.ts

207 lines
7.5 KiB
TypeScript
Raw Normal View History

2021-12-20 18:04:00 +01:00
import * as fs from "fs";
import * as path from "path";
2018-05-15 03:19:49 +02:00
2022-06-14 17:10:53 +02:00
import { ApiService } from "@bitwarden/common/abstractions/api.service";
import { CipherService } from "@bitwarden/common/abstractions/cipher.service";
import { CryptoService } from "@bitwarden/common/abstractions/crypto.service";
import { FolderService } from "@bitwarden/common/abstractions/folder.service";
import { StateService } from "@bitwarden/common/abstractions/state.service";
import { Utils } from "@bitwarden/common/misc/utils";
import { CipherExport } from "@bitwarden/common/models/export/cipherExport";
import { CollectionExport } from "@bitwarden/common/models/export/collectionExport";
import { FolderExport } from "@bitwarden/common/models/export/folderExport";
import { CollectionRequest } from "@bitwarden/common/models/request/collectionRequest";
import { SelectionReadOnlyRequest } from "@bitwarden/common/models/request/selectionReadOnlyRequest";
import { Response } from "@bitwarden/node/cli/models/response";
2019-03-16 03:34:59 +01:00
2022-03-03 18:24:41 +01:00
import { OrganizationCollectionRequest } from "../models/request/organizationCollectionRequest";
2021-12-20 18:04:00 +01:00
import { CipherResponse } from "../models/response/cipherResponse";
import { FolderResponse } from "../models/response/folderResponse";
import { OrganizationCollectionResponse } from "../models/response/organizationCollectionResponse";
import { CliUtils } from "../utils";
2018-05-15 03:19:49 +02:00
export class CreateCommand {
2021-12-20 18:04:00 +01:00
constructor(
private cipherService: CipherService,
private folderService: FolderService,
private stateService: StateService,
2021-12-20 18:04:00 +01:00
private cryptoService: CryptoService,
private apiService: ApiService
) {}
2022-01-19 16:45:14 +01:00
async run(
object: string,
requestJson: string,
cmdOptions: Record<string, any>,
additionalData: any = null
): Promise<Response> {
2021-12-20 18:04:00 +01:00
let req: any = null;
if (object !== "attachment") {
2022-01-19 16:45:14 +01:00
if (process.env.BW_SERVE !== "true" && (requestJson == null || requestJson === "")) {
2021-12-20 18:04:00 +01:00
requestJson = await CliUtils.readStdin();
}
if (requestJson == null || requestJson === "") {
return Response.badRequest("`requestJson` was not provided.");
}
2022-01-19 16:45:14 +01:00
if (typeof requestJson !== "string") {
req = requestJson;
} else {
try {
const reqJson = Buffer.from(requestJson, "base64").toString();
req = JSON.parse(reqJson);
} catch (e) {
return Response.badRequest("Error parsing the encoded request data.");
}
2021-12-20 18:04:00 +01:00
}
2018-05-15 03:19:49 +02:00
}
2022-01-19 16:45:14 +01:00
const normalizedOptions = new Options(cmdOptions);
2021-12-20 18:04:00 +01:00
switch (object.toLowerCase()) {
case "item":
return await this.createCipher(req);
case "attachment":
2022-01-19 16:45:14 +01:00
return await this.createAttachment(normalizedOptions, additionalData);
2021-12-20 18:04:00 +01:00
case "folder":
return await this.createFolder(req);
case "org-collection":
2022-01-19 16:45:14 +01:00
return await this.createOrganizationCollection(req, normalizedOptions);
2021-12-20 18:04:00 +01:00
default:
return Response.badRequest("Unknown object.");
}
}
private async createCipher(req: CipherExport) {
const cipher = await this.cipherService.encrypt(CipherExport.toView(req));
2021-12-20 18:04:00 +01:00
try {
await this.cipherService.saveWithServer(cipher);
const newCipher = await this.cipherService.get(cipher.id);
const decCipher = await newCipher.decrypt();
const res = new CipherResponse(decCipher);
return Response.success(res);
} catch (e) {
return Response.error(e);
}
}
2022-01-19 16:45:14 +01:00
private async createAttachment(options: Options, additionalData: any) {
if (options.itemId == null || options.itemId === "") {
return Response.badRequest("`itemid` option is required.");
2021-12-20 18:04:00 +01:00
}
2022-01-19 16:45:14 +01:00
let fileBuf: Buffer = null;
let fileName: string = null;
if (process.env.BW_SERVE === "true") {
fileBuf = additionalData.fileBuffer;
fileName = additionalData.fileName;
} else {
if (options.file == null || options.file === "") {
return Response.badRequest("`file` option is required.");
}
const filePath = path.resolve(options.file);
if (!fs.existsSync(options.file)) {
return Response.badRequest("Cannot find file at " + filePath);
}
fileBuf = fs.readFileSync(filePath);
fileName = path.basename(filePath);
}
if (fileBuf == null) {
return Response.badRequest("File not provided.");
2021-12-20 18:04:00 +01:00
}
2022-01-19 16:45:14 +01:00
if (fileName == null || fileName.trim() === "") {
return Response.badRequest("File name not provided.");
2021-12-20 18:04:00 +01:00
}
2022-01-19 16:45:14 +01:00
const itemId = options.itemId.toLowerCase();
2021-12-20 18:04:00 +01:00
const cipher = await this.cipherService.get(itemId);
if (cipher == null) {
return Response.notFound();
2018-05-15 03:19:49 +02:00
}
if (cipher.organizationId == null && !(await this.stateService.getCanAccessPremium())) {
2021-12-20 18:04:00 +01:00
return Response.error("Premium status is required to use this feature.");
2018-05-17 21:33:36 +02:00
}
2021-12-20 18:04:00 +01:00
const encKey = await this.cryptoService.getEncKey();
if (encKey == null) {
return Response.error(
"You must update your encryption key before you can use this feature. " +
"See https://help.bitwarden.com/article/update-encryption-key/"
);
2018-05-15 03:19:49 +02:00
}
2019-09-25 22:08:56 +02:00
2021-12-20 18:04:00 +01:00
try {
await this.cipherService.saveAttachmentRawWithServer(
cipher,
2022-01-19 16:45:14 +01:00
fileName,
2021-12-20 18:04:00 +01:00
new Uint8Array(fileBuf).buffer
);
const updatedCipher = await this.cipherService.get(cipher.id);
const decCipher = await updatedCipher.decrypt();
return Response.success(new CipherResponse(decCipher));
2021-12-20 18:04:00 +01:00
} catch (e) {
return Response.error(e);
}
}
private async createFolder(req: FolderExport) {
const folder = await this.folderService.encrypt(FolderExport.toView(req));
2021-12-20 18:04:00 +01:00
try {
await this.folderService.saveWithServer(folder);
const newFolder = await this.folderService.get(folder.id);
const decFolder = await newFolder.decrypt();
const res = new FolderResponse(decFolder);
return Response.success(res);
} catch (e) {
return Response.error(e);
}
}
2022-01-19 16:45:14 +01:00
private async createOrganizationCollection(req: OrganizationCollectionRequest, options: Options) {
if (options.organizationId == null || options.organizationId === "") {
return Response.badRequest("`organizationid` option is required.");
2021-12-20 18:04:00 +01:00
}
2022-01-19 16:45:14 +01:00
if (!Utils.isGuid(options.organizationId)) {
return Response.badRequest("`" + options.organizationId + "` is not a GUID.");
2021-12-20 18:04:00 +01:00
}
2022-01-19 16:45:14 +01:00
if (options.organizationId !== req.organizationId) {
return Response.badRequest("`organizationid` option does not match request object.");
2021-12-20 18:04:00 +01:00
}
try {
const orgKey = await this.cryptoService.getOrgKey(req.organizationId);
if (orgKey == null) {
throw new Error("No encryption key for this organization.");
}
const groups =
req.groups == null
? null
: req.groups.map((g) => new SelectionReadOnlyRequest(g.id, g.readOnly, g.hidePasswords));
const request = new CollectionRequest();
request.name = (await this.cryptoService.encrypt(req.name, orgKey)).encryptedString;
request.externalId = req.externalId;
request.groups = groups;
const response = await this.apiService.postCollection(req.organizationId, request);
const view = CollectionExport.toView(req);
2021-12-20 18:04:00 +01:00
view.id = response.id;
const res = new OrganizationCollectionResponse(view, groups);
return Response.success(res);
} catch (e) {
return Response.error(e);
2019-09-25 22:08:56 +02:00
}
2021-12-20 18:04:00 +01:00
}
2018-05-15 03:19:49 +02:00
}
2022-01-19 16:45:14 +01:00
class Options {
itemId: string;
organizationId: string;
file: string;
constructor(passedOptions: Record<string, any>) {
this.organizationId = passedOptions?.organizationid || passedOptions?.organizationId;
this.itemId = passedOptions?.itemid || passedOptions?.itemId;
this.file = passedOptions?.file;
2022-01-19 16:45:14 +01:00
}
}