IMove itself is positioned as “a logically reusable, function-oriented, process-oriented JavaScript tool library.”

For developers, iMove is the ideal tool to accomplish these goals. Move the mouse, write a node function, code export, into the specific project can be directly used, is not very convenient?

So why do we do iMove online execution code? From a user perspective, it is necessary for developers to select nodes and right click to execute code online. In this way, the tool attributes of iMove can be achieved to the extreme, so that developers have a better experience and easier operation.

Last time on the Github Trend list, iMove principle technology revealed! In this article we sold a mystery, today and you share the next iMove is how to achieve online execution of the node code ~

primers

It all started with a friend who was using iMove for the first time and asked me how to test and run the code I’d just written. Without further ado, I went on her computer and operated:

  1. The installation@imove/cli
  2. Execute the command in the project root directory:imove -d
  3. Find the components in the project and add the codelogic.invoke('trigger')
  4. Start the project, open the console panel, and filterloginformation
  5. . .

“I just want to run the code I just wrote and see what the output is…”

“Cumbersome steps”, “troublesome operation”, “cost to get started”, “not intuitive operation results”… At this moment, the above words are the first time in my mind, and then I look at my partner’s puzzled eyes, as if she will be persuaded to leave the next moment…

How can this work? The experience has to be optimized and the cost of using it has to come down! This gives you the ability to right-click and execute code online.

In fact, the appeal of the above problem is very simple: after writing the code of the node, there is a run button next to it, and you can directly see the result of running on the line. Let’s take a look at how this optimization can change things:

  • No installation tools required, 0 cost to get started: Code can run in the browser, no need to install command line tools, greatly reducing learning costs.
  • Results visualization, simple and clear: the most commonly used in the development and debugging process is the log, every time to open the console is very inconvenient, direct node running results in a visual way to show, simple and intuitive.
  • Online mock input for easy testing:Each node of the code will have a variety of possible inputs, if can be directlymockInput, test efficiency will be greatly improved.
  • Test cases can be saved to ensure code quality: In addition to facilitating the testing of node code, if pairs of input/output can be saved as test cases to gradually form a complete set of test cases, the code quality of node can be further guaranteed.
  • . .

It’s a nice idea, but how do you do it? Continue to read ~

Exploration 1: Where does the node code run

First, the first question to solve is: Where does the node code run?

After evaluation, we believe there are two main options:

  1. The browser side directly runs the node code;
  2. Start a service locally, send node information to the local, after the local build is compiledbundleSend back to the browser for execution.

For the above two schemes, iMove chooses the former for a simple reason: the initial intention of running node code online is to reduce the cost of getting started. If a service needs to be started locally, users will have to learn a new command, which virtually raises the threshold of use.

However, there are many obstacles to running node code directly in a browser, and eval is not enough

Exploration 2: How do I run import/export

As you can see from the iMove output code, each node code in the flowchart will eventually be compiled into a separate JS file. Therefore, each node is capable of importing other NPM packages, which is why eval cannot be called directly.

1) Browser native support ES Module

Of course, you’re probably smart enough to already know that browsers support Native ES Modules, including the popular Vite. The underlying principle is based on this. To that end, let’s take a quick look at how to get import/export code to run in the browser

Let’s start with the official documentation on MDN:

As you can see, modern browsers already support it, so we don’t have to worry about compatibility. In addition, as you continue to browse through the documentation, you will see that the key to getting your browser to support ES Module is to add the type=” Module “attribute to the script tag. Here’s an example:

# file directory index.html main.mjs <! --index.html--><script type="module">
  import sayHello from './main.js';
  say('Hello iMove! ');
</script>

// main.js 
const say = (words) = > console.log(words);
Copy the code

Note: do not open the index.html file directly locally when running the example above, because browsers will default to using the file:// protocol and cross-domain requests for main.js resources. This problem can be solved by opening an HTTP service locally, or trying it on CodesandBox.

As you can see, getting the browser to run ES Module code is pretty easy and almost free, but is the problem really solved? Try adding a line import get from ‘lodash.get’ to your code.

Can see the console error, the original is lodash get loaded by the browser as relative path, the browser is actually support HTTP load path, so we can change it to import the get the from ‘https://unpkg.com/lodash.get’.

Unfortunately, the console still reported an error. The reason for the error is that the lodash.get package follows the CJS specification, but the browser only recognizes the ESM specification, so it cannot parse and execute the package code.

2) SystemJS tries

Since browsers only support esM code natively, is there a way to support CJS code? After some research, we found the SystemJS library (portal: github.com/systemjs/sy…). .

According to its description, it is a tool to solve the browser running ES Module. But when we fumbled around with the documentation, we found that loading the lodash.get package still failed and reported the same error… Luckily, we found something else in its issue section: ES Modules and CommonJS? (PS: Thus, the problem is still widespread.)

