Skip to content

Latest commit

 

History

History
389 lines (288 loc) · 12.3 KB

README.md

File metadata and controls

389 lines (288 loc) · 12.3 KB

Build Status npm

UiPath CoreIpc client for JavaScript

Introduction

The UiPath CoreIpc client for JavaScript API is complementary to the UiPath CoreIpc for .NET API. It facilitates Node.js and Web apps to interop with .NET servers via RPC.

The JavaScript API is provided as two NPM packages:

NPM Package Target Transports
npm @uipath/coreipc Node.js (includes Electron) Named Pipes, Web Sockets
npm @uipath/coreipc-web Web (Angular, React) Web Sockets

diagram |

Getting started

  • To install the packages, configure your .npmrc as described in GitHub's documentation.

    Your .npmrc should be similar to:

    //npm.pkg.github.com/:_authToken=<ANY VALID AUTH TOKEN>
    @uipath:registry=https://npm.pkg.github.com/
    
  • For a Node.js app, run:

    npm install @uipath/coreipc

    while for a Web app, run:

    npm install @uipath/coreipc-web
  • Import the module and start using it:

    import { ipc } from '@uipath/coreipc'; // for Node.js
    import { ipc } from '@uipath/coreipc-web'; // for Web
    
    async function main(): Promise<void> {
        ipc...
    }
    
    const _ = main();

Using the API

Contracts

.NET CoreIpc contracts are defined in the form of .NET interfaces which are reflected on at runtime. Each interface leveraged by the .NET CoreIpc Server must be translated to a TypeScript class so that the JavaScript CoreIpc client can reflect on it at runtime.

For example, consider the following .NET interface:

using System.Threading.Tasks;

public interface IComputingService
{
    Task<double> Sum(double x, double y);
}

It's translation to TypeScript will be:

import { ipc } from '@uipath/coreipc';

@ipc.$service
export class IComputingService {

    @ipc.$operation
    public Sum(x: number, y: number): Promise<number> {
        /* this method body will never be executed */
        throw void 0;
    }

}

NOTE: The contract class methods bodies will never be called at runtime.

Task<T>Promise<T> and TaskPromise<void>:

.NET CoreIpc requires that all contract methods return either Task<T> or Task where Task (non-generic) is a marker for fire-and-forget operations. If an operation should be awaited by the client, even though it doesn't return a value, then Task<T> (traditionally, users of CoreIpc use Task<bool>) should be used nevertheless to go around the fire-and-forget connotation.

The CoreIpc client for JavaScript follows the same logic and restrictions as .NET CoreIpc and it uses the idiomatic Promise<T> thusly:

.NET Type JavaScript counterpart Notes
Task<T> Promise<T> Task<int> and Task<double> would both be translated to Promise<number>
Task Promise<void> Typescript uses erasure for generics meaning that Promise is not distinct from Promise<T>. It also allows void to be a generic argument.

Basic usage

Consider an example where a .NET CoreIpc Server exposes the IComputingService contract over the "DemoServer" named pipe and our JavaScript code is part of a Node.js app. Connecting and calling its method remotely can be achieved thusly:

import { ipc } from '@uipath/coreipc'; // for Node.js
import { ipc } from '@uipath/coreipc-web'; // for Web
import { IComputingService } from './translated-contract';

async function main(): Promise<void> {
    // connecting via NamedPipes (exclusive to Node.js)
    const proxy = ipc.proxy
        .withAddress(options => options.isPipe("TestPipe"))
        .withService(IComputingService);

    // alternatively, connecting via WebSockets (for both Web and Node.js)
    const proxy = ipc.proxy
        .withAddress(options => options.isWebSocket("ws://127.0.0.1:61234"))
        .withService(IComputingService);

    console.log(`1 + 2 === ${await computingService.Sum(1, 2)}`);
}

const _ = main();

Canceling a call

CoreIpc client for JavaScript exports CancellationToken and CancellationTokenSource. These types mirror their .NET counterparts and are employed in CoreIpc contracts.

Considering the following .NET operation exposed over CoreIpc:

public class Operations : IOperations
{
    public async Task<int> LongRunningOperation(int x, bool y, string z, CancellationToken ct)
    {
        ...
        while (true)
        {
            ...
            await Task.Delay(1000, ct);
            ...
            ct.ThrowIfCancellationRequested();
            ...
        }
        ...
    }
}

a JavaScript client can cancel calls by using the CancellationTokenSource and CancellationToken classes:

import { ipc, CancellationToken, CancellationTokenSource, OperationCanceledError } from '@uipath/coreipc';     // for Node.js
import { ipc, CancellationToken, CancellationTokenSource, OperationCanceledError } from '@uipath/coreipc-web'; // for Web

class IOperations {
    public LongRunningOperation(x: number, y: boolean, z: string, ct: CancellationToken): Promise<number> {
        throw void 0;
    }
}

export class Mechanism {
    private _latest: number | undefined;
    private _cts: CancellationTokenSource | undefined;

    public get latest(): number | undefined { return this._latest; }

    constructor(private readonly _proxy: IOperations) {
    }

