Export an Apple iPhoto image library to a new directory (using hardlinks) with all metadata saved in XMP sidecar files. Useful if you do not want your iPhoto library moved to a cloud-centric Photos app with less features.
This script will export your Apple iPhoto library to a new directory structure (folders according to iPhoto "Events") using a standard metadata format (XMP sidecar files) wherever possible, so that you can import your image library to a different application (DAM = Digital Asset Management) and keep all your image metadata intact.
Hard links are used to copy the actual images and videos, so very little additional disk space is required. This requires that the target directory is on the same volume as the iPhoto library.
Original images and your iPhoto library are not modified in any way.
You do not need to run this on OS X to read the iPhoto library, you only need a copy of the library. Tested on Linux (Ubuntu 14.04).
Apple's public AlbumData.xml API does not expose all of this information, so this script uses SQLite3 to access the iPhoto library data directly.
EXIF and other data which was in the original images is of course preserved but NOT copied to the XMP sidecar file.
require 'progressbar' # required for eye candy during conversion
require 'find' # required to find orphaned images
require 'fileutils' # required to move and link files around
require 'sqlite3' # required to access iPhoto database
require 'time' # required to convert integer timestamps
require 'cfpropertylist' # required to read binary plist blobs in SQLite3 dbs, 'plist' gem can't do this
require 'erb' # template engine
require 'pp' # to pretty print PList extractions
For Ubuntu and Ruby 2.2, this means e.g.
sudo apt-get install ruby2.2 ruby2.2-dev libsqlite3-dev
sudo gem2.2 install sqlite3 progressbar CFPropertyList
ruby iphoto2xmp.rb "~/Pictures/My iPhoto library" "~/Pictures/Export Here"
Use a DEBUG
environment variable to print out debugging information. For example, DEBUG=1
will print out basic information about all found images. DEBUG=3
will print out all metadata found in all images including faces.
DEBUG=1 ruby iphoto2xmp.rb "~/Pictures/My iPhoto library" "~/Pictures/Export Here"
The original idea was taken from https://gist.github.com/lpar/2191225, but the script has been heavily modified to access more iPhoto metadata (not just AlbumData.xml), distinguish between original and modified photos, and not rely on exiftool. This also brings a huge speed improvement.
The script can currently export the following metadata:
- Image filenames (duh)
- All EXIF data within the original image (preserved inside the files)
- Captions / Titles
- Descriptions
- Keywords (iPhoto does not use hierarchical tags)
- Event names (used for the folder structure, not exported into XMP)
- GPS coordinates
- Edited and original images, edit operation (eg. "Crop", "WhiteBalance", ...)
- Face names and face coordinates
- Face names and face coordinates in rotated or cropped images (TODO: still buggy if the image had EXIF rotation flags set since then iPhoto saves weird position values)
- Hidden, Starred, Flagged, Editable, Original, isInTrash flags (as tags)
- iPhoto and iOS edit operations as additional *.plist sidecar files (so far, not all are decoded)
- Albums as tag collections (Library:RKFolder/RKAlbum, Library:RKAlbumVersion) into "TopLevelAlbums/" tag hierarchy
- iPhoto's Slideshows, Calendars, Cards, Books as tag collections (to identify which photos were used) into "TopLevelKeepsakes/" tag hierarchy
- Smart Album rules into a separate text file so they can be recreated in the target application (the structure is not decoded yet but can be looked at)
Some image properties cannot (properly) be converted into metadata suitable for XMP sidecar files. They must be
patched into the target application's database after the import process. This requires exceuting sqlite
scripts
after starting Digikam at least once and letting it update the image database.
iphoto2xmp writes several SQL scripts into the destination folder which can be executed against the digikam4.db
SQLite database after the import. Note that this should only be done with a backup, in case something goes wrong.
These might include (depending on what features were used in iPhoto):
- iPhoto <= 9.1 Event notes (as Album descriptions):
event_notes.sql
- Event minimum date and thumbnail (as Album date and thumbnail):
event_metadata.sql
- Group original & modified images (as groups & versions):
grouped_images.sql
Usage for each file (grouped_images.sql as an example):
sqlite3 ~/Pictures/digikam4.db < grouped_images.sql
If there is no output, everything went fine. If there is a lot of output, there is a problem with the SQL. Post the output as an issue here on Github.
The script should (at some point) also do the following. Note: This is your chance to fork and create a pull request ;-)
- Avoid saving duplicate versions of non-modified images (e.g. "RawDecodeOperation" is not a modification)
- Optionally rotate videos (that have the "Orientation" flag set) so that they display correctly in Digikam.
- Export iPhoto "hidden" status as group commands and include a dummy image as first image with a "hidden" icon (hidden photos are already tagged accordingly so you can do anything you want with them)
- use XMP DerivedFrom to automatically group "Original" and "Modified" photos from RKVersion.isOriginal und masterUuid
- GPS coordinate names (Country, City, etc.). These are in Properties.apdb::RKPlace, RKPlaceName
- Fix face coordinates for EXIF rotated images (see above)
- export an image's edit history at least as a descriptive text, perhaps as XMP (e.g. digikam:history tag)
- correctly identify iOS Edit operations (which create their own proprietary XMP sidecar file)
The script will additionally identify
- orphaned images in your iPhoto Library (ie. images which are referenced nowhere) and
- missing images (images which are in the database but have no associated file).
and optionally copy orphaned images to the export root directory
iPhoto does not use hierarchical tags, but some users might have created a pseudo-hierarchical structure in iPhoto using dots or slashes, naming tags like "Places/Ottawa" or "People/John Doe". The tags are converted verbatim, so it is up to your new DAM to make sense of these tags. Or fork, write a conversion, and create a pull request! ;-)
I plan to document the iPhoto Library structure in this repository when I have the time. Meanwhile, look at the source code comments.
The license of this script is GPL2 as of now. If this causes problems with your intended usage please contact me.
Contact me at jens-github@spamfreemail.de or via Github.