This project aims to make Bisq Network accesible in Mobile Platforms following the philosofy of Bisq2 - to make it easier for both, experienced and newcomers, to trade Bitcoin in a decentralized way as well as defending Bisq motto: exchange, decentralized, private & secure.
To achieve this goal, we are building a total of 3 mobile apps that can be divided in 2 categories:
- An Android app (called
androidNode
) that runs bisq core in its veins aiming to bring a fully featured trading version ofBisq2
(also referred to as its main protocol -Bisq Easy
) to Mobile for full privacy & security.
- A thin Bisq App
Client
(coming inandroid
&iOS
flavours) able to be configured to connect to a trusted Bisq node (over clearnet) to cater for people willing to try Bisq from somebody they really trust (popularily described as "Uncle Jim") who is willing to share his Bisq node with them.
We follow Bisq standard guidelines for contributions, fork + PR, etc. Please refer to Contributor Checklist
We are currently working in the project definition and Github issues will soon be available for contributors to pick what would they like to help with. Stay tuned for updates.
For now follow along to learn how to run this project. If you are a mobile enthusiast and feel driven by Bisq goals, please reach out!
- Java: 17.0.12.fx-zulu JDK (sdkman env file is avail in project root)
- Ruby: v3+ (for iOS Cocoapods 1.15+)
- IDE: We use and recommend Fleet, but you may as well use the IDE of your preference. For iOS testing you will need XCode.
note: at the time of writing Fleet is in preview, it can get unstable so it's recommended to switch to Android Studio / Xcode as needed. For example, the first time you try to run the iosClient most probably you will need to do it from Xcode.
- Download this repo code
- Open Fleet IDE and open the folder you've just downloaded.
- Wait for the smart mode to run the
Pre-flight
. This will let you know what's missing in your machine to run the project. Follow its instructions to install everything. The project has an sdkman file, if you have sdkman installed the right java version will be picked up for you. - Once the preflight is successful, you should see all the items checked
- If you are on a MacOS computer building the iOS app you can go ahead and open the subfolder iosClient with your Xcode, build the project and run it in your device or emulator. After that you can just do it from Fleet
- For Android it can run on any machine, just run the preconfigured configurations
androidClient
and/orandroidNode
Alternatively, you could run ./gradlew clean build
(1) first from terminal and then open with your IDE of preference.
Addicionally, for the androidNode
module to build you need to have its dependent Bisq2 jars in your local maven2 repository ('~/.m2/repository`). Here are the steps to do that
- download Bisq2 if you don't have it already
- follow Bisq2 root
README.md
steps to build the project - run
./gradlew publishAll
// this will install all the jars you need in your m2 repo
Done! Alternatively if you are interested only in contributing for the xClients
you can just build them individually instead of building the whole project.
Node
You just need to run a local bisq seed node from the bisq2 project. By default port 8000 is used
Clients
You need to run the seed node as explained above + the http-api module with the following VM parameters
-Dapplication.appName=bisq2_restApi_clear
-Dapplication.network.supportedTransportTypes.2=CLEAR
-Dapplication.devMode=true
Default networking setup for the WebSocket (WS) connection can be found in gradle.properties
file. You can change there for locally building pointing at the ip you are interested in.
Designs
androidNode + xClient screens are designed in Figma. Yet to differentiate between which screens goes into which.
Figma link: https://www.figma.com/design/IPnuicxGKIZXq28gybxOgp/Xchange?node-id=7-759&t=LV9Gx9XgJRvXu5YQ-1
Though the figma design captures most of the functionality, it's an evolving document. It will be updated with new screens, flow updates, based on discussions happening in GH issues / matrix.
Navigation Implementation
Please refer to this README
- Some Apple M chips have trouble with cocoapods, follow this guide to fix it
- On MacOS: non-homebrew versions of Ruby will cause problems
- On MacOS: If Fleet Pre-flight gives error "Gradle not found" and running the (1) terminal command doesn't even run, you need to install gradle with
homebrew
and then rungradle wrapper
on the root. Then reopen Fleet and try the Pre-flight again.
Though this can evolve, this is the initial structure of this KMP project:
- shared:domain: Domain module has models (KOJOs) and components that provide them.
- shared:presentation: Contains UI shared code using Kotlin MultiPlatform Compose Implementation forr all the apps, its Presenter's behaviour contracts (interfaces) and default presenter implementations that connects UI with domain.
- iosClient: Xcode project that generates the thin iOS client from sharedUI
- androidClient: Kotlin Compose Android thin app. This app as well should have most if not all of the code shared with the iosClient.
- androidNode: Bisq2 Implementation in Android, will contain the dependencies to Java 17 Bisq2 core jars.
This project uses the MVP (Model-View-Presenter) Design Pattern with small variations (introducing Repositories & we allow reusal of presenters under specific conditions) in the following way:
- Dumb Views: Each View will define it's desired presenter behaviour. For example, for the
AppView
it would define theAppPresenter
interface. This includes which data its interested in observing and the commands it needs to trigger from user interactions. - UI indepdently builtThe view will react to changes in the presenter observed data, and call the methods it needs to inform the presenter about user actions. In this way each view can be created idependently without strictly needing anything else
- Encourage Rich Domain well-test models Same goes for the Models, they can be built (and unit tested) without needing anything else, simple POKOs (Plain Old Kotlin Objects - meaning no external deps). Ideally business logic should go here and the result of executing a business model logic should be put back into the repository for all observers to know.
- Presenters guide the orchestra When you want to bring interaction to life, create a presenter (or reuse one if the view is small enough) and implement the interface you defined when doing the view (
AppPresenter
interface for example). That presenter will generally modify/observe the models through a repository and/or a service. The most important thing is that mutable/unmutable observability should happen here connecting those fields that the view needs with the real data as appropiate case-by-case. - Repositories key for reactive UI Now for the presenter to connect to the domain models we use repositories which is basically a storage of data (that abstracts where that data is stored in). The repositories also expose the data in an observable way, so the presenter can satisfy the requested data from the view from the data of the domain model in the ways it see fit. Sometimes it would just be a pathrough. The resposities could also have caching strategy, and persistance. For most of the use cases so far we don't see a strong need for persistance in most of them (with the exception of settings-related repositories) - more on this soon
- Services allow us to have different networking sources we are developing 3 apps divided in 2 groups:
node
andclient
. Each group has a very distinct networking setup. We need each type of app build to have only the networking it needs. The proposed separation of concerns not only allows a clean architecture but also allows faster development focus on each complexity separately. We found that for theandroidNode
it makes sense to handle all the domain stuff directly using domain models in the services without connecting to a repository since the bisq-core jars manage all the persistance. You have the option to decide how to connect this in your presenter.
As per original specs single-activity
pattern (or single-viewcontroller
in iOS) is sufficient for this project. Which means, unless we find a specific use case for it, we'll stick to a single Activity/ViewController for the whole lifecycle of the app.
The app's architecture BasePresenter
allows a tree like behaviour where a presenter can be a root with dependent child presenters.
We leverage this by having:
- A
MainPresenter
that acts as root in each and all of the apps - The rest of the presenters require the main presenter as construction parameter to be notified about lifecycle events.
It's ok to reuse an existing presenter for your view if:
- Your view is a very small part of a bigger view that renders together (commonly called
Screen
) and you can't foresee reusal for it - Your view is a very small part of a bigger view and even if its reused the presenter required implementation is minimal
To reuse an existing presenter you would have to make it extend your view defined presenter interface and do the right Koin bind
on its Koin repository definition.
Then you can inject it in the @Composable
function using koinInject()
.
- Native Performance
- Allows us to focus on the "easiest" platform first for the Node (Because of Apple restrictions on Tor and networking in general). Althought unexpected, if situation changes in the future we could cater for an iOS Node.
- Flexibility without the security/privacy concerns of its competitors
- (Node)JVM language allows us to port much of the optimised Bisq code already existing in the Desktop apps
- Kotlin Compose UI allows us to share UI code easily across the 3 apps.
If you are interested in seeing the POCs related to the R&D before this project kicked-off please refer to this branch