Basic CLI usage

Display a file from the file system

From inside the src folder:

$ cat sample.txt
This is a text file.
$ ori sample.txt
This is a text file.

At this basic level, ori behaves like the cat command, but it can handle more than just files.

When you invoke ori:

  1. It parses its arguments as an expression.
  2. It evaluates that expression, looking up identifiers in the current scope (defined below).
  3. If the value of the expression is a JavaScript module, ori imports the module and obtains its default export. If it’s a JavaScript function, ori executes it.
  4. It displays the result.

Here ori parses the expression sample.txt as an identifier. In JavaScript, sample.txt is not a valid identifier because it contains a period, but ori’s expression parser can recognize file names as identifiers. ori looks up that identifier in the current scope. By default, the scope includes:

  • the files in the current folder
  • the functions exported by JavaScript modules in the current folder
  • the functions built into ori

In this case, ori finds that “sample.txt” is the name of a file, and reads that file from the current folder. The file’s contents become the result of the expression, which ori then renders to the console.

Invoke a function

If you ask ori for the sample file greet.js, it returns the contents of that file:

$ ori greet.js
export default (name = "world") => `Hello, ${name}.`;

You can invoke the function exported by that module by adding parentheses. Since most command-line shells interpret parentheses, you’ll need to quote the expression:

$ ori "greet.js()"
Hello, world.

When you ask ori to evaluate greet.js():

  • ori sees if greet.js exists. In this case, it finds a JavaScript module with that name.
  • ori dynamically imports the module and obtains the default export (a function).
  • Because the result is a JavaScript function, ori executes it.
  • The function’s result is the string “Hello, world.”, which ori displays.

Pass a string to a function

You can pass arguments to JavaScript functions from the shell:

$ ori "greet.js('Alice')"
Hello, Alice.

ori accepts strings in single quotes or double quotes. In this case, the double quotes above are parsed by the shell, and are necessary because the bash shell shown here would otherwise prevent ori from seeing the single quotes.

In the async tree paradigm discussed later, any function can be treated like a tree, and vice versa. This means you can use path syntax as a convenient alternative to specify a string argument:

$ ori greet.js/Alice
Hello, Alice.

In this path syntax, the first path segment (greet) will be looked up in the current scope. All subsequent path segments (like Alice) are treated as plain text. Otherwise, both ways of passing arguments behave the same.

In this way, ori lets you call a JavaScript function from the shell without needing to write JavaScript code to parse command line arguments.

Aside: Loading functions as ES modules

The Origami project you’re working in includes a file called package.json that instructs Node to load .js files as ES (EcmaScript) modules.

By default, Node imports .js files as CommonJS modules. To allow ori to dynamically import JavaScript files in your own projects as ES modules, you will need to include a package.json file in the folder with your .js file or in any parent folder. That package.json should include the entry "type": "module".

Use ori as a general-purpose JavaScript shell tool

The code samples include a small collection of functions:

$ ori double.js
export default (x) => `${x}${x}`;
$ ori greet.js
export default (name = "world") => `Hello, ${name}. `;
$ ori uppercase.js
export default (x) => x.toString().toUpperCase();

You can then use ori to mix and match these functions from the shell:

$ ori uppercase.js/hi
HI
$ ori greet.js uppercase.js/there
Hello, THERE.
$ ori uppercase.js greet.js
EXPORT DEFAULT (NAME = "WORLD") => `HELLO, ${NAME}. `
$ ori uppercase.js greet.js/world
HELLO, WORLD.
$ ori double.js greet.js/everybody
Hello, everybody. Hello, everybody.
$ ori double.js greet.js uppercase.js/there
Hello, THERE. Hello, THERE.

Here are the equivalent verbose forms with parentheses:

$ ori "uppercase.js('hi')"
HI
$ ori "greet.js(uppercase.js('there'))"
Hello, THERE.
$ ori "uppercase.js(greet.js)"
EXPORT DEFAULT (NAME = "WORLD") => `HELLO, ${NAME}. `
$ ori "uppercase.js(greet.js('world'))"
HELLO, WORLD.
$ ori "double.js(greet.js('everybody'))"
Hello, everybody. Hello, everybody.
$ ori "double.js(greet.js(uppercase.js('there')))"
Hello, THERE. Hello, THERE.

The expression ori uppercase.js greet.js performs the dubious work of uppercasing the greet.js code. To invoke greet.js and pass its result to uppercase.js without using parenthesis, the correct form includes an argument for greet.js like ori uppercase.js greet.js/world.

ori lets you use the shell as a basic JavaScript console, so you can invoke and compose functions in any combination without having to write permanent code. This can be useful when you’re experimenting, testing, or need to do one-off operations from the shell.

Read files with ori

You can feed a file to a JavaScript function:

$ ori sample.txt
This is a text file.
$ ori uppercase.js
export default (x) => x.toString().toUpperCase();
$ ori uppercase.js sample.txt
THIS IS A TEXT FILE.

This lets you pass files to your JavaScript functions without you having to write code to deal with files.

In this example, ori ends up passing a file buffer to the uppercase.js function. The uppercase.js function includes a toString() call which here will extract the text from the file buffer. It can then do its uppercasing work on the resulting text.

Reading input from stdin

You can pipe data into JavaScript functions with the built-in stdin function:

$ echo This is input from the shell | ori uppercase.js @stdin/
THIS IS INPUT FROM THE SHELL

Note the trailing slash in @stdin/, which invokes the @stdin function to obtain the complete standard input. This lets you pipe data to a JavaScript function that accepts a simple argument instead of needing specialize the function specifically to read data from the console.

Writing output to a file

You can use regular shell features to pipe the output from your JavaScript functions to a file:

$ ori sample.txt
This is a text file.
$ ori uppercase.js sample.txt > uppercase.txt
$ ori uppercase.txt
THIS IS A TEXT FILE.

 

Next: Using trees with the ori CLI ยป