Skip to main content

Overview

The Royalti.io notification system keeps users informed about important events across your workspace through multiple delivery channels. This guide covers how to retrieve, manage, and customize notification preferences for your users.

Why Use Notifications?

  • Stay Informed: Get real-time updates about critical events in your workspace
  • Multi-Channel Delivery: Receive notifications via in-app, email, and push notifications
  • Customizable: Control which events trigger notifications and through which channels
  • Centralized: All notifications are accessible through a unified API
  • Actionable: Notifications link directly to relevant resources for quick action

Notification Channels

Royalti.io supports three notification channels:
ChannelDescriptionUse Case
IN_APPIn-app notifications displayed in the Royalti dashboardReal-time updates while users are actively working
EMAILEmail notifications sent to user’s registered email addressImportant updates that require user attention
PUSHPush notifications sent to mobile devicesUrgent updates when users are away from the dashboard
Users can customize which channels they want to receive for each notification type through their preferences.

Notification Types

Royalti.io generates notifications for events across multiple categories:

User & Collaboration Events

Event TypeDescriptionDefault Channels
USER_CREATEDNew user added to workspaceIN_APP, EMAIL
USER_UPDATEDUser profile or permissions modifiedIN_APP
USER_DELETEDUser removed from workspaceIN_APP, EMAIL
USER_INVITATION_SENTUser invited to workspaceEMAIL
USER_ADDED_TO_SPLITUser added to revenue splitIN_APP, EMAIL
USER_REMOVED_FROM_SPLITUser removed from revenue splitIN_APP, EMAIL

Catalog Management Events

Event TypeDescriptionDefault Channels
ARTIST_CREATEDNew artist profile createdIN_APP
ARTIST_UPDATEDArtist profile modifiedIN_APP
ARTIST_DELETEDArtist profile removedIN_APP
ASSET_CREATEDNew asset (track) addedIN_APP
ASSET_UPDATEDAsset modifiedIN_APP
ASSET_DELETEDAsset removedIN_APP
PRODUCT_CREATEDNew product (album/release) createdIN_APP
PRODUCT_UPDATEDProduct modifiedIN_APP
PRODUCT_DELETEDProduct removedIN_APP

Financial Events

Event TypeDescriptionDefault Channels
PAYMENT_REQUEST_SENTPayment request createdIN_APP, EMAIL
PAYMENT_REQUEST_APPROVEDPayment request approvedIN_APP, EMAIL
PAYMENT_REQUEST_REJECTEDPayment request rejectedIN_APP, EMAIL
PAYMENT_MADE_PROCESSINGPayment being processedIN_APP
PAYMENT_MADE_COMPLETEDPayment successfully completedIN_APP, EMAIL
PAYMENT_MADE_FAILEDPayment failedIN_APP, EMAIL
PAYMENT_DELETEDPayment record deletedIN_APP
REVENUE_CREATEDRevenue entry createdIN_APP
REVENUE_UPDATEDRevenue entry modifiedIN_APP
REVENUE_DELETEDRevenue entry deletedIN_APP
EXPENSE_CREATEDExpense entry createdIN_APP
EXPENSE_UPDATEDExpense entry modifiedIN_APP
EXPENSE_DELETEDExpense entry deletedIN_APP

Quick Start

1

Get Notifications

Retrieve notifications for the authenticated user:
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications?page=1&pageSize=20" \
  -H "Authorization: Bearer YOUR_TOKEN"
Response:
{
  "status": "success",
  "message": "Notifications retrieved successfully",
  "data": {
    "notifications": [
      {
        "id": "dd12f9bc-3ed5-453e-a929-69b6d5267f4b",
        "type": "ASSET_CREATED",
        "title": "Track Created",
        "content": "Track \"Midnight City\" has been created.",
        "resourceType": "Asset",
        "resourceId": "0b13028e-74a9-4d1d-bb9f-12cef06ff236",
        "actionUrl": "/assets/0b13028e-74a9-4d1d-bb9f-12cef06ff236",
        "isRead": false,
        "isImportant": false,
        "createdAt": "2025-08-21T11:42:37.717Z"
      }
    ],
    "total": 34,
    "page": 1,
    "pageSize": 20,
    "totalPages": 2
  }
}
2

Check Unread Count

Get the count of unread notifications:
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/unread/count" \
  -H "Authorization: Bearer YOUR_TOKEN"
Response:
{
  "status": "success",
  "message": "Unread notification count retrieved successfully",
  "data": {
    "count": 5
  }
}
3

Mark as Read

Mark a specific notification as read:
curl -X PATCH "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/dd12f9bc-3ed5-453e-a929-69b6d5267f4b/read" \
  -H "Authorization: Bearer YOUR_TOKEN"
