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

@fullstory/server-api-client

v1.1.1

Published

The official FullStory server API client SDK for NodeJS.

Downloads

8,400

Readme

FullStory NodeJS Client SDK

CircleCI

The Official FullStory server API client SDK for NodeJS. The SDK is designed for easy usage to make server-to-server HTTP API requests. For more information visit our developer site.

To use our in-browser module, we also provide a FullStory browser SDK.

NodeJS Support

  • Node.js >= 14

Getting Started

Installation

# npm
npm install @fullstory/server-api-client
# yarn
yarn add @fullstory/server-api-client

Usage

Initializing the Client

Use the init function to initialize the FullStory client with your API key. The FullStoryOptions used in the init function will be used for all server API requests.

import { init } from '@fullstory/server-api-client';

const fsOpts: FullStoryOptions = {
  // apiKey is required to init the FullStory client
  apiKey: '<YOUR_API_KEY>',
  
  // integrationSource is generally intended for FullStory developers
  // or FullStory partners while building integrations.
  //Generally it should be left empty.
  integrationSource: 'INTEGRATION_NAME' 
};

const fsClient = init(fsOpts);

Users

  • Use users to access the users api

    const { users } = fsClient;
  • Create a user

    const createResponse = await users.create({
      body: {
        uid: 'user123',
        display_name: 'Display Name',
        email: '[email protected]',
        properties: {
          pricing_plan: 'paid',
          popup_help: true,
          total_spent: 14.55,
          },
        },
      // optionally provide an idempotencyKey to make
      // the request idempotent
      idempotencyKey: "YOUR_KEY"
      });
  • Get a user

    // get user by the FullStory assigned id 
    const getResponse = await users.get({
      id: '123456',
      includeSchema: true
    });
  • Get users

    // get all users by the application-specified uid
    const listResponse = await users.list({ uid: 'user123' });
  • Update a user

    // update user by the fullstory assigned id
    const updatedResponse = await users.update({
      id: '123456',
      body: { display_name: 'New Display Name' }
    });
  • Delete a user

    // delete user by the FullStory assigned id
    await users.delete({ id: '123456' });
    
    // delete user by the application-specified uid
    await users.delete({ uid: 'xyz123' });

Events

  • Use events to access the events api

    const { events } = fsClient;
  • Create a event

    const createResponse = await events.create({
      body: {
        session: {
          id: '123:456',
        },
        context: {
          browser: {
            url: 'https://www.example.com',
            initial_referrer: 'https://www.referrer.com',
          },
          location: {
            latitude: 33.748997,
            longitude: -84.387985,
          },
        },
        name: 'Support Ticket',
        timestamp: '2022-03-15T14:23:23Z',
        properties: {
          id: 424242,
          priority: 'Normal',
          source: 'Email',
          title: 'Account locked out',
        }
      }
    });

Batch Import Job

  • Creating import job objects allows clients to batch import items such as users or events. First create a job object:

    // array of users to be imported
    const requests = [
      {
        uid: 'user123',
        display_name: 'Display Name',
        email: '[email protected]',
        properties: {
          pricing_plan: 'paid',
          popup_help: true,
          total_spent: 14.5,
        },
      },
      {
        uid: 'user456',
      },
      {
        uid: 'user789',
        display_name: 'A New User',
      },
    ];
    
    // create a job object
    const job = users.batchCreate({
      body: { requests },
      // include schema in the server response
      // when retrieving imported users
      includeSchema: true,
    });
    
    // you can add more requests before executing
    job.add(
      { display_name: 'Someone Else' },
      { display_name: 'Last User' },
    );
  • Create a batch events import job

    // array of events requests to be imported
    const requests = [
      {
        user: {
          id: '123456',
        },
        session: {
          use_most_recent: true,
        },
        context: {
          web: {
            url: 'https://www.example.com',
          },
        },
        name: 'Support Ticket',
        timestamp: '2022-03-15T14:23:23Z',
        properties: {
          source: 'Email',
          title: 'Account locked out',
        },
      },
    ];
    
    // create a job object
    const job = events.batchCreate({ 
      body: { requests }
      includeSchema: true,
    });
    
    // you can add more requests before executing
    job.add({
        user: {
          id: '999999',
        },
        name: 'Events - 1',
        timestamp: '2022-03-15T14:23:23Z',
      },{
        user: {
          id: '100000',
        },
        name: 'Events - 2',
        timestamp: '2022-03-15T14:23:23Z',
      });
      
    job
      // register listeners before executing
      .on('created', job => {
        console.log('batch job successfully created', job.getId());
      })
      .on('processing', job => {
        console.log('get notified when job status polled and is still processing', job.getId());
      })
      .on('error', error => {
        console.log('an error had occurred during the import', error);
      })
      .on('abort', error => {
        console.error('an unrecoverable error had occurred and the job had been aborted', error);
      })
      .on('done', (imported, failed) => {
        console.log('the batch imported job is done');
        console.log('items successfully imported', imported);
        console.log('items failed to be imported', failed);
      })
    
      // execute the job by calling the server API
    job.execute();