According to the official reply, we can extract the following key information:

  • Before version 0.21SystemJSIs to supportcjsStandard, but no longer supported.
  • Previous versions supported itcjsThe main reason: the previous approach was to download the code string and match with the rerequireParsing dependencies before executing code; Now the compile and parse work depends on the browser itself.
  • The previous solution has performance problems, so it will not be considered in the new version.

For this, we find the documentation for version 0.21: github.com/systemjs/sy…

You can see that this version of SystemJS seemed to meet our needs, but ultimately iMove didn’t. Because it is no longer officially recommended and maintained, the road is closed again…

3) CDN hosting of a new generation of JS modules

The above problem seems to be a dead end, but if you think about it, the essence of the problem is “browsers don’t support the CJS specification” **.

On the other hand, why should browsers support the CJS specification? CJS to ESM (CJS to ESM); Wolf uncle this article “2021 to see Deno (CDN for JavaScript Modules thinking)” said very clearly, the conversion of this matter in CDN to do more appropriate ~

To recap, that’s what JSPM does.

To verify the effect, here is a screenshot of lodash.get loaded by the browser from unpkg and JSPM.

Exploration 3: Merge multiple files into a single file execution

The “how to run import/export problem” mentioned above seems to have been solved perfectly, but in practice one detail is still missing: IMove’s output code is a multi-file organization, so browsers will import other files in relative paths, which means an HTTP service is required to load these resources, which is not acceptable.

How to solve? The simplest way to do this is to merge multiple files into a single file, which naturally eliminates the relative path problem. Let’s look at another example:

// a.js
import get from 'lodash.get';
const obj = {
  text: 'a'.say: () = > console.log(get(obj, 'text'))};export default obj;

// b.js
import get from 'lodash.get';
const obj = {
  text: 'b'.say: () = > console.log(get(obj, 'text'))};export default obj;

// main.js
import a from './a';
import b from './b';
a.say();
b.say();
Copy the code

The above is the pre-merge multi-file organization. It is not difficult to merge them, just note the following two points:

  • importWhen a file, the code for that file needs to be executed immediately
  • Global variable names may be the same in different files, and naming contamination needs to be addressed
// merged.js
const run = async() = > {const modA = await (async() = > {const get = (await import('https://jspm.dev/lodash.get')).default;
    const obj = {
      text: 'a'.say: () = > console.log(get(obj, 'text'))};returnobj; }) ();const modB = await (async() = > {const get = (await import('https://jspm.dev/lodash.get')).default;
  const obj = {
    text: 'b'.say: () = > console.log(get(obj, 'text'))};returnobj; }) (); modA.say(); modB.say(); }; run();Copy the code

Note that the original import is in the function body due to merge, so you need to use dynamic import to load the network package. As long as the rest of the regular expression match substitution or AST transformation can achieve the corresponding effect, this article will not expand on the details.

Exploration 4: How do Scripts communicate by value

At this point, it can be said that “everything is ready, only the east wind”. After the multi-file merge, we just need to run the code and get the resulting display. It is important to note that since the code to be executed follows the ESM specification, there is one prerequisite for executing code with eval: the current code must also be under the script tag of type=”module”. Instead, we can execute string code without eval:

const script = document.createElement('script');
script.type = 'module';
script.text = 'code here';
document.body.appendChild(script);
Copy the code

As shown above, we can execute code by dynamically inserting script tags. But then the question becomes: how do we trigger a callback to get the result display after the code has run? Moreover, these two pieces of code are still under two script tags.

The easiest way to do this is in a jSONP-like fashion, where one party registers a unique method under the global window and the other party calls the method after the code has executed. Here, we introduce a more elegant way to solve this problem using event communication

MDN official document portal: Custom Events

/ / listen for an event document. AddEventListener (‘ customEvent ‘, function (evt) {the console. The log (evt) detail); }, false); Document.dispatchevent (new CustomEvent(‘ CustomEvent ‘, {detail: {text: ‘iMove’}}));

conclusion

This article has introduced iMove online node code this step through the pit, ultimately mainly rely on the IDEA of HTTP-import to solve the problem. Deno has certainly changed the way you think about package management. Because deno is small and trial-and-error cheap, it really does set a trend. This isn’t new, but it’s been around for a long time, it’s easy to use, it’s efficient, and it’s even spawned a lot of thinking about CDN for JavaScript modules. If you have better ideas, welcome to communicate with us.

In addition, making iMove online execution code function is a decision made from the perspective of users, which is highly recognized by my friends. In the process of landing, they can explore a new way, which is very commendable. IMove itself is an open source project designed to empower teams that can define problems, solve them, build confidence, and inspire enthusiasm for technology.