Arbitrarily synchronizable, encrypted todo list mobile web application that supports a multitude of storage backends.
The todo lists in this application can be synchronized with a storage provider of your choice (a very simple HTTP REST interface is enough) and also works offline without losing any data. Because of the synchronization features, todo lists can be easily shared among different users and conflicts will automatically be resolved.
If a storage backend supports CORS and has a http REST interface, it is most probably usable by synclist. These backends include remoteStorage, ownCloud, or a simple WebDAV server.
It's here: http://peter-x.github.io/synclist You can just use it, no installation required!
You can create items, mark them as done, move them between categories and also move them inside the list (just drag on the little arrow symbol in the menu).
Per default, everything will be stored locally in your browser, but you can add a remote storage to synchronize between different browsers (and smartphones). The easiest way to do this is to create an account for RemoteStorage at 5apps. In the application, add the storage by giving your account name and don't worry: Everything stored remotely will be encrypted if you supply a key.
If you don't want to enter the URL in your phone every time: There is a very comfortable Android package! The package does not yet automatically update itself, so please come back here regularly for updates.
Synclist on your Android Phone
Have fun and please give us some feedback if you have time!
The data is stored as a collection of versioned items, where each item has the following properties:
- string id
- string revid
- list previous revisions
- int creation (timestamp)
- int resolution (timestamp)
- int last modification (timestamp)
- string category
- string text
- float sort position
The id
is chosen randomly on creation, revid
contains the revision number
(incremented on each change) plus a hash value of the json-encoding of the
object (without the hash field) - possibly after encryption.
previous revisions
is a list of all previous revisions of this object (needed
for resolving conflicts).
Each item is stored json-encoded under the file name id + '-' + revid
.
id
and revid
are omitted from the json encoding to avoid inconsistencies.
An item is considered resolved iff its resolution timestamp is nonzero.
Pushing a change to the storage consists of simply adding a file to the storage. There will be no write conflicts unless we find a hash collision. The current version of the item is always the lexicographically largest filename that starts with the id of the item.
Conflicts are detected whenever there is a revid for an item that is not part of
the previous revisions field. In this case, the software tries to resolve the
conflict or asks the user to decide. Once the conflict is resolved, a new
revision is created that contains the previously unknown revision in its
previous revisions
field. Automatic conflict resolution should always be
deterministic, so that absolutely the same file will be generated.
Since no subdirectories are needed, a simple key-value storage is already sufficient as storage engine.
Synchronization (without conflict resolution) can already be achieved by simply copying all files, other changes will not be overwritten (assuming there are no hash collisions).
How can changes to the storage be detected?
Large changes to the text should probably rather create a new item. Otherwise, automatic merges can seem counter-intuitive: If Alice changes "buy chocolate" to "buy milk" and Bob marks the item as resolved at the same time, they most probably still need milk.
For simplicity, access keys for remote storages are (can be) all stored locally, nobody wants to enter passwords for a mobile application.
Items can possibly be encrypted, the encryption standard to be used is stored in the item.
A conflict is detected if there is a revision that is not contained in the
previous revisions
field of the most recent revision of an item.
Since there can be multiple such revisions, the newest is merged first to reduce
the number of merges needed.
From the previous revisions
field of both items, the most recent shared
revision is determined to form the base of the merging. If there is no common
base, split the item (explained below).
Fields that can conflict and how to resolve them:
previous revisions
: not a real conflict, compute the union (and sort)creation
: should actually be read-only, thus use the baseresolution
: if both versions are zero or both are non-zero, take the minimum, otherwise take the value that differs from the base with respect to being zero or non-zero.last modification
: take the maximumcategory
: take the one that differs from the base, if both differ, join them sorted and with a comma as separator.text
: take the one that differs from the base, if it differs too much or if both differ, split the item.sort position
: take the one that differs from the base, and the minimum if both differ.
If two revisions of an item are too different (especially when the text changes drastically), splitting the item into two could be reasonable (this needs to be investigated). For this, an independent copy of the item is created, where the id is the hash value of the revision that is to be cloned (the one with the smaller revision is used). This method could result in duplicates and its feasibility has to be determined.
Note that there is no way to explicitly delete items, since conflicts in deleted items cannot be resolved. To keep the storage small, though, items are automatically removed that are resolved and have not been modified for a specific amount of time.