mirror of
https://github.com/bitwarden/browser
synced 2025-12-20 10:13:31 +00:00
* Rename service-factory folder * Move cryptographic service factories * Move crypto models * Move crypto services * Move domain base class * Platform code owners * Move desktop log services * Move log files * Establish component library ownership * Move background listeners * Move background background * Move localization to Platform * Move browser alarms to Platform * Move browser state to Platform * Move CLI state to Platform * Move Desktop native concerns to Platform * Move flag and misc to Platform * Lint fixes * Move electron state to platform * Move web state to Platform * Move lib state to Platform * Fix broken tests * Rename interface to idiomatic TS * `npm run prettier` 🤖 * Resolve review feedback * Set platform as owners of web core and shared * Expand moved services * Fix test types --------- Co-authored-by: Hinton <hinton@users.noreply.github.com>
163 lines
5.4 KiB
TypeScript
163 lines
5.4 KiB
TypeScript
import { ServiceUtils } from "../../misc/serviceUtils";
|
|
import { TreeNode } from "../../models/domain/tree-node";
|
|
import { CryptoService } from "../../platform/abstractions/crypto.service";
|
|
import { I18nService } from "../../platform/abstractions/i18n.service";
|
|
import { StateService } from "../../platform/abstractions/state.service";
|
|
import { Utils } from "../../platform/misc/utils";
|
|
import { CollectionService as CollectionServiceAbstraction } from "../abstractions/collection.service";
|
|
import { CollectionData } from "../models/data/collection.data";
|
|
import { Collection } from "../models/domain/collection";
|
|
import { CollectionView } from "../models/view/collection.view";
|
|
|
|
const NestingDelimiter = "/";
|
|
|
|
export class CollectionService implements CollectionServiceAbstraction {
|
|
constructor(
|
|
private cryptoService: CryptoService,
|
|
private i18nService: I18nService,
|
|
private stateService: StateService
|
|
) {}
|
|
|
|
async clearCache(userId?: string): Promise<void> {
|
|
await this.stateService.setDecryptedCollections(null, { userId: userId });
|
|
}
|
|
|
|
async encrypt(model: CollectionView): Promise<Collection> {
|
|
if (model.organizationId == null) {
|
|
throw new Error("Collection has no organization id.");
|
|
}
|
|
const key = await this.cryptoService.getOrgKey(model.organizationId);
|
|
if (key == null) {
|
|
throw new Error("No key for this collection's organization.");
|
|
}
|
|
const collection = new Collection();
|
|
collection.id = model.id;
|
|
collection.organizationId = model.organizationId;
|
|
collection.readOnly = model.readOnly;
|
|
collection.name = await this.cryptoService.encrypt(model.name, key);
|
|
return collection;
|
|
}
|
|
|
|
async decryptMany(collections: Collection[]): Promise<CollectionView[]> {
|
|
if (collections == null) {
|
|
return [];
|
|
}
|
|
const decCollections: CollectionView[] = [];
|
|
const promises: Promise<any>[] = [];
|
|
collections.forEach((collection) => {
|
|
promises.push(collection.decrypt().then((c) => decCollections.push(c)));
|
|
});
|
|
await Promise.all(promises);
|
|
return decCollections.sort(Utils.getSortFunction(this.i18nService, "name"));
|
|
}
|
|
|
|
async get(id: string): Promise<Collection> {
|
|
const collections = await this.stateService.getEncryptedCollections();
|
|
// eslint-disable-next-line
|
|
if (collections == null || !collections.hasOwnProperty(id)) {
|
|
return null;
|
|
}
|
|
|
|
return new Collection(collections[id]);
|
|
}
|
|
|
|
async getAll(): Promise<Collection[]> {
|
|
const collections = await this.stateService.getEncryptedCollections();
|
|
const response: Collection[] = [];
|
|
for (const id in collections) {
|
|
// eslint-disable-next-line
|
|
if (collections.hasOwnProperty(id)) {
|
|
response.push(new Collection(collections[id]));
|
|
}
|
|
}
|
|
return response;
|
|
}
|
|
|
|
async getAllDecrypted(): Promise<CollectionView[]> {
|
|
let decryptedCollections = await this.stateService.getDecryptedCollections();
|
|
if (decryptedCollections != null) {
|
|
return decryptedCollections;
|
|
}
|
|
|
|
const hasKey = await this.cryptoService.hasKey();
|
|
if (!hasKey) {
|
|
throw new Error("No key.");
|
|
}
|
|
|
|
const collections = await this.getAll();
|
|
decryptedCollections = await this.decryptMany(collections);
|
|
|
|
await this.stateService.setDecryptedCollections(decryptedCollections);
|
|
return decryptedCollections;
|
|
}
|
|
|
|
async getAllNested(collections: CollectionView[] = null): Promise<TreeNode<CollectionView>[]> {
|
|
if (collections == null) {
|
|
collections = await this.getAllDecrypted();
|
|
}
|
|
const nodes: TreeNode<CollectionView>[] = [];
|
|
collections.forEach((c) => {
|
|
const collectionCopy = new CollectionView();
|
|
collectionCopy.id = c.id;
|
|
collectionCopy.organizationId = c.organizationId;
|
|
const parts = c.name != null ? c.name.replace(/^\/+|\/+$/g, "").split(NestingDelimiter) : [];
|
|
ServiceUtils.nestedTraverse(nodes, 0, parts, collectionCopy, null, NestingDelimiter);
|
|
});
|
|
return nodes;
|
|
}
|
|
|
|
/**
|
|
* @deprecated August 30 2022: Moved to new Vault Filter Service
|
|
* Remove when Desktop and Browser are updated
|
|
*/
|
|
async getNested(id: string): Promise<TreeNode<CollectionView>> {
|
|
const collections = await this.getAllNested();
|
|
return ServiceUtils.getTreeNodeObjectFromList(collections, id) as TreeNode<CollectionView>;
|
|
}
|
|
|
|
async upsert(collection: CollectionData | CollectionData[]): Promise<any> {
|
|
let collections = await this.stateService.getEncryptedCollections();
|
|
if (collections == null) {
|
|
collections = {};
|
|
}
|
|
|
|
if (collection instanceof CollectionData) {
|
|
const c = collection as CollectionData;
|
|
collections[c.id] = c;
|
|
} else {
|
|
(collection as CollectionData[]).forEach((c) => {
|
|
collections[c.id] = c;
|
|
});
|
|
}
|
|
|
|
await this.replace(collections);
|
|
}
|
|
|
|
async replace(collections: { [id: string]: CollectionData }): Promise<any> {
|
|
await this.clearCache();
|
|
await this.stateService.setEncryptedCollections(collections);
|
|
}
|
|
|
|
async clear(userId?: string): Promise<any> {
|
|
await this.clearCache(userId);
|
|
await this.stateService.setEncryptedCollections(null, { userId: userId });
|
|
}
|
|
|
|
async delete(id: string | string[]): Promise<any> {
|
|
const collections = await this.stateService.getEncryptedCollections();
|
|
if (collections == null) {
|
|
return;
|
|
}
|
|
|
|
if (typeof id === "string") {
|
|
delete collections[id];
|
|
} else {
|
|
(id as string[]).forEach((i) => {
|
|
delete collections[i];
|
|
});
|
|
}
|
|
|
|
await this.replace(collections);
|
|
}
|
|
}
|