diff --git a/.circleci/config.yml b/.circleci/config.yml
index 72d4b3d46..ba6254a5c 100644
--- a/.circleci/config.yml
+++ b/.circleci/config.yml
@@ -6,10 +6,12 @@ executors:
- auth:
username: $DOCKERHUB_USERNAME
password: $DOCKERHUB_ACCESS_TOKEN
- image: cimg/python:3.12
+ image: cimg/python:3.13
+ environment:
+ ENVIRONMENT: production
orbs:
- python: circleci/python@2.1.1
+ python: circleci/python@3.0.0
jobs:
build:
@@ -18,12 +20,11 @@ jobs:
steps:
- checkout
- - python/install-packages:
- app-dir: ~/project
+ - python/install-packages
- run:
name: Build documentation
- command: sphinx-build -nW -b html -d build/doctrees source build/html
+ command: sphinx-build -nW -b dirhtml -d build/doctrees source build/html
workflows:
workflow:
@@ -31,8 +32,3 @@ workflows:
- build:
context:
- org-global
- filters:
- branches:
- ignore:
- - gh-pages
- - /.*-gh-pages/
diff --git a/.env.template b/.env.template
new file mode 100644
index 000000000..7d3d25946
--- /dev/null
+++ b/.env.template
@@ -0,0 +1,10 @@
+# The port to use to populate the documentation on your local machine or server
+LOCAL_PORT=8080
+
+# Possible values: development, staging, production
+ENVIRONMENT=development
+
+# Base urls used for the canonical urls compilation
+DEVELOPMENT_HOST=localhost
+STAGING_HOST=staging-docs.talkable.com
+PRODUCTION_HOST=docs.talkable.com
diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md
index 5d752ddc8..cca63a8aa 100644
--- a/.github/PULL_REQUEST_TEMPLATE.md
+++ b/.github/PULL_REQUEST_TEMPLATE.md
@@ -1,9 +1,5 @@
-## Demo
-
-https://deploy-preview-{{id}}--talkable-docs.netlify.app/
-
## Related Stories
[](https://talkable.atlassian.net/browse/PR-1234)
diff --git a/.gitignore b/.gitignore
index 14ec1a9a3..c3f20a68f 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,7 +1,19 @@
+# IDE
+.idea
+.vscode
+
+# Local
+*.log
+.env
+
+# Sphinx
+_build
+
+# Python
+.venv
+__pycache__
+
+# Ruby
/build
/.bundle
Gemfile.lock
-source/_static/test
-.pivotalrc
-.idea
-*.log
diff --git a/.ruby-gemset b/.ruby-gemset
deleted file mode 100644
index 10b85f5b3..000000000
--- a/.ruby-gemset
+++ /dev/null
@@ -1 +0,0 @@
-talkable-docs
diff --git a/.ruby-version b/.ruby-version
deleted file mode 100644
index 47b322c97..000000000
--- a/.ruby-version
+++ /dev/null
@@ -1 +0,0 @@
-3.4.1
diff --git a/CNAME b/CNAME
deleted file mode 100644
index fe80f6e25..000000000
--- a/CNAME
+++ /dev/null
@@ -1 +0,0 @@
-docs.talkable.com
diff --git a/Dockerfile b/Dockerfile
new file mode 100644
index 000000000..7697fa96f
--- /dev/null
+++ b/Dockerfile
@@ -0,0 +1,14 @@
+FROM python:3.13-alpine3.21
+
+# Install dependencies
+WORKDIR /docs
+ADD requirements.txt /docs
+RUN python3 -m pip install -r requirements.txt
+
+CMD if [ "$ENVIRONMENT" = "development" ]; then \
+ echo "Running Sphinx in Development mode"; \
+ sphinx-autobuild -b dirhtml /docs/source /docs/_build; \
+ else \
+ echo "Running Sphinx in Staging/Production mode"; \
+ sphinx-build -b dirhtml /docs/source /docs/_build; \
+ fi
\ No newline at end of file
diff --git a/Gemfile b/Gemfile
deleted file mode 100644
index 11fcd5bc1..000000000
--- a/Gemfile
+++ /dev/null
@@ -1,11 +0,0 @@
-source 'https://rubygems.org'
-
-ruby file: '.ruby-version'
-
-gem 'base64' # Removed from standard library in Ruby 3.4
-gem 'foreman'
-gem 'guard-livereload'
-gem 'guard-shell'
-gem 'rake'
-gem 'reline' # Removed from standard library in Ruby 3.5
-gem 'webrick'
diff --git a/Guardfile b/Guardfile
deleted file mode 100644
index dfe8ecf09..000000000
--- a/Guardfile
+++ /dev/null
@@ -1,14 +0,0 @@
-guard 'shell' do
- watch(%r{(.*)\.rst}) do |m|
- system("sphinx-build -b html -d build/doctrees source build/html")
- end
-
- watch(%r{^source/_static/}) do |m|
- system("rsync -az source/_static build/html")
- end
-end
-
-guard 'livereload' do
- watch(%r{(.*)\.rst})
- watch(%r{^source/_static/})
-end
diff --git a/Procfile b/Procfile
deleted file mode 100644
index 2da95c518..000000000
--- a/Procfile
+++ /dev/null
@@ -1,2 +0,0 @@
-web: ruby -run -e httpd -- --port=5001 build/html
-guard: bundle exec guard
diff --git a/README-devops.md b/README-devops.md
new file mode 100644
index 000000000..b09d2e06a
--- /dev/null
+++ b/README-devops.md
@@ -0,0 +1,93 @@
+## Overview
+
+The Talkable documentation stack is a containerized system that uses Docker to simplify deployment and management. It is designed to generate, serve, and manage static documentation across multiple environments, such as staging and production.
+
+## Key Features
+
+1. **Containerized Components**:
+
+ - **Nginx**: Handles HTTP requests, serves static HTML files, manages URL redirection, and dynamically serves environment-specific `robots.txt` files.
+ - **Sphinx Autobuilder**: Generates static HTML files from source documentation and stores them in a persistent volume shared with Nginx.
+
+2. **Environment-Specific Behavior**:
+
+ - Configured via a `.env` file for flexibility.
+ - Supports dynamic environment-specific behavior, such as serving different `robots.txt` files based on the `ENVIRONMENT` variable.
+
+3. **Efficient Architecture**:
+
+ - Deployed on Amazon AWS Virtual Private Servers (VPS) with an AWS load balancer directing user traffic to the appropriate environment.
+ - Sphinx generates content on a persistent volume that Nginx serves directly.
+
+## Deployment Process
+
+### Prerequisites
+
+- Ensure Docker and Docker Compose are installed on the target VPS.
+- Clone the repository containing the stack configuration.
+
+### Steps
+
+1. **Clone the Repository**:
+
+ ```bash
+ git clone git@github.com:talkable/talkable-docs.git
+ ```
+
+2. **Switch to the Appropriate Branch**:
+
+ - Use the `master` branch for production.
+
+ ```bash
+ git checkout master
+ ```
+
+ - Use the `staging` branch for staging.
+
+ ```bash
+ git checkout staging
+ ```
+
+3. **Create and Configure the `.env` File**:
+
+ - Copy `.env.template` to `.env`:
+
+ ```bash
+ cp .env.template .env
+ ```
+
+ - Update the following variables:
+
+ - **`ENVIRONMENT`**: Set to `development`, `staging`, or `production`.
+ - **`LOCAL_PORT`**: Adjust if the default port (`8080`) is already in use.
+ - Leave `_HOST` variables unchanged unless domain names for staging or production servers are updated.
+
+4. **Deploy the Stack**:
+
+ ```bash
+ docker compose up -d --build
+ ```
+
+## Environment-Specific Configuration
+
+### Handling `robots.txt`
+
+- The repository includes separate `robots.txt` files for each environment.
+- The correct file is dynamically selected based on the `ENVIRONMENT` variable and mapped to the container:
+
+ ```yaml
+ volumes:
+ - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt
+ ```
+
+- Nginx serves the file at `/var/www/robots.txt` in response to `robots.txt` requests.
+
+## Persistent Data Sharing
+
+- **Static HTML Files**:
+ - Generated by Sphinx and stored in a shared volume.
+ - Served by Nginx without regeneration.
+
+- **Volume Management**:
+ - Shared volumes allow seamless access and updates between containers.
+ - Ensures efficient and consistent behavior across environments.
diff --git a/README-maintainer.md b/README-maintainer.md
new file mode 100644
index 000000000..e9891b437
--- /dev/null
+++ b/README-maintainer.md
@@ -0,0 +1,215 @@
+# Talkable Documentation Maintenance Routine
+
+This documentation provides instructions for maintaining the Sphinx Builder **framework** used to generate the Talkable Documentation.
+
+It outlines the routine for maintaining the framework and associated workflows.
+
+> **Note**:
+>
+> This guide does not cover updating the documentation content.
+> Refer to [README.md](README.md) for details on updating the Talkable documentation source, which is available at [https://docs.talkable.com/](https://docs.talkable.com/).
+
+## Scope
+
+The maintenance routine includes the following tasks:
+
+- Updating dependencies:
+ - Sphinx and other Python packages
+ - Nginx container (application server)
+ - Python container
+- Adding new extensions
+- Introducing Talkable-specific customizations (Python scripts)
+
+## Preparations
+
+1. Clone the documentation repository.
+
+ ```bash
+ git clone git@github.com:talkable/talkable-docs.git
+ cd talkable-docs
+ ```
+
+2. Create a new branch from `master`.
+
+ ```bash
+ git branch new-branch
+ git checkout new-branch
+ ```
+
+3. Generate a `.env` file from the `.env.template`.
+
+ ```bash
+ cp .env.template .env
+ ```
+
+## Updating Packages
+
+The goal is to update `requirements.txt` with the latest versions of dependencies.
+
+1. Replace `requirements.txt` with the `packages.txt` file.
+
+ ```bash
+ cp packages.txt requirements.txt
+ ```
+
+2. Build the Sphinx container.
+
+ ```bash
+ docker compose up -d --build
+ ```
+
+ This starts the framework and allows you to load the documentation at http://localhost:8080.
+
+ If the documentation fails to load, check the Sphinx container logs:
+
+ ```bash
+ docker logs -f sphinx
+ ```
+
+ ...and the Nginx logs:
+
+ ```bash
+ docker logs -f nginx
+ ```
+
+3. Test and freeze `requirements.txt`.
+
+ Ensure everything works as expected locally. Once confirmed, update `requirements.txt` to include all installed dependencies with their versions.
+
+ > **Note:**
+ >
+ > In addition to the packages listed in [packages.txt](packages.txt), `requirements.txt` will include transitive dependencies.
+
+ Save the dependencies with the following command:
+
+ ```bash
+ docker exec sphinx pip freeze > requirements.txt
+ ```
+
+4. Stop the containers.
+
+ Once the documentation is fully functional, stop the containers:
+
+ ```bash
+ docker compose down -v
+ ```
+
+5. Push the updated `requirements.txt` to GitHub.
+
+ Commit and push the updated `requirements.txt` for testing and production.
+
+## Updating the Nginx Version
+
+The goal is to update the `docker-compose.yml` file with the latest Nginx image tag.
+
+1. Check the DockerHub Nginx page for newer versions: https://hub.docker.com/_/nginx.
+
+ > **Note:**
+ >
+ > Use only Alpine-based images.
+
+2. Update the `docker-compose.yml` file with the new version tag:
+
+ ```yaml
+ image: nginx:1.27-alpine3.20
+ ```
+
+3. Deploy the updated image.
+
+ Test the new image:
+
+ ```bash
+ docker compose up -d --build
+ ```
+
+ Verify that the documentation loads at http://localhost:8080.
+
+4. Finalize the update.
+
+ Commit the updated `docker-compose.yml` to the repository for testing and production.
+
+ Stop the containers:
+
+ ```bash
+ docker compose down -v
+ ```
+
+## Updating the Python Container
+
+The Sphinx framework uses a Python Docker image from DockerHub.
+
+1. Check for the latest Python image on DockerHub: https://hub.docker.com/_/python.
+
+2. Update the image name in the [Dockerfile](./Dockerfile):
+
+ ```dockerfile
+ FROM python:3.13-alpine3.21
+ ```
+
+3. Test the deployment.
+
+ Deploy the Sphinx container to verify the updates:
+
+ ```bash
+ docker compose up -d --build
+ ```
+
+ Confirm that the documentation loads at http://localhost:8080.
+
+4. Finalize the update.
+
+ Commit the updated `Dockerfile` to the repository for testing and production.
+
+ Stop the containers:
+
+ ```bash
+ docker compose down -v
+ ```
+
+## Adding New Extensions
+
+Sphinx is a highly customizable documentation framework. You can extend its functionality with official or third-party extensions.
+
+Here are some resources:
+
+- https://sphinx-extensions.readthedocs.io/en/latest/
+- https://www.sphinx-doc.org/en/master/development/index.html
+- https://github.com/sphinx-contrib
+
+To add extensions, follow these steps:
+
+1. [Install additional Python packages](#installing-additional-packages).
+2. [Adjust the conf.py file](#modifying-configuration-files).
+3. Add Python scripts to the [./source/](./source/) directory if necessary.
+
+Start by deploying the framework container:
+
+```bash
+docker compose up -d --build
+```
+
+### Installing Additional Packages
+
+1. Add the package to `requirements.txt`.
+
+ Append the package name to `requirements.txt` (version specification is optional at this stage).
+
+ > **Note:**
+ >
+ > Version pinning can be done later.
+
+2. Rebuild the container.
+
+ Rebuild the container after modifying `requirements.txt`:
+
+ ```bash
+ docker compose up -d --build
+ ```
+
+### Modifying Configuration Files
+
+Most changes involve editing the [./source/conf.py](./source/conf.py) file or other files in the [./source/](./source/) directory.
+
+> **Note:**
+>
+> Rebuilding the container is unnecessary for changes to [./source/conf.py](./source/conf.py) or [./source/](./source/). These changes are applied automatically within 1 second.
diff --git a/README.md b/README.md
index f2c07f7a8..ed2a8ba2c 100644
--- a/README.md
+++ b/README.md
@@ -1,30 +1,83 @@
-Talkable Documentation
-======================
+## What is Talkable Documentation?
-[](https://circleci.com/gh/talkable/workflows/talkable-docs)
+The set of articles describing Talkable's capabilities, publicly available at [docs.talkable.com](https://docs.talkable.com).
-This GitHub repository represents Talkable’s documentation site, located at [docs.talkable.com](https://docs.talkable.com).
+It uses [reStructuredText](https://docutils.sourceforge.io/rst.html) as its markup language, an easy-to-read, what-you-see-is-what-you-get plaintext markup syntax. All reStructuredText formatting capabilities can be found in [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html).
-The Talkable documentation uses [reStructuredText](https://docutils.sourceforge.io/rst.html) as its markup language and is built using [Sphinx](https://www.sphinx-doc.org).
+It is built using [Sphinx](https://www.sphinx-doc.org), an open-source documentation generation tool that transforms plain text files into beautifully formatted documentation. For more details, see [The Sphinx Documentation](https://www.sphinx-doc.org).
-Sphinx
-------
+## Where is it stored?
-For more details see [The Sphinx Documentation](https://www.sphinx-doc.org).
+It's stored in a dedicated GitHub repository ([talkable-docs](https://github.com/talkable/talkable-docs)).
-reStructuredText
-----------------
+The repository consists of the following branches:
+- [master](https://github.com/talkable/talkable-docs/tree/master): The main branch used to keep the most recent stable version available at [docs.talkable.com](https://docs.talkable.com).
+- [staging](https://github.com/talkable/talkable-docs/tree/staging): A staging branch used for testing by QA. It is available at [staging-docs.talkable.com](https://staging-docs.talkable.com).
+- Feature branches created from `master` by individual contributors/developers.
-For more details see [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html).
+## What is the documentation update workflow?
+
+1. Pull changes from `master`.
+2. Checkout a new branch from `master`.
+3. Deploy the local/development environment.
+4. Make changes and test them locally.
+5. Commit the changes to the `staging` branch.
+6. Get the documentation tested by QA.
+7. Create a pull request to the `master` branch, providing the staging URL of the changed page in the pull request description.
+8. Merge the pull request once it passes the review.
+
+## How to deploy the local environment?
+
+0. **Install Docker**
+
+ Follow the [official Docker documentation](https://docs.docker.com/compose/install/).
+
+1. **Navigate to the repository root directory.**
+
+ Ensure the `docker-compose.yml` file is located there.
+
+2. **Create an `.env` file by copying `.env.template`.**
+
+ Review and update the variable values if needed.
+
+ For a **development/local environment**, all default settings should work out of the box. The only value you may need to change is `LOCAL_PORT` if `8080` is already in use on your local machine.
+
+3. **Run the local environment deployment.**
+
+ Run the command:
+
+ ```bash
+ docker compose up -d
+ ```
+
+ If everything is set up correctly, the documentation will be available at [http://localhost:8080](http://localhost:8080). Make sure you use the port number defined in the `.env` file.
+
+ If the documentation does not load, check the **Troubleshooting** section.
+
+## How to deploy changes to production and staging?
+
+You should not deploy it manually!
+
+The deployment is handled by Jenkins jobs.
+
+All you need to do is commit your changes to the corresponding branch to deploy them to the appropriate server:
+- Commit to the [staging](https://github.com/talkable/talkable-docs/tree/staging) branch => [staging-docs.talkable.com](https://staging-docs.talkable.com/).
+- Commit to the [master](https://github.com/talkable/talkable-docs/tree/master) branch => [docs.talkable.com](https://docs.talkable.com/).
+
+## How do I make the actual changes?
+
+Navigate to the [source](./source/) directory and update the files using `reStructuredText` syntax. Refer to [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html) for syntax details.
+
+Here are some formatting examples:
### Sections
Section headings are very flexible in reST. We use the following convention in the Talkable documentation:
-* `#` for module headings
-* `=` for sections
-* `-` for subsections
-* `.` for subsubsections
+- `#` for module headings
+- `=` for sections
+- `-` for subsections
+- `.` for subsubsections
### Cross-referencing
@@ -57,103 +110,28 @@ Here is a reference to "talkable section": :ref:`talkable-section` which will ha
name "Talkable Section".
```
-### General Formatting Rules for Documentation
-
-**Directives (`..` syntax)**:
-
-* All directives starting with `..` (e.g., `.. image::`, `.. note::`) must have **2 spaces** indentation for their content. Examples:
-
- ```rst
- .. meta::
- :description: This is an example of a meta directive.
-
- .. image:: /_static/img/example.jpg
- :alt: Example Image
-
- .. note::
- This is an important note.
-
- .. code-block:: html
-