This is my cobbled together exporter of Apple Notes into Markdown. It exports every note into its own Markdown file. Contrary to other exporters I’ve found, it supports checklists (including nested checklists), attachments, and embedded links. Tested on macOS Big Sur.
Based on the work of Jon Baumann and Ciofeca Forensics, original README below:
This program is a parser for the current version of Apple Notes data syncable with iCloud as seen on Apple handsets in iOS 9 and later.
This program was made as an update to the previous Perl script which did not well handle the protobufs used by Apple in Apple Notes.
That script and this program are needed because data that was stored in plaintext in the versions of Apple's Notes prior to iOS 9 in its notes.sqlite
database is now gzipped before storage in the iCloud Notes database NoteStore.sqlite
and the amount of embedded objects inside of Notes is far higher.
While the data is not necessarily encrypted, although some is using the password feature, it is not as searchable to the examiner, given its compressed nature.
This program intends to make the plaintext stored in the note and its embedded attachments far more usable.
This program was implemented in Ruby. The classes underlying this represent all the necessary features to write other programs to interface with an Apple Notes backup, including exporting data to another format, or writing better search functions. In addition, this program and its classes attempts to abstract away the work needed to understand the type of backup and how it stores files. While examiners must understand those backups, this will provide its own internal interfaces for identifying where media files are kept. For example, if the backup is from iTunes, this program will use the Manifest.db to identify the hashed name of the included file, and copy out/rename the image to the appropriate name, without the examiner having to do that manually.
This program will:
- Parse legacy (pre-iOS9) Notes files (but those are already plaintext, so not much to be gained)
- Parse iOS 9-14 Cloud Notes files
- ... decrypting notes if the password is known
- ... generating CSV roll-ups of each account, folder, note, and embedded object within them
- ... rebuilding the notes as an HTML file to browse and see as they would be displayed on the phone
- ... amending the NoteStore.sqlite database to include plaintext and decompressed objects to interact with in other tools
- ... from iTunes logical backups, physical backups, single files, and directly from Mac versions
- ... displaying tables as actual tables and ripping the embedded images from the backup and putting them into a folder with the other output files for review
- ... identifying the CloudKit participants involved in any shared items.
This program is run by Ruby on a command line, using either rake
or ruby
.
The easiest use, which is the same as the original Perl script, is to drop your exported NoteStore.sqlite
into the same directory as this program and then run rake
(which is the same as running ruby notes_cloud_ripper.rb --file NoteStore.sqlite
).
If you are more comfortable with the command line, you can point the program anywhere you would like on your computer and specify the type of backup you are looking at, such as: ruby notes_cloud_ripper.rb --itunes-dir ~/.iTunes/backup1/
or ruby notes_cloud_ripper.rb --file ~/.iTunes/backup1/4f/4f98687d8ab0d6d1a371110e6b7300f6e465bef2
.
The benefit of pointing at full backups is this program can pull files out of them as needed, such as drawings and pictures.
The options that are currently supported are:
2. -f | --file FILE
: Tells the program to look only at a specific file that is identified.
-i | --itunes-dir DIRECTORY
: Tells the program to look at an iTunes backup folder.-m | --mac DIRECTORY
: Tells the program to look at a folder from a Mac.-o | --output-dir DIRECTORY
: Changes the output folder from the default./output
to the specified one.-p | --physical DIRECTORY
: Tells the program to look at a physical backup folder.-w | --password-file FILE
: Tells the program which password list to use-h | --help
: Prints the usage information.
For backups created with iTunes MobileSync, that include a wide range of hashed files inside of folders named after the filename, this program expects to be given the root folder of that backup. With that, it will compute the path to the NoteStore.sqlite file. If it exists, that file will be copied to the output directory and the copy, not the original, will be opened.
For example, if you had an iTunes backup located in /home/user/phone_rips/iphone/[deviceid]/
(Which means the Manifest.db is located at /home/whatever/phone_rips/iphone/[deviceid]/Manifest.db
) you would run: ruby notes_cloud_ripper.rb -i /home/user/phone_rips/iphone/[deviceid]/
For backups created with a full file system (or at least the /private
directory) from your tool of choice. This program expects to be given the root folder of that backup. With that, it will compute the path to the NoteStore.sqlite file. If it exists, that file will be copied to the output directory and the copy, not the original, will be opened.
For example, if you had a physical backup located in /home/user/phone_rips/iphone/physical/
(Which means the phone's /private
directory is located at /home/whatever/phone_rips/iphone/physical/private/
) you would run: ruby notes_cloud_ripper.rb -p /home/user/phone_rips/iphone/physical
For single file "backups", this program expects to be given the path of the NoteStore.sqlite file directly, although filename does not matter. If it exists, that file will be copied to the output directory and the copy, not the original, will be opened.
For example, if you had a NoteStore.sqlite file located in /home/user/phone_rips/iphone/files/NoteStore.sqlite
you would run: ruby notes_cloud_ripper.rb -f /home/user/phone_rips/iphone/files/NoteStore.sqlite
For backups created from the Notes app as installed on a Mac. This program expects to be given the group.com.apple.notes folder of that Mac. With that, it will compute the path to the NoteStore.sqlite file. If it exists, that file will be copied to the output directory and the copy, not the original, will be opened.
For example, if you were running this on data from a Mac used by 'Logitech' and had the full file system available, you would run: ruby notes_cloud_ripper.rb -m /Users/Logitech/Library/Group Containers/group.com.apple.notes/
For backups that may have encrypted notes within them, this option tells the program where to find its password list. This list should have one password per row and any passwords that correctly decrypt an encrypted note will be tried before the rest for future encrypted notes.
For example, if you were running this on data from a Mac used by 'Logitech,' had the full file system available, and wanted to use a file called "passwords.txt"you would run: ruby notes_cloud_ripper.rb -m /Users/Logitech/Library/Group Containers/group.com.apple.notes/
-w passwords.txt
Once the NoteStore file is opened, the program will create new AppleNotesAccount, AppleNotesFolder, and AppleNote objects based on the contents of that file.
For each note, it takes the gzipped blob in the ZDATA field, gunzips it, and parses the protobuf that is inside.
It will then add the plaintext from the protobuf of each note back into the NoteStore.sqlite file's ZICNOTEDATA
table as a new column, ZPLAINTEXTDATA
and create AppleNotesEmbeddedObject objects for each of the embedded objects it identifies.
All of the output from this program will go into the output folder (it will be created if it doesn't exist), which defaults to [location of this program]/output
.
Within that folder, sub-folders will be created based on the current date and time, to the second.
For example, if this program was run on December 17, 2019 at 10:24:28 local, the output for that run would be in [location of this program]/output/2019_12_17-10_24_28/
.
If the type of backup used has the original files referenced in attached media, this program will copy the file into the output directory, under [location of this program]/output/[date of run]/files
, and beneath that following the file path on disk, relative to the Notes application folder.
This program will produce four CSV files summarizing the information stored in [location of this program]/output/[date of run]/csv
: note_store_accounts.csv
, note_store_embedded_objects.csv
, note_store_folders.csv
, and note_store_notes.sqlite
.
Finally, it will produce an HTML dump of the notes to reflect the text and table formatting which may be meaningful in [location of this program]/output/[date of run]/html
.
Because Apple devices often have more than one version of Notes, it is important to note (pun intended) that all of the output is suffixed by a number, starting at 1, to identify which of the backups it corresponds to. In all cases where more than one is found, care is taken to produce output that assigns the suffix of 1 for the modern version, and the suffix of 2 for the legacy version.
This program has been tested with the following versions of Ruby:
Ruby Version | OS | Status |
---|---|---|
2.3.0 | Linux | ✔️ |
2.3.1 | Linux | ✔️ |
2.4.3 | Linux | ✔️ |
2.5.1 | Linux | ✔️ |
2.6.5 | Linux | ✔️ |
2.7.1 | Linux | ✔️ |
2.4.3 | macOS 10.13 | ✔️ |
2.5.1 | macOS 10.13 | ✔️ |
2.6.5 | macOS 10.13 | ✔️ |
2.7.1 | macOS 10.13 | ✔️ |
2.6.5 | Windows 8 | ✔️ |
2.6.5 | Windows 10 Enterprise | ✔️ |
This program requires the following Ruby gems which can be installed by running bundle install
or gem install [gemname]
:
- fileutils
- google-protobuf
- Note: If you use Ruby 2.7, you must have version 3.12 of this gem, or newer
- Note: If you use Ruby 2.3 or 2.4, you must not have any version later than 3.11.4
- sqlite3
- zlib
- openssl
- aes_key_wrap
- keyed_archive
Below are instructions, generally preferring the command line, for each of Linux, Mac, and Windows. The user can choose to use Git if they want to be able to keep up with changes, or just download the tool once, you do not need to do both. On each OS, you will want to:
- Install Ruby (at least version 2.3.0, preferably 2.5 or later), its development headers, and bundler if not already installed.
- Install development headers for SQLite3 if not already installed.
- Get this code
- Clone this repository with Git or
- Download the Zip file and unzip it
- Enter the repository's directory.
- Use bundler to install the required gems.
- Run the program (see Usage section)!
sudo apt-get install build-essential libsqlite3-dev zlib1g-dev git ruby-full ruby-bundler
git clone https://github.com/threeplanetssoftware/apple_cloud_notes_parser.git
cd apple_cloud_notes_parser
bundle install
sudo apt-get install build-essential libsqlite3-dev zlib1g-dev git ruby-full ruby-bundler
curl https://codeload.github.com/threeplanetssoftware/apple_cloud_notes_parser/zip/master -o apple_cloud_notes_parser.zip
unzip apple_cloud_notes_parser.zip
cd apple_cloud_notes_parser-master
bundle install
sudo yum groupinstall "Development Tools"
sudo yum install sqlite sqlite-devel zlib zlib-devel openssl openssl-devel ruby ruby-devel rubygem-bundler
git clone https://github.com/threeplanetssoftware/apple_cloud_notes_parser.git
cd apple_cloud_notes_parser
bundle install
sudo gem pristine sqlite3 zlib openssl aes_key_wrap keyed_archive
sudo yum groupinstall "Development Tools"
sudo yum install sqlite sqlite-devel zlib zlib-devel openssl openssl-devel ruby ruby-devel rubygem-bundler
curl https://codeload.github.com/threeplanetssoftware/apple_cloud_notes_parser/zip/master -o apple_cloud_notes_parser.zip
unzip apple_cloud_notes_parser.zip
cd apple_cloud_notes_parser-master
bundle install
sudo gem pristine sqlite3 zlib openssl aes_key_wrap keyed_archive
git clone https://github.com/threeplanetssoftware/apple_cloud_notes_parser.git
cd apple_cloud_notes_parser
bundle install
curl https://codeload.github.com/threeplanetssoftware/apple_cloud_notes_parser/zip/master -o apple_cloud_notes_parser.zip
unzip apple_cloud_notes_parser.zip
cd apple_cloud_notes_parser-master
bundle install
- Download the 2.6.5 64-bit RubyInstaller with DevKit (Google Protobuf isn't yet working on 2.7)
- Run RubyInstaller using default settings.
- Download the latest SQLite amalgamation souce code and 64-bit SQLite Precompiled DLL
- Install SQLite
- Create a folder C:\sqlite
- Unzip the source code into C:\sqlite (you should now have C:\sqlite\sqlite3.c and C:\sqlite\sqlite.h, among others)
- Unzip the DLL into C:\sqlite (you should now have C:\sqlite\sqlite3.dll, among others)
- Download this Apple Cloud Notes Parser as a zip archive
- Unzip the Zip archive
- Launch a powershell window and navigate to where you unzipped the archive
- Execute the following commands (these set the PATH so SQLite files can be found install SQLite's Gem specifically pointing to them, and then installs the rest of the gems):
$env:Path += ";C:\sqlite"
[Environment]::SetEnvironmentVariable("Path", $env:Path + ";C:\sqlite", "User")
gem install sqlite3 --platform=ruby -- --with-sqlite-3-dir=C:/sqlite --with-sqlite-3-include=C:/sqlite
bundle install
For reference, the structure of this program is as follows:
apple_cloud_notes_parser
|
|-lib
| |
| |-notestore_pb.rb: Protobuf representation generated with protoc
| |-Apple\*.rb: Ruby classes dealing with various aspects of Notes
|
|-output (created after run)
| |
| |-[folders for each date/time run]
| |
| |-csv: This folder holds the CSV output
| |-debug_log.txt: A more verbose log to assist with debugging
| |-files: This folder holds files copied out of the backup, such as pictures
| |-html: This folder holds the generated HTML copy of the Notestore
| |-Manifest.db: If run on an iTunes backup, this is a copy of the Manifest.db
| |-NoteStore.sqlite: If run on a modern version, this copy of the target file will include plaintext versions of the Notes
| |-notes.sqlite: If run on a legacy version, this copy is just a copy for ease of use
|
|-.gitignore
|-.travis.yml
|-Gemfile
|-LICENSE
|-README.md
|-Rakefile
|-notes_cloud_ripper.rb: The main program itself
Example: /var/lib/gems/2.3.0/gems/sqlite3-1.4.1/lib/sqlite3/database.rb:147:in 'initialize': no such column: ZICCLOUDSYNCINGOBJECT.ZSERVERRECORDDATA (SQLite3::SQLException)
Apple changed the format of its Notes database in different versions of iOS. While the supported versions should be supported, interesting cases may come up. Please open an issue and include the stack trace and the following information:
- iOS version (including any versions it may have upgraded from)
- The results of
SELECT name,sql FROM sqlite_master WHERE type="table"
when the database is open in sqlitebrowser (or your editor of choice). This can be in any columned format (Excel, CSV, SQL, etc) - If possible, the database file directly (I can receive it through other means if it needs to stay confidential). If this is possible, the above results are not needed.
Programming languages are like human languages, there are many and which you choose (for those with multiple) can largely be a personal preference, assuming mutual intelligibility. I chose Ruby as the previous Perl code was a nice little script, but I wanted a bit more substance behind with with solid object oriented programming principals. For those new to Ruby, hopefully these classes will give you an idea of what Ruby has to offer and spark an interest in trying a new language.
- MildSunrise's protobuf-inspector drove most of my analysis into the Notes protobufs.
- Previous work by dunhamsteve proved invaluable to finally understanding the embedded table aspects.