base on Script that organizes the Google Takeout archive into one big chronological folder [![AUR](https://img.shields.io/aur/version/gpth-bin?logo=arch-linux)](https://aur.archlinux.org/packages/gpth-bin) [![total Github Releases downloads](https://img.shields.io/github/downloads/TheLastGimbus/GooglePhotosTakeoutHelper/total?label=total%20downloads)](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/releases/) [![latest version downloads](https://img.shields.io/github/downloads/TheLastGimbus/GooglePhotosTakeoutHelper/latest/total?label=latest%20version%20downloads)](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/releases/latest) [![resolved Github issues](https://img.shields.io/github/issues-closed/TheLastGimbus/GooglePhotosTakeoutHelper?label=resolved%20issues)](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/issues) [![commit activity](https://img.shields.io/github/commit-activity/y/TheLastGimbus/GooglePhotosTakeoutHelper)](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/graphs/contributors) # Google Photos Takeout Helper πŸ“ΈπŸ†˜ ## What is this for 🧐 If you ever want to move from Google Photos to other platform/solution, your fastest choice to export all photos is [Google Takeout πŸ₯‘](https://takeout.google.com/) But when you download it, you will find yourself with zips with hundreds of little folders with weird `.json` files inside 🍝. What if you want to just have one folder with all photos, in chronological order? Good luck copying all of that πŸ™ƒ This script does just that - it organizes and cleans up your Takeout for you 🧹😌 It will take all of those folders, find all photos in them, set their `file last modified` correctly, and put it in one big folder (or folders divided by a month) πŸ—„ ## How to use: Since `v3.2.0`, `gpth` is interactive πŸŽ‰ - you don't need to type any complicated arguments - just get your takeout, run gpth, and follow prompted instructions πŸ’ƒ If you want to run it on Synology, have problems with interactive, or just love cmd, look at ["Running manually with cmd"](#running-manually-with-cmd). Otherwise, just: ### 1. Get all your photos from [Google Takeout](https://takeout.google.com/) πŸ“₯ "deselect all" and then select only Google Photos <img width="75%" alt="gpth usage image tutorial" src="https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/assets/40139196/8e85f58c-9958-466a-a176-51af85bb73dd"> ### 2. Unzip them all and merge into one, so that all "Takeout" folders become one **NOTE:** Keep those original zips, you may need them if anything goes wrong <img width="75%" alt="Unzip image tutorial" src="https://user-images.githubusercontent.com/40139196/229361367-b9803ab9-2724-4ddf-9af5-4df507e02dfe.png"> ### 3. Download the executable for your system from [releases tab](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/releases) πŸ›’ ([also available on AUR 😏](https://aur.archlinux.org/packages/gpth-bin)) ### 4. Run `gpth` - On Windoza: just double-click the downloaded `.exe` πŸŽ‰ - tell windoza defender that it's safe, and follow prompted instructions 🧾 - On Mac/Linux: open terminal, `cd` to the folder with downloaded executable and run it: ```bash # if you have Mac with M1/M2 chip, you need to enable x86 emulation # otherwise, just skip it softwareupdate --install-rosetta cd Downloads # probably # add execute permission for file chmod +x gpth-macos # or gpth-linux # tell MacOS Gatekeeper to not worry xattr -r -d com.apple.quarantine gpth-macos # run it πŸƒ ./gpth-macos # or ./gpth-linux # follow prompted instructions πŸ₯° ``` **Fun fact:** `gpth` *moves* files around by default - so if anything goes wrong mid-way, re-export the zips again :pray: ### 5. Most of your photos should have correct original EXIFs (metadata), but if you want, you can insert them everywhere with `exiftool`, so you won't lose their creation time - Download Phil Harvey's exiftool: https://exiftool.sourceforge.net/ - Open the cmd/terminal, and run ```bash # cd to folder where you downloaded exiftool cd Downloads # run it on your output folder: # (the '-r' means "run on all files/in sub-folders" aka recursively) # (Make sure you didn't change file modify dates aka didn't do anything with the files after running gpth exiftool -overwrite_original -r -if 'not defined DateTimeOriginal' -P "-AllDates<FileModifyDate" "your/output/folder/ALL_PHOTOS/" ``` Done! Enjoy your photos!!! ### Running manually with cmd You may still need this mode if: - You want to run on Synology where there are no ui programs required for interactive - You can read/discuss in [#157](https://github.com/TheLastGimbus/GooglePhotosTakeoutHelper/discussions/157) for any help - ~~Interactive unzipping crashes for you (known issue in windoza 😒 #178)~~ - disabled for now - Want to use this in other script/automation In that case: 1. Manually unzip all your takeout zips and merge them into one folder 2. Open cmd and: - For windoza: ```bash # psst: in windoza cmd, you can just drag and drop files/folders to type them in # 1. change working directory to where gpth.exe is: cd Downloads # Most probably # run it, selecting input and output folders with options like this: # (you can try to drag and drop them) gpth.exe --input "Downloads\you\input\folder" --output "C:\some\other\location" --albums "shortcut" # select which album solution you like - see --help for all of them # remember to use "" ! ``` - For Linux/macOS: ```bash # ssh/whatever to where you're running it cd Downloads # folder with gpth chmod +x gpth # add execute permission # tell MacOS Gatekeeper to not worry xattr -r -d com.apple.quarantine gpth-macos ./gpth --input "/some/input/folder" --output "other/output/folder" --albums "shortcut" # select which album solution you like - see --help for all of them ``` You can check all cmd flags by running `gpth --help` - for example, the `--divide-to-dates` flag ## If I helped you, you can consider donating me β˜• I spent **a lot of** time fixing bugs and making this work stable πŸ’– - would be super thankful for any donations πŸ₯° [![Donate](https://img.shields.io/badge/Donate-PayPal-blue.svg?logo=paypal&style=for-the-badge)](https://www.paypal.me/TheLastGimbus) [![Donate using ko-fi](https://ko-fi.com/img/githubbutton_sm.svg)](https://ko-fi.com/A0A6HO71P) [![Donate using Liberapay](https://liberapay.com/assets/widgets/donate.svg)](https://liberapay.com/TheLastGimbus/donate) ## After exporting πŸ€” ### Be aware if you move your photos on your Android phone... ☝ (99% of the times), if you move some files in Android, their creation and modification time is reset to current. "Simple Gallery" app usually keeps original file creation time when moving and copying (but I don't guarantee it). It's also pretty cool - check it out: https://github.com/SimpleMobileTools/Simple-Gallery ### What to do when you got rid of Google Photos? What are the alternatives? πŸ—Ί - I really recommend you using [Syncthing](https://syncthing.net/) for syncing your photos and files across devices. It does so through your local Wi-Fi, so you're not dependent on any service or internet connection. It will also keep original file creation date and metadata, so it resolves Android issue that I mentioned before. - [Immich](https://immich.app/) aims to be full blown GPhotos replacement - it's still under development, but already looks great! - Same with [Photoprism](https://photoprism.org/), tho this one is in development longer (may be more mature) - If you want something more centralized but also self-hosted, [Nextcloud](https://nextcloud.com) is a nice choice, but its approach to photos is still not perfect. (And you need to set up your own server) ### Other Takeout projects I used this tool to export my notes to markdown - you can then edit them with any markdown editor you like :) https://github.com/vHanda/google-keep-exporter ### Where is the Python script 🐍 ?? Yeah, the whole thing got re-written in Dart, and now it's way more stable and faster. If you still want Python for some reason, check out v2.x - in releases/tags ### TODO (Pull Requests welcome): - [ ] GPS data: from JSON to Exif - ~~Thank you @DalenW πŸ’–~~ still thank you, but it is now missing in the Dart version - [ ] Writing data from `.json`s back to `EXIF` data - [x] Some way to handle albums - THANK YOU @bitsondatadev 😘 πŸŽ‰ πŸ’ƒ ", Assign "at most 3 tags" to the expected json: {"id":"9550","tags":[]} "only from the tags list I provide: [{"id":77,"name":"3d"},{"id":89,"name":"agent"},{"id":17,"name":"ai"},{"id":54,"name":"algorithm"},{"id":24,"name":"api"},{"id":44,"name":"authentication"},{"id":3,"name":"aws"},{"id":27,"name":"backend"},{"id":60,"name":"benchmark"},{"id":72,"name":"best-practices"},{"id":39,"name":"bitcoin"},{"id":37,"name":"blockchain"},{"id":1,"name":"blog"},{"id":45,"name":"bundler"},{"id":58,"name":"cache"},{"id":21,"name":"chat"},{"id":49,"name":"cicd"},{"id":4,"name":"cli"},{"id":64,"name":"cloud-native"},{"id":48,"name":"cms"},{"id":61,"name":"compiler"},{"id":68,"name":"containerization"},{"id":92,"name":"crm"},{"id":34,"name":"data"},{"id":47,"name":"database"},{"id":8,"name":"declarative-gui "},{"id":9,"name":"deploy-tool"},{"id":53,"name":"desktop-app"},{"id":6,"name":"dev-exp-lib"},{"id":59,"name":"dev-tool"},{"id":13,"name":"ecommerce"},{"id":26,"name":"editor"},{"id":66,"name":"emulator"},{"id":62,"name":"filesystem"},{"id":80,"name":"finance"},{"id":15,"name":"firmware"},{"id":73,"name":"for-fun"},{"id":2,"name":"framework"},{"id":11,"name":"frontend"},{"id":22,"name":"game"},{"id":81,"name":"game-engine "},{"id":23,"name":"graphql"},{"id":84,"name":"gui"},{"id":91,"name":"http"},{"id":5,"name":"http-client"},{"id":51,"name":"iac"},{"id":30,"name":"ide"},{"id":78,"name":"iot"},{"id":40,"name":"json"},{"id":83,"name":"julian"},{"id":38,"name":"k8s"},{"id":31,"name":"language"},{"id":10,"name":"learning-resource"},{"id":33,"name":"lib"},{"id":41,"name":"linter"},{"id":28,"name":"lms"},{"id":16,"name":"logging"},{"id":76,"name":"low-code"},{"id":90,"name":"message-queue"},{"id":42,"name":"mobile-app"},{"id":18,"name":"monitoring"},{"id":36,"name":"networking"},{"id":7,"name":"node-version"},{"id":55,"name":"nosql"},{"id":57,"name":"observability"},{"id":46,"name":"orm"},{"id":52,"name":"os"},{"id":14,"name":"parser"},{"id":74,"name":"react"},{"id":82,"name":"real-time"},{"id":56,"name":"robot"},{"id":65,"name":"runtime"},{"id":32,"name":"sdk"},{"id":71,"name":"search"},{"id":63,"name":"secrets"},{"id":25,"name":"security"},{"id":85,"name":"server"},{"id":86,"name":"serverless"},{"id":70,"name":"storage"},{"id":75,"name":"system-design"},{"id":79,"name":"terminal"},{"id":29,"name":"testing"},{"id":12,"name":"ui"},{"id":50,"name":"ux"},{"id":88,"name":"video"},{"id":20,"name":"web-app"},{"id":35,"name":"web-server"},{"id":43,"name":"webassembly"},{"id":69,"name":"workflow"},{"id":87,"name":"yaml"}]" returns me the "expected json"