| Updated 25.07.2025 | Languages: EN, FR, CZ, PL |

Contributing guide

Compiling with SQLCipher encryption enabled

Add cabal.project.local to project root with the location of OpenSSL headers and libraries and flag setting encryption mode:

cp scripts/cabal.project.local.mac cabal.project.local
# or
# cp scripts/cabal.project.local.linux cabal.project.local

OpenSSL on MacOS

MacOS comes with LibreSSL as default, OpenSSL must be installed to compile SimpleX from source.

OpenSSL can be installed with brew install openssl@3.0

You will have to add /opt/homebrew/opt/openssl@3.0/bin to your PATH in order to have things working properly

Project branches

In simplex-chat repo

  • stable - stable release of the apps, can be used for updates to the previous stable release (GHC 9.6.3).

  • stable-android - used to build stable Android core library with Nix (GHC 8.10.7) - only for Android armv7a.

  • master - branch for beta version releases (compatible with both GHC 9.6.3 and 8.10.7).

  • master-android - used to build beta Android core library with Nix (GHC 8.10.7) - only for Android armv7a.

In simplexmq repo

  • master - compatible with both GHC 9.6.3 and 8.10.7.

Development & release process

  1. Make PRs to master branch only for both simplex-chat and simplexmq repos.

  2. To build core libraries for Android, iOS and windows:

  • merge master branch to master-android branch.
  • push to GitHub.
  1. All libraries should be built from master branch, Android armv7a - from master-android branch.

  2. To build Desktop and CLI apps, make tag in master branch, APK files should be attached to the release.

  3. After the public release to App Store and Play Store, merge:

  • master to stable
  • master to master-android (and compile/update code)
  • master-android to stable-android
  1. Independently, master branch of simplexmq repo should be merged to stable branch on stable releases.

Branches and PRs

Use change scope (or comma separated scopes) as the first word in the PR names, followed by the colon. Commit name itself should be lowercase, in present tense.

The PR names in simplex-chat repo are used in release notes, they should describe the solved problem and not the change. Possible PR scopes:

  • ios
  • android
  • desktop
  • core
  • docs
  • website
  • ci

We squash PRs, do not rewrite branch history after the review.

For some complex features we create feature branches that will be merged once ready - do not make commits directly to them, make PRs to feature branches.

Differences between GHC 8.10.7 and GHC 9.6.3

  1. The main difference is related to DuplicateRecordFields extension.

It is no longer possible in GHC 9.6.3 to specify type when using selectors, instead OverloadedRecordDot extension and syntax are used that need to be removed in GHC 8.10.7:

{-# LANGUAGE DuplicateRecordFields #-}
-- use this in GHC 9.6.3 when needed
{-# LANGUAGE OverloadedRecordDot #-}

-- GHC 9.6.3 syntax
let x = record.field

-- GHC 8.10.7 syntax removed in GHC 9.6.3
let x = field (record :: Record)

It is still possible to specify type when using record update syntax, use this pragma to suppress compiler warning:

-- use this in GHC 9.6.3 when needed
{-# OPTIONS_GHC -fno-warn-ambiguous-fields #-}

let r' = (record :: Record) {field = value}
  1. Most monad functions now have to be imported from Control.Monad, and not from specific monad modules (e.g. Control.Monad.Except).
-- use this in GHC 9.6.3 when needed
import Control.Monad

This PR has all the differences.

Improving compatibility between versions for remote desktop connection

UI already can handle failed JSON conversions of chats and chat items, and it helps both debugging and downgrading.

While we can increase versions for remote connections to make different versions incompatible, it degrades remote connection UX, as in many cases users can't upgrade mobile or desktop apps at the same time because of different release cycles.

It is especially problematic for Android app users, as they can only downgrade via Export/Import - older version can't be installed on top of newer version.

PR #6105 improved it by:

  • adding CInfoInvalidJSON constructor, so that chats that cannot be parsed will show as "invalid chat" via remote connection (as when UI has field not present in API),
  • changing JSON parsing for CIContent, so that it falls back to CIInvalidJSON in platform-specific JSON parser.

To avoid "invalid" chats in the list we need to maintain forward compatibility on JSON encoding level of AChat type and subtypes:

  • add new fields as optional to these types,
  • add omittedField method to FromJSON instances of types of new fields to provide a default value, where appropriate,
  • define primitive non-optional fields as newtype with omittedField in JSON instance.

To avoid fallback to invalid JSON in chat items we should do the same for ChatItem type and subtypes. It's especially important when adding fields to types used for all CIContent, as otherwise all items will be broken.

© 2020-2025 SimpleX | Open-Source Project