2020-02-15 00:10:58 +03:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Bit.Core.Abstractions;
|
|
|
|
|
using Bit.Core.Enums;
|
|
|
|
|
using Bit.Core.Models.Export;
|
|
|
|
|
using Bit.Core.Models.View;
|
|
|
|
|
using Bit.Core.Utilities;
|
|
|
|
|
using CsvHelper;
|
|
|
|
|
using CsvHelper.Configuration.Attributes;
|
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
|
using Newtonsoft.Json.Serialization;
|
|
|
|
|
|
|
|
|
|
namespace Bit.Core.Services
|
|
|
|
|
{
|
|
|
|
|
public class ExportService : IExportService
|
|
|
|
|
{
|
|
|
|
|
private readonly IFolderService _folderService;
|
|
|
|
|
private readonly ICipherService _cipherService;
|
2021-05-28 23:16:19 +03:00
|
|
|
|
private readonly ICryptoService _cryptoService;
|
2020-02-15 00:10:58 +03:00
|
|
|
|
|
|
|
|
|
public ExportService(
|
|
|
|
|
IFolderService folderService,
|
2021-05-28 23:16:19 +03:00
|
|
|
|
ICipherService cipherService,
|
|
|
|
|
ICryptoService cryptoService)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
|
|
|
|
_folderService = folderService;
|
|
|
|
|
_cipherService = cipherService;
|
2021-05-28 23:16:19 +03:00
|
|
|
|
_cryptoService = cryptoService;
|
2020-02-15 00:10:58 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<string> GetExport(string format = "csv")
|
|
|
|
|
{
|
2020-12-14 20:56:13 +03:00
|
|
|
|
if (format == "encrypted_json")
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
2020-12-14 20:56:13 +03:00
|
|
|
|
var folders = (await _folderService.GetAllAsync()).Where(f => f.Id != null).Select(f => new FolderWithId(f));
|
2020-12-29 20:38:12 +03:00
|
|
|
|
var items = (await _cipherService.GetAllAsync()).Where(c => c.OrganizationId == null && c.DeletedDate == null)
|
|
|
|
|
.Select(c => new CipherWithId(c));
|
2020-02-15 00:10:58 +03:00
|
|
|
|
|
2021-05-28 23:16:19 +03:00
|
|
|
|
return await ExportEncryptedJson(folders, items);
|
2020-02-15 00:10:58 +03:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-12-14 20:56:13 +03:00
|
|
|
|
var decryptedFolders = await _folderService.GetAllDecryptedAsync();
|
2020-12-29 20:38:12 +03:00
|
|
|
|
var decryptedCiphers = (await _cipherService.GetAllDecryptedAsync()).Where(c => c.DeletedDate == null);
|
2020-02-15 00:10:58 +03:00
|
|
|
|
|
2020-12-14 20:56:13 +03:00
|
|
|
|
return format == "csv" ? ExportCsv(decryptedFolders, decryptedCiphers) : ExportJson(decryptedFolders, decryptedCiphers);
|
2020-02-15 00:10:58 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Task<string> GetOrganizationExport(string organizationId, string format = "csv")
|
|
|
|
|
{
|
|
|
|
|
throw new NotImplementedException();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string GetFileName(string prefix = null, string extension = "csv")
|
|
|
|
|
{
|
|
|
|
|
var dateString = DateTime.Now.ToString("yyyyMMddHHmmss");
|
|
|
|
|
|
|
|
|
|
return string.Format("bitwarden{0}_export_{1}.{2}",
|
|
|
|
|
!string.IsNullOrEmpty(prefix) ? ("_" + prefix) : string.Empty, dateString, extension);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void BuildCommonCipher(ExportCipher cipher, CipherView c)
|
|
|
|
|
{
|
|
|
|
|
cipher.Type = null;
|
|
|
|
|
cipher.Name = c.Name;
|
|
|
|
|
cipher.Notes = c.Notes;
|
|
|
|
|
cipher.Fields = null;
|
|
|
|
|
// Login props
|
|
|
|
|
cipher.LoginUris = null;
|
|
|
|
|
cipher.LoginUsername = null;
|
|
|
|
|
cipher.LoginPassword = null;
|
|
|
|
|
cipher.LoginTotp = null;
|
|
|
|
|
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (c.Fields != null)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
foreach (var f in c.Fields)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (cipher.Fields == null)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
|
|
|
|
cipher.Fields = "";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
cipher.Fields += "\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cipher.Fields += (f.Name ?? "") + ": " + f.Value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-28 16:16:28 +03:00
|
|
|
|
switch (c.Type)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
|
|
|
|
case CipherType.Login:
|
|
|
|
|
cipher.Type = "login";
|
|
|
|
|
cipher.LoginUsername = c.Login.Username;
|
|
|
|
|
cipher.LoginPassword = c.Login.Password;
|
|
|
|
|
cipher.LoginTotp = c.Login.Totp;
|
|
|
|
|
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (c.Login.Uris != null)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
foreach (var u in c.Login.Uris)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
2020-03-28 16:16:28 +03:00
|
|
|
|
if (cipher.LoginUris == null)
|
2020-02-15 00:10:58 +03:00
|
|
|
|
{
|
|
|
|
|
cipher.LoginUris = "";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
cipher.LoginUris += ",";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cipher.LoginUris += u.Uri;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
case CipherType.SecureNote:
|
|
|
|
|
cipher.Type = "note";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-14 20:56:13 +03:00
|
|
|
|
private string ExportCsv(IEnumerable<FolderView> decryptedFolders, IEnumerable<CipherView> decryptedCiphers)
|
|
|
|
|
{
|
|
|
|
|
var foldersMap = decryptedFolders.Where(f => f.Id != null).ToDictionary(f => f.Id);
|
|
|
|
|
|
|
|
|
|
var exportCiphers = new List<ExportCipher>();
|
|
|
|
|
foreach (var c in decryptedCiphers)
|
|
|
|
|
{
|
|
|
|
|
// only export logins and secure notes
|
|
|
|
|
if (c.Type != CipherType.Login && c.Type != CipherType.SecureNote)
|
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (c.OrganizationId != null)
|
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cipher = new ExportCipher();
|
|
|
|
|
cipher.Folder = c.FolderId != null && foldersMap.ContainsKey(c.FolderId)
|
|
|
|
|
? foldersMap[c.FolderId].Name : null;
|
|
|
|
|
cipher.Favorite = c.Favorite ? "1" : null;
|
|
|
|
|
BuildCommonCipher(cipher, c);
|
|
|
|
|
exportCiphers.Add(cipher);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using (var writer = new StringWriter())
|
|
|
|
|
using (var csv = new CsvWriter(writer, CultureInfo.InvariantCulture))
|
|
|
|
|
{
|
|
|
|
|
csv.WriteRecords(exportCiphers);
|
|
|
|
|
csv.Flush();
|
|
|
|
|
return writer.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string ExportJson(IEnumerable<FolderView> decryptedFolders, IEnumerable<CipherView> decryptedCiphers)
|
|
|
|
|
{
|
|
|
|
|
var jsonDoc = new
|
|
|
|
|
{
|
2020-12-19 01:17:04 +03:00
|
|
|
|
Encrypted = false,
|
2020-12-14 20:56:13 +03:00
|
|
|
|
Folders = decryptedFolders.Where(f => f.Id != null).Select(f => new FolderWithId(f)),
|
|
|
|
|
Items = decryptedCiphers.Where(c => c.OrganizationId == null)
|
|
|
|
|
.Select(c => new CipherWithId(c) { CollectionIds = null })
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return CoreHelpers.SerializeJson(jsonDoc,
|
|
|
|
|
new JsonSerializerSettings
|
|
|
|
|
{
|
|
|
|
|
Formatting = Formatting.Indented,
|
|
|
|
|
ContractResolver = new CamelCasePropertyNamesContractResolver()
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-28 23:16:19 +03:00
|
|
|
|
private async Task<string> ExportEncryptedJson(IEnumerable<FolderWithId> folders, IEnumerable<CipherWithId> ciphers)
|
2020-12-14 20:56:13 +03:00
|
|
|
|
{
|
2021-05-28 23:16:19 +03:00
|
|
|
|
var encKeyValidation = await _cryptoService.EncryptAsync(Guid.NewGuid().ToString());
|
|
|
|
|
|
2020-12-14 20:56:13 +03:00
|
|
|
|
var jsonDoc = new
|
|
|
|
|
{
|
2020-12-19 01:17:04 +03:00
|
|
|
|
Encrypted = true,
|
2021-05-28 23:16:19 +03:00
|
|
|
|
EncKeyValidation_DO_NOT_EDIT = encKeyValidation.EncryptedString,
|
2020-12-14 20:56:13 +03:00
|
|
|
|
Folders = folders,
|
|
|
|
|
Items = ciphers,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return CoreHelpers.SerializeJson(jsonDoc,
|
|
|
|
|
new JsonSerializerSettings
|
|
|
|
|
{
|
|
|
|
|
Formatting = Formatting.Indented,
|
|
|
|
|
ContractResolver = new CamelCasePropertyNamesContractResolver()
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-15 00:10:58 +03:00
|
|
|
|
private class ExportCipher
|
|
|
|
|
{
|
|
|
|
|
[Name("folder")]
|
|
|
|
|
public string Folder { get; set; }
|
|
|
|
|
[Name("favorite")]
|
|
|
|
|
public string Favorite { get; set; }
|
|
|
|
|
[Name("type")]
|
|
|
|
|
public string Type { get; set; }
|
|
|
|
|
[Name("name")]
|
|
|
|
|
public string Name { get; set; }
|
|
|
|
|
[Name("notes")]
|
|
|
|
|
public string Notes { get; set; }
|
|
|
|
|
[Name("fields")]
|
|
|
|
|
public string Fields { get; set; }
|
|
|
|
|
[Name("login_uri")]
|
|
|
|
|
public string LoginUris { get; set; }
|
|
|
|
|
[Name("login_username")]
|
|
|
|
|
public string LoginUsername { get; set; }
|
|
|
|
|
[Name("login_password")]
|
|
|
|
|
public string LoginPassword { get; set; }
|
|
|
|
|
[Name("login_totp")]
|
|
|
|
|
public string LoginTotp { get; set; }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|