Skip to main content
  • This is currently only compatible with setDocuments method.
  • Ensure that the data providers are set prior to calling identify method.
  • The data provider methods must return the correct status code (e.g. 200 for success, 500 for errors) and success boolean in the response object. This ensures proper error handling and retries.

Overview

Velt supports self-hosting your recording annotation PII data:
  • Recording annotation identity (user from field), transcription, and attachment URLs can be stored on your own infrastructure, with only necessary identifiers on Velt servers.
  • Velt Components automatically hydrate recording data in the frontend by fetching from your configured data provider.
  • This gives you full control over recording PII while maintaining all Velt recorder features.

How does it work?

When recorder annotations are created, updated, or deleted:
  1. The SDK uses your configured RecorderAnnotationDataProvider to handle storage and retrieval.
  2. Your data provider implements three optional methods:
    • get: Fetches recording annotation PII from your database
    • save: Stores PII fields and returns updated transcription/attachments
    • delete: Removes PII fields from your database
The process works as follows: When a recording annotation operation occurs:
  1. The SDK first attempts the operation on your storage infrastructure
  2. If successful:
    • The SDK updates Velt’s servers with minimal identifiers
    • The RecorderAnnotation object is updated with isRecorderResolverUsed: true while PII is being fetched
    • Once PII is available, the annotation is hydrated with identity, transcription, and attachment data
  3. If the operation fails, no changes are made to Velt’s servers and the operation is retried if you have configured retries.

Implementation Approaches

You can implement recorder self-hosting using either of these approaches:
  1. Endpoint based: Provide endpoint URLs and let the SDK handle HTTP requests
  2. Function based: Implement get, save, and delete methods yourself
Both approaches are fully backward compatible and can be used together.
FeatureFunction basedEndpoint based
Best ForComplex setups requiring middleware logic, dynamic headers, or transformation before sendingStandard REST APIs where you just need to pass the request “as-is” to the backend
ImplementationYou write the fetch() or axios codeYou provide the url string and headers object
FlexibilityHighMedium
SpeedMediumHigh

Endpoint based DataProvider

Instead of implementing custom methods, you can configure endpoints directly and let the SDK handle HTTP requests.

getConfig

