Skip to content

Latest commit

 

History

History
258 lines (206 loc) · 11.9 KB

porting_to_new_environment.md

File metadata and controls

258 lines (206 loc) · 11.9 KB

Guide to Porting ConfluxPortal to a New Environment

ConfluxPortal has been under continuous development for nearly one year now, and we’ve gradually discovered some useful abstractions that have allowed us to grow more easily. A couple of those layers together allow ConfluxPortal to be ported to new environments and contexts increasingly easily (although it still could be easier, and please let us know if you get stuck!)

Before we get started, it's worth becoming familiar with our basic architecture:

ConfluxPortal-architecture-diagram

The ConfluxPortal-background describes the file at app/scripts/background.js, which is the web extension singleton. This context instantiates an instance of the ConfluxPortal Controller, which represents the user's accounts, a connection to the blockchain, and the interaction with new Dapps.

When a new site is visited, the WebExtension creates a new ContentScript in that page's context, which can be seen at app/scripts/portal-contentscript.js. This script represents a per-page setup process, which creates the per-page web3 api, connects it to the background script via the Port API (wrapped in a stream abstraction), and injected into the DOM before anything loads.

You can choose to use this streaming interface to connect to the ConfluxPortal controller over any transport you can wrap with a stream, by connecting it to the confluxportal-inpage-provider, but you can also construct a provider per domain like this:

const providerFromEngine = require('@yqrashawn/cfx-json-rpc-middleware/providerFromEngine')

/**
 * returns a provider restricted to the requesting domain
 **/
function incomingConnection(domain, getSiteMetadata) {
  const engine = confluxPortalController.setupProviderEngine(
    domain,
    getSiteMetadata
  )
  const provider = providerFromEngine(engine)
  return provider
}

Please note if you take this approach, you are responsible for cleaning up the filters when the page is closed:

const filterMiddleware = engine._middleware.filter(mid => mid.name === 'filterMiddleware')[0]
filterMiddleware.destroy()

getSiteMetadata()

This method is used to enhance our confirmation screens with images and text representing the requesting domain.

It should return a promise that resolves with an object with the following properties:

  • name: The requesting site's name.
  • icon: A URI representing the site's logo.

Using the Streams Interface

Only use this if you intend to construct the confluxportal-inpage-provider over a stream!

The most confusing part about porting ConfluxPortal to a new platform is the way we provide the js-conflux-sdk API over a series of streams between contexts. Once you understand how we create the ConfluxportalInpageProvider in the portal-inpage.js script, you will be able to understand how the extension-port-stream is just a thin wrapper around the Port postMessage API (see the Chrome documentation for extension message passing for more information). A similar stream API can be wrapped around any communication channel to communicate with the ConfluxPortalController via its setupUntrustedCommunication(stream, domain) method.

The ConfluxPortal Controller

The core functionality of ConfluxPortal all lives in what we call The ConfluxPortal Controller. Our goal for this file is for it to eventually be its own javascript module that can be imported into any JS-compatible context, allowing it to fully manage an app's relationship to Conflux.

Constructor

When calling new ConfluxPortal(opts), many platform-specific options are configured. The keys on opts are as follows:

  • initState: The last emitted state, used for restoring persistent state between sessions.
  • platform: The platform object defines a variety of platform-specific functions, including opening the confirmation view, and opening websites.
  • encryptor - An object that provides access to the desired encryption methods.
Encryptor

An object that provides two simple methods, which can encrypt in any format you prefer. This parameter is optional, and will default to the browser-native WebCrypto API.

  • encrypt(password, object) - returns a Promise of a string that is ready for storage.
  • decrypt(password, encryptedString) - Accepts the encrypted output of encrypt and returns a Promise of a restored object as it was encrypted.
Platform Options

The platform object has a variety of options:

  • reload (function) - Will be called when ConfluxPortal would like to reload its own context.
  • openWindow ({ url }) - Will be called when ConfluxPortal would like to open a web page. It will be passed a single options object with a url key, with a string value.
  • getVersion() - Should return the current ConfluxPortal version, as described in the current CHANGELOG.md or app/manifest.json.

