Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/adrielcafe/broker
💬 Publish-Subscribe (a.k.a Pub/Sub, EventBus) library for Android and JVM built with Coroutines
https://github.com/adrielcafe/broker
android android-library broker event-bus eventbus kotlin kotlin-android kotlin-coroutines kotlin-library message-broker message-bus pub-sub publish-subscribe publisher-subscriber pubsub
Last synced: 2 months ago
JSON representation
💬 Publish-Subscribe (a.k.a Pub/Sub, EventBus) library for Android and JVM built with Coroutines
- Host: GitHub
- URL: https://github.com/adrielcafe/broker
- Owner: adrielcafe
- License: mit
- Created: 2020-04-21T13:41:21.000Z (almost 5 years ago)
- Default Branch: master
- Last Pushed: 2021-08-14T01:53:19.000Z (over 3 years ago)
- Last Synced: 2024-11-07T13:38:16.265Z (3 months ago)
- Topics: android, android-library, broker, event-bus, eventbus, kotlin, kotlin-android, kotlin-coroutines, kotlin-library, message-broker, message-bus, pub-sub, publish-subscribe, publisher-subscriber, pubsub
- Language: Kotlin
- Homepage:
- Size: 229 KB
- Stars: 96
- Watchers: 4
- Forks: 9
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
- License: LICENSE.md
Awesome Lists containing this project
- awesome-list - adrielcafe/broker - 💬 Publish-Subscribe (a.k.a Pub/Sub, EventBus) library for Android and JVM built with Coroutines (Kotlin)
README
[![JitPack](https://img.shields.io/jitpack/v/github/adrielcafe/broker.svg?style=for-the-badge)](https://jitpack.io/#adrielcafe/broker)
[![Android API](https://img.shields.io/badge/api-16%2B-brightgreen.svg?style=for-the-badge)](https://android-arsenal.com/api?level=16)
[![Github Actions](https://img.shields.io/github/workflow/status/adrielcafe/broker/main/master?style=for-the-badge)](https://github.com/adrielcafe/broker/actions)
[![Codacy](https://img.shields.io/codacy/grade/ae430c15c7834ac088b23d27e4890dc0.svg?style=for-the-badge)](https://www.codacy.com/app/adriel_cafe/broker)
[![Codecov](https://img.shields.io/codecov/c/github/adrielcafe/broker/master.svg?style=for-the-badge)](https://codecov.io/gh/adrielcafe/broker)
[![kotlin](https://img.shields.io/github/languages/top/adrielcafe/broker.svg?style=for-the-badge)](https://kotlinlang.org/)
[![ktlint](https://img.shields.io/badge/code%20style-%E2%9D%A4-FF4081.svg?style=for-the-badge)](https://ktlint.github.io/)
[![License MIT](https://img.shields.io/github/license/adrielcafe/broker.svg?style=for-the-badge&color=yellow)](https://opensource.org/licenses/MIT)# Broker
Broker is a [Publish-Subscribe](https://en.wikipedia.org/wiki/Publish%E2%80%93subscribe_pattern) (a.k.a Pub/Sub, EventBus) library for Android and JVM built with [Coroutines](https://github.com/Kotlin/kotlinx.coroutines).
```kotlin
class MyActivity : AppCompatActivity(), GlobalBroker.Subscriber {override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
subscribe(this) { event ->
// Handle event
}
}
}class MyViewModel : ViewModel(), GlobalBroker.Publisher {
fun doSomething() {
publish(MyEvent(payload))
}
}
```**Features**
* Helps to decouple your code: publishers are loosely coupled to subscribers, and don't even need to know of their existence
* Works great with Activity, Fragment, Service, Custom View, ViewModel...
* Provides a [global instance](#global-pubsub) by default and lets you create [your own instances](#local-pubsub)
* Also provides useful extension functions to avoid boilerplate code
* [Android Lifecycle-aware](#android-lifecycle-aware): unsubscribe to events automatically
* [Retained event](#retained-events): cache the last published events
* Thread-safe: you can publish/subscribe from any thread
* Fast: all work is done outside the main thread and the events are delivered through a [Coroutines Flow](https://kotlin.github.io/kotlinx.coroutines/kotlinx-coroutines-core/kotlinx.coroutines.flow/-flow/)
* Small: [~30kb](https://jitpack.io/com/github/adrielcafe/broker/broker-core/1.1.0/)## Usage
Take a look at the [sample app](https://github.com/adrielcafe/broker/tree/master/sample/src/main/java/cafe/adriel/broker/sample) for working examples.### Creating events
Events (a.k.a Topic, Message) can be represented as `object` (without payload) and `data class` (with payload).
```kotlin
object EventA
data class EventB(val message: String)
```You can also group your events inside a `sealed class`, this way you can organize events by module, feature, scope, or similar.
```kotlin
sealed class MyEvent {
object EventA : MyEvent()
data class EventB(val message: String) : MyEvent()
}
```### Global Pub/Sub
Broker provides a global instance by default with some useful extension functions.Call `GlobalBroker.subscribe()` to subscribe to an event and `GlobalBroker.unsubscribe()` to unsubscribe to all events.
To subscribe, you should pass as parameters:
* The subscriber (usually the current class but can be a `String`, `Int`, `object`...)
* A `CoroutineScope` (tip: use the built-in [lifecycleScope and viewModelScope](https://developer.android.com/topic/libraries/architecture/coroutines))
* An *optional* `CoroutineContext` to run your lambda (default is `Dispatchers.Main`)
* A `suspend` lambda used to handle the incoming eventsCall `subscribe()` in `onStart()` (for Activity and Fragment) and `onAttachedToWindow()` (for Custom View), and call `unsubscribe()` in `onStop()` (for Activity and Fragment) and `onDetachedFromWindow()` (for Custom View).
```kotlin
class MyActivity : AppCompatActivity() {override fun onStart() {
super.onStart()
GlobalBroker.subscribe(this, lifecycleScope) { event ->
// Handle event
}
}override fun onStop() {
GlobalBroker.unsubscribe(this)
super.onStop()
}
}
```To publish events just call `GlobalBroker.publish()` passing the event as parameter. It can be called from any thread.
```kotlin
class MyViewModel : ViewModel() {fun doSomething() {
GlobalBroker.publish(MyEvent)
}
}
```#### GlobalBroker.Publisher & GlobalBroker.Subscriber
You can avoid some boilerplate code by implementing the `GlobalBroker.Publisher` and `GlobalBroker.Subscriber` interfaces. This also helps to identify the role of your class: is it a Publisher or Subscriber?
```kotlin
class MyActivity : AppCompatActivity(), GlobalBroker.Subscriber {override fun onStart() {
super.onStart()
subscribe(lifecycleScope) { event ->
// Handle event
}
}override fun onStop() {
unsubscribe()
super.onStop()
}
}class MyViewModel : ViewModel(), GlobalBroker.Publisher {
fun doSomething() {
publish(MyEvent)
}
}
```### Local Pub/Sub
In some situations a global instance is not a good option, because of that you can also create your own Broker instance.In the example below, we use [Koin](https://github.com/InsertKoinIO/koin) to inject a Broker instance in the `MyActivity` scope.
```kotlin
val myModule = module {scope {
scoped { Broker() }viewModel { MyViewModel(broker = get()) }
}
}
```And now we can inject a local Broker instance:
```kotlin
class MyActivity : AppCompatActivity() {private val broker by instance()
override fun onStart() {
super.onStart()
broker.subscribe(this, lifecycleScope) { event ->
// Handle event
}
}override fun onStop() {
broker.unsubscribe(this)
super.onStop()
}
}class MyViewModel(broker: Broker) : ViewModel() {
fun doSomething() {
broker.publish(MyEvent)
}
}
```#### BrokerPublisher & BrokerSubscriber
Broker class implements two interfaces: `BrokerPublisher` and `BrokerSubscriber`. You can use this to inject only the necessary behavior into your class.Let's back to the previous example. Instead of provide a Broker instance directly we can provide two injections, one for publishers and another for subscribers.
```kotlin
val myModule = module {scope {
val broker = Broker()scoped { broker }
scoped { broker }
viewModel { MyViewModel(broker = get()) }
}
}
```Now we can inject only what our class needs:
```kotlin
class MyActivity : AppCompatActivity() {private val broker by instance()
}class MyViewModel(broker: BrokerPublisher) : ViewModel()
```### Android Lifecycle-aware
Broker's subscribers can be [lifecycle-aware](https://developer.android.com/topic/libraries/architecture/lifecycle)! Works for global and local instances.Instead of subscribe in `onStart()` and unsubscribe in `onStop()` just subscribe in `onCreate()` and pass the `lifecycleOnwer` as parameter. Your events will now be automatically unsubscribed!
```kotlin
class MyActivity : AppCompatActivity(), GlobalBroker.Subscriber {override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
subscribe(owner = this) { event ->
// Handle event
}
}
}
```### Retained Events
It's possible to retain the last published event of a given type. Every time you subscribe to a retained event it will be emitted immediately.Just use the flags `emitRetained` when subscribing and `retain` when publishing:
```kotlin
subscribe(this, emitRetained = true) { event ->
// Handle event
}publish(MyEvent, retain = true)
```At any moment you can query for a retained event (will return null if there are none):
```kotlin
val lastEvent = getRetained()// removeRetained() will also return the last retained event
val lastEvent = removeRetained()
```### Error handling
If the subscriber's lambda throws an error, Broker will catch it and publish the event `BrokerExceptionEvent`. Just subscribe to it if you want to handle the exceptions.
```kotlin
subscribe(lifecycleScope) { event ->
// Handle error
}
```## Import to your project
1. Add the JitPack repository in your root build.gradle at the end of repositories:
```gradle
allprojects {
repositories {
maven { url 'https://jitpack.io' }
}
}
```2. Next, add the desired dependencies to your module:
```gradle
dependencies {
// Core
implementation "com.github.adrielcafe.broker:broker-core:$currentVersion"// Android Lifecycle support
implementation "com.github.adrielcafe.broker:broker-lifecycle:$currentVersion"
}
```
Current version: [![JitPack](https://img.shields.io/jitpack/v/github/adrielcafe/broker.svg?style=flat-square)](https://jitpack.io/#adrielcafe/broker)### Platform compatibility
| | `broker-core` | `broker-lifecycle` |
|---------|---------------|--------------------|
| Android | ✓ | ✓ |
| JVM | ✓ | |