1.0.0 • Published 8 years ago

@rgbkrk/synchronized-object-diff-spec v1.0.0

Weekly downloads
1
License
Apache-2.0
Repository
github
Last release
8 years ago

Thinking on Synchronized Objects for Jupyter

I'd love to see a general synchronized objects specification for Jupyter/nteract, extracting the core of what ipywidgets provides.

Model reduction

Let's think about how a model could get updates.

const simpleUpdate = (model, change) =>
  Object.assign({}, model, change);

If they rely on Immutable.js

const mergedUpdate = (model, change) =>
  model.merge(Immutable.fromJS(change))

Or if they like assuming the change is the whole thing

const directed = (model, change) =>
  change

The :key: here would be that this is a contract for a frontend component to coordinate with a backend. They could be using protobufs, bitmasks on binary arrays, etc. - it's up to the libraries! The component expects updates to their model and that it too may send updates.

It's up to the implementer for how big or small they want to be in terms of what's diff'ed.

Out-of-synchronization

Could these get out of sync? Yes!

We probably want some lineage so an actor can ask for the whole model again. If they missed a dependent, they ask for either the entire model or all the patches they need.

Truth:  A --> B --> C --> D

Kernel: A --> B --> C --> D

Client: A --> B --> D       ❌
 "Ruh roh, I have up to B, just got D."

Kernel:              C

Client:  A --> B --> C --> D ✅

All actors can send to request the whole model or a collection of patches - all while receiving patches on top of the initial model. Similar to other real-time models, the resolution is done by one actor, the kernel, amongst "competing" actors while providing an optimistic "merged" view to clients. If the model gets out of sync, the kernel can request either the model or all the changes they missed and vice versa.

Things to think on

  • Is it good/bad that we expect the component to have a local state?
    • Should they provide the reducer, we pass them the final model state -- so its in the state tree and notebook doc, potentially easier for synchronization amongst multiple users

Proposed plugin API for nteract

We'll extend on top of the transform API

class Transform extends React.Component {
  ...
}

Transform.mimetype = 'application/vnd.tf.v1+json'

To provide an optional reducer:

Transform.reducer = (model, update) =>
  Object.assign({}, model, update);

Note that it's up to the implementer to declare this reducer.

When a new model is created, with a modelID, we register the reducer and apply it to our list of models. Later, as changes flow through, we update the state of that model.

  [MODEL_UPDATE]: (state, action) => {
    const id = action.modelID;
    const model = state.getIn(['models', id]);
    return state.setIn(['models', id, 'state'],
      model.reducer(model.state, action.change));
  }
}

Changes to that model get reflected back to registered views. React (and the component itself) handle the rest of the changes.

Initiating the model

It's tempting to rely on any of the messages that return a mime bundle, so that we can display interactive models anywhere that has a display-like area.

I'd like the first message to both provide the initial model state and the model id.

content = {
  "data" : {
    "application/custom+json": {
      "value": 2
    }
  },
  "metadata" : {
    "model_id": "e191c04e-4428-4648-9f76-dc7e643bd980"
  }
}

In our case, this means that the frontend can now track this model_id.

Knowing whether or not to use this model_id is a matter of checking to see if Transform.reducer exists for the Transform with this mimetype (yes, I'm ignoring the bundle).

Perhaps if we want to be explicit about a matching model_id to the mimetype given:

content = {
  "data" : {
    "application/custom+json": {
      "value": 2
    }
  },
  "metadata" : {
    "models": {
      "application/custom+json": "e191c04e-4428-4648-9f76-dc7e643bd980"
    }
  }
}

We now have enough to register a new model (or ensure a prior one).

Questions to explore and answer

Questions to answer:

  • What's the message spec?
  • How do we handle differences?
  • How do we initiate the model?
    • Tempted to rely on display_data/execute_reply/pager, relying on the mimebundles
  • How do we handle binary payloads (from the message spec)?
1.0.0

8 years ago