mirror of
https://github.com/bitwarden/browser
synced 2025-12-15 07:43:35 +00:00
* Simplify password protected file format * no items to import is not an error * Await inner importer * Add export format type * Error if import file is password protected * Update tests * Test password protected with normat json importer * Simplify imports * Ignore code coverage directory * Expand importer options without changing display options * Import password require import error handling * Use interface * Fix curlies * linter fixes * Add null of empty util * Lint fixes * run prettier * Move import options to separate enum file * Fix imports
180 lines
6.1 KiB
TypeScript
180 lines
6.1 KiB
TypeScript
import { CryptoService } from "../abstractions/crypto.service";
|
|
import { I18nService } from "../abstractions/i18n.service";
|
|
import { EncString } from "../models/domain/encString";
|
|
import { ImportResult } from "../models/domain/importResult";
|
|
import { CipherWithIds } from "../models/export/cipherWithIds";
|
|
import { CollectionWithId } from "../models/export/collectionWithId";
|
|
import { FolderWithId } from "../models/export/folderWithId";
|
|
|
|
import { BaseImporter } from "./baseImporter";
|
|
import { Importer } from "./importer";
|
|
|
|
export class BitwardenJsonImporter extends BaseImporter implements Importer {
|
|
private results: any;
|
|
private result: ImportResult;
|
|
|
|
constructor(protected cryptoService: CryptoService, protected i18nService: I18nService) {
|
|
super();
|
|
}
|
|
|
|
async parse(data: string): Promise<ImportResult> {
|
|
this.result = new ImportResult();
|
|
this.results = JSON.parse(data);
|
|
if (this.results == null || this.results.items == null) {
|
|
if (this.results?.passwordProtected) {
|
|
this.result.success = false;
|
|
this.result.missingPassword = true;
|
|
this.result.errorMessage = this.i18nService.t("importPasswordRequired");
|
|
return this.result;
|
|
}
|
|
|
|
this.result.success = false;
|
|
return this.result;
|
|
}
|
|
|
|
if (this.results.encrypted) {
|
|
await this.parseEncrypted();
|
|
} else {
|
|
this.parseDecrypted();
|
|
}
|
|
|
|
return this.result;
|
|
}
|
|
|
|
private async parseEncrypted() {
|
|
if (this.results.encKeyValidation_DO_NOT_EDIT != null) {
|
|
const orgKey = await this.cryptoService.getOrgKey(this.organizationId);
|
|
const encKeyValidation = new EncString(this.results.encKeyValidation_DO_NOT_EDIT);
|
|
const encKeyValidationDecrypt = await this.cryptoService.decryptToUtf8(
|
|
encKeyValidation,
|
|
orgKey
|
|
);
|
|
if (encKeyValidationDecrypt === null) {
|
|
this.result.success = false;
|
|
this.result.errorMessage = this.i18nService.t("importEncKeyError");
|
|
return;
|
|
}
|
|
}
|
|
|
|
const groupingsMap = new Map<string, number>();
|
|
|
|
if (this.organization && this.results.collections != null) {
|
|
for (const c of this.results.collections as CollectionWithId[]) {
|
|
const collection = CollectionWithId.toDomain(c);
|
|
if (collection != null) {
|
|
collection.id = null;
|
|
collection.organizationId = this.organizationId;
|
|
const view = await collection.decrypt();
|
|
groupingsMap.set(c.id, this.result.collections.length);
|
|
this.result.collections.push(view);
|
|
}
|
|
}
|
|
} else if (!this.organization && this.results.folders != null) {
|
|
for (const f of this.results.folders as FolderWithId[]) {
|
|
const folder = FolderWithId.toDomain(f);
|
|
if (folder != null) {
|
|
folder.id = null;
|
|
const view = await folder.decrypt();
|
|
groupingsMap.set(f.id, this.result.folders.length);
|
|
this.result.folders.push(view);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const c of this.results.items as CipherWithIds[]) {
|
|
const cipher = CipherWithIds.toDomain(c);
|
|
// reset ids incase they were set for some reason
|
|
cipher.id = null;
|
|
cipher.folderId = null;
|
|
cipher.organizationId = this.organizationId;
|
|
cipher.collectionIds = null;
|
|
|
|
// make sure password history is limited
|
|
if (cipher.passwordHistory != null && cipher.passwordHistory.length > 5) {
|
|
cipher.passwordHistory = cipher.passwordHistory.slice(0, 5);
|
|
}
|
|
|
|
if (!this.organization && c.folderId != null && groupingsMap.has(c.folderId)) {
|
|
this.result.folderRelationships.push([
|
|
this.result.ciphers.length,
|
|
groupingsMap.get(c.folderId),
|
|
]);
|
|
} else if (this.organization && c.collectionIds != null) {
|
|
c.collectionIds.forEach((cId) => {
|
|
if (groupingsMap.has(cId)) {
|
|
this.result.collectionRelationships.push([
|
|
this.result.ciphers.length,
|
|
groupingsMap.get(cId),
|
|
]);
|
|
}
|
|
});
|
|
}
|
|
|
|
const view = await cipher.decrypt();
|
|
this.cleanupCipher(view);
|
|
this.result.ciphers.push(view);
|
|
}
|
|
|
|
this.result.success = true;
|
|
}
|
|
|
|
private parseDecrypted() {
|
|
const groupingsMap = new Map<string, number>();
|
|
if (this.organization && this.results.collections != null) {
|
|
this.results.collections.forEach((c: CollectionWithId) => {
|
|
const collection = CollectionWithId.toView(c);
|
|
if (collection != null) {
|
|
collection.id = null;
|
|
collection.organizationId = null;
|
|
groupingsMap.set(c.id, this.result.collections.length);
|
|
this.result.collections.push(collection);
|
|
}
|
|
});
|
|
} else if (!this.organization && this.results.folders != null) {
|
|
this.results.folders.forEach((f: FolderWithId) => {
|
|
const folder = FolderWithId.toView(f);
|
|
if (folder != null) {
|
|
folder.id = null;
|
|
groupingsMap.set(f.id, this.result.folders.length);
|
|
this.result.folders.push(folder);
|
|
}
|
|
});
|
|
}
|
|
|
|
this.results.items.forEach((c: CipherWithIds) => {
|
|
const cipher = CipherWithIds.toView(c);
|
|
// reset ids incase they were set for some reason
|
|
cipher.id = null;
|
|
cipher.folderId = null;
|
|
cipher.organizationId = null;
|
|
cipher.collectionIds = null;
|
|
|
|
// make sure password history is limited
|
|
if (cipher.passwordHistory != null && cipher.passwordHistory.length > 5) {
|
|
cipher.passwordHistory = cipher.passwordHistory.slice(0, 5);
|
|
}
|
|
|
|
if (!this.organization && c.folderId != null && groupingsMap.has(c.folderId)) {
|
|
this.result.folderRelationships.push([
|
|
this.result.ciphers.length,
|
|
groupingsMap.get(c.folderId),
|
|
]);
|
|
} else if (this.organization && c.collectionIds != null) {
|
|
c.collectionIds.forEach((cId) => {
|
|
if (groupingsMap.has(cId)) {
|
|
this.result.collectionRelationships.push([
|
|
this.result.ciphers.length,
|
|
groupingsMap.get(cId),
|
|
]);
|
|
}
|
|
});
|
|
}
|
|
|
|
this.cleanupCipher(cipher);
|
|
this.result.ciphers.push(cipher);
|
|
});
|
|
|
|
this.result.success = true;
|
|
}
|
|
}
|