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 🙏

© 2026 – Pkg Stats / Ryan Hefner

@tvlabs/wdio-service

v0.1.13

Published

WebdriverIO service that provides a better integration into TV Labs

Downloads

26,995

Readme

Introduction

npm

The @tvlabs/wdio-service package uses a websocket to connect to the TV Labs platform before an Appium session begins, logging events relating to build upload and session creation as they occur. This offloads the responsibility of creating the TV Labs session from the POST /session Webdriver endpoint, leading to more reliable session requests and creation.

If a build path is provided, the service first makes a build upload request to the TV Labs platform, and then sets the tvlabs:build capability to the newly created build ID.

The service then makes a session request and then subscribes to events for that request. Once the session has been filled and is ready for the Webdriver script to begin, the service receives a ready event with the TV Labs session ID. This session ID is injected into the capabilities as tvlabs:session_id on the Webdriver session create request.

Additionally, the service automatically injects an Authorization header with your API key (formatted as Bearer ${apiKey}) into all requests to the TV Labs platform. If you have already set an Authorization header in your configuration, the service will respect your existing header and not override it.

The service also adds a unique request ID for each request made. The service will generate and attach an x-request-id header before each request to the TV Labs platform. This can be used to correlate requests in the client side logs to the Appium server logs.

Installation

In your WebdriverIO project, run one of the following commands to install:

NPM

npm i --save @tvlabs/wdio-service

Yarn

yarn add @tvlabs/wdio-service

Usage

WebdriverIO Test Runner

To use this as a WebdriverIO test runner service, include the service in your WebdriverIO configuration file (e.g. wdio.conf.ts) with your TV Labs API key set in the options.

import { TVLabsService } from '@tvlabs/wdio-service';

export const config = {
  // ...
  services: [[TVLabsService, { apiKey: process.env.TVLABS_API_KEY }]],
  // ...
};

Standalone (remote)

To use this with WebdriverIO remote but without the test runner, call beforeSession() before instantiating the remote.

import { remote } from 'webdriverio';
import { TVLabsService } from '@tvlabs/wdio-service';

async function run() {
  const capabilities = { ... };

  const wdOpts = {
    capabilities,
    hostname: 'appium.tvlabs.ai',
    port: 4723
  };

  // NOTE: wdOpts must be the same reference passed to remote()
  const service = new TVLabsService(
    { apiKey: process.env.TVLABS_API_KEY },
    capabilities,
    wdOpts,
  );

  await service.beforeSession(wdOpts, capabilities, [], '');

  const driver = await remote(wdOpts);

  try {
    // ...
  } finally {
    await driver.deleteSession();
  }
}

run();

Attaching to an Existing Session

WebdriverIO's attach() binds a driver to a session that was created earlier — possibly by a different process. Use TVLabsService.fromSession() alongside attach() to keep TV Labs telemetry (lastRequestId(), requestMetadata(), sessionMetadata()) working for the attaching driver, without recreating the session.

The portable handle is the Appium session ID — the value of driver.sessionId after the original remote() call. Pass it through whatever channel fits your setup (env var, fixture, queue message, file).

Creating the session

import { remote } from 'webdriverio';
import { TVLabsService } from '@tvlabs/wdio-service';

const capabilities = { ... };
const wdOpts = {
  capabilities,
  hostname: 'appium.tvlabs.ai',
  port: 4723,
};

const service = new TVLabsService(
  { apiKey: process.env.TVLABS_API_KEY },
  capabilities,
  wdOpts,
);

await service.beforeSession(wdOpts, capabilities, [], '');
const driver = await remote(wdOpts);

// Hand off driver.sessionId to wherever the attaching process will pick it up
const appiumSessionId = driver.sessionId;

Attaching to the session

import { attach } from 'webdriverio';
import { TVLabsService } from '@tvlabs/wdio-service';

const wdOpts = {
  capabilities: {},
  hostname: 'appium.tvlabs.ai',
  port: 4723,
};

// appiumSessionId received from the session-creating process
const service = TVLabsService.fromSession(
  appiumSessionId,
  { apiKey: process.env.TVLABS_API_KEY },
  wdOpts,
);

const driver = await attach({
  sessionId: appiumSessionId,
  ...wdOpts,
  options: wdOpts,
});

try {
  const element = await driver.$('#my-button');
  await element.click();

  const requestId = service.lastRequestId();
  const metadata = await service.requestMetadata(appiumSessionId, requestId);
  const session = await service.sessionMetadata(appiumSessionId);
} finally {
  await service.disconnect();
  // Only delete the session if this process owns its lifecycle.
}

Note: lastRequestId() is per-instance — each TVLabsService tracks request IDs that flow through its own transformRequest hook. Use requestMetadata() and sessionMetadata() (both keyed by Appium session ID) for telemetry that needs to span instances.

Options

apiKey

  • Type: string
  • Required: Yes
  • Description: TV Labs API key used for authentication to the platform

buildPath

  • Type: string
  • Required: No
  • Description: Path to the packaged build to use for the session. When provided, this will perform a build upload before the session is created, and sets the tvlabs:build capability to the newly created build ID. The build is uploaded under the organizations default App unless the app option is provided

app

  • Type: string
  • Required: No
  • Description: The slug of the app on the TV Labs platform to use to upload the build. When not provided, the organization's default app is used. You may find or create an app on the Apps page in the TV Labs platform.