    public EnsureRefreshStarted(): void {
        if (this._cts !== undefined) {
            return;
        }

        (async function Refresh(): Promise<void> {
            this._cts = new CancellationTokenSource();

            try {
                try {
                    this._latest = await this._proxy.LongRunningOperation(1, true, "foo", this._cts.token);
                } catch (err) {
                    if (err instanceof OperationCanceledError) {
                        console.log('OCE was caught.');
                    }
                }
            } finally {
                this._cts.dispose();
                this._cts = undefined;
            }

        })();
    }

    public MaybeCancel(): void {
        if (this._cts !== undefined) {
            this._cts.cancel();
        }
    }
}

const proxy = ipc.proxy
    .withAddress(options => options.isPipe("DemoServer"))
    .withService(IOperations);

const mechanism = new Mechanism(proxy);

mechanism.Refresh();

setTimeout(
    () => {
        mechanism.MaybeCancel();
        console.log(`latest === ${mechanism.latest}`);
    },
    1000);

Configuring the CoreIpc default request timeout

For any request:

import { ipc, TimeSpan } from '@uipath/coreipc';

const _3seconds = TimeSpan.fromSeconds(3);

ipc.config
    .forAnyAddress()
    .forAnyService()
    .setRequestTimeout(_3seconds);

For a particular pipe name or Web socket URL:

import { ipc, TimeSpan } from '@uipath/coreipc';

const _3seconds = TimeSpan.fromSeconds(3);
const whichPipe = 'this one';

ipc.config
    .forAddress(options => options.isPipe(whichPipe))
    .forAnyService()
    .setRequestTimeout(_3seconds);

// alternatively, for Web sockets:

ipc.config
    .forAddress(options => options.isWebSocket("...URL..."))
    .forAnyService()
    .setRequestTimeout(_3seconds);

For a particular contract:

import { ipc, TimeSpan } from '@uipath/coreipc';
import { ISample } from './SampleContract';

const _3seconds = TimeSpan.fromSeconds(3);

ipc.config
    .forAnyAddress()
    .forService(ISample)
    .setRequestTimeout(_3seconds);

For a particular contract and pipe name/Web socket URL:

import { ipc, TimeSpan } from '@uipath/coreipc';
import { ISample } from './SampleContract';

const whichPipe = 'this one';
const _3seconds = TimeSpan.fromSeconds(3);

ipc.config
    .forAddress(options => options.isPipe(whichPipe))
    .forService(ISample)
    .setRequestTimeout(_3seconds);

// alternatively, for Web Sockets:

ipc.config
    .forAddress(options => options.isWebSocket("...URL..."))
    .forService(ISample)
    .setRequestTimeout(_3seconds);

NOTES:

  • All configurations are stored but choosing the same pipe name / contract in a 2nd configuration call will overwrite the previous configuration.

  • Specific configurations outweigh generic ones, i.e, given the following configurations:

    - set the request timeout in general to 1 second
    - set the request timeout for pipe name "foo" to 2 seconds
    - set the request timeout for pipe name "foo" and contract ISample to 3 seconds
    

    the following facts are correct:

    - the request timeout for pipe name "bar" and contract IAlgebra is 1 second
    - the request timeout for pipe name "foo" and contract IAlgebra is 2 seconds
    - the request timeout for pipe name "foo" and contract ISample is 3 seconds
    

Notes for Contributors

Codebase structure

From a top level, the folders are:

Folder Remarks
📂 src ✅checked in source ✅produces deliverables
📂 test ✅checked in source 🛑doesn't produce deliverables
📂 dist 🛑git-ignored ⭐temporary artifacts during build
📂 dist-packages 🛑git-ignored ⭐the NPM packages

The src folder

The src folder is split into 3 subfolders: node, web and std.

std is the largest and is used in the production of both NPM packages.

node and web are specific to their respective NPM packages: the on targeting Node.js and the one targeting Web respectively.

📂 src
   ╔════════════════╗
   ║  ✅📂 node    ║
   ║    ✅📄 ..    ║
   ║    ✅📄 ..    ║
   ║    ✅📄 ..    ║
 ┌─╫─────────────┐  ║
 │ ║  ✅📂 std  │  ║
 │ ║    ✅📄 .. │  ╠════════╗
 │ ║    ✅📄.   │  ║        ║
 │ ║    ✅📄 .. │  ║        ║
 │ ╚═════════════╪══╝        ║
 │    ✅📂 web  │           ║
 │      ✅📄 .. ├───────────╫─────────────────┐
 │      ✅📄 .. │           ║                 │
 │      ✅📄 .. │           ║             web + std
 └───────────────┘      node + std        produce the
      📂 test          produce the        Web package
        📄 ..        Node.js package           │
        📄 ..                ║                 │
      📂 dist                ║                 │
        📄 ..                ║                 │
        📄 ..                ║                 │
      📂 dist-packages       ↓                 │
        💻 uipath-coreipc-{Version}.tgz        │
        🕸️ uipath-coreipc-web-{Version}.tgz ←──┘

General Guide for contributors

Prerequisites

  • Visual Studio Code
  • Node.js and NPM
  • .NET SDK

Steps from cloning the repo to running the integration tests

  • Clone the repo

  • Open a terminal and navigate to $(REPO)/src/Clients/js/dotnet.

  • Run dotnet build.

  • Navigate up, to $(REPO)/src/Clients/js.

  • Run npm ci.

  • Run npm run build

  • Run npm test