Or mark all notifications as read:
curl -X PATCH "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/mark-all-read" \
  -H "Authorization: Bearer YOUR_TOKEN"
4

Configure Preferences

Customize notification preferences:
curl -X PUT "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/preferences" \
  -H "Authorization: Bearer YOUR_TOKEN" \
  -H "Content-Type: application/json" \
  -d '{
    "preferences": [
      {
        "notificationType": "PAYMENT_MADE_COMPLETED",
        "channels": ["EMAIL", "IN_APP", "PUSH"],
        "isEnabled": true
      },
      {
        "notificationType": "ASSET_CREATED",
        "channels": ["IN_APP"],
        "isEnabled": true
      }
    ]
  }'

Notification Structure

All notifications follow a standardized structure:
{
  "id": "dd12f9bc-3ed5-453e-a929-69b6d5267f4b",
  "type": "PAYMENT_MADE_COMPLETED",
  "title": "Payment Completed",
  "content": "Payment of $1,500.00 to Artist Name has been completed successfully.",
  "resourceType": "payment",
  "resourceId": "pay_123abc",
  "actionUrl": "/payments/pay_123abc",
  "isRead": false,
  "isImportant": true,
  "isArchived": false,
  "metadata": {
    "amount": 1500.00,
    "currency": "USD",
    "recipientName": "Artist Name",
    "method": "stripe"
  },
  "expiresAt": null,
  "TenantId": 2,
  "TenantUserId": "c75c54e3-b6e0-4b89-b443-928727b9a931",
  "createdAt": "2025-01-22T14:30:00.000Z",
  "updatedAt": "2025-01-22T14:30:00.000Z"
}

Field Descriptions

FieldTypeDescription
idstring (UUID)Unique notification identifier
typestringNotification type from NotificationType enum
titlestringShort notification title
contentstringDetailed notification message
resourceTypestringType of resource the notification relates to
resourceIdstringID of the related resource
actionUrlstringURL to navigate to when notification is clicked
isReadbooleanWhether the notification has been read
isImportantbooleanWhether the notification is marked as important
isArchivedbooleanWhether the notification is archived
metadataobjectAdditional event-specific data
expiresAtISO 8601When the notification expires (null for no expiration)
TenantIdintegerWorkspace ID
TenantUserIdstring (UUID)User ID the notification is for
createdAtISO 8601When the notification was created
updatedAtISO 8601When the notification was last updated

API Endpoints

Get Notifications

Retrieve paginated notifications with optional filtering. Endpoint: GET /notifications Query Parameters:
ParameterTypeDefaultDescription
pageinteger1Page number
pageSizeinteger20Number of items per page
isReadboolean-Filter by read status
isArchivedboolean-Filter by archived status
typestring-Filter by notification type
Example:
# Get unread notifications
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications?isRead=false&page=1&pageSize=10" \
  -H "Authorization: Bearer YOUR_TOKEN"

# Get payment notifications
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications?type=PAYMENT_MADE_COMPLETED" \
  -H "Authorization: Bearer YOUR_TOKEN"

Get Unread Count

Get the count of unread notifications for quick badge display. Endpoint: GET /notifications/unread/count Example:
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/unread/count" \
  -H "Authorization: Bearer YOUR_TOKEN"
Response:
{
  "status": "success",
  "message": "Unread notification count retrieved successfully",
  "data": {
    "count": 12
  }
}

Mark Notification as Read

Mark a specific notification as read. Endpoint: PATCH /notifications/{notificationId}/read Path Parameters:
ParameterTypeRequiredDescription
notificationIdstring (UUID)YesID of the notification
Example:
curl -X PATCH "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/dd12f9bc-3ed5-453e-a929-69b6d5267f4b/read" \
  -H "Authorization: Bearer YOUR_TOKEN"

Mark All Notifications as Read

Mark all notifications as read for the authenticated user. Endpoint: PATCH /notifications/mark-all-read Example:
curl -X PATCH "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/mark-all-read" \
  -H "Authorization: Bearer YOUR_TOKEN"
Response:
{
  "status": "success",
  "message": "All notifications marked as read"
}

Notification Preferences

Users can customize which notifications they receive and through which channels.

Get Notification Preferences

Retrieve the current notification preferences for the authenticated user. Endpoint: GET /notifications/preferences Example:
curl -X GET "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/preferences" \
  -H "Authorization: Bearer YOUR_TOKEN"