retries

  • Type: number
  • Required: No
  • Default: 3
  • Description: Maximum number of attempts to create a session before failing

reconnectRetries

  • Type: number
  • Required: No
  • Default: 5
  • Description: Maximum number of attempts to re-connect if the connection to TV Labs is lost.

attachRequestId

  • Type: boolean
  • Required: No
  • Default: true
  • Description: Controls whether or not to attach an x-request-id header to each request made to the TV Labs platform.

continueOnError

  • Type: boolean
  • Required: No
  • Default: false
  • Description: Whether to continue the session request if any step fails. When true, the session request will still be made with the original provided capabilities. When false, the service will exit with a non-zero code.

Methods

lastRequestId()

  • Returns: string | undefined
  • Description: Returns the last request ID attached to a request made to the TV Labs platform. Useful for correlating client-side logs with server-side logs. Returns undefined if no request has been made yet or if attachRequestId is false.

requestMetadata()

  • Parameters: appiumSessionId: string, requestIds: string | string[]
  • Returns: Promise<TVLabsRequestMetadata | TVLabsRequestMetadataResponse>
  • Description: Fetches metadata for one or more request IDs from the TV Labs platform. Pass a single request ID to get its metadata directly; pass an array to get a map keyed by request ID.

Note: Request metadata is processed asynchronously on the TV Labs platform. Fetch after the session ends, or wait a few seconds after the request, to ensure the data is available.

Example

import { remote } from 'webdriverio';
import { TVLabsService } from '@tvlabs/wdio-service';

const capabilities = { ... };
const wdOpts = { ... };

const service = new TVLabsService(
  { apiKey: process.env.TVLABS_API_KEY },
  capabilities,
  wdOpts,
);

await service.beforeSession(wdOpts, capabilities, [], '');

const driver = await remote(wdOpts);
let requestId;

try {
  const element = await driver.$('#my-button');
  await element.click();
  requestId = service.lastRequestId();
} finally {
  await driver.deleteSession();
}

// Single request
if (requestId) {
  const metadata = await service.requestMetadata(driver.sessionId, requestId);
  console.log('Request metadata:', metadata);
}

// Multiple requests
const multiMetadata = await service.requestMetadata(driver.sessionId, [
  'request-id-123',
  'request-id-456',
  'request-id-789',
]);
console.log('Multiple requests metadata:', multiMetadata);

sessionMetadata()

  • Parameters: appiumSessionId: string
  • Returns: Promise<TVLabsSessionMetadataResponse>
  • Description: Fetches metadata for a session by Appium session ID from the TV Labs platform.

Note: Partial metadata will be available immediately after session creation. Fields such as recording end time, session end time, and session duration are populated asynchronously after the session ends — they will be null until then.

Tip: The response includes both IDs — data.id is the TV Labs session ID (useful for deep-linking into https://tvlabs.ai/app/sessions/:id) and data.appium.session_id is the Appium session ID. Consumers that only have the Appium session ID can resolve the TV Labs session ID from this response.

Example

import { remote } from 'webdriverio';
import { TVLabsService } from '@tvlabs/wdio-service';

const capabilities = { ... };
const wdOpts = { ... };

const service = new TVLabsService(
  { apiKey: process.env.TVLABS_API_KEY },
  capabilities,
  wdOpts,
);

await service.beforeSession(wdOpts, capabilities, [], '');

const driver = await remote(wdOpts);
const { sessionId } = driver;

try {
  // ...
} finally {
  await driver.deleteSession();
}

const sessionMetadata = await service.sessionMetadata(sessionId);

const { data: {
  device: {
    make: { display_name: make },
    model: { display_name: model, year },
  },
  recording: { start_time },
} } = sessionMetadata;

console.log(`Recording on ${year} ${make} ${model} started at ${start_time}`);

disconnect()

  • Returns: Promise<void>
  • Description: Closes any long-lived connections held by the service (currently the metadata channel WebSocket opened by requestMetadata()). Call this when you are finished with the service so the process can exit. Safe to call when no channel was opened, and safe to call multiple times.

Note: Await all in-flight requestMetadata() calls before calling disconnect(). Tearing down the metadata channel while a request is pending will cause that call to reject.

Static Factory

TVLabsService.fromSession()

  • Parameters: appiumSessionId: string, options: TVLabsServiceOptions, wdOpts: Options.WebdriverIO
  • Returns: TVLabsService
  • Description: Creates a TVLabsService instance bound to an existing Appium session without calling beforeSession(). Use this when you have an Appium session ID (the value of driver.sessionId after a previous remote() call) and want to attach to that session via WebdriverIO's attach().

The factory:

  • Injects the Authorization header on wdOpts.
  • Installs the x-request-id tracking hook on wdOpts (unless attachRequestId: false), so lastRequestId() works on the attached driver.
  • Records the Appium session ID for use by appiumSessionId, requestMetadata(), and sessionMetadata().

Important: wdOpts must be the same reference later passed to attach().

Note: Calling beforeSession() on a service created via fromSession() throws a SevereServiceError to prevent accidentally creating a duplicate session.

Tip: To verify the session exists before using it, call await service.sessionMetadata(appiumSessionId) immediately after construction. It throws if the session is not found or the API key cannot access it.

appiumSessionId

  • Type: getter
  • Returns: string | undefined
  • Description: Returns the Appium session ID bound via fromSession(), or undefined for service instances created through the standard constructor.