diff --git a/.env.example b/.env.example
new file mode 100644
index 0000000..a4eee53
--- /dev/null
+++ b/.env.example
@@ -0,0 +1,17 @@
+# Matrix Client
+MATRIX_HOST=http://localhost:8008
+MATRIX_ADMIN_TOKEN=secret
+
+# PostgreSQL
+POSTGRES_USER=synapse_user
+POSTGRES_PASSWORD=secretpassword
+POSTGRES_DB=synapse
+# https://www.postgresql.org/docs/current/app-initdb.html
+POSTGRES_INITDB_ARGS='--no-locale --encoding=UTF8'
+
+# Synapse
+SYNAPSE_SERVER_NAME=matrix.localhost
+SYNAPSE_REPORT_STATS=yes
+SYNAPSE_ENABLE_REGISTRATION=yes
+SYNAPSE_NO_TLS=true
+SYNAPSE_USER_DIR_SEARCH_ALL_USERS=true
diff --git a/.gitignore b/.gitignore
index 6985cf1..d673cb1 100644
--- a/.gitignore
+++ b/.gitignore
@@ -12,3 +12,12 @@ Cargo.lock
# MSVC Windows builds of rustc generate these, which store debugging information
*.pdb
+
+# Development
+/docker/*
+!/docker/.gitkeep
+!/docker/postgre
+.env
+
+# System Specific
+.DS_Store
diff --git a/Justfile b/Justfile
new file mode 100644
index 0000000..ab74a98
--- /dev/null
+++ b/Justfile
@@ -0,0 +1,29 @@
+set positional-arguments
+
+# Lists all available commands
+default:
+ just --list
+
+# Creates the `.env` file if it doesn't exist
+dotenv:
+ cp -n .env.example .env || true
+
+# Generates the synapse configuration file and saves it
+gen_synapse_conf: dotenv
+ docker run -it --rm \
+ -v ./docker/synapse:/data \
+ --env-file .env \
+ matrixdotorg/synapse:v1.96.1 generate
+
+# Runs backend dependency services
+backend: dotenv
+ docker compose up --build
+
+# Stops backend dependency services
+stop:
+ docker compose down
+
+# Removes oll Docker related config, volumes and containers for this project
+clear: stop
+ docker compose rm --all --force --volumes --stop
+ docker volume rm commune_synapse_database || true
diff --git a/README.md b/README.md
index bf84f02..e31545b 100644
--- a/README.md
+++ b/README.md
@@ -1,2 +1,51 @@
-# commune
-Commune Server
+
+
commune
+
+ Commune Server
+
+
+
+## Development
+
+### Requirements
+
+- [Docker](https://www.docker.com/get-started/)
+- [Justfile](https://github.com/casey/just)
+- [Rust](https://rustup.rs)
+
+### Getting Started
+
+1. Generate `Synapse` server configuration
+
+```bash
+just gen_synapse_conf
+```
+
+2. Run Synapse Server (and other containerized services) using Docker Compose
+via:
+
+```bash
+just backend
+```
+
+**When you are ready**
+
+Teardown services using `just stop`. If you want to perform a complete cleanup
+use `just clear`.
+
+> **Warning** `just clear` will remove all containers and images.
+
+### Application Layout
+
+
+
+
Application Layout Overview
+
+
+The client, any HTTP Client, comunicates with the Commune Server which may or
+may not communicate with Matrix's server _Synapse_ which runs along with its
+database in a Docker container.
+
+## License
+
+This project is licensed under the Apache License Version 2.0
diff --git a/docker-compose.yml b/docker-compose.yml
new file mode 100644
index 0000000..fce631b
--- /dev/null
+++ b/docker-compose.yml
@@ -0,0 +1,28 @@
+version: '3'
+
+services:
+ synapse_database:
+ image: 'postgres:16'
+ ports:
+ - '5432:5432'
+ volumes:
+ - synapse_database:/var/lib/postgresql/data
+ env_file:
+ - .env
+ restart: always
+
+ synapse:
+ image: 'matrixdotorg/synapse:v1.96.1'
+ ports:
+ - '8008:8008'
+ - '8448:8448'
+ volumes:
+ - ./docker/synapse:/data
+ env_file:
+ - .env
+ restart: always
+ depends_on:
+ - synapse_database
+
+volumes:
+ synapse_database:
diff --git a/docker/.gitkeep b/docker/.gitkeep
new file mode 100644
index 0000000..e69de29
diff --git a/docs/diagrams/diagram.excalidraw b/docs/diagrams/diagram.excalidraw
new file mode 100644
index 0000000..eaa5793
--- /dev/null
+++ b/docs/diagrams/diagram.excalidraw
@@ -0,0 +1,475 @@
+{
+ "type": "excalidraw",
+ "version": 2,
+ "source": "https://excalidraw.com",
+ "elements": [
+ {
+ "id": "RYL6z1yNI4ryhDrjxemyL",
+ "type": "rectangle",
+ "x": 364,
+ "y": 272,
+ "width": 130,
+ "height": 119,
+ "angle": 0,
+ "strokeColor": "#1971c2",
+ "backgroundColor": "#a5d8ff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "seed": 1352799417,
+ "version": 94,
+ "versionNonce": 297889623,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "NwNRXHacaPiTQKk-sS3iv"
+ },
+ {
+ "id": "jR76iIplo4TVvNKZBH6vT",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "NwNRXHacaPiTQKk-sS3iv",
+ "type": "text",
+ "x": 393.84375,
+ "y": 319.5,
+ "width": 70.3125,
+ "height": 24,
+ "angle": 0,
+ "strokeColor": "#1971c2",
+ "backgroundColor": "#ffec99",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "seed": 683172535,
+ "version": 80,
+ "versionNonce": 581421529,
+ "isDeleted": false,
+ "boundElements": null,
+ "updated": 1700421905506,
+ "link": null,
+ "locked": false,
+ "text": "Client",
+ "fontSize": 20,
+ "fontFamily": 3,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "baseline": 19,
+ "containerId": "RYL6z1yNI4ryhDrjxemyL",
+ "originalText": "Client",
+ "lineHeight": 1.2
+ },
+ {
+ "type": "rectangle",
+ "version": 207,
+ "versionNonce": 145190007,
+ "isDeleted": false,
+ "id": "igO3c0IESW8ZSDcllatP-",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "angle": 0,
+ "x": 582,
+ "y": 269.5,
+ "strokeColor": "#2f9e44",
+ "backgroundColor": "#b2f2bb",
+ "width": 130,
+ "height": 119,
+ "seed": 2026968473,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "E1cQEf95N_pZWvm755-z4"
+ },
+ {
+ "id": "jR76iIplo4TVvNKZBH6vT",
+ "type": "arrow"
+ },
+ {
+ "id": "I--VThaGzeWEpMH77IyuH",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false
+ },
+ {
+ "type": "text",
+ "version": 206,
+ "versionNonce": 648226489,
+ "isDeleted": false,
+ "id": "E1cQEf95N_pZWvm755-z4",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "angle": 0,
+ "x": 605.984375,
+ "y": 305,
+ "strokeColor": "#2f9e44",
+ "backgroundColor": "#ffec99",
+ "width": 82.03125,
+ "height": 48,
+ "seed": 815919225,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "boundElements": [],
+ "updated": 1700421905506,
+ "link": null,
+ "locked": false,
+ "fontSize": 20,
+ "fontFamily": 3,
+ "text": "Commune\nServer",
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "igO3c0IESW8ZSDcllatP-",
+ "originalText": "Commune\nServer",
+ "lineHeight": 1.2,
+ "baseline": 43
+ },
+ {
+ "id": "jR76iIplo4TVvNKZBH6vT",
+ "type": "arrow",
+ "x": 495,
+ "y": 328.38774886792805,
+ "width": 86,
+ "height": 0.220972375229735,
+ "angle": 0,
+ "strokeColor": "#fa5252",
+ "backgroundColor": "#a5d8ff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "seed": 1227684473,
+ "version": 56,
+ "versionNonce": 1745929623,
+ "isDeleted": false,
+ "boundElements": null,
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false,
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ 86,
+ -0.220972375229735
+ ]
+ ],
+ "lastCommittedPoint": null,
+ "startBinding": {
+ "elementId": "RYL6z1yNI4ryhDrjxemyL",
+ "focus": -0.04931816566337021,
+ "gap": 1
+ },
+ "endBinding": {
+ "elementId": "igO3c0IESW8ZSDcllatP-",
+ "focus": 0.016806722689075442,
+ "gap": 1
+ },
+ "startArrowhead": null,
+ "endArrowhead": "triangle"
+ },
+ {
+ "id": "BBin9aP3kMIsghlY9Q3Xb",
+ "type": "rectangle",
+ "x": 803,
+ "y": 235,
+ "width": 305,
+ "height": 200,
+ "angle": 0,
+ "strokeColor": "#868e96",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "dashed",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": {
+ "type": 3
+ },
+ "seed": 1920549721,
+ "version": 263,
+ "versionNonce": 2133981879,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "id": "I--VThaGzeWEpMH77IyuH",
+ "type": "arrow"
+ }
+ ],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "mxgPLlWP_Rswux186mo6F",
+ "type": "image",
+ "x": 906,
+ "y": 149,
+ "width": 85,
+ "height": 85,
+ "angle": 0,
+ "strokeColor": "transparent",
+ "backgroundColor": "transparent",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "dashed",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "seed": 1091137593,
+ "version": 301,
+ "versionNonce": 1118800855,
+ "isDeleted": false,
+ "boundElements": null,
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false,
+ "status": "saved",
+ "fileId": "674c6c1c00fa6878a440ca4cb9c65ee6ce0b9d51",
+ "scale": [
+ 1,
+ 1
+ ]
+ },
+ {
+ "id": "4N7vb2roINeGhvAK-EnUr",
+ "type": "rectangle",
+ "x": 831,
+ "y": 278,
+ "width": 114,
+ "height": 108,
+ "angle": 0,
+ "strokeColor": "#6741d9",
+ "backgroundColor": "#d0bfff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": {
+ "type": 3
+ },
+ "seed": 717291801,
+ "version": 341,
+ "versionNonce": 513669367,
+ "isDeleted": false,
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "4YZ1HMXFK8M48eFnDMR4v"
+ }
+ ],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false
+ },
+ {
+ "id": "4YZ1HMXFK8M48eFnDMR4v",
+ "type": "text",
+ "x": 846.984375,
+ "y": 308,
+ "width": 82.03125,
+ "height": 48,
+ "angle": 0,
+ "strokeColor": "#6741d9",
+ "backgroundColor": "#d0bfff",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "seed": 1393907641,
+ "version": 262,
+ "versionNonce": 1356851097,
+ "isDeleted": false,
+ "boundElements": null,
+ "updated": 1700421905507,
+ "link": null,
+ "locked": false,
+ "text": "Matrix\nSynapse",
+ "fontSize": 20,
+ "fontFamily": 3,
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "baseline": 43,
+ "containerId": "4N7vb2roINeGhvAK-EnUr",
+ "originalText": "Matrix\nSynapse",
+ "lineHeight": 1.2
+ },
+ {
+ "type": "rectangle",
+ "version": 494,
+ "versionNonce": 142721559,
+ "isDeleted": false,
+ "id": "9lVssKY2uk0fr5TCbd5kT",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "angle": 0,
+ "x": 969,
+ "y": 277,
+ "strokeColor": "#6741d9",
+ "backgroundColor": "#d0bfff",
+ "width": 114,
+ "height": 108,
+ "seed": 487814649,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": {
+ "type": 3
+ },
+ "boundElements": [
+ {
+ "type": "text",
+ "id": "QJwL5UFm3QWsAGT_IMZZG"
+ }
+ ],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false
+ },
+ {
+ "type": "text",
+ "version": 431,
+ "versionNonce": 462523513,
+ "isDeleted": false,
+ "id": "QJwL5UFm3QWsAGT_IMZZG",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 1,
+ "opacity": 100,
+ "angle": 0,
+ "x": 979.125,
+ "y": 321.4,
+ "strokeColor": "#6741d9",
+ "backgroundColor": "#d0bfff",
+ "width": 93.75,
+ "height": 19.2,
+ "seed": 1463634583,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "boundElements": [],
+ "updated": 1700421905508,
+ "link": null,
+ "locked": false,
+ "fontSize": 16,
+ "fontFamily": 3,
+ "text": "PostgreSQL",
+ "textAlign": "center",
+ "verticalAlign": "middle",
+ "containerId": "9lVssKY2uk0fr5TCbd5kT",
+ "originalText": "PostgreSQL",
+ "lineHeight": 1.2,
+ "baseline": 15
+ },
+ {
+ "type": "arrow",
+ "version": 257,
+ "versionNonce": 1364158263,
+ "isDeleted": false,
+ "id": "I--VThaGzeWEpMH77IyuH",
+ "fillStyle": "solid",
+ "strokeWidth": 2,
+ "strokeStyle": "solid",
+ "roughness": 0,
+ "opacity": 100,
+ "angle": 0,
+ "x": 714,
+ "y": 329.467361039966,
+ "strokeColor": "#e8590c",
+ "backgroundColor": "#a5d8ff",
+ "width": 86,
+ "height": 0.2401904503404353,
+ "seed": 184831705,
+ "groupIds": [],
+ "frameId": null,
+ "roundness": null,
+ "boundElements": [],
+ "updated": 1700421911263,
+ "link": null,
+ "locked": false,
+ "startBinding": {
+ "elementId": "igO3c0IESW8ZSDcllatP-",
+ "gap": 2,
+ "focus": 0.010959919320073687
+ },
+ "endBinding": {
+ "elementId": "BBin9aP3kMIsghlY9Q3Xb",
+ "gap": 3,
+ "focus": 0.06180802042331033
+ },
+ "lastCommittedPoint": null,
+ "startArrowhead": null,
+ "endArrowhead": "triangle",
+ "points": [
+ [
+ 0,
+ 0
+ ],
+ [
+ 86,
+ -0.2401904503404353
+ ]
+ ]
+ }
+ ],
+ "appState": {
+ "gridSize": null,
+ "viewBackgroundColor": "#ffffff"
+ },
+ "files": {
+ "674c6c1c00fa6878a440ca4cb9c65ee6ce0b9d51": {
+ "mimeType": "image/png",
+ "id": "674c6c1c00fa6878a440ca4cb9c65ee6ce0b9d51",
+ "dataURL": "",
+ "created": 1700421758712,
+ "lastRetrieved": 1700421758712
+ }
+ }
+}
\ No newline at end of file
diff --git a/docs/diagrams/diagram.png b/docs/diagrams/diagram.png
new file mode 100644
index 0000000..bb027bd
Binary files /dev/null and b/docs/diagrams/diagram.png differ