Demonstration release of the principles underpinning krsd. master
authorRick van Rein <rick@openfortress.nl>
Fri, 31 Jan 2014 09:25:22 +0000 (09:25 +0000)
committerRick van Rein <rick@openfortress.nl>
Fri, 31 Jan 2014 09:25:22 +0000 (09:25 +0000)
This is based on rs-serve, present on https://github.com/remotestorage/rs-serve
It includes a demonstration from https://github.com/tastejs/todomvc

Changes relative to rs-serve are manyfold, thus leading to this fork:
 * It is made for virtual hosting, rather than local user accounts
 * It uses Kerberos authentication instead of explicit, JS-programmed OAuth
 * It uses a .k5remotestorage configfile on the filesystem for authorisation
 * It provides service for static web content

24 files changed:
CHANGES [new file with mode: 0644]
LIMITATIONS [new file with mode: 0644]
README
demo/README.md [new file with mode: 0644]
demo/todo/assets/base.css [new file with mode: 0644]
demo/todo/assets/base.js [new file with mode: 0644]
demo/todo/assets/bg.png [new file with mode: 0644]
demo/todo/index.html [new file with mode: 0644]
demo/todo/js/app.js [new file with mode: 0644]
demo/todo/js/remotestorage.js [new file with mode: 0644]
demo/todo/js/tasks.js [new file with mode: 0644]
demo/todo/license.md [new file with mode: 0644]
demo/todo/readme.md [new file with mode: 0644]
init-script.sh
src/config.c
src/config.h
src/handler/auth.c
src/handler/auth.h
src/handler/dispatch.c
src/handler/storage.c
src/handler/storage.h
src/handler/webfinger.c
src/process/main.c
src/rs-serve.h

diff --git a/CHANGES b/CHANGES
new file mode 100644 (file)
index 0000000..6972393
--- /dev/null
+++ b/CHANGES
@@ -0,0 +1,42 @@
+# Changes of krsd relative to rs-serve
+
+The reason to create krsd from rs-serve was to demonstrate a different mode of using remoteStorage.  Instead of making all authentication and authorisation code explicitly a part of JavaScript code, this deamon presents one way of having this arranged by the browser, outside of the potentially dangerous grip of web code.
+
+Two mechanisms are generally available in browsers to do such things:
+
+ * X.509 client-side certificates
+ * Kerberos
+
+Both can behave as a single sign-on system, thus leading to a high level of end user satisfaction.  The version implemented by krsd is based on Kerberos.
+
+
+## Virtual Hosting
+
+The design of rs-serve is based on local user accounts.  This makes it highly suitable for a NAS or local desktop or server system, which is ultimately what remoteStorage is after.  But to get less technical users going I would also love to see such a component provided by domain hosting providers.  So I’ve built in virtual hosting facilities.
+
+Currently, I parse /domain/user/ instead of /user/ from the URI, and put that into the disk_path, but I’m not sure this is the best scheme ever.  It might take the hostname into account instead (hardly a briliiant idea…) or perhaps the SNI-offered hostname.  This is likely to change.
+
+
+## User Accounts
+
+I removed all code relating to local user accounts, simply because that would not match with virtual hosting.  This also means that all data is stored with the uid of krsd, which is a pitty.  I do see a way out of this with an OpenAFS backend, but that is a future scenario.
+
+
+## Kerberos
+
+This is the point I am trying to prove — remoteStorage can use HTTP-level authentication, such as X.509 client certificates or, as I’ve implemented into krsd, Kerberos tickets passed over SPNEGO (the WWW-Authenticate method is called Negotiate and exchanges GSSAPI packets) and so there is no need for authentication / authorisation code in JavaScript — which is an insecure place to have it anyway.  I modified remoteStorage accordingly, and will offer patches to that end.
+
+The authorisation has also moved.  It does not take place before parsing the headers (which may leak information on locally available UIDs if you’re not careful) but it done afterwards.  When paths are constructed, it digs up a file .k5remotestorage which should list the Kerberos Principal Name on a line on its own.
+
+In future versions, I expect this to move into LDAP, to permit centrally managed systems to facilitate remoteStorage.  Again, this is with virtual hosting of remoteStorage in mind.
+
+
+## Filesystem
+
+The future idea we’re having is that we could have a (remote) filesystem hosted as OpenAFS, store remoteStorage on it (using Kerberos’ Constrained Delegation to access it as the authenticated user) and permit the user to mount that filesystem on their working places as well.  We imagine other hosted / mounted combinations, such as MSRP-based file transfers, either within a phone call or pushed to a user’s inbox for documents.  This is why I decided to drop the user accounts from rs-serve.
+
+
+## Static web content
+
+I included a facility to present static web content, by adding a default callback to libevhtp.  I used it for the demo setup without zooming in on CORS yet, but it should be generally useful to be able to present an index.html or so.
+
diff --git a/LIMITATIONS b/LIMITATIONS
new file mode 100644 (file)
index 0000000..6287983
--- /dev/null
@@ -0,0 +1,54 @@
+KNOWN LIMITATIONS OF KRSD
+=========================
+
+*Not all things that you might like work as you think...*
+
+
+Limited use of CORS
+-------------------
+
+The CORS approach may not work.  Think of it -- you're a browser and a
+(potentially rogue) program attempts to approach a remote site and serve
+it with your Kerberos ticket.  This may not be what the user is after, as
+it may give rise to a MITM attack.
+
+CORS in general has a fairly rocky basis when it comes to security.  This
+is especially true for the implicit security operations that are provided
+by a browser: Negoatiate for GSSAPI/Kerberos, and X.509 client certificates
+over a TLS connection.
+
+The point is, these implicit security operations are more promising in a
+secure web model than CORS can ever be.  The separation of application
+program dynamicity from solid HTTP handling is very useful in keeping
+rogue programs away from (the use of) credentials.  It could even be
+implemented along the way, in an "authenticating HTTP proxy".
+
+
+Limited utility for POSIX accounts
+----------------------------------
+
+Where rs-serve was designed for local user accounts that were made available
+for web storage, the krsd is instead designed to service infrastructure,
+in a manner that can be hosted en masse by hosting providers.  This should
+make the work accessible to a larger audience.
+
+The use of POSIX accounts works well for a local setup, but less so for the
+infrastructural variety.  For that reason, files are stored under the user
+of the krsd and the user privileges are not modified.
+
+
+Filesystem backing on OpenAFS
+-----------------------------
+
+There is a potential filesystem approach to this, though.  It is feasible
+to use the online filesystem AFS as a backend, and employ Kerberos to
+authorize any changes to it.  If this is done, S2U4Proxy can be used to
+pass on the SPNEGO credential to access the AFS mount.  This would work if
+the KDC has Constrained Delegation configured to permit the krsd HTTP service
+to access the filesystem, or when a krsd credential is permitted in the ACL
+of the AFS partition.
+
+Note how this approach makes the stored data available to the end user, in
+much the same way as with a local storage directory under a POSIX account.
+
+
diff --git a/README b/README
index 70b2c6e..7202df9 100644 (file)
--- a/README
+++ b/README
@@ -72,14 +72,14 @@ be able to run on another system, please open an issue to ask for help)
 2.1) remotestorage
 ------------------
 
-The currently implemented protocol version is "draft-dejong-remotestorage-01".
+The currently implemented protocol version is "draft-dejong-remotestorage-01".  It has been modified to permit implied, that is HTTP-level, authentication and authorisation.  Demo code is available in a separate directory.
 
 Currently the following features are supported:
-* CORS support for all verbs
+* CORS support for all verbs (TODO: May not work at present)
 * GET, PUT, DELETE requests on files and folders
 * Opaque version strings (in directory listings and "ETag" header)
 * Conditional GET, PUT and DELETE requests ("If-Match", "If-None-Match" headers)
-* Protection of all non-public paths via Bearer token authorization.
+* Protection of all non-public paths via authentication by the browser at the HTTP level, and authorisation based on a .k5remotestorage file in the destination file system
 * Special handling of public paths (i.e. those starting with /public/), such that
   requests on non-directory paths succeed without authorization.
 * HEAD requests on files and folders with "Content-Length" header
@@ -96,6 +96,12 @@ overridden with the --hostname option.
 Virtual hosting (== hosting storage for multiple domains from a single
 instance) is currently not supported.
 
+The pathname returned ermits for the krsd to parse out two components of
+the pathname to which it stores: /storage/domain.tld/user/ should be at
+the beginning of the URI if it is to be accepted as a remoteStorage URI
+on krsd.  Anything after this is taken as a path into the storage
+structures used.
+
 2.3) authorization
 ------------------
 
@@ -182,9 +188,9 @@ to these steps:
        http/chitchat.arpa2.net will append http/chitchat.arpa2.net/ to
        the path found so far.
 
-  3. Thirdly, a fixed directory name such as webstorage/ is appended to
+  3. Thirdly, a fixed directory name such as remotestorage/ is appended to
      the path name.  This is done to separate remote storage from local
-     storage and from other remote storeage components, and to make all the
+     storage and from other remote storage components, and to make all the
      other directories unavailable.  This is of no use to local storage, but
      it is immensely useful when dealing with storage on an OpenAFS share
      that is also employed for other purposes.
@@ -195,21 +201,28 @@ be helpful at this point.
 Example.  The user john@ARPA2.NET could be found in DNS zone arpa2.net, and
 dependent on local settings his files could end up in mounted locations like:
 
-  /afs/arpa2.net/john/webstorage/...
-  /var/lib/krs/arpa2.net/john/webstorage/...
+  /afs/arpa2.net/john/remotestorage/...
+  /var/lib/krs/arpa2.net/john/remotestorage/...
 
 Example.  After changing user-ID to john/admin under the same realm, the
 files accessible to John are found in places like:
 
-  /afs/arpa2.net/john/admin/webstorage/...
-  /var/lib/krs/arpa2.net/john/admin/webstorage/...
+  /afs/arpa2.net/john/admin/remotestorage/...
+  /var/lib/krs/arpa2.net/john/admin/remotestorage/...
 
 Example.  When a server is not acting on behalf of a user (through S4U with
 Constrained Delegation) but on its own title, it depends on its principal
 name.  For example, xmpp/xmpp.arpa2.net@ARPA2.NET could be found on:
 
-  /afs/arpa2.net/xmpp/xmpp.arpa2.net/webstorage/...
-  /var/lib/krs/arpa2.net/xmpp/xmpp.arpa2.net/webstorage/...
+  /afs/arpa2.net/xmpp/xmpp.arpa2.net/remotestorage/...
+  /var/lib/krs/arpa2.net/xmpp/xmpp.arpa2.net/remotestorage/...
+
+The filesystem path is configured in the webfinger profile, and may or may
+not relate to the Kerberos Principal Name used to access the resource.
+For full flexibility, the remotestorage directory should hold a file named
+.k5remotestorage which must hold the Kerberos Principal Name on a line of
+its own, if it is to have read/write access to anything underneath this
+directory.
 
 The reliance on filesystem paths implies a few noteworthy restrictions:
 
@@ -220,7 +233,7 @@ The reliance on filesystem paths implies a few noteworthy restrictions:
   remotestorage (as far as I know).
 
 * MIME types may not be exact for files that were added "out-of-band", that is
-  not added via the remotestorage protocol, but by copying to the ~/storage/
+  not added via the remotestorage protocol, but by copying to the remotestorage/
   directory by other means. krsd stores MIME type and character encoding
   under the "user.mime_type" and "user.charset" extended attributes, given these
   are supported by the underlying filesystem. When these attributes aren't set,
