/dirac

A Chrome DevTools fork for ClojureScript developers

Primary LanguageJavaScriptOtherNOASSERTION

Dirac GitHub license Clojars Project Travis

Dirac is a Chrome DevTools fork with extra features to aid development in ClojureScript.

Features

  • REPL integrated into DevTools Javascript console
    • eval ClojureScript in the context of currently selected stack frame (activated breakpoint)
    • Parinfer goodness
  • custom formatters enabled by default (for cljs-devtools)
  • inlined custom formatters in source code during debugging
  • friendly display of ClojureScript properties in 'Object Properties' panels
    • macro-generated names are renamed to friendly names using sub-indexes
    • properties are grouped into clusters, the most important properties at the top
      • properties with nice names go first
      • then macro-generated names
      • then null values
      • then undefined values

ClojureScript REPL

Source Panel Enhancements

Motivation

Currently I spend most of my time developing web apps on client side in ClojureScript. And for page inspection and debugging I stay most of the time in the DevTools. Over time I realized that a few possible DevTools tweaks would greatly improve my ClojureScript experience. Especially integrating ClojureScript REPL into DevTools Javascript Console would be a pretty huge deal.

A bit of history

Chrome/Blink developers have been pretty open for adding extensibility points to their DevTools platform. Traditionally one has been able to implement a Chrome extension which could register additional panels in Chrome DevTools. Actually Suprematic have implemented their ClojureScript REPL extension this way back in 2014. Also more recently React Developer Tools used this approach.

The problem with extra panels is that that they don't integrate well with existing panels. They are separate islands, separate web "apps" injected into DevTools UI. In my opinion. for best user experience the ClojureScript REPL should be very closely integrated with Javascript Console. Ideally, there should be no switching between panels.

A big improvement was an introduction of custom formatters in early 2015. This allowed to enhance presentation of ClojureScript values directly in Javascript Console and without the need to install a Chrome extension. I have implemented the cljs-devtools library which leverages this feature.

Also I have filled a request for a new extension point which would potentially allow me to implement REPL functionality in a similar way to custom formatters. Unfortunately existing DevTools extension APIs weren't suitable for what I wanted to do. There has been some progress made on that issue but after six months of waiting I decided to fork the DevTools.

A divergent fork?

I don't have an ambition to merge this project upstream into official DevTools. Dirac changes are too specific for ClojureScript (and mostly implemented in ClojureScript). Instead, the idea is to maintain a set of patches rolling on top of official DevTools branch.

Turns out, it is not that big deal at the end of the day. I also provide the Dirac Chrome Extension which wraps Dirac DevTools to make installation and life a bit easier for Dirac users. The experience should be very similar to integrated DevTools - you just use a different keyboard shortcut to open it.

Also I view Dirac as an experimental test ground where new DevTools features can be implemented. Not only for ClojureScript but maybe for other compile-to-js languages as well. It could serve as a proof-of-concept implementation of some niche features or extension points requested or planned in the official DevTools.

In future some Dirac features could move into official DevTools eventually, because there could be a way how to implement them using newly introduced extension points. But until then we want to maintain a live fork so we don't have to wait for Chrome developers to add "our" requested features.

A demo time!

Launch latest Chrome Canary from command-line. I will want you to install the Dirac Chrome Extension there, so I recommend to run it with a dedicated user profile:

mkdir demo-workspace
cd demo-workspace
mkdir .dirac-chrome-profile

Also you have to run it with remote debugging enabled on port 9222 (better make an alias of this command):

/Applications/Google\ Chrome\ Canary.app/Contents/MacOS/Google\ Chrome\ Canary \
  --remote-debugging-port=9222 \
  --no-first-run \
  --user-data-dir=.dirac-chrome-profile

Now you can install Dirac DevTools Chrome extension. After installation, should see a new extension icon next to your address bar.

Now you should clone cljs-devtools-sample and test it with Dirac DevTools:

cd demo-workspace
git clone https://github.com/binaryage/cljs-devtools-sample.git
cd cljs-devtools-sample
lein dirac

At this point you should have a demo website running at http://localhost:7000.

Clicking on Dirac extension icon while on the localhost:7000 page should open you a new window with Dirac DevTools app. It will look almost the same as stock DevTools, but you can tell the difference at first glance: active tab highlight will be green instead of blue (see the screenshots above).

Ok, now you can switch to Javascript Console in Dirac DevTools. Focus prompt field and press PageUp or PageDown. This will switch prompt from Javascript REPL to ClojureScript REPL. (Note you might need to refresh the page once to see custom formatters - DevTools do not render them first time for some reason).

