mirror of
https://github.com/bitwarden/server.git
synced 2025-05-21 11:34:31 -05:00
return attachments from API
This commit is contained in:
parent
284078e946
commit
8ea81a74ae
@ -22,6 +22,7 @@ namespace Bit.Api.Controllers
|
|||||||
private readonly IUserService _userService;
|
private readonly IUserService _userService;
|
||||||
private readonly IAttachmentStorageService _attachmentStorageService;
|
private readonly IAttachmentStorageService _attachmentStorageService;
|
||||||
private readonly CurrentContext _currentContext;
|
private readonly CurrentContext _currentContext;
|
||||||
|
private readonly GlobalSettings _globalSettings;
|
||||||
|
|
||||||
public CiphersController(
|
public CiphersController(
|
||||||
ICipherRepository cipherRepository,
|
ICipherRepository cipherRepository,
|
||||||
@ -29,7 +30,8 @@ namespace Bit.Api.Controllers
|
|||||||
ICipherService cipherService,
|
ICipherService cipherService,
|
||||||
IUserService userService,
|
IUserService userService,
|
||||||
IAttachmentStorageService attachmentStorageService,
|
IAttachmentStorageService attachmentStorageService,
|
||||||
CurrentContext currentContext)
|
CurrentContext currentContext,
|
||||||
|
GlobalSettings globalSettings)
|
||||||
{
|
{
|
||||||
_cipherRepository = cipherRepository;
|
_cipherRepository = cipherRepository;
|
||||||
_collectionCipherRepository = collectionCipherRepository;
|
_collectionCipherRepository = collectionCipherRepository;
|
||||||
@ -37,6 +39,7 @@ namespace Bit.Api.Controllers
|
|||||||
_userService = userService;
|
_userService = userService;
|
||||||
_attachmentStorageService = attachmentStorageService;
|
_attachmentStorageService = attachmentStorageService;
|
||||||
_currentContext = currentContext;
|
_currentContext = currentContext;
|
||||||
|
_globalSettings = globalSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}")]
|
[HttpGet("{id}")]
|
||||||
@ -49,7 +52,7 @@ namespace Bit.Api.Controllers
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
return new CipherResponseModel(cipher);
|
return new CipherResponseModel(cipher, _globalSettings);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}/full-details")]
|
[HttpGet("{id}/full-details")]
|
||||||
@ -65,7 +68,7 @@ namespace Bit.Api.Controllers
|
|||||||
}
|
}
|
||||||
|
|
||||||
var collectionCiphers = await _collectionCipherRepository.GetManyByUserIdCipherIdAsync(userId, cipherId);
|
var collectionCiphers = await _collectionCipherRepository.GetManyByUserIdCipherIdAsync(userId, cipherId);
|
||||||
return new CipherDetailsResponseModel(cipher, collectionCiphers);
|
return new CipherDetailsResponseModel(cipher, _globalSettings, collectionCiphers);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("")]
|
[HttpGet("")]
|
||||||
@ -73,7 +76,7 @@ namespace Bit.Api.Controllers
|
|||||||
{
|
{
|
||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
var ciphers = await _cipherRepository.GetManyByUserIdAsync(userId);
|
var ciphers = await _cipherRepository.GetManyByUserIdAsync(userId);
|
||||||
var responses = ciphers.Select(c => new CipherResponseModel(c)).ToList();
|
var responses = ciphers.Select(c => new CipherResponseModel(c, _globalSettings)).ToList();
|
||||||
return new ListResponseModel<CipherResponseModel>(responses);
|
return new ListResponseModel<CipherResponseModel>(responses);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -86,7 +89,7 @@ namespace Bit.Api.Controllers
|
|||||||
var collectionCiphers = await _collectionCipherRepository.GetManyByUserIdAsync(userId);
|
var collectionCiphers = await _collectionCipherRepository.GetManyByUserIdAsync(userId);
|
||||||
var collectionCiphersGroupDict = collectionCiphers.GroupBy(c => c.CipherId).ToDictionary(s => s.Key);
|
var collectionCiphersGroupDict = collectionCiphers.GroupBy(c => c.CipherId).ToDictionary(s => s.Key);
|
||||||
|
|
||||||
var responses = ciphers.Select(c => new CipherDetailsResponseModel(c, collectionCiphersGroupDict));
|
var responses = ciphers.Select(c => new CipherDetailsResponseModel(c, _globalSettings, collectionCiphersGroupDict));
|
||||||
return new ListResponseModel<CipherDetailsResponseModel>(responses);
|
return new ListResponseModel<CipherDetailsResponseModel>(responses);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -105,7 +108,8 @@ namespace Bit.Api.Controllers
|
|||||||
var collectionCiphers = await _collectionCipherRepository.GetManyByOrganizationIdAsync(orgIdGuid);
|
var collectionCiphers = await _collectionCipherRepository.GetManyByOrganizationIdAsync(orgIdGuid);
|
||||||
var collectionCiphersGroupDict = collectionCiphers.GroupBy(c => c.CipherId).ToDictionary(s => s.Key);
|
var collectionCiphersGroupDict = collectionCiphers.GroupBy(c => c.CipherId).ToDictionary(s => s.Key);
|
||||||
|
|
||||||
var responses = ciphers.Select(c => new CipherMiniDetailsResponseModel(c, collectionCiphersGroupDict));
|
var responses = ciphers.Select(c => new CipherMiniDetailsResponseModel(c, _globalSettings,
|
||||||
|
collectionCiphersGroupDict));
|
||||||
return new ListResponseModel<CipherMiniDetailsResponseModel>(responses);
|
return new ListResponseModel<CipherMiniDetailsResponseModel>(responses);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -228,6 +232,11 @@ namespace Bit.Api.Controllers
|
|||||||
throw new BadRequestException("Invalid content.");
|
throw new BadRequestException("Invalid content.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(Request.ContentLength > 105906176) // 101 MB, give em' 1 extra MB for cushion
|
||||||
|
{
|
||||||
|
throw new BadRequestException("Max file size is 100 MB.");
|
||||||
|
}
|
||||||
|
|
||||||
var idGuid = new Guid(id);
|
var idGuid = new Guid(id);
|
||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
var cipher = await _cipherRepository.GetByIdAsync(idGuid, userId);
|
var cipher = await _cipherRepository.GetByIdAsync(idGuid, userId);
|
||||||
|
@ -21,17 +21,20 @@ namespace Bit.Api.Controllers
|
|||||||
private readonly ICipherService _cipherService;
|
private readonly ICipherService _cipherService;
|
||||||
private readonly IUserService _userService;
|
private readonly IUserService _userService;
|
||||||
private readonly CurrentContext _currentContext;
|
private readonly CurrentContext _currentContext;
|
||||||
|
private readonly GlobalSettings _globalSettings;
|
||||||
|
|
||||||
public LoginsController(
|
public LoginsController(
|
||||||
ICipherRepository cipherRepository,
|
ICipherRepository cipherRepository,
|
||||||
ICipherService cipherService,
|
ICipherService cipherService,
|
||||||
IUserService userService,
|
IUserService userService,
|
||||||
CurrentContext currentContext)
|
CurrentContext currentContext,
|
||||||
|
GlobalSettings globalSettings)
|
||||||
{
|
{
|
||||||
_cipherRepository = cipherRepository;
|
_cipherRepository = cipherRepository;
|
||||||
_cipherService = cipherService;
|
_cipherService = cipherService;
|
||||||
_userService = userService;
|
_userService = userService;
|
||||||
_currentContext = currentContext;
|
_currentContext = currentContext;
|
||||||
|
_globalSettings = globalSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("{id}")]
|
[HttpGet("{id}")]
|
||||||
@ -44,7 +47,7 @@ namespace Bit.Api.Controllers
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -58,7 +61,7 @@ namespace Bit.Api.Controllers
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,7 +70,7 @@ namespace Bit.Api.Controllers
|
|||||||
{
|
{
|
||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
var logins = await _cipherRepository.GetManyByTypeAndUserIdAsync(Core.Enums.CipherType.Login, userId);
|
var logins = await _cipherRepository.GetManyByTypeAndUserIdAsync(Core.Enums.CipherType.Login, userId);
|
||||||
var responses = logins.Select(l => new LoginResponseModel(l)).ToList();
|
var responses = logins.Select(l => new LoginResponseModel(l, _globalSettings)).ToList();
|
||||||
return new ListResponseModel<LoginResponseModel>(responses);
|
return new ListResponseModel<LoginResponseModel>(responses);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -78,7 +81,7 @@ namespace Bit.Api.Controllers
|
|||||||
var login = model.ToCipherDetails(userId);
|
var login = model.ToCipherDetails(userId);
|
||||||
await _cipherService.SaveDetailsAsync(login, userId);
|
await _cipherService.SaveDetailsAsync(login, userId);
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -94,7 +97,7 @@ namespace Bit.Api.Controllers
|
|||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
await _cipherService.SaveAsync(login, userId, true);
|
await _cipherService.SaveAsync(login, userId, true);
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -118,7 +121,7 @@ namespace Bit.Api.Controllers
|
|||||||
|
|
||||||
await _cipherService.SaveDetailsAsync(model.ToCipherDetails(login), userId);
|
await _cipherService.SaveDetailsAsync(model.ToCipherDetails(login), userId);
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -136,7 +139,7 @@ namespace Bit.Api.Controllers
|
|||||||
var userId = _userService.GetProperUserId(User).Value;
|
var userId = _userService.GetProperUserId(User).Value;
|
||||||
await _cipherService.SaveAsync(model.ToCipher(login), userId, true);
|
await _cipherService.SaveAsync(model.ToCipher(login), userId, true);
|
||||||
|
|
||||||
var response = new LoginResponseModel(login);
|
var response = new LoginResponseModel(login, _globalSettings);
|
||||||
return response;
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
},
|
},
|
||||||
"attachment": {
|
"attachment": {
|
||||||
"connectionString": "SECRET",
|
"connectionString": "SECRET",
|
||||||
"baseUrl": "http://localhost:4000/"
|
"baseUrl": "http://localhost:4000/attachments/"
|
||||||
},
|
},
|
||||||
"documentDb": {
|
"documentDb": {
|
||||||
"uri": "SECRET",
|
"uri": "SECRET",
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
<aspNetCore processPath="%LAUNCHER_PATH%" arguments="%LAUNCHER_ARGS%" stdoutLogEnabled="false" stdoutLogFile=".\logs\stdout" forwardWindowsAuthToken="false"/>
|
<aspNetCore processPath="%LAUNCHER_PATH%" arguments="%LAUNCHER_ARGS%" stdoutLogEnabled="false" stdoutLogFile=".\logs\stdout" forwardWindowsAuthToken="false"/>
|
||||||
<security>
|
<security>
|
||||||
<requestFiltering>
|
<requestFiltering>
|
||||||
<requestLimits maxQueryString="5120"/>
|
<requestLimits maxQueryString="5120" maxAllowedContentLength="105906176"/>
|
||||||
</requestFiltering>
|
</requestFiltering>
|
||||||
</security>
|
</security>
|
||||||
</system.webServer>
|
</system.webServer>
|
||||||
|
46
src/Core/Models/Api/Response/AttachmentResponseModel.cs
Normal file
46
src/Core/Models/Api/Response/AttachmentResponseModel.cs
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
using Bit.Core.Models.Data;
|
||||||
|
using Bit.Core.Models.Table;
|
||||||
|
using Newtonsoft.Json;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
|
||||||
|
namespace Bit.Core.Models.Api
|
||||||
|
{
|
||||||
|
public class AttachmentResponseModel : ResponseModel
|
||||||
|
{
|
||||||
|
public AttachmentResponseModel(string id, CipherAttachment.MetaData data, Cipher cipher, GlobalSettings globalSettings)
|
||||||
|
: base("attachment")
|
||||||
|
{
|
||||||
|
Id = id;
|
||||||
|
Url = $"{globalSettings.Attachment.BaseUrl}{cipher.Id}/{id}";
|
||||||
|
FileName = data.FileName;
|
||||||
|
Size = data.SizeString;
|
||||||
|
SizeName = Utilities.CoreHelpers.ReadableBytesSize(data.Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
public string Id { get; set; }
|
||||||
|
public string Url { get; set; }
|
||||||
|
public string FileName { get; set; }
|
||||||
|
public string Size { get; set; }
|
||||||
|
public string SizeName { get; set; }
|
||||||
|
|
||||||
|
public static IEnumerable<AttachmentResponseModel> FromCipher(Cipher cipher, GlobalSettings globalSettings)
|
||||||
|
{
|
||||||
|
if(string.IsNullOrWhiteSpace(cipher.Attachments))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var attachments =
|
||||||
|
JsonConvert.DeserializeObject<Dictionary<string, CipherAttachment.MetaData>>(cipher.Attachments);
|
||||||
|
return attachments.Select(a => new AttachmentResponseModel(a.Key, a.Value, cipher, globalSettings));
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -8,7 +8,7 @@ namespace Bit.Core.Models.Api
|
|||||||
{
|
{
|
||||||
public class CipherMiniResponseModel : ResponseModel
|
public class CipherMiniResponseModel : ResponseModel
|
||||||
{
|
{
|
||||||
public CipherMiniResponseModel(Cipher cipher, string obj = "cipherMini")
|
public CipherMiniResponseModel(Cipher cipher, GlobalSettings globalSettings, string obj = "cipherMini")
|
||||||
: base(obj)
|
: base(obj)
|
||||||
{
|
{
|
||||||
if(cipher == null)
|
if(cipher == null)
|
||||||
@ -20,6 +20,7 @@ namespace Bit.Core.Models.Api
|
|||||||
Type = cipher.Type;
|
Type = cipher.Type;
|
||||||
RevisionDate = cipher.RevisionDate;
|
RevisionDate = cipher.RevisionDate;
|
||||||
OrganizationId = cipher.OrganizationId?.ToString();
|
OrganizationId = cipher.OrganizationId?.ToString();
|
||||||
|
Attachments = AttachmentResponseModel.FromCipher(cipher, globalSettings);
|
||||||
|
|
||||||
switch(cipher.Type)
|
switch(cipher.Type)
|
||||||
{
|
{
|
||||||
@ -35,13 +36,14 @@ namespace Bit.Core.Models.Api
|
|||||||
public string OrganizationId { get; set; }
|
public string OrganizationId { get; set; }
|
||||||
public Enums.CipherType Type { get; set; }
|
public Enums.CipherType Type { get; set; }
|
||||||
public dynamic Data { get; set; }
|
public dynamic Data { get; set; }
|
||||||
|
public IEnumerable<AttachmentResponseModel> Attachments { get; set; }
|
||||||
public DateTime RevisionDate { get; set; }
|
public DateTime RevisionDate { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class CipherResponseModel : CipherMiniResponseModel
|
public class CipherResponseModel : CipherMiniResponseModel
|
||||||
{
|
{
|
||||||
public CipherResponseModel(CipherDetails cipher, string obj = "cipher")
|
public CipherResponseModel(CipherDetails cipher, GlobalSettings globalSettings, string obj = "cipher")
|
||||||
: base(cipher, obj)
|
: base(cipher, globalSettings, obj)
|
||||||
{
|
{
|
||||||
FolderId = cipher.FolderId?.ToString();
|
FolderId = cipher.FolderId?.ToString();
|
||||||
Favorite = cipher.Favorite;
|
Favorite = cipher.Favorite;
|
||||||
@ -55,9 +57,9 @@ namespace Bit.Core.Models.Api
|
|||||||
|
|
||||||
public class CipherDetailsResponseModel : CipherResponseModel
|
public class CipherDetailsResponseModel : CipherResponseModel
|
||||||
{
|
{
|
||||||
public CipherDetailsResponseModel(CipherDetails cipher,
|
public CipherDetailsResponseModel(CipherDetails cipher, GlobalSettings globalSettings,
|
||||||
IDictionary<Guid, IGrouping<Guid, CollectionCipher>> collectionCiphers, string obj = "cipherDetails")
|
IDictionary<Guid, IGrouping<Guid, CollectionCipher>> collectionCiphers, string obj = "cipherDetails")
|
||||||
: base(cipher, obj)
|
: base(cipher, globalSettings, obj)
|
||||||
{
|
{
|
||||||
if(collectionCiphers.ContainsKey(cipher.Id))
|
if(collectionCiphers.ContainsKey(cipher.Id))
|
||||||
{
|
{
|
||||||
@ -69,9 +71,9 @@ namespace Bit.Core.Models.Api
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public CipherDetailsResponseModel(CipherDetails cipher, IEnumerable<CollectionCipher> collectionCiphers,
|
public CipherDetailsResponseModel(CipherDetails cipher, GlobalSettings globalSettings,
|
||||||
string obj = "cipherDetails")
|
IEnumerable<CollectionCipher> collectionCiphers, string obj = "cipherDetails")
|
||||||
: base(cipher, obj)
|
: base(cipher, globalSettings, obj)
|
||||||
{
|
{
|
||||||
CollectionIds = collectionCiphers.Select(c => c.CollectionId);
|
CollectionIds = collectionCiphers.Select(c => c.CollectionId);
|
||||||
}
|
}
|
||||||
@ -81,9 +83,9 @@ namespace Bit.Core.Models.Api
|
|||||||
|
|
||||||
public class CipherMiniDetailsResponseModel : CipherMiniResponseModel
|
public class CipherMiniDetailsResponseModel : CipherMiniResponseModel
|
||||||
{
|
{
|
||||||
public CipherMiniDetailsResponseModel(Cipher cipher,
|
public CipherMiniDetailsResponseModel(Cipher cipher, GlobalSettings globalSettings,
|
||||||
IDictionary<Guid, IGrouping<Guid, CollectionCipher>> collectionCiphers, string obj = "cipherMiniDetails")
|
IDictionary<Guid, IGrouping<Guid, CollectionCipher>> collectionCiphers, string obj = "cipherMiniDetails")
|
||||||
: base(cipher, obj)
|
: base(cipher, globalSettings, obj)
|
||||||
{
|
{
|
||||||
if(collectionCiphers.ContainsKey(cipher.Id))
|
if(collectionCiphers.ContainsKey(cipher.Id))
|
||||||
{
|
{
|
||||||
|
@ -1,12 +1,13 @@
|
|||||||
using System;
|
using System;
|
||||||
using Core.Models.Data;
|
using Core.Models.Data;
|
||||||
using Bit.Core.Models.Table;
|
using Bit.Core.Models.Table;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Bit.Core.Models.Api
|
namespace Bit.Core.Models.Api
|
||||||
{
|
{
|
||||||
public class LoginResponseModel : ResponseModel
|
public class LoginResponseModel : ResponseModel
|
||||||
{
|
{
|
||||||
public LoginResponseModel(Cipher cipher, string obj = "login")
|
public LoginResponseModel(Cipher cipher, GlobalSettings globalSettings, string obj = "login")
|
||||||
: base(obj)
|
: base(obj)
|
||||||
{
|
{
|
||||||
if(cipher == null)
|
if(cipher == null)
|
||||||
@ -30,10 +31,11 @@ namespace Bit.Core.Models.Api
|
|||||||
Notes = data.Notes;
|
Notes = data.Notes;
|
||||||
RevisionDate = cipher.RevisionDate;
|
RevisionDate = cipher.RevisionDate;
|
||||||
Edit = true;
|
Edit = true;
|
||||||
|
Attachments = AttachmentResponseModel.FromCipher(cipher, globalSettings);
|
||||||
}
|
}
|
||||||
|
|
||||||
public LoginResponseModel(CipherDetails cipher, string obj = "login")
|
public LoginResponseModel(CipherDetails cipher, GlobalSettings globalSettings, string obj = "login")
|
||||||
: this(cipher as Cipher, obj)
|
: this(cipher as Cipher, globalSettings, obj)
|
||||||
{
|
{
|
||||||
FolderId = cipher.FolderId?.ToString();
|
FolderId = cipher.FolderId?.ToString();
|
||||||
Favorite = cipher.Favorite;
|
Favorite = cipher.Favorite;
|
||||||
@ -50,6 +52,7 @@ namespace Bit.Core.Models.Api
|
|||||||
public string Username { get; set; }
|
public string Username { get; set; }
|
||||||
public string Password { get; set; }
|
public string Password { get; set; }
|
||||||
public string Notes { get; set; }
|
public string Notes { get; set; }
|
||||||
|
public IEnumerable<AttachmentResponseModel> Attachments { get; set; }
|
||||||
public DateTime RevisionDate { get; set; }
|
public DateTime RevisionDate { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,7 @@ namespace Bit.Core.Services
|
|||||||
public AzureAttachmentStorageService(
|
public AzureAttachmentStorageService(
|
||||||
GlobalSettings globalSettings)
|
GlobalSettings globalSettings)
|
||||||
{
|
{
|
||||||
var storageAccount = CloudStorageAccount.Parse(globalSettings.Storage.ConnectionString);
|
var storageAccount = CloudStorageAccount.Parse(globalSettings.Attachment.ConnectionString);
|
||||||
_blobClient = storageAccount.CreateCloudBlobClient();
|
_blobClient = storageAccount.CreateCloudBlobClient();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -38,7 +38,7 @@ namespace Bit.Core.Services
|
|||||||
if(_attachmentsContainer == null)
|
if(_attachmentsContainer == null)
|
||||||
{
|
{
|
||||||
_attachmentsContainer = _blobClient.GetContainerReference(AttchmentContainerName);
|
_attachmentsContainer = _blobClient.GetContainerReference(AttchmentContainerName);
|
||||||
await _attachmentsContainer.CreateIfNotExistsAsync();
|
await _attachmentsContainer.CreateIfNotExistsAsync(BlobContainerPublicAccessType.Blob, null, null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -204,5 +204,43 @@ namespace Bit.Core.Utilities
|
|||||||
return sb.ToString();
|
return sb.ToString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ref: https://stackoverflow.com/a/11124118/1090359
|
||||||
|
// Returns the human-readable file size for an arbitrary 64-bit file size .
|
||||||
|
// The format is "0.## XB", ex: "4.2 KB" or "1.43 GB"
|
||||||
|
public static string ReadableBytesSize(long size)
|
||||||
|
{
|
||||||
|
// Get absolute value
|
||||||
|
var absoluteSize = (size < 0 ? -size : size);
|
||||||
|
|
||||||
|
// Determine the suffix and readable value
|
||||||
|
string suffix;
|
||||||
|
double readable;
|
||||||
|
if(absoluteSize >= 0x40000000) // 1 Gigabyte
|
||||||
|
{
|
||||||
|
suffix = "GB";
|
||||||
|
readable = (size >> 20);
|
||||||
|
}
|
||||||
|
else if(absoluteSize >= 0x100000) // 1 Megabyte
|
||||||
|
{
|
||||||
|
suffix = "MB";
|
||||||
|
readable = (size >> 10);
|
||||||
|
}
|
||||||
|
else if(absoluteSize >= 0x400) // 1 Kilobyte
|
||||||
|
{
|
||||||
|
suffix = "KB";
|
||||||
|
readable = size;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return absoluteSize.ToString("0 Bytes"); // Byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// Divide by 1024 to get fractional value
|
||||||
|
readable = (readable / 1024);
|
||||||
|
|
||||||
|
// Return formatted number with suffix
|
||||||
|
return readable.ToString("0.## ") + suffix;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user