2018-05-15 03:19:49 +02:00
|
|
|
import * as program from "commander";
|
2018-05-17 21:33:36 +02:00
|
|
|
import * as fs from "fs";
|
|
|
|
import * as path from "path";
|
2018-05-15 03:19:49 +02:00
|
|
|
|
2021-06-07 19:25:55 +02:00
|
|
|
import { ApiService } from "jslib-common/abstractions/api.service";
|
|
|
|
import { CipherService } from "jslib-common/abstractions/cipher.service";
|
|
|
|
import { CryptoService } from "jslib-common/abstractions/crypto.service";
|
|
|
|
import { FolderService } from "jslib-common/abstractions/folder.service";
|
|
|
|
import { UserService } from "jslib-common/abstractions/user.service";
|
2018-05-15 03:19:49 +02:00
|
|
|
|
2021-06-07 19:25:55 +02:00
|
|
|
import { Cipher } from "jslib-common/models/export/cipher";
|
|
|
|
import { Collection } from "jslib-common/models/export/collection";
|
|
|
|
import { Folder } from "jslib-common/models/export/folder";
|
2018-12-17 16:30:06 +01:00
|
|
|
|
2021-06-07 19:25:55 +02:00
|
|
|
import { CollectionRequest } from "jslib-common/models/request/collectionRequest";
|
|
|
|
import { SelectionReadOnlyRequest } from "jslib-common/models/request/selectionReadOnlyRequest";
|
2019-09-25 22:08:56 +02:00
|
|
|
|
2021-06-07 19:25:55 +02:00
|
|
|
import { Response } from "jslib-node/cli/models/response";
|
2019-03-16 03:34:59 +01:00
|
|
|
|
2018-05-19 20:53:28 +02:00
|
|
|
import { CipherResponse } from "../models/response/cipherResponse";
|
|
|
|
import { FolderResponse } from "../models/response/folderResponse";
|
2019-09-25 22:08:56 +02:00
|
|
|
import { OrganizationCollectionResponse } from "../models/response/organizationCollectionResponse";
|
|
|
|
|
|
|
|
import { OrganizationCollectionRequest } from "../models/request/organizationCollectionRequest";
|
2018-05-15 03:19:49 +02:00
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
import { CliUtils } from "../utils";
|
|
|
|
|
2021-06-07 19:25:55 +02:00
|
|
|
import { Utils } from "jslib-common/misc/utils";
|
2019-10-01 17:16:24 +02:00
|
|
|
|
2018-05-15 03:19:49 +02:00
|
|
|
export class CreateCommand {
|
2018-05-18 16:55:50 +02:00
|
|
|
constructor(
|
|
|
|
private cipherService: CipherService,
|
|
|
|
private folderService: FolderService,
|
2019-09-25 22:08:56 +02:00
|
|
|
private userService: UserService,
|
|
|
|
private cryptoService: CryptoService,
|
|
|
|
private apiService: ApiService
|
|
|
|
) {}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-05-17 04:40:48 +02:00
|
|
|
async run(object: string, requestJson: string, cmd: program.Command): Promise<Response> {
|
2018-05-17 21:33:36 +02:00
|
|
|
let req: any = null;
|
|
|
|
if (object !== "attachment") {
|
|
|
|
if (requestJson == null || requestJson === "") {
|
|
|
|
requestJson = await CliUtils.readStdin();
|
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-05-17 21:33:36 +02:00
|
|
|
if (requestJson == null || requestJson === "") {
|
|
|
|
return Response.badRequest("`requestJson` was not provided.");
|
2018-05-15 16:50:06 +02:00
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2018-05-15 03:19:49 +02:00
|
|
|
try {
|
2019-10-01 17:16:24 +02:00
|
|
|
const reqJson = Buffer.from(requestJson, "base64").toString();
|
|
|
|
req = JSON.parse(reqJson);
|
|
|
|
} catch (e) {
|
2018-05-15 03:19:49 +02:00
|
|
|
return Response.badRequest("Error parsing the encoded request data.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (object.toLowerCase()) {
|
2021-12-20 18:04:00 +01:00
|
|
|
case "item":
|
2018-05-15 03:19:49 +02:00
|
|
|
return await this.createCipher(req);
|
2018-05-17 21:33:36 +02:00
|
|
|
case "attachment":
|
|
|
|
return await this.createAttachment(cmd);
|
2018-05-15 03:19:49 +02:00
|
|
|
case "folder":
|
|
|
|
return await this.createFolder(req);
|
2019-09-25 22:08:56 +02:00
|
|
|
case "org-collection":
|
2019-10-01 17:16:24 +02:00
|
|
|
return await this.createOrganizationCollection(req, cmd);
|
2021-12-20 18:04:00 +01:00
|
|
|
default:
|
2019-10-01 17:16:24 +02:00
|
|
|
return Response.badRequest("Unknown object.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-15 18:18:47 +02:00
|
|
|
private async createCipher(req: Cipher) {
|
|
|
|
const cipher = await this.cipherService.encrypt(Cipher.toView(req));
|
2018-05-15 03:19:49 +02:00
|
|
|
try {
|
|
|
|
await this.cipherService.saveWithServer(cipher);
|
2018-05-19 20:53:28 +02:00
|
|
|
const newCipher = await this.cipherService.get(cipher.id);
|
|
|
|
const decCipher = await newCipher.decrypt();
|
|
|
|
const res = new CipherResponse(decCipher);
|
|
|
|
return Response.success(res);
|
2018-05-15 03:19:49 +02:00
|
|
|
} catch (e) {
|
2018-05-15 17:08:55 +02:00
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-03 18:44:33 +01:00
|
|
|
private async createAttachment(options: program.OptionValues) {
|
|
|
|
if (options.itemid == null || options.itemid === "") {
|
2018-05-17 21:33:36 +02:00
|
|
|
return Response.badRequest("--itemid <itemid> required.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2021-02-03 18:44:33 +01:00
|
|
|
if (options.file == null || options.file === "") {
|
2018-05-17 21:33:36 +02:00
|
|
|
return Response.badRequest("--file <file> required.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2021-02-03 18:44:33 +01:00
|
|
|
const filePath = path.resolve(options.file);
|
|
|
|
if (!fs.existsSync(options.file)) {
|
2018-05-17 21:33:36 +02:00
|
|
|
return Response.badRequest("Cannot find file at " + filePath);
|
2018-05-15 03:19:49 +02:00
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2021-02-03 18:44:33 +01:00
|
|
|
const itemId = options.itemid.toLowerCase();
|
2018-05-17 21:55:44 +02:00
|
|
|
const cipher = await this.cipherService.get(itemId);
|
2018-05-17 21:33:36 +02:00
|
|
|
if (cipher == null) {
|
|
|
|
return Response.notFound();
|
2018-05-15 03:19:49 +02:00
|
|
|
}
|
|
|
|
|
2018-08-29 15:22:24 +02:00
|
|
|
if (cipher.organizationId == null && !(await this.userService.canAccessPremium())) {
|
|
|
|
return Response.error("Premium status is required to use this feature.");
|
2018-05-17 21:33:36 +02:00
|
|
|
}
|
|
|
|
|
2018-05-15 18:18:47 +02:00
|
|
|
const encKey = await this.cryptoService.getEncKey();
|
2018-05-15 03:19:49 +02:00
|
|
|
if (encKey == null) {
|
2018-05-15 17:08:55 +02:00
|
|
|
return Response.error(
|
2018-05-18 16:55:50 +02:00
|
|
|
"You must update your encryption key before you can use this feature. " +
|
|
|
|
"See https://help.bitwarden.com/article/update-encryption-key/"
|
2018-05-15 17:08:55 +02:00
|
|
|
);
|
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 {
|
2021-02-03 18:44:33 +01:00
|
|
|
const fileBuf = fs.readFileSync(filePath);
|
|
|
|
await this.cipherService.saveAttachmentRawWithServer(
|
2021-12-20 18:04:00 +01:00
|
|
|
cipher,
|
2021-02-03 18:44:33 +01:00
|
|
|
path.basename(filePath),
|
2018-05-17 21:33:36 +02:00
|
|
|
new Uint8Array(fileBuf).buffer
|
2021-12-20 18:04:00 +01:00
|
|
|
);
|
2021-02-03 18:44:33 +01:00
|
|
|
const updatedCipher = await this.cipherService.get(cipher.id);
|
|
|
|
const decCipher = await updatedCipher.decrypt();
|
|
|
|
const res = new CipherResponse(decCipher);
|
|
|
|
return Response.success(res);
|
2018-05-17 21:33:36 +02:00
|
|
|
} catch (e) {
|
2021-02-03 18:44:33 +01:00
|
|
|
return Response.error(e);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-03 18:44:33 +01:00
|
|
|
private async createFolder(req: Folder) {
|
|
|
|
const folder = await this.folderService.encrypt(Folder.toView(req));
|
|
|
|
try {
|
2018-05-15 03:19:49 +02:00
|
|
|
await this.folderService.saveWithServer(folder);
|
2021-02-03 18:44:33 +01:00
|
|
|
const newFolder = await this.folderService.get(folder.id);
|
|
|
|
const decFolder = await newFolder.decrypt();
|
2019-10-01 17:16:24 +02:00
|
|
|
const res = new FolderResponse(decFolder);
|
|
|
|
return Response.success(res);
|
2021-02-03 18:44:33 +01: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 createOrganizationCollection(
|
|
|
|
req: OrganizationCollectionRequest,
|
|
|
|
options: program.OptionValues
|
2021-12-20 18:04:00 +01:00
|
|
|
) {
|
2021-02-03 18:44:33 +01:00
|
|
|
if (options.organizationid == null || options.organizationid === "") {
|
2019-10-01 17:16:24 +02:00
|
|
|
return Response.badRequest("--organizationid <organizationid> required.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2021-02-03 18:44:33 +01:00
|
|
|
if (!Utils.isGuid(options.organizationid)) {
|
2019-10-01 17:16:24 +02:00
|
|
|
return Response.error("`" + options.organizationid + "` is not a GUID.");
|
|
|
|
}
|
2021-02-03 18:44:33 +01:00
|
|
|
if (options.organizationid !== req.organizationId) {
|
2019-10-01 17:16:24 +02:00
|
|
|
return Response.error("--organizationid <organizationid> does not match request object.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2019-09-25 22:08:56 +02:00
|
|
|
try {
|
|
|
|
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
|
|
|
|
2019-09-25 22:08:56 +02:00
|
|
|
const groups =
|
|
|
|
req.groups == null
|
|
|
|
? null
|
2021-02-04 05:51:59 +01:00
|
|
|
: req.groups.map((g) => new SelectionReadOnlyRequest(g.id, g.readOnly, g.hidePasswords));
|
2019-09-25 22:08:56 +02:00
|
|
|
const request = new CollectionRequest();
|
|
|
|
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.postCollection(req.organizationId, request);
|
|
|
|
const view = Collection.toView(req);
|
|
|
|
view.id = response.id;
|
|
|
|
const res = new OrganizationCollectionResponse(view, groups);
|
2019-09-25 22:08:56 +02:00
|
|
|
return Response.success(res);
|
|
|
|
} catch (e) {
|
|
|
|
return Response.error(e);
|
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2018-05-15 03:19:49 +02:00
|
|
|
}
|