Archive your photos and videos to Azure Archive Storage with the lowest cost of β¬0.00084/GB/mo or β¬0.8/TB/mo. See detailed pricing.
This app uploads and optionally encrypts your files like IMG_20190727_123456.jpg
or DSC_5438.NEF
, groups them by date into directories like 2019
/07
/27
and sets their tiers to Archive to save cost.
Requirements:
- Microsoft Azure subscription
- Azure Storage Account (General Purpose v2 or Blob)
- .NET Core 9 Runtime installed on your machine
Download executable from Releases or clone the source code and build it yourself. Read the best practices.
For basic usage, you have to specify only the path to the folder to upload and the connection string to your storage account.
Run on Windows:
.\PhotoArchiver.exe upload "D:\OneDrive\Camera Roll"
Run on Linux/Unix:
dotnet PhotoArchiver.dll upload "D:\OneDrive\Camera Roll"
You can also save your credentials to a configuration file. See below.
Start download of blobs that are immediately accessible.
.\PhotoArchiver.exe download "2019-08-14" "D:\OneDrive\Camera Roll"
You may see that your archived blobs are not accessible right now and they are being rehydrated. In that case, check back after 12 hours and continue:
.\PhotoArchiver.exe rehydrate "2019-08-14"
.\PhotoArchiver.exe download "2019-08-14" "D:\OneDrive\Camera Roll"
Configuration is based on the .NET Standard library and the application reads it from JSON file and/or command-line arguments.
Storage
properties of Storage AccountConnectionString
: the connection string for your Azure StorageContainer
(default"photos"
): the name of the container to upload files toDirectoryFormat
(default"{0:yyyy}/{0:MM}/{0:dd}"
): format string for blob directories that blob are organized into (see docs)
Upload
upload settingsPath
: the directory to upload the files fromAccessTier
(default"Cool"
): possible values:"Hot"
,"Cold"
,"Archive"
SearchPattern
(default"**/*"
): glob search pattern for files to upload (see docs)Delete
(defaultfalse
): delete files after successful uploadVerify
(defaulttrue
): verifies uploaded blob by its MD5 hashSkip
(default0
): skip the first given number of files (by name ascending)Take
(defaultnull
): upload only the first given number of files (by name ascending), can be combined withSkip
ConflictResolution
(default"Skip"
) possible values:"Skip"
: whenever a conflict is found, log as warning and skip"KeepBoth"
: the hash of the file to be uploaded is appended to its file name, right before its extension, and gets uploaded. The already existing blob is kept and not modified."Overwrite"
: the existing blob gets overwritten, if it is in Archive tier, deleted and then reuploaded with the same name"SnapshotAndOverwrite"
: a snapshot is taken of the existing blob and then it gets overwritten (see"Overwrite"
option). If the blob is in Archive tier, taking a snapshot is not possible, so it is skipped and logged as a warning.
ParallelBlockCount
(default1
): the number of blocks of a single blob that may be simultaneously uploaded (see docs)
Thumbnails
thumbnail settingsMaxWidth
: maximum width of the thumbnail imagesMaxHeight
: maximum height of the thumbnail imagesQuality
(default0.50
): quality of JPEG compression (0-1: min-max quality)Container
(default"photos-thumbnails"
): blob container to put the thumbnails toForce
(defaultfalse
): create thumbnails for existing blobs
Download
download settingsPath
: the directory to download the blobs toDate
(orStartDate
andEndDate
): the creation date of the media to downloadStartDate
: the earliest creation date of the media to downloadEndDate
: the latest creation date of the media to downloadTags
: filter by tags (tagged by Computer Vision)People
: filter by IDs of people (tagged by Face API)RehydrationTier
(default"Hot"
): rehydrate archived blobs to this tier temporarily for downloadVerify
(defaultfalse
): verify downloaded files by MD5 hashArchive
(defaultfalse
): set access tier of blobs toArchive
after download
You can persist your configuration into a file named appsettings.json
, for example:
{
"Storage": {
"ConnectionString": "SECRET"
},
"Upload": {
"Delete": true
}
}
Or supply as CLI arguments:
.\PhotoArchiver.exe --Upload:Path "D:\OneDrive\Camera Roll" --Storage:Archive false --Upload:Delete true
Estimates transaction and storage costs for your uploaded data.
Requirements:
- Configuration, get pricing from Azure Storage Pricing
Configuration:
Costs
set the prices based on your region and redundancy for cost estimations (see pricing)Currency
(default"$"
): currency to display costsListOrCreateContainerPricePer10000
ReadPricePer10000
WritePricePer10000
OtherPricePer10000
DataStoragePricePerGB
GRSDataTransferPricePerGB
: leave it empty if your Storage Account is not geo-replicatedKeyVaultTransactionPricePer10000
: leave it empty if you don't use Key VaultComputerVisionDescribeTransactionPricePer1000
: leave if empty if you don't use Computer VisionOutboundDataTransferPricePerGB
Collect detailed operational logs.
Requirements:
Configuration:
ApplicationInsights
see ApplicationInsightsServiceOptions for referenceConnectionString
: connection string for the resourceEnableAdaptiveSampling
: enable/disable Adaptive Sampling
Configure update options.
Configuration:
Update
Enabled
: check for updates at startupStop
: do not proceed if an update is found
Supported file types:
- Photos
- Any JPEG with EXIF
- Android
- iOS (.JPG, .HEIF, .HEIC)
- Office Lens
- Windows Phone
- Videos
- Any video with Quick Time metadata (.MOV, .MP4)
- Any video with RIFF IDIT metadat (.AVI)
- Android (.MP4)
- iOS (.MOV)
- Windows Phone (.MP4)
- RAW (with EXIF, matching JPEG or date in file name)
- Canon Raw Version 2 (.CR2)
- Digital Negative (.DNG), iOS/Windows Phone RAW
- Nikon Electric Format (.NEF)
- GoPro (.GPR, .WAV)
Supported sources to upload from:
- Local File System
- Synced cloud storage
- OneDrive
- Google Drive
- USB, CD, DVD drives
Metadata appended to blobs:
OriginalFileName
: the full path of the file on the diskOriginalFileSize
: the size of the unencrypted file which may be different when encryptedCreatedAt
: the detected date and time the media was originally createdCaption
: the image description generated by Computer VisionTags
: tags generated by Computer VisionPeople
: IDs of persons identified by Face API
A single file is uploaded at a time.
Use at your own risk. The creator of this software takes no responsibility in moving, storing, deleting or processing your data in any form. It is your responsibility to keep your encryption keys safe. Also, cost estimations are only for information purposes, to get exact and detailed information view Azure Storage pricing page, or check your actual consumption in Azure Portal.
Requirements:
- Visual Studio 2022
- .NET 9 SDK
Also, set Update:Enabled
to false
, if you want to turn off update checks.
- Make sure you have a valid key for your Storage Account. You can try it in another tool like Storage Explorer.
- Make sure your Storage Account is accessible from your network. You can check this on the Firewall tab of your Storage Account.
- If you want to archive blobs, make sure your Storage Account is the new kind (GPv2 or Blob), so it supports blob level tiers.
- Make sure you (and this application) have at least Read permission to the folder you want to upload.
- If you want to delete the files as well, make sure you have Delete permission.
- Also, that the files are not in use anywhere else.
- If you want to encrypt using Key Vault, make sure your AD App is defined in the Key Vault access policies.
- Computer Vision and Face API supports files up to 4 MB in size and only JPEG images.
- Create a Storage Account
- Manage blobs on Azure Portal
- Manage blobs using Azure Storage Explorer
- Use lifecycle management policies to archive blobs
- ImageSharp library for resizing images
- MetadataExtractor library for reading metadata from videos