diff --git a/demo/README.md b/demo/README.md
new file mode 100644 (file)
index 0000000..55b6c7f
--- /dev/null
@@ -0,0 +1,32 @@
+# Demonstration of krsd
+
+This demonstration uses TODOMVC, a common example used to demonstratie
+remoteStorage and Unhosted principles.  It is available under a MIT
+license on https://github.com/tastejs/todomvc and included here in a
+modified form.
+
+The current demonstration is not suited for CORS yet, so you will have
+to serve these files as part of your website's static content.  Do not
+forget to setup the filesystem extended attribute user.mime_type on
+each of the files so the krsd knows how to deliver them.
+
+To use this, you can run krsd as follows:
+
+krsd -p 443 --debug -n krsd.domain.tld --ssl --key-path=/etc/ssl/private/krsd.domain.tld-2014.pem --cert-path=/etc/ssl/certs/krsd.domain.tld-2014.pem --ca-path=/etc/ssl/certs/cacert.org.pem
+
+You should create a /home/domain.tld/user/remotestorage path for your user data,
+and setup a .k5remotestorage file with the Kerberos Principal Name that gets
+R/W access to the data underneath it.
+
+And of course, you'll need to setup Kerberos, and tell your browser to
+use it on this site.  This may involve whitelisting the domain being
+accessed, in a browser-specific manner.
+
+This work has been shown to work with Safari, FireFox and Chrome, all run
+under Mac OS X, using the builtin Kerberos support of Mac OS X.
+
+The work done to modify remoteStorage in this demo will be ported back to
+the remoteStorage main line and offered.  But it will first be integrated
+into GitHub location https://github.com/arpa2/remotestorage.js in a branch
+named "http-implied-auth".
+
diff --git a/demo/todo/assets/base.css b/demo/todo/assets/base.css
new file mode 100644 (file)
index 0000000..8d1db3a
--- /dev/null
@@ -0,0 +1,414 @@
+html,
+body {
+       margin: 0;
+       padding: 0;
+}
+
+button {
+       margin: 0;
+       padding: 0;
+       border: 0;
+       background: none;
+       font-size: 100%;
+       vertical-align: baseline;
+       font-family: inherit;
+       color: inherit;
+       -webkit-appearance: none;
+       /*-moz-appearance: none;*/
+       -ms-appearance: none;
+       -o-appearance: none;
+       appearance: none;
+}
+
+body {
+       font: 14px 'Helvetica Neue', Helvetica, Arial, sans-serif;
+       line-height: 1.4em;
+       background: #eaeaea url('bg.png');
+       color: #4d4d4d;
+       width: 550px;
+       margin: 0 auto;
+       -webkit-font-smoothing: antialiased;
+       -moz-font-smoothing: antialiased;
+       -ms-font-smoothing: antialiased;
+       -o-font-smoothing: antialiased;
+       font-smoothing: antialiased;
+}
+
+#todoapp {
+       background: #fff;
+       background: rgba(255, 255, 255, 0.9);
+       margin: 130px 0 40px 0;
+       border: 1px solid #ccc;
+       position: relative;
+       border-top-left-radius: 2px;
+       border-top-right-radius: 2px;
+       box-shadow: 0 2px 6px 0 rgba(0, 0, 0, 0.2),
+                               0 25px 50px 0 rgba(0, 0, 0, 0.15);
+}
+
+#todoapp:before {
+       content: '';
+       border-left: 1px solid #f5d6d6;
+       border-right: 1px solid #f5d6d6;
+       width: 2px;
+       position: absolute;
+       top: 0;
+       left: 40px;
+       height: 100%;
+}
+
+#todoapp input::-webkit-input-placeholder {
+       font-style: italic;
+}
+
+#todoapp input:-moz-placeholder {
+       font-style: italic;
+       color: #a9a9a9;
+}
+
+#todoapp h1 {
+       position: absolute;
+       top: -120px;
+       width: 100%;
+       font-size: 70px;
+       font-weight: bold;
+       text-align: center;
+       color: #b3b3b3;
+       color: rgba(255, 255, 255, 0.3);
+       text-shadow: -1px -1px rgba(0, 0, 0, 0.2);
+       -webkit-text-rendering: optimizeLegibility;
+       -moz-text-rendering: optimizeLegibility;
+       -ms-text-rendering: optimizeLegibility;
+       -o-text-rendering: optimizeLegibility;
+       text-rendering: optimizeLegibility;
+}
+
+#header {
+       padding-top: 15px;
+       border-radius: inherit;
+}
+
+#header:before {
+       content: '';
+       position: absolute;
+       top: 0;
+       right: 0;
+       left: 0;
+       height: 15px;
+       z-index: 2;
+       border-bottom: 1px solid #6c615c;
+       background: #8d7d77;
+       background: -webkit-gradient(linear, left top, left bottom, from(rgba(132, 110, 100, 0.8)),to(rgba(101, 84, 76, 0.8)));
+       background: -webkit-linear-gradient(top, rgba(132, 110, 100, 0.8), rgba(101, 84, 76, 0.8));
+       background: -moz-linear-gradient(top, rgba(132, 110, 100, 0.8), rgba(101, 84, 76, 0.8));
+       background: -o-linear-gradient(top, rgba(132, 110, 100, 0.8), rgba(101, 84, 76, 0.8));
+       background: -ms-linear-gradient(top, rgba(132, 110, 100, 0.8), rgba(101, 84, 76, 0.8));
+       background: linear-gradient(top, rgba(132, 110, 100, 0.8), rgba(101, 84, 76, 0.8));
+       filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0,StartColorStr='#9d8b83', EndColorStr='#847670');
+       border-top-left-radius: 1px;
+       border-top-right-radius: 1px;
+}
+
+#new-todo,
+.edit {
+       position: relative;
+       margin: 0;
+       width: 100%;
+       font-size: 24px;
+       font-family: inherit;
+       line-height: 1.4em;
+       border: 0;
+       outline: none;
+       color: inherit;
+       padding: 6px;
+       border: 1px solid #999;
+       box-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2);
+       -webkit-box-sizing: border-box;
+       -moz-box-sizing: border-box;
+       -ms-box-sizing: border-box;
+       -o-box-sizing: border-box;
+       box-sizing: border-box;
+       -webkit-font-smoothing: antialiased;
+       -moz-font-smoothing: antialiased;
+       -ms-font-smoothing: antialiased;
+       -o-font-smoothing: antialiased;
+       font-smoothing: antialiased;
+}
+
+#new-todo {
+       padding: 16px 16px 16px 60px;
+       border: none;
+       background: rgba(0, 0, 0, 0.02);
+       z-index: 2;
+       box-shadow: none;
+}
+
+#main {
+       position: relative;
+       z-index: 2;
+       border-top: 1px dotted #adadad;
+}
+
+label[for='toggle-all'] {
+       display: none;
+}
+
+#toggle-all {
+       position: absolute;
+       top: -42px;
+       left: -4px;
+       width: 40px;
+       text-align: center;
+       border: none; /* Mobile Safari */
+}
+
+#toggle-all:before {
+       content: '»';
+       font-size: 28px;
+       color: #d9d9d9;
+       padding: 0 25px 7px;
+}
+
+#toggle-all:checked:before {
+       color: #737373;
+}
+
+#todo-list {
+       margin: 0;
+       padding: 0;
+       list-style: none;
+}
+
+#todo-list li {
+       position: relative;
+       font-size: 24px;
+       border-bottom: 1px dotted #ccc;
+}
+
+#todo-list li:last-child {
+       border-bottom: none;
+}
+
+#todo-list li.editing {
+       border-bottom: none;
+       padding: 0;
+}
+
+#todo-list li.editing .edit {
+       display: block;
+       width: 506px;
+       padding: 13px 17px 12px 17px;
+       margin: 0 0 0 43px;
+}
+
+#todo-list li.editing .view {
+       display: none;
+}
+
+#todo-list li .toggle {
+       text-align: center;
+       width: 40px;
+       /* auto, since non-WebKit browsers doesn't support input styling */
+       height: auto;
+       position: absolute;
+       top: 0;
+       bottom: 0;
+       margin: auto 0;
+       border: none; /* Mobile Safari */
+       -webkit-appearance: none;
+       /*-moz-appearance: none;*/
+       -ms-appearance: none;
+       -o-appearance: none;
+       appearance: none;
+}
+
+#todo-list li .toggle:after {
+       content: '✔';
+       line-height: 43px; /* 40 + a couple of pixels visual adjustment */
+       font-size: 20px;
+       color: #d9d9d9;
+       text-shadow: 0 -1px 0 #bfbfbf;
+}
+
+#todo-list li .toggle:checked:after {
+       color: #85ada7;
+       text-shadow: 0 1px 0 #669991;
+       bottom: 1px;
+       position: relative;
+}
+
+#todo-list li label {
+       word-break: break-word;
+       padding: 15px;
+       margin-left: 45px;
+       display: block;
+       line-height: 1.2;
+       -webkit-transition: color 0.4s;
+       -moz-transition: color 0.4s;
+       -ms-transition: color 0.4s;
+       -o-transition: color 0.4s;
+       transition: color 0.4s;
+}
+
+#todo-list li.completed label {
+       color: #a9a9a9;
+       text-decoration: line-through;
+}
+
+#todo-list li .destroy {
+       display: none;
+       position: absolute;
+       top: 0;
+       right: 10px;
+       bottom: 0;
+       width: 40px;
+       height: 40px;
+       margin: auto 0;
+       font-size: 22px;
+       color: #a88a8a;
+       -webkit-transition: all 0.2s;
+       -moz-transition: all 0.2s;
+       -ms-transition: all 0.2s;
+       -o-transition: all 0.2s;
+       transition: all 0.2s;
+}
+
+#todo-list li .destroy:hover {
+       text-shadow: 0 0 1px #000,
+                                0 0 10px rgba(199, 107, 107, 0.8);
+       -webkit-transform: scale(1.3);
+       -moz-transform: scale(1.3);
+       -ms-transform: scale(1.3);
+       -o-transform: scale(1.3);
+       transform: scale(1.3);
+}
+
+#todo-list li .destroy:after {
+       content: '✖';
+}
+
+#todo-list li:hover .destroy {
+       display: block;
+}
+
+#todo-list li .edit {
+       display: none;
+}
+
+#todo-list li.editing:last-child {
+       margin-bottom: -1px;
+}
+
+#footer {
+       color: #777;
+       padding: 0 15px;
+       position: absolute;
+       right: 0;
+       bottom: -31px;
+       left: 0;
+       height: 20px;
+       z-index: 1;
+       text-align: center;
+}
+
+#footer:before {
+       content: '';
+       position: absolute;
+       right: 0;
+       bottom: 31px;
+       left: 0;
+       height: 50px;
+       z-index: -1;
+       box-shadow: 0 1px 1px rgba(0, 0, 0, 0.3),
+                               0 6px 0 -3px rgba(255, 255, 255, 0.8),
+                               0 7px 1px -3px rgba(0, 0, 0, 0.3),
+                               0 43px 0 -6px rgba(255, 255, 255, 0.8),
+                               0 44px 2px -6px rgba(0, 0, 0, 0.2);
+}
+
+#todo-count {
+       float: left;
+       text-align: left;
+}
+
+#filters {
+       margin: 0;
+       padding: 0;
+       list-style: none;
+       position: absolute;
+       right: 0;
+       left: 0;
+}
+
+#filters li {
+       display: inline;
+}
+
+#filters li a {
+       color: #83756f;
+       margin: 2px;
+       text-decoration: none;
+}
+
+#filters li a.selected {
+       font-weight: bold;
+}
+
+#clear-completed {
+       float: right;
+       position: relative;
+       line-height: 20px;
+       text-decoration: none;
+       background: rgba(0, 0, 0, 0.1);
+       font-size: 11px;
+       padding: 0 10px;
+       border-radius: 3px;
+       box-shadow: 0 -1px 0 0 rgba(0, 0, 0, 0.2);
+}
+
+#clear-completed:hover {
+       background: rgba(0, 0, 0, 0.15);
+       box-shadow: 0 -1px 0 0 rgba(0, 0, 0, 0.3);
+}
+
+#info {
+       margin: 65px auto 0;
+       color: #a6a6a6;
+       font-size: 12px;
+       text-shadow: 0 1px 0 rgba(255, 255, 255, 0.7);
+       text-align: center;
+}
+
+#info a {
+       color: inherit;
+}
+
+/*
+       Hack to remove background from Mobile Safari.
+       Can't use it globally since it destroys checkboxes in Firefox and Opera
+*/
+@media screen and (-webkit-min-device-pixel-ratio:0) {
+       #toggle-all,
+       #todo-list li .toggle {
+               background: none;
+       }
+
+       #todo-list li .toggle {
+               height: 40px;
+       }
+
+       #toggle-all {
+               top: -56px;
+               left: -15px;
+               width: 65px;
+               height: 41px;
+               -webkit-transform: rotate(90deg);
+               transform: rotate(90deg);
+               -webkit-appearance: none;
+               appearance: none;
+       }
+}
+
+.hidden{
+       display:none;
+}
diff --git a/demo/todo/assets/base.js b/demo/todo/assets/base.js
new file mode 100644 (file)
index 0000000..6cf09a2
--- /dev/null
@@ -0,0 +1,7 @@
+(function( window ) {
+       'use strict';
+
+       if ( location.hostname === 'todomvc.com' ) {
+               var _gaq=[['_setAccount','UA-31081062-1'],['_trackPageview']];(function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0];g.src='//www.google-analytics.com/ga.js';s.parentNode.insertBefore(g,s)}(document,'script'));
+       }
+})( window );
diff --git a/demo/todo/assets/bg.png b/demo/todo/assets/bg.png
new file mode 100644 (file)
index 0000000..b2a7600
Binary files /dev/null and b/demo/todo/assets/bg.png differ
diff --git a/demo/todo/index.html b/demo/todo/index.html
new file mode 100644 (file)
index 0000000..9d77144
--- /dev/null
@@ -0,0 +1,39 @@
+<!doctype html>
+<html lang="en">
+<head>
+       <meta charset="utf-8">
+       <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
+       <title>remotestorage • TodoMVC</title>
+       <link rel="stylesheet" href="assets/base.css">
+       <!--[if IE]>
+       <script src="assets/ie.js"></script>
+       <![endif]-->
+</head>
+<body>
+       <section id="todoapp">
+               <header id="header">
+                       <h1>todos</h1>
+                       <input id="new-todo" placeholder="What needs to be done?" autofocus>
+               </header>
+               <section id="main">
+                       <input id="toggle-all" type="checkbox">
+                       <label for="toggle-all">Mark all as complete</label>
+                       <ul id="todo-list"></ul>
+               </section>
+               <footer id="footer">
+                       <span id="todo-count"></span>
+                       <button id="clear-completed">Clear completed</button>
+               </footer>
+       </section>
+       <footer id="info">
+               <p>Double-click to edit a todo</p>
+               <p>Template by <a href="http://github.com/sindresorhus">Sindre Sorhus</a></p>
+               <p>Created by <a href="http://twitter.com/ffesseler">Florian Fesseler</a></p>
+               <p>Cleanup, edits by <a href="http://github.com/boushley">Aaron Boushley</a></p>
+       </footer>
+       <script src="assets/base.js"></script>
+       <script src="js/remotestorage.js"></script>
+       <script src="js/tasks.js"></script>
+       <script src="js/app.js"></script>
+</body>
+</html>
diff --git a/demo/todo/js/app.js b/demo/todo/js/app.js
new file mode 100644 (file)
index 0000000..c75d30b
--- /dev/null
@@ -0,0 +1,229 @@
+(function() {
+  'use strict';
+
+  var stat = {},
+    ENTER_KEY = 13;
+
+  window.addEventListener( 'load', windowLoadHandler, false );
+
+  function Stat() {
+    this.todoLeft = 0;
+    this.todoCompleted = 0;
+    this.totalTodo = 0;
+  }
+
+  function windowLoadHandler() {
+    remoteStorage.access.claim('tasks', 'rw');
+    remoteStorage.displayWidget();
+    remoteStorage.tasks.init();
+    paintAll();
+    addEventListeners();
+
+    remoteStorage.tasks.onChange(paintAll);
+    remoteStorage.on('disconnect', paintAll);
+  }
+
+  function addEventListeners() {
+    document.getElementById('new-todo').addEventListener( 'keypress', newTodoKeyPressHandler, false );
+    document.getElementById('toggle-all').addEventListener( 'change', toggleAllChangeHandler, false );
+  }
+
+  function inputEditTodoKeyPressHandler( event ) {
+    var inputEditTodo = event.target,
+      trimmedText = inputEditTodo.value.trim(),
+      todoId = event.target.id.slice( 6 );
+
+    if ( trimmedText ) {
+      if ( event.keyCode === ENTER_KEY ) {
+        remoteStorage.tasks.setTodoText( todoId, trimmedText );
+      }
+    } else {
+      remoteStorage.tasks.removeTodo( todoId );
+    }
+  }
+
+  function inputEditTodoBlurHandler( event ) {
+    var inputEditTodo = event.target,
+      todoId = event.target.id.slice( 6 );
+
+    remoteStorage.tasks.setTodoText( todoId, inputEditTodo.value );
+  }
+
+  function newTodoKeyPressHandler( event ) {
+    var trimmedText = document.getElementById('new-todo').value.trim();
+    if ( event.keyCode === ENTER_KEY && trimmedText ) {
+      remoteStorage.tasks.addTodo( trimmedText );
+    }
+  }
+
+  function toggleAllChangeHandler( event ) {
+    remoteStorage.tasks.setAllTodosCompleted( event.target.checked );
+  }
+
+  function spanDeleteClickHandler( event ) {
+    remoteStorage.tasks.removeTodo( event.target.getAttribute('data-todo-id') );
+  }
+
+  function hrefClearClickHandler() {
+    remoteStorage.tasks.removeAllCompletedTodos();
+  }
+
+  function todoContentHandler( event ) {
+    var todoId = event.target.getAttribute('data-todo-id'),
+      div = document.getElementById( 'li_' + todoId ),
+      inputEditTodo = document.getElementById( 'input_' + todoId );
+
+    div.className = 'editing';
+    inputEditTodo.focus();
+  }
+
+  function checkboxChangeHandler( event ) {
+    var checkbox = event.target;
+    remoteStorage.tasks.setTodoCompleted( checkbox.getAttribute('data-todo-id'), checkbox.checked );
+  }
+
+  function paintAll() {
+    remoteStorage.tasks.getTodos().then( function( todosMap ) {
+      var todosArr = [], i;
+      for( i in  todosMap ) {
+        todosArr.push( todosMap[i] );
+      }
+      computeStats( todosArr );
+      redrawTodosUI( todosArr );
+      redrawStatsUI( todosArr );
+      changeToggleAllCheckboxState( todosArr );
+    }, function(err) {
+      console.log('error during getTodos', err);
+    });
+  }
+
+  function computeStats( todos ) {
+    var i, l;
+
+    stat = new Stat();
+    stat.totalTodo = todos.length;
+
+    for ( i = 0, l = todos.length; i < l; i++ ) {
+      if ( todos[ i ].completed ) {
+        stat.todoCompleted++;
+      }
+    }
+
+    stat.todoLeft = stat.totalTodo - stat.todoCompleted;
+  }
+
+
+  function redrawTodosUI( todos ) {
+
+    var todo, checkbox, label, deleteLink, divDisplay, inputEditTodo, li, i, l,
+      ul = document.getElementById('todo-list');
+
+    document.getElementById('main').style.display = todos.length ? 'block' : 'none';
+
+    ul.innerHTML = '';
+    document.getElementById('new-todo').value = '';
+
+    for ( i = 0, l = todos.length; i < l; i++ ) {
+      todo = todos[ i ];
+
+      // create checkbox
+      checkbox = document.createElement('input');
+      checkbox.className = 'toggle';
+      checkbox.setAttribute( 'data-todo-id', todo.id );
+      checkbox.type = 'checkbox';
+      checkbox.addEventListener( 'change', checkboxChangeHandler );
+
+      // create div text
+      label = document.createElement('label');
+      label.setAttribute( 'data-todo-id', todo.id );
+      label.appendChild( document.createTextNode( todo.title ) );
+      label.addEventListener( 'dblclick', todoContentHandler );
+
+
+      // create delete button
+      deleteLink = document.createElement('button');
+      deleteLink.className = 'destroy';
+      deleteLink.setAttribute( 'data-todo-id', todo.id );
+      deleteLink.addEventListener( 'click', spanDeleteClickHandler );
+
+      // create divDisplay
+      divDisplay = document.createElement('div');
+      divDisplay.className = 'view';
+      divDisplay.setAttribute( 'data-todo-id', todo.id );
+      divDisplay.appendChild( checkbox );
+      divDisplay.appendChild( label );
+      divDisplay.appendChild( deleteLink );
+
+      // create todo input
+      inputEditTodo = document.createElement('input');
+      inputEditTodo.id = 'input_' + todo.id;
+      inputEditTodo.className = 'edit';
+      inputEditTodo.value = todo.title;
+      inputEditTodo.addEventListener( 'keypress', inputEditTodoKeyPressHandler );
+      inputEditTodo.addEventListener( 'blur', inputEditTodoBlurHandler );
+
+
+      // create li
+      li = document.createElement('li');
+      li.id = 'li_' + todo.id;
+      li.appendChild( divDisplay );
+      li.appendChild( inputEditTodo );
+
+
+      if ( todo.completed ) {
+        li.className += 'complete';
+        checkbox.checked = true;
+      }
+
+      ul.appendChild( li );
+    }
+  }
+
+  function changeToggleAllCheckboxState( todos ) {
+    var toggleAll = document.getElementById('toggle-all');
+
+    toggleAll.checked = stat.todoCompleted === todos.length;
+  }
+
+  function redrawStatsUI( todos ) {
+    removeChildren( document.getElementsByTagName('footer')[0] );
+    document.getElementById('footer').style.display = todos.length ? 'block' : 'none';
+
+    if ( stat.todoCompleted ) {
+      drawTodoClear();
+    }
+
+    if ( stat.totalTodo ) {
+      drawTodoCount();
+    }
+  }
+
+  function drawTodoCount() {
+    var number = document.createElement('strong'),
+      remaining = document.createElement('span'),
+      text = ' ' + ( stat.todoLeft === 1 ? 'item' : 'items' ) + ' left';
+
+    // create remaining count
+    number.innerHTML = stat.todoLeft;
+
+    remaining.id = 'todo-count';
+    remaining.appendChild( number );
+    remaining.appendChild( document.createTextNode( text ) );
+
+    document.getElementsByTagName('footer')[0].appendChild( remaining );
+  }
+
+  function drawTodoClear() {
+    var buttonClear = document.createElement('button');
+
+    buttonClear.id = 'clear-completed';
+    buttonClear.addEventListener( 'click', hrefClearClickHandler );
+    buttonClear.innerHTML = 'Clear completed (' + stat.todoCompleted + ')';
+
+    document.getElementsByTagName('footer')[0].appendChild( buttonClear );
+  }
+
+  function removeChildren( node ) {
+    node.innerHTML = '';
+  }
+})();
diff --git a/demo/todo/js/remotestorage.js b/demo/todo/js/remotestorage.js
new file mode 100644 (file)
index 0000000..621aaaf
--- /dev/null
@@ -0,0 +1,4800 @@
+/** remotestorage.js 0.8.0-head remotestorage.io, MIT-licensed **/
+
+/** FILE: lib/promising.js **/
+(function(global) {
+  function getPromise(builder) {
+    var promise;
+
+    if(typeof(builder) === 'function') {
+      setTimeout(function() {
+        try {
+          builder(promise);
+        } catch(e) {
+          promise.reject(e);
+        }
+      }, 0);
+    }
+
+    var consumers = [], success, result;
+
+    function notifyConsumer(consumer) {
+      if(success) {
+        var nextValue;
+        if(consumer.fulfilled) {
+          try {
+            nextValue = [consumer.fulfilled.apply(null, result)];
+          } catch(exc) {
+            consumer.promise.reject(exc);
+            return;
+          }
+        } else {
+          nextValue = result;
+        }
+        if(nextValue[0] && typeof(nextValue[0].then) === 'function') {
+          nextValue[0].then(consumer.promise.fulfill, consumer.promise.reject);
+        } else {
+          consumer.promise.fulfill.apply(null, nextValue);
+        }
+      } else {
+        if(consumer.rejected) {
+          var ret;
+          try {
+            ret = consumer.rejected.apply(null, result);
+          } catch(exc) {
+            consumer.promise.reject(exc);
+            return;
+          }
+          if(ret && typeof(ret.then) === 'function') {
+            ret.then(consumer.promise.fulfill, consumer.promise.reject);
+          } else {
+            consumer.promise.fulfill(ret);
+          }
+        } else {
+          consumer.promise.reject.apply(null, result);
+        }
+      }
+    }
+
+    function resolve(succ, res) {
+      if(result) {
+        console.error("WARNING: Can't resolve promise, already resolved!");
+        return;
+      }
+      success = succ;
+      result = Array.prototype.slice.call(res);
+      setTimeout(function() {
+        var cl = consumers.length;
+        if(cl === 0 && (! success)) {
+          console.error("Possibly uncaught error: ", result, result[0] && result[0].stack);
+        }
+        for(var i=0;i<cl;i++) {
+          notifyConsumer(consumers[i]);
+        }
+        consumers = undefined;
+      }, 0);
+    }
+
+    promise = {
+
+      then: function(fulfilled, rejected) {
+        var consumer = {
+          fulfilled: typeof(fulfilled) === 'function' ? fulfilled : undefined,
+          rejected: typeof(rejected) === 'function' ? rejected : undefined,
+          promise: getPromise()
+        };
+        if(result) {
+          setTimeout(function() {
+            notifyConsumer(consumer)
+          }, 0);
+        } else {
+          consumers.push(consumer);
+        }
+        return consumer.promise;
+      },
+
+      fulfill: function() {
+        resolve(true, arguments);
+        return this;
+      },
+      
+      reject: function() {
+        resolve(false, arguments);
+        return this;
+      }
+      
+    };
+
+    return promise;
+  };
+
+  global.promising = getPromise;
+
+})(this);
+
+
+/** FILE: src/remotestorage.js **/
+(function(global) {
+
+  var SyncedGetPutDelete = {
+    get: function(path) {
+      if(this.caching.cachePath(path)) {
+        return this.local.get(path);
+      } else {
+        return this.remote.get(path);
+      }
+    },
+
+    put: function(path, body, contentType) {
+      if(this.caching.cachePath(path)) {
+        return this.local.put(path, body, contentType);
+      } else {
+        return SyncedGetPutDelete._wrapBusyDone.call(this, this.remote.put(path, body, contentType));
+      }
+    },
+
+    'delete': function(path) {
+      if(this.caching.cachePath(path)) {
+        return this.local.delete(path);
+      } else {
+        return SyncedGetPutDelete._wrapBusyDone.call(this, this.remote.delete(path));
+      }
+    },
+
+    _wrapBusyDone: function(result) {
+      this._emit('sync-busy');
+      return result.then(function() {
+        var promise = promising();
+        this._emit('sync-done');
+        return promise.fulfill.apply(promise, arguments);
+      }.bind(this), function(err) {
+        throw err;
+      });
+    }
+  }
+
+  /**
+   * Class: RemoteStorage
+   *
+   * Constructor for global remoteStorage object.
+   *
+   * This class primarily contains feature detection code and a global convenience API.
+   *
+   * Depending on which features are built in, it contains different attributes and
+   * functions. See the individual features for more information.
+   *
+   */
+  var RemoteStorage = function() {
+    RemoteStorage.eventHandling(
+      this, 'ready', 'disconnected', 'disconnect', 'conflict', 'error',
+      'features-loaded', 'connecting', 'authing', 'sync-busy', 'sync-done'
+    );
+    // pending get/put/delete calls.
+    this._pending = [];
+    this._setGPD({
+      get: this._pendingGPD('get'),
+      put: this._pendingGPD('put'),
+      delete: this._pendingGPD('delete')
+    });
+    this._cleanups = [];
+    this._pathHandlers = { change: {}, conflict: {} };
+
+    var origOn = this.on;
+    this.on = function(eventName, handler) {
+      if(eventName == 'ready' && this.remote.connected && this._allLoaded) {
+        setTimeout(handler, 0);
+      } else if(eventName == 'features-loaded' && this._allLoaded) {
+        setTimeout(handler, 0);
+      }
+      return origOn.call(this, eventName, handler);
+    }
+
+    this._init();
+
+    this.on('ready', function() {
+      if(this.local) {
+        setTimeout(this.local.fireInitial.bind(this.local), 0);
+      }
+    }.bind(this));
+  };
+
+  RemoteStorage.DiscoveryError = function(message) {
+    Error.apply(this, arguments);
+    this.message = message;
+  };
+  RemoteStorage.DiscoveryError.prototype = Object.create(Error.prototype);
+
+  RemoteStorage.Unauthorized = function() { Error.apply(this, arguments); };
+  RemoteStorage.Unauthorized.prototype = Object.create(Error.prototype);
+
+  /**
+   * Method: RemoteStorage.log
+   *
+   * Logging using console.log, when logging is enabled.
+   */
+  RemoteStorage.log = function() {
+    if(RemoteStorage._log) {
+      console.log.apply(console, arguments);
+    }
+  };
+
+  RemoteStorage.prototype = {
+
+    /**
+     ** PUBLIC INTERFACE
+     **/
+
+    /**
+     * Method: connect
+     *
+     * Connect to a remotestorage server.
+     *
+     * Parameters:
+     *   userAddress - The user address (user@host) to connect to.
+     *
+     * Discovers the webfinger profile of the given user address and
+     * initiates the OAuth dance.
+     *
+     * This method must be called *after* all required access has been claimed.
+     *
+     */
+    connect: function(userAddress) {
+      if( userAddress.indexOf('@') < 0) {
+        this._emit('error', new RemoteStorage.DiscoveryError("user adress doesn't contain an @"));
+        return;
+      }
+      this._emit('connecting');
+      this.remote.configure(userAddress);
+      RemoteStorage.Discover(userAddress,function(href, storageApi, authURL){
+        if(!href){
+          this._emit('error', new RemoteStorage.DiscoveryError('failed to contact storage server'));
+          return;
+        }
+        this._emit('authing');
+        this.remote.configure(userAddress, href, storageApi);
+        if(! this.remote.connected) {
+          if(authURL) {
+            this.authorize(authURL);
+          } else {
+           // In lieu of an excplicit authURL, assume that the browser
+           // and server handle any authorization needs; for instance,
+           // TLS may trigger the browser to use a client certificate,
+           // or a 401 Not Authorized response may make the browser
+           // send a Kerberos ticket using the SPNEGO method.
+           this.impliedauth();
+         }
+        }
+      }.bind(this));
+    },
+
+    /**
+     * Method: disconnect
+     *
+     * "Disconnect" from remotestorage server to terminate current session.
+     * This method clears all stored settings and deletes the entire local cache.
+     *
+     * Once the disconnect is complete, the "disconnected" event will be fired.
+     * From that point on you can connect again (using <connect>).
+     */
+    disconnect: function() {
+      if(this.remote) {
+        this.remote.configure(null, null, null, null);
+      }
+      this._setGPD({
+        get: this._pendingGPD('get'),
+        put: this._pendingGPD('put'),
+        delete: this._pendingGPD('delete')
+      });
+      var n = this._cleanups.length, i = 0;
+      var oneDone = function() {
+        i++;
+        if(i == n) {
+          this._init();
+          this._emit('disconnected');
+          this._emit('disconnect');// DEPRECATED?
+        }
+      }.bind(this);
+      this._cleanups.forEach(function(cleanup) {
+        var cleanupResult = cleanup(this);
+        if(typeof(cleanup) == 'object' && typeof(cleanup.then) == 'function') {
+          cleanupResult.then(oneDone);
+        } else {
+          oneDone();
+        }
+      }.bind(this));
+    },
+
+    /**
+     * Method: onChange
+     *
+     * Adds a 'change' event handler to the given path.
+     * Whenever a 'change' happens (as determined by the backend, such
+     * as <RemoteStorage.IndexedDB>) and the affected path is equal to
+     * or below the given 'path', the given handler is called.
+     *
+     * You shouldn't need to use this method directly, but instead use
+     * the "change" events provided by <RemoteStorage.BaseClient>.
+     *
+     * Parameters:
+     *   path    - Absolute path to attach handler to.
+     *   handler - Handler function.
+     */
+    onChange: function(path, handler) {
+      if(! this._pathHandlers.change[path]) {
+        this._pathHandlers.change[path] = [];
+      }
+      this._pathHandlers.change[path].push(handler);
+    },
+
+    onConflict: function(path, handler) {
+      if(! this._conflictBound) {
+        this.on('features-loaded', function() {
+          if(this.local) {
+            this.local.on('conflict', this._dispatchEvent.bind(this, 'conflict'));
+          }
+        }.bind(this));
+        this._conflictBound = true;
+      }
+      if(! this._pathHandlers.conflict[path]) {
+        this._pathHandlers.conflict[path] = [];
+      }
+      this._pathHandlers.conflict[path].push(handler);
+    },
+
+    /**
+     * Method: enableLog
+     *
+     * enable logging
+     */
+    enableLog: function() {
+      RemoteStorage._log = true;
+    },
+
+    /**
+     * Method: disableLog
+     *
+     * disable logging
+     */
+    disableLog: function() {
+      RemoteStorage._log = false;
+    },
+
+    /**
+     * Method: log
+     *
+     * The same as <RemoteStorage.log>.
+     */
+    log: function() {
+      RemoteStorage.log.apply(RemoteStorage, arguments);
+    },
+
+    /**
+     ** INITIALIZATION
+     **/
+
+    _init: function() {
+      this._loadFeatures(function(features) {
+        this.log('all features loaded');
+        this.local = features.local && new features.local();
+        // (this.remote set by WireClient._rs_init
+        //  as lazy property on RS.prototype)
+
+        if(this.local && this.remote) {
+          this._setGPD(SyncedGetPutDelete, this);
+          this._bindChange(this.local);
+        } else if(this.remote) {
+          this._setGPD(this.remote, this.remote);
+        }
+
+        if(this.remote) {
+          this.remote.on('connected', function() {
+            try {
+              this._emit('ready');
+            } catch(e) {
+              console.error("'ready' failed: ", e, e.stack);
+              this._emit('error', e);
+            };
+          }.bind(this));
+          if(this.remote.connected) {
+            try {
+              this._emit('ready');
+            } catch(e) {
+              console.error("'ready' failed: ", e, e.stack);
+              this._emit('error', e);
+            };
+          }
+        }
+
+        var fl = features.length;
+        for(var i=0;i<fl;i++) {
+          var cleanup = features[i].cleanup;
+          if(cleanup) {
+            this._cleanups.push(cleanup);
+          }
+        }
+
+        try {
+          this._allLoaded = true;
+          this._emit('features-loaded');
+        } catch(exc) {
+          console.error("remoteStorage#ready block failed: ");
+          if(typeof(exc) == 'string') {
+            console.error(exc);
+          } else {
+            console.error(exc.message, exc.stack);
+          }
+          this._emit('error', exc);
+        }
+        this._processPending();
+      });
+    },
+
+    /**
+     ** FEATURE DETECTION
+     **/
+
+    _detectFeatures: function() {
+      // determine availability
+      var features = [
+        'WireClient',
+        'Access',
+        'Caching',
+        'Discover',
+        'Authorize',
+             'Widget',
+        'IndexedDB',
+        'LocalStorage',
+        'Sync',
+        'BaseClient'
+      ].map(function(featureName) {
+        var impl = RemoteStorage[featureName];
+        return {
+          name: featureName,
+          init: (impl && impl._rs_init),
+          supported: impl && (impl._rs_supported ? impl._rs_supported() : true),
+          cleanup: ( impl && impl._rs_cleanup )
+        };
+      }).filter(function(feature) {
+        var supported = !! (feature.init && feature.supported);
+        this.log("[FEATURE " + feature.name + "] " + (supported ? '' : 'not ') + 'supported.');
+        return supported;
+      }.bind(this));
+
+      features.forEach(function(feature) {
+        if(feature.name == 'IndexedDB') {
+          features.local = RemoteStorage.IndexedDB;
+        } else if(feature.name == 'LocalStorage' && ! features.local) {
+          features.local = RemoteStorage.LocalStorage;
+        }
+      });
+      features.caching = !!RemoteStorage.Caching;
+      features.sync = !!RemoteStorage.Sync;
+
+      this.features = features;
+
+      return features;
+    },
+
+    _loadFeatures: function(callback) {
+      var features = this._detectFeatures();
+      var n = features.length, i = 0;
+      var self = this;
+      function featureDoneCb(name) {
+        return function() {
+          i++;
+          self.log("[FEATURE " + name + "] initialized. (" + i + "/" + n + ")");
+          if(i == n)
+            setTimeout(function() {
+              callback.apply(self, [features]);
+            }, 0);
+        }
+      }
+      features.forEach(function(feature) {
+        self.log("[FEATURE " + feature.name + "] initializing...");
+        var initResult = feature.init(self);
+        var cb = featureDoneCb(feature.name);
+        if(typeof(initResult) == 'object' && typeof(initResult.then) == 'function') {
+          initResult.then(cb);
+        } else {
+          cb();
+        }
+      });
+    },
+
+    /**
+     ** GET/PUT/DELETE INTERFACE HELPERS
+     **/
+
+    _setGPD: function(impl, context) {
+      this.get = impl.get.bind(context);
+      this.put = impl.put.bind(context);
+      this.delete = impl.delete.bind(context);
+    },
+
+    _pendingGPD: function(methodName) {
+      return function() {
+        var promise = promising();
+        this._pending.push({
+          method: methodName,
+          args: Array.prototype.slice.call(arguments),
+          promise: promise
+        });
+        return promise;
+      }.bind(this);
+    },
+
+    _processPending: function() {
+      this._pending.forEach(function(pending) {
+        this[pending.method].apply(this, pending.args).then(pending.promise.fulfill, pending.promise.reject);
+      }.bind(this));
+      this._pending = [];
+    },
+
+    /**
+     ** CHANGE EVENT HANDLING
+     **/
+
+    _bindChange: function(object) {
+      object.on('change', this._dispatchEvent.bind(this, 'change'));
+    },
+
+    _dispatchEvent: function(eventName, event) {
+      for(var path in this._pathHandlers[eventName]) {
+        var pl = path.length;
+        this._pathHandlers[eventName][path].forEach(function(handler) {
+          if(event.path.substr(0, pl) == path) {
+            var ev = {};
+            for(var key in event) { ev[key] = event[key]; }
+            ev.relativePath = event.path.replace(new RegExp('^' + path), '');
+            try {
+              handler(ev);
+            } catch(e) {
+              console.error("'change' handler failed: ", e, e.stack);
+              this._emit('error', e);
+            }
+          }
+        }.bind(this));
+      }
+    }
+  };
+
+  /**
+   * Method: claimAccess
+   *
+   * High-level method to claim access on one or multiple scopes and enable
+   * caching for them. WARNING: when using Caching control, use remoteStorage.access.claim instead,
+   * see https://github.com/remotestorage/remotestorage.js/issues/380
+   *
+   * Examples:
+   *   (start code)
+   *     remoteStorage.claimAccess('foo', 'rw');
+   *     // is equivalent to:
+   *     remoteStorage.claimAccess({ foo: 'rw' });
+   *
+   *     // is equivalent to:
+   *     remoteStorage.access.claim('foo', 'rw');
+   *     remoteStorage.caching.enable('/foo/');
+   *     remoteStorage.caching.enable('/public/foo/');
+   *   (end code)
+   */
+
+  /**
+   * Property: connected
+   *
+   * Boolean property indicating if remoteStorage is currently connected.
+   */
+  Object.defineProperty(RemoteStorage.prototype, 'connected', {
+    get: function() {
+      return this.remote.connected;
+    }
+  });
+
+  /**
+   * Property: access
+   *
+   * Tracking claimed access scopes. A <RemoteStorage.Access> instance.
+   *
+   *
+   * Property: caching
+   *
+   * Caching settings. A <RemoteStorage.Caching> instance.
+   *
+   * (only available when caching is built in)
+   *
+   *
+   * Property: remote
+   *
+   * Access to the remote backend used. Usually a <RemoteStorage.WireClient>.
+   *
+   *
+   * Property: local
+   *
+   * Access to the local caching backend used.
+   * Only available when caching is built in.
+   * Usually either a <RemoteStorage.IndexedDB> or <RemoteStorage.LocalStorage>
+   * instance.
+   */
+
+  global.RemoteStorage = RemoteStorage;
+
+})(this);
+
+
+/** FILE: src/eventhandling.js **/
+(function(global) {
+  /**
+   * Class: eventhandling
+   */
+  var methods = {
+    /**
+     * Method: addEventListener
+     *
+     * Install an event handler for the given event name.
+     */
+    addEventListener: function(eventName, handler) {
+      this._validateEvent(eventName);
+      this._handlers[eventName].push(handler);
+    },
+
+    /**
+     * Method: removeEventListener
+     *
+     * Remove a previously installed event handler
+     */
+    removeEventListener: function(eventName, handler) {
+      this._validateEvent(eventName);
+      var hl = this._handlers[eventName].length;
+      for(var i=0;i<hl;i++) {
+        if(this._handlers[eventName][i] === handler) {
+          this._handlers[eventName].splice(i, 1);
+          return;
+        }
+      }
+    },
+
+    _emit: function(eventName) {
+      this._validateEvent(eventName);
+      var args = Array.prototype.slice.call(arguments, 1);
+      this._handlers[eventName].forEach(function(handler) {
+        handler.apply(this, args);
+      });
+    },
+
+    _validateEvent: function(eventName) {
+      if(! (eventName in this._handlers)) {
+        throw new Error("Unknown event: " + eventName);
+      }
+    },
+
+    _delegateEvent: function(eventName, target) {
+      target.on(eventName, function(event) {
+        this._emit(eventName, event);
+      }.bind(this));
+    },
+
+    _addEvent: function(eventName) {
+      this._handlers[eventName] = [];
+    }
+  };
+
+  // Method: eventhandling.on
+  // Alias for <addEventListener>
+  methods.on = methods.addEventListener;
+
+  /**
+   * Function: eventHandling
+   *
+   * Mixes event handling functionality into an object.
+   *
+   * The first parameter is always the object to be extended.
+   * All remaining parameter are expected to be strings, interpreted as valid event
+   * names.
+   *
+   * Example:
+   *   (start code)
+   *   var MyConstructor = function() {
+   *     eventHandling(this, 'connected', 'disconnected');
+   *
+   *     this._emit('connected');
+   *     this._emit('disconnected');
+   *     // this would throw an exception:
+   *     //this._emit('something-else');
+   *   };
+   *
+   *   var myObject = new MyConstructor();
+   *   myObject.on('connected', function() { console.log('connected'); });
+   *   myObject.on('disconnected', function() { console.log('disconnected'); });
+   *   // this would throw an exception as well:
+   *   //myObject.on('something-else', function() {});
+   *
+   *   (end code)
+   */
+  RemoteStorage.eventHandling = function(object) {
+    var eventNames = Array.prototype.slice.call(arguments, 1);
+    for(var key in methods) {
+      object[key] = methods[key];
+    }
+    object._handlers = {};
+    eventNames.forEach(function(eventName) {
+      object._addEvent(eventName);
+    });
+  };
+})(this);
+
+
+/** FILE: src/wireclient.js **/
+(function(global) {
+  var RS = RemoteStorage;
+
+  /**
+   * WireClient Interface
+   * --------------------
+   *
+   * This file exposes a get/put/delete interface on top of XMLHttpRequest.
+   * It requires to be configured with parameters about the remotestorage server to
+   * connect to.
+   * Each instance of WireClient is always associated with a single remotestorage
+   * server and access token.
+   *
+   * Usually the WireClient instance can be accessed via `remoteStorage.remote`.
+   *
+   * This is the get/put/delete interface:
+   *
+   *   - #get() takes a path and optionally a ifNoneMatch option carrying a version
+   *     string to check. It returns a promise that will be fulfilled with the HTTP
+   *     response status, the response body, the MIME type as returned in the
+   *     'Content-Type' header and the current revision, as returned in the 'ETag'
+   *     header.
+   *   - #put() takes a path, the request body and a content type string. It also
+   *     accepts the ifMatch and ifNoneMatch options, that map to the If-Match and
+   *     If-None-Match headers respectively. See the remotestorage-01 specification
+   *     for details on handling these headers. It returns a promise, fulfilled with
+   *     the same values as the one for #get().
+   *   - #delete() takes a path and the ifMatch option as well. It returns a promise
+   *     fulfilled with the same values as the one for #get().
+   *
+   * In addition to this, the WireClient has some compatibility features to work with
+   * remotestorage 2012.04 compatible storages. For example it will cache revisions
+   * from directory listings in-memory and return them accordingly as the "revision"
+   * parameter in response to #get() requests. Similarly it will return 404 when it
+   * receives an empty directory listing, to mimic remotestorage-01 behavior. Note
+   * that it is not always possible to know the revision beforehand, hence it may
+   * be undefined at times (especially for caching-roots).
+   */
+
+  var haveLocalStorage;
+  var SETTINGS_KEY = "remotestorage:wireclient";
+
+  var API_2012 = 1, API_00 = 2, API_01 = 3, API_HEAD = 4;
+
+  var STORAGE_APIS = {
+    'draft-dejong-remotestorage-00': API_00,
+    'draft-dejong-remotestorage-01': API_01,
+    'https://www.w3.org/community/rww/wiki/read-write-web-00#simple': API_2012
+  };
+
+  var isArrayBufferView;
+  if(typeof(ArrayBufferView) === 'function') {
+    isArrayBufferView = function(object) { return object && (object instanceof ArrayBufferView); };
+  } else {
+    var arrayBufferViews = [
+      Int8Array, Uint8Array, Int16Array, Uint16Array,
+      Int32Array, Uint32Array, Float32Array, Float64Array
+    ];
+    isArrayBufferView = function(object) {
+      for(var i=0;i<8;i++) {
+        if(object instanceof arrayBufferViews[i]) {
+          return true;
+        }
+      }
+      return false;
+    };
+  }
+
+  function request(method, uri, token, headers, body, getEtag, fakeRevision) {
+    if((method == 'PUT' || method == 'DELETE') && uri[uri.length - 1] == '/') {
+      throw "Don't " + method + " on directories!";
+    }
+
+    var timedOut = false;
+    var timer = setTimeout(function() {
+      timedOut = true;
+      promise.reject('timeout');
+    }, RS.WireClient.REQUEST_TIMEOUT);
+
+    var promise = promising();
+    RemoteStorage.log(method, uri);
+    var xhr = new XMLHttpRequest();
+    xhr.open(method, uri, true);
+    if (token != 'implied') {
+      xhr.setRequestHeader('Authorization', 'Bearer ' + token);
+    }
+    for(var key in headers) {
+      if(typeof(headers[key]) !== 'undefined') {
+        xhr.setRequestHeader(key, headers[key]);
+      }
+    }
+    xhr.onload = function() {
+      if(timedOut) return;
+      clearTimeout(timer);
+      if(xhr.status == 404) return promise.fulfill(xhr.status);
+      var mimeType = xhr.getResponseHeader('Content-Type');
+      var body;
+      var revision = getEtag ? xhr.getResponseHeader('ETag') : (xhr.status == 200 ? fakeRevision : undefined);
+      if((! mimeType) || mimeType.match(/charset=binary/)) {
+        var blob = new Blob([xhr.response], {type: mimeType});
+        var reader = new FileReader();
+        reader.addEventListener("loadend", function() {
+          // reader.result contains the contents of blob as a typed array
+          promise.fulfill(xhr.status, reader.result, mimeType, revision);
+        });
+        reader.readAsArrayBuffer(blob);
+      } else {
+        body = mimeType && mimeType.match(/^application\/json/) ? JSON.parse(xhr.responseText) : xhr.responseText;
+        promise.fulfill(xhr.status, body, mimeType, revision);
+      }
+    };
+    xhr.onerror = function(error) {
+      if(timedOut) return;
+      clearTimeout(timer);
+      promise.reject(error);
+    };
+    if(typeof(body) === 'object') {
+      if(isArrayBufferView(body)) { /* alright. */ }
+      else if(body instanceof ArrayBuffer) {
+        body = new Uint8Array(body);
+      } else {
+        body = JSON.stringify(body);
+      }
+    }
+    xhr.send(body);
+    return promise;
+  }
+
+  function cleanPath(path) {
+    // strip duplicate slashes.
+    return path.replace(/\/+/g, '/');
+  }
+
+  RS.WireClient = function(rs) {
+    this.connected = false;
+    RS.eventHandling(this, 'change', 'connected');
+    rs.on('error', function(error){
+      if(error instanceof RemoteStorage.Unauthorized) {
+        this.configure(undefined, undefined, undefined, null);
+      }
+    }.bind(this))
+    if(haveLocalStorage) {
+      var settings;
+      try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {};
+      if(settings) {
+        this.configure(settings.userAddress, settings.href, settings.storageApi, settings.token);
+      }
+    }
+
+    this._revisionCache = {};
+
+    if(this.connected) {
+      setTimeout(this._emit.bind(this), 0, 'connected');
+    }
+  };
+
+  RS.WireClient.REQUEST_TIMEOUT = 30000;
+
+  RS.WireClient.prototype = {
+
+    /**
+     * Property: token
+     *
+     * Holds the bearer token of this WireClient, as obtained in the OAuth dance
+     *
+     * Example:
+     *   (start code)
+     *
+     *   remoteStorage.remote.token
+     *   // -> 'DEADBEEF01=='
+     */
+
+    /**
+     * Property: href
+     *
+     * Holds the server's base URL, as obtained in the Webfinger discovery
+     *
+     * Example:
+     *   (start code)
+     *
+     *   remoteStorage.remote.href
+     *   // -> 'https://storage.example.com/users/jblogg/'
+     */
+
+    /**
+     * Property: storageApi
+     *
+     * Holds the spec version the server claims to be compatible with
+     *
+     * Example:
+     *   (start code)
+     *
+     *   remoteStorage.remote.storageApi
+     *   // -> 'draft-dejong-remotestorage-01'
+     */
+
+
+    configure: function(userAddress, href, storageApi, token) {
+      if(typeof(userAddress) !== 'undefined') this.userAddress = userAddress;
+      if(typeof(href) !== 'undefined') this.href = href;
+      if(typeof(storageApi) !== 'undefined') this.storageApi = storageApi;
+      if(typeof(token) !== 'undefined') this.token = token;
+      if(typeof(this.storageApi) !== 'undefined') {
+        this._storageApi = STORAGE_APIS[this.storageApi] || API_HEAD;
+        this.supportsRevs = this._storageApi >= API_00;
+      }
+      if(this.href && this.token) {
+        this.connected = true;
+        this._emit('connected');
+      } else {
+        this.connected = false;
+      }
+      if(haveLocalStorage) {
+        localStorage[SETTINGS_KEY] = JSON.stringify({
+          userAddress: this.userAddress,
+          href: this.href,
+          token: this.token,
+          storageApi: this.storageApi
+        });
+      }
+    },
+
+    get: function(path, options) {
+      if(! this.connected) throw new Error("not connected (path: " + path + ")");
+      if(!options) options = {};
+      var headers = {};
+      if(this.supportsRevs) {
+        // setting '' causes the browser (at least chromium) to ommit
+        // the If-None-Match header it would normally send.
+        headers['If-None-Match'] = options.ifNoneMatch || '';
+      } else if(options.ifNoneMatch) {
+        var oldRev = this._revisionCache[path];
+        if(oldRev === options.ifNoneMatch) {
+//since sync descends for allKeys(local, remote), this causes
+// https://github.com/remotestorage/remotestorage.js/issues/399
+//commenting this out so that it gets the actual 404 from the server.
+//this only affects legacy servers (this.supportsRevs==false):
+//
+//           return promising().fulfill(412);
+        }
+      }
+      var promise = request('GET', this.href + cleanPath(path), this.token, headers,
+                            undefined, this.supportsRevs, this._revisionCache[path]);
+      if(this.supportsRevs || path.substr(-1) != '/') {
+        return promise;
+      } else {
+        return promise.then(function(status, body, contentType, revision) {
+          if(status == 200 && typeof(body) == 'object') {
+            if(Object.keys(body).length === 0) {
+              // no children (coerce response to 'not found')
+              status = 404;
+            } else {
+              for(var key in body) {
+                this._revisionCache[path + key] = body[key];
+              }
+            }
+          }
+          return promising().fulfill(status, body, contentType, revision);
+        }.bind(this));
+      }
+    },
+
+    put: function(path, body, contentType, options) {
+      if(! this.connected) throw new Error("not connected (path: " + path + ")");
+      if(!options) options = {};
+      if(! contentType.match(/charset=/)) {
+        contentType += '; charset=' + ((body instanceof ArrayBuffer || isArrayBufferView(body)) ? 'binary' : 'utf-8');
+      }
+      var headers = { 'Content-Type': contentType };
+      if(this.supportsRevs) {
+        headers['If-Match'] = options.ifMatch;
+        headers['If-None-Match'] = options.ifNoneMatch;
+      }
+      return request('PUT', this.href + cleanPath(path), this.token,
+                     headers, body, this.supportsRevs);
+    },
+
+    'delete': function(path, options) {
+      if(! this.connected) throw new Error("not connected (path: " + path + ")");
+      if(!options) options = {};
+      return request('DELETE', this.href + cleanPath(path), this.token,
+                     this.supportsRevs ? { 'If-Match': options.ifMatch } : {},
+                     undefined, this.supportsRevs);
+    }
+
+  };
+
+  RS.WireClient._rs_init = function() {
+    Object.defineProperty(RS.prototype, 'remote', {
+      configurable: true,
+      get: function() {
+        var wireclient = new RS.WireClient(this);
+        Object.defineProperty(this, 'remote', {
+          value: wireclient
+        });
+        return wireclient;
+      }
+    });
+  };
+
+  RS.WireClient._rs_supported = function() {
+    haveLocalStorage = 'localStorage' in global;
+    return !! global.XMLHttpRequest;
+  };
+
+  RS.WireClient._rs_cleanup = function(){
+    if(haveLocalStorage){
+      delete localStorage[SETTINGS_KEY];
+    }
+  }
+
+
+})(this);
+
+
+/** FILE: src/discover.js **/
+(function(global) {
+
+  // feature detection flags
+  var haveXMLHttpRequest, haveLocalStorage;
+  // used to store settings in localStorage
+  var SETTINGS_KEY = 'remotestorage:discover';
+  // cache loaded from localStorage
+  var cachedInfo = {};
+
+  RemoteStorage.Discover = function(userAddress, callback) {
+    if(userAddress in cachedInfo) {
+      var info = cachedInfo[userAddress];
+      callback(info.href, info.type, info.authURL);
+      return;
+    }
+    var hostname = userAddress.split('@')[1]
+    var params = '?resource=' + encodeURIComponent('acct:' + userAddress);
+    var urls = [
+      'https://' + hostname + '/.well-known/webfinger' + params,
+      'https://' + hostname + '/.well-known/host-meta.json' + params,
+      'http://' + hostname + '/.well-known/webfinger' + params,
+      'http://' + hostname + '/.well-known/host-meta.json' + params
+    ];
+    function tryOne() {
+      var xhr = new XMLHttpRequest();
+      var url = urls.shift();
+      if(! url) return callback();
+      RemoteStorage.log('try url', url);
+      xhr.open('GET', url, true);
+      xhr.onabort = xhr.onerror = function() {
+        console.error("webfinger error", arguments, '(', url, ')');
+        tryOne();
+      }
+      xhr.onload = function() {
+        if(xhr.status != 200) return tryOne();
+        var profile;
+         
+        try {
+          profile = JSON.parse(xhr.responseText);
+        } catch(e) {
+          RemoteStorage.log("Failed to parse profile ", xhr.responseText, e);
+          tryOne();
+          return;
+        }
+
+        if (!profile.links) {
+          RemoteStorage.log("profile has no links section ", JSON.stringify(profile));
+          tryOne();
+          return;
+        }
+
+        var link;
+        profile.links.forEach(function(l) {
+          if(l.rel == 'remotestorage') {
+            link = l;
+          } else if(l.rel == 'remoteStorage' && !link) {
+            link = l;
+          }
+        });
+        RemoteStorage.log('got profile', profile, 'and link', link);
+        if(link) {
+          var authURL = link.properties['auth-endpoint'] ||
+            link.properties['http://tools.ietf.org/html/rfc6749#section-4.2'];
+          cachedInfo[userAddress] = { href: link.href, type: link.type, authURL: authURL };
+          if(haveLocalStorage) {
+            localStorage[SETTINGS_KEY] = JSON.stringify({ cache: cachedInfo });
+          }
+          callback(link.href, link.type, authURL);
+        } else {
+          tryOne();
+        }
+      }
+      xhr.send();
+    }
+    tryOne();
+  },
+
+
+
+  RemoteStorage.Discover._rs_init = function(remoteStorage) {
+    if(haveLocalStorage) {
+      var settings;
+      try { settings = JSON.parse(localStorage[SETTINGS_KEY]); } catch(e) {};
+      if(settings) {
+        cachedInfo = settings.cache;
+      }
+    }
+  };
+
+  RemoteStorage.Discover._rs_supported = function() {
+    haveLocalStorage = !! global.localStorage;
+    haveXMLHttpRequest = !! global.XMLHttpRequest;
+    return haveXMLHttpRequest;
+  }
+
+  RemoteStorage.Discover._rs_cleanup = function() {
+    if(haveLocalStorage) {
+      delete localStorage[SETTINGS_KEY];
+    }
+  };
+
+})(this);
+
+
+/** FILE: src/authorize.js **/
+(function() {
+
+  function extractParams() {
+    //FF already decodes the URL fragment in document.location.hash, so use this instead:
+    if(! document.location.href) {//bit ugly way to fix unit tests
+      document.location.href = document.location.hash;
+    }
+    var hashPos = document.location.href.indexOf('#');
+    if(hashPos == -1) return;
+    var hash = document.location.href.substring(hashPos+1);
+    return hash.split('&').reduce(function(m, kvs) {
+      var kv = kvs.split('=');
+      m[decodeURIComponent(kv[0])] = decodeURIComponent(kv[1]);
+      return m;
+    }, {});
+  };
+
+  RemoteStorage.Authorize = function(authURL, storageApi, scopes, redirectUri) {
+    RemoteStorage.log('Authorize authURL = ',authURL)
+    var scope = [];
+    for(var key in scopes) {
+      var mode = scopes[key];
+      if(key == 'root') {
+        if(! storageApi.match(/^draft-dejong-remotestorage-/)) {
+          key = '';
+        }
+      }
+      scope.push(key + ':' + mode);
+    }
+    scope = scope.join(' ');
+
+    var clientId = redirectUri.match(/^(https?:\/\/[^\/]+)/)[0];
+
+    var url = authURL;
+    url += authURL.indexOf('?') > 0 ? '&' : '?';
+    url += 'redirect_uri=' + encodeURIComponent(redirectUri.replace(/#.*$/, ''));
+    url += '&scope=' + encodeURIComponent(scope);
+    url += '&client_id=' + encodeURIComponent(clientId);
+    url += '&response_type=token';
+    document.location = url;
+  };
+
+  RemoteStorage.ImpliedAuth = function(storageApi, redirectUri) {
+    RemoteStorage.log('ImpliedAuth proceeding due to absent authURL; storageApi = ' + storageApi + ' redirectUri = ' + redirectUri);
+    // Set a fixed access token, signalling to not send it as Bearer
+    remoteStorage.remote.configure(undefined, undefined, undefined, 'implied');
+    document.location = redirectUri;
+  };
+
+  RemoteStorage.prototype.authorize = function(authURL) {
+    RemoteStorage.Authorize(authURL, this.remote.storageApi, this.access.scopeModeMap, String(document.location));
+  };
+
+  RemoteStorage.prototype.impliedauth = function() {
+    RemoteStorage.ImpliedAuth(this.remote.storageApi, String(document.location));
+  };
+
+  RemoteStorage.Authorize._rs_init = function(remoteStorage) {
+    var params = extractParams();
+    if(params) {
+      document.location.hash = '';
+    }
+    remoteStorage.on('features-loaded', function() {
+      if(params) {
+        if(params.access_token) {
+          remoteStorage.remote.configure(undefined, undefined, undefined, params.access_token);
+        }
+        if(params.remotestorage) {
+          remoteStorage.connect(params.remotestorage);
+        }
+        if(params.error) {
+          throw "Authorization server errored: " + params.error;
+        }
+      }
+    });
+  }
+
+})();
+
+
+/** FILE: src/access.js **/
+(function(global) {
+
+  var haveLocalStorage = 'localStorage' in global;
+  var SETTINGS_KEY = "remotestorage:access";
+
+  /**
+   * Class: RemoteStorage.Access
+   *
+   * Keeps track of claimed access and scopes.
+   */
+  RemoteStorage.Access = function() {
+    this.reset();
+
+    if(haveLocalStorage) {
+      var rawSettings = localStorage[SETTINGS_KEY];
+      if(rawSettings) {
+        var savedSettings = JSON.parse(rawSettings);
+        for(var key in savedSettings) {
+          this.set(key, savedSettings[key]);
+        }
+      }
+    }
+  };
+
+  RemoteStorage.Access.prototype = {
+    // not sure yet, if 'set' or 'claim' is better...
+
+    /**
+     * Method: claim
+     *
+     * Claim access on a given scope with given mode.
+     *
+     * Parameters:
+     *   scope - An access scope, such as "contacts" or "calendar".
+     *   mode  - Access mode to use. Either "r" or "rw".
+     */
+    claim: function() {
+      this.set.apply(this, arguments);
+    },
+
+    set: function(scope, mode) {
+      this._adjustRootPaths(scope);
+      this.scopeModeMap[scope] = mode;
+      this._persist();
+    },
+
+    get: function(scope) {
+      return this.scopeModeMap[scope];
+    },
+
+    remove: function(scope) {
+      var savedMap = {};
+      for(var name in this.scopeModeMap) {
+        savedMap[name] = this.scopeModeMap[name];
+      }
+      this.reset();
+      delete savedMap[scope];
+      for(var name in savedMap) {
+        this.set(name, savedMap[name]);
+      }
+      this._persist();
+    },
+
+    check: function(scope, mode) {
+      var actualMode = this.get(scope);
+      return actualMode && (mode === 'r' || actualMode === 'rw');
+    },
+
+    reset: function() {
+      this.rootPaths = [];
+      this.scopeModeMap = {};
+    },
+
+    _adjustRootPaths: function(newScope) {
+      if('root' in this.scopeModeMap || newScope === 'root') {
+        this.rootPaths = ['/'];
+      } else if(! (newScope in this.scopeModeMap)) {
+        this.rootPaths.push('/' + newScope + '/');
+        this.rootPaths.push('/public/' + newScope + '/');
+      }
+    },
+
+    _persist: function() {
+      if(haveLocalStorage) {
+        localStorage[SETTINGS_KEY] = JSON.stringify(this.scopeModeMap);
+      }
+    },
+
+    setStorageType: function(type) {
+      this.storageType = type;
+    }
+  };
+
+  /**
+   * Property: scopes
+   *
+   * Holds an array of claimed scopes in the form
+   * > { name: "<scope-name>", mode: "<mode>" }
+   *
+   * Example:
+   *   (start code)
+   *   remoteStorage.access.claim('foo', 'r');
+   *   remoteStorage.access.claim('bar', 'rw');
+   *
+   *   remoteStorage.access.scopes
+   *   // -> [ { name: 'foo', mode: 'r' }, { name: 'bar', mode: 'rw' } ]
+   */
+  Object.defineProperty(RemoteStorage.Access.prototype, 'scopes', {
+    get: function() {
+      return Object.keys(this.scopeModeMap).map(function(key) {
+        return { name: key, mode: this.scopeModeMap[key] };
+      }.bind(this));
+    }
+  });
+
+  Object.defineProperty(RemoteStorage.Access.prototype, 'scopeParameter', {
+    get: function() {
+      return this.scopes.map(function(scope) {
+        return (scope.name === 'root' && this.storageType === '2012.04' ? '' : scope.name) + ':' + scope.mode;
+      }.bind(this)).join(' ');
+    }
+  });
+
+  // documented in src/remotestorage.js
+  Object.defineProperty(RemoteStorage.prototype, 'access', {
+    get: function() {
+      var access = new RemoteStorage.Access();
+      Object.defineProperty(this, 'access', {
+        value: access
+      });
+      return access;
+    },
+    configurable: true
+  });
+
+  function setModuleCaching(remoteStorage, key) {
+    if(key == 'root' || key === '') {
+      remoteStorage.caching.set('/', { data: true });
+    } else {
+      remoteStorage.caching.set('/' + key + '/', { data: true });
+      remoteStorage.caching.set('/public/' + key + '/', { data: true });
+    }
+  }
+
+  // documented in src/remotestorage.js
+  RemoteStorage.prototype.claimAccess = function(scopes) {
+    console.log("DEPRECATION WARNING: remoteStorage.claimAccess may mess with your caching control - if you use cache control directives, then see https://github.com/remotestorage/remotestorage.js/issues/380 and use remoteStorage.access.claim instead.");
+    if(typeof(scopes) === 'object') {
+      for(var key in scopes) {
+        this.access.claim(key, scopes[key]);
+        setModuleCaching(this, key); // legacy hack
+      }
+    } else {
+      this.access.claim(arguments[0], arguments[1])
+      setModuleCaching(this, arguments[0]); // legacy hack;
+    }
+  };
+
+  RemoteStorage.Access._rs_init = function() {};
+
+})(this);
+
+
+/** FILE: src/assets.js **/
+/** THIS FILE WAS GENERATED BY build/compile-assets.js. DO NOT CHANGE IT MANUALLY, BUT INSTEAD CHANGE THE ASSETS IN assets/. **/
+RemoteStorage.Assets = {
+
+  connectIcon: '',
+  disconnectIcon: '',
+  remoteStorageIcon: '',
+  remoteStorageIconError: '',
+  remoteStorageIconOffline: '',
+  syncIcon: '',
+  widget: ' <div class="rs-bubble rs-hidden">   <div class="rs-bubble-text remotestorage-initial remotestorage-error remotestorage-authing remotestorage-offline">     <span class="rs-status-text">       Connect <strong>remotestorage</strong>     </span>   </div>   <div class="rs-bubble-expandable">     <!-- error -->     <div class="remotestorage-error">       <pre class="rs-status-text rs-error-msg">ERROR</pre>          <button class="remotestorage-reset">get me out of here</button>     <p class="rs-centered-text"> If this problem persists, please <a href="http://remotestorage.io/community/" target="_blank">let us know</a>!</p>     </div>     <!-- connected -->     <div class="rs-bubble-text remotestorage-connected">       <strong class="userAddress"> User Name </strong>       <span class="remotestorage-unauthorized">         <br/>Unauthorized! Click to reconnect.<br/>       </span>     </div>     <div class="content remotestorage-connected">       <button class="rs-sync" title="sync">  <img>  </button>       <button class="rs-disconnect" title="disconnect">  <img>  </button>     </div>     <!-- initial -->     <form novalidate class="remotestorage-initial">       <input  type="email" placeholder="user@host" name="userAddress" novalidate>       <button class="connect" name="connect" title="connect" disabled="disabled">         <img>       </button>     </form>     <div class="rs-info-msg remotestorage-initial">       This app allows you to use your own storage! Find more info on       <a href="http://remotestorage.io/" target="_blank">remotestorage.io</a>     </div>      </div> </div> <img class="rs-cube rs-action">  ',
+  widgetCss: '/** encoding:utf-8 **/ /* RESET */ #remotestorage-widget{text-align:left;}#remotestorage-widget input, #remotestorage-widget button{font-size:11px;}#remotestorage-widget form input[type=email]{margin-bottom:0;/* HTML5 Boilerplate */}#remotestorage-widget form input[type=submit]{margin-top:0;/* HTML5 Boilerplate */}/* /RESET */ #remotestorage-widget, #remotestorage-widget *{-moz-box-sizing:border-box;box-sizing:border-box;}#remotestorage-widget{position:absolute;right:10px;top:10px;font:normal 16px/100% sans-serif !important;user-select:none;-webkit-user-select:none;-moz-user-select:-moz-none;cursor:default;z-index:10000;}#remotestorage-widget .rs-bubble{background:rgba(80, 80, 80, .7);border-radius:5px 15px 5px 5px;color:white;font-size:0.8em;padding:5px;position:absolute;right:3px;top:9px;min-height:24px;white-space:nowrap;text-decoration:none;}#remotestorage-widget .rs-bubble-text{padding-right:32px;/* make sure the bubble doesn\'t "jump" when initially opening. */ min-width:182px;}#remotestorage-widget .rs-action{cursor:pointer;}/* less obtrusive cube when connected */ #remotestorage-widget.remotestorage-state-connected .rs-cube, #remotestorage-widget.remotestorage-state-busy .rs-cube{opacity:.3;-webkit-transition:opacity .3s ease;-moz-transition:opacity .3s ease;-ms-transition:opacity .3s ease;-o-transition:opacity .3s ease;transition:opacity .3s ease;}#remotestorage-widget.remotestorage-state-connected:hover .rs-cube, #remotestorage-widget.remotestorage-state-busy:hover .rs-cube, #remotestorage-widget.remotestorage-state-connected .rs-bubble:not(.rs-hidden) + .rs-cube{opacity:1 !important;}#remotestorage-widget .rs-cube{position:relative;top:5px;right:0;}/* pulsing animation for cube when loading */ #remotestorage-widget .rs-cube.remotestorage-loading{-webkit-animation:remotestorage-loading .5s ease-in-out infinite alternate;-moz-animation:remotestorage-loading .5s ease-in-out infinite alternate;-o-animation:remotestorage-loading .5s ease-in-out infinite alternate;-ms-animation:remotestorage-loading .5s ease-in-out infinite alternate;animation:remotestorage-loading .5s ease-in-out infinite alternate;}@-webkit-keyframes remotestorage-loading{to{opacity:.7}}@-moz-keyframes remotestorage-loading{to{opacity:.7}}@-o-keyframes remotestorage-loading{to{opacity:.7}}@-ms-keyframes remotestorage-loading{to{opacity:.7}}@keyframes remotestorage-loading{to{opacity:.7}}#remotestorage-widget a{text-decoration:underline;color:inherit;}#remotestorage-widget form{margin-top:.7em;position:relative;}#remotestorage-widget form input{display:table-cell;vertical-align:top;border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:2em;}#remotestorage-widget form input:disabled{color:#999;background:#444 !important;cursor:default !important;}#remotestorage-widget form input[type=email]{background:#000;width:100%;height:26px;padding:0 30px 0 5px;border-top:1px solid #111;border-bottom:1px solid #999;}#remotestorage-widget button:focus, #remotestorage-widget input:focus{box-shadow:0 0 4px #ccc;}#remotestorage-widget form input[type=email]::-webkit-input-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]::-moz-placeholder{color:#999;}#remotestorage-widget form input[type=email]:-ms-input-placeholder{color:#999;}#remotestorage-widget form input[type=submit]{background:#000;cursor:pointer;padding:0 5px;}#remotestorage-widget form input[type=submit]:hover{background:#333;}#remotestorage-widget .rs-info-msg{font-size:10px;color:#eee;margin-top:0.7em;white-space:normal;}#remotestorage-widget .rs-info-msg.last-synced-message{display:inline;white-space:nowrap;margin-bottom:.7em}#remotestorage-widget .rs-info-msg a:hover, #remotestorage-widget .rs-info-msg a:active{color:#fff;}#remotestorage-widget button img{vertical-align:baseline;}#remotestorage-widget button{border:none;border-radius:6px;font-weight:bold;color:white;outline:none;line-height:1.5em;height:26px;width:26px;background:#000;cursor:pointer;margin:0;padding:5px;}#remotestorage-widget button:hover{background:#333;}#remotestorage-widget .rs-bubble button.connect{display:block;background:none;position:absolute;right:0;top:0;opacity:1;/* increase clickable area of connect button */ margin:-5px;padding:10px;width:36px;height:36px;}#remotestorage-widget .rs-bubble button.connect:not([disabled]):hover{background:rgba(150,150,150,.5);}#remotestorage-widget .rs-bubble button.connect[disabled]{opacity:.5;cursor:default !important;}#remotestorage-widget .rs-bubble button.rs-sync{position:relative;left:-5px;bottom:-5px;padding:4px 4px 0 4px;background:#555;}#remotestorage-widget .rs-bubble button.rs-sync:hover{background:#444;}#remotestorage-widget .rs-bubble button.rs-disconnect{background:#721;position:absolute;right:0;bottom:0;padding:4px 4px 0 4px;}#remotestorage-widget .rs-bubble button.rs-disconnect:hover{background:#921;}#remotestorage-widget .remotestorage-error-info{color:#f92;}#remotestorage-widget .remotestorage-reset{width:100%;background:#721;}#remotestorage-widget .remotestorage-reset:hover{background:#921;}#remotestorage-widget .rs-bubble .content{margin-top:7px;}#remotestorage-widget pre{user-select:initial;-webkit-user-select:initial;-moz-user-select:text;max-width:27em;margin-top:1em;overflow:auto;}#remotestorage-widget .rs-centered-text{text-align:center;}#remotestorage-widget .rs-bubble.rs-hidden{padding-bottom:2px;border-radius:5px 15px 15px 5px;}#remotestorage-widget .rs-error-msg{min-height:5em;}.rs-bubble.rs-hidden .rs-bubble-expandable{display:none;}.remotestorage-state-connected .rs-bubble.rs-hidden{display:none;}.remotestorage-connected{display:none;}.remotestorage-state-connected .remotestorage-connected{display:block;}.remotestorage-initial{display:none;}.remotestorage-state-initial .remotestorage-initial{display:block;}.remotestorage-error{display:none;}.remotestorage-state-error .remotestorage-error{display:block;}.remotestorage-state-authing .remotestorage-authing{display:block;}.remotestorage-state-offline .remotestorage-connected, .remotestorage-state-offline .remotestorage-offline{display:block;}.remotestorage-unauthorized{display:none;}.remotestorage-state-unauthorized .rs-bubble.rs-hidden{display:none;}.remotestorage-state-unauthorized .remotestorage-connected, .remotestorage-state-unauthorized .remotestorage-unauthorized{display:block;}.remotestorage-state-unauthorized .rs-sync{display:none;}.remotestorage-state-busy .rs-bubble{display:none;}.remotestorage-state-authing .rs-bubble-expandable{display:none;}'
+};
+
+
+/** FILE: src/widget.js **/
+(function(window) {
+
+  var haveLocalStorage;
+  var LS_STATE_KEY = "remotestorage:widget:state";
+  // states allowed to immediately jump into after a reload.
+  var VALID_ENTRY_STATES = {
+    initial: true, connected: true, offline: true
+  };
+
+  function stateSetter(widget, state) {
+    return function() {
+      if(haveLocalStorage) {
+        localStorage[LS_STATE_KEY] = state;
+      }
+      if(widget.view) {
+        if(widget.rs.remote) {
+          widget.view.setUserAddress(widget.rs.remote.userAddress);
+        }
+        widget.view.setState(state, arguments);
+      } else {
+        widget._rememberedState = state;
+      }
+    };
+  }
+  function errorsHandler(widget){
+    //decided to not store error state
+    return function(error){
+      if(error instanceof RemoteStorage.DiscoveryError) {
+        console.error('discovery failed',  error, '"' + error.message + '"');
+        widget.view.setState('initial', [error.message]);
+      } else if(error instanceof RemoteStorage.SyncError) {
+        widget.view.setState('offline', []);
+      } else if(error instanceof RemoteStorage.Unauthorized){
+        widget.view.setState('unauthorized')
+      } else {
+        widget.view.setState('error', [error]);
+      }
+    }
+  }
+  RemoteStorage.Widget = function(remoteStorage) {
+
+    // setting event listeners on rs events to put
+    // the widget into corresponding states
+    this.rs = remoteStorage;
+    this.rs.on('ready', stateSetter(this, 'connected'));
+    this.rs.on('disconnected', stateSetter(this, 'initial'));
+    this.rs.on('authing', stateSetter(this, 'authing'));
+    this.rs.on('sync-busy', stateSetter(this, 'busy'));
+    this.rs.on('sync-done', stateSetter(this, 'connected'));
+    this.rs.on('error', errorsHandler(this) );
+    if(haveLocalStorage) {
+      var state = localStorage[LS_STATE_KEY];
+      if(state && VALID_ENTRY_STATES[state]) {
+        this._rememberedState = state;
+
+        if(state == 'connected' && ! remoteStorage.connected) {
+          this._rememberedState = 'initial';
+        }
+      }
+    }
+  };
+
+  RemoteStorage.Widget.prototype = {
+    // Methods :
+    //   display(domID)
+    //     displays the widget via the view.display method
+    //    returns: this
+    //
+    //   setView(view)
+    //     sets the view and initializes event listeners to
+    //     react on widget events
+    //
+
+    display: function(domID) {
+      if(! this.view) {
+        this.setView(new RemoteStorage.Widget.View(domID));
+      }
+      this.view.display.apply(this.view, arguments);
+      return this;
+    },
+
+    setView: function(view) {
+      this.view = view;
+      this.view.on('connect', this.rs.connect.bind(this.rs));
+      this.view.on('disconnect', this.rs.disconnect.bind(this.rs));
+      if(this.rs.sync) {
+        this.view.on('sync', this.rs.sync.bind(this.rs));
+      }
+      try {
+        this.view.on('reset', function(){
+          this.rs.on('disconnected', document.location.reload.bind(document.location))
+          this.rs.disconnect()
+        }.bind(this));
+      } catch(e) {
+        if(e.message && e.message.match(/Unknown event/)) {
+          // ignored. (the 0.7 widget-view interface didn't have a 'reset' event)
+        } else {
+          throw e;
+        }
+      }
+
+      if(this._rememberedState) {
+        setTimeout(stateSetter(this, this._rememberedState), 0);
+        delete this._rememberedState;
+      }
+    }
+  };
+
+  RemoteStorage.prototype.displayWidget = function(domID) {
+    this.widget.display(domID);
+  };
+
+  RemoteStorage.Widget._rs_init = function(remoteStorage) {
+    if(! remoteStorage.widget) {
+      remoteStorage.widget = new RemoteStorage.Widget(remoteStorage);
+    }
+  };
+
+  RemoteStorage.Widget._rs_supported = function(remoteStorage) {
+    haveLocalStorage = 'localStorage' in window;
+    return true;
+  };
+
+})(this);
+
+
+/** FILE: src/view.js **/
+(function(window){
+
+
+  //
+  // helper methods
+  //
+  var cEl = document.createElement.bind(document);
+  function gCl(parent, className) {
+    return parent.getElementsByClassName(className)[0];
+  }
+  function gTl(parent, className) {
+    return parent.getElementsByTagName(className)[0];
+  }
+
+  function removeClass(el, className) {
+    return el.classList.remove(className);
+  }
+
+  function addClass(el, className) {
+    return el.classList.add(className);
+  }
+
+  function stop_propagation(event) {
+    if(typeof(event.stopPropagation) == 'function') {
+      event.stopPropagation();
+    } else {
+      event.cancelBubble = true;
+    }
+  }
+
+
+  RemoteStorage.Widget.View = function() {
+    if(typeof(document) === 'undefined') {
+      throw "Widget not supported";
+    }
+    RemoteStorage.eventHandling(this,
+                                'connect',
+                                'disconnect',
+                                'sync',
+                                'display',
+                                'reset');
+
+    // re-binding the event so they can be called from the window
+    for(var event in this.events){
+      this.events[event] = this.events[event].bind(this);
+    }
+
+
+    // bubble toggling stuff
+    this.toggle_bubble = function(event) {
+      if(this.bubble.className.search('rs-hidden') < 0) {
+        this.hide_bubble(event);
+      } else {
+        this.show_bubble(event);
+      }
+    }.bind(this);
+
+    this.hide_bubble = function(){
+      //console.log('hide bubble',this);
+      addClass(this.bubble, 'rs-hidden')
+      document.body.removeEventListener('click', hide_bubble_on_body_click);
+    }.bind(this);
+
+    var hide_bubble_on_body_click = function (event) {
+      for(var p = event.target; p != document.body; p = p.parentElement) {
+        if(p.id == 'remotestorage-widget') {
+          return;
+        }
+      }
+      this.hide_bubble();
+    }.bind(this);
+
+    this.show_bubble = function(event){
+      //console.log('show bubble',this.bubble,event)
+      removeClass(this.bubble, 'rs-hidden');
+      if(typeof(event) != 'undefined') {
+         stop_propagation(event);
+       }
+      document.body.addEventListener('click', hide_bubble_on_body_click);
+      gTl(this.bubble,'form').userAddress.focus();
+    }.bind(this);
+
+
+    this.display = function(domID) {
+
+      if(typeof(this.div) !== 'undefined')
+        return this.div;
+
+      var element = cEl('div');
+      var style = cEl('style');
+      style.innerHTML = RemoteStorage.Assets.widgetCss;
+
+      element.id = "remotestorage-widget";
+
+      element.innerHTML = RemoteStorage.Assets.widget;
+
+
+      element.appendChild(style);
+      if(domID) {
+        var parent = document.getElementById(domID);
+        if(! parent) {
+          throw "Failed to find target DOM element with id=\"" + domID + "\"";
+        }
+        parent.appendChild(element);
+      } else {
+        document.body.appendChild(element);
+      }
+
+      var el;
+      //sync button
+      el = gCl(element, 'rs-sync');
+      gTl(el, 'img').src = RemoteStorage.Assets.syncIcon;
+      el.addEventListener('click', this.events.sync);
+
+      //disconnect button
+      el = gCl(element, 'rs-disconnect');
+      gTl(el, 'img').src = RemoteStorage.Assets.disconnectIcon;
+      el.addEventListener('click', this.events.disconnect);
+
+
+      //get me out of here
+      var el = gCl(element, 'remotestorage-reset').addEventListener('click', this.events.reset);
+      //connect button
+      var cb = gCl(element,'connect');
+      gTl(cb, 'img').src = RemoteStorage.Assets.connectIcon;
+      cb.addEventListener('click', this.events.connect);
+
+
+      // input
+      this.form = gTl(element, 'form')
+      el = this.form.userAddress;
+      el.addEventListener('keyup', function(event) {
+        if(event.target.value) cb.removeAttribute('disabled');
+        else cb.setAttribute('disabled','disabled');
+      });
+      if(this.userAddress) {
+        el.value = this.userAddress;
+      }
+
+      //the cube
+      el = gCl(element, 'rs-cube');
+      el.src = RemoteStorage.Assets.remoteStorageIcon;
+      el.addEventListener('click', this.toggle_bubble);
+      this.cube = el
+
+      //the bubble
+      this.bubble = gCl(element,'rs-bubble');
+      // what is the meaning of this hiding the b
+      var bubbleDontCatch = { INPUT: true, BUTTON: true, IMG: true };
+      this.bubble.addEventListener('click', function(event) {
+        if(! bubbleDontCatch[event.target.tagName] && ! (this.div.classList.contains('remotestorage-state-unauthorized') )) {
+
+          this.show_bubble(event);
+        };
+      }.bind(this))
+      this.hide_bubble();
+
+      this.div = element;
+
+      this.states.initial.call(this);
+      this.events.display.call(this);
+      return this.div;
+    };
+
+  }
+
+  RemoteStorage.Widget.View.prototype = {
+
+    // Methods:
+    //
+    //  display(domID)
+    //    draws the widget inside of the dom element with the id domID
+    //   returns: the widget div
+    //
+    //  showBubble()
+    //    shows the bubble
+    //  hideBubble()
+    //    hides the bubble
+    //  toggleBubble()
+    //    shows the bubble when hidden and the other way around
+    //
+    //  setState(state, args)
+    //    calls states[state]
+    //    args are the arguments for the
+    //    state(errors mostly)
+    //
+    // setUserAddres
+    //    set userAddress of the input field
+    //
+    // States:
+    //  initial      - not connected
+    //  authing      - in auth flow
+    //  connected    - connected to remote storage, not syncing at the moment
+    //  busy         - connected, syncing at the moment
+    //  offline      - connected, but no network connectivity
+    //  error        - connected, but sync error happened
+    //  unauthorized - connected, but request returned 401
+    //
+    // Events:
+    // connect    : fired when the connect button is clicked
+    // sync       : fired when the sync button is clicked
+    // disconnect : fired when the disconnect button is clicked
+    // reset      : fired after crash triggers disconnect
+    // display    : fired when finished displaying the widget
+    setState : function(state, args) {
+      RemoteStorage.log('widget.view.setState(',state,',',args,');');
+      var s = this.states[state];
+      if(typeof(s) === 'undefined') {
+        throw new Error("Bad State assigned to view: " + state);
+      }
+      s.apply(this,args);
+    },
+    setUserAddress : function(addr) {
+      this.userAddress = addr || '';
+
+      var el;
+      if(this.div && (el = gTl(this.div, 'form').userAddress)) {
+        el.value = this.userAddress;
+      }
+    },
+
+    states :  {
+      initial : function(message) {
+        var cube = this.cube;
+        var info = message || 'This app allows you to use your own storage! Find more info on <a href="http://remotestorage.io/" target="_blank">remotestorage.io';
+        if(message) {
+          cube.src = RemoteStorage.Assets.remoteStorageIconError;
+          removeClass(this.cube, 'remotestorage-loading');
+          this.show_bubble();
+          setTimeout(function(){
+            cube.src = RemoteStorage.Assets.remoteStorageIcon;
+          },5000)//show the red error cube for 5 seconds, then show the normal orange one again
+        } else {
+          this.hide_bubble();
+        }
+        this.div.className = "remotestorage-state-initial";
+        gCl(this.div, 'rs-status-text').innerHTML = "Connect <strong>remotestorage</strong>";
+
+        //if address not empty connect button enabled
+        //TODO check if this works
+        var cb = gCl(this.div, 'connect')
+        if(cb.value)
+          cb.removeAttribute('disabled');
+
+        var infoEl = gCl(this.div, 'rs-info-msg');
+        infoEl.innerHTML = info;
+
+        if(message) {
+          infoEl.classList.add('remotestorage-error-info');
+        } else {
+          infoEl.classList.remove('remotestorage-error-info');
+        }
+
+      },
+      authing : function() {
+        this.div.removeEventListener('click', this.events.connect);
+        this.div.className = "remotestorage-state-authing";
+        gCl(this.div, 'rs-status-text').innerHTML = "Connecting <strong>"+this.userAddress+"</strong>";
+        addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone, when is that neccesary
+      },
+      connected : function() {
+        this.div.className = "remotestorage-state-connected";
+        gCl(this.div, 'userAddress').innerHTML = this.userAddress;
+        this.cube.src = RemoteStorage.Assets.remoteStorageIcon;
+        removeClass(this.cube, 'remotestorage-loading');
+      },
+      busy : function() {
+        this.div.className = "remotestorage-state-busy";
+        addClass(this.cube, 'remotestorage-loading'); //TODO needs to be undone when is that neccesary
+        this.hide_bubble();
+      },
+      offline : function() {
+        this.div.className = "remotestorage-state-offline";
+        this.cube.src = RemoteStorage.Assets.remoteStorageIconOffline;
+        gCl(this.div, 'rs-status-text').innerHTML = 'Offline';
+      },
+      error : function(err) {
+        var errorMsg = err;
+        this.div.className = "remotestorage-state-error";
+
+        gCl(this.div, 'rs-bubble-text').innerHTML = '<strong> Sorry! An error occured.</strong>'
+        if(err instanceof Error /*|| err instanceof DOMError*/) { //I don't know what an DOMError is and my browser doesn't know too(how to handle this?)
+          errorMsg = err.message + '\n\n' +
+            err.stack;
+        }
+        gCl(this.div, 'rs-error-msg').textContent = errorMsg;
+        this.cube.src = RemoteStorage.Assets.remoteStorageIconError;
+        this.show_bubble();
+      },
+      unauthorized : function() {
+        this.div.className = "remotestorage-state-unauthorized";
+        this.cube.src = RemoteStorage.Assets.remoteStorageIconError;
+        this.show_bubble();
+        this.div.addEventListener('click', this.events.connect);
+      }
+    },
+
+    events : {
+      connect : function(event) {
+        stop_propagation(event);
+        event.preventDefault();
+        this._emit('connect', gTl(this.div, 'form').userAddress.value);
+      },
+      sync : function(event) {
+        stop_propagation(event);
+        event.preventDefault();
+
+        this._emit('sync');
+      },
+      disconnect : function(event) {
+        stop_propagation(event);
+        event.preventDefault();
+        this._emit('disconnect');
+      },
+      reset : function(event){
+        event.preventDefault();
+        var result = window.confirm("Are you sure you want to reset everything? That will probably make the error go away, but also clear your entire localStorage and reload the page. Please make sure you know what you are doing, before clicking 'yes' :-)");
+        if(result){
+          this._emit('reset');
+        }
+      },
+      display : function(event) {
+        if(event)
+          event.preventDefault();
+        this._emit('display');
+      }
+    }
+  };
+})(this);
+
+
+/** FILE: lib/tv4.js **/
+/**
+Author: Geraint Luff and others
+Year: 2013
+
+This code is released into the "public domain" by its author(s).  Anybody may use, alter and distribute the code without restriction.  The author makes no guarantees, and takes no liability of any kind for use of this code.
+
+If you find a bug or make an improvement, it would be courteous to let the author know, but it is not compulsory.
+**/
+
+(function (global) {
+var ValidatorContext = function (parent, collectMultiple) {
+       this.missing = [];
+       this.schemas = parent ? Object.create(parent.schemas) : {};
+       this.collectMultiple = collectMultiple;
+       this.errors = [];
+       this.handleError = collectMultiple ? this.collectError : this.returnError;
+};
+ValidatorContext.prototype.returnError = function (error) {
+       return error;
+};
+ValidatorContext.prototype.collectError = function (error) {
+       if (error) {
+               this.errors.push(error);
+       }
+       return null;
+}
+ValidatorContext.prototype.prefixErrors = function (startIndex, dataPath, schemaPath) {
+       for (var i = startIndex; i < this.errors.length; i++) {
+               this.errors[i] = this.errors[i].prefixWith(dataPath, schemaPath);
+       }
+       return this;
+}
+
+ValidatorContext.prototype.getSchema = function (url) {
+       if (this.schemas[url] != undefined) {
+               var schema = this.schemas[url];
+               return schema;
+       }
+       var baseUrl = url;
+       var fragment = "";
+       if (url.indexOf('#') != -1) {
+               fragment = url.substring(url.indexOf("#") + 1);
+               baseUrl = url.substring(0, url.indexOf("#"));
+       }
+       if (this.schemas[baseUrl] != undefined) {
+               var schema = this.schemas[baseUrl];
+               var pointerPath = decodeURIComponent(fragment);
+               if (pointerPath == "") {
+                       return schema;
+               } else if (pointerPath.charAt(0) != "/") {
+                       return undefined;
+               }
+               var parts = pointerPath.split("/").slice(1);
+               for (var i = 0; i < parts.length; i++) {
+                       var component = parts[i].replace("~1", "/").replace("~0", "~");
+                       if (schema[component] == undefined) {
+                               schema = undefined;
+                               break;
+                       }
+                       schema = schema[component];
+               }
+               if (schema != undefined) {
+                       return schema;
+               }
+       }
+       if (this.missing[baseUrl] == undefined) {
+               this.missing.push(baseUrl);
+               this.missing[baseUrl] = baseUrl;
+       }
+};
+ValidatorContext.prototype.addSchema = function (url, schema) {
+       var map = {};
+       map[url] = schema;
+       normSchema(schema, url);
+       searchForTrustedSchemas(map, schema, url);
+       for (var key in map) {
+               this.schemas[key] = map[key];
+       }
+       return map;
+};
+       
+ValidatorContext.prototype.validateAll = function validateAll(data, schema, dataPathParts, schemaPathParts) {
+       if (schema['$ref'] != undefined) {
+               schema = this.getSchema(schema['$ref']);
+               if (!schema) {
+                       return null;
+               }
+       }
+       
+       var errorCount = this.errors.length;
+       var error = this.validateBasic(data, schema)
+               || this.validateNumeric(data, schema)
+               || this.validateString(data, schema)
+               || this.validateArray(data, schema)
+               || this.validateObject(data, schema)
+               || this.validateCombinations(data, schema)
+               || null
+       if (error || errorCount != this.errors.length) {
+               while ((dataPathParts && dataPathParts.length) || (schemaPathParts && schemaPathParts.length)) {
+                       var dataPart = (dataPathParts && dataPathParts.length) ? "" + dataPathParts.pop() : null;
+                       var schemaPart = (schemaPathParts && schemaPathParts.length) ? "" + schemaPathParts.pop() : null;
+                       if (error) {
+                               error = error.prefixWith(dataPart, schemaPart);
+                       }
+                       this.prefixErrors(errorCount, dataPart, schemaPart);
+               }
+       }
+               
+       return this.handleError(error);
+}
+
+function recursiveCompare(A, B) {
+       if (A === B) {
+               return true;
+       }
+       if (typeof A == "object" && typeof B == "object") {
+               if (Array.isArray(A) != Array.isArray(B)) {
+                       return false;
+               } else if (Array.isArray(A)) {
+                       if (A.length != B.length) {
+                               return false
+                       }
+                       for (var i = 0; i < A.length; i++) {
+                               if (!recursiveCompare(A[i], B[i])) {
+                                       return false;
+                               }
+                       }
+               } else {
+                       for (var key in A) {
+                               if (B[key] === undefined && A[key] !== undefined) {
+                                       return false;
+                               }
+                       }
+                       for (var key in B) {
+                               if (A[key] === undefined && B[key] !== undefined) {
+                                       return false;
+                               }
+                       }
+                       for (var key in A) {
+                               if (!recursiveCompare(A[key], B[key])) {
+                                       return false;
+                               }
+                       }
+               }
+               return true;
+       }
+       return false;
+}
+
+ValidatorContext.prototype.validateBasic = function validateBasic(data, schema) {
+       var error;
+       if (error = this.validateType(data, schema)) {
+               return error.prefixWith(null, "type");
+       }
+       if (error = this.validateEnum(data, schema)) {
+               return error.prefixWith(null, "type");
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateType = function validateType(data, schema) {
+       if (schema.type == undefined) {
+               return null;
+       }
+       var dataType = typeof data;
+       if (data == null) {
+               dataType = "null";
+       } else if (Array.isArray(data)) {
+               dataType = "array";
+       }
+       var allowedTypes = schema.type;
+       if (typeof allowedTypes != "object") {
+               allowedTypes = [allowedTypes];
+       }
+       
+       for (var i = 0; i < allowedTypes.length; i++) {
+               var type = allowedTypes[i];
+               if (type == dataType || (type == "integer" && dataType == "number" && (data%1 == 0))) {
+                       return null;
+               }
+       }
+       return new ValidationError(ErrorCodes.INVALID_TYPE, "invalid data type: " + dataType);
+}
+
+ValidatorContext.prototype.validateEnum = function validateEnum(data, schema) {
+       if (schema["enum"] == undefined) {
+               return null;
+       }
+       for (var i = 0; i < schema["enum"].length; i++) {
+               var enumVal = schema["enum"][i];
+               if (recursiveCompare(data, enumVal)) {
+                       return null;
+               }
+       }
+       return new ValidationError(ErrorCodes.ENUM_MISMATCH, "No enum match for: " + JSON.stringify(data));
+}
+ValidatorContext.prototype.validateNumeric = function validateNumeric(data, schema) {
+       return this.validateMultipleOf(data, schema)
+               || this.validateMinMax(data, schema)
+               || null;
+}
+
+ValidatorContext.prototype.validateMultipleOf = function validateMultipleOf(data, schema) {
+       var multipleOf = schema.multipleOf || schema.divisibleBy;
+       if (multipleOf == undefined) {
+               return null;
+       }
+       if (typeof data == "number") {
+               if (data%multipleOf != 0) {
+                       return new ValidationError(ErrorCodes.NUMBER_MULTIPLE_OF, "Value " + data + " is not a multiple of " + multipleOf);
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateMinMax = function validateMinMax(data, schema) {
+       if (typeof data != "number") {
+               return null;
+       }
+       if (schema.minimum != undefined) {
+               if (data < schema.minimum) {
+                       return new ValidationError(ErrorCodes.NUMBER_MINIMUM, "Value " + data + " is less than minimum " + schema.minimum).prefixWith(null, "minimum");
+               }
+               if (schema.exclusiveMinimum && data == schema.minimum) {
+                       return new ValidationError(ErrorCodes.NUMBER_MINIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive minimum " + schema.minimum).prefixWith(null, "exclusiveMinimum");
+               }
+       }
+       if (schema.maximum != undefined) {
+               if (data > schema.maximum) {
+                       return new ValidationError(ErrorCodes.NUMBER_MAXIMUM, "Value " + data + " is greater than maximum " + schema.maximum).prefixWith(null, "maximum");
+               }
+               if (schema.exclusiveMaximum && data == schema.maximum) {
+                       return new ValidationError(ErrorCodes.NUMBER_MAXIMUM_EXCLUSIVE, "Value "+ data + " is equal to exclusive maximum " + schema.maximum).prefixWith(null, "exclusiveMaximum");
+               }
+       }
+       return null;
+}
+ValidatorContext.prototype.validateString = function validateString(data, schema) {
+       return this.validateStringLength(data, schema)
+               || this.validateStringPattern(data, schema)
+               || null;
+}
+
+ValidatorContext.prototype.validateStringLength = function validateStringLength(data, schema) {
+       if (typeof data != "string") {
+               return null;
+       }
+       if (schema.minLength != undefined) {
+               if (data.length < schema.minLength) {
+                       return new ValidationError(ErrorCodes.STRING_LENGTH_SHORT, "String is too short (" + data.length + " chars), minimum " + schema.minLength).prefixWith(null, "minLength");
+               }
+       }
+       if (schema.maxLength != undefined) {
+               if (data.length > schema.maxLength) {
+                       return new ValidationError(ErrorCodes.STRING_LENGTH_LONG, "String is too long (" + data.length + " chars), maximum " + schema.maxLength).prefixWith(null, "maxLength");
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateStringPattern = function validateStringPattern(data, schema) {
+       if (typeof data != "string" || schema.pattern == undefined) {
+               return null;
+       }
+       var regexp = new RegExp(schema.pattern);
+       if (!regexp.test(data)) {
+               return new ValidationError(ErrorCodes.STRING_PATTERN, "String does not match pattern").prefixWith(null, "pattern");
+       }
+       return null;
+}
+ValidatorContext.prototype.validateArray = function validateArray(data, schema) {
+       if (!Array.isArray(data)) {
+               return null;
+       }
+       return this.validateArrayLength(data, schema)
+               || this.validateArrayUniqueItems(data, schema)
+               || this.validateArrayItems(data, schema)
+               || null;
+}
+
+ValidatorContext.prototype.validateArrayLength = function validateArrayLength(data, schema) {
+       if (schema.minItems != undefined) {
+               if (data.length < schema.minItems) {
+                       var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_SHORT, "Array is too short (" + data.length + "), minimum " + schema.minItems)).prefixWith(null, "minItems");
+                       if (this.handleError(error)) {
+                               return error;
+                       }
+               }
+       }
+       if (schema.maxItems != undefined) {
+               if (data.length > schema.maxItems) {
+                       var error = (new ValidationError(ErrorCodes.ARRAY_LENGTH_LONG, "Array is too long (" + data.length + " chars), maximum " + schema.maxItems)).prefixWith(null, "maxItems");
+                       if (this.handleError(error)) {
+                               return error;
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateArrayUniqueItems = function validateArrayUniqueItems(data, schema) {
+       if (schema.uniqueItems) {
+               for (var i = 0; i < data.length; i++) {
+                       for (var j = i + 1; j < data.length; j++) {
+                               if (recursiveCompare(data[i], data[j])) {
+                                       var error = (new ValidationError(ErrorCodes.ARRAY_UNIQUE, "Array items are not unique (indices " + i + " and " + j + ")")).prefixWith(null, "uniqueItems");
+                                       if (this.handleError(error)) {
+                                               return error;
+                                       }
+                               }
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateArrayItems = function validateArrayItems(data, schema) {
+       if (schema.items == undefined) {
+               return null;
+       }
+       var error;
+       if (Array.isArray(schema.items)) {
+               for (var i = 0; i < data.length; i++) {
+                       if (i < schema.items.length) {
+                               if (error = this.validateAll(data[i], schema.items[i], [i], ["items", i])) {
+                                       return error;
+                               }
+                       } else if (schema.additionalItems != undefined) {
+                               if (typeof schema.additionalItems == "boolean") {
+                                       if (!schema.additionalItems) {
+                                               error = (new ValidationError(ErrorCodes.ARRAY_ADDITIONAL_ITEMS, "Additional items not allowed")).prefixWith("" + i, "additionalItems");
+                                               if (this.handleError(error)) {
+                                                       return error;
+                                               }
+                                       }
+                               } else if (error = this.validateAll(data[i], schema.additionalItems, [i], ["additionalItems"])) {
+                                       return error;
+                               }
+                       }
+               }
+       } else {
+               for (var i = 0; i < data.length; i++) {
+                       if (error = this.validateAll(data[i], schema.items, [i], ["items"])) {
+                               return error;
+                       }
+               }
+       }
+       return null;
+}
+ValidatorContext.prototype.validateObject = function validateObject(data, schema) {
+       if (typeof data != "object" || data == null || Array.isArray(data)) {
+               return null;
+       }
+       return this.validateObjectMinMaxProperties(data, schema)
+               || this.validateObjectRequiredProperties(data, schema)
+               || this.validateObjectProperties(data, schema)
+               || this.validateObjectDependencies(data, schema)
+               || null;
+}
+
+ValidatorContext.prototype.validateObjectMinMaxProperties = function validateObjectMinMaxProperties(data, schema) {
+       var keys = Object.keys(data);
+       if (schema.minProperties != undefined) {
+               if (keys.length < schema.minProperties) {
+                       var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MINIMUM, "Too few properties defined (" + keys.length + "), minimum " + schema.minProperties).prefixWith(null, "minProperties");
+                       if (this.handleError(error)) {
+                               return error;
+                       }
+               }
+       }
+       if (schema.maxProperties != undefined) {
+               if (keys.length > schema.maxProperties) {
+                       var error = new ValidationError(ErrorCodes.OBJECT_PROPERTIES_MAXIMUM, "Too many properties defined (" + keys.length + "), maximum " + schema.maxProperties).prefixWith(null, "maxProperties");
+                       if (this.handleError(error)) {
+                               return error;
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateObjectRequiredProperties = function validateObjectRequiredProperties(data, schema) {
+       if (schema.required != undefined) {
+               for (var i = 0; i < schema.required.length; i++) {
+                       var key = schema.required[i];
+                       if (data[key] === undefined) {
+                               var error = new ValidationError(ErrorCodes.OBJECT_REQUIRED, "Missing required property: " + key).prefixWith(null, "" + i).prefixWith(null, "required");
+                               if (this.handleError(error)) {
+                                       return error;
+                               }
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateObjectProperties = function validateObjectProperties(data, schema) {
+       var error;
+       for (var key in data) {
+               var foundMatch = false;
+               if (schema.properties != undefined && schema.properties[key] != undefined) {
+                       foundMatch = true;
+                       if (error = this.validateAll(data[key], schema.properties[key], [key], ["properties", key])) {
+                               return error;
+                       }
+               }
+               if (schema.patternProperties != undefined) {
+                       for (var patternKey in schema.patternProperties) {
+                               var regexp = new RegExp(patternKey);
+                               if (regexp.test(key)) {
+                                       foundMatch = true;
+                                       if (error = this.validateAll(data[key], schema.patternProperties[patternKey], [key], ["patternProperties", patternKey])) {
+                                               return error;
+                                       }
+                               }
+                       }
+               }
+               if (!foundMatch && schema.additionalProperties != undefined) {
+                       if (typeof schema.additionalProperties == "boolean") {
+                               if (!schema.additionalProperties) {
+                                       error = new ValidationError(ErrorCodes.OBJECT_ADDITIONAL_PROPERTIES, "Additional properties not allowed").prefixWith(key, "additionalProperties");
+                                       if (this.handleError(error)) {
+                                               return error;
+                                       }
+                               }
+                       } else {
+                               if (error = this.validateAll(data[key], schema.additionalProperties, [key], ["additionalProperties"])) {
+                                       return error;
+                               }
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateObjectDependencies = function validateObjectDependencies(data, schema) {
+       var error;
+       if (schema.dependencies != undefined) {
+               for (var depKey in schema.dependencies) {
+                       if (data[depKey] !== undefined) {
+                               var dep = schema.dependencies[depKey];
+                               if (typeof dep == "string") {
+                                       if (data[dep] === undefined) {
+                                               error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + dep).prefixWith(null, depKey).prefixWith(null, "dependencies");
+                                               if (this.handleError(error)) {
+                                                       return error;
+                                               }
+                                       }
+                               } else if (Array.isArray(dep)) {
+                                       for (var i = 0; i < dep.length; i++) {
+                                               var requiredKey = dep[i];
+                                               if (data[requiredKey] === undefined) {
+                                                       error = new ValidationError(ErrorCodes.OBJECT_DEPENDENCY_KEY, "Dependency failed - key must exist: " + requiredKey).prefixWith(null, "" + i).prefixWith(null, depKey).prefixWith(null, "dependencies");
+                                                       if (this.handleError(error)) {
+                                                               return error;
+                                                       }
+                                               }
+                                       }
+                               } else {
+                                       if (error = this.validateAll(data, dep, [], ["dependencies", depKey])) {
+                                               return error;
+                                       }
+                               }
+                       }
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateCombinations = function validateCombinations(data, schema) {
+       var error;
+       return this.validateAllOf(data, schema)
+               || this.validateAnyOf(data, schema)
+               || this.validateOneOf(data, schema)
+               || this.validateNot(data, schema)
+               || null;
+}
+
+ValidatorContext.prototype.validateAllOf = function validateAllOf(data, schema) {
+       if (schema.allOf == undefined) {
+               return null;
+       }
+       var error;
+       for (var i = 0; i < schema.allOf.length; i++) {
+               var subSchema = schema.allOf[i];
+               if (error = this.validateAll(data, subSchema, [], ["allOf", i])) {
+                       return error;
+               }
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateAnyOf = function validateAnyOf(data, schema) {
+       if (schema.anyOf == undefined) {
+               return null;
+       }
+       var errors = [];
+       var startErrorCount = this.errors.length;
+       for (var i = 0; i < schema.anyOf.length; i++) {
+               var subSchema = schema.anyOf[i];
+
+               var errorCount = this.errors.length;
+               var error = this.validateAll(data, subSchema, [], ["anyOf", i]);
+
+               if (error == null && errorCount == this.errors.length) {
+                       this.errors = this.errors.slice(0, startErrorCount);
+                       return null;
+               }
+               if (error) {
+                       errors.push(error.prefixWith(null, "" + i).prefixWith(null, "anyOf"));
+               }
+       }
+       errors = errors.concat(this.errors.slice(startErrorCount));
+       this.errors = this.errors.slice(0, startErrorCount);
+       return new ValidationError(ErrorCodes.ANY_OF_MISSING, "Data does not match any schemas from \"anyOf\"", "", "/anyOf", errors);
+}
+
+ValidatorContext.prototype.validateOneOf = function validateOneOf(data, schema) {
+       if (schema.oneOf == undefined) {
+               return null;
+       }
+       var validIndex = null;
+       var errors = [];
+       var startErrorCount = this.errors.length;
+       for (var i = 0; i < schema.oneOf.length; i++) {
+               var subSchema = schema.oneOf[i];
+               
+               var errorCount = this.errors.length;
+               var error = this.validateAll(data, subSchema, [], ["oneOf", i]);
+               
+               if (error == null && errorCount == this.errors.length) {
+                       if (validIndex == null) {
+                               validIndex = i;
+                       } else {
+                               this.errors = this.errors.slice(0, startErrorCount);
+                               return new ValidationError(ErrorCodes.ONE_OF_MULTIPLE, "Data is valid against more than one schema from \"oneOf\": indices " + validIndex + " and " + i, "", "/oneOf");
+                       }
+               } else if (error) {
+                       errors.push(error.prefixWith(null, "" + i).prefixWith(null, "oneOf"));
+               }
+       }
+       if (validIndex == null) {
+               errors = errors.concat(this.errors.slice(startErrorCount));
+               this.errors = this.errors.slice(0, startErrorCount);
+               return new ValidationError(ErrorCodes.ONE_OF_MISSING, "Data does not match any schemas from \"oneOf\"", "", "/oneOf", errors);
+       } else {
+               this.errors = this.errors.slice(0, startErrorCount);
+       }
+       return null;
+}
+
+ValidatorContext.prototype.validateNot = function validateNot(data, schema) {
+       if (schema.not == undefined) {
+               return null;
+       }
+       var oldErrorCount = this.errors.length;
+       var error = this.validateAll(data, schema.not);
+       var notErrors = this.errors.slice(oldErrorCount);
+       this.errors = this.errors.slice(0, oldErrorCount);
+       if (error == null && notErrors.length == 0) {
+               return new ValidationError(ErrorCodes.NOT_PASSED, "Data matches schema from \"not\"", "", "/not")
+       }
+       return null;
+}
+
+// parseURI() and resolveUrl() are from https://gist.github.com/1088850
+//   -  released as public domain by author ("Yaffle") - see comments on gist
+
+function parseURI(url) {
+       var m = String(url).replace(/^\s+|\s+$/g, '').match(/^([^:\/?#]+:)?(\/\/(?:[^:@]*(?::[^:@]*)?@)?(([^:\/?#]*)(?::(\d*))?))?([^?#]*)(\?[^#]*)?(#[\s\S]*)?/);
+       // authority = '//' + user + ':' + pass '@' + hostname + ':' port
+       return (m ? {
+               href     : m[0] || '',
+               protocol : m[1] || '',
+               authority: m[2] || '',
+               host     : m[3] || '',
+               hostname : m[4] || '',
+               port     : m[5] || '',
+               pathname : m[6] || '',
+               search   : m[7] || '',
+               hash     : m[8] || ''
+       } : null);
+}
+
+function resolveUrl(base, href) {// RFC 3986
+
+       function removeDotSegments(input) {
+               var output = [];
+               input.replace(/^(\.\.?(\/|$))+/, '')
+                       .replace(/\/(\.(\/|$))+/g, '/')
+                       .replace(/\/\.\.$/, '/../')
+                       .replace(/\/?[^\/]*/g, function (p) {
+                               if (p === '/..') {
+                                       output.pop();
+                               } else {
+                                       output.push(p);
+                               }
+               });
+               return output.join('').replace(/^\//, input.charAt(0) === '/' ? '/' : '');
+       }
+
+       href = parseURI(href || '');
+       base = parseURI(base || '');
+
+       return !href || !base ? null : (href.protocol || base.protocol) +
+               (href.protocol || href.authority ? href.authority : base.authority) +
+               removeDotSegments(href.protocol || href.authority || href.pathname.charAt(0) === '/' ? href.pathname : (href.pathname ? ((base.authority && !base.pathname ? '/' : '') + base.pathname.slice(0, base.pathname.lastIndexOf('/') + 1) + href.pathname) : base.pathname)) +
+               (href.protocol || href.authority || href.pathname ? href.search : (href.search || base.search)) +
+               href.hash;
+}
+
+function normSchema(schema, baseUri) {
+       if (baseUri == undefined) {
+               baseUri = schema.id;
+       } else if (typeof schema.id == "string") {
+               baseUri = resolveUrl(baseUri, schema.id);
+               schema.id = baseUri;
+       }
+       if (typeof schema == "object") {
+               if (Array.isArray(schema)) {
+                       for (var i = 0; i < schema.length; i++) {
+                               normSchema(schema[i], baseUri);
+                       }
+               } else if (typeof schema['$ref'] == "string") {
+                       schema['$ref'] = resolveUrl(baseUri, schema['$ref']);
+               } else {
+                       for (var key in schema) {
+                               if (key != "enum") {
+                                       normSchema(schema[key], baseUri);
+                               }
+                       }
+               }
+       }
+}
+
+var ErrorCodes = {
+       INVALID_TYPE: 0,
+       ENUM_MISMATCH: 1,
+       ANY_OF_MISSING: 10,
+       ONE_OF_MISSING: 11,
+       ONE_OF_MULTIPLE: 12,
+       NOT_PASSED: 13,
+       // Numeric errors
+       NUMBER_MULTIPLE_OF: 100,
+       NUMBER_MINIMUM: 101,
+       NUMBER_MINIMUM_EXCLUSIVE: 102,
+       NUMBER_MAXIMUM: 103,
+       NUMBER_MAXIMUM_EXCLUSIVE: 104,
+       // String errors
+       STRING_LENGTH_SHORT: 200,
+       STRING_LENGTH_LONG: 201,
+       STRING_PATTERN: 202,
+       // Object errors
+       OBJECT_PROPERTIES_MINIMUM: 300,
+       OBJECT_PROPERTIES_MAXIMUM: 301,
+       OBJECT_REQUIRED: 302,
+       OBJECT_ADDITIONAL_PROPERTIES: 303,
+       OBJECT_DEPENDENCY_KEY: 304,
+       // Array errors
+       ARRAY_LENGTH_SHORT: 400,
+       ARRAY_LENGTH_LONG: 401,
+       ARRAY_UNIQUE: 402,
+       ARRAY_ADDITIONAL_ITEMS: 403
+};
+
+function ValidationError(code, message, dataPath, schemaPath, subErrors) {
+       if (code == undefined) {
+               throw new Error ("No code supplied for error: "+ message);
+       }
+       this.code = code;
+       this.message = message;
+       this.dataPath = dataPath ? dataPath : "";
+       this.schemaPath = schemaPath ? schemaPath : "";
+       this.subErrors = subErrors ? subErrors : null;
+}
+ValidationError.prototype = {
+       prefixWith: function (dataPrefix, schemaPrefix) {
+               if (dataPrefix != null) {
+                       dataPrefix = dataPrefix.replace("~", "~0").replace("/", "~1");
+                       this.dataPath = "/" + dataPrefix + this.dataPath;
+               }
+               if (schemaPrefix != null) {
+                       schemaPrefix = schemaPrefix.replace("~", "~0").replace("/", "~1");
+                       this.schemaPath = "/" + schemaPrefix + this.schemaPath;
+               }
+               if (this.subErrors != null) {
+                       for (var i = 0; i < this.subErrors.length; i++) {
+                               this.subErrors[i].prefixWith(dataPrefix, schemaPrefix);
+                       }
+               }
+               return this;
+       }
+};
+
+function searchForTrustedSchemas(map, schema, url) {
+       if (typeof schema.id == "string") {
+               if (schema.id.substring(0, url.length) == url) {
+                       var remainder = schema.id.substring(url.length);
+                       if ((url.length > 0 && url.charAt(url.length - 1) == "/")
+                               || remainder.charAt(0) == "#"
+                               || remainder.charAt(0) == "?") {
+                               if (map[schema.id] == undefined) {
+                                       map[schema.id] = schema;
+                               }
+                       }
+               }
+       }
+       if (typeof schema == "object") {
+               for (var key in schema) {
+                       if (key != "enum" && typeof schema[key] == "object") {
+                               searchForTrustedSchemas(map, schema[key], url);
+                       }
+               }
+       }
+       return map;
+}
+
+var globalContext = new ValidatorContext();
+
+var publicApi = {
+       validate: function (data, schema) {
+               var context = new ValidatorContext(globalContext);
+               if (typeof schema == "string") {
+                       schema = {"$ref": schema};
+               }
+               var added = context.addSchema("", schema);
+               var error = context.validateAll(data, schema);
+               this.error = error;
+               this.missing = context.missing;
+               this.valid = (error == null);
+               return this.valid;
+       },
+       validateResult: function () {
+               var result = {};
+               this.validate.apply(result, arguments);
+               return result;
+       },
+       validateMultiple: function (data, schema) {
+               var context = new ValidatorContext(globalContext, true);
+               if (typeof schema == "string") {
+                       schema = {"$ref": schema};
+               }
+               context.addSchema("", schema);
+               context.validateAll(data, schema);
+               var result = {};
+               result.errors = context.errors;
+               result.missing = context.missing;
+               result.valid = (result.errors.length == 0);
+               return result;
+       },
+       addSchema: function (url, schema) {
+               return globalContext.addSchema(url, schema);
+       },
+       getSchema: function (url) {
+               return globalContext.getSchema(url);
+       },
+       missing: [],
+       error: null,
+       normSchema: normSchema,
+       resolveUrl: resolveUrl,
+       errorCodes: ErrorCodes
+};
+
+global.tv4 = publicApi;
+
+})((typeof module !== 'undefined' && module.exports) ? exports : this);
+
+
+
+/** FILE: lib/Math.uuid.js **/
+/*!
+  Math.uuid.js (v1.4)
+  http://www.broofa.com
+  mailto:robert@broofa.com
+
+  Copyright (c) 2010 Robert Kieffer
+  Dual licensed under the MIT and GPL licenses.
+
+  ********
+
+  Changes within remoteStorage.js:
+  2012-10-31:
+  - added AMD wrapper <niklas@unhosted.org>
+  - moved extensions for Math object into exported object.
+*/
+
+/*
+ * Generate a random uuid.
+ *
+ * USAGE: Math.uuid(length, radix)
+ *   length - the desired number of characters
+ *   radix  - the number of allowable values for each character.
+ *
+ * EXAMPLES:
+ *   // No arguments  - returns RFC4122, version 4 ID
+ *   >>> Math.uuid()
+ *   "92329D39-6F5C-4520-ABFC-AAB64544E172"
+ *
+ *   // One argument - returns ID of the specified length
+ *   >>> Math.uuid(15)     // 15 character ID (default base=62)
+ *   "VcydxgltxrVZSTV"
+ *
+ *   // Two arguments - returns ID of the specified length, and radix. (Radix must be <= 62)
+ *   >>> Math.uuid(8, 2)  // 8 character ID (base=2)
+ *   "01001010"
+ *   >>> Math.uuid(8, 10) // 8 character ID (base=10)
+ *   "47473046"
+ *   >>> Math.uuid(8, 16) // 8 character ID (base=16)
+ *   "098F4D35"
+ */
+  // Private array of chars to use
+  var CHARS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split('');
+
+Math.uuid = function (len, radix) {
+  var chars = CHARS, uuid = [], i;
+  radix = radix || chars.length;
+
+  if (len) {
+    // Compact form
+    for (i = 0; i < len; i++) uuid[i] = chars[0 | Math.random()*radix];
+  } else {
+    // rfc4122, version 4 form
+    var r;
+
+    // rfc4122 requires these characters
+    uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-';
+    uuid[14] = '4';
+
+    // Fill in random data.  At i==19 set the high bits of clock sequence as
+    // per rfc4122, sec. 4.1.5
+    for (i = 0; i < 36; i++) {
+      if (!uuid[i]) {
+        r = 0 | Math.random()*16;
+        uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];
+      }
+    }
+  }
+
+  return uuid.join('');
+};
+
+
+/** FILE: src/baseclient.js **/
+
+(function(global) {
+
+  function deprecate(thing, replacement) {
+    console.log('WARNING: ' + thing + ' is deprecated. Use ' +
+                replacement + ' instead.');
+  }
+
+  var RS = RemoteStorage;
+
+  /**
+   * Class: RemoteStorage.BaseClient
+   *
+   * Provides a high-level interface to access data below a given root path.
+   *
+   * A BaseClient deals with three types of data: folders, objects and files.
+   *
+   * <getListing> returns a list of all items within a folder. Items that end
+   * with a forward slash ("/") are child folders.
+   *
+   * <getObject> / <storeObject> operate on JSON objects. Each object has a type.
+   *
+   * <getFile> / <storeFile> operates on files. Each file has a MIME type.
+   *
+   * <remove> operates on either objects or files (but not folders, folders are
+   * created and removed implictly).
+   */
+  RS.BaseClient = function(storage, base) {
+    if(base[base.length - 1] != '/') {
+      throw "Not a directory: " + base;
+    }
+    /**
+     * Property: storage
+     *
+     * The <RemoteStorage> instance this <BaseClient> operates on.
+     */
+    this.storage = storage;
+    /**
+     * Property: base
+     *
+     * Base path this <BaseClient> operates on.
+     *
+     * For the module's privateClient this would be /<moduleName>/, for the
+     * corresponding publicClient /public/<moduleName>/.
+     */
+    this.base = base;
+
+    var parts = this.base.split('/');
+    if(parts.length > 2) {
+      this.moduleName = parts[1];
+    } else {
+      this.moduleName = 'root';
+    }
+
+    RS.eventHandling(this, 'change', 'conflict');
+    this.on = this.on.bind(this);
+    storage.onChange(this.base, this._fireChange.bind(this));
+    storage.onConflict(this.base, this._fireConflict.bind(this));
+  };
+
+  RS.BaseClient.prototype = {
+
+    // BEGIN LEGACY
+    use: function(path) {
+      deprecate('BaseClient#use(path)', 'BaseClient#cache(path)');
+      return this.cache(path);
+    },
+
+    release: function(path) {
+      deprecate('BaseClient#release(path)', 'BaseClient#cache(path, false)');
+      return this.cache(path, false);
+    },
+    // END LEGACY
+
+    extend: function(object) {
+      for(var key in object) {
+        this[key] = object[key];
+      }
+      return this;
+    },
+
+    /**
+     * Method: scope
+     *
+     * Returns a new <BaseClient> operating on a subpath of the current <base> path.
+     */
+    scope: function(path) {
+      return new RS.BaseClient(this.storage, this.makePath(path));
+    },
+
+    // folder operations
+
+    /**
+     * Method: getListing
+     *
+     * Get a list of child nodes below a given path.
+     *
+     * The callback semantics of getListing are identical to those of getObject.
+     *
+     * Parameters:
+     *   path     - The path to query. It MUST end with a forward slash.
+     *
+     * Returns:
+     *   A promise for an Array of keys, representing child nodes.
+     *   Those keys ending in a forward slash, represent *directory nodes*, all
+     *   other keys represent *data nodes*.
+     *
+     * Example:
+     *   (start code)
+     *   client.getListing('').then(function(listing) {
+     *     listing.forEach(function(item) {
+     *       console.log('- ' + item);
+     *     });
+     *   });
+     *   (end code)
+     */
+    getListing: function(path) {
+      if(typeof(path) == 'undefined') {
+        path = '';
+      } else if(path.length > 0 && path[path.length - 1] != '/') {
+        throw "Not a directory: " + path;
+      }
+      return this.storage.get(this.makePath(path)).then(function(status, body) {
+        if(status == 404) return;
+        return typeof(body) === 'object' ? Object.keys(body) : undefined;
+      });
+    },
+
+    /**
+     * Method: getAll
+     *
+     * Get all objects directly below a given path.
+     *
+     * Parameters:
+     *   path      - path to the direcotry
+     *   typeAlias - (optional) local type-alias to filter for
+     *
+     * Returns:
+     *   a promise for an object in the form { path : object, ... }
+     *
+     * Example:
+     *   (start code)
+     *   client.getAll('').then(function(objects) {
+     *     for(var key in objects) {
+     *       console.log('- ' + key + ': ', objects[key]);
+     *     }
+     *   });
+     *   (end code)
+     */
+    getAll: function(path) {
+      if(typeof(path) == 'undefined') {
+        path = '';
+      } else if(path.length > 0 && path[path.length - 1] != '/') {
+        throw "Not a directory: " + path;
+      }
+      return this.storage.get(this.makePath(path)).then(function(status, body) {
+        if(status == 404) return;
+        if(typeof(body) === 'object') {
+          var promise = promising();
+          var count = Object.keys(body).length, i = 0;
+          if(count == 0) {
+            // treat this like 404. it probably means a directory listing that
+            // has changes that haven't been pushed out yet.
+            return;
+          }
+          for(var key in body) {
+            this.storage.get(this.makePath(path + key)).
+              then(function(status, b) {
+                body[this.key] = b;
+                i++;
+                if(i == count) promise.fulfill(body);
+              }.bind({ key: key }));
+          }
+          return promise;
+        }
+      }.bind(this));
+    },
+
+    // file operations
+
+    /**
+     * Method: getFile
+     *
+     * Get the file at the given path. A file is raw data, as opposed to
+     * a JSON object (use <getObject> for that).
+     *
+     * Except for the return value structure, getFile works exactly like
+     * getObject.
+     *
+     * Parameters:
+     *   path     - see getObject
+     *
+     * Returns:
+     *   A promise for an object:
+     *
+     *   mimeType - String representing the MIME Type of the document.
+     *   data     - Raw data of the document (either a string or an ArrayBuffer)
+     *
+     * Example:
+     *   (start code)
+     *   // Display an image:
+     *   client.getFile('path/to/some/image').then(function(file) {
+     *     var blob = new Blob([file.data], { type: file.mimeType });
+     *     var targetElement = document.findElementById('my-image-element');
+     *     targetElement.src = window.URL.createObjectURL(blob);
+     *   });
+     *   (end code)
+     */
+    getFile: function(path) {
+      return this.storage.get(this.makePath(path)).then(function(status, body, mimeType, revision) {
+        return {
+          data: body,
+          mimeType: mimeType,
+          revision: revision // (this is new)
+        };
+      });
+    },
+
+    /**
+     * Method: storeFile
+     *
+     * Store raw data at a given path.
+     *
+     * Parameters:
+     *   mimeType - MIME media type of the data being stored
+     *   path     - path relative to the module root. MAY NOT end in a forward slash.
+     *   data     - string, ArrayBuffer or ArrayBufferView of raw data to store
+     *
+     * The given mimeType will later be returned, when retrieving the data
+     * using <getFile>.
+     *
+     * Example (UTF-8 data):
+     *   (start code)
+     *   client.storeFile('text/html', 'index.html', '<h1>Hello World!</h1>');
+     *   (end code)
+     *
+     * Example (Binary data):
+     *   (start code)
+     *   // MARKUP:
+     *   <input type="file" id="file-input">
+     *   // CODE:
+     *   var input = document.getElementById('file-input');
+     *   var file = input.files[0];
+     *   var fileReader = new FileReader();
+     *
+     *   fileReader.onload = function() {
+     *     client.storeFile(file.type, file.name, fileReader.result);
+     *   };
+     *
+     *   fileReader.readAsArrayBuffer(file);
+     *   (end code)
+     *
+     */
+    storeFile: function(mimeType, path, body) {
+      return this.storage.put(this.makePath(path), body, mimeType).then(function(status, _body, _mimeType, revision) {
+        if(status == 200 || status == 201) {
+          return revision;
+        } else {
+          throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status;
+        }
+      });
+    },
+
+    // object operations
+
+    /**
+     * Method: getObject
+     *
+     * Get a JSON object from given path.
+     *
+     * Parameters:
+     *   path     - relative path from the module root (without leading slash)
+     *
+     * Returns:
+     *   A promise for the object.
+     *
+     * Example:
+     *   (start code)
+     *   client.getObject('/path/to/object').
+     *     then(function(object) {
+     *       // object is either an object or null
+     *     });
+     *   (end code)
+     */
+    getObject: function(path) {
+      return this.storage.get(this.makePath(path)).then(function(status, body, mimeType, revision) {
+        if(typeof(body) == 'object') {
+          return body;
+        } else if(typeof(body) !== 'undefined' && status == 200) {
+          throw "Not an object: " + this.makePath(path);
+        }
+      });
+    },
+
+    /**
+     * Method: storeObject
+     *
+     * Store object at given path. Triggers synchronization.
+     *
+     * Parameters:
+     *
+     *   type     - unique type of this object within this module. See description below.
+     *   path     - path relative to the module root.
+     *   object   - an object to be saved to the given node. It must be serializable as JSON.
+     *
+     * Returns:
+     *   A promise to store the object. The promise fails with a ValidationError, when validations fail.
+     *
+     *
+     * What about the type?:
+     *
+     *   A great thing about having data on the web, is to be able to link to
+     *   it and rearrange it to fit the current circumstances. To facilitate
+     *   that, eventually you need to know how the data at hand is structured.
+     *   For documents on the web, this is usually done via a MIME type. The
+     *   MIME type of JSON objects however, is always application/json.
+     *   To add that extra layer of "knowing what this object is", remoteStorage
+     *   aims to use <JSON-LD at http://json-ld.org/>.
+     *   A first step in that direction, is to add a *@context attribute* to all
+     *   JSON data put into remoteStorage.
+     *   Now that is what the *type* is for.
+     *
+     *   Within remoteStorage.js, @context values are built using three components:
+     *     http://remotestoragejs.com/spec/modules/ - A prefix to guarantee unqiueness
+     *     the module name     - module names should be unique as well
+     *     the type given here - naming this particular kind of object within this module
+     *
+     *   In retrospect that means, that whenever you introduce a new "type" in calls to
+     *   storeObject, you should make sure that once your code is in the wild, future
+     *   versions of the code are compatible with the same JSON structure.
+     *
+     * How to define types?:
+     *
+     *   See <declareType> for examples.
+     */
+    storeObject: function(typeAlias, path, object) {
+      this._attachType(object, typeAlias);
+      try {
+        var validationResult = this.validate(object);
+        if(! validationResult.valid) {
+          return promising(function(p) { p.reject(validationResult); });
+        }
+      } catch(exc) {
+        if(exc instanceof RS.BaseClient.Types.SchemaNotFound) {
+          // ignore.
+        } else {
+          return promising().reject(exc);
+        }
+      }
+      return this.storage.put(this.makePath(path), object, 'application/json; charset=UTF-8').then(function(status, _body, _mimeType, revision) {
+        if(status == 200 || status == 201) {
+          return revision;
+        } else {
+          throw "Request (PUT " + this.makePath(path) + ") failed with status: " + status;
+        }
+      }.bind(this));
+    },
+
+    // generic operations
+
+    /**
+     * Method: remove
+     *
+     * Remove node at given path from storage. Triggers synchronization.
+     *
+     * Parameters:
+     *   path     - Path relative to the module root.
+     */
+    remove: function(path) {
+      return this.storage.delete(this.makePath(path));
+    },
+
+    cache: function(path, disable) {
+      this.storage.caching[disable !== false ? 'enable' : 'disable'](
+        this.makePath(path)
+      );
+      return this;
+    },
+
+    makePath: function(path) {
+      return this.base + (path || '');
+    },
+
+    _fireChange: function(event) {
+      this._emit('change', event);
+    },
+
+    _fireConflict: function(event) {
+      if(this._handlers.conflict.length > 0) {
+        this._emit('conflict', event);
+      } else {
+        event.resolve('remote');
+      }
+    },
+
+    getItemURL: function(path) {
+      if(this.storage.connected) {
+        return this.storage.remote.href + this.makePath(path);
+      } else {
+        return undefined;
+      }
+    },
+
+    uuid: function() {
+      return Math.uuid();
+    }
+
+  };
+
+  /**
+   * Method: RS#scope
+   *
+   * Returns a new <RS.BaseClient> scoped to the given path.
+   *
+   * Parameters:
+   *   path - Root path of new BaseClient.
+   *
+   *
+   * Example:
+   *   (start code)
+   *
+   *   var foo = remoteStorage.scope('/foo/');
+   *
+   *   // PUTs data "baz" to path /foo/bar
+   *   foo.storeFile('text/plain', 'bar', 'baz');
+   *
+   *   var something = foo.scope('something/');
+   *
+   *   // GETs listing from path /foo/something/bla/
+   *   something.getListing('bla/');
+   *
+   *   (end code)
+   *
+   */
+
+
+  RS.BaseClient._rs_init = function() {
+    RS.prototype.scope = function(path) {
+      return new RS.BaseClient(this, path);
+    };
+  };
+
+  /* e.g.:
+  remoteStorage.defineModule('locations', function(priv, pub) {
+    return {
+      exports: {
+        features: priv.scope('features/').defaultType('feature'),
+        collections: priv.scope('collections/').defaultType('feature-collection');
+      }
+    };
+  });
+  */
+
+})(this);
+
+
+/** FILE: src/baseclient/types.js **/
+
+(function(global) {
+
+  RemoteStorage.BaseClient.Types = {
+    // <alias> -> <uri>
+    uris: {},
+    // <uri> -> <schema>
+    schemas: {},
+    // <uri> -> <alias>
+    aliases: {},
+
+    declare: function(moduleName, alias, uri, schema) {
+      var fullAlias = moduleName + '/' + alias;
+
+      if(schema.extends) {
+        var extendedAlias;
+        var parts = schema.extends.split('/');
+        if(parts.length === 1) {
+          extendedAlias = moduleName + '/' + parts.shift();
+        } else {
+          extendedAlias = parts.join('/');
+        }
+        var extendedUri = this.uris[extendedAlias];
+        if(! extendedUri) {
+          throw "Type '" + fullAlias + "' tries to extend unknown schema '" + extendedAlias + "'";
+        }
+        schema.extends = this.schemas[extendedUri];
+      }
+      
+      this.uris[fullAlias] = uri;
+      this.aliases[uri] = fullAlias;
+      this.schemas[uri] = schema;
+    },
+
+    resolveAlias: function(alias) {
+      return this.uris[alias];
+    },
+
+    getSchema: function(uri) {
+      return this.schemas[uri];
+    },
+
+    inScope: function(moduleName) {
+      var ml = moduleName.length;
+      var schemas = {};
+      for(var alias in this.uris) {
+        if(alias.substr(0, ml + 1) == moduleName + '/') {
+          var uri = this.uris[alias];
+          schemas[uri] = this.schemas[uri];
+        }
+      }
+      return schemas;
+    }
+  };
+
+  var SchemaNotFound = function(uri) {
+    var error = Error("Schema not found: " + uri);
+    error.name = "SchemaNotFound";
+    return error;
+  };
+  SchemaNotFound.prototype = Error.prototype;
+
+  RemoteStorage.BaseClient.Types.SchemaNotFound = SchemaNotFound;
+
+  RemoteStorage.BaseClient.prototype.extend({
+
+    validate: function(object) {
+      var schema = RemoteStorage.BaseClient.Types.getSchema(object['@context']);
+      if(schema) {
+        return tv4.validateResult(object, schema);
+      } else {
+        throw new SchemaNotFound(object['@context']);
+      }
+    },
+
+    // client.declareType(alias, schema);
+    //  /* OR */
+    // client.declareType(alias, uri, schema);
+    declareType: function(alias, uri, schema) {
+      if(! schema) {
+        schema = uri;
+        uri = this._defaultTypeURI(alias);
+      }
+      RemoteStorage.BaseClient.Types.declare(this.moduleName, alias, uri, schema);
+    },
+
+    _defaultTypeURI: function(alias) {
+      return 'http://remotestoragejs.com/spec/modules/' + this.moduleName + '/' + alias;
+    },
+
+    _attachType: function(object, alias) {
+      object['@context'] = RemoteStorage.BaseClient.Types.resolveAlias(alias) || this._defaultTypeURI(alias);
+    }
+  });
+
+  Object.defineProperty(RemoteStorage.BaseClient.prototype, 'schemas', {
+    configurable: true,
+    get: function() {
+      return RemoteStorage.BaseClient.Types.inScope(this.moduleName);
+    }
+  });
+
+})(this);
+
+
+/** FILE: src/caching.js **/
+(function(global) {
+
+  var haveLocalStorage = 'localStorage' in global;
+  var SETTINGS_KEY = "remotestorage:caching";
+
+  function containingDir(path) {
+    if(path === '') return '/';
+    if(! path) throw "Path not given!";
+    return path.replace(/\/+/g, '/').replace(/[^\/]+\/?$/, '');
+  }
+
+  function isDir(path) {
+    return path.substr(-1) == '/';
+  }
+
+  function pathContains(a, b) {
+    return a.slice(0, b.length) === b;
+  }
+
+  /**
+   * Class: RemoteStorage.Caching
+   *
+   * Holds caching configuration.
+   */
+  RemoteStorage.Caching = function() {
+    this.reset();
+
+    if(haveLocalStorage) {
+      var settings = localStorage[SETTINGS_KEY];
+      if(settings) {
+        this._pathSettingsMap = JSON.parse(settings);
+        this._updateRoots();
+      }
+    }
+  };
+
+  RemoteStorage.Caching.prototype = {
+
+    /**
+     * Method: enable
+     *
+     * Enable caching for the given path.
+     *
+     * Parameters:
+     *   path - Absolute path to a directory.
+     */
+    enable: function(path) { this.set(path, { data: true }); },
+    /**
+     * Method: disable
+     *
+     * Disable caching for the given path.
+     *
+     * Parameters:
+     *   path - Absolute path to a directory.
+     */
+    disable: function(path) { this.remove(path); },
+
+    /**
+     ** configuration methods
+     **/
+
+    get: function(path) {
+      this._validateDirPath(path);
+      return this._pathSettingsMap[path];
+    },
+
+    set: function(path, settings) {
+      this._validateDirPath(path);
+      if(typeof(settings) !== 'object') {
+        throw new Error("settings is required");
+      }
+      this._pathSettingsMap[path] = settings;
+      this._updateRoots();
+    },
+
+    remove: function(path) {
+      this._validateDirPath(path);
+      delete this._pathSettingsMap[path];
+      this._updateRoots();
+    },
+
+    reset: function() {
+      this.rootPaths = [];
+      this._pathSettingsMap = {};
+    },
+
+    /**
+     ** query methods
+     **/
+
+    // Method: descendIntoPath
+    //
+    // Checks if the given directory path should be followed.
+    //
+    // Returns: true or false
+    descendIntoPath: function(path) {
+      this._validateDirPath(path);
+      return !! this._query(path);
+    },
+
+    // Method: cachePath
+    //
+    // Checks if given path should be cached.
+    //
+    // Returns: true or false
+    cachePath: function(path) {
+      this._validatePath(path);
+      var settings = this._query(path);
+      return settings && (isDir(path) || settings.data);
+    },
+
+    /**
+     ** private methods
+     **/
+
+    // gets settings for given path. walks up the path until it finds something.
+    _query: function(path) {
+      return this._pathSettingsMap[path] ||
+        path !== '/' &&
+        this._query(containingDir(path));
+    },
+
+    _validatePath: function(path) {
+      if(typeof(path) !== 'string') {
+        throw new Error("path is required");
+      }
+    },
+
+    _validateDirPath: function(path) {
+      this._validatePath(path);
+      if(! isDir(path)) {
+        throw new Error("not a directory path: " + path);
+      }
+      if(path[0] !== '/') {
+        throw new Error("path not absolute: " + path);
+      }
+    },
+
+    _updateRoots: function() {
+      var roots = {}
+      for(var a in this._pathSettingsMap) {
+        // already a root
+        if(roots[a]) {
+          continue;
+        }
+        var added = false;
+        for(var b in this._pathSettingsMap) {
+          if(pathContains(a, b)) {
+            roots[b] = true;
+            added = true;
+            break;
+          }
+        }
+        if(! added) {
+          roots[a] = true;
+        }
+      }
+      this.rootPaths = Object.keys(roots);
+      if(haveLocalStorage) {
+        localStorage[SETTINGS_KEY] = JSON.stringify(this._pathSettingsMap);
+      }
+    },
+
+  };
+
+  Object.defineProperty(RemoteStorage.Caching.prototype, 'list', {
+    get: function() {
+      var list = [];
+      for(var path in this._pathSettingsMap) {
+        list.push({ path: path, settings: this._pathSettingsMap[path] });
+      }
+      return list;
+    }
+  });
+
+
+  Object.defineProperty(RemoteStorage.prototype, 'caching', {
+    configurable: true,
+    get: function() {
+      var caching = new RemoteStorage.Caching();
+      Object.defineProperty(this, 'caching', {
+        value: caching
+      });
+      return caching;
+    }
+  });
+
+  RemoteStorage.Caching._rs_init = function() {};
+
+})(this);
+
+
+/** FILE: src/sync.js **/
+(function(global) {
+
+  var SYNC_INTERVAL = 10000;
+
+  //
+  // The synchronization algorithm is as follows:
+  //
+  // (for each path in caching.rootPaths)
+  //
+  // (1) Fetch all pending changes from 'local'
+  // (2) Try to push pending changes to 'remote', if that fails mark a
+  //     conflict, otherwise clear the change.
+  // (3) Folder items: GET a listing
+  //     File items: GET the file
+  // (4) Compare versions. If they match the locally cached one, return.
+  //     Otherwise continue.
+  // (5) Folder items: For each child item, run this algorithm starting at (3).
+  //     File items: Fetch remote data and replace locally cached copy.
+  //
+  // Depending on the API version the server supports, the version comparison
+  // can either happen on the server (through ETag, If-Match, If-None-Match
+  // headers), or on the client (through versions specified in the parent listing).
+  //
+
+  function isDir(path) {
+    return path[path.length - 1] == '/';
+  }
+
+  function descendInto(remote, local, path, keys, promise) {
+    var n = keys.length, i = 0;
+    if(n == 0) promise.fulfill();
+    function oneDone() {
+      i++;
+      if(i == n) promise.fulfill();
+    }
+    keys.forEach(function(key) {
+      synchronize(remote, local, path + key).then(oneDone);
+    });
+  }
+
+  function updateLocal(remote, local, path, body, contentType, revision, promise) {
+    if(isDir(path)) {
+      descendInto(remote, local, path, Object.keys(body), promise);
+    } else {
+      local.put(path, body, contentType, true).then(function() {
+        return local.setRevision(path, revision)
+      }).then(function() {
+        promise.fulfill();
+      });
+    }
+  }
+
+  function allKeys(a, b) {
+    var keyObject = {};
+    for(var ak in a) keyObject[ak] = true;
+    for(var bk in b) keyObject[bk] = true;
+    return Object.keys(keyObject);
+  }
+  function promiseDeleteLocal(local, path) {
+    var promise = promising();
+    deleteLocal(local, path, promise);
+    return promise;
+  }
+  function deleteLocal(local, path, promise) {
+    if(isDir(path)) {
+      local.get(path).then(function(localStatus, localBody, localContentType, localRevision) {
+        var keys = [], failed = false;
+        for(item in localBody) {
+          keys.push(item);
+        }
+        //console.log('deleting keys', keys, 'from', path, localBody);
+        var n = keys.length, i = 0;
+        if(n == 0) promise.fulfill();
+        function oneDone() {
+          i++;
+          if(i == n && !failed) promise.fulfill();
+        }
+        function oneFailed(error) {
+          if(!failed) {
+            failed = true;
+            promise.reject(error);
+          }
+        }
+        keys.forEach(function(key) {
+          promiseDeleteLocal(local, path + key).then(oneDone, oneFailed);
+        });
+      });
+    } else {
+      //console.log('deleting local item', path);
+      local.delete(path, true).then(promise.fulfill, promise.reject);
+    }
+  }
+  function synchronize(remote, local, path, options) {
+    var promise = promising();
+    local.get(path).then(function(localStatus, localBody, localContentType, localRevision) {
+      remote.get(path, {
+        ifNoneMatch: localRevision
+      }).then(function(remoteStatus, remoteBody, remoteContentType, remoteRevision) {
+        //TEST// if(remoteStatus == 401 || remoteStatus == 403) {
+        if(remoteStatus == 401999 || remoteStatus == 403) {
+          throw new RemoteStorage.Unauthorized();
+        } else if(remoteStatus == 412 || remoteStatus == 304) {
+          // up to date.
+          promise.fulfill();
+        } else if(localStatus == 404 && remoteStatus == 200) {
+          // local doesn't exist, remote does.
+          updateLocal(remote, local, path, remoteBody, remoteContentType, remoteRevision, promise);
+        } else if(localStatus == 200 && remoteStatus == 404) {
+          // remote doesn't exist, local does.
+          deleteLocal(local, path, promise);
+        } else if(localStatus == 200 && remoteStatus == 200) {
+          if(isDir(path)) {
+            local.setRevision(path, remoteRevision).then(function() {
+              descendInto(remote, local, path, allKeys(localBody, remoteBody), promise);
+            });
+          } else {
+            updateLocal(remote, local, path, remoteBody, remoteContentType, remoteRevision, promise);
+          }
+        } else {
+          // do nothing.
+          promise.fulfill();
+        }
+      }).then(undefined, promise.reject);
+    }).then(undefined, promise.reject);
+    return promise;
+  }
+
+  function fireConflict(local, path, attributes) {
+    local.setConflict(path, attributes);
+  }
+
+  function pushChanges(remote, local, path) {
+    return local.changesBelow(path).then(function(changes) {
+      var n = changes.length, i = 0;
+      var promise = promising();
+      function oneDone(path) {
+        function done() {
+          i++;
+          if(i == n) promise.fulfill();
+        }
+        if(path) {
+          // change was propagated -> clear.
+          local.clearChange(path).then(done);
+        } else {
+          // change wasn't propagated (conflict?) -> handle it later.
+          done();
+        }
+      }
+      if(n > 0) {
+        function errored(err) {
+          console.error("pushChanges aborted due to error: ", err, err.stack);
+          promise.reject(err);
+        }
+        changes.forEach(function(change) {
+          if(change.conflict) {
+            var res = change.conflict.resolution;
+            if(res) {
+              RemoteStorage.log('about to resolve', res);
+              // ready to be resolved.
+              change.action = (res == 'remote' ? change.remoteAction : change.localAction);
+              change.force = true;
+            } else {
+              RemoteStorage.log('conflict pending for ', change.path);
+              // pending conflict, won't do anything.
+              return oneDone();
+            }
+          }
+          switch(change.action) {
+          case 'PUT':
+            var options = {};
+            if(! change.force) {
+              if(change.revision) {
+                options.ifMatch = change.revision;
+              } else {
+                options.ifNoneMatch = '*';
+              }
+            }
+            local.get(change.path).then(function(status, body, contentType) {
+              if(status == 200) {
+                return remote.put(change.path, body, contentType, options);
+              } else {
+                return 200; // fake 200 so the change is cleared.
+              }
+            }).then(function(status) {
+              if(status == 412) {
+                fireConflict(local, change.path, {
+                  localAction: 'PUT',
+                  remoteAction: 'PUT'
+                });
+                oneDone();
+              } else {
+                oneDone(change.path);
+              }
+            }).then(undefined, errored);
+            break;
+          case 'DELETE':
+            remote.delete(change.path, {
+              ifMatch: change.force ? undefined : change.revision
+            }).then(function(status) {
+              if(status == 412) {
+                fireConflict(local, change.path, {
+                  remoteAction: 'PUT',
+                  localAction: 'DELETE'
+                });
+                oneDone();
+              } else {
+                oneDone(change.path);
+              }
+            }).then(undefined, errored);
+            break;
+          }
+        });
+        return promise;
+      }
+    });
+  }
+
+  RemoteStorage.Sync = {
+    sync: function(remote, local, path) {
+      return pushChanges(remote, local, path).
+        then(function() {
+          return synchronize(remote, local, path);
+        });
+    },
+
+    syncTree: function(remote, local, path) {
+      return synchronize(remote, local, path, {
+        data: false
+      });
+    }
+  };
+
+  var SyncError = function(originalError) {
+    var msg = 'Sync failed: ';
+    if(typeof(originalError) == 'object' && 'message' in originalError) {
+      msg += originalError.message;
+    } else {
+      msg += originalError;
+    }
+    this.originalError = originalError;
+    Error.apply(this, [msg]);
+  };
+
+  SyncError.prototype = Object.create(Error.prototype);
+
+  RemoteStorage.prototype.sync = function() {
+    if(! (this.local && this.caching)) {
+      throw "Sync requires 'local' and 'caching'!";
+    }
+    if(! this.remote.connected) {
+      return promising().fulfill();
+    }
+    var roots = this.caching.rootPaths.slice(0);
+    var n = roots.length, i = 0;
+    var aborted = false;
+    var rs = this;
+    return promising(function(promise) {
+      if(n == 0) {
+        rs._emit('sync-busy');
+        rs._emit('sync-done');
+        return promise.fulfill();
+      }
+      rs._emit('sync-busy');
+      var path;
+      while((path = roots.shift())) {
+        (function (path) {
+          //console.log('syncing '+path);
+          RemoteStorage.Sync.sync(rs.remote, rs.local, path, rs.caching.get(path)).
+            then(function() {
+              //console.log('syncing '+path+' success');
+              if(aborted) return;
+              i++;
+              if(n == i) {
+                rs._emit('sync-done');
+                promise.fulfill();
+              }
+            }, function(error) {
+              console.error('syncing', path, 'failed:', error);
+              if(aborted) return;
+              aborted = true;
+              rs._emit('sync-done');
+              if(error instanceof RemoteStorage.Unauthorized) {
+                rs._emit('error', error);
+              } else {
+                rs._emit('error', new SyncError(error));
+              }
+              promise.reject(error);
+            });
+        })(path);
+      }
+    });
+  };
+
+  RemoteStorage.SyncError = SyncError;
+
+  RemoteStorage.prototype.syncCycle = function() {
+    this.sync().then(function() {
+      this.stopSync();
+      this._syncTimer = setTimeout(this.syncCycle.bind(this), SYNC_INTERVAL);
+    }.bind(this),
+    function() {
+      console.log('sync error, retrying');
+      this.stopSync();
+      this._syncTimer = setTimeout(this.syncCycle.bind(this), SYNC_INTERVAL);
+    }.bind(this));
+  };
+
+  RemoteStorage.prototype.stopSync = function() {
+    if(this._syncTimer) {
+      clearTimeout(this._syncTimer);
+      delete this._syncTimer;
+    }
+  };
+
+  RemoteStorage.Sync._rs_init = function(remoteStorage) {
+    remoteStorage.on('ready', function() {
+      remoteStorage.syncCycle();
+    });
+  };
+
+  RemoteStorage.Sync._rs_cleanup = function(remoteStorage) {
+    remoteStorage.stopSync();
+  };
+
+})(this);
+
+
+/** FILE: src/indexeddb.js **/
+(function(global) {
+
+  /**
+   * Class: RemoteStorage.IndexedDB
+   *
+   *
+   * IndexedDB Interface
+   * -------------------
+   *
+   * This file exposes a get/put/delete interface, accessing data in an indexedDB.
+   *
+   * There are multiple parts to this interface:
+   *
+   *   The RemoteStorage integration:
+   *     - RemoteStorage.IndexedDB._rs_supported() determines if indexedDB support
+   *       is available. If it isn't, RemoteStorage won't initialize the feature.
+   *     - RemoteStorage.IndexedDB._rs_init() initializes the feature. It returns
+   *       a promise that is fulfilled as soon as the database has been opened and
+   *       migrated.
+   *
+   *   The storage interface (RemoteStorage.IndexedDB object):
+   *     - Usually this is accessible via "remoteStorage.local"
+   *     - #get() takes a path and returns a promise.
+   *     - #put() takes a path, body and contentType and also returns a promise.
+   *       In addition it also takes a 'incoming' flag, which indicates that the
+   *       change is not fresh, but synchronized from remote.
+   *     - #delete() takes a path and also returns a promise. It also supports
+   *       the 'incoming' flag described for #put().
+   *     - #on('change', ...) events, being fired whenever something changes in
+   *       the storage. Change events roughly follow the StorageEvent pattern.
+   *       They have "oldValue" and "newValue" properties, which can be used to
+   *       distinguish create/update/delete operations and analyze changes in
+   *       change handlers. In addition they carry a "origin" property, which
+   *       is either "window" or "remote". "remote" events are fired whenever the
+   *       "incoming" flag is passed to #put() or #delete(). This is usually done
+   *       by RemoteStorage.Sync.
+   *
+   *   The revision interface (also on RemoteStorage.IndexedDB object):
+   *     - #setRevision(path, revision) sets the current revision for the given
+   *       path. Revisions are only generated by the remotestorage server, so
+   *       this is usually done from RemoteStorage.Sync once a pending change
+   *       has been pushed out.
+   *     - #setRevisions(revisions) takes path/revision pairs in the form:
+   *       [[path1, rev1], [path2, rev2], ...] and updates all revisions in a
+   *       single transaction.
+   *     - #getRevision(path) returns the currently stored revision for the given
+   *       path.
+   *
+   *   The changes interface (also on RemoteStorage.IndexedDB object):
+   *     - Used to record local changes between sync cycles.
+   *     - Changes are stored in a separate ObjectStore called "changes".
+   *     - #_recordChange() records a change and is called by #put() and #delete(),
+   *       given the "incoming" flag evaluates to false. It is private andshould
+   *       never be used from the outside.
+   *     - #changesBelow() takes a path and returns a promise that will be fulfilled
+   *       with an Array of changes that are pending for the given path or below.
+   *       This is usually done in a sync cycle to push out pending changes.
+   *     - #clearChange removes the change for a given path. This is usually done
+   *       RemoteStorage.Sync once a change has successfully been pushed out.
+   *     - #setConflict sets conflict attributes on a change. It also fires the
+   *       "conflict" event.
+   *     - #on('conflict', ...) event. Conflict events usually have the following
+   *       attributes: path, localAction and remoteAction. Both actions are either
+   *       "PUT" or "DELETE". They also bring a "resolve" method, which can be
+   *       called with either of the strings "remote" and "local" to mark the
+   *       conflict as resolved. The actual resolution will usually take place in
+   *       the next sync cycle.
+   */
+
+  var RS = RemoteStorage;
+
+  var DEFAULT_DB_NAME = 'remotestorage';
+  var DEFAULT_DB;
+
+  function keepDirNode(node) {
+    return Object.keys(node.body).length > 0 ||
+      Object.keys(node.cached).length > 0;
+  }
+
+  function removeFromParent(nodes, path, key) {
+    var parts = path.match(/^(.*\/)([^\/]+\/?)$/);
+    if(parts) {
+      var dirname = parts[1], basename = parts[2];
+      nodes.get(dirname).onsuccess = function(evt) {
+        var node = evt.target.result;
+        if(!node) {//attempt to remove something from a non-existing directory
+          return;
+        }
+        delete node[key][basename];
+        if(keepDirNode(node)) {
+          nodes.put(node);
+        } else {
+          nodes.delete(node.path).onsuccess = function() {
+            if(dirname != '/') {
+              removeFromParent(nodes, dirname, key);
+            }
+          };
+        }
+      };
+    }
+  }
+
+  function makeNode(path) {
+    var node = { path: path };
+    if(path[path.length - 1] == '/') {
+      node.body = {};
+      node.cached = {};
+      node.contentType = 'application/json';
+    }
+    return node;
+  }
+
+  function addToParent(nodes, path, key) {
+    var parts = path.match(/^(.*\/)([^\/]+\/?)$/);
+    if(parts) {
+      var dirname = parts[1], basename = parts[2];
+      nodes.get(dirname).onsuccess = function(evt) {
+        var node = evt.target.result || makeNode(dirname);
+        node[key][basename] = true;
+        nodes.put(node).onsuccess = function() {
+          if(dirname != '/') {
+            addToParent(nodes, dirname, key);
+          }
+        };
+      };
+    }
+  }
+
+  RS.IndexedDB = function(database) {
+    this.db = database || DEFAULT_DB;
+    if(! this.db) {
+      if(RemoteStorage.LocalStorage) {
+        RemoteStorage.log("Failed to open indexedDB, falling back to localStorage");
+        return new RemoteStorage.LocalStorage();
+      } else {
+        throw "Failed to open indexedDB and localStorage fallback not available!";
+      }
+    }
+    RS.eventHandling(this, 'change', 'conflict');
+  };
+  RS.IndexedDB.prototype = {
+
+    get: function(path) {
+      var promise = promising();
+      var transaction = this.db.transaction(['nodes'], 'readonly');
+      var nodes = transaction.objectStore('nodes');
+      var nodeReq = nodes.get(path);
+      var node;
+      nodeReq.onsuccess = function() {
+        node = nodeReq.result;
+      };
+      transaction.oncomplete = function() {
+        if(node) {
+          promise.fulfill(200, node.body, node.contentType, node.revision);
+        } else {
+          promise.fulfill(404);
+        }
+      };
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    put: function(path, body, contentType, incoming) {
+      var promise = promising();
+      if(path[path.length - 1] == '/') { throw "Bad: don't PUT folders"; }
+      var transaction = this.db.transaction(['nodes'], 'readwrite');
+      var nodes = transaction.objectStore('nodes');
+      var oldNode;
+      var done;
+      nodes.get(path).onsuccess = function(evt) {
+        try {
+          oldNode = evt.target.result;
+          var node = {
+            path: path, contentType: contentType, body: body
+          };
+          nodes.put(node).onsuccess = function() {
+            try {
+              addToParent(nodes, path, 'body');
+            } catch(e) {
+              if(typeof(done) === 'undefined') {
+                done = true;
+                promise.reject(e);
+              }
+            };
+          };
+        } catch(e) {
+          if(typeof(done) === 'undefined') {
+            done = true;
+            promise.reject(e);
+          }
+        };
+      };
+      transaction.oncomplete = function() {
+        this._emit('change', {
+          path: path,
+          origin: incoming ? 'remote' : 'window',
+          oldValue: oldNode ? oldNode.body : undefined,
+          newValue: body
+        });
+        if(! incoming) {
+          this._recordChange(path, { action: 'PUT', revision: oldNode ? oldNode.revision : undefined });
+        }
+        if(typeof(done) === 'undefined') {
+          done = true;
+          promise.fulfill(200);
+        }
+      }.bind(this);
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    delete: function(path, incoming) {
+      var promise = promising();
+      if(path[path.length - 1] == '/') { throw "Bad: don't DELETE folders"; }
+      var transaction = this.db.transaction(['nodes'], 'readwrite');
+      var nodes = transaction.objectStore('nodes');
+      var oldNode;
+      nodes.get(path).onsuccess = function(evt) {
+        oldNode = evt.target.result;
+        nodes.delete(path).onsuccess = function() {
+          removeFromParent(nodes, path, 'body', incoming);
+        };
+      }
+      transaction.oncomplete = function() {
+        if(oldNode) {
+          this._emit('change', {
+            path: path,
+            origin: incoming ? 'remote' : 'window',
+            oldValue: oldNode.body,
+            newValue: undefined
+          });
+        }
+        if(! incoming) {
+          this._recordChange(path, { action: 'DELETE', revision: oldNode ? oldNode.revision : undefined });
+        }
+        promise.fulfill(200);
+      }.bind(this);
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    setRevision: function(path, revision) {
+      return this.setRevisions([[path, revision]]);
+    },
+
+    setRevisions: function(revs) {
+      var promise = promising();
+      var transaction = this.db.transaction(['nodes'], 'readwrite');
+      revs.forEach(function(rev) {
+        var nodes = transaction.objectStore('nodes');
+        nodes.get(rev[0]).onsuccess = function(event) {
+          var node = event.target.result || makeNode(rev[0]);
+          node.revision = rev[1];
+          nodes.put(node).onsuccess = function() {
+            addToParent(nodes, rev[0], 'cached');
+          };
+        };
+      });
+      transaction.oncomplete = function() {
+        promise.fulfill();
+      };
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    getRevision: function(path) {
+      var promise = promising();
+      var transaction = this.db.transaction(['nodes'], 'readonly');
+      var rev;
+      transaction.objectStore('nodes').
+        get(path).onsuccess = function(evt) {
+          if(evt.target.result) {
+            rev = evt.target.result.revision;
+          }
+        };
+      transaction.oncomplete = function() {
+        promise.fulfill(rev);
+      };
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    getCached: function(path) {
+      if(path[path.length - 1] != '/') {
+        return this.get(path);
+      }
+      var promise = promising();
+      var transaction = this.db.transaction(['nodes'], 'readonly');
+      var nodes = transaction.objectStore('nodes');
+      nodes.get(path).onsuccess = function(evt) {
+        var node = evt.target.result || {};
+        promise.fulfill(200, node.cached, node.contentType, node.revision);
+      };
+      return promise;
+    },
+
+    reset: function(callback) {
+      var dbName = this.db.name;
+      this.db.close();
+      var self = this;
+      RS.IndexedDB.clean(this.db.name, function() {
+        RS.IndexedDB.open(dbName, function(other) {
+          // hacky!
+          self.db = other.db;
+          callback(self);
+        });
+      });
+    },
+
+    fireInitial: function() {
+      var transaction = this.db.transaction(['nodes'], 'readonly');
+      var cursorReq = transaction.objectStore('nodes').openCursor();
+      cursorReq.onsuccess = function(evt) {
+        var cursor = evt.target.result;
+        if(cursor) {
+          var path = cursor.key;
+          if(path.substr(-1) != '/') {
+            this._emit('change', {
+              path: path,
+              origin: 'remote',
+              oldValue: undefined,
+              newValue: cursor.value.body
+            });
+          }
+          cursor.continue();
+        }
+      }.bind(this);
+    },
+
+    _recordChange: function(path, attributes) {
+      var promise = promising();
+      var transaction = this.db.transaction(['changes'], 'readwrite');
+      var changes = transaction.objectStore('changes');
+      var change;
+      changes.get(path).onsuccess = function(evt) {
+        change = evt.target.result || {};
+        change.path = path;
+        for(var key in attributes) {
+          change[key] = attributes[key];
+        }
+        changes.put(change);
+      };
+      transaction.oncomplete = promise.fulfill;
+      transaction.onerror = transaction.onabort = promise.reject;
+      return promise;
+    },
+
+    clearChange: function(path) {
+      var promise = promising();
+      var transaction = this.db.transaction(['changes'], 'readwrite');
+      var changes = transaction.objectStore('changes');
+      changes.delete(path);
+      transaction.oncomplete = function() {
+        promise.fulfill();
+      }
+      return promise;
+    },
+
+    changesBelow: function(path) {
+      var promise = promising();
+      var transaction = this.db.transaction(['changes'], 'readonly');
+      var cursorReq = transaction.objectStore('changes').
+        openCursor(IDBKeyRange.lowerBound(path));
+      var pl = path.length;
+      var changes = [];
+      cursorReq.onsuccess = function() {
+        var cursor = cursorReq.result;
+        if(cursor) {
+          if(cursor.key.substr(0, pl) == path) {
+            changes.push(cursor.value);
+            cursor.continue();
+          }
+        }
+      };
+      transaction.oncomplete = function() {
+        promise.fulfill(changes);
+      };
+      return promise;
+    },
+
+    setConflict: function(path, attributes) {
+      var event = { path: path };
+      for(var key in attributes) {
+        event[key] = attributes[key];
+      }
+      this._recordChange(path, { conflict: attributes }).
+        then(function() {
+          // fire conflict once conflict has been recorded.
+          if(this._handlers.conflict.length > 0) {
+            this._emit('conflict', event);
+          } else {
+            setTimeout(function() { event.resolve('remote'); }, 0);
+          }
+        }.bind(this));
+      event.resolve = function(resolution) {
+        if(resolution == 'remote' || resolution == 'local') {
+          attributes.resolution = resolution;
+          this._recordChange(path, { conflict: attributes });
+        } else {
+          throw "Invalid resolution: " + resolution;
+        }
+      }.bind(this);
+    },
+
+    closeDB: function() {
+      this.db.close();
+    }
+
+  };
+
+  var DB_VERSION = 2;
+  RS.IndexedDB.open = function(name, callback) {
+    var timer = setTimeout(function() {
+      callback("timeout trying to open db");
+    }, 3500);
+
+    var dbOpen = indexedDB.open(name, DB_VERSION);
+    dbOpen.onerror = function() {
+      console.error('opening db failed', dbOpen);
+      clearTimeout(timer);
+      callback(dbOpen.error);
+    };
+    dbOpen.onupgradeneeded = function(event) {
+      RemoteStorage.log("[IndexedDB] Upgrade: from ", event.oldVersion, " to ", event.newVersion);
+      var db = dbOpen.result;
+      if(event.oldVersion != 1) {
+        RemoteStorage.log("[IndexedDB] Creating object store: nodes");
+        db.createObjectStore('nodes', { keyPath: 'path' });
+      }
+      RemoteStorage.log("[IndexedDB] Creating object store: changes");
+      db.createObjectStore('changes', { keyPath: 'path' });
+    }
+    dbOpen.onsuccess = function() {
+      clearTimeout(timer);
+      callback(null, dbOpen.result);
+    };
+  };
+
+  RS.IndexedDB.clean = function(databaseName, callback) {
+    var req = indexedDB.deleteDatabase(databaseName);
+    req.onsuccess = function() {
+      RemoteStorage.log('done removing db');
+      callback();
+    };
+    req.onerror = req.onabort = function(evt) {
+      console.error('failed to remove database "' + databaseName + '"', evt);
+    };
+  };
+
+  RS.IndexedDB._rs_init = function(remoteStorage) {
+    var promise = promising();
+    RS.IndexedDB.open(DEFAULT_DB_NAME, function(err, db) {
+      if(err) {
+        if(err.name == 'InvalidStateError') {
+          // firefox throws this when trying to open an indexedDB in private browsing mode
+          var err = new Error("IndexedDB couldn't be opened.");
+          // instead of a stack trace, display some explaination:
+          err.stack = "If you are using Firefox, please disable\nprivate browsing mode.\n\nOtherwise please report your problem\nusing the link below";
+          remoteStorage._emit('error', err);
+        } else {
+        }
+      } else {
+        DEFAULT_DB = db;
+        db.onerror = function() { remoteStorage._emit('error', err); };
+        promise.fulfill();
+      }
+    });
+    return promise;
+  };
+
+  RS.IndexedDB._rs_supported = function() {
+    return 'indexedDB' in global;
+  }
+
+  RS.IndexedDB._rs_cleanup = function(remoteStorage) {
+    if(remoteStorage.local) {
+      remoteStorage.local.closeDB();
+    }
+    var promise = promising();
+    RS.IndexedDB.clean(DEFAULT_DB_NAME, function() {
+      promise.fulfill();
+    });
+    return promise;
+  }
+
+})(this);
+
+
+/** FILE: src/localstorage.js **/
+(function(global) {
+
+  var NODES_PREFIX = "remotestorage:cache:nodes:";
+  var CHANGES_PREFIX = "remotestorage:cache:changes:";
+
+  RemoteStorage.LocalStorage = function() {
+    RemoteStorage.eventHandling(this, 'change', 'conflict');
+  };
+
+  function makeNode(path) {
+    var node = { path: path };
+    if(path[path.length - 1] == '/') {
+      node.body = {};
+      node.cached = {};
+      node.contentType = 'application/json';
+    }
+    return node;
+  }
+
+  RemoteStorage.LocalStorage.prototype = {
+
+    get: function(path) {
+      var node = this._get(path);
+      if(node) {
+        return promising().fulfill(200, node.body, node.contentType, node.revision);
+      } else {
+        return promising().fulfill(404);
+      }
+    },
+
+    put: function(path, body, contentType, incoming) {
+      var oldNode = this._get(path);
+      var node = {
+        path: path, contentType: contentType, body: body
+      };
+      localStorage[NODES_PREFIX + path] = JSON.stringify(node);
+      this._addToParent(path);
+      this._emit('change', {
+        path: path,
+        origin: incoming ? 'remote' : 'window',
+        oldValue: oldNode ? oldNode.body : undefined,
+        newValue: body
+      });
+      if(! incoming) {
+        this._recordChange(path, { action: 'PUT' });
+      }
+      return promising().fulfill(200);
+    },
+
+    'delete': function(path, incoming) {
+      var oldNode = this._get(path);
+      delete localStorage[NODES_PREFIX + path];
+      this._removeFromParent(path);
+      if(oldNode) {
+        this._emit('change', {
+          path: path,
+          origin: incoming ? 'remote' : 'window',
+          oldValue: oldNode.body,
+          newValue: undefined
+        });
+      }
+      if(! incoming) {
+        this._recordChange(path, { action: 'DELETE' });
+      }
+      return promising().fulfill(200);
+    },
+
+    setRevision: function(path, revision) {
+      var node = this._get(path) || makeNode(path);
+      node.revision = revision;
+      localStorage[NODES_PREFIX + path] = JSON.stringify(node);
+      return promising().fulfill();
+    },
+
+    getRevision: function(path) {
+      var node = this._get(path);
+      return promising.fulfill(node ? node.revision : undefined);
+    },
+
+    _get: function(path) {
+      var node;
+      try {
+        node = JSON.parse(localStorage[NODES_PREFIX + path]);
+      } catch(e) { /* ignored */ }
+      return node;
+    },
+
+    _recordChange: function(path, attributes) {
+      var change;
+      try {
+        change = JSON.parse(localStorage[CHANGES_PREFIX + path]);
+      } catch(e) {
+        change = {};
+      }
+      for(var key in attributes) {
+        change[key] = attributes[key];
+      }
+      change.path = path;
+      localStorage[CHANGES_PREFIX + path] = JSON.stringify(change);
+    },
+
+    clearChange: function(path) {
+      delete localStorage[CHANGES_PREFIX + path];
+      return promising().fulfill();
+    },
+
+    changesBelow: function(path) {
+      var changes = [];
+      var kl = localStorage.length;
+      var prefix = CHANGES_PREFIX + path, pl = prefix.length;
+      for(var i=0;i<kl;i++) {
+        var key = localStorage.key(i);
+        if(key.substr(0, pl) == prefix) {
+          changes.push(JSON.parse(localStorage[key]));
+        }
+      }
+      return promising().fulfill(changes);
+    },
+
+    setConflict: function(path, attributes) {
+      var event = { path: path };
+      for(var key in attributes) {
+        event[key] = attributes[key];
+      }
+      this._recordChange(path, { conflict: attributes });
+      event.resolve = function(resolution) {
+        if(resolution == 'remote' || resolution == 'local') {
+          attributes.resolution = resolution;
+          this._recordChange(path, { conflict: attributes });
+        } else {
+          throw "Invalid resolution: " + resolution;
+        }
+      }.bind(this);
+      this._emit('conflict', event);
+    },
+
+    _addToParent: function(path) {
+      var parts = path.match(/^(.*\/)([^\/]+\/?)$/);
+      if(parts) {
+        var dirname = parts[1], basename = parts[2];
+        var node = this._get(dirname) || makeNode(dirname);
+        node.body[basename] = true;
+        localStorage[NODES_PREFIX + dirname] = JSON.stringify(node);
+        if(dirname != '/') {
+          this._addToParent(dirname);
+        }
+      }
+    },
+
+    _removeFromParent: function(path) {
+      var parts = path.match(/^(.*\/)([^\/]+\/?)$/);
+      if(parts) {
+        var dirname = parts[1], basename = parts[2];
+        var node = this._get(dirname);
+        if(node) {
+          delete node.body[basename];
+          if(Object.keys(node.body).length > 0) {
+            localStorage[NODES_PREFIX + dirname] = JSON.stringify(node);
+          } else {
+            delete localStorage[NODES_PREFIX + dirname];
+            if(dirname != '/') {
+              this._removeFromParent(dirname);
+            }
+          }
+        }
+      }
+    },
+
+    fireInitial: function() {
+      var l = localStorage.length, npl = NODES_PREFIX.length;
+      for(var i=0;i<l;i++) {
+        var key = localStorage.key(i);
+        if(key.substr(0, npl) == NODES_PREFIX) {
+          var path = key.substr(npl);
+          var node = this._get(path);
+          this._emit('change', {
+            path: path,
+            origin: 'remote',
+            oldValue: undefined,
+            newValue: node.body
+          });
+        }
+      }
+    }
+
+  };
+
+  RemoteStorage.LocalStorage._rs_init = function() {};
+
+  RemoteStorage.LocalStorage._rs_supported = function() {
+    return 'localStorage' in global;
+  };
+
+  RemoteStorage.LocalStorage._rs_cleanup = function() {
+    var l = localStorage.length;
+    var npl = NODES_PREFIX.length, cpl = CHANGES_PREFIX.length;
+    var remove = [];
+    for(var i=0;i<l;i++) {
+      var key = localStorage.key(i);
+      if(key.substr(0, npl) == NODES_PREFIX ||
+         key.substr(0, cpl) == CHANGES_PREFIX) {
+        remove.push(key);
+      }
+    }
+    remove.forEach(function(key) {
+      console.log('removing', key);
+      delete localStorage[key];
+    });
+  };
+
+})(this);
+
+
+/** FILE: src/modules.js **/
+(function() {
+
+  RemoteStorage.MODULES = {};
+
+  RemoteStorage.defineModule = function(moduleName, builder) {
+    RemoteStorage.MODULES[moduleName] = builder;
+
+    Object.defineProperty(RemoteStorage.prototype, moduleName, {
+      configurable: true,
+      get: function() {
+        var instance = this._loadModule(moduleName);
+        Object.defineProperty(this, moduleName, {
+          value: instance
+        });
+        return instance;
+      }
+    });
+
+    if(moduleName.indexOf('-') != -1) {
+      var camelizedName = moduleName.replace(/\-[a-z]/g, function(s) {
+        return s[1].toUpperCase();
+      });
+      Object.defineProperty(RemoteStorage.prototype, camelizedName, {
+        get: function() {
+          return this[moduleName];
+        }
+      });
+    }
+  };
+
+  RemoteStorage.prototype._loadModule = function(moduleName) {
+    var builder = RemoteStorage.MODULES[moduleName];
+    if(builder) {
+      var module = builder(new RemoteStorage.BaseClient(this, '/' + moduleName + '/'),
+                           new RemoteStorage.BaseClient(this, '/public/' + moduleName + '/'));
+      return module.exports;
+    } else {
+      throw "Unknown module: " + moduleName;
+    }
+  };
+
+  RemoteStorage.prototype.defineModule = function(moduleName) {
+    console.log("remoteStorage.defineModule is deprecated, use RemoteStorage.defineModule instead!");
+    RemoteStorage.defineModule.apply(RemoteStorage, arguments);
+  };
+
+})();
+
+
+/** FILE: src/debug/inspect.js **/
+(function() {
+  function loadTable(table, storage, paths) {
+    table.setAttribute('border', '1');
+    table.style.margin = '8px';
+    table.innerHTML = '';
+    var thead = document.createElement('thead');
+    table.appendChild(thead);
+    var titleRow = document.createElement('tr');
+    thead.appendChild(titleRow);
+    ['Path', 'Content-Type', 'Revision'].forEach(function(label) {
+      var th = document.createElement('th');
+      th.textContent = label;
+      thead.appendChild(th);
+    });
+
+    var tbody = document.createElement('tbody');
+    table.appendChild(tbody);
+
+    function renderRow(tr, path, contentType, revision) {
+      [path, contentType, revision].forEach(function(value) {
+        var td = document.createElement('td');
+        td.textContent = value || '';
+        tr.appendChild(td);
+      });      
+    }
+
+    function loadRow(path) {
+      if(storage.connected === false) return;
+      function processRow(status, body, contentType, revision) {
+        if(status == 200) {
+          var tr = document.createElement('tr');
+          tbody.appendChild(tr);
+          renderRow(tr, path, contentType, revision);
+          if(path[path.length - 1] == '/') {
+            for(var key in body) {
+              loadRow(path + key);
+            }
+          }
+        }
+      }
+      storage.get(path).then(processRow);
+    }
+
+    paths.forEach(loadRow);
+  }
+
+
+  function renderWrapper(title, table, storage, paths) {
+    var wrapper = document.createElement('div');
+    //wrapper.style.display = 'inline-block';
+    var heading = document.createElement('h2');
+    heading.textContent = title;
+    wrapper.appendChild(heading);
+    var updateButton = document.createElement('button');
+    updateButton.textContent = "Refresh";
+    updateButton.onclick = function() { loadTable(table, storage, paths); };
+    wrapper.appendChild(updateButton);
+    if(storage.reset) {
+      var resetButton = document.createElement('button');
+      resetButton.textContent = "Reset";
+      resetButton.onclick = function() {
+        storage.reset(function(newStorage) {
+          storage = newStorage;
+          loadTable(table, storage, paths);
+        });
+      };
+      wrapper.appendChild(resetButton);
+    }
+    wrapper.appendChild(table);
+    loadTable(table, storage, paths);
+    return wrapper;
+  }
+
+  function renderLocalChanges(local) {
+    var wrapper = document.createElement('div');
+    //wrapper.style.display = 'inline-block';
+    var heading = document.createElement('h2');
+    heading.textContent = "Outgoing changes";
+    wrapper.appendChild(heading);
+    var updateButton = document.createElement('button');
+    updateButton.textContent = "Refresh";
+    wrapper.appendChild(updateButton);
+    var list = document.createElement('ul');
+    list.style.fontFamily = 'courier';
+    wrapper.appendChild(list);
+
+    function updateList() {
+      local.changesBelow('/').then(function(changes) {
+        list.innerHTML = '';
+        changes.forEach(function(change) {
+          var el = document.createElement('li');
+          el.textContent = JSON.stringify(change);
+          list.appendChild(el);
+        });
+      });
+    }
+
+    updateButton.onclick = updateList;
+    updateList();
+    return wrapper;
+  }
+
+  RemoteStorage.prototype.inspect = function() {
+
+    var widget = document.createElement('div');
+    widget.id = 'remotestorage-inspect';
+    widget.style.position = 'absolute';
+    widget.style.top = 0;
+    widget.style.left = 0;
+    widget.style.background = 'black';
+    widget.style.color = 'white';
+    widget.style.border = 'groove 5px #ccc';
+
+    var controls = document.createElement('div');
+    controls.style.position = 'absolute';
+    controls.style.top = 0;
+    controls.style.left = 0;
+
+    var heading = document.createElement('strong');
+    heading.textContent = " remotestorage.js inspector ";
+
+    controls.appendChild(heading);
+
+    if(this.local) {
+      var syncButton = document.createElement('button');
+      syncButton.textContent = "Synchronize";
+      controls.appendChild(syncButton);
+    }
+
+    var closeButton = document.createElement('button');
+    closeButton.textContent = "Close";
+    closeButton.onclick = function() {
+      document.body.removeChild(widget);
+    }
+    controls.appendChild(closeButton);
+
+    widget.appendChild(controls);
+
+    var remoteTable = document.createElement('table');
+    var localTable = document.createElement('table');
+    widget.appendChild(renderWrapper("Remote", remoteTable, this.remote, this.caching.rootPaths));
+    if(this.local) {
+      widget.appendChild(renderWrapper("Local", localTable, this.local, ['/']));
+      widget.appendChild(renderLocalChanges(this.local));
+
+      syncButton.onclick = function() {
+        this.log('sync clicked');
+        this.sync().then(function() {
+          this.log('SYNC FINISHED');
+          loadTable(localTable, this.local, ['/'])
+        }.bind(this), function(err) {
+          console.error("SYNC FAILED", err, err.stack);
+        });
+      }.bind(this);
+    }
+
+    document.body.appendChild(widget);
+  };
+
+})();
+
+
+/** FILE: src/legacy.js **/
+
+(function() {
+  var util = {
+    getEventEmitter: function() {
+      var object = {};
+      var args = Array.prototype.slice.call(arguments);
+      args.unshift(object);
+      RemoteStorage.eventHandling.apply(RemoteStorage, args);
+      object.emit = object._emit;
+      return object;
+    },
+
+    extend: function(target) {
+      var sources = Array.prototype.slice.call(arguments, 1);
+      sources.forEach(function(source) {
+        for(var key in source) {
+          target[key] = source[key];
+        }
+      });
+      return target;
+    },
+
+    asyncEach: function(array, callback) {
+      return this.asyncMap(array, callback).
+        then(function() { return array; });
+    },
+
+    asyncMap: function(array, callback) {
+      var promise = promising();
+      var n = array.length, i = 0;
+      var results = [], errors = [];
+      function oneDone() {
+        i++;
+        if(i == n) {
+          promise.fulfill(results, errors);
+        }
+      }
+      array.forEach(function(item, index) {
+        try {
+          var result = callback(item);
+        } catch(exc) {
+          oneDone();
+          errors[index] = exc;
+        }
+        if(typeof(result) == 'object' && typeof(result.then) == 'function') {
+          result.then(function(res) { results[index] = res; oneDone(); },
+                      function(error) { errors[index] = res; oneDone(); });
+        } else {
+          oneDone();
+          results[index] = result;
+        }
+      });
+      return promise;
+    },
+
+    containingDir: function(path) {
+      var dir = path.replace(/[^\/]+\/?$/, '');
+      return dir == path ? null : dir;
+    },
+
+    isDir: function(path) {
+      return path.substr(-1) == '/';
+    },
+
+    baseName: function(path) {
+      var parts = path.split('/');
+      if(util.isDir(path)) {
+        return parts[parts.length-2]+'/';
+      } else {
+        return parts[parts.length-1];
+      }
+    },
+
+    bindAll: function(object) {
+      for(var key in this) {
+        if(typeof(object[key]) == 'function') {
+          object[key] = object[key].bind(object);
+        }
+      }
+    }
+  };
+
+  Object.defineProperty(RemoteStorage.prototype, 'util', {
+    get: function() {
+      console.log("DEPRECATION WARNING: remoteStorage.util is deprecated and will be removed with the next major release.");
+      return util;
+    }
+  });
+
+})();
+
+remoteStorage = new RemoteStorage();
diff --git a/demo/todo/js/tasks.js b/demo/todo/js/tasks.js
new file mode 100644 (file)
index 0000000..ce91ed9
--- /dev/null
@@ -0,0 +1,84 @@
+RemoteStorage.defineModule('tasks', function(privateClient, publicClient) {
+  function init() {
+    privateClient.cache('todos/', true);
+  }
+  function getUuid() {
+    var i, random, uuid = '';
+    for (i=0; i<32; i++) {
+      random = Math.random()*16 | 0;
+      if(i === 8 || i === 12 || i === 16 || i === 20) {
+        uuid += '-';
+      }
+      uuid += (i === 12 ? 4 : (i === 16 ? (random & 3 | 8) : random) ).toString(16);
+    }
+    return uuid;
+  }
+  function Todo(title) {
+    this.id = getUuid();
+    this.title = title;
+    this.completed = false;
+  }
+
+  return {
+    exports: {
+      init: init,
+      getTodos: function() {
+        return privateClient.getAll('todos/');
+      },
+      addTodo: function(text) {
+        var todo = new Todo(text);
+        privateClient.storeObject('todo-list-item', 'todos/'+todo.id, todo);
+      },
+      setTodo: function(id, todo) {
+        privateClient.storeObject('todo-list-item', 'todos/'+id, todo);
+      },
+      setTodoText: function(id, text) {
+        privateClient.getObject('todos/'+id).then(function(obj) {
+          obj.title = text;
+          privateClient.storeObject('todo-list-item', 'todos/'+id, obj);
+        }, function(err) {
+          console.log('error in setTodoText', err);
+        });
+      },
+      setTodoCompleted: function(id, value) {
+        privateClient.getObject('todos/'+id).then(function(obj) {
+          obj.completed = value;
+          privateClient.storeObject('todo-list-item', 'todos/'+id, obj);
+        }, function(err) {
+          console.log('error in setTodoCompleted', err);
+        });
+      },
+      setAllTodosCompleted: function(value) {
+        privateClient.getAll('todos/').then(function(objs) {
+          for(var i in objs) {
+            if(objs[i].completed != value) {
+              objs[i].completed = value;
+              privateClient.storeObject('todo-list-item', 'todos/'+i, objs[i]);
+            }
+          }
+        }, function(err) {
+          console.log('error in setAllTodosCompleted', err);
+        });
+      },
+      removeTodo: function(id) {
+        privateClient.remove('todos/'+id);
+      },
+      removeAllCompletedTodos: function() {
+        privateClient.getAll('todos/').then(function(objs) {
+          for(var i in objs) {
+            if(objs[i].completed) {
+              privateClient.remove('todos/'+i);
+            }
+          }
+        }, function(err) {
+          console.log('error in removeAllCompletedTodos', err);
+        });
+      },
+      onChange: function( cb ) {
+        privateClient.on('change', function(event) {
+          cb(event.oldValue, event.newValue);
+        });
+      }
+    }
+  };
+});
diff --git a/demo/todo/license.md b/demo/todo/license.md
new file mode 100644 (file)
index 0000000..9275d43
--- /dev/null
@@ -0,0 +1,20 @@
+Everything in this repo is MIT License unless otherwise specified.
+
+Copyright (c) Addy Osmani, Sindre Sorhus, Pascal Hartig, Stephen Sawchuk.
+
+Permission is hereby granted, free of charge, to any person obtaining a copy of
+this software and associated documentation files (the "Software"), to deal in
+the Software without restriction, including without limitation the rights to
+use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
+the Software, and to permit persons to whom the Software is furnished to do so,
+subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all
+copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
+FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
+COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
+IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
+CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
diff --git a/demo/todo/readme.md b/demo/todo/readme.md
new file mode 100644 (file)
index 0000000..ee39598
--- /dev/null
@@ -0,0 +1,157 @@
+# ![TodoMVC](https://raw.github.com/tastejs/todomvc/gh-pages/media/logo.png)
+
+## Helping you select an MV\* framework
+
+
+Developers these days are spoiled with choice when it comes to selecting an MV\* framework for structuring and organizing JavaScript web apps.
+
+Backbone, Ember, AngularJS, Spine... the list of new and stable solutions goes on and on, but just how do you decide on which to use in a sea of so many options?
+
+To help solve this problem, we created TodoMVC - a project which offers the same Todo application implemented using MV* concepts in most of the popular JavaScript MV\* frameworks of today.
+
+
+#### Todo apps are included for:
+
+- [Backbone.js](http://documentcloud.github.com/backbone)
+- [Ember.js](http://emberjs.com)
+- [AngularJS](http://angularjs.org)
+- [Spine](http://spinejs.com)
+- [KnockoutJS](http://knockoutjs.com) (MVVM)
+- [Dojo](http://dojotoolkit.org)
+- [YUI](http://yuilibrary.com)
+- [Batman.js](http://batmanjs.org)
+- [Closure](http://code.google.com/closure/library/)
+- [Agility.js](http://agilityjs.com)
+- [Knockback.js](http://kmalakoff.github.com/knockback)
+- [Google Web Toolkit](https://developers.google.com/web-toolkit/)
+- [Flight by Twitter](https://github.com/twitter/flight)
+
+
+###### Non-framework Implementations
+
+- [jQuery](http://jquery.com)
+- [Vanilla JS](http://todomvc.com/vanilla-examples/vanillajs/)
+
+###### RequireJS
+
+- [Backbone.js](http://documentcloud.github.com/backbone) + [RequireJS](http://requirejs.org)
+- [AngularJS](http://angularjs.org) + [RequireJS](http://requirejs.org)
+- [KnockoutJS](http://knockoutjs.com) + [RequireJS](http://requirejs.org)
+- [CanJS](http://canjs.us) + [RequireJS](http://requirejs.org)
+
+#### Labs
+
+We also have a number of in-progress applications in Labs:
+
+- [Ractive](http://ractivejs.org)
+- [React](http://facebook.github.io/react)
+- [Meteor](http://meteor.com)
+- [Derby](http://derbyjs.com)
+- [CanJS](http://canjs.us)
+- [Kendo UI](http://kendoui.com)
+- [Maria.js](https://github.com/petermichaux/maria)
+- [cujoJS](http://cujojs.com)
+- [MarionetteJS](http://marionettejs.com/)
+- [SocketStream](http://www.socketstream.org) + [jQuery](http://jquery.com)
+- [Ext.js](http://www.sencha.com/products/extjs)
+- [Sammy.js](http://sammyjs.org)
+- [dermis](https://github.com/wearefractal/dermis)
+- [Stapes.js](http://hay.github.com/stapes)
+- [Epitome](http://dimitarchristoff.github.com/Epitome)
+- [TroopJS](https://github.com/troopjs)
+- [soma.js](http://somajs.github.com/somajs)
+- [DUEL](https://bitbucket.org/mckamey/duel/wiki/Home)
+- [Olives](https://github.com/flams/olives)
+- [PlastronJS](https://github.com/rhysbrettbowen/PlastronJS)
+- [Dijon](https://github.com/creynders/dijon-framework)
+- [rAppid.js](http://www.rappidjs.com)
+- [Thorax](http://walmartlabs.github.com/thorax/)
+- [KnockoutJS](http://knockoutjs.com) + [ClassBindingProvider](https://github.com/rniemeyer/knockout-classBindingProvider) (using Ryan Niemeyer's Class Binding Provider)
+- [KnockoutJS](http://knockoutjs.com) + [RequireJS](http://requirejs.org) (using AMD)
+- [AngularJS](http://angularjs.org) + [RequireJS](http://requirejs.org) (using AMD)
+- [CanJS](http://canjs.us) + [RequireJS](http://requirejs.org) (using AMD)
+- [AngularJS](http://angularjs.org) (optimized)
+- [Backbone.xmpp](https://github.com/ggozad/Backbone.xmpp)
+- [Dart](http://dartlang.org)
+- [Chaplin](http://chaplinjs.org) + [Brunch](http://brunch.io)
+- [Serenade.js](https://github.com/elabs/serenade.js)
+- [TypeScript](http://www.typescriptlang.org/)
+- [DeftJS](http://deftjs.org/)
+- [Aria Templates](http://ariatemplates.com/)
+- [Enyo + Backbone.js](http://enyojs.com/)
+- [SAPUI5](http://scn.sap.com/community/developer-center/front-end)
+- [Lavaca](http://getlavaca.com) + [RequireJS](http://requirejs.org) (using AMD)
+- [ComponentJS](http://componentjs.com)
+
+## Live demos
+
+Live demos are available on our [website](http://todomvc.com)
+
+## View & Run in Web IDE
+
+Click on the button below to view the code in a Web IDE. Feel free to edit the code and then run it all from your browser.
+
+[![IDE](https://raw.github.com/tastejs/todomvc/gh-pages/site-assets/editcloud9.png)](https://c9.io/open/git/?url=git://github.com/tastejs/todomvc.git)
+[![IDE](https://codio-public.s3.amazonaws.com/sharing/demo-in-ide.png)](https://codio.com/p/create/?from_github=tastejs/todomvc)
+
+
+## Screenshot
+
+![screenshot](https://raw.github.com/tastejs/todomvc/gh-pages/site-assets/screenshot.png)
+
+
+## Team
+
+TodoMVC would not be possible without a strong team of [contributors](https://github.com/tastejs/todomvc/contributors) helping push the project forward each day. In addition, we have a core project team composed of:
+
+#### [Addy Osmani](http://github.com/addyosmani) - Founder/Lead
+
+<img align="left" width="40" height="40" src="http://www.gravatar.com/avatar/96270e4c3e5e9806cf7245475c00b275.png?s=40">
+Addy is a Developer Platform Engineer at Google who originally created TodoMVC. He oversees the project direction, drives expansion and helps lead core development with Sindre Sorhus (by far our most active contributor!).
+
+#### [Sindre Sorhus](https://github.com/sindresorhus) - Lead Developer
+
+<img align="left" width="40" height="40" src="http://www.gravatar.com/avatar/d36a92237c75c5337c17b60d90686bf9.png?s=40">
+Sindre is a Web Developer who drives core development, quality control and application design for the project. His contributions have helped us ensure consistency and best practices are enforced wherever possible.
+
+#### [Pascal Hartig](https://github.com/passy) - Developer
+
+<img align="left" width="40" height="40" src="http://www.gravatar.com/avatar/be451fcdbf0e5ff07f23ed16cb5c90a3.png?s=40">
+Pascal is a Front-end Engineer at Twitter with a deep passion for consistency. He watches pull requests and helps developers getting their contributions integrated with TodoMVC.
+
+#### [Stephen Sawchuk](https://github.com/stephenplusplus) - Developer
+
+<img align="left" width="40" height="40" src="https://secure.gravatar.com/avatar/098cfe2d360e77c3229f2cd5298354c4?s=40">
+Stephen is a Front-end Engineer at Quicken Loans that cares about improving the maintainability and developer experience of open-source projects. His recent contributions include helping us move all apps over to using Bower and implementing the new information bar.
+
+#### [Colin Eberhardt](https://github.com/colineberhardt) - Developer
+
+<img align="left" width="40" height="40" src="https://secure.gravatar.com/avatar/73bba00b41ff1c9ecc3ee29487bace7d?s=40">
+Colin is a software consultant at Scott Logic who is passionate about all software - from JavaScript to Java, and C# to Objective-C. His recent contribution to the project has been a fully automated test suite.
+
+#### [Gianni Chiappetta](https://github.com/gf3) - Logo designer
+
+<img align="left" width="40" height="40" src="http://www.gravatar.com/avatar/4b0209ae3652cc5a7d53545e759fbe39.png?s=40">
+Gianni is a programmer and designer currently working as the Chief Rigger at MetaLab.
+
+## Disclaimer
+
+<img align="right" width="230" height="230" src="https://raw.github.com/tastejs/todomvc/gh-pages/media/icon-small.png">
+
+TodoMVC has been called many things including the 'Speed-dating' and 'Rosetta Stone' of MV* frameworks. Whilst we hope that this project is able to offer assistance in deciding what frameworks are worth spending more time looking at, remember that the Todo application offers a limited view of what a framework may be capable of.
+
+It is meant to be used as a gateway to reviewing how a basic application using a framework may be structured and we heavily recommend investing time researching a solution in more depth before opting to use it.
+
+
+## Getting Involved
+
+Whilst we enjoy implementing and improving existing Todo apps, we're always interested in speaking to framework authors (and users) wishing to share Todo app implementations in their framework/solution of choice.
+
+Check out our [contribution docs](contributing.md) for more info.
+
+
+## License
+
+Everything in this repo is MIT License unless otherwise specified.
+
+MIT © Addy Osmani, Sindre Sorhus, Pascal Hartig, Stephen Sawchuk.
index 594fcdf..762c435 100644 (file)
@@ -1,5 +1,5 @@
 ### BEGIN INIT INFO
-# Provides:          krs-serve
+# Provides:          krsd
 # Required-Start:    networking
 # Required-Stop:     networking
 # Default-Start:     2 3 4 5
 # Using the lsb functions to perform the operations.
 . /lib/lsb/init-functions
 # Process name ( For display )
-NAME=krs-serve
+NAME=krsd
 # Daemon name, where is the actual executable
-DAEMON=/usr/bin/krs-serve
+DAEMON=/usr/bin/krsd
 # pid file for the daemon
-PIDFILE=/var/run/krs-serve.pid
+PIDFILE=/var/run/krsd.pid
 
 # If the daemon is not there, then exit.
 test -x $DAEMON || exit 5
 
-. /etc/default/krs-serve
+. /etc/default/krsd
 
 case $1 in
  start)
index ad17e3f..ae281cb 100644 (file)
@@ -28,7 +28,9 @@ static void print_help(const char *progname) {
           "                                  future output will be lost.\n"
           "  --dir=<directory-name>        - Name of the directory relative to the user's\n"
           "                                  home directory to serve data from.\n"
-          "                                  Defaults to: storage\n"
+          "                                  Defaults to: remotestorage\n"
+          "  --static=<directory>          - Directory from which to serve static files.\n"
+         "                                  Defaults to: /var/www/krsd\n"
           "  --pid-file=<file>             - Write PID to given file.\n"
           "  --stop                        - Stop a running rs-serve process. The process\n"
           "                                  is identified by the PID file specified via\n"
@@ -71,6 +73,8 @@ FILE *rs_pid_file = NULL;
 char *rs_pid_file_path = NULL;
 char *rs_home_serve_root = NULL;
 int rs_home_serve_root_len = 0;
+char *rs_static_dir = NULL;
+int rs_static_dir_len = 0;
 int rs_stop_other = 0;
 char *rs_auth_uri = NULL;
 int rs_auth_uri_len = 0;
@@ -95,6 +99,7 @@ static struct option long_options[] = {
   { "detach", no_argument, 0, 'd' },
   { "help", no_argument, 0, 'h' },
   { "version", no_argument, 0, 'v' },
+  { "static", required_argument, 0, 0 },
 #if 0
   { "auth-uri", required_argument, 0, 0 },
 #endif
@@ -186,6 +191,14 @@ void init_config(int argc, char **argv) {
           rs_home_serve_root[--len] = 0;
         }
         rs_home_serve_root_len = len;
+      } else if(strcmp(arg_name, "static") == 0) { // --static=<dirname>
+        rs_static_dir = optarg;
+        int len = strlen(rs_static_dir);
+        if(rs_static_dir[len - 1] == '/') {
+          // strip trailing slash.
+          rs_static_dir[--len] = 0;
+        }
+        rs_static_dir = len;
 #if 0
       } else if(strcmp(arg_name, "auth-uri") == 0) { // --auth-uri=<uri-template>
         rs_auth_uri = optarg;
@@ -209,8 +222,13 @@ void init_config(int argc, char **argv) {
   }
 
   if(rs_home_serve_root == NULL) {
-    rs_home_serve_root = "storage";
-    rs_home_serve_root_len = 7;
+    rs_home_serve_root = "remotestorage";
+    rs_home_serve_root_len = 13;
+  }
+
+  if(rs_static_dir == NULL) {
+    rs_static_dir = "/var/www/krsd";
+    rs_static_dir_len = 13;
   }
 
   if(rs_stop_other) {
index e57633a..c18cbe5 100644 (file)
@@ -86,6 +86,11 @@ extern char *rs_home_serve_root;
 extern int rs_home_serve_root_len;
 #define RS_HOME_SERVE_ROOT_LEN rs_home_serve_root_len
 
+extern char *rs_static_dir;
+#define RS_STATIC_DIR rs_static_dir
+extern int rs_static_dir_len;
+#define RS_STATIC_DIR_LEN rs_static_dir_len
+
 extern struct rs_header rs_default_headers;
 #define RS_DEFAULT_HEADERS rs_default_headers
 
index 9b6553d..92dc8c1 100644 (file)
 #include <alloca.h>
 
 #include <sys/signalfd.h>
-#include <gssapi.h>
 
 #include "rs-serve.h"
 
 #define IS_READ(r) (r->method == htp_method_GET || r->method == htp_method_HEAD)
 
-// SPNEGO mechanism is 1.3.6.1.5.5.2 -- tag 0x06, lenght 6, 1.3 packed funny:
+// SPNEGO mechanism is 1.3.6.1.5.5.2 -- tag 0x06, length 6, 1.3 packed funny:
 // uint8_t OID_SPNEGO_bytes [] = { 1*40+3, 6, 1, 5, 5, 2 };
 // const gss_OID_desc OID_SPNEGO = {
 //     .length = 6,
@@ -76,13 +75,17 @@ int b64_decode (gss_buffer_t out, const char *in) {
                uint32_t digit = (uint32_t) b64_decode_table [(uint8_t) *in];
                if (digit <= 63) {
                        block |= digit << shift;
-                       shift -= 6;
                        in++;
+               } else if (shift == 3*6) {
+                       break;
                }
+               shift -= 6;
                if (shift < 0) {
-                       ((uint8_t *) out->value) [out->length++] = digit >> 16;
-                       ((uint8_t *) out->value) [out->length++] = digit >>  8;
-                       ((uint8_t *) out->value) [out->length++] = digit      ;
+                       ((uint8_t *) out->value) [out->length++] = block >> 16;
+                       ((uint8_t *) out->value) [out->length++] = block >>  8;
+                       ((uint8_t *) out->value) [out->length++] = block      ;
+                       //DEBUG// log_debug("%02x %02x %02x", (block >> 16) & 0x00ff, (block >> 8) & 0x00ff, block & 0x00ff);
+                       block = 0;
                        if (digit <= 63) {
                                shift = 3*6;
                        }
@@ -97,19 +100,63 @@ int b64_decode (gss_buffer_t out, const char *in) {
        return 0;
 }
 
+static uint8_t b64_encode_table [64] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
 
-int authorize_request(evhtp_request_t *req) {
+int b64_encode (char *out, gss_buffer_t in) {
+       int len64 = 0;
+       int pos256 = 0;
+       int totalbits = in->length * 4;
+       uint32_t chunk;
+       while (3 * len64 < totalbits) {
+               int maskpos = len64 & 0x03;
+               if (maskpos == 0) {
+                       chunk = (((unsigned char *) in->value) [pos256+0] << 16) |
+                               (((unsigned char *) in->value) [pos256+1] <<  8) |
+                                ((unsigned char *) in->value) [pos256+2];
+                       pos256 += 3;
+               }
+               *out++ = b64_encode_table [0x3f & (chunk >> (18 - maskpos * 6))];
+               len64++;
+       }
+       switch (in->length % 3) {
+       case 1:
+               // Append "=="
+               out [len64++] = '=';
+               // ...continue into next case...
+       case 2:
+               // Append "="
+               out [len64++] = '=';
+               // ...continue into next case...
+       case 0:
+               // Append no '=' characters
+               break;
+       }
+       out [len64] = '\0';
+       return len64;
+}
+
+
+int authorize_request(evhtp_request_t *req, gss_buffer_t username) {
 #if 0
   char *username = REQUEST_GET_USER(req);
 #endif
   const char *auth_header = evhtp_header_find(req->headers_in, "Authorization");
-  evhtp_header_t *wwwauth;
+  evhtp_header_t *wwwauth = NULL;
   gss_ctx_id_t ctxh = GSS_C_NO_CONTEXT;
+  gss_cred_id_t deleg = GSS_C_NO_CREDENTIAL;
+  OM_uint32 flgs = 0;
   int b64len;
-  gss_buffer_desc gssbuf;
-  gss_buffer_desc gssout;
+  gss_buffer_desc gssbuf = GSS_C_EMPTY_BUFFER;
+  gss_buffer_desc gssout = GSS_C_EMPTY_BUFFER;
+  gss_name_t intname = GSS_C_NO_NAME;
+  gss_OID mech_oid;
   OM_uint32 major, minor;
-  log_debug("Got auth header: %s", auth_header);
+  if(auth_header == NULL) {
+    log_debug("Got no auth header to work on; requesting SPNEGO");
+    ADD_RESP_HEADER(req, "WWW-Authenticate", "Negotiate");
+    return -1;
+  }
+  //DEBUG// log_debug("Got auth header: %s", auth_header);
   const char *token;
   int retval = 0;
   if(auth_header) {
@@ -122,35 +169,62 @@ int authorize_request(evhtp_request_t *req) {
       }
       gssbuf.length = (b64len * 3 + 3) >> 2; /* Perhaps slightly too much */
       gssbuf.value = alloca (gssbuf.length);   /* No NULL return */
-      log_debug("Got SPNEGO token: %s", token);
+      log_debug("Got SPNEGO token of size %d: %s", strlen (token), token);
       if (b64_decode (&gssbuf, token) < 0) {
        log_error("Rejecting faulty base64 coding in SPNEGO token.");
        return -2;
       }
+      log_debug("SPNEGO decoded length: %d (should be around %f minus trailing '=' signs)", gssbuf.length, strlen (token) * 6.0 / 8.0);
       //TODO// Sessions are _only_ needed for additional client authentication
       //TODO// So skip: Was a usable GSSAPI context already created?
-      major = gss_accept_sec_context (&minor, &ctxh, GSS_C_NO_CREDENTIAL, &gssbuf, GSS_C_NO_CHANNEL_BINDINGS, NULL, NULL /**NONEED:OUTPUT** OID_SPNEGO */, &gssout, NULL /**TODO** &GSS_C_REPLAY_FLAG */, NULL, NULL /**TODO** or lure credential into this varptr with &GSS_C_DELEG_FLAG? */);
-      //TODO// Supply token to session's GSSAPI accept environment
+      major = gss_accept_sec_context (
+                       &minor,
+                       &ctxh,
+                       GSS_C_NO_CREDENTIAL,    //MODKRB: server_creds (with that value)
+                       &gssbuf,
+                       GSS_C_NO_CHANNEL_BINDINGS,
+                       &intname,
+                       &mech_oid,
+                       &gssout,
+                       &flgs,
+                       NULL,
+                       &deleg);
       if (major != GSS_S_COMPLETE) {
         if (major == GSS_S_CONTINUE_NEEDED) {
          log_error("GSSAPI requires continued negotiation, which is not supported.");
          retval = -2;
         } else {
-         log_error("GSSAPI returns error code %d and minor %d.", major, minor);
+         log_error("GSSAPI initiation of security context returns error code %d and minor %d.", major, minor);
          retval = -1;
         }
       }
-      if (gssout.length > 0) {
+      if ((major == GSS_S_CONTINUE_NEEDED) && (gssout.length > 0)) {
        if (retval == 0) {
-          //TODO// The construct WWW-Authenticate header for the 200 reply
-         log_debug("GSSAPI wants to send data back, which is not supported yet.");
+         // We know that major == GSS_S_COMPLETE (otherwise retval < 0)
+         char *hdrval = malloc (11 + gssout.length * 4 / 3 + 3 + 1);
+         if(hdrval) {
+           memcpy (hdrval,      "Negotiate ", 10);
+           b64_encode (hdrval + 10, &gssout);
+           evhtp_headers_add_header(req->headers_out,
+                       evhtp_header_new ("WWW-Authenticate", hdrval, 0, 1));
+           free (hdrval);
+         }
+         gss_release_buffer (NULL, &gssout);
        }
-       gss_release_buffer (NULL, &gssout);
        /* Keep retval == 0 --> the client may fail, but we are satisfied */
       }
       if (ctxh != GSS_C_NO_CONTEXT) {
        gss_delete_sec_context (NULL, &ctxh, GSS_C_NO_BUFFER);
       }
+      /* Pickup opaque intname handle, map it to a gss_buffer_t string */
+      if ((major == GSS_S_COMPLETE) && (retval == 0)) {
+       major = gss_display_name (&minor, intname, username, NULL);
+       if (major != GSS_S_COMPLETE) {
+         log_error("GSSAPI display name returns error code %d and minor %d.", major, minor);
+         return -2;
+       }
+       log_debug("GSSAPI accepted credential named %.*s.", username->length, username->value);
+      }
       //TODO// Accepted and no sessions?  Then report success right now
       if (major == GSS_S_COMPLETE) {
        return 0;
index 0e11872..ea3a708 100644 (file)
@@ -13,7 +13,7 @@
 #ifndef RS_AUTH_H
 #define RS_AUTH_H
 
-int authorize_request(evhtp_request_t *req);
+int authorize_request(evhtp_request_t *req, gss_buffer_desc *user);
 
 #endif /* !RS_AUTH_H */
 
index 11f45df..a43feff 100644 (file)
@@ -19,6 +19,7 @@ static void add_cors_headers(evhtp_request_t *req) {
   ADD_RESP_HEADER(req, "Access-Control-Expose-Headers", RS_EXPOSE_HEADERS);
 }
 
+#if 0
 static void verify_user(evhtp_request_t *req) {
   char *username = REQUEST_GET_USER(req);
   uid_t uid = user_get_uid(username);
@@ -33,27 +34,32 @@ static void verify_user(evhtp_request_t *req) {
     log_debug("User found: %s (uid: %ld)", username, uid);
   }
 }
+#endif
 
 void dispatch_storage(evhtp_request_t *req, void *arg) {
   req->status = 0;
 
   do {
+    gss_buffer_desc username = GSS_C_EMPTY_BUFFER;
 
+    // TODO: It remains to be seen if CORS works with Kerberos authentication
     add_cors_headers(req);
 
+#if 0
     // validate user
     verify_user(req);
 
     if(req->status) break; // bail
+#endif
 
     // authorize request
     if(req->method != htp_method_OPTIONS) {
-      int auth_result = authorize_request(req);
+      int auth_result = authorize_request(req, &username);
       if(auth_result == 0) {
-        log_debug("Request authorized.");
+        log_info("Requester authenticated as %.*s.", username.length, username.value);
       } else if(auth_result == -1) {
         log_info("Request NOT authorized.");
-        //OK/Negotiate// req->status = EVHTP_RES_UNAUTH;
+        req->status = EVHTP_RES_UNAUTH;
       } else if(auth_result == -2) {
         log_error("An error occured while authorizing request.");    
         req->status = EVHTP_RES_SERVERR; 
@@ -69,22 +75,31 @@ void dispatch_storage(evhtp_request_t *req, void *arg) {
         req->status = EVHTP_RES_NOCONTENT;
         break;
       case htp_method_GET:
-        req->status = storage_handle_get(req);
+        req->status = storage_handle_get(req, &username);
         break;
       case htp_method_HEAD:
-        req->status = storage_handle_head(req);
+        req->status = storage_handle_head(req, &username);
         break;
       case htp_method_PUT:
-        req->status = storage_handle_put(req);
+        req->status = storage_handle_put(req, &username);
         break;
       case htp_method_DELETE:
-        req->status = storage_handle_delete(req);
+        req->status = storage_handle_delete(req, &username);
         break;
       default:
         req->status = EVHTP_RES_METHNALLOWED;
       }
     }
 
+    // Cleanup username
+    if (username.value != NULL) {
+      OM_uint32 major, minor;
+      major = gss_release_buffer (&minor, &username);
+      if (major != GSS_S_COMPLETE) {
+       log_error("GSSAPI cleanup of username buffer returns error code %d and minor %d.", major, minor);
+      }
+    }
+
   } while(0);
 
   // send reply, if status was set
index 01501ff..d3579bd 100644 (file)
  *
  */
 
-static char *make_disk_path(char *user, char *path, char **storage_root);
+static char *make_disk_path(char *dom_user, char *path, gss_buffer_t authuser, char **storage_root);
 static evhtp_res serve_directory(evhtp_request_t *request, char *disk_path,
                                  struct stat *stat_buf);
 static evhtp_res serve_file_head(evhtp_request_t *request_t, char *disk_path,
                            struct stat *stat_buf,const char *mime_type);
 static evhtp_res serve_file(evhtp_request_t *request, const char *disk_path,
                       struct stat *stat_buf);
-static evhtp_res handle_get_or_head(evhtp_request_t *request, int include_body);
+static evhtp_res handle_get_or_head(evhtp_request_t *request, gss_buffer_t authuser, int include_body);
 
-evhtp_res storage_handle_head(evhtp_request_t *request) {
+evhtp_res storage_handle_head(evhtp_request_t *request, gss_buffer_t authuser) {
   if(RS_EXPERIMENTAL) {
-    return handle_get_or_head(request, 0);
+    return handle_get_or_head(request, authuser, 0);
   } else {
     return EVHTP_RES_METHNALLOWED;
   }
 }
 
-evhtp_res storage_handle_get(evhtp_request_t *request) {
+evhtp_res storage_handle_get(evhtp_request_t *request, gss_buffer_t authuser) {
   log_debug("storage_handle_get()");
-  return handle_get_or_head(request, 1);
+  return handle_get_or_head(request, authuser, 1);
 }
 
-evhtp_res storage_handle_put(evhtp_request_t *request) {
+evhtp_res storage_handle_put(evhtp_request_t *request, gss_buffer_t authuser) {
   log_debug("HANDLE PUT");
 
   if(request->uri->path->file == NULL) {
@@ -54,6 +54,7 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
   char *storage_root = NULL;
   char *disk_path = make_disk_path(REQUEST_GET_USER(request),
                                    REQUEST_GET_PATH(request),
+                                  authuser,
                                    &storage_root);
   if(disk_path == NULL) {
     return EVHTP_RES_SERVERR;
@@ -87,6 +88,7 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
 
   } while(0);
 
+#if 0
   // look up uid and gid of current user, so we can chown() correctly.
   uid_t uid;
   gid_t gid;
@@ -102,6 +104,7 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
       return EVHTP_RES_SERVERR;
     }
   } while(0);
+#endif
 
   // create parent directories
   do {
@@ -154,9 +157,11 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
           return EVHTP_RES_SERVERR;
         }
 
+#if 0
         if(fchownat(dirfd, dir_name, uid, gid, AT_SYMLINK_NOFOLLOW) != 0) {
           log_warn("failed to chown() newly created directory: %s", strerror(errno));
         }
+#endif
       }
       prevfd = dirfd;
       dirfd = openat(prevfd, dir_name, O_RDONLY);
@@ -188,9 +193,11 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
   }
 
   if(! exists) {
+#if 0
     if(fchown(fd, uid, gid) != 0) {
       log_warn("Failed to chown() newly created file: %s", strerror(errno));
     }
+#endif
   }
 
   // write buffered data
@@ -230,7 +237,7 @@ evhtp_res storage_handle_put(evhtp_request_t *request) {
   return exists ? EVHTP_RES_OK : EVHTP_RES_CREATED;
 }
 
-evhtp_res storage_handle_delete(evhtp_request_t *request) {
+evhtp_res storage_handle_delete(evhtp_request_t *request, gss_buffer_t authuser) {
 
   if(request->uri->path->file == NULL) {
     // DELETE to directories aren't allowed
@@ -240,6 +247,7 @@ evhtp_res storage_handle_delete(evhtp_request_t *request) {
   char *storage_root = NULL;
   char *disk_path = make_disk_path(REQUEST_GET_USER(request),
                                    REQUEST_GET_PATH(request),
+                                  authuser,
                                    &storage_root);
   if(disk_path == NULL) {
     return EVHTP_RES_SERVERR;
@@ -485,28 +493,66 @@ static evhtp_res serve_file(evhtp_request_t *request, const char *disk_path, str
   return EVHTP_RES_OK;
 }
 
-static char *make_disk_path(char *user, char *path, char **storage_root) {
+static char *make_disk_path(char *dom_user, char *path, gss_buffer_t authuser, char **storage_root) {
 
   // FIXME: use passwd->pwdir instead of /home/{user}/
 
   // calculate maximum length of path
-  int pathlen = ( strlen(user) + strlen(path) +
+  int pathlen = ( strlen(dom_user) + strlen(path) +
                   6 + // "/home/"
                   1 + // another slash
                   RS_HOME_SERVE_ROOT_LEN );
   char *disk_path = malloc(pathlen + 1);
+  char *xsfile = NULL;
+  FILE *xsf;
+  char principal [1026];
+  bool authorized;
   if(disk_path == NULL) {
     log_error("malloc() failed: %s", strerror(errno));
     return NULL;
   }
-  if(storage_root) {
-    *storage_root = malloc( 7 + RS_HOME_SERVE_ROOT_LEN + strlen(user) + 1);
-    if(*storage_root == NULL) {
-      log_error("malloc() failed: %s", strerror(errno));
-      free(disk_path);
-      return NULL;
+  log_debug("Constructing disk_path for dom_user = \"%s\"", dom_user);
+  xsfile = malloc( 7 + RS_HOME_SERVE_ROOT_LEN + strlen(dom_user) + 1 + 17);
+  if(xsfile == NULL) {
+    log_error("malloc() failed: %s", strerror(errno));
+    free(disk_path);
+    return NULL;
+  }
+  sprintf(xsfile, "/home/%s/%s/.k5remotestorage", dom_user, RS_HOME_SERVE_ROOT);
+  log_debug("Access control list = \"%s\"", xsfile);
+  xsf = fopen (xsfile, "r");
+  authorized = false;
+  if (xsf) {
+    while ((!authorized) && fgets (principal, sizeof (principal)-1, xsf)) {
+      int len = strlen (principal);
+      if ((len > 1) && (principal [len-1] == '\n')) {
+        principal [--len] = '\0';
+      }
+      log_debug("Considering acceptable principal \"%s\"", principal);
+      authorized = (len == authuser->length) && (0 == memcmp (principal, authuser->value, len));
     }
-    sprintf(*storage_root, "/home/%s/%s", user, RS_HOME_SERVE_ROOT);
+    fclose (xsf);
+  } else {
+    log_error ("Failed to open access control list");
+    free(xsfile);
+    free(disk_path);
+    return NULL;
+  }
+  if (!authorized) {
+    log_error ("Access control list does not contain authorized user");
+    free(xsfile);
+    free(disk_path);
+    return NULL;
+  }
+  log_debug ("xsfile = \"%s\"", xsfile);
+  if(storage_root) {
+    // Cut off .k5remotestorage and reuse for *storage_root
+    xsfile [7 + RS_HOME_SERVE_ROOT_LEN + strlen (dom_user)] = '\0';
+    *storage_root = xsfile;
+    log_debug ("storage_root = \"%s\"", storage_root);
+  } else {
+    free (xsfile);
+    xsfile = NULL;
   }
   // remove all /.. segments
   // (we don't try to resolve them, but instead treat them as garbage)
@@ -523,16 +569,18 @@ static char *make_disk_path(char *user, char *path, char **storage_root) {
     pos[restlen] = 0;
   }
   // build path
-  sprintf(disk_path, "/home/%s/%s%s", user, RS_HOME_SERVE_ROOT, path);
+  sprintf(disk_path, "/home/%s/%s%s", dom_user, RS_HOME_SERVE_ROOT, path);
+  log_debug ("disk_path = \"%s\"", disk_path);
   return disk_path;
 }
 
-static evhtp_res handle_get_or_head(evhtp_request_t *request, int include_body) {
+static evhtp_res handle_get_or_head(evhtp_request_t *request, gss_buffer_t authuser, int include_body) {
 
   log_debug("HANDLE GET / HEAD (body: %s)", include_body ? "true" : "false");
 
   char *disk_path = make_disk_path(REQUEST_GET_USER(request),
                                    REQUEST_GET_PATH(request),
+                                  authuser,
                                    NULL);
   if(disk_path == NULL) {
     return EVHTP_RES_SERVERR;
index 2404f76..10c3825 100644 (file)
@@ -13,9 +13,9 @@
 #ifndef RS_HANDLER_STORAGE_H
 #define RS_HANDLER_STORAGE_H
 
-evhtp_res storage_handle_head(evhtp_request_t *request);
-evhtp_res storage_handle_get(evhtp_request_t *request);
-evhtp_res storage_handle_put(evhtp_request_t *request);
-evhtp_res storage_handle_delete(evhtp_request_t *request);
+evhtp_res storage_handle_head(evhtp_request_t *request, gss_buffer_t username);
+evhtp_res storage_handle_get(evhtp_request_t *request, gss_buffer_t username);
+evhtp_res storage_handle_put(evhtp_request_t *request, gss_buffer_t username);
+evhtp_res storage_handle_delete(evhtp_request_t *request, gss_buffer_t username);
 
 #endif /* !RS_HANDLER_STORAGE_H */
index 8620502..ff6e22c 100644 (file)
@@ -29,9 +29,9 @@ void init_webfinger() {
   }
   sprintf(lrdd_template, "%s://%s/.well-known/webfinger?resource={uri}",
           RS_SCHEME, RS_HOSTNAME);
-  storage_uri_format_len = strlen(RS_SCHEME) + strlen(RS_HOSTNAME) + 12;
+  storage_uri_format_len = strlen(RS_SCHEME) + strlen(RS_HOSTNAME) + 13;
   storage_uri_format = malloc(storage_uri_format_len + 2 + 1);
-  sprintf(storage_uri_format, "%s://%s/storage/%%s", RS_SCHEME, RS_HOSTNAME);
+  sprintf(storage_uri_format, "%s://%s/storage/%%s/%%s", RS_SCHEME, RS_HOSTNAME);
 }
 
 static size_t json_writer(char *buf, size_t count, void *arg) {
@@ -76,13 +76,15 @@ static int process_resource(const char *resource, char **storage_uri) {
       log_debug("hostname: %s", hostname);
       // check hostname
       if(strcmp(hostname, RS_HOSTNAME) == 0) {
+#if 0
         uid_t uid = user_get_uid(local_part);
         log_debug("got uid: %d (RS_MIN_UID: %d, allowed: %d)", uid,
                   RS_MIN_UID, UID_ALLOWED(uid));
         // check if user is valid
         if(UID_ALLOWED(uid)) {
-          *storage_uri = malloc(storage_uri_format_len + strlen(local_part) + 1);
-          sprintf(*storage_uri, storage_uri_format, local_part);
+#endif
+          *storage_uri = malloc(storage_uri_format_len + strlen(hostname) + strlen(local_part) + 1);
+          sprintf(*storage_uri, storage_uri_format, hostname, local_part);
 #if 0
           *auth_uri = malloc(RS_AUTH_URI_LEN + strlen(local_part) + 1);
           sprintf(*auth_uri, RS_AUTH_URI, local_part);
@@ -90,7 +92,9 @@ static int process_resource(const char *resource, char **storage_uri) {
 
           free(resource_buf);
           return 0; // success!
+#if 0
         }
+#endif
       }
     }
 
index afd1d4a..2b2a355 100644 (file)
@@ -95,6 +95,76 @@ static void handle_storage(evhtp_request_t *req, void *arg) {
   dispatch_storage(req, arg);
 }
 
+static void handle_static(evhtp_request_t *req, void *arg) {
+  log_info("static URI path \"%s\"", req->uri->path->full);
+  if (req->method != htp_method_GET) {
+    req->status = EVHTP_RES_METHNALLOWED;
+  } else if ((req->uri->path->full [0] != '/') || strstr (req->uri->path->full, "/..")) {
+    req->status = EVHTP_RES_SERVERR;
+  } else {
+    char *path = malloc (RS_STATIC_DIR_LEN + strlen (req->uri->path->full) + 10 + 1);
+    if (!path) {
+      log_error("malloc() failed");
+      req->status = EVHTP_RES_SERVERR;
+    } else {
+      int fh;
+      char *mime_type = NULL;
+      int free_mime_type = 0;
+
+      sprintf (path, "%s%s", RS_STATIC_DIR, req->uri->path->full);
+      if (path [strlen (path) - 1] == '/') {
+       strcat (path, "index.html");
+      }
+      log_debug("static filename \"%s\"", path);
+    
+      // mime type is either passed in ... (such as for directory listings)
+      if(mime_type == NULL) {
+        // ... or detected based on xattr
+        mime_type = content_type_from_xattr(path);
+        if(mime_type == NULL) {
+#if 0
+          // ... or guessed by libmagic
+          log_debug("mime type not given, detecting...");
+          mime_type = magic_file(magic_cookie, path);
+#endif
+          if(mime_type == NULL) {
+            // ... or defaulted to "application/octet-stream"
+            log_error("magic failed: %s", magic_error(magic_cookie));
+            mime_type = "application/octet-stream; charset=binary";
+          }
+        } else {
+          // xattr detected mime type and allocated memory for it
+          free_mime_type = 1;
+        }
+      }
+    
+      if (mime_type) {
+        log_info("setting Content-Type of %s: %s", req->uri->path->full, mime_type);
+        ADD_RESP_HEADER_CP(req, "Content-Type", mime_type);
+      }
+    
+      if(free_mime_type) {
+        free((char*)mime_type);
+      }
+
+      fh = open (path, O_RDONLY);
+      free(path);
+      if (fh < 0) {
+        req->status = EVHTP_RES_NOTFOUND;
+      } else {
+       char buf [1024];
+       size_t rdlen;
+       while ((rdlen = read (fh, buf, 1024)) > 0) {
+         evbuffer_add (req->buffer_out, buf, rdlen);
+       }
+        close (fh);
+        req->status = EVHTP_RES_OK;
+      }
+    }
+  }
+  evhtp_send_reply(req, req->status);
+}
+
 static int dummy_ssl_verify_callback(int ok, X509_STORE_CTX * x509_store) {
   return 1;
 }
@@ -188,10 +258,14 @@ int main(int argc, char **argv) {
 
   /* REMOTESTORAGE */
 
-  evhtp_callback_t *storage_cb = evhtp_set_regex_cb(server, "^/storage/([^/]+)/.*$", handle_storage, NULL);
+  evhtp_callback_t *storage_cb = evhtp_set_regex_cb(server, "^/storage/([^/]+/[^/]+)/.*$", handle_storage, NULL);
 
   evhtp_set_hook(&storage_cb->hooks, evhtp_hook_on_request_fini, finish_request, NULL);
 
+  /* STATIC CONTENT */
+
+  evhtp_set_gencb(server, handle_static, NULL);
+
   if(evhtp_bind_sockaddr(server, (struct sockaddr*)&sin, sizeof(sin), 1024) != 0) {
     log_error("evhtp_bind_sockaddr() failed: %s", strerror(errno));
     exit(EXIT_FAILURE);
index 24588fa..3134c13 100644 (file)
@@ -19,6 +19,8 @@
 #include <stdlib.h>
 #include <stdio.h>
 #include <string.h>
+#include <stdint.h>
+#include <stdbool.h>
 #include <errno.h>
 #include <time.h>
 #include <unistd.h>
 
 #include <openssl/sha.h>
 
+// kerberos headers
+#include <krb5.h>
+#include <gssapi.h>
+#include <gssapi/gssapi_krb5.h>
+
 // rs-serve headers
 
 #include "version.h"