Note: Any error, abort or done listeners registered after the job has been executed, the callback may be called immediately with any historical data from the job, if any error had occurred, or if the job is already aborted or done, respectively.

  • Batch import job lifecycle.

    Optional events listeners can be added for the batch import job. The batch import job will manage the following lifecycle:

    1. When the job's execute method is called, an API request is made to the server to create the import the job:
    1. Once the job is created successfully, created listeners are invoked with the job's metadata.

    2. The job then starts to poll on the server API to get the latest job status at an interval:

    • users

    • events

      Each successful poll will invoke the processing listeners.

    1. When the job status reaches a done state (either COMPLETED or FAILED), the job automatically starts to retrieve the imported items, by calling the "get batch imports" APIs:
    • users

    • events

      or "get batch errors" APIs if there were errors:

    • users

    • events

      And lastly the done listeners are invoked with the results.

    1. The error listeners are called anytime an error is encountered, may be called more than once.

    2. The abort listeners is called only once per job, if non-recoverable errors had occurred, such as multiple API failures had been encountered that exceeds the max number of retries.

  • Restart a job

    If a job had been successfully created, but aborted for some reason, you may restart the polling of the same job by calling restart.

    // restart failed job
    const job = events.batchCreate({
      body: { requests }
    });
    job.on('abort', () => {
      // logic to determine if should restart
      job.restart();
    });
      
    // Or
    // restart a failed job with a job id
    const job = events.batchCreate().restart('your-job-id');

Batch Job Options

Each job can be created with different BatchJobOptions when needed.

const options: BatchJobOptions = {
  // poll job status every one minute
  pollInterval: 60000,
  // retry 5 times on API errors before aborting
  maxRetry: 5,
}

const createResponse = await users.batchCreate(
    { 
      body: { requests: [{ uid: 'user123' }] },
      includeSchema: true,
    },
    options
);

Multiple batch import jobs

It is recommended to have one batch import job of a resource type at a given time. However in case you need to create multiple batch import jobs by calling batchCreate multiple times. The jobs may be concurrently executed. In this case that the server APIs may return rate limiting errors. It is recommended to adjust the pollInterval option accordingly.

The batch import job execution will retry if rate limit or other transient errors are encountered up to a max number of retries.

Error Handling

  • init may throw an error when the required fields are not satisfied.

  • Functions in the FullStory client may throw FSError objects. For import jobs, the FSError object is provided to the on error callbacks.

  • If needed, the name field identifies the type of error. The SDK also provides isFSError function to check if an error object is a typed FSError.

See errors for more information.

try {
  ...
} catch (err: unknown) {
  if (isFSError(err)) {
    switch (err.name) {
       case FSErrorName.ERROR_INVALID_ARGUMENT:
        console.log('An argument provided to the SDK is invalid.');
        break;
      case FSErrorName.ERROR_RATE_LIMITED:
        console.log('FullStory server API returned HTTP 429 Too Many Requests.');
        console.log(`received 'retry - after' header, retry in ${err.getRetryAfter()} milliseconds.`);
        break;
      case FSErrorName.ERROR_PARSE_RESPONSE:
        console.log('Unable to parse FullStory server API responses.');
        console.log(`Raw response received ${(err as FSParserError).fsErrorPayload}.`);
        break;
      case FSErrorName.ERROR_FULLSTORY:
        // API errors except 429s
        console.log('FullStory server API returned non-2xx responses.');
        console.log(`Status: ${(err as FSApiError).httpStatusCode}.`);
        console.log(`Response: ${(err as FSApiError).fsErrorPayload}.`);
        break;
      case FSErrorName.ERROR_TIMEOUT:
        console.log('Timeout exceeded while making FullStory server API requests.');
        break;
      case FSErrorName.ERROR_MAX_RETRY:
        console.log('Max number of retries exceeded during batch import job execution.');
        break;
      case FSErrorName.ERROR_UNKNOWN:
        console.log('Unexpected error occurred.');
        break;
    }
  }
}