npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@jolie/jo

v1.1.0

Published

A thin JavaScript library for accessing microservice APIs

Downloads

4

Readme

Jo

Access web APIs as JavaScript objects. Jo (pronounced "Yo!") is a modest and thin library to write simpler client code.

Chuck Norris, Jo

Jo supports both verb-oriented APIs (e.g., /getJoke?{id:1}) and resource-oriented APIs (e.g., /jokes/1). You can use both interchangeably, which is useful when you have to interact with microservices that adopt different API styles.

Jo can be used with any web server. It uses JSON as data format (more formats could be added in the future). It includes native support for Jolie API Gateways (aka Jolie redirections).

Installation

npm i @jolie/jo

Usage: Verb-oriented APIs

Verb-oriented APIs can be accessed through the global Jo object.

Invoking an operation from the server

Syntax: Jo.operation( [data, params] ) where

  • operation is the operation you want to invoke;
  • the optional data to be sent is a JSON object;
  • the optional params are parameters for the underlying fetch operation (https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API), for example in case you need to specify HTTP headers.

Invoking this method returns a promise.

This method uses POST as default HTTP method. To change it, use the optional parameters. For example, to use GET: Jo.operation( data, { method: 'GET' } ).

Example

Suppose the originating web server offers an operation greet that returns a string given a tree with subnode name. (Nanoservices are great for examples, not so much for production: do this at home only!)

You can invoke it as follows.

import {Jo} from '@jolie/jo'

Jo.greet( { name: "Homer" } )
	.then( response => console.log( response.greeting ) ) // Jo uses promises
	.catch( error => {		// an error occurred
		if ( error.isFault ) {	// It's an application error
			console.log( JSON.stringify( error.fault ) );
		} else {		// It's a middleware error
			console.log( error.message );
		}
	} );

Here is how this operation would be implemented in a Jolie service.

greet( request )( response ) {
	response.greeting = "Hello " + request.name
}

Catching errors, the alternative way

Distinguishing application and middleware errors might be boring. Use JoHelp.parseError (JoHelp is pronounced "Yo! Help!") to get that part done for you automatically. You will get a string containing the error message, if it is a middleware error, or a JSON.stringify of the carried data, if it is an application error.

Jo.greet( { name: "Homer" } )
	.then( response => console.log( response.greeting ) )
	.catch( JoHelp.parseError ).catch( console.log );

Jolie Redirections

Jo supports redirection (not to be confused with HTTP redirections), the Jolie primitive to build API gateways with named subservices. (Unnamed subservices in the gateway, obtained by aggregation, are available as normal operations, so they can be called with the previous syntax.)

Suppose that the originating Jolie service has a redirection table as follows.

Redirects: Greeter => GreeterService

If GreeterService has our operation greet, we can invoke it as follows.

Jo("Greeter").greet( { name: "Homer" } )
	.then( response => console.log( response.greeting ) )
	.catch( JoHelp.parseError ).catch( console.log );

If your Jolie API gateway points to another API gateway, you can nest them!

Jo("SubGateway1/SubGateway2/Greeter").greet( { name: "Homer" } )
	.then( response => console.log( response.greeting ) )
	.catch( JoHelp.parseError ).catch( console.log );

Usage: Resource-oriented APIs

Resource-oriented APIs can be accessed through the global Jor object.

Syntax: Jor.resource.http_method( data [, params] ) where

  • resource is the name of the resource you want to access;
  • http_method is an HTTP method (use lowercase): get, post, delete, put, head, patch, or options;
  • the data to be sent is a JSON object;
  • the optional params are parameters for the underlying fetch operation (https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API).

If resource has a name that cannot be written in JavaScript, you can use the alternative syntax Jor["resource"].http_method( data [, params] ).

Example

Suppose the web server offers a resource /jokes, and you want to get all of them.

import {Jor} from '@jolie/jo'

Jor.jokes.get()
	.then( response => /* handle all the jokes */ )
	.catch( JoHelp.parseError ).catch( console.log );

Redirections

Redirections are supported by Jor just as for verb-based APIs. Suppose the server offers the /jokes resource through the subservice ChuckNorris. Then we can access it as follows.

Jor("ChuckNorris").jokes.get()
	.then( response => /* handle all the jokes */ )
	.catch( JoHelp.parseError ).catch( console.log );

Installation

<script type="text/javascript" src="https://raw.githubusercontent.com/fmontesi/jo/master/lib/jo.js"></script>

Or, download Jo (https://raw.githubusercontent.com/fmontesi/jo/master/lib/jo.js) and use it locally.

Pull requests with better ways to distribute Jo are welcome.

Dependencies

There are no dependencies on other libraries. However, Jo uses some recent features offered by web browsers.

  • Jo uses fetch to perform asynchronous calls. If you want to use Jo with older browsers, use a polyfill for fetch. Check which browsers support fetch here: https://caniuse.com/#feat=fetch.

  • Jo uses some modern JavaScript features. Proxy is used to implement the magic of calling the operations of your Jolie server as if they were native methods (Jo.operation). We also use Arrow functions. Check which browsers support Proxy at https://caniuse.com/#feat=proxy, and which support arrow functions at https://caniuse.com/#feat=arrow-functions. If you want to use Jo in browsers that do not support these features, you can try compiling Jo with Babel.

FAQ

How do I handle basic values in root nodes sent by Jolie? (AKA response.$)

In JSON, an element can either be a basic value (e.g., strings, numbers), an object, or an array. In Jolie, there are no restrictions: an element is always a tree, and each node can contain both a basic value and subnodes (similarly to markup languages). For example, this is valid Jolie: "Homer" { .children[0] = "Bart", .children[1] = "Lisa" }. It gives a tree containing the string Homer in its root node, which has an array subnode children with two elements. If you receive this tree using Jo in variable response, you can access the value contained in the root node ("Homer" in our example) by response.$.