You should see a red message on a green background: Dirac Agent is disconnected. Check your nREPL tunnel at ws://localhost:8231.

That's correct. Dirac REPL uses nREPL protocol, so we have to provide it with some nREPL server. Luckily enough leiningen offers nREPL server by simply running:

cd demo-workspace/cljs-devtools-sample
lein repl

After your nREPL starts your Dirac DevTools should eventually reconnect. Or close Dirac window and open it again if you don't want to wait for reconnection countdown.

Connected? The red message should go away and you should see cljs.user indicating your current namespace instead. REPL is ready for your input at this point. You can try:

(+ 1 2)
js/window
(doc filter)
(filter odd? (range 42))
(in-ns)
(in-ns 'my.ns)

If you see something very similar to the first screenshot at the top, you have Dirac running properly. Congratulations!

You can also test evaluation of ClojureScript in the context of selected stack frame when paused on a breakpoint:

  1. set breakpoint to line 40 of more.cljs (on Mac, you can use use CMD+P to quickly open a file at source panel)
  2. click the "breakpoint test" button on the page
  3. debugger should pause on the line (similar to the second screenshot at the top)

You could notice that custom formatters are presented everywhere including inlined values in the source code. Also property names in the scope panel are sorted and displayed in a more friendly way.

Now hit ESC to bring up console drawer. Make sure you are switched to Dirac REPL and then enter:

js/rng

You should see actual value of rng variable from local scope (formatted by custom formatters from cljs-devtools). Same way as you would expect when evaluating a Javascript command. Actually you can try it. Hit "Page Up" to switch to Javascript console prompt and enter:

rng

This should return the same value.

And now return back to Dirac REPL by pressing "Page Up" again and enter:

(take 3 js/rng)

This is a proof that Dirac REPL can execute arbitrary ClojureScript code in the context of selected stack frame.

Installation

Dirac integration with your project requires some effort and can be configured in many ways. I will first document standard configuration with Leiningen. In later sections we will discuss alternative and/or optional setups.

Here are the ingredients you are going to need:

  1. the Dirac Chrome Extension installed in your Chrome Canary
  2. the cljs-devtools, the library must be installed in your page with :dirac feature enabled
  3. an nREPL server (+ Dirac nREPL middleware)
  4. the Dirac Agent

I assume you went through the demo section above, so you roughly know what to expect.

Setup Dirac Chrome Extension

Please note that you should always use the latest Chrome Canary with Dirac DevTools to prevent any compatibility issues.

As I wrote in the demo section, you probably want to run your Chrome Canary with dedicated user profile. And you have to run it with remote debugging enabled.

/Applications/Google\ Chrome\ Canary.app/Contents/MacOS/Google\ Chrome\ Canary \
  --remote-debugging-port=9222 \
  --no-first-run \
  --user-data-dir=$A_PATH_TO_YOUR_USER_PROFILE_DIRECTORY

Please note that --remote-debugging-port should be 9222 by default. But you can change it in the Dirac Extension options page if needed.

Now install Dirac Chrome Extension.

Chrome should keep it up-to-date for you.

Install cljs-devtools with dirac feature enabled

Please follow cljs-devtools installation instructions to include the latest version in your page. Also make sure that you call (devtools/enable-feature! :dirac) before (devtools/install!) - Dirac feature is not enabled by default!

(devtools/enable-feature! :dirac)
(devtools/install!)
Start nREPL server

There are many ways how to start an nREPL server. We will use Leiningen's nREPL server here.

By default you should run it on port 8230 and with dirac.nrepl.middleware/dirac-repl middleware. Please note that Dirac middleware was implemented as a Piggieback middleware fork, so you cannot run both. Think of Dirac middleware as Piggieback middleware replacement with some extra features specific for Dirac DevTools.

The configuration snippet could look something like this:

:dependencies [[binaryage/dirac "<DIRAC-VERSION>"]]

:repl-options {:port 8230
               :nrepl-middleware [dirac.nrepl.middleware/dirac-repl]}

Replace <DIRAC-VERSION> with actual version of your Dirac Agent.

I tend to put this extra config under :dev profile in my project.clj files (see an example here).

Start Dirac Agent

Dirac Agent is a program which connects to an existing nREPL server and acts as a proxy providing nREPL connections to the browser.

Please note that Dirac DevTools is "just" a web app. It cannot open classic socket connection and talk to the nREPL server directly. Instead it connects to a Dirac Agent instance which listens for web socket connections on port 8231. Dirac Agent has also an open connection to your nREPL server at port 8230 so it can bridge messages between those two. Tunneling messages between the browser and the nREPL server is the main feature of Dirac Agent, sometimes you might see messages mentioning "nREPL Tunnel", which is the main component of Dirac Agent.

Actually Dirac Agent is a bit smarter than that. It allows one-to-many scenario, where multiple Dirac DevTools instances can connect to a singe Dirac Agent which talks to a single nREPL server. Each Dirac DevTools instance is assigned its own nREPL session, so they don't step on each others' toes. Thanks to sessions you can open multiple pages with different Dirac DevTools and they all can have their own independent REPLs.

Unfortunately this is the hardest part of the setup and most fragile. If you run into issues, it is pretty difficult to troubleshoot it without deeper understanding how nREPL works internally, what Piggieback is, how Weasel comes into play and how Dirac Agent orchestrates all this.

If you hit a wall, you can try to ask for help in the #dirac channel at http://clojurians.slack.com (ask for an invitation here).

Now let's get back to launching the Dirac Agent. You can wrap it as a command-line tool and run it. The source for cli tool is here. Here is a custom launcher script which uses maven to download dirac jar and execute agent-cli with your current java env from command-line:

https://github.com/binaryage/dirac/blob/master/scripts/agent-launcher.sh

But I have a better idea. Why to spin yet another JVM just to run this tiny server? Let's use our existing nREPL process and start the Dirac Agent there. This will effectively run nREPL server and Dirac Agent in one process.

You can configure it as :init command which will be executed every time you start your REPL. Simply add this to your :repl-options for Leiningen:

:repl-options {:init (do
                       (require 'dirac.agent)
                       (dirac.agent/boot!))}

Now when you start lein repl from command-line, you should see something like this:

nREPL server started on port 8230 on host 127.0.0.1 - nrepl://127.0.0.1:8230
REPL-y 0.3.7, nREPL 0.2.10
Clojure 1.8.0
Java HotSpot(TM) 64-Bit Server VM 1.8.0_60-b27
    Docs: (doc function-name-here)
          (find-doc "part-of-name-here")
  Source: (source function-name-here)
 Javadoc: (javadoc java-object-or-class-here)
    Exit: Control+D or (exit) or (quit)
 Results: Stored in vars *1, *2, *3, an exception in *e

user=>
Dirac Agent v0.1.2
Connected to nREPL server at nrepl://localhost:8230.
Tunnel is accepting connections at ws://localhost:8231.

The last line should remind you that Dirac Agent started successfully and listens for browser connections on port 8231.

Now you should be able to use REPL from any of your Dirac DevTools instances.

Good job! 👍

Custom configuration

Installation instructions above relied on default settings. Depending on your setup you might need to provide your custom configuration.

There are three areas where custom configuration can be provided:

  1. Dirac Agent
  2. Dirac nREPL middleware
  3. page-specific configuration of a DevTools instance

In general, the configuration can come from different sources:

  1. hard-coded defaults
  2. environment
  3. a config map provided as an argument.

Configuration options from later source override settings from previous sources.

Environmental variables have always a prefix "DIRAC" and are named after option key in the config map. We use environ library for reading environmental settings, so you have multiple ways where to define them: environmental variables, Java system properties or .lein-env file. Please refer to their docs, you can for example use lein-environ plugin to specify environmental variables directly in project.clj (in your leiningen profiles). Alternatively lein-boot for boot scenario.

Dirac Agent configuration

Please consult this file for possible defaults and their environmental counterparts.

If you are invoking dirac.agent/boot! from code you can override config options with a config map specified as an argument. Actually this is what dirac.agent-cli does by reading command-line arguments and converting some of them to config options.

Dirac nREPL middleware configuration

Please consult this file for possible defaults and their environmental counterparts.

If you are invoking dirac.nrepl/boot-cljs-repl! from code you can override config options with a config map specified as an argument.

DevTools page-specific configuration

When you open/refresh Dirac DevTools, it asks current page for effective page-specific Dirac config (see devtools.dirac/get-effective-config. The most important setting is a connection info for Dirac Agent. This allows flexibility of configuring individual ClojureScript projects with their own settings instead of providing one system-wide configuration. In theory you can have different projects connecting do different Dirac Agents (which are connected to different nREPL servers).

Please consult this file for possible defaults and their environmental counterparts.

Credits