Response:
{
  "status": "success",
  "message": "Notification preferences retrieved successfully",
  "data": {
    "preferences": [
      {
        "id": "65fdcd80-cada-4c76-83ec-70d8e895774d",
        "notificationType": "PAYMENT_MADE_COMPLETED",
        "channels": ["EMAIL", "IN_APP", "PUSH"],
        "isEnabled": true,
        "createdAt": "2025-05-27T19:55:02.423Z",
        "updatedAt": "2025-05-27T19:55:02.423Z"
      },
      {
        "id": "9e522898-358c-4208-9f7d-94bac43b7cf4",
        "notificationType": "ASSET_CREATED",
        "channels": ["IN_APP"],
        "isEnabled": true,
        "createdAt": "2025-05-27T21:17:56.973Z",
        "updatedAt": "2025-05-27T21:17:56.973Z"
      }
    ]
  }
}

Update Notification Preferences

Customize which notifications to receive and through which channels. Endpoint: PUT /notifications/preferences Request Body:
{
  "preferences": [
    {
      "notificationType": "PAYMENT_MADE_COMPLETED",
      "channels": ["EMAIL", "IN_APP", "PUSH"],
      "isEnabled": true
    },
    {
      "notificationType": "PAYMENT_MADE_FAILED",
      "channels": ["EMAIL", "PUSH"],
      "isEnabled": true
    },
    {
      "notificationType": "ASSET_CREATED",
      "channels": ["IN_APP"],
      "isEnabled": false
    }
  ]
}
Example:
curl -X PUT "https://server26-dot-royalti-project.uc.r.appspot.com/notifications/preferences" \
  -H "Authorization: Bearer YOUR_TOKEN" \
  -H "Content-Type: application/json" \
  -d '{
    "preferences": [
      {
        "notificationType": "PAYMENT_MADE_COMPLETED",
        "channels": ["EMAIL", "IN_APP", "PUSH"],
        "isEnabled": true
      },
      {
        "notificationType": "ASSET_CREATED",
        "channels": ["IN_APP"],
        "isEnabled": false
      }
    ]
  }'
Response:
{
  "status": "success",
  "message": "Notification preferences updated successfully",
  "data": {
    "preferences": [
      {
        "id": "65fdcd80-cada-4c76-83ec-70d8e895774d",
        "notificationType": "PAYMENT_MADE_COMPLETED",
        "channels": ["EMAIL", "IN_APP", "PUSH"],
        "isEnabled": true,
        "createdAt": "2025-05-27T19:55:02.423Z",
        "updatedAt": "2025-01-22T15:30:00.000Z"
      }
    ]
  }
}

Preference Configuration

Available Channels

ChannelDescriptionDelivery Time
EMAILEmail to registered addressImmediate or batched (hourly aggregation)
IN_APPIn-app notification in dashboardReal-time
PUSHPush notification to mobile deviceReal-time

Notification Batching

Some notification types support batching to reduce email volume:
  • Immediate: Critical notifications (payments, failures) are sent immediately
  • Hourly Aggregation: Non-critical notifications (asset created, updates) are batched and sent hourly
  • Daily Cleanup: Expired notifications are automatically removed

Integration Patterns

Real-Time Notification Display

Implement a notification bell with unread count:
async function updateNotificationBell() {
  const response = await fetch(
    'https://server26-dot-royalti-project.uc.r.appspot.com/notifications/unread/count',
    {
      headers: {
        'Authorization': `Bearer ${token}`
      }
    }
  );

  const { data } = await response.json();

  // Update UI badge
  document.getElementById('notification-badge').textContent = data.count;
  document.getElementById('notification-badge').style.display =
    data.count > 0 ? 'block' : 'none';
}

// Poll every 30 seconds
setInterval(updateNotificationBell, 30000);

Notification List Component

Fetch and display notifications with pagination:
async function loadNotifications(page = 1) {
  const response = await fetch(
    `https://server26-dot-royalti-project.uc.r.appspot.com/notifications?page=${page}&pageSize=10&isRead=false`,
    {
      headers: {
        'Authorization': `Bearer ${token}`
      }
    }
  );

  const { data } = await response.json();

  data.notifications.forEach(notification => {
    renderNotification(notification);
  });

  // Handle pagination
  renderPagination(data.page, data.totalPages);
}

function renderNotification(notification) {
  const item = document.createElement('div');
  item.className = notification.isImportant ? 'notification important' : 'notification';
  item.innerHTML = `
    <h4>${notification.title}</h4>
    <p>${notification.content}</p>
    <small>${new Date(notification.createdAt).toLocaleString()}</small>
    ${notification.actionUrl ? `<a href="${notification.actionUrl}">View</a>` : ''}
  `;

  item.addEventListener('click', () => markAsRead(notification.id));

  document.getElementById('notification-list').appendChild(item);
}

