bitwarden-browser/libs/common/src/services/export.service.ts

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

440 lines
13 KiB
TypeScript
Raw Normal View History

2018-05-17 16:52:06 +02:00
import * as papa from "papaparse";
2018-07-05 20:39:58 +02:00
import { ApiService } from "../abstractions/api.service";
2018-05-17 16:52:06 +02:00
import { CipherService } from "../abstractions/cipher.service";
import { CryptoService } from "../abstractions/crypto.service";
import { CryptoFunctionService } from "../abstractions/cryptoFunction.service";
import {
ExportFormat,
ExportService as ExportServiceAbstraction,
} from "../abstractions/export.service";
import { FolderService } from "../abstractions/folder/folder.service.abstraction";
2022-02-22 15:39:11 +01:00
import { CipherType } from "../enums/cipherType";
import { DEFAULT_PBKDF2_ITERATIONS, KdfType } from "../enums/kdfType";
2022-02-22 15:39:11 +01:00
import { Utils } from "../misc/utils";
import { CipherData } from "../models/data/cipher.data";
import { CollectionData } from "../models/data/collection.data";
2018-07-05 20:39:58 +02:00
import { Cipher } from "../models/domain/cipher";
import { Collection } from "../models/domain/collection";
import { Folder } from "../models/domain/folder";
import { CipherWithIdExport as CipherExport } from "../models/export/cipher-with-ids.export";
import { CollectionWithIdExport as CollectionExport } from "../models/export/collection-with-id.export";
import { EventExport } from "../models/export/event.export";
import { FolderWithIdExport as FolderExport } from "../models/export/folder-with-id.export";
import { CollectionDetailsResponse } from "../models/response/collection.response";
import { CipherView } from "../models/view/cipher.view";
import { CollectionView } from "../models/view/collection.view";
import { EventView } from "../models/view/event.view";
import { FolderView } from "../models/view/folder.view";
2018-05-17 16:52:06 +02:00
export class ExportService implements ExportServiceAbstraction {
2018-07-05 20:39:58 +02:00
constructor(
private folderService: FolderService,
private cipherService: CipherService,
private apiService: ApiService,
private cryptoService: CryptoService,
private cryptoFunctionService: CryptoFunctionService
) {}
2021-12-16 13:36:21 +01:00
async getExport(format: ExportFormat = "csv", organizationId?: string): Promise<string> {
if (organizationId) {
return await this.getOrganizationExport(organizationId, format);
}
if (format === "encrypted_json") {
return this.getEncryptedExport();
} else {
return this.getDecryptedExport(format);
}
2021-12-16 13:36:21 +01:00
}
async getPasswordProtectedExport(password: string, organizationId?: string): Promise<string> {
const clearText = organizationId
? await this.getOrganizationExport(organizationId, "json")
: await this.getExport("json");
const salt = Utils.fromBufferToB64(await this.cryptoFunctionService.randomBytes(16));
const kdfIterations = DEFAULT_PBKDF2_ITERATIONS;
const key = await this.cryptoService.makePinKey(
password,
salt,
KdfType.PBKDF2_SHA256,
kdfIterations
);
const encKeyValidation = await this.cryptoService.encrypt(Utils.newGuid(), key);
const encText = await this.cryptoService.encrypt(clearText, key);
const jsonDoc: any = {
encrypted: true,
passwordProtected: true,
salt: salt,
kdfIterations: kdfIterations,
kdfType: KdfType.PBKDF2_SHA256,
encKeyValidation_DO_NOT_EDIT: encKeyValidation.encryptedString,
data: encText.encryptedString,
};
return JSON.stringify(jsonDoc, null, " ");
}
async getOrganizationExport(
organizationId: string,
format: ExportFormat = "csv"
): Promise<string> {
if (format === "encrypted_json") {
return this.getOrganizationEncryptedExport(organizationId);
} else {
return this.getOrganizationDecryptedExport(organizationId, format);
}
2021-12-16 13:36:21 +01:00
}
async getEventExport(events: EventView[]): Promise<string> {
2022-04-19 13:03:04 +02:00
return papa.unparse(events.map((e) => new EventExport(e)));
2021-12-16 13:36:21 +01:00
}
2022-02-22 15:39:11 +01:00
getFileName(prefix: string = null, extension = "csv"): string {
const now = new Date();
const dateString =
now.getFullYear() +
2021-12-16 13:36:21 +01:00
"" +
this.padNumber(now.getMonth() + 1, 2) +
2021-12-16 13:36:21 +01:00
"" +
this.padNumber(now.getDate(), 2) +
this.padNumber(now.getHours(), 2) +
2021-12-16 13:36:21 +01:00
"" +
this.padNumber(now.getMinutes(), 2) +
this.padNumber(now.getSeconds(), 2);
2021-12-16 13:36:21 +01:00
return "bitwarden" + (prefix ? "_" + prefix : "") + "_export_" + dateString + "." + extension;
2021-12-16 13:36:21 +01:00
}
private async getDecryptedExport(format: "json" | "csv"): Promise<string> {
2018-05-17 16:52:06 +02:00
let decFolders: FolderView[] = [];
let decCiphers: CipherView[] = [];
const promises = [];
2021-12-16 13:36:21 +01:00
promises.push(
this.folderService.getAllDecryptedFromState().then((folders) => {
2018-05-17 16:52:06 +02:00
decFolders = folders;
2021-12-16 13:36:21 +01:00
})
);
promises.push(
this.cipherService.getAllDecrypted().then((ciphers) => {
decCiphers = ciphers.filter((f) => f.deletedDate == null);
2021-12-16 13:36:21 +01:00
})
);
2018-05-17 16:52:06 +02:00
await Promise.all(promises);
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
if (format === "csv") {
2018-12-17 16:32:02 +01:00
const foldersMap = new Map<string, FolderView>();
decFolders.forEach((f) => {
if (f.id != null) {
foldersMap.set(f.id, f);
2021-12-16 13:36:21 +01:00
}
});
const exportCiphers: any[] = [];
decCiphers.forEach((c) => {
// only export logins and secure notes
if (c.type !== CipherType.Login && c.type !== CipherType.SecureNote) {
return;
2021-12-16 13:36:21 +01:00
}
2018-12-17 16:32:02 +01:00
if (c.organizationId != null) {
2021-12-16 13:36:21 +01:00
return;
}
const cipher: any = {};
cipher.folder =
c.folderId != null && foldersMap.has(c.folderId) ? foldersMap.get(c.folderId).name : null;
2018-12-17 16:32:02 +01:00
cipher.favorite = c.favorite ? 1 : null;
this.buildCommonCipher(cipher, c);
2018-12-17 16:32:02 +01:00
exportCiphers.push(cipher);
2021-12-16 13:36:21 +01:00
});
return papa.unparse(exportCiphers);
2021-12-16 13:36:21 +01:00
} else {
const jsonDoc: any = {
encrypted: false,
2018-12-17 16:32:02 +01:00
folders: [],
items: [],
2021-12-16 13:36:21 +01:00
};
decFolders.forEach((f) => {
if (f.id == null) {
2021-12-16 13:36:21 +01:00
return;
}
const folder = new FolderExport();
2018-12-17 16:32:02 +01:00
folder.build(f);
jsonDoc.folders.push(folder);
2021-12-16 13:36:21 +01:00
});
decCiphers.forEach((c) => {
if (c.organizationId != null) {
return;
2021-12-16 13:36:21 +01:00
}
const cipher = new CipherExport();
cipher.build(c);
cipher.collectionIds = null;
jsonDoc.items.push(cipher);
2021-12-16 13:36:21 +01:00
});
return JSON.stringify(jsonDoc, null, " ");
}
2021-12-16 13:36:21 +01:00
}
private async getEncryptedExport(): Promise<string> {
2018-05-17 16:52:06 +02:00
let folders: Folder[] = [];
let ciphers: Cipher[] = [];
const promises = [];
2021-12-16 13:36:21 +01:00
promises.push(
this.folderService.getAllFromState().then((f) => {
2018-05-17 16:52:06 +02:00
folders = f;
})
);
2021-12-16 13:36:21 +01:00
promises.push(
this.cipherService.getAll().then((c) => {
ciphers = c.filter((f) => f.deletedDate == null);
2018-05-17 16:52:06 +02:00
})
);
2021-12-16 13:36:21 +01:00
2018-05-17 16:52:06 +02:00
await Promise.all(promises);
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
const encKeyValidation = await this.cryptoService.encrypt(Utils.newGuid());
2021-12-16 13:36:21 +01:00
const jsonDoc: any = {
encrypted: true,
encKeyValidation_DO_NOT_EDIT: encKeyValidation.encryptedString,
folders: [],
items: [],
2021-12-16 13:36:21 +01:00
};
2018-07-05 20:39:58 +02:00
folders.forEach((f) => {
if (f.id == null) {
2021-12-16 13:36:21 +01:00
return;
}
2018-12-17 16:32:02 +01:00
const folder = new FolderExport();
folder.build(f);
jsonDoc.folders.push(folder);
2018-12-17 16:32:02 +01:00
});
2021-12-16 13:36:21 +01:00
ciphers.forEach((c) => {
2018-12-17 16:32:02 +01:00
if (c.organizationId != null) {
return;
}
const cipher = new CipherExport();
cipher.build(c);
cipher.collectionIds = null;
jsonDoc.items.push(cipher);
});
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
return JSON.stringify(jsonDoc, null, " ");
2021-12-16 13:36:21 +01:00
}
2018-07-05 20:39:58 +02:00
private async getOrganizationDecryptedExport(
organizationId: string,
format: "json" | "csv"
): Promise<string> {
2018-07-05 20:39:58 +02:00
const decCollections: CollectionView[] = [];
const decCiphers: CipherView[] = [];
const promises = [];
2021-12-16 13:36:21 +01:00
promises.push(
this.apiService.getOrganizationExport(organizationId).then((exportData) => {
const exportPromises: any = [];
if (exportData != null) {
if (exportData.collections != null && exportData.collections.length > 0) {
exportData.collections.forEach((c) => {
const collection = new Collection(new CollectionData(c as CollectionDetailsResponse));
exportPromises.push(
collection.decrypt().then((decCol) => {
decCollections.push(decCol);
2021-12-16 13:36:21 +01:00
})
);
2018-12-17 16:32:02 +01:00
});
}
if (exportData.ciphers != null && exportData.ciphers.length > 0) {
exportData.ciphers
.filter((c) => c.deletedDate === null)
.forEach((c) => {
const cipher = new Cipher(new CipherData(c));
exportPromises.push(
cipher.decrypt().then((decCipher) => {
decCiphers.push(decCipher);
})
);
});
}
2018-07-05 20:39:58 +02:00
}
return Promise.all(exportPromises);
2021-12-16 13:36:21 +01:00
})
);
2018-07-05 20:39:58 +02:00
await Promise.all(promises);
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
if (format === "csv") {
2018-12-17 16:32:02 +01:00
const collectionsMap = new Map<string, CollectionView>();
decCollections.forEach((c) => {
2018-12-17 16:32:02 +01:00
collectionsMap.set(c.id, c);
2021-12-16 13:36:21 +01:00
});
2018-12-17 16:32:02 +01:00
const exportCiphers: any[] = [];
decCiphers.forEach((c) => {
2018-12-17 16:32:02 +01:00
// only export logins and secure notes
if (c.type !== CipherType.Login && c.type !== CipherType.SecureNote) {
2021-12-16 13:36:21 +01:00
return;
2018-07-05 20:39:58 +02:00
}
const cipher: any = {};
cipher.collections = [];
2018-12-17 16:32:02 +01:00
if (c.collectionIds != null) {
cipher.collections = c.collectionIds
.filter((id) => collectionsMap.has(id))
.map((id) => collectionsMap.get(id).name);
2021-12-16 13:36:21 +01:00
}
this.buildCommonCipher(cipher, c);
exportCiphers.push(cipher);
2021-12-16 13:36:21 +01:00
});
return papa.unparse(exportCiphers);
} else {
const jsonDoc: any = {
encrypted: false,
collections: [],
items: [],
};
2021-12-16 13:36:21 +01:00
decCollections.forEach((c) => {
const collection = new CollectionExport();
collection.build(c);
jsonDoc.collections.push(collection);
});
2021-12-16 13:36:21 +01:00
decCiphers.forEach((c) => {
const cipher = new CipherExport();
cipher.build(c);
jsonDoc.items.push(cipher);
});
return JSON.stringify(jsonDoc, null, " ");
}
2021-12-16 13:36:21 +01:00
}
private async getOrganizationEncryptedExport(organizationId: string): Promise<string> {
2018-07-05 20:39:58 +02:00
const collections: Collection[] = [];
const ciphers: Cipher[] = [];
const promises = [];
2021-12-16 13:36:21 +01:00
promises.push(
this.apiService.getCollections(organizationId).then((c) => {
2018-07-05 20:49:48 +02:00
const collectionPromises: any = [];
2018-07-05 20:39:58 +02:00
if (c != null && c.data != null && c.data.length > 0) {
c.data.forEach((r) => {
2018-07-05 20:39:58 +02:00
const collection = new Collection(new CollectionData(r as CollectionDetailsResponse));
collections.push(collection);
2018-05-17 16:52:06 +02:00
});
}
2018-07-05 20:39:58 +02:00
return Promise.all(collectionPromises);
})
);
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
promises.push(
this.apiService.getCiphersOrganization(organizationId).then((c) => {
2018-12-17 16:32:02 +01:00
const cipherPromises: any = [];
if (c != null && c.data != null && c.data.length > 0) {
c.data
.filter((item) => item.deletedDate === null)
.forEach((item) => {
2018-12-17 16:32:02 +01:00
const cipher = new Cipher(new CipherData(item));
ciphers.push(cipher);
});
2021-12-16 13:36:21 +01:00
}
2018-07-05 20:39:58 +02:00
return Promise.all(cipherPromises);
2021-12-16 13:36:21 +01:00
})
);
await Promise.all(promises);
2021-12-16 13:36:21 +01:00
const orgKey = await this.cryptoService.getOrgKey(organizationId);
const encKeyValidation = await this.cryptoService.encrypt(Utils.newGuid(), orgKey);
2021-12-16 13:36:21 +01:00
const jsonDoc: any = {
encrypted: true,
encKeyValidation_DO_NOT_EDIT: encKeyValidation.encryptedString,
collections: [],
items: [],
2021-12-16 13:36:21 +01:00
};
collections.forEach((c) => {
2018-07-05 20:39:58 +02:00
const collection = new CollectionExport();
collection.build(c);
jsonDoc.collections.push(collection);
2021-12-16 13:36:21 +01:00
});
2018-07-05 20:39:58 +02:00
ciphers.forEach((c) => {
const cipher = new CipherExport();
cipher.build(c);
jsonDoc.items.push(cipher);
2021-12-16 13:36:21 +01:00
});
return JSON.stringify(jsonDoc, null, " ");
2021-12-16 13:36:21 +01:00
}
2022-02-22 15:39:11 +01:00
private padNumber(num: number, width: number, padCharacter = "0"): string {
2018-05-17 16:52:06 +02:00
const numString = num.toString();
return numString.length >= width
? numString
2018-07-05 20:39:58 +02:00
: new Array(width - numString.length + 1).join(padCharacter) + numString;
2021-12-16 13:36:21 +01:00
}
2018-07-05 20:39:58 +02:00
private buildCommonCipher(cipher: any, c: CipherView) {
cipher.type = null;
cipher.name = c.name;
cipher.notes = c.notes;
cipher.fields = null;
cipher.reprompt = c.reprompt;
2018-07-05 20:39:58 +02:00
// Login props
cipher.login_uri = null;
cipher.login_username = null;
cipher.login_password = null;
cipher.login_totp = null;
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
if (c.fields) {
c.fields.forEach((f: any) => {
if (!cipher.fields) {
cipher.fields = "";
} else {
2018-12-17 16:32:02 +01:00
cipher.fields += "\n";
2018-07-05 20:39:58 +02:00
}
2018-05-17 16:52:06 +02:00
cipher.fields += (f.name || "") + ": " + f.value;
2021-12-16 13:36:21 +01:00
});
2018-05-17 16:52:06 +02:00
}
2018-07-05 20:39:58 +02:00
switch (c.type) {
case CipherType.Login:
cipher.type = "login";
cipher.login_username = c.login.username;
cipher.login_password = c.login.password;
cipher.login_totp = c.login.totp;
2021-12-16 13:36:21 +01:00
2018-07-05 20:39:58 +02:00
if (c.login.uris) {
cipher.login_uri = [];
c.login.uris.forEach((u) => {
2018-07-05 20:39:58 +02:00
cipher.login_uri.push(u.uri);
});
}
break;
case CipherType.SecureNote:
cipher.type = "note";
2021-12-16 13:36:21 +01:00
break;
default:
2018-07-05 20:39:58 +02:00
return;
}
2021-12-16 13:36:21 +01:00
return cipher;
2021-12-16 13:36:21 +01:00
}
2018-05-17 16:52:06 +02:00
}