Works with Expo ββ’β Read the Documentation ββ’β Report Issues
munim-bluetooth is a comprehensive React Native library for all your Bluetooth Low Energy (BLE) needs, supporting both peripheral and central roles. This library allows your React Native app to act as a BLE peripheral (advertising services and characteristics) or as a BLE central (scanning, connecting, and communicating with devices).
Fully compatible with Expo! Works seamlessly with both Expo managed and bare workflows.
Built with React Native's Nitro modules architecture for high performance and reliability.
Note: This library focuses on reliability and platform compatibility. It supports the core BLE features that work consistently across both Android and iOS platforms.
- π Documentation
- π Features
- π¦ Installation
- β‘ Quick Start
- π§ API Reference
- π Usage Examples
- π Troubleshooting
- π Contributing
- π License
Learn about building BLE apps in our documentation!
- π΅ BLE Peripheral Mode: Transform your React Native app into a BLE peripheral device
- π‘ Service Advertising: Advertise custom GATT services with multiple characteristics
- π Real-time Communication: Support for read, write, and notify operations
- β Platform-Supported BLE Advertising: Support for core BLE advertising data types that work reliably on both platforms
- π§ Dynamic Updates: Update advertising data while advertising is active
- π Device Scanning: Scan for BLE devices with filtering options
- π Device Connection: Connect and disconnect from BLE devices
- π GATT Operations: Discover services, read/write characteristics
- π Notifications: Subscribe to characteristic notifications/indications
- πΆ RSSI Monitoring: Read signal strength for connected devices
- π± Cross-platform: Works on both iOS and Android
- π― TypeScript Support: Full TypeScript definitions included
- β‘ High Performance: Built with React Native's Nitro modules architecture
- π Expo Compatible: Works seamlessly with Expo managed and bare workflows
- π Permission Handling: Built-in permission request helpers
npm install munim-bluetooth react-native-nitro-modules
# or
yarn add munim-bluetooth react-native-nitro-modulesnpx expo install munim-bluetooth react-native-nitro-modulesNote: This library requires Expo SDK 50+ and works with both managed and bare workflows. To support Nitro modules, you need React Native version v0.78.0 or higher.
For iOS, the library is automatically linked. However, you need to add the following to your Info.plist:
<key>NSBluetoothAlwaysUsageDescription</key>
<string>This app uses Bluetooth for BLE communication</string>
<key>NSBluetoothPeripheralUsageDescription</key>
<string>This app uses Bluetooth to create a peripheral device</string>For Expo projects, add these permissions to your app.json:
{
"expo": {
"ios": {
"infoPlist": {
"NSBluetoothAlwaysUsageDescription": "This app uses Bluetooth for BLE communication",
"NSBluetoothPeripheralUsageDescription": "This app uses Bluetooth to create a peripheral device"
}
}
}
}For Android, add the following permissions to your AndroidManifest.xml:
<uses-permission android:name="android.permission.BLUETOOTH" />
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
<uses-permission android:name="android.permission.BLUETOOTH_ADVERTISE" />
<uses-permission android:name="android.permission.BLUETOOTH_SCAN" />
<uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />For Expo projects, add these permissions to your app.json:
{
"expo": {
"android": {
"permissions": [
"android.permission.BLUETOOTH",
"android.permission.BLUETOOTH_ADMIN",
"android.permission.BLUETOOTH_ADVERTISE",
"android.permission.BLUETOOTH_SCAN",
"android.permission.BLUETOOTH_CONNECT",
"android.permission.ACCESS_FINE_LOCATION",
"android.permission.ACCESS_COARSE_LOCATION"
]
}
}
}import { startAdvertising, stopAdvertising, setServices } from 'munim-bluetooth'
// Start advertising with basic options
startAdvertising({
serviceUUIDs: ['180D', '180F'],
localName: 'My Device',
manufacturerData: '0102030405',
})
// Set GATT services
setServices([
{
uuid: '180D',
characteristics: [
{
uuid: '2A37',
properties: ['read', 'notify'],
value: 'Hello World',
},
],
},
])
// Stop advertising
stopAdvertising()import {
startScan,
stopScan,
connect,
discoverServices,
readCharacteristic,
subscribeToCharacteristic,
} from 'munim-bluetooth'
// Start scanning
startScan({
serviceUUIDs: ['180D'],
allowDuplicates: false,
scanMode: 'balanced',
})
// Connect to a device (deviceId from deviceFound event)
await connect('device-id-here')
// Discover services
const services = await discoverServices('device-id-here')
// Read a characteristic
const value = await readCharacteristic('device-id-here', '180D', '2A37')
// Subscribe to notifications
subscribeToCharacteristic('device-id-here', '180D', '2A37')import {
startAdvertising,
updateAdvertisingData,
getAdvertisingData,
type AdvertisingDataTypes,
} from 'munim-bluetooth'
// Platform-supported advertising data configuration
const advertisingData: AdvertisingDataTypes = {
// 0x01 - Flags (LE General Discoverable Mode, BR/EDR Not Supported)
flags: 0x06,
// 0x02-0x07 - Service UUIDs (fully supported)
completeServiceUUIDs16: ['180D', '180F'],
incompleteServiceUUIDs128: ['0000180D-0000-1000-8000-00805F9B34FB'],
// 0x08-0x09 - Local Name (fully supported)
completeLocalName: 'My Smart Device',
shortenedLocalName: 'SmartDev',
// 0x0A - Tx Power Level (fully supported)
txPowerLevel: -12,
// 0x14-0x15 - Service Solicitation (fully supported)
serviceSolicitationUUIDs16: ['180D'],
serviceSolicitationUUIDs128: ['0000180D-0000-1000-8000-00805F9B34FB'],
// 0x16, 0x20, 0x21 - Service Data (fully supported)
serviceData16: [
{ uuid: '180D', data: '0102030405' },
{ uuid: '180F', data: '060708090A' },
],
serviceData32: [
{ uuid: '0000180D-0000-1000-8000-00805F9B34FB', data: '0B0C0D0E0F' },
],
// 0x19 - Appearance (partial support)
appearance: 0x03c0, // Generic Watch
// 0x1F - Service Solicitation (32-bit) (fully supported)
serviceSolicitationUUIDs32: ['0000180D'],
// 0xFF - Manufacturer Specific Data (fully supported)
manufacturerData: '4C000215FDA50693A4E24FB1AFCFC6EB0764782500010001C5',
}
// Start advertising with supported data
startAdvertising({
serviceUUIDs: ['180D', '180F'],
advertisingData: advertisingData,
})
// Update advertising data dynamically
updateAdvertisingData({
flags: 0x04,
completeLocalName: 'Updated Device Name',
txPowerLevel: -8,
})
// Get current advertising data
const currentData = await getAdvertisingData()
console.log('Current advertising data:', currentData)Starts BLE advertising with the specified options.
Parameters:
options(object):serviceUUIDs(string[]): Array of service UUIDs to advertiselocalName?(string): Device name (legacy support)manufacturerData?(string): Manufacturer data in hex format (legacy support)advertisingData?(AdvertisingDataTypes): Platform-supported advertising data
Updates the advertising data while advertising is active.
Parameters:
advertisingData(AdvertisingDataTypes): New advertising data
Returns a Promise that resolves to the current advertising data.
Returns: Promise
Stops BLE advertising.
Sets GATT services and characteristics.
Parameters:
services(array): Array of service objects
Checks if Bluetooth is enabled on the device.
Returns: Promise
Requests Bluetooth permissions (Android) or checks authorization status (iOS).
Returns: Promise
Starts scanning for BLE devices.
Parameters:
options?(object):serviceUUIDs?(string[]): Filter by service UUIDsallowDuplicates?(boolean): Allow duplicate scan resultsscanMode?('lowPower' | 'balanced' | 'lowLatency'): Scan mode
Stops scanning for BLE devices.
Connects to a BLE device.
Parameters:
deviceId(string): The unique identifier of the device
Returns: Promise
Disconnects from a BLE device.
Parameters:
deviceId(string): The unique identifier of the device
Discovers GATT services for a connected device.
Parameters:
deviceId(string): The unique identifier of the connected device
Returns: Promise<GATTService[]>
Reads a characteristic value from a connected device.
Parameters:
deviceId(string): The unique identifier of the connected deviceserviceUUID(string): The UUID of the servicecharacteristicUUID(string): The UUID of the characteristic
Returns: Promise
Writes a value to a characteristic on a connected device.
Parameters:
deviceId(string): The unique identifier of the connected deviceserviceUUID(string): The UUID of the servicecharacteristicUUID(string): The UUID of the characteristicvalue(string): The value to write (hex string)writeType?('write' | 'writeWithoutResponse'): Write type
Returns: Promise
Subscribes to notifications/indications from a characteristic.
Parameters:
deviceId(string): The unique identifier of the connected deviceserviceUUID(string): The UUID of the servicecharacteristicUUID(string): The UUID of the characteristic
Unsubscribes from notifications/indications from a characteristic.
Parameters:
deviceId(string): The unique identifier of the connected deviceserviceUUID(string): The UUID of the servicecharacteristicUUID(string): The UUID of the characteristic
Gets list of currently connected devices.
Returns: Promise<string[]>
Reads RSSI (signal strength) for a connected device.
Parameters:
deviceId(string): The unique identifier of the connected device
Returns: Promise
Platform-supported interface for BLE advertising data types:
interface AdvertisingDataTypes {
// 0x01 - Flags (partial support)
flags?: number
// 0x02-0x07 - Service UUIDs (fully supported)
incompleteServiceUUIDs16?: string[]
completeServiceUUIDs16?: string[]
incompleteServiceUUIDs32?: string[]
completeServiceUUIDs32?: string[]
incompleteServiceUUIDs128?: string[]
completeServiceUUIDs128?: string[]
// 0x08-0x09 - Local Name (fully supported)
shortenedLocalName?: string
completeLocalName?: string
// 0x0A - Tx Power Level (fully supported)
txPowerLevel?: number
// 0x14-0x15 - Service Solicitation (fully supported)
serviceSolicitationUUIDs16?: string[]
serviceSolicitationUUIDs128?: string[]
// 0x16, 0x20, 0x21 - Service Data (fully supported)
serviceData16?: Array<{
uuid: string
data: string
}>
serviceData32?: Array<{
uuid: string
data: string
}>
serviceData128?: Array<{
uuid: string
data: string
}>
// 0x19 - Appearance (partial support)
appearance?: number
// 0x1F - Service Solicitation (32-bit) (fully supported)
serviceSolicitationUUIDs32?: string[]
// 0xFF - Manufacturer Specific Data (fully supported)
manufacturerData?: string
}| Hex | Type Name | Description | Support Level | Example |
|---|---|---|---|---|
| 0x01 | Flags | Basic device capabilities | Partial | flags: 0x06 |
| 0x02-0x07 | Service UUIDs | Service UUIDs offered | Full | completeServiceUUIDs16: ['180D'] |
| 0x08-0x09 | Local Name | Device name | Full | completeLocalName: 'My Device' |
| 0x0A | Tx Power Level | Transmit power in dBm | Full | txPowerLevel: -12 |
| 0x14-0x15 | Service Solicitation | Services being sought | Full | serviceSolicitationUUIDs16: ['180D'] |
| 0x16, 0x20, 0x21 | Service Data | Data associated with services | Full | serviceData16: [{uuid: '180D', data: '010203'}] |
| 0x19 | Appearance | Appearance category | Partial | appearance: 0x03C0 |
| 0x1F | Service Solicitation (32-bit) | 32-bit services being solicited | Full | serviceSolicitationUUIDs32: ['0000180D'] |
| 0xFF | Manufacturer Specific Data | Vendor-defined data | Full | manufacturerData: '4748494A4B4C4D4E' |
Note: This library focuses on reliability and platform compatibility. Advanced BLE features like mesh networking, LE Audio, indoor positioning, etc., are not supported due to platform limitations.
import { startAdvertising, setServices } from 'munim-bluetooth'
// Health device advertising
startAdvertising({
serviceUUIDs: ['180D', '180F'], // Heart Rate, Battery Service
advertisingData: {
flags: 0x06, // LE General Discoverable Mode, BR/EDR Not Supported
completeLocalName: 'Health Monitor',
appearance: 0x03c0, // Generic Watch
txPowerLevel: -8,
manufacturerData: '0102030405', // Custom health data
serviceData16: [
{ uuid: '180D', data: '6400' }, // Heart rate: 100 bpm
{ uuid: '180F', data: '64' }, // Battery: 100%
],
},
})
// Set up GATT services
setServices([
{
uuid: '180D', // Heart Rate Service
characteristics: [
{
uuid: '2A37', // Heart Rate Measurement
properties: ['read', 'notify'],
value: '6400', // 100 bpm
},
],
},
{
uuid: '180F', // Battery Service
characteristics: [
{
uuid: '2A19', // Battery Level
properties: ['read', 'notify'],
value: '64', // 100%
},
],
},
])import { startAdvertising, updateAdvertisingData } from 'munim-bluetooth'
// Smart home device
startAdvertising({
serviceUUIDs: ['1812', '180F'], // HID, Battery Service
advertisingData: {
flags: 0x04, // LE General Discoverable Mode
completeLocalName: 'Smart Light Bulb',
appearance: 0x03c1, // Generic Light Fixture
manufacturerData: '0102030405', // Custom light data
serviceData16: [
{ uuid: '1812', data: '01' }, // HID: Keyboard
{ uuid: '180F', data: '64' }, // Battery: 100%
],
},
})
// Update advertising data when light state changes
updateAdvertisingData({
manufacturerData: '0102030406', // Updated light data
serviceData16: [
{ uuid: '1812', data: '02' }, // HID: Mouse
{ uuid: '180F', data: '50' }, // Battery: 80%
],
})import React, { useEffect } from 'react'
import {
startAdvertising,
stopAdvertising,
setServices,
addListener,
removeListeners,
} from 'munim-bluetooth'
const MyPeripheral = () => {
useEffect(() => {
// Configure services
setServices([
{
uuid: '1800', // Generic Access Service
characteristics: [
{
uuid: '2a00', // Device Name
properties: ['read'],
value: 'MyDevice',
},
{
uuid: '2a01', // Appearance
properties: ['read'],
value: '0x03C0', // Generic Computer
},
],
},
{
uuid: '1801', // Generic Attribute Service
characteristics: [
{
uuid: '2a05', // Service Changed
properties: ['indicate'],
},
],
},
])
// Start advertising
startAdvertising({
serviceUUIDs: ['1800', '1801'],
localName: 'MyReactNativePeripheral',
})
// Cleanup on unmount
return () => {
stopAdvertising()
removeListeners('connectionStateChanged')
}
}, [])
return <Text>Peripheral is running...</Text>
}import React, { useState, useEffect } from 'react'
import {
startScan,
stopScan,
connect,
discoverServices,
readCharacteristic,
subscribeToCharacteristic,
addListener,
} from 'munim-bluetooth'
const DeviceScanner = () => {
const [devices, setDevices] = useState([])
const [connectedDevice, setConnectedDevice] = useState(null)
useEffect(() => {
// Start scanning
startScan({
serviceUUIDs: ['180D'], // Filter by Heart Rate service
allowDuplicates: false,
scanMode: 'balanced',
})
// Listen for discovered devices
addListener('deviceFound')
// Handle deviceFound events to update devices state
// Cleanup
return () => {
stopScan()
if (connectedDevice) {
disconnect(connectedDevice)
}
}
}, [])
const handleConnect = async (deviceId) => {
await connect(deviceId)
setConnectedDevice(deviceId)
// Discover services
const services = await discoverServices(deviceId)
console.log('Services:', services)
// Read a characteristic
const value = await readCharacteristic(deviceId, '180D', '2A37')
console.log('Heart Rate:', value)
// Subscribe to notifications
subscribeToCharacteristic(deviceId, '180D', '2A37')
// Listen for value changes
addListener('characteristicValueChanged')
}
return (
<View>
<Text>Found {devices.length} devices</Text>
{/* Render device list */}
</View>
)
}- Permission Denied: Ensure you have the necessary Bluetooth permissions in your app
- Advertising Not Starting: Check that Bluetooth is enabled on the device
- Services Not Visible: Verify that your service UUIDs are properly formatted
- Scanning Not Working: On Android 6.0+, ensure location permissions are granted
- Connection Fails: Verify the device is in range and advertising
- Development Build Required: This library requires a development build in Expo. Use
npx expo run:iosornpx expo run:android - Permissions Not Working: Make sure you've added the permissions to your
app.jsonas shown in the setup section - Build Errors: Ensure you're using Expo SDK 50+ and have the latest Expo CLI
- Nitro Modules: Make sure you have
react-native-nitro-modulesinstalled and configured
Enable debug logging by setting the following environment variable:
export REACT_NATIVE_BLUETOOTH_DEBUG=1We welcome contributions! Please see our Contributing Guide for details on how to submit pull requests, report issues, and contribute to the project.
This project is licensed under the MIT License - see the LICENSE file for details.