async function markAsRead(notificationId) {
  await fetch(
    `https://server26-dot-royalti-project.uc.r.appspot.com/notifications/${notificationId}/read`,
    {
      method: 'PATCH',
      headers: {
        'Authorization': `Bearer ${token}`
      }
    }
  );

  // Refresh notification list
  loadNotifications();
}

Preference Management UI

Allow users to customize notification preferences:
async function loadPreferences() {
  const response = await fetch(
    'https://server26-dot-royalti-project.uc.r.appspot.com/notifications/preferences',
    {
      headers: {
        'Authorization': `Bearer ${token}`
      }
    }
  );

  const { data } = await response.json();

  data.preferences.forEach(pref => {
    renderPreferenceControl(pref);
  });
}

async function updatePreference(notificationType, channels, isEnabled) {
  const response = await fetch(
    'https://server26-dot-royalti-project.uc.r.appspot.com/notifications/preferences',
    {
      method: 'PUT',
      headers: {
        'Authorization': `Bearer ${token}`,
        'Content-Type': 'application/json'
      },
      body: JSON.stringify({
        preferences: [
          {
            notificationType,
            channels,
            isEnabled
          }
        ]
      })
    }
  );

  const result = await response.json();
  console.log('Preferences updated:', result);
}

Best Practices

✅ User Experience
  1. Display unread count prominently in your UI (bell icon with badge)
  2. Mark notifications as read when user views them
  3. Provide action links to relevant resources when applicable
  4. Group similar notifications to reduce clutter
  5. Auto-refresh notification count periodically (every 30-60 seconds)
  6. Show timestamps in user-friendly format (e.g., “5 minutes ago”)
  7. Highlight important notifications with visual indicators
  8. Allow bulk actions (mark all as read, archive, etc.)
✅ Performance Optimization
  1. Use pagination to limit the number of notifications loaded at once
  2. Cache unread count to reduce API calls
  3. Implement lazy loading for notification details
  4. Batch preference updates when changing multiple settings
  5. Use WebSockets or polling for real-time updates (if available)
✅ Preference Management
  1. Provide granular controls for each notification type
  2. Set sensible defaults based on notification importance
  3. Allow channel-specific preferences (email vs. in-app vs. push)
  4. Explain notification types so users understand what they’re opting into
  5. Respect user preferences immediately - no delay in applying changes
⚠️ Common Pitfalls
  1. Don’t overwhelm users with too many notifications
  2. Don’t ignore unread status - always mark as read when viewed
  3. Don’t hardcode notification types - use the API to get available types
  4. Don’t skip error handling - API calls may fail
  5. Don’t forget to implement pagination - notification lists can grow large

Troubleshooting

Notifications Not Appearing

✅ Checklist:
  1. Check notification preferences - is the event type enabled?
  2. Verify the user has the correct permissions
  3. Check if notifications are filtered by isRead or isArchived
  4. Ensure the tenant ID matches the authenticated user’s workspace
  5. Check the API response for errors

Missing Notification Preferences

✅ Solutions:
  1. Preferences are created on-demand when first updated
  2. Default preferences are used if none are set
  3. Call GET /notifications/preferences to see current state
  4. Update preferences with PUT /notifications/preferences

High Unread Count

✅ Management:
  1. Implement “Mark all as read” functionality
  2. Allow users to archive old notifications
  3. Set expiration dates for less important notifications
  4. Encourage users to customize preferences to reduce noise

FAQ

Yes, notifications are workspace-scoped. When events occur in your workspace (e.g., a teammate creates an asset), all relevant users receive notifications based on their preferences and permissions.
Notifications are stored indefinitely unless they have an expiresAt date. Expired notifications are automatically removed by the daily cleanup job at 2 AM UTC.
Yes, you can disable notifications by setting isEnabled: false for each notification type in your preferences. Alternatively, you can choose to only enable the IN_APP channel and manage notifications within the dashboard.
In-app and push notifications are sent in real-time. Email notifications for non-critical events may be batched and sent hourly to reduce email volume. Critical notifications (payments, failures) are always sent immediately.
Currently, notifications are workspace-level and cannot be filtered by specific resources. You can control notification types and channels, but not individual resources. This feature may be added in future updates.
Failed notifications are automatically retried based on the channel’s retry strategy. If all retry attempts fail, the notification is logged for debugging but the in-app notification remains accessible through the API.
Notification templates are currently managed by Royalti.io and cannot be customized per workspace. However, you can control which notifications you receive through your preferences.

Support & Resources

API Documentation: Related Guides: External Resources:
Last Updated: January 2025 Version: 1.0