This method returns a javascript object representing the current ConfluxPortal state. This includes things like known accounts, sent transactions, current exchange rates, and more! The controller is also an event emitter, so you can subscribe to state updates via confluxportal.on('update', handleStateUpdate). State examples available here under the confluxportal key. (Warning: some are outdated)

Returns a JavaScript object filled with callback functions representing every operation our user interface ever performs. Everything from creating new accounts, changing the current network, to sending a transaction, is provided via these API methods. We export this external API on an object because it allows us to easily expose this API over a port using dnode, which is how our WebExtension's UI works!

The UI

The ConfluxPortal UI is essentially just a website that can be configured by passing it the API and state subscriptions from above. Anyone could make a UI that consumes these, effectively re-skinning ConfluxPortal.

You can see this in action in our file ui/index.js. There you can see the background connection being passed in, which is essentially the ConfluxPortal controller. With access to that object, the UI is able to initialize a whole React/Redux app that relies on this API for its account/blockchain-related/persistent states.

Putting it Together

As an example, a WebExtension is always defined by a manifest.json file. In ours, you can see that background.js is defined as a script to run in the background, and this is the file that we use to initialize the ConfluxPortal controller.

In that file, there's a lot going on, so it's maybe worth focusing on our ConfluxPortal controller constructor to start. It looks something like this:

const controller = new ConfluxPortalController({
  // User confirmation callbacks:
  showUnconfirmedMessage: triggerUi,
  showUnapprovedTx: triggerUi,
  // initial state
  initState,
  // platform specific api
  platform,
})

Since background.js is essentially the Extension setup file, we can see it doing all the things specific to the extension platform:

  • Defining how to open the UI for new messages, transactions, and even requests to unlock (reveal to the site) their account.
  • Provide the instance's initial state, leaving ConfluxPortal persistence to the platform.
  • Providing a platform object. This is becoming our catch-all adapter for platforms to define a few other platform-variant features we require, like opening a web link. (Soon we will be moving encryption out here too, since our browser-encryption isn't portable enough!)

Ports, streams, and js-conflux-sdk!

Everything so far has been enough to create a ConfluxPortal wallet on virtually any platform that runs JS, but ConfluxPortal's most unique feature isn't being a wallet, it's providing an Conflux-enabled JavaScript context to websites.

ConfluxPortal has two kinds of duplex stream APIs that it exposes:

js-conflux-sdk as a Stream

If you are making a ConfluxPortal-powered browser for a new platform, one of the trickiest tasks will be injecting the js-conflux-sdk API into websites that are visited. On WebExtensions, we actually have to pipe data through a total of three JS contexts just to let sites talk to our background process (site -> contentscript -> background).

To see how we do that, you can refer to the inpage script that we inject into every website. There you can see it creates a multiplex stream to the background, and uses it to initialize what we call the ConfluxPortalInpageProvider, which you can see stubs a few methods out, but mostly just passes calls to sendAsync through the stream it's passed! That's really all the magic that's needed to create a web3-like API in a remote context, once you have a stream to ConfluxPortal available.

In portal-inpage.js you can see we create a postMessage Stream, that's just a class we use to wrap WebExtension postMessage as streams, so we can reuse our favorite stream abstraction over the more irregular API surface of the WebExtension. In a new platform, you will probably need to construct this stream differently. The key is that you need to construct a stream that talks from the site context to the background. Once you have that set up, it works like magic!

If streams seem new and confusing to you, that's ok, they can seem strange at first. To help learn them, we highly recommend reading Substack's Stream Handbook, or going through NodeSchool's interactive command-line class Stream Adventure, also maintained by Substack.

Conclusion

I hope this has been helpful to you! If you have any other questions, or points you think need clarification in this guide, please open an issue on our GitHub!