Config-based endpoint for fetching recording annotation PII. The SDK automatically makes HTTP POST requests with the request body.
const recorderDataProvider = {
  config: {
    getConfig: {
      url: 'https://your-backend.com/api/velt/recordings/get',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    }
  }
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

saveConfig

Config-based endpoint for saving recording annotation PII. The SDK automatically makes HTTP POST requests with the request body.
const recorderDataProvider = {
  config: {
    saveConfig: {
      url: 'https://your-backend.com/api/velt/recordings/save',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    }
  }
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

deleteConfig

Config-based endpoint for deleting recording annotation PII. The SDK automatically makes HTTP POST requests with the request body.
const recorderDataProvider = {
  config: {
    deleteConfig: {
      url: 'https://your-backend.com/api/velt/recordings/delete',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    }
  }
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

Endpoint based Complete Example

const recorderDataProvider = {
  config: {
    getConfig: {
      url: 'https://your-backend.com/api/velt/recordings/get',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    },
    saveConfig: {
      url: 'https://your-backend.com/api/velt/recordings/save',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    },
    deleteConfig: {
      url: 'https://your-backend.com/api/velt/recordings/delete',
      headers: { 'Authorization': 'Bearer YOUR_TOKEN' }
    },
    additionalFields: ['customField1', 'customField2'],
    resolveTimeout: 5000,
    getRetryConfig: { retryCount: 3, retryDelay: 2000 },
    saveRetryConfig: { retryCount: 3, retryDelay: 2000 },
    deleteRetryConfig: { retryCount: 3, retryDelay: 2000 }
  },
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

Function based DataProvider

Implement custom methods to handle data operations yourself.

get

Fetch recording annotation PII data from your database. Called when annotations need to be hydrated in the frontend.
const getRecordingFromDB = async (request) => {
  const response = await fetch('/api/velt/recordings/get', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const recorderDataProvider = {
  get: getRecordingFromDB,
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

save

Store recording annotation PII fields. Called when a recorder annotation is created or updated.
const saveRecordingToDB = async (request) => {
  const response = await fetch('/api/velt/recordings/save', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const recorderDataProvider = {
  save: saveRecordingToDB,
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

delete

Remove recording annotation PII from your database. Called when a recorder annotation is deleted.
const deleteRecordingFromDB = async (request) => {
  const response = await fetch('/api/velt/recordings/delete', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const recorderDataProvider = {
  delete: deleteRecordingFromDB,
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

config

Configuration for the recorder data provider.
  • Type: ResolverConfig. Relevant properties:
    • resolveTimeout: Timeout duration (in milliseconds) for resolver operations
    • getRetryConfig: RetryConfig. Configure retry behavior for get operations.
    • saveRetryConfig: RetryConfig. Configure retry behavior for save operations.
    • deleteRetryConfig: RetryConfig. Configure retry behavior for delete operations.
const recorderResolverConfig = {
  resolveTimeout: 5000,
  getRetryConfig: { retryCount: 3, retryDelay: 2000 },
  saveRetryConfig: { retryCount: 3, retryDelay: 2000 },
  deleteRetryConfig: { retryCount: 3, retryDelay: 2000 }
};

uploadChunks

Controls whether recording media is uploaded as individual chunks during recording, or as a single full recording after the annotationId is assigned.
  • Type: boolean
  • Default: false (upload full recording)
Set to true to upload individual chunks as they are recorded, which can be useful for long recordings or unreliable network conditions.

storage

A scoped AttachmentDataProvider for recorder media files. Use this to route recording file uploads to a separate storage backend from your default attachment provider.
const recorderDataProvider = {
  storage: {
    save: async (request) => {
      const formData = new FormData();
      formData.append('file', request.file);
      formData.append('metadata', JSON.stringify(request.metadata));
      const response = await fetch('/api/velt/recordings/upload', {
        method: 'POST',
        body: formData
      });
      return await response.json();
    },
    delete: async (request) => {
      const response = await fetch('/api/velt/recordings/upload/delete', {
        method: 'POST',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify(request)
      });
      return await response.json();
    },
  },
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

Function based Complete Example

const getRecordingFromDB = async (request) => {
  const response = await fetch('/api/velt/recordings/get', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const saveRecordingToDB = async (request) => {
  const response = await fetch('/api/velt/recordings/save', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const deleteRecordingFromDB = async (request) => {
  const response = await fetch('/api/velt/recordings/delete', {
    method: 'POST',
    headers: { 'Content-Type': 'application/json' },
    body: JSON.stringify(request)
  });
  return await response.json();
};

const recorderResolverConfig = {
  resolveTimeout: 5000,
  getRetryConfig: { retryCount: 3, retryDelay: 2000 },
  saveRetryConfig: { retryCount: 3, retryDelay: 2000 },
  deleteRetryConfig: { retryCount: 3, retryDelay: 2000 }
};

const recorderDataProvider = {
  get: getRecordingFromDB,
  save: saveRecordingToDB,
  delete: deleteRecordingFromDB,
  config: recorderResolverConfig,
  uploadChunks: false,
  storage: {
    save: async (request) => {
      const formData = new FormData();
      formData.append('file', request.file);
      formData.append('metadata', JSON.stringify(request.metadata));
      const response = await fetch('/api/velt/recordings/upload', {
        method: 'POST',
        body: formData
      });
      return await response.json();
    },
    delete: async (request) => {
      const response = await fetch('/api/velt/recordings/upload/delete', {
        method: 'POST',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify(request)
      });
      return await response.json();
    },
  },
};

<VeltProvider
  apiKey='YOUR_API_KEY'
  dataProviders={{ recorder: recorderDataProvider }}
>
</VeltProvider>

Loading and Upload State Fields

Two fields on RecorderAnnotation indicate recorder resolver status:
FieldTypeDescription
isRecorderResolverUsedbooleantrue while PII is being fetched from the recorder resolver. Use this field to show a loading state in your UI.
isUrlAvailablebooleantrue once the recording URL has been uploaded and is available (not a local blob URL). Use this field to indicate upload progress.

Debugging

You can subscribe to dataProvider events to monitor and debug recorder resolver operations. Filter for the RecorderResolverModuleName.GET_RECORDER_ANNOTATIONS module name to isolate recorder-specific events. You can also use the Velt Chrome DevTools extension to inspect and debug your Velt implementation. RecorderResolverModuleName — Enum of module names for recorder resolver events.
import { useVeltClient } from '@veltdev/react';

const { client } = useVeltClient();

useEffect(() => {
  if (!client) return;

  const subscription = client.on('dataProvider').subscribe((event) => {
    console.log('Data Provider Event:', event);
  });

  return () => subscription?.unsubscribe();
}, [client]);