bitwarden-estensione-browser/src/commands/get.command.ts

346 lines
13 KiB
TypeScript
Raw Normal View History

2018-05-14 19:37:52 +02:00
import * as program from 'commander';
2018-05-17 19:28:22 +02:00
import * as fet from 'node-fetch';
2018-05-14 19:37:52 +02:00
import { CipherType } from 'jslib/enums/cipherType';
2018-05-17 17:07:53 +02:00
import { AuditService } from 'jslib/abstractions/audit.service';
2018-05-14 19:37:52 +02:00
import { CipherService } from 'jslib/abstractions/cipher.service';
import { CollectionService } from 'jslib/abstractions/collection.service';
2018-05-17 19:28:22 +02:00
import { CryptoService } from 'jslib/abstractions/crypto.service';
2018-05-14 19:37:52 +02:00
import { FolderService } from 'jslib/abstractions/folder.service';
2018-05-18 16:55:50 +02:00
import { TokenService } from 'jslib/abstractions/token.service';
2018-05-14 19:37:52 +02:00
import { TotpService } from 'jslib/abstractions/totp.service';
2018-05-16 17:54:59 +02:00
import { CipherView } from 'jslib/models/view/cipherView';
import { CollectionView } from 'jslib/models/view/collectionView';
import { FolderView } from 'jslib/models/view/folderView';
2018-05-14 20:54:19 +02:00
import { Response } from '../models/response';
import { CipherResponse } from '../models/response/cipherResponse';
import { CollectionResponse } from '../models/response/collectionResponse';
import { FolderResponse } from '../models/response/folderResponse';
2018-05-17 19:28:22 +02:00
import { MessageResponse } from '../models/response/messageResponse';
2018-05-14 20:54:19 +02:00
import { StringResponse } from '../models/response/stringResponse';
2018-05-14 22:25:14 +02:00
import { TemplateResponse } from '../models/response/templateResponse';
2018-05-15 18:18:47 +02:00
import { Card } from '../models/card';
import { Cipher } from '../models/cipher';
2018-05-15 18:53:08 +02:00
import { Collection } from '../models/collection';
2018-05-15 18:18:47 +02:00
import { Field } from '../models/field';
2018-05-15 18:53:08 +02:00
import { Folder } from '../models/folder';
2018-05-15 18:18:47 +02:00
import { Identity } from '../models/identity';
import { Login } from '../models/login';
import { LoginUri } from '../models/loginUri';
import { SecureNote } from '../models/secureNote';
2018-05-14 20:54:19 +02:00
2018-05-16 17:54:59 +02:00
import { CliUtils } from '../utils';
2018-05-14 19:37:52 +02:00
export class GetCommand {
constructor(private cipherService: CipherService, private folderService: FolderService,
2018-05-17 17:07:53 +02:00
private collectionService: CollectionService, private totpService: TotpService,
2018-05-18 16:55:50 +02:00
private auditService: AuditService, private cryptoService: CryptoService,
private tokenService: TokenService) { }
2018-05-14 19:37:52 +02:00
2018-05-14 20:54:19 +02:00
async run(object: string, id: string, cmd: program.Command): Promise<Response> {
2018-05-16 17:54:59 +02:00
if (id != null) {
id = id.toLowerCase();
}
2018-05-14 22:25:14 +02:00
switch (object.toLowerCase()) {
2018-05-14 19:37:52 +02:00
case 'item':
2018-05-17 21:55:44 +02:00
return await this.getCipher(id);
2018-05-16 19:59:08 +02:00
case 'username':
return await this.getUsername(id);
case 'password':
return await this.getPassword(id);
case 'uri':
return await this.getUri(id);
2018-05-14 19:37:52 +02:00
case 'totp':
2018-05-14 20:54:19 +02:00
return await this.getTotp(id);
2018-05-17 17:07:53 +02:00
case 'exposed':
return await this.getExposed(id);
2018-05-17 21:55:44 +02:00
case 'attachment':
return await this.getAttachment(id, cmd);
2018-05-14 19:37:52 +02:00
case 'folder':
2018-05-14 20:54:19 +02:00
return await this.getFolder(id);
2018-05-14 19:37:52 +02:00
case 'collection':
2018-05-14 20:54:19 +02:00
return await this.getCollection(id);
2018-05-14 22:25:14 +02:00
case 'template':
return await this.getTemplate(id);
2018-05-14 19:37:52 +02:00
default:
2018-05-14 20:54:19 +02:00
return Response.badRequest('Unknown object.');
2018-05-14 19:37:52 +02:00
}
}
private async getCipher(id: string) {
2018-05-16 17:54:59 +02:00
let decCipher: CipherView = null;
if (this.isGuid(id)) {
const cipher = await this.cipherService.get(id);
if (cipher != null) {
decCipher = await cipher.decrypt();
}
} else if (id.trim() !== '') {
let ciphers = await this.cipherService.getAllDecrypted();
ciphers = CliUtils.searchCiphers(ciphers, id);
if (ciphers.length > 1) {
2018-05-16 18:00:40 +02:00
return Response.multipleResults(ciphers.map((c) => c.id));
2018-05-16 17:54:59 +02:00
}
if (ciphers.length > 0) {
decCipher = ciphers[0];
}
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
if (decCipher == null) {
return Response.notFound();
}
2018-05-14 20:54:19 +02:00
const res = new CipherResponse(decCipher);
return Response.success(res);
2018-05-14 19:37:52 +02:00
}
2018-05-16 19:59:08 +02:00
private async getUsername(id: string) {
const cipherResponse = await this.getCipher(id);
if (!cipherResponse.success) {
return cipherResponse;
}
const cipher = cipherResponse.data as CipherResponse;
if (cipher.type !== CipherType.Login) {
return Response.badRequest('Not a login.');
}
if (cipher.login.username == null || cipher.login.username === '') {
return Response.error('No username available for this login.');
}
const res = new StringResponse(cipher.login.username);
return Response.success(res);
}
private async getPassword(id: string) {
const cipherResponse = await this.getCipher(id);
if (!cipherResponse.success) {
return cipherResponse;
}
const cipher = cipherResponse.data as CipherResponse;
if (cipher.type !== CipherType.Login) {
return Response.badRequest('Not a login.');
}
if (cipher.login.password == null || cipher.login.password === '') {
return Response.error('No password available for this login.');
}
const res = new StringResponse(cipher.login.password);
return Response.success(res);
}
private async getUri(id: string) {
const cipherResponse = await this.getCipher(id);
if (!cipherResponse.success) {
return cipherResponse;
}
const cipher = cipherResponse.data as CipherResponse;
if (cipher.type !== CipherType.Login) {
return Response.badRequest('Not a login.');
}
if (cipher.login.uris == null || cipher.login.uris.length === 0 || cipher.login.uris[0].uri === '') {
return Response.error('No uri available for this login.');
}
const res = new StringResponse(cipher.login.uris[0].uri);
return Response.success(res);
}
2018-05-14 19:37:52 +02:00
private async getTotp(id: string) {
2018-05-16 17:54:59 +02:00
const cipherResponse = await this.getCipher(id);
if (!cipherResponse.success) {
return cipherResponse;
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
const cipher = cipherResponse.data as CipherResponse;
2018-05-14 19:37:52 +02:00
if (cipher.type !== CipherType.Login) {
2018-05-14 20:54:19 +02:00
return Response.badRequest('Not a login.');
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
if (cipher.login.totp == null || cipher.login.totp === '') {
2018-05-14 20:54:19 +02:00
return Response.error('No TOTP available for this login.');
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
const totp = await this.totpService.getCode(cipher.login.totp);
2018-05-14 19:37:52 +02:00
if (totp == null) {
2018-05-14 20:54:19 +02:00
return Response.error('Couldn\'t generate TOTP code.');
2018-05-14 19:37:52 +02:00
}
2018-05-18 16:55:50 +02:00
if (!this.tokenService.getPremium()) {
const originalCipher = await this.cipherService.get(id);
if (originalCipher == null || originalCipher.organizationId == null ||
!originalCipher.organizationUseTotp) {
return Response.error('A premium membership is required to use this feature.');
}
}
2018-05-14 20:54:19 +02:00
const res = new StringResponse(totp);
return Response.success(res);
2018-05-14 19:37:52 +02:00
}
2018-05-17 17:07:53 +02:00
private async getExposed(id: string) {
const passwordResponse = await this.getPassword(id);
if (!passwordResponse.success) {
return passwordResponse;
}
const exposedNumber = await this.auditService.passwordLeaked((passwordResponse.data as StringResponse).data);
const res = new StringResponse(exposedNumber.toString());
return Response.success(res);
}
2018-05-17 21:55:44 +02:00
private async getAttachment(id: string, cmd: program.Command) {
if (cmd.itemid == null || cmd.itemid === '') {
return Response.badRequest('--itemid <itemid> required.');
}
2018-05-17 19:43:53 +02:00
2018-05-17 21:55:44 +02:00
const itemId = cmd.itemid.toLowerCase();
const cipherResponse = await this.getCipher(itemId);
2018-05-17 19:28:22 +02:00
if (!cipherResponse.success) {
return cipherResponse;
}
const cipher = cipherResponse.data as CipherResponse;
if (cipher.attachments == null || cipher.attachments.length === 0) {
return Response.error('No attachments available for this item.');
}
2018-05-17 21:55:44 +02:00
const attachments = cipher.attachments.filter((a) => a.id.toLowerCase() === id ||
(a.fileName != null && a.fileName.toLowerCase().indexOf(id) > -1));
2018-05-17 19:43:53 +02:00
if (attachments.length === 0) {
2018-05-17 21:55:44 +02:00
return Response.error('Attachment `' + id + '` was not found.');
2018-05-17 19:28:22 +02:00
}
2018-05-17 19:43:53 +02:00
if (attachments.length > 1) {
return Response.multipleResults(attachments.map((a) => a.id));
2018-05-17 19:28:22 +02:00
}
2018-05-18 16:55:50 +02:00
if (!this.tokenService.getPremium()) {
const originalCipher = await this.cipherService.get(cipher.id);
if (originalCipher == null || originalCipher.organizationId == null) {
return Response.error('A premium membership is required to use this feature.');
}
}
2018-05-17 19:43:53 +02:00
const response = await fet.default(new fet.Request(attachments[0].url, { headers: { cache: 'no-cache' } }));
2018-05-17 19:28:22 +02:00
if (response.status !== 200) {
return Response.error('A ' + response.status + ' error occurred while downloading the attachment.');
}
try {
const buf = await response.arrayBuffer();
const key = await this.cryptoService.getOrgKey(cipher.organizationId);
const decBuf = await this.cryptoService.decryptFromBytes(buf, key);
2018-05-17 19:43:53 +02:00
const filePath = await CliUtils.saveFile(new Buffer(decBuf), cmd.output, attachments[0].fileName);
2018-05-17 19:28:22 +02:00
const res = new MessageResponse('Saved ' + filePath, null);
res.raw = filePath;
return Response.success(res);
} catch (e) {
if (typeof (e) === 'string') {
return Response.error(e);
} else {
return Response.error('An error occurred while saving the attachment.');
}
}
}
2018-05-14 19:37:52 +02:00
private async getFolder(id: string) {
2018-05-16 17:54:59 +02:00
let decFolder: FolderView = null;
if (this.isGuid(id)) {
const folder = await this.folderService.get(id);
if (folder != null) {
decFolder = await folder.decrypt();
}
} else if (id.trim() !== '') {
let folders = await this.folderService.getAllDecrypted();
folders = CliUtils.searchFolders(folders, id);
if (folders.length > 1) {
2018-05-16 18:00:40 +02:00
return Response.multipleResults(folders.map((f) => f.id));
2018-05-16 17:54:59 +02:00
}
if (folders.length > 0) {
decFolder = folders[0];
}
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
if (decFolder == null) {
return Response.notFound();
}
2018-05-14 20:54:19 +02:00
const res = new FolderResponse(decFolder);
return Response.success(res);
2018-05-14 19:37:52 +02:00
}
private async getCollection(id: string) {
2018-05-16 17:54:59 +02:00
let decCollection: CollectionView = null;
if (this.isGuid(id)) {
const collection = await this.collectionService.get(id);
if (collection != null) {
decCollection = await collection.decrypt();
}
} else if (id.trim() !== '') {
let collections = await this.collectionService.getAllDecrypted();
collections = CliUtils.searchCollections(collections, id);
if (collections.length > 1) {
2018-05-16 18:00:40 +02:00
return Response.multipleResults(collections.map((c) => c.id));
2018-05-16 17:54:59 +02:00
}
if (collections.length > 0) {
decCollection = collections[0];
}
2018-05-14 19:37:52 +02:00
}
2018-05-16 17:54:59 +02:00
if (decCollection == null) {
return Response.notFound();
}
2018-05-14 20:54:19 +02:00
const res = new CollectionResponse(decCollection);
return Response.success(res);
2018-05-14 19:37:52 +02:00
}
2018-05-14 22:25:14 +02:00
2018-05-16 17:54:59 +02:00
private isGuid(id: string) {
return RegExp(/^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/, 'i').test(id);
}
2018-05-14 22:25:14 +02:00
private async getTemplate(id: string) {
let template: any = null;
switch (id.toLowerCase()) {
case 'item':
2018-05-15 18:18:47 +02:00
template = Cipher.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.field':
2018-05-15 18:18:47 +02:00
template = Field.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.login':
2018-05-15 18:18:47 +02:00
template = Login.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.login.uri':
2018-05-15 18:18:47 +02:00
template = LoginUri.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.card':
2018-05-15 18:18:47 +02:00
template = Card.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.identity':
2018-05-15 18:18:47 +02:00
template = Identity.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-18 15:16:34 +02:00
case 'item.securenote':
2018-05-15 18:18:47 +02:00
template = SecureNote.template();
2018-05-14 22:25:14 +02:00
break;
2018-05-15 18:53:08 +02:00
case 'folder':
template = Folder.template();
break;
case 'collection':
template = Collection.template();
break;
2018-05-14 22:25:14 +02:00
default:
2018-05-14 23:13:57 +02:00
return Response.badRequest('Unknown template object.');
2018-05-14 22:25:14 +02:00
}
2018-05-15 05:40:11 +02:00
2018-05-14 22:25:14 +02:00
const res = new TemplateResponse(template);
return Response.success(res);
}
2018-05-14 19:37:52 +02:00
}