Skip to content

bridgefy/sdk-android-beta

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

16 Commits
 
 
 
 
 
 
 
 

Repository files navigation

The Bridgefy Software Development Kit (SDK) is a state-of-the-art, plug-and-play package that will let people use your mobile app when they don’t have access to the Internet, by using Bluetooth mesh networks.

Integrate the Bridgefy SDK into your Android and iOS app to reach the 3.5 billion people that don’t always have access to an Internet connection, and watch engagement and revenue grow!

Website. https://developer.bridgefy.me
Email. contact@bridgefy.me
witter. https://twitter.com/bridgefy
Facebook. https://www.facebook.com/bridgefy

Overview

Bridgefy creates mesh networks in which devices connect directly to each other in a decentralized manner. This allows users to communicate with nearby devices within a certain range, forming a network without the need for a centralized server or Internet access.

networking

The Bridgefy SDK provides a set of tools and APIs that developers can use to incorporate offline messaging, data transfer, and real-time communication features into their applications. It allows users to send data directly to other nearby devices using Bluetooth Low Energy.

Table of contents

  1. Setup
  2. Android Permissions
  3. Summary of available permissions
  4. Usage
  5. Secure connections
  6. Using ProGuard
  7. Supported Devices
  8. Contact & Support

Setup

Bridgefy SDK is available in our public repository. To install it you must follow the instructions:

val bridgefy_release_maven_url = "http://34.82.5.94:8081/artifactory/libs-release-local"

allprojects {
    repositories {
        maven {
            url = java.net.URI(bridgefy_release_maven_url)
            isAllowInsecureProtocol = true
        }
    }
}

In the app module build.gradle :

/**
 * Declare dependencies
 * @see http://www.gradle.org/docs/current/userguide/userguide_single.html#sec:how_to_declare_your_dependencies
 */
dependencies {
    implementation (group = "me.bridgefy", name = "android-sdk", version = "1.0.0", ext = "aar") {
        isTransitive = true
    }
}

Compatibility

Supported on Android 6 (minSdk = 23) and above with compileSdk >= 31

Android Permissions

Android requires additional permissions declared in the manifest for an app to run a BLE scan since API 23 (6.0 / Marshmallow) and perform a Bluetooth Low Energy connection since API 31 (Android 12). These permissions currently assume scanning is only used when the App is in the foreground, and that the App wants to derive the user's location from Bluetooth Low Energy signal (on API >= 23). Below are a number of additions you can make to your AndroidManifext.xml for your specific use case.

Location permission for Bluetooth Low Energy Scanning

Bridgefy uses the uses-permission-sdk-23 tag to require location only on APIs >= 23, you can request the required permissions by adding the following to your AndroidManifest.xml:

<uses-permission-sdk-23 android:name="android.permission.ACCESS_COARSE_LOCATION"
    android:maxSdkVersion="30"
    tools:node="replace" />
    
<uses-permission-sdk-23 android:name="android.permission.ACCESS_FINE_LOCATION"
    android:maxSdkVersion="32"
    tools:node="replace" />

Scan in the background and support APIs 29 & 32

You should add the following to your AndroidManifest.xml:

<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION"
    android:maxSdkVersion="32" />

If you want to access the user's location in the background on APIs > 30, remove the android:maxSdkVersion attribute.

Location from BLE scanning in API >= 31

API 31 (Android 12) introduced new Bluetooth permissions. Bridgefy uses the android:usesPermissionFlags="neverForLocation" attribute on the BLUETOOTH_SCAN permission, which indicates scanning will not be used to derive the user's location, so location permissions are not required. If you need to locate the user with BLE scanning, use this instead, but keep in mind that you will still need ACCESS_FINE_LOCATION:

<uses-permission android:name="android.permission.BLUETOOTH_SCAN" tools:node="replace" />

Connect peripherals

You add the BLUETOOTH_CONNECT permission that Bridgefy requests in APIs >= 31:

<uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />

Summary of available permissions

Required permissions

A summary of available runtime permissions used for BLE:

from API to API (inclusive) Acceptable runtime permissions
18 22 (No runtime permissions needed)
23 28 One of below:
- android.permission.ACCESS_COARSE_LOCATION
- android.permission.ACCESS_FINE_LOCATION
29 30 - android.permission.ACCESS_FINE_LOCATION
- android.permission.ACCESS_BACKGROUND_LOCATION*
31 current - android.permission.ACCESS_FINE_LOCATION**
- android.permission.BLUETOOTH_SCAN
- android.permission.BLUETOOTH_ADVERTISE
- android.permission.BLUETOOTH_CONNECT

Usage

Start the SDK

The following code shows how to start the SDK (using your API key) and how to assign the delegate.

Android Manifest

Locate the AndroidManifest.xml file within your project. It is usually located in the "app" or "src/main" directory. Add the element as application element inside the component tag.

<manifest>

    <application>
        
        <meta-data
            android:name="com.bridgefy.sdk.API_KEY"
            android:value="..." />
        
    </application>
    
</manifest>

Bridgefy initialization

By default the SDK starts using the Standard propagation profile mode

    /**
 * Init bridgefy SDK
 * - customPeerID: The client can set a custom id, if null Bridgefy create one
 * - bridgefyApiKey: Bridgefy API KEY
 * - propagationProfile: Default PropagationProfile.Standard
 * - delegate: The delegate object listens for all bridgefy actions.
 */
fun init(
    customPeerID: UUID?,
    bridgefyApiKey: UUID,
    propagationProfile: PropagationProfile = PropagationProfile.Standard,
    delegate: BridgefyDelegate?
)

The string “bridgefyApiKey” represents a valid API key. An Internet connection is needed at least for the first time in order to validate the license.

To stop it, use the following code:

/**
 * Stop bridgefy sdk operations
 *
 */
bridgefy.stop()

To start bridgefy operations:

/**
 * Start bridgefy sdk operations after init SDK
 */
bridgefy.start()

Nearby peer detection

The following method is invoked when a peer has established connection:

    val delegate: BridgefyDelegate = object : BridgefyDelegate {

    override fun onConnected(peerID: UUID) {
        ...
    }

}

peerID: Identifier of the user that has established a connection.

When a peer is disconnected(out of range), the following method will be invoked:

    val delegate: BridgefyDelegate = object : BridgefyDelegate {

    override fun onDisconnected(peerID: UUID) {
        ...
    }

}

peerID: Identifier of the disconnected user.

When a device is detected, notifies the list of connected users:

    val delegate: BridgefyDelegate = object : BridgefyDelegate {

    /**
     * On connected peers
     *
     * @param connectedPeers
     */
    fun onConnectedPeers(connectedPeers: List<UUID>) {
        ...
    }

}

connectedPeers: List of identifiers of the connected user.

Send data

The following method is used to send data using a transmission mode. This method returns the message ID (messageID) to the client.

import me.bridgefy.commons.TransmissionMode
// Bridgefy instance
val bridgefy: Bridgefy

// After start sdk, send ByteArray
val messageId = bridgefy.send(
    "Sample text!".toByteArray(Charsets.UTF_8),
    TransmissionMode.P2P(peerID)
)

messageID: Unique identifier related to the message.

Transmission Modes:

open class TransmissionMode {
    /**
     * Broadcast type propagate message on mesh network
     */
    data class Broadcast(val sender: UUID) : TransmissionMode()

    /**
     * Mesh type propagate message and find receiver on mesh network
     *
     */
    data class Mesh(val receiver: UUID) : TransmissionMode()

    /**
     * Direct type allow direct message and if receiver isn't connected,
     * the SDK change and propagate message with Mesh type
     *
     */
    data class P2P(val receiver: UUID) : TransmissionMode()
}

There are several modes for sending packets:

P2P(val receiver: String): Sends the packet only when the receiver is in range.
Mesh(val receiver: String): Sends the packet using mesh to only once receiver. It doesn't need the receiver to be in range. Receiver can be in range of a third receiver located within range of both sender and receiver at the same time, or receiver can be out of range of all other nodes, but eventually come within range of a node that at some point received the packet. Mesh messages can be received my multiple nodes, but can only be read by the intended receiver.
Broadcast: Sends a packet using mesh without a defined receiver. The packet is broadcast to all nearby users that are in range, who then broadcast it to all receivers that are in their range, and so on. If a user isn't in range, the packet will be delivered the next time said user comes within range of another user who did receive the packet. Broadcast messages can be read by all nodes that receive it.

If there is no error when sending the message, will be received with the message id (messageID)

    val delegate: BridgefyDelegate = object : BridgefyDelegate {
    /**
     * On send
     *
     * @param messageID
     */
    override fun onSend(messageID: UUID) {
        ...
    }

}

otherwise, the following method will be received

    val delegate: BridgefyDelegate = object : BridgefyDelegate {
    /**
     * On fail to send
     *
     * @param messageID
     */
    override fun onFailToSend(messageID: UUID) {
        ...
    }
}

Direct and Mesh transmission

Direct transmission is a mechanism used to deliver packets to a user that is nearby or visible (a connection has been detected). Mesh transmission is a mechanism used to deliver offline packets even when the receiving user isn’t nearby or visible. It can be achieved taking advantage of other nearby peers; these receive the package, hold it, and forward to other peers trying to find the receiver.

Receive Data

When a packet has been received, the following method will be invoked:

val delegate: BridgefyDelegate = object : BridgefyDelegate {
    /**
     * On receive
     *
     * @param data
     * @param messageID
     * @param transmissionMode
     */
    override fun onReceive(
        data: ByteArray,
        messageID: UUID,
        transmissionMode: TransmissionMode,
    ) {
        ...
    }
}

data: Received Data object
messageID: Unique identifier related to the message.
transmissionMode: The transmission mode used when sending the message

Propagation Profiles

/**
 * Propagation profile
 */
enum class PropagationProfile {
    Standard,
    HighDensityEnvironment,
    SparseEnvironment,
    LongReach,
    ShortReach
}
Profile Hops limit TTL(s) Sharing Time Maximum Propagation Track list limit
Standard 100 86400 (1 d) 15000 200 50
High Density Environment 50 3600 (1 h) 10000 50 50
Sparse Environment 100 302400 (3.5 d) 10000 250 50
Long Reach 250 604800 (7 d) 15000 1000 50
Short Reach 50 1800 10000 50 50
  • Hops limit: The maximum number of hops a message can get. Each time a message is forwarded, is considered a hop.
  • TTL: Time to live, is the maximum amount of time a message can be propagated since its creation.
  • Sharing time: The maximum amount of time a message will be kept for forwarding.
  • Maximum propagation: The maximum number of times a message will be forwarded from a device.
  • Track list limit: The maximum number of UUID’s stored in an array to prevent sending the message to a peer which already forwarded the message.

Secure connections

Part of Bridgefy's functionality is its ability to provide a secure connection for sending data within a mesh network. To ensure the privacy and security of sensitive data, Bridgefy SDK employs encryption techniques. Encryption involves transforming data into an unreadable format, which can only be deciphered by authorized recipients who possess the correct decryption key.

Bridgefy utilizes the Signal Protocol, a widely recognized and trusted encryption protocol, to encrypt sensitive data exchanged between devices in a mesh network. The Signal Protocol provides end-to-end encryption, meaning the data remains encrypted throughout its entire journey from the sender to the intended recipient. This ensures that even if someone intercepts the data, they won't be able to access its contents without the proper decryption key.

However, companies and developers who use the Bridgefy SDK in their mobile apps also have the option to implement their own custom encryption if they prefer, which doesn't require the establishment of a secure connection but needs robust encryption-key management practices

Established secure connection

Bridgefy SDK offers the option to establish secure connections within the mesh network, encrypting the data traveling on the mesh using the Signal protocol. This ensures a secure connection and protects the data from unauthorized access.

When node is connected you can try to establishing a secure connection with follow method:

// Bridgefy instance
val bridgefy: Bridgefy

bridgefy.establishSecureConnection(connectedNodeID)

When the establishment is done, the secure connection is notified on delegate:

 val delegate: BridgefyDelegate = object : BridgefyDelegate {

    /**
     * Connected node secured
     */
    fun onConnectedSecurely(peerID: UUID) {
        ...
    }

Bridgefy also allows users to implement their own custom encryption if they prefer not to use the Signal Protocol. In such cases, where custom encryption is utilized, it's important to note that Bridgefy does not require the establishment of a secure connection between devices.

Recommendations for using a secure connection

While it's common to think that a secure connection needs to be established simultaneously with other connections, it's not always the case. In fact, many secure connection protocols allow for the establishment of connections at different times, depending on the specific requirements and circumstances.

By recommending a secure connection that is not simultaneous, it means that the focus should be on establishing a secure connection whenever it is feasible and appropriate, without being bound to the constraints of synchronizing the connection with other devices.

This approach allows for flexibility and adaptability in securing the connection. For example, a device may establish a secure connection with one device at a time, ensuring that each connection is properly encrypted and authenticated. This way, the security of the connection is not compromised, even if it takes place at different times.

Additionally, requesting a secure connection that is not simultaneous can also be beneficial in terms of resource management. Establishing secure connections simultaneously with multiple devices may impose a heavier burden on the device's resources, such as processing power and network bandwidth. By prioritizing security over simultaneous connections, the device can allocate resources effectively and maintain a high level of security for each connection.

Using ProGuard

If you are using ProGuard in your project, include the following lines to your configuration file:

 -keep class me.bridgefy.** { *; }
 -dontwarn me.bridgefy.**
 -keepclassmembernames class me.bridgefy.crypto.** { *; }
 -keep, allowshrinking class org.signal.libsignal.** { *; }
 -dontwarn me.bridgefy.crypto.**

Supported Devices

Bridgefy's support for devices extends to smartphones and tablets, as long as they run iOS or Android, and have BLE. This makes Bridgefy a versatile platform that caters to various communication needs. By enabling connectivity across a diverse range of devices, Bridgefy ensures that users can easily establish connections and exchange information regardless of whether they have an internet connection or not.

Contact & Support

© 2023 Bridgefy Inc. All rights reserved

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published