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 { 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";
|
2018-05-19 20:53:28 +02:00
|
|
|
import { CipherResponse } from "../models/response/cipherResponse";
|
|
|
|
import { FolderResponse } from "../models/response/folderResponse";
|
2019-10-01 17:16:24 +02:00
|
|
|
import { OrganizationCollectionResponse } from "../models/response/organizationCollectionResponse";
|
2018-05-17 05:23:12 +02:00
|
|
|
import { CliUtils } from "../utils";
|
2018-05-17 04:40:48 +02:00
|
|
|
|
2018-05-15 17:30:56 +02:00
|
|
|
export class EditCommand {
|
2019-10-01 17:16:24 +02:00
|
|
|
constructor(
|
2018-05-17 04:40:48 +02:00
|
|
|
private cipherService: CipherService,
|
2018-05-15 17:30:56 +02:00
|
|
|
private folderService: FolderService,
|
2018-05-15 18:18:47 +02:00
|
|
|
private cryptoService: CryptoService,
|
2018-05-15 17:30:56 +02:00
|
|
|
private apiService: ApiService
|
2018-10-23 23:31:59 +02:00
|
|
|
) {}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-10-23 23:31:59 +02:00
|
|
|
async run(
|
2018-10-24 04:56:15 +02:00
|
|
|
object: string,
|
2018-05-15 18:18:47 +02:00
|
|
|
id: string,
|
2022-01-19 16:45:14 +01:00
|
|
|
requestJson: any,
|
|
|
|
cmdOptions: Record<string, any>
|
2019-10-01 17:16:24 +02:00
|
|
|
): Promise<Response> {
|
2022-01-19 16:45:14 +01:00
|
|
|
if (process.env.BW_SERVE !== "true" && (requestJson == null || requestJson === "")) {
|
2020-10-20 15:38:11 +02:00
|
|
|
requestJson = await CliUtils.readStdin();
|
2019-10-01 17:16:24 +02:00
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
if (requestJson == null || requestJson === "") {
|
|
|
|
return Response.badRequest("`requestJson` was not provided.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2018-05-15 17:30:56 +02:00
|
|
|
let req: any = null;
|
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-16 17:54:59 +02:00
|
|
|
if (id != null) {
|
|
|
|
id = id.toLowerCase();
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2022-01-19 16:45:14 +01:00
|
|
|
const normalizedOptions = new Options(cmdOptions);
|
2018-05-15 17:30:56 +02:00
|
|
|
switch (object.toLowerCase()) {
|
2021-12-20 18:04:00 +01:00
|
|
|
case "item":
|
2018-05-15 17:30:56 +02:00
|
|
|
return await this.editCipher(id, req);
|
2018-10-23 23:31:59 +02:00
|
|
|
case "item-collections":
|
|
|
|
return await this.editCipherCollections(id, req);
|
2021-12-20 18:04:00 +01:00
|
|
|
case "folder":
|
2018-05-15 17:30:56 +02:00
|
|
|
return await this.editFolder(id, req);
|
2019-10-01 17:16:24 +02:00
|
|
|
case "org-collection":
|
2022-01-19 16:45:14 +01:00
|
|
|
return await this.editOrganizationCollection(id, req, normalizedOptions);
|
2021-12-20 18:04:00 +01:00
|
|
|
default:
|
2018-05-15 17:30:56 +02:00
|
|
|
return Response.badRequest("Unknown object.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-16 03:58:36 +02:00
|
|
|
private async editCipher(id: string, req: CipherExport) {
|
2018-05-15 17:30:56 +02:00
|
|
|
const cipher = await this.cipherService.get(id);
|
|
|
|
if (cipher == null) {
|
|
|
|
return Response.notFound();
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2018-05-15 17:30:56 +02:00
|
|
|
let cipherView = await cipher.decrypt();
|
2020-04-14 19:04:19 +02:00
|
|
|
if (cipherView.isDeleted) {
|
2022-01-19 16:45:14 +01:00
|
|
|
return Response.badRequest("You may not edit a deleted item. Use the restore command first.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2022-05-16 03:58:36 +02:00
|
|
|
cipherView = CipherExport.toView(req, cipherView);
|
2018-05-15 17:30:56 +02:00
|
|
|
const encCipher = await this.cipherService.encrypt(cipherView);
|
2021-12-20 18:04:00 +01:00
|
|
|
try {
|
2018-05-15 17:30:56 +02:00
|
|
|
await this.cipherService.saveWithServer(encCipher);
|
2018-10-23 23:31:59 +02:00
|
|
|
const updatedCipher = await this.cipherService.get(cipher.id);
|
2018-05-19 20:53:28 +02:00
|
|
|
const decCipher = await updatedCipher.decrypt();
|
|
|
|
const res = new CipherResponse(decCipher);
|
|
|
|
return Response.success(res);
|
2018-05-15 17:30:56 +02:00
|
|
|
} catch (e) {
|
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-23 23:31:59 +02:00
|
|
|
private async editCipherCollections(id: string, req: string[]) {
|
|
|
|
const cipher = await this.cipherService.get(id);
|
|
|
|
if (cipher == null) {
|
2018-05-15 17:30:56 +02:00
|
|
|
return Response.notFound();
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2018-10-24 01:04:32 +02:00
|
|
|
if (cipher.organizationId == null) {
|
2018-10-24 04:56:15 +02:00
|
|
|
return Response.badRequest(
|
2022-01-19 16:45:14 +01:00
|
|
|
"Item does not belong to an organization. Consider moving it first."
|
2021-12-20 18:04:00 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-10-23 23:31:59 +02:00
|
|
|
cipher.collectionIds = req;
|
2021-12-20 18:04:00 +01:00
|
|
|
try {
|
2018-10-23 23:31:59 +02:00
|
|
|
await this.cipherService.saveCollectionsWithServer(cipher);
|
|
|
|
const updatedCipher = await this.cipherService.get(cipher.id);
|
2018-05-19 20:53:28 +02:00
|
|
|
const decCipher = await updatedCipher.decrypt();
|
|
|
|
const res = new CipherResponse(decCipher);
|
|
|
|
return Response.success(res);
|
2018-05-15 17:30:56 +02:00
|
|
|
} catch (e) {
|
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-16 03:58:36 +02:00
|
|
|
private async editFolder(id: string, req: FolderExport) {
|
2018-05-15 17:30:56 +02:00
|
|
|
const folder = await this.folderService.get(id);
|
|
|
|
if (folder == null) {
|
|
|
|
return Response.notFound();
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2018-05-15 17:30:56 +02:00
|
|
|
let folderView = await folder.decrypt();
|
2022-05-16 03:58:36 +02:00
|
|
|
folderView = FolderExport.toView(req, folderView);
|
2018-05-15 17:30:56 +02:00
|
|
|
const encFolder = await this.folderService.encrypt(folderView);
|
2021-12-20 18:04:00 +01:00
|
|
|
try {
|
2018-05-15 17:30:56 +02:00
|
|
|
await this.folderService.saveWithServer(encFolder);
|
2018-05-19 20:53:28 +02:00
|
|
|
const updatedFolder = await this.folderService.get(folder.id);
|
|
|
|
const decFolder = await updatedFolder.decrypt();
|
|
|
|
const res = new FolderResponse(decFolder);
|
|
|
|
return Response.success(res);
|
2018-05-15 17:30:56 +02:00
|
|
|
} catch (e) {
|
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-03 18:44:33 +01:00
|
|
|
private async editOrganizationCollection(
|
2018-05-17 04:40:48 +02:00
|
|
|
id: string,
|
2021-02-03 18:44:33 +01:00
|
|
|
req: OrganizationCollectionRequest,
|
2022-01-19 16:45:14 +01:00
|
|
|
options: Options
|
2021-12-20 18:04:00 +01:00
|
|
|
) {
|
2022-01-19 16:45:14 +01:00
|
|
|
if (options.organizationId == null || options.organizationId === "") {
|
|
|
|
return Response.badRequest("`organizationid` option is required.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2019-10-01 17:16:24 +02:00
|
|
|
if (!Utils.isGuid(id)) {
|
2022-01-19 16:45:14 +01:00
|
|
|
return Response.badRequest("`" + id + "` is not a GUID.");
|
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 {
|
2019-10-01 17:16:24 +02:00
|
|
|
const orgKey = await this.cryptoService.getOrgKey(req.organizationId);
|
|
|
|
if (orgKey == null) {
|
|
|
|
throw new Error("No encryption key for this organization.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2020-10-20 15:38:11 +02:00
|
|
|
const groups =
|
2019-10-01 17:16:24 +02:00
|
|
|
req.groups == null
|
2021-12-20 18:04:00 +01:00
|
|
|
? null
|
2021-02-04 05:51:59 +01:00
|
|
|
: req.groups.map((g) => new SelectionReadOnlyRequest(g.id, g.readOnly, g.hidePasswords));
|
2020-10-20 15:38:11 +02:00
|
|
|
const request = new CollectionRequest();
|
2019-10-01 17:16:24 +02:00
|
|
|
request.name = (await this.cryptoService.encrypt(req.name, orgKey)).encryptedString;
|
|
|
|
request.externalId = req.externalId;
|
|
|
|
request.groups = groups;
|
2020-10-20 15:38:11 +02:00
|
|
|
const response = await this.apiService.putCollection(req.organizationId, id, request);
|
2022-05-16 03:58:36 +02:00
|
|
|
const view = CollectionExport.toView(req);
|
2020-10-20 15:38:11 +02:00
|
|
|
view.id = response.id;
|
|
|
|
const res = new OrganizationCollectionResponse(view, groups);
|
2019-10-01 17:16:24 +02:00
|
|
|
return Response.success(res);
|
|
|
|
} catch (e) {
|
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
2018-05-15 17:30:56 +02:00
|
|
|
}
|
2022-01-19 16:45:14 +01:00
|
|
|
|
|
|
|
class Options {
|
|
|
|
organizationId: string;
|
|
|
|
|
|
|
|
constructor(passedOptions: Record<string, any>) {
|
2022-01-26 17:28:56 +01:00
|
|
|
this.organizationId = passedOptions?.organizationid || passedOptions?.organizationId;
|
2022-01-19 16:45:14 +01:00
|
|
|
}
|
|
|
|
}
|