Skip to content

tmelliott/rserve-ts

Repository files navigation

rserve-ts: Typescript wrapper for rserve library

NPM Version

This is a typescript wrapper for the rserve library. The goal is to provide modern async API alternatives with typescript support (via zod).

The package wraps the main connection function and allows users to evaluate arbitrary R code on the server. This will be extended to full OCAP support.

Installation

npm install rserve-ts

Usage

You need to have an Rserve server running for the following to work.

import RserveClient from "rserve-ts";

(async () => {
  const R = await RserveClient.create({
    host: "http://127.0.0.1:8081",
  });

  const rversion = await R.eval("R.version.string", R.character());
  console.log("Connected to ", rversion);
})();

OCAP mode

In OCAP mode, only pre-defined functions can be called. This is useful for restricting the R code that can be executed on the server, or for developing applications.

Zod is used to define function schemas, with additional R types defined.

# ocap.R
library(Rserve)
oc.init <- function() {
  ocap(function() {
    list(
      # Ocaps are functions that javascript can call
      add <- ocap(function(a, b) {
        a + b
      }),
      dist <- ocap(function(which = c('normal', 'uniform')) {
        # Ocaps can return new ocaps, too!
        # This could be useful for progressively revealing functionality, etc.
        switch(which,
          normal = list(sample = ocap(function(n) rnorm(n))),
          uniform = list(sample = ocap(function(n) runif(n)))
        )
      })
    )
  })
}
// app.ts
import { z } from "zod";
import { Robj } from "../index";

export const appSchema = {
  add: Robj.ocap([z.number(), z.number()], Robj.numeric(1)),
  dist: Robj.ocap(
    [z.enum(["normal", "uniform"])],
    Robj.list({
      sample: Robj.ocap([z.number()], Robj.numeric()),
    })
  ),
};
// app.ts
import R from "../index";
import { appSchema } from "./app";

(async () => {
  const s = await R.create({ host: "ws://localhost:8181" });
  const app = await s.ocap(appSchema);

  const sum = await app.add(16, 32);
  const cart = { total: sum };
  cart.total;

  const { sample } = await app.dist("uniform");
  const x = await sample(235);
  if (typeof x === "number") x + 5;
  else {
    // deal with an array
  }
})();