sms-db is a tool to build an SQLite database out of collections of SMS and MMS messages in various formats. The database can then be queried using standard SQLite queries, and the messages in the database can be exported in various formats.
This README is for sms-db version 0.3, which uses sms-db format 2.
sms-db has several design goals:
-
Providing message storage in a simple, standard, easily understandable format.
-
Enabling the combination of messages from multiple sources into a single message store.
-
Enabling the querying of message collections (via the SQLite framework).
-
Enabling the output of messages in various formats, potentially allowing their import into other systems, including Android's internal message storage.
All versions of sms-db are available here. sms-db depends on the following Perl modules (Debian packages are in parentheses):
- XML::LibXML (libxml-libxml-perl)
- DBD::SQLite (libdbd-sqlite3-perl)
- Try::Tiny (libtry-tiny-perl)
- Encode (libencode-perl)
- Data::Dump (libdata-dump-perl)
No special installation of sms-db itself is required.
sms-db.pl [-d database-file.db] [-t message_type] -f format (-i input_file | -o output_file)
If -d
is not given, the default is sms-db.db
in the current directory. If the database or its tables do not exist, they will be created.
-t
controls which message types will be imported or exported. It should be either sms
, mms
, or all
(the default).
To import from multiple files with a single command, you can do something like:
find path/to/files -name "sms*xml" -exec sms-db.pl -f xml -i '{}' -d ./sms.db \;
(This does not add any processing efficiency over invoking sms-db multiple times - it just saves typing. A future version of the program may allow the import of multiple files with a single invocation of sms-db and a single database connection.)
For examples of querying the database, see querying.
IMPORTANT: to avoid data loss, all message sources should be retained after import by sms-db, both because of the possibility of corruption of the database due to bugs, and due to the fact that the database format is not yet entirely stable and may change with future versions of sms-db. In both of these cases it may be necessary to rebuild the database from the original message sources.
The following input formats are currently (at least partially) supported:
Bugle SQLite database files (bugle_db
) used internally by some versions of Android. These can be obtained in various ways, such as via the Android backup apps oandbackup / OAndBackupX.
Both SMS and MMS metadata and text parts are imported, but MMS non-text parts (and their filenames) are not currently imported. I can't find them in the Bugle database itself, and I don't know how to use the content://mms/part/nnnn
URIs in the uri
field of the Bugle parts
table to get the data. The information I found online explains how to use the URI to get the data programatically from a running Android system, but I could not find an explanation of how to use it with a standalone Bugle database.
XML files produced by the Android app SMS Backup & Restore. These XML files contain both SMSs and MMSs, and both are imported.
The XML format is documented here. Note that many of the individual fields, particularly for MMS messages, are undocumented. Additionally, Synctech links to what it describes as "The XSD schema", but the schema is apparently either wrong or outdated, since it does not allow for the presence of the addrs
element present in both its webpage describing the XML fields as well as in actual XML backups produced by the app!
sms-db does not currently distinguish between the various recipient address types (To
, CC
, and BCC
). More generally, sbs-db ignores many of the fields in the XML, particularly with respect to MMS messages.
Encrypted backups produced by Signal for Android, decrypted and unpacked by either Signal backup decode or Signal for Android decryption. (sms-db as currently written is unlikely to work with the output of other tools that decode Signal backups, such as signal back or signalbackup-tools, but it would likely not be difficult to add support for them.)
When using the Signal format, set -i
to the root directory of the decrypted backup (e.g. -i signal-yyyy-mm-dd-nn-nn-nn
or -i my-signal-backup
); this directory should contain the file signal_backup.db
or database.sqlite
(depending on which decryption tool is used), which contains the SMS and MMS metadata and text parts, as well as the directory attachment
or attachments
, which contains the MMS attachments stored as individual files. All this is imported.
Signal has updated its database format many times, and will likely continue to do so in the future. This can break sms-db's Signal import capability. sms-db will attempt to maintain compatibility with the latest version of the Signal database, which may result in incompatibility between more recent versions of sms-db and backups produced by earlier versions of Signal. In such cases, the recommended solution is to leverage Signal's own database migration code to upgrade the database to the current format, via the following procedure (which ought to work, although it has not been tested):
- Install the latest version of Signal, either on actual hardware, or on an Android virtual device run on the Android Emulator.
- In Signal, restore from the old backup file.
- In Signal, create a new backup.
The new backup will hopefully be compatible with the latest version of sms-db. (If it isn't, open an issue to report the problem.)
My Signal databases contain only ordinary SMS and MMS messages, and not Signal's end-to-end encrypted messages, so I do not know whether sms-db will properly import such messages. (See here.)
I have not fully understood the meaning of all the fields in the various formats, and even with regard to those that I do understand, I have not currently chosen to import all of them (e.g., charsets and part sequence numbers). Future versions of sms-db may expand the scope of data and metadata imported (e.g., attachment filenames).
Currently, the only supported output format is Synctech's XML format. This should work well for SMS messages, but MMS support should be considered experimental at best. Note that I have not yet done any testing on whether and to what extent sms-db's XML output is correctly handled by SMS Backup and Restore.
Synctech does not document many of the MMS fields at all, and sms-db currently does not preserve most of them (and certainly does not generate the information from MMS messages imported from other formats). In order to generate "valid" XML somewhat resembling the XML produced by Synctech's app, sms-db sets many of the fields to null
and other arbitrary, and sometimes even known incorrect, values. The resulting XML mostly validates with Synctech's XSD schema (except for the fact that we include the addrs
element - see above), but how well Synctech's import functionality will work, given all the missing and bogus information, is untested.
sms-db does not produce CSV output, but such output can be easily generated by the SQLite executable, e.g.:
sqlite3 -csv sms-db.db "select sender_name,recipient_name,timestamp,data from messages INNER JOIN parts ON messages._id = parts.message_id where content_type LIKE 'text%' ORDER BY timestamp;"
(See querying for further examples of queries against the database.)
sms-db stores messages in a simple SQLite database, in two tables: messages
and parts
. Message metadata is stored in the former, and message data in the latter. (The database structure is a much simplified version of the Bugle one used by Android, with many tables and columns omitted. No notion of 'conversation' is preserved, and sender and recipient metadata is stored in the messages
table.)
sms-db tries to avoid storing duplicate messages in its database. This is currently implemented by storing SHA-256 hashes of the stored message metadata alongside that metadata, and checking the hashes of potential additions to the message store for uniqueness. (It might be feasible to use a less computationally expensive hash algorithm, or even a non-cryptographic hash, but the performance of the current implementation seems quite adequate, and we don't seem to be paying too high a price for the collision resistance of SHA-256.) This is less likely to work with copies of messages stored in different formats, in which case sms-db may wind up storing duplicate copies of what is actually the same message. (This has been observed to occur with outgoing SMS messages that appear in both Bugle and Signal databases, since with outgoing Bugle messages, sms-db records the actual sender (i.e., owner) name and number, while with outgoing Signal messages, the sender name and number are recorded as '', and the messages are therefore not considered identical. It is a design goal to never err in the opposite direction of failing to store messages which are actually not duplicates.)
Currently, all timestamps are in Epoch time format, in milliseconds. To convert to or from a human readable date, use date -d'@nnnnnnnnnn.nnn'
/ date -d'human_readable_date' +%s
(the decimal point must be inserted in the input, and '000' must be added to the output, since the date
command date string format is in seconds rather than milliseconds), or use this online converter.
Many of the columns in the sms-db database schema should be self-explanatory; following is some explanation of some of the less obvious ones:
-
msg_box
values have the meaning they have in the Synctech XML format: 1 = Received, 2 = Sent, 3 = Draft, 4 = Outbox. (Synctech usestype
for SMSs andmsg_box
for MMS; we usemsg_box
for both.) -
message_type
records the type of message: 0 = SMS, 1 = MMS. -
source_format
records the format of the message collection of a message's origin: 0 = XML, 1 = Bugle, 2 = Signal.
SMS-Tools, a "Multipurpose import / export / merge tool for your text message history", apparently does what sms-db does and more. It supports many more file formats, and can output to most of the formats it supports. On the other hand, it does not advertise Signal support. I have not used SMS-Tools or looked at its code. The original, last updated in 2014, is here; a fork, with a single commit in 2017, adding support for Bugle, is here.
Probably, particularly inaccurate interpretation of message metadata in the various supported input formats. For bug reports, feature requests, or general feedback, use the sms-db issue tracker.