Skip to main content

Using Logs for Troubleshooting

Description — Couchbase Lite on React Native — Using Logs for Troubleshooting
Related Content — Troubleshooting Queries | Troubleshooting Crashes

note
  • The retrieval of logs from the device is out of scope of this feature.

Introduction

Couchbase Lite provides a robust Logging API — see: API References for Logging classes — which make debugging and troubleshooting easier during development and in production. It delivers flexibility in terms of how logs are generated and retained, whilst also maintaining the level of logging required by Couchbase Support for investigation of issues.

Log output is split into the following streams:

  • File based logging

    Here logs are written to separate log files filtered by log level, with each log level supporting individual retention policies.

  • Console based logging

    You can independently configure and control console logs, which provides a convenient method of accessing diagnostic information during debugging scenarios. With console logging, you can fine-tune diagnostic output to suit specific debug scenarios, without interfering with any logging required by Couchbase Support for the investigation of issues.

  • Custom logging

    For greater flexibility you can implement a custom logging class using the ILogger interface.

Log Sink API

Version 1.0 introduces the Log Sink API which provides three types of log sinks for flexible logging control.

Log Levels

LevelValueDescription
LogLevel.DEBUG0Most verbose - all logs
LogLevel.VERBOSE1Detailed diagnostic logs
LogLevel.INFO2Informational messages
LogLevel.WARNING3Warning messages only
LogLevel.ERROR4Error messages only
LogLevel.NONE5No logging

Log Domains

DomainDescription
LogDomain.DATABASEDatabase operations
LogDomain.QUERYQuery execution and planning
LogDomain.REPLICATORReplication activity
LogDomain.NETWORKNetwork operations
LogDomain.LISTENERChange listeners
LogDomain.ALLAll domains (new in 1.0)

Console Log Sink

Console based logging outputs logs to the system console (stdout/stderr), useful for development and debugging.

Example 1. Enable Console Logging

import { LogSinks, LogLevel, LogDomain } from 'cbl-reactnative';

// Enable verbose logging for all domains
await LogSinks.setConsole({
level: LogLevel.VERBOSE,
domains: [LogDomain.ALL]
});

Example 2. Console with Specific Domains

// Log only replication and network activity
await LogSinks.setConsole({
level: LogLevel.INFO,
domains: [LogDomain.REPLICATOR, LogDomain.NETWORK]
});

Example 3. Disable Console Logging

// Disable console logging
await LogSinks.setConsole(null);

File Log Sink

File logging writes logs to files on the device with automatic rotation and retention policies.

Example 4. Enable File Logging

import { Platform } from 'react-native';
import RNFS from 'react-native-fs';

// Determine platform-specific log directory
const logDirectory = Platform.OS === 'ios'
? RNFS.DocumentDirectoryPath + '/logs'
: RNFS.ExternalDirectoryPath + '/logs';

await LogSinks.setFile({
level: LogLevel.INFO,
directory: logDirectory,
maxKeptFiles: 5, // Keep 5 old log files
maxFileSize: 1024 * 1024, // 1MB max file size
usePlaintext: true // Use plaintext format
});
File Rotation

When a log file reaches maxFileSize, it's closed and a new one is created. Old files exceeding maxKeptFiles are automatically deleted.

Example 5. Disable File Logging

await LogSinks.setFile(null);

Custom Log Sink

Custom logging allows you to implement your own logging logic with a callback function.

Example 6. Custom Logging with Callback

await LogSinks.setCustom({
level: LogLevel.ERROR,
domains: [LogDomain.ALL],
callback: (level, domain, message) => {
const timestamp = new Date().toISOString();
console.log(`[${timestamp}] [${domain}] ${message}`);

// You can also send to analytics, log to database, etc.
}
});

Example 7. Disable Custom Logging

await LogSinks.setCustom(null);

Using Multiple Log Sinks

You can enable multiple log sinks simultaneously for different purposes.

Example 8. Development and Production Configuration

if (__DEV__) {
// Development: Verbose console logging
await LogSinks.setConsole({
level: LogLevel.VERBOSE,
domains: [LogDomain.ALL]
});
} else {
// Production: File logging for warnings and errors
await LogSinks.setFile({
level: LogLevel.WARNING,
directory: logDirectory,
maxKeptFiles: 7,
maxFileSize: 2 * 1024 * 1024,
usePlaintext: true
});

// Also send errors to analytics
await LogSinks.setCustom({
level: LogLevel.ERROR,
domains: [LogDomain.ALL],
callback: (level, domain, message) => {
Analytics.logError({ level, domain, message });
}
});
}

Platform Considerations

iOS:

  • Log files are stored in the app's Documents directory
  • Path: RNFS.DocumentDirectoryPath + '/logs'
  • Accessible via iTunes File Sharing if enabled in Info.plist

Android:

  • Log files are stored in the app's external directory
  • Path: RNFS.ExternalDirectoryPath + '/logs'
  • May require storage permissions in AndroidManifest.xml