From 506bae7de681ec120202eaf7d44f98a4ff04e1a3 Mon Sep 17 00:00:00 2001 From: en-ver Date: Wed, 22 Jan 2025 19:46:56 +0300 Subject: [PATCH 01/28] added venv didrectories to exclusions --- .gitignore | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.gitignore b/.gitignore index 14ec1a9a3..e76bee5da 100644 --- a/.gitignore +++ b/.gitignore @@ -1,7 +1,11 @@ /build +_build /.bundle Gemfile.lock source/_static/test .pivotalrc .idea *.log + +.env +.venv \ No newline at end of file From d003f91001fd03cad37bf538e48af814e58546bf Mon Sep 17 00:00:00 2001 From: en-ver Date: Wed, 22 Jan 2025 19:49:27 +0300 Subject: [PATCH 02/28] refactoring --- Dockerfile | 26 ++++++++++++++++++++++++++ README-dev.md | 5 +++++ docker-compose.yml | 0 requirements-dev.txt | 5 +++++ requirements.txt | 3 +++ 5 files changed, 39 insertions(+) create mode 100644 Dockerfile create mode 100644 README-dev.md create mode 100644 docker-compose.yml create mode 100644 requirements-dev.txt diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 000000000..c11bb7d0e --- /dev/null +++ b/Dockerfile @@ -0,0 +1,26 @@ +# Use the official Python 3.13 Alpine image as the base +FROM python:3.13-alpine3.21 + +# Set environment variables to prevent Python from writing .pyc files +ENV PYTHONDONTWRITEBYTECODE=1 +ENV PYTHONUNBUFFERED=1 + +# Install system dependencies +RUN apk add --no-cache \ + build-base \ + libffi-dev \ + openssl-dev \ + git + +# Install Sphinx and common dependencies +RUN pip install --upgrade pip && \ + pip install sphinx sphinx_rtd_theme myst_parser + +# Create and set the working directory +WORKDIR /docs + +# Copy documentation source files into the container +COPY . /docs + +# Default command to build the Sphinx documentation +CMD ["sphinx-build", "-b", "html", ".", "_build"] \ No newline at end of file diff --git a/README-dev.md b/README-dev.md new file mode 100644 index 000000000..706e953f8 --- /dev/null +++ b/README-dev.md @@ -0,0 +1,5 @@ +Run shinx so it builds teh directory for every page to hide `.html` extension from the path + +```bash +sphinx-autobuild -b dirhtml source _build/html +``` \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 000000000..e69de29bb diff --git a/requirements-dev.txt b/requirements-dev.txt new file mode 100644 index 000000000..7315bc22c --- /dev/null +++ b/requirements-dev.txt @@ -0,0 +1,5 @@ +furo +sphinx-copybutton +# sphinx +sphinx-sitemap +sphinx-autobuild \ No newline at end of file diff --git a/requirements.txt b/requirements.txt index ffd6786dc..3c134f214 100644 --- a/requirements.txt +++ b/requirements.txt @@ -2,3 +2,6 @@ furo sphinx-copybutton Sphinx==8.1.3 sphinx-sitemap==2.6.0 + + +sphinx-autobuild \ No newline at end of file From 418cb1aacd4b3a73b39486b2dce5da20e36208ac Mon Sep 17 00:00:00 2001 From: en-ver Date: Thu, 23 Jan 2025 17:42:33 +0300 Subject: [PATCH 03/28] creanup from the ruby way of deployment --- .circleci/config.yml | 38 ----------- .github/PULL_REQUEST_TEMPLATE.md | 9 --- .gitignore | 10 +-- .ruby-gemset | 1 - .ruby-version | 1 - CNAME | 1 - Gemfile | 11 ---- Guardfile | 14 ---- Procfile | 2 - README-dev.md | 5 -- Rakefile | 109 ------------------------------- robots.txt | 3 - 12 files changed, 3 insertions(+), 201 deletions(-) delete mode 100644 .circleci/config.yml delete mode 100644 .github/PULL_REQUEST_TEMPLATE.md delete mode 100644 .ruby-gemset delete mode 100644 .ruby-version delete mode 100644 CNAME delete mode 100644 Gemfile delete mode 100644 Guardfile delete mode 100644 Procfile delete mode 100644 README-dev.md delete mode 100644 Rakefile delete mode 100644 robots.txt diff --git a/.circleci/config.yml b/.circleci/config.yml deleted file mode 100644 index 72d4b3d46..000000000 --- a/.circleci/config.yml +++ /dev/null @@ -1,38 +0,0 @@ -version: 2.1 - -executors: - default: - docker: - - auth: - username: $DOCKERHUB_USERNAME - password: $DOCKERHUB_ACCESS_TOKEN - image: cimg/python:3.12 - -orbs: - python: circleci/python@2.1.1 - -jobs: - build: - executor: default - - steps: - - checkout - - - python/install-packages: - app-dir: ~/project - - - run: - name: Build documentation - command: sphinx-build -nW -b html -d build/doctrees source build/html - -workflows: - workflow: - jobs: - - build: - context: - - org-global - filters: - branches: - ignore: - - gh-pages - - /.*-gh-pages/ diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md deleted file mode 100644 index 5d752ddc8..000000000 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ /dev/null @@ -1,9 +0,0 @@ - - -## Demo - -https://deploy-preview-{{id}}--talkable-docs.netlify.app/ - -## Related Stories - -[![](http://proxies.talkable.com/talkable/PR-1234)](https://talkable.atlassian.net/browse/PR-1234) diff --git a/.gitignore b/.gitignore index e76bee5da..e0b54580d 100644 --- a/.gitignore +++ b/.gitignore @@ -1,11 +1,7 @@ -/build -_build -/.bundle -Gemfile.lock -source/_static/test -.pivotalrc .idea *.log .env -.venv \ No newline at end of file +.venv + +_build \ No newline at end of file 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/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-dev.md b/README-dev.md deleted file mode 100644 index 706e953f8..000000000 --- a/README-dev.md +++ /dev/null @@ -1,5 +0,0 @@ -Run shinx so it builds teh directory for every page to hide `.html` extension from the path - -```bash -sphinx-autobuild -b dirhtml source _build/html -``` \ No newline at end of file diff --git a/Rakefile b/Rakefile deleted file mode 100644 index f47ca6c00..000000000 --- a/Rakefile +++ /dev/null @@ -1,109 +0,0 @@ -# frozen_string_literal: true -require 'mkmf' - -SPHINX_BUILD = ENV['SPHINX_BUILD'] || 'sphinx-build' -SOURCE_DIR = 'source' -BUILD_DIR = 'build' -SPHINX_OPTS = "-b html -d #{BUILD_DIR}/doctrees #{SOURCE_DIR} #{BUILD_DIR}/html" - -task :default => :help -task :help do - system 'rake --tasks' -end - -desc 'Set up deploys' -task :setup do - sh 'git remote add staging git@github.com:talkable/void-talkable-docs.git' -end - -task :environment do - ENV['LANG'] = ENV['LC_ALL'] = 'en_US.UTF-8' - find_executable(SPHINX_BUILD) || abort("The '#{SPHINX_BUILD}' command was not found. Make sure you have Sphinx installed, then set the SPHINX_BUILD environment variable to point to the full path of the '#{SPHINX_BUILD}' executable. Alternatively you can add the directory with the executable to your PATH. If you do not have Sphinx installed, grab it from http://sphinx-doc.org/") - - regexp = /(\d+\.)?(\d+\.)?(\*|\d+)/ - required = File.read('requirements.txt').match(regexp).to_s - installed = `#{SPHINX_BUILD} --version`.match(regexp).to_s rescue '' - if !required.empty? && !installed.empty? && Gem::Version.new(required) > Gem::Version.new(installed) - abort "\nYou are running an outdated version of Sphinx #{installed}. Required version is #{required}. Run `pip3 install -r requirements.txt --break-system-packages` to upgrade Sphinx." - end -end - -desc 'Run build in test mode' -task :test => :environment do - sh "#{SPHINX_BUILD} -nW #{SPHINX_OPTS}" -end - -task :build => :environment do - sh "#{SPHINX_BUILD} #{SPHINX_OPTS}" - - Rake::FileList["#{BUILD_DIR}/html/**/*.html"].each do |filename| - File.open(filename, "r+") do |file| - old_content = file.read - new_content = old_content.gsub(%r{ [:clean, :build] - -desc 'Run the server on localhost:5001 and open a browser' -task :server => :preview do - sh '(sleep 2 && open "http://localhost:5001")&' - sh 'bundle exec foreman start' -end - -desc 'Commit and deploy changes to https://docs.talkable.com' -task :deploy => :'deploy:production' - -namespace :deploy do - def deploy(domain:, html_branch:, source_branch:, disallow_robots:, push_command:) - sh "git checkout #{html_branch}" - sh 'git pull' - sh "find . -not -path './.git' -not -path './.git/*' -not -path './Rakefile' -not -path './.circleci' -not -path './.circleci/*' -delete" - sh "git checkout #{source_branch} #{SOURCE_DIR} .gitignore requirements.txt" - sh 'git reset HEAD' - Rake::Task[:build].invoke - sh "(cd #{BUILD_DIR}/html && tar c ./) | (cd ./ && tar xf -)" - sh "rm -rf #{BUILD_DIR} #{SOURCE_DIR} .buildinfo" - File.write('.nojekyll', '') - File.write('CNAME', domain) - File.write('robots.txt', "User-agent: *\nDisallow: /") if disallow_robots - sh 'git add -A' - sh "git commit -m \"Generated gh-pages for `git log #{source_branch} -1 --pretty=short --abbrev-commit`\" && #{push_command} ; git checkout #{source_branch}" - - puts "\nDeployment finished. Check updated docs at https://#{domain}" - end - - task :production do - deploy( - domain: 'docs.talkable.com', - html_branch: 'gh-pages', - source_branch: 'master', - disallow_robots: false, - push_command: 'git push origin gh-pages' - ) - end - - desc 'Commit and deploy changes to https://void-docs.talkable.com' - task :staging do - sh 'git remote show staging' do |ok, _| - Rake::Task[:setup].invoke unless ok - end - - deploy( - domain: 'void-docs.talkable.com', - html_branch: 'void-gh-pages', - source_branch: 'void', - disallow_robots: true, - push_command: 'git push -f origin void-gh-pages && git push -f staging void-gh-pages:gh-pages' - ) - end -end diff --git a/robots.txt b/robots.txt deleted file mode 100644 index 60ed3d057..000000000 --- a/robots.txt +++ /dev/null @@ -1,3 +0,0 @@ -User-agent: * -Disallow: - From c5dafe36cedc54d3a560b09689f15ff6ca1c0777 Mon Sep 17 00:00:00 2001 From: en-ver Date: Fri, 24 Jan 2025 10:39:56 +0300 Subject: [PATCH 04/28] move robots.txt to nginx level --- nginx/robots-local.txt | 2 ++ nginx/robots-production.txt | 4 ++++ nginx/robots-staging.txt | 2 ++ source/robots.txt | 3 --- 4 files changed, 8 insertions(+), 3 deletions(-) create mode 100644 nginx/robots-local.txt create mode 100644 nginx/robots-production.txt create mode 100644 nginx/robots-staging.txt delete mode 100644 source/robots.txt diff --git a/nginx/robots-local.txt b/nginx/robots-local.txt new file mode 100644 index 000000000..77470cb39 --- /dev/null +++ b/nginx/robots-local.txt @@ -0,0 +1,2 @@ +User-agent: * +Disallow: / \ No newline at end of file diff --git a/nginx/robots-production.txt b/nginx/robots-production.txt new file mode 100644 index 000000000..0580d3269 --- /dev/null +++ b/nginx/robots-production.txt @@ -0,0 +1,4 @@ +User-agent: * +Disallow: + +Sitemap: https://docs.talkable.com/sitemap.xml \ No newline at end of file diff --git a/nginx/robots-staging.txt b/nginx/robots-staging.txt new file mode 100644 index 000000000..77470cb39 --- /dev/null +++ b/nginx/robots-staging.txt @@ -0,0 +1,2 @@ +User-agent: * +Disallow: / \ No newline at end of file diff --git a/source/robots.txt b/source/robots.txt deleted file mode 100644 index f3fc00670..000000000 --- a/source/robots.txt +++ /dev/null @@ -1,3 +0,0 @@ -User-agent: * - -Sitemap: https://docs.talkable.com/sitemap.xml From a373777c2bfec0dec1534289cac679472eb3f4ac Mon Sep 17 00:00:00 2001 From: en-ver Date: Fri, 24 Jan 2025 21:03:37 +0300 Subject: [PATCH 05/28] dockerized deployment introduction --- Dockerfile | 26 +-- README.md | 188 ++++++++++++------ docker-compose.yml | 34 ++++ nginx/conf.d/default.conf | 25 +++ .../development.txt} | 0 .../production.txt} | 2 +- .../staging.txt} | 0 requirements-dev.txt | 4 +- requirements.txt | 10 +- source/conf.py | 115 ++++++----- 10 files changed, 260 insertions(+), 144 deletions(-) create mode 100644 nginx/conf.d/default.conf rename nginx/{robots-local.txt => robots/development.txt} (100%) rename nginx/{robots-production.txt => robots/production.txt} (85%) rename nginx/{robots-staging.txt => robots/staging.txt} (100%) diff --git a/Dockerfile b/Dockerfile index c11bb7d0e..70a1c498f 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,26 +1,6 @@ -# Use the official Python 3.13 Alpine image as the base FROM python:3.13-alpine3.21 -# Set environment variables to prevent Python from writing .pyc files -ENV PYTHONDONTWRITEBYTECODE=1 -ENV PYTHONUNBUFFERED=1 - -# Install system dependencies -RUN apk add --no-cache \ - build-base \ - libffi-dev \ - openssl-dev \ - git - -# Install Sphinx and common dependencies -RUN pip install --upgrade pip && \ - pip install sphinx sphinx_rtd_theme myst_parser - -# Create and set the working directory +# Install dependencies WORKDIR /docs - -# Copy documentation source files into the container -COPY . /docs - -# Default command to build the Sphinx documentation -CMD ["sphinx-build", "-b", "html", ".", "_build"] \ No newline at end of file +ADD requirements.txt /docs +RUN python3 -m pip install -r requirements.txt \ No newline at end of file diff --git a/README.md b/README.md index 6f23ceb9c..394178994 100644 --- a/README.md +++ b/README.md @@ -1,21 +1,134 @@ Talkable Documentation ====================== -[![Build Status](https://circleci.com/gh/talkable/talkable-docs.svg?style=svg&circle-token=cc33458158e7b0c1f6f8cbf1bcbf74f00ee28a8e)](https://circleci.com/gh/talkable/workflows/talkable-docs) +Overview +-------- This GitHub repository represents Talkable’s documentation site, located at [docs.talkable.com](https://docs.talkable.com). 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). -Sphinx ------- - For more details see [The Sphinx Documentation](https://www.sphinx-doc.org). -reStructuredText ----------------- +Concepts +-------- + +#### reStructuredText + +An easy-to-read, what-you-see-is-what-you-get plaintext markup syntax. +All reStructuredText formatting capabilities can be found at [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html). + +#### Documentation Builder + +Dockerized set of Sphinx and Nginx containers. Used to build the static pages from .rst and other format files into html static pages (Sphinx) and populate tyem in form of the documentation pages (Nginx) available to the user. + +#### Documentation Building Process + +The .rst files need to be buit to static html files. That doesn't require any involvement from developers since is handled by the [sphinx-autobuild](https://github.com/sphinx-doc/sphinx-autobuild) package included in to the Documentation Builder set mentoned above. + +#### Changes Deployment + +The local deployment doesn't require any efforts since available instantly after the files are changed on the local machine storage. The Builder buils teh changes just after indicated the change. +The staging and producxtion deployment are handled by the corresponding Jenkins jobs just after the changes are pushed to the proper branch (staging or master) + +Environments +------------ + +### Local + +Used to make teh local changes on developer's machine before pushing them to staging environment for testing by QA. +Deployed in form of docker container on the developer's machine. +Code managed in the dedicated github branch created from `master` branch. + +### Staging + +Used by QA team to test the documentation before pushing it to master branch/production environment. +Deployed in dockerised AWS infrastructure. +Code managed in `staging-bastion` branch. +Available under the following url `url to be provided` + +### Production + +The public version of the documentation available to the users. +Deployed in dockerised AWS infrastructure. +Code managed in `master` branch. +Available under the following url `url to be provided` + +Workflows +--------- + +### General Flow: + +1. Pull changes from master +2. Checkout your new branch from master +3. Deploy local Sphinx container +4. Make changes, test your changes locally +5. Commit the changes to staging branch +6. Get the documentation tested by QA +7. Create a Pull Request to "main" branch, providing the staging URL to the changed page in Pull Request’s description. +8. Merge pull request once it passes the review + +### Builder Container Deployment + +0. Install Docker + + Follow the [official Docker documentation](https://docs.docker.com/compose/install/) + + +1. Navigate to the repo root directory. + + Make sure `docker-compose.yaml` file is located there. + +2. Create `.env` file and set the value of the port you want the documentation to be available on your local machine. Use `.env.example` file as a template. + + ``` + NGINX_PORT=8080 + ENVIRONMENT=local + ``` + + Set the port number to the value you want to use as a port number while browsing the documentation locally. + For teh example above you would use `http://localhost:8080`. Chose whichever free port. + + Possible values for `ENVIRONMENT` variable: + - `local` for local development environment deployment + - `staging` for staging + - `production` for production + +3. Run the local environment deployment + + Run the command to deploy the Builder -For more details see [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html). + ``` + docker-compose up -d + ``` + +4. Check the successfull deployment + + Try follow the link [http://localhost:8080](http://localhost:8080) + Make sure you use the port number define din `.env` file + If you've done everything right the documentation will open. + If it doesn't check teh Troubleshooting section + +Troubleshooting +--------------- + +#### Can't view the documentation locally in browser + +1. Make sure you use correct post number and protocol. + The port number should equal the number you've provided in `.env` as `NGINX_PORT` value + +2. Check `nginx` logs: + ```shell + docker logs -f nginx + ``` + +3. Check `Sphinx` logs + ```shell + dcoker logs -f sphinx + ``` + +Formatting examples +------------------- ### Sections @@ -57,65 +170,12 @@ Here is a reference to "talkable section": :ref:`talkable-section` which will ha name "Talkable Section". ``` -Build the documentation ------------------------ - -First install [Sphinx](https://www.sphinx-doc.org). See below. - -### Installing Sphinx on macOS - -* Install [Homebrew](https://brew.sh). - -* Install Ruby and [Bundler](https://bundler.io), and run `bundle install` to install dependencies. - -* Install Python (>= 3.9) and pip: - - ``` - brew install python - ``` - - More information in case of trouble: https://docs.brew.sh/Homebrew-and-Python - -* Install dependencies: - - ``` - pip3 install -r requirements.txt --break-system-package - ``` - - If you have problems, try adding `-I` flag (`--ignore-installed`) to the `pip install` command. - -If you get the error "unknown locale: UTF-8" when generating the documentation, -the solution is to define the following environment variables: - - export LANG=en_US.UTF-8 - export LC_ALL=en_US.UTF-8 - -### Building - -Run `rake preview` from "master" branch. - -#### Setting up LiveReload - -Run `rake server` from "master" branch and open `http://localhost:5000` in browser. - -### Deploying - -If you’re deploying for the first time make sure you have `gh-pages` branch locally. Otherwise run the following command to create it: `git checkout -b gh-pages origin/gh-pages`. - -General flow: -1. Pull changes from master -2. Checkout your new branch from master -3. Make changes -4. Deploy your changes to staging (see the instruction below) -4. Create a Pull Request to "master" branch, providing the demo URL to the changed page in Pull Request’s description. -5. Merge pull request once it passes the review -6. Deploy -If you did everything right, deploying is as easy as `rake deploy` from "master" branch. +Deployment +---------- -#### Deploying to Staging +#### Local Enviro -If it’s your first time deploying to staging, run `rake setup` to setup git remote. 1. Switch to local branch "void" and pull the latest changes from the remote: `git checkout void; git pull` @@ -123,8 +183,6 @@ If it’s your first time deploying to staging, run `rake setup` to setup git re `git merge YOUR_BRANCH_NAME` 3. Push the changes to the remote branch "void": ```git push origin void``` -4. Deploy: - ```rake deploy:staging``` --- diff --git a/docker-compose.yml b/docker-compose.yml index e69de29bb..eedfc3839 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -0,0 +1,34 @@ +services: + + sphinx: + build: + context: . + dockerfile: Dockerfile + container_name: sphinx + volumes: + - ./source:/docs/source + environment: + - ENVIRONMENT=${ENVIRONMENT} + - NGINX_PORT=${NGINX_PORT} + command: sphinx-autobuild -b dirhtml --host 0.0.0.0 /docs/source /docs/_build + networks: + - docs_net + + nginx: + image: nginx:1.27.3-alpine3.20 + container_name: nginx + volumes: + - ./nginx/conf.d:/etc/nginx/conf.d + networks: + - docs_net + ports: + - ${NGINX_PORT}:80 + depends_on: + - sphinx + +networks: + docs_net: + name: docs_net + driver: bridge + + diff --git a/nginx/conf.d/default.conf b/nginx/conf.d/default.conf new file mode 100644 index 000000000..2ab66a10c --- /dev/null +++ b/nginx/conf.d/default.conf @@ -0,0 +1,25 @@ +server { + listen 80; + server_name _; + + # root /var/www/html; + # index index.html; + + # Redirect URLs ending with /path/index.html to /path/ + location ~* ^(.*/)?index\.html$ { + return 301 $scheme://$http_host$1; + } + + # Redirect all other .html URLs to corresponding directory + location ~* ^(.+)\.html$ { + return 301 $scheme://$http_host$1/; + } + + location / { + proxy_pass http://sphinx:8000; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + } +} \ No newline at end of file diff --git a/nginx/robots-local.txt b/nginx/robots/development.txt similarity index 100% rename from nginx/robots-local.txt rename to nginx/robots/development.txt diff --git a/nginx/robots-production.txt b/nginx/robots/production.txt similarity index 85% rename from nginx/robots-production.txt rename to nginx/robots/production.txt index 0580d3269..85f681f30 100644 --- a/nginx/robots-production.txt +++ b/nginx/robots/production.txt @@ -1,4 +1,4 @@ User-agent: * -Disallow: +Allow: / Sitemap: https://docs.talkable.com/sitemap.xml \ No newline at end of file diff --git a/nginx/robots-staging.txt b/nginx/robots/staging.txt similarity index 100% rename from nginx/robots-staging.txt rename to nginx/robots/staging.txt diff --git a/requirements-dev.txt b/requirements-dev.txt index 7315bc22c..83c514c9d 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -1,5 +1,5 @@ furo sphinx-copybutton -# sphinx sphinx-sitemap -sphinx-autobuild \ No newline at end of file +sphinx-autobuild +sphinx-favicon \ No newline at end of file diff --git a/requirements.txt b/requirements.txt index 3c134f214..e358dfdf7 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,7 +1,5 @@ -furo -sphinx-copybutton -Sphinx==8.1.3 +furo==2024.8.6 +sphinx-copybutton==0.5.2 sphinx-sitemap==2.6.0 - - -sphinx-autobuild \ No newline at end of file +sphinx-autobuild==2024.10.3 +sphinx-favicon==1.0.1 \ No newline at end of file diff --git a/source/conf.py b/source/conf.py index 9ba32d7fc..91d996d65 100644 --- a/source/conf.py +++ b/source/conf.py @@ -13,87 +13,105 @@ # serve to show the default. # import sys -# import os +import os + +environment = os.getenv("ENVIRONMENT", "production") +nginx_port = os.getenv("NGINX_PORT", "") + + +def get_baseurl(environment) -> str: + if environment == "development": + return f"http://localhost:{nginx_port}/" + elif environment == "staging": + return "https://void-docs.talkable.com/" + else: + return "https://docs.talkable.com/" + # If extensions (or modules to document with autodoc) are in another directory, # add these directories to sys.path here. If the directory is relative to the # documentation root, use os.path.abspath to make it absolute, like shown here. -#sys.path.insert(0, os.path.abspath('.')) +# sys.path.insert(0, os.path.abspath('.')) # -- General configuration ------------------------------------------------ # If your documentation needs a minimal Sphinx version, state it here. -#needs_sphinx = '1.0' +# needs_sphinx = '1.0' # Add any Sphinx extension module names here, as strings. They can be # extensions coming with Sphinx (named 'sphinx.ext.*') or your custom # ones. -extensions = ['sphinx_sitemap', 'sphinx_copybutton'] +extensions = [ + "sphinx_sitemap", + "sphinx_copybutton", + "sphinx_favicon", +] + sitemap_url_scheme = "{link}" # Add any paths that contain templates here, relative to this directory. -templates_path = ['_templates'] +templates_path = ["_templates"] # The suffix of source filenames. -source_suffix = '.rst' +source_suffix = ".rst" # The encoding of source files. -#source_encoding = 'utf-8-sig' +# source_encoding = 'utf-8-sig' # The master toctree document. -master_doc = 'index' +master_doc = "index" # General information about the project. -project = u'Talkable' -copyright = u'Talkable Inc.' -author = u'Talkable Inc.' +project = "Talkable" +copyright = "Talkable Inc." +author = "Talkable Inc." # The version info for the project you’re documenting, acts as replacement for # |version| and |release|, also used in various other places throughout the # built documents. # # The short X.Y version. -version = u'1.0' +version = "1.0" # The full version, including alpha/beta/rc tags. -release = u'1.0' +release = "1.0" # The language for content autogenerated by Sphinx. Refer to documentation # for a list of supported languages. -#language = None +# language = None # There are two options for replacing |today|: either, you set today to some # non-false value, then it is used: -#today = '' +# today = '' # Else, today_fmt is used as the format for a strftime call. -#today_fmt = '%B %d, %Y' +# today_fmt = '%B %d, %Y' # List of patterns, relative to source directory, that match files and # directories to ignore when looking for source files. -exclude_patterns = ['_build', 'partials', 'samples', '.DS_Store'] +exclude_patterns = ["_build", "partials", "samples", ".DS_Store"] # The reST default role (used for this markup: `text`) to use for all # documents. -#default_role = None +# default_role = None # If true, '()' will be appended to :func: etc. cross-reference text. -#add_function_parentheses = True +# add_function_parentheses = True # If true, the current module name will be prepended to all description # unit titles (such as .. function::). -#add_module_names = True +# add_module_names = True # If true, sectionauthor and moduleauthor directives will be shown in the # output. They are ignored by default. -#show_authors = False +# show_authors = False # The name of the Pygments (syntax highlighting) style to use. -pygments_style = 'sphinx' +pygments_style = "sphinx" # A list of ignored prefixes for module index sorting. -#modindex_common_prefix = [] +# modindex_common_prefix = [] # If true, keep warnings as "system message" paragraphs in the built documents. -#keep_warnings = False +# keep_warnings = False # -- Options for HTML output ---------------------------------------------- @@ -111,30 +129,36 @@ html_show_sourcelink = False # Add any paths that contain custom themes here, relative to this directory. -#html_theme_path = [] +# html_theme_path = [] # The name for this set of Sphinx documents. If None, it defaults to # " v documentation". -html_title = u'' +html_title = "" # A shorter title for the navigation bar. Default is the same as html_title. -#html_short_title = None +# html_short_title = None # The name of an image file (relative to this directory) to place at the top # of the sidebar. -#html_logo = None +# html_logo = None # The name of an image file (within the static path) to use as favicon of the # docs. This file should be a Windows icon file (.ico) being 16x16 or 32x32 # pixels large. -#html_favicon = None +# html_favicon = None + +favicons = { + "rel": "icon", + "href": "img/favicon.ico", + "type": "image/vnd.microsoft.icon", +} # Add any paths that contain custom static files (such as style sheets) here, # relative to this directory. They are copied after the builtin static files, # so a file named "default.css" will overwrite the builtin "default.css". -html_static_path = ['_static'] +html_static_path = ["_static"] html_css_files = [ - 'talkable.css', + "talkable.css", ] html_theme_options = { "light_logo": "img/logo.svg", @@ -149,7 +173,7 @@ # Add any extra paths that contain custom files (such as robots.txt or # .htaccess) here, relative to this directory. These files are copied # directly to the root of the documentation. -html_extra_path = ['robots.txt'] +html_extra_path = ["robots.txt"] # If not None, a 'Last updated on:' timestamp is inserted at every page bottom, # using the given strftime format. @@ -158,26 +182,26 @@ # If true, SmartyPants will be used to convert quotes and dashes to # typographically correct entities. -#html_use_smartypants = True +# html_use_smartypants = True # Custom sidebar templates, maps document names to template names. -#html_sidebars = {} +# html_sidebars = {} # Additional templates that should be rendered to pages, maps page names to # template names. -#html_additional_pages = {} +# html_additional_pages = {} # If false, no module index is generated. -#html_domain_indices = True +# html_domain_indices = True # If false, no index is generated. html_use_index = False # If true, the index is split into individual pages for each letter. -#html_split_index = False +# html_split_index = False # If true, links to the reST sources are added to the pages. -#html_show_sourcelink = True +# html_show_sourcelink = True # If true, "Created using Sphinx" is shown in the HTML footer. Default is True. html_show_sphinx = False @@ -188,26 +212,23 @@ # If true, an OpenSearch description file will be output, and all pages will # contain a tag referring to it. The value of this option must be the # base URL from which the finished HTML is served. -#html_use_opensearch = '' +# html_use_opensearch = '' # This is the file name suffix for HTML files (e.g. ".xhtml"). -#html_file_suffix = None +# html_file_suffix = None # Output file base name for HTML help builder. -htmlhelp_basename = 'Talkabledoc' +htmlhelp_basename = "Talkabledoc" # The URL which points to the root of the HTML documentation. # It is used to indicate the location of document like canonical_url. -html_baseurl = 'https://docs.talkable.com/' +html_baseurl = get_baseurl(environment) # -- Options for manual page output --------------------------------------- # One entry per manual page. List of tuples # (source start file, name, description, authors, manual section). -man_pages = [ - (master_doc, 'talkable', u'Talkable Documentation', - [author], 1) -] +man_pages = [(master_doc, "talkable", "Talkable Documentation", [author], 1)] # If true, show URL addresses after external links. -#man_show_urls = False +# man_show_urls = False From ad0d42a7deb7386d4e0671f13360ffc8089f9a23 Mon Sep 17 00:00:00 2001 From: en-ver Date: Sat, 25 Jan 2025 19:39:30 +0300 Subject: [PATCH 06/28] refactoring + documentation completion --- .env.template | 10 +++ .gitignore | 3 +- README-devops.md | 78 +++++++++++++++++ README.md | 171 ++++++++++++++------------------------ docker-compose.yml | 25 +++--- nginx/conf.d/default.conf | 41 ++++++--- nginx/robots/staging.txt | 4 +- source/_utils/__init__.py | 1 + source/_utils/baseurl.py | 15 ++++ source/conf.py | 24 ++---- 10 files changed, 218 insertions(+), 154 deletions(-) create mode 100644 .env.template create mode 100644 README-devops.md create mode 100644 source/_utils/__init__.py create mode 100644 source/_utils/baseurl.py diff --git a/.env.template b/.env.template new file mode 100644 index 000000000..161f4b646 --- /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=bastion-docs.talkable.com +PRODUCTION_HOST=docs.talkable.com \ No newline at end of file diff --git a/.gitignore b/.gitignore index e0b54580d..e1d1a7ce4 100644 --- a/.gitignore +++ b/.gitignore @@ -4,4 +4,5 @@ .env .venv -_build \ No newline at end of file +_build +__pycache__ \ No newline at end of file diff --git a/README-devops.md b/README-devops.md new file mode 100644 index 000000000..0dc56da33 --- /dev/null +++ b/README-devops.md @@ -0,0 +1,78 @@ +## 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 https://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-bastion` branch for staging. + ```bash + git checkout staging-bastion + ``` + +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 + ``` + +## 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.md b/README.md index 147387d89..45ea5cb41 100644 --- a/README.md +++ b/README.md @@ -1,128 +1,74 @@ -# Talkable Documentation +## What is Talkable Documentation? -## Overview +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). -For more details see [The Sphinx Documentation](https://www.sphinx-doc.org). +## Where is it stored? -## Concepts +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-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion): A staging branch used for testing by QA. It is available at [bastion-docs.talkable.com](https://bastion-docs.talkable.com). +- Feature branches created from `master` by individual contributors/developers. -An easy-to-read, what-you-see-is-what-you-get plaintext markup syntax. -All reStructuredText formatting capabilities can be found at [The reST Quickref](https://docutils.sourceforge.io/docs/user/rst/quickref.html). +## What is the documentation update workflow? -#### Documentation Builder +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. -Dockerized set of Sphinx and Nginx containers. Used to build the static pages from .rst and other format files into html static pages (Sphinx) and populate tyem in form of the documentation pages (Nginx) available to the user. +## How to deploy the local environment? -#### Documentation Building Process +0. **Install Docker** -The .rst files need to be buit to static html files. That doesn't require any involvement from developers since is handled by the [sphinx-autobuild](https://github.com/sphinx-doc/sphinx-autobuild) package included in to the Documentation Builder set mentoned above. + Follow the [official Docker documentation](https://docs.docker.com/compose/install/). -#### Changes Deployment +1. **Navigate to the repository root directory.** -The local deployment doesn't require any efforts since available instantly after the files are changed on the local machine storage. The Builder buils teh changes just after indicated the change. -The staging and producxtion deployment are handled by the corresponding Jenkins jobs just after the changes are pushed to the proper branch (staging or master) + Ensure the `docker-compose.yaml` file is located there. -## Environments +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. -### Local +3. **Run the local environment deployment.** -Used to make teh local changes on developer's machine before pushing them to staging environment for testing by QA. -Deployed in form of docker container on the developer's machine. -Code managed in the dedicated github branch created from `master` branch. + Run the command: -### Staging - -Used by QA team to test the documentation before pushing it to master branch/production environment. -Deployed in dockerised AWS infrastructure. -Code managed in `staging-bastion` branch. -Available under the following url `url to be provided` - -### Production - -The public version of the documentation available to the users. -Deployed in dockerised AWS infrastructure. -Code managed in `master` branch. -Available under the following url `url to be provided` - -## Workflows - -### General Flow: - -1. Pull changes from master -2. Checkout your new branch from master -3. Deploy local Sphinx container -4. Make changes, test your changes locally -5. Commit the changes to staging branch -6. Get the documentation tested by QA -7. Create a Pull Request to "main" branch, providing the staging URL to the changed page in Pull Request’s description. -8. Merge pull request once it passes the review - -### Builder Container Deployment - -0. Install Docker - - Follow the [official Docker documentation](https://docs.docker.com/compose/install/) - -1. Navigate to the repo root directory. - - Make sure `docker-compose.yaml` file is located there. - -2. Create `.env` file and set the value of the port you want the documentation to be available on your local machine. Use `.env.example` file as a template. - - ``` - NGINX_PORT=8080 - ENVIRONMENT=local - ``` - - Set the port number to the value you want to use as a port number while browsing the documentation locally. - For teh example above you would use `http://localhost:8080`. Chose whichever free port. - - Possible values for `ENVIRONMENT` variable: - - - `local` for local development environment deployment - - `staging` for staging - - `production` for production - -3. Run the local environment deployment - - Run the command to deploy the Builder - - ``` + ```bash docker-compose up -d ``` -4. Check the successfull deployment + 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. - Try follow the link [http://localhost:8080](http://localhost:8080) - Make sure you use the port number define din `.env` file - If you've done everything right the documentation will open. - If it doesn't check teh Troubleshooting section + If the documentation does not load, check the **Troubleshooting** section. -## Troubleshooting +## How to deploy changes to production and staging? -#### Can't view the documentation locally in browser +You should not deploy it manually! -1. Make sure you use correct post number and protocol. - The port number should equal the number you've provided in `.env` as `NGINX_PORT` value +The deployment is handled by Jenkins jobs. -2. Check `nginx` logs: +All you need to do is commit your changes to the corresponding branch to deploy them to the appropriate server: +- Commit to the [staging-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion) branch => [bastion-docs.talkable.com](https://bastion-docs.talkable.com/). +- Commit to the [master](https://github.com/talkable/talkable-docs/tree/master) branch => [docs.talkable.com](https://docs.talkable.com/). - ```shell - docker logs -f nginx - ``` +## How do I make the actual changes? -3. Check `Sphinx` logs - ```shell - dcoker logs -f sphinx - ``` +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. -## Formatting examples +Here are some formatting examples: ### Sections @@ -164,21 +110,28 @@ Here is a reference to "talkable section": :ref:`talkable-section` which will ha name "Talkable Section". ``` -## Deployment +## Troubleshooting + +#### Can't view the documentation locally in the browser? + +1. Ensure you are using the correct port number and protocol. + The port number should match the value provided in `.env` as `LOCAL_PORT`. -#### Local Enviro +2. Check `nginx` logs: -1. Switch to local branch "void" and pull the latest changes from the remote: - `git checkout void; git pull` -2. Merge your branch into local branch "void": - `git merge YOUR_BRANCH_NAME` -3. Push the changes to the remote branch "void": - `git push origin void` + ```bash + docker logs -f nginx + ``` ---- +3. Check `Sphinx` logs: -See "master" branch: https://github.com/talkable/talkable-docs + ```bash + docker logs -f sphinx + ``` -See "gh-pages" branch: https://github.com/talkable/talkable-docs/tree/gh-pages +## Links -See GitHub Page (auto generated): https://docs.talkable.com +- GitHub "staging" branch: [staging-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion) +- Staging web server: [bastion-docs.talkable.com](https://bastion-docs.talkable.com/) +- GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) +- Production web server: [docs.talkable.com](https://docs.talkable.com/) \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml index eedfc3839..bbf7b1bc5 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -7,28 +7,27 @@ services: container_name: sphinx volumes: - ./source:/docs/source + - docs_build:/docs/_build environment: - ENVIRONMENT=${ENVIRONMENT} - - NGINX_PORT=${NGINX_PORT} - command: sphinx-autobuild -b dirhtml --host 0.0.0.0 /docs/source /docs/_build - networks: - - docs_net + - LOCAL_PORT=${LOCAL_PORT} + - PRODUCTION_HOST=${PRODUCTION_HOST} + - STAGING_HOST=${STAGING_HOST} + - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} + command: sphinx-autobuild -b dirhtml /docs/source /docs/_build nginx: image: nginx:1.27.3-alpine3.20 container_name: nginx volumes: + - docs_build:/var/www/html - ./nginx/conf.d:/etc/nginx/conf.d - networks: - - docs_net + - ./nginx/robots/${ENVIRONMENT}.txt:/var/www/robots.txt ports: - - ${NGINX_PORT}:80 + - ${LOCAL_PORT}:80 depends_on: - sphinx -networks: - docs_net: - name: docs_net - driver: bridge - - +volumes: + docs_build: + driver: local diff --git a/nginx/conf.d/default.conf b/nginx/conf.d/default.conf index 2ab66a10c..9e41dbfca 100644 --- a/nginx/conf.d/default.conf +++ b/nginx/conf.d/default.conf @@ -2,24 +2,39 @@ server { listen 80; server_name _; - # root /var/www/html; - # index index.html; + root /var/www/html; + index index.html; - # Redirect URLs ending with /path/index.html to /path/ - location ~* ^(.*/)?index\.html$ { - return 301 $scheme://$http_host$1; + # Helps preserve the port in redirects when not on :80/:443 + absolute_redirect off; + port_in_redirect on; + + # Single location to catch ANY URL ending in .html + location ~* ^/.+\.html$ { + # Avoid redirect loops if this is an internal sub-request (fallback) + if ($uri != $request_uri) { + break; + } + + # 1) If it ends with "index.html", remove "index.html" + # e.g. /path/index.html => /path/ + # e.g. /index.html => / + rewrite "^/(.*)index\.html$" /$1 permanent; + + # 2) Otherwise remove ".html" and append slash + # e.g. /overview.html => /overview/ + rewrite "^/(.+)\.html$" /$1/ permanent; } - # Redirect all other .html URLs to corresponding directory - location ~* ^(.+)\.html$ { - return 301 $scheme://$http_host$1/; + # Serve robots.txt from a specific file + location = /robots.txt { + alias /var/www/robots.txt; + access_log off; + log_not_found off; } + # Default static-file behavior location / { - proxy_pass http://sphinx:8000; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; + try_files $uri $uri/ =404; } } \ No newline at end of file diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt index 77470cb39..e3c680508 100644 --- a/nginx/robots/staging.txt +++ b/nginx/robots/staging.txt @@ -1,2 +1,4 @@ User-agent: * -Disallow: / \ No newline at end of file +Disallow: / + +Sitemap: https://bastion-docs.talkable.com/sitemap.xml \ No newline at end of file diff --git a/source/_utils/__init__.py b/source/_utils/__init__.py new file mode 100644 index 000000000..5712dd70e --- /dev/null +++ b/source/_utils/__init__.py @@ -0,0 +1 @@ +from .baseurl import baseurl diff --git a/source/_utils/baseurl.py b/source/_utils/baseurl.py new file mode 100644 index 000000000..d15df347d --- /dev/null +++ b/source/_utils/baseurl.py @@ -0,0 +1,15 @@ +import os + +local_port = os.getenv("LOCAL_PORT") +environment = os.getenv("ENVIRONMENT") +production_host = os.getenv("PRODUCTION_HOST") +staging_host = os.getenv("STAGING_HOST") +development_host = os.getenv("DEVELOPMENT_HOST") + +baseurl_json = { + "development": f"http://{development_host}:{local_port}/", + "staging": f"https://{staging_host}/", + "production": f"https://{production_host}/", +} + +baseurl = baseurl_json.get(environment) diff --git a/source/conf.py b/source/conf.py index b76ffba3d..296acbe88 100644 --- a/source/conf.py +++ b/source/conf.py @@ -12,26 +12,16 @@ # All configuration values have a default; values that are commented out # serve to show the default. -# import sys +import sys import os -environment = os.getenv("ENVIRONMENT", "production") -nginx_port = os.getenv("NGINX_PORT", "") - - -def get_baseurl(environment) -> str: - if environment == "development": - return f"http://localhost:{nginx_port}/" - elif environment == "staging": - return "https://void-docs.talkable.com/" - else: - return "https://docs.talkable.com/" - - # If extensions (or modules to document with autodoc) are in another directory, # add these directories to sys.path here. If the directory is relative to the # documentation root, use os.path.abspath to make it absolute, like shown here. -# sys.path.insert(0, os.path.abspath('.')) +sys.path.insert(0, os.path.abspath(".")) + +# Import Talkable custom code +from _utils import baseurl # -- General configuration ------------------------------------------------ @@ -174,7 +164,7 @@ def get_baseurl(environment) -> str: # Add any extra paths that contain custom files (such as robots.txt or # .htaccess) here, relative to this directory. These files are copied # directly to the root of the documentation. -html_extra_path = ["robots.txt"] +# html_extra_path = ["robots.txt"] # If not None, a 'Last updated on:' timestamp is inserted at every page bottom, # using the given strftime format. @@ -223,7 +213,7 @@ def get_baseurl(environment) -> str: # The URL which points to the root of the HTML documentation. # It is used to indicate the location of document like canonical_url. -html_baseurl = get_baseurl(environment) +html_baseurl = baseurl # -- Options for manual page output --------------------------------------- From eb557d05ff2374cd247f839bf3888394b6162477 Mon Sep 17 00:00:00 2001 From: en-ver Date: Sat, 25 Jan 2025 20:20:16 +0300 Subject: [PATCH 07/28] doc refactoring --- README-devops.md | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/README-devops.md b/README-devops.md index 0dc56da33..8268d3d6e 100644 --- a/README-devops.md +++ b/README-devops.md @@ -5,14 +5,17 @@ The Talkable documentation stack is a containerized system that uses Docker to s ## 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. @@ -26,22 +29,29 @@ The Talkable documentation stack is a containerized system that uses Docker to s ### Steps 1. **Clone the Repository**: + ```bash git clone https://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-bastion` branch for staging. + ```bash git checkout staging-bastion ``` 3. **Create and Configure the `.env` File**: + - Copy `.env.template` to `.env`: + ```bash cp .env.template .env ``` @@ -51,6 +61,7 @@ The Talkable documentation stack is a containerized system that uses Docker to s - Leave `_HOST` variables unchanged unless domain names for staging or production servers are updated. 4. **Deploy the Stack**: + ```bash docker-compose up -d ``` @@ -61,10 +72,12 @@ The Talkable documentation stack is a containerized system that uses Docker to s - 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 From 0dda506a15aae008fcb47d6ed8bae9a00a43c63c Mon Sep 17 00:00:00 2001 From: en-ver Date: Sat, 25 Jan 2025 20:30:04 +0300 Subject: [PATCH 08/28] docs reformat --- README-devops.md | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/README-devops.md b/README-devops.md index 8268d3d6e..bda87f345 100644 --- a/README-devops.md +++ b/README-devops.md @@ -38,15 +38,15 @@ The Talkable documentation stack is a containerized system that uses Docker to s - Use the `master` branch for production. - ```bash - git checkout master - ``` + ```bash + git checkout master + ``` - Use the `staging-bastion` branch for staging. - ```bash - git checkout staging-bastion - ``` + ```bash + git checkout staging-bastion + ``` 3. **Create and Configure the `.env` File**: @@ -55,7 +55,9 @@ The Talkable documentation stack is a containerized system that uses Docker to s ```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. From c2624ea1ac20b205569a6e549bb5895433691fbe Mon Sep 17 00:00:00 2001 From: Enver Date: Sun, 26 Jan 2025 14:21:24 +0300 Subject: [PATCH 09/28] Update README-devops.md change to use SSH keys instead of passwords for authentication Co-authored-by: Bohdan Zhuravel --- README-devops.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README-devops.md b/README-devops.md index bda87f345..33d735452 100644 --- a/README-devops.md +++ b/README-devops.md @@ -31,7 +31,7 @@ The Talkable documentation stack is a containerized system that uses Docker to s 1. **Clone the Repository**: ```bash - git clone https://github.com/talkable/talkable-docs.git + git clone git@github.com:talkable/talkable-docs.git ``` 2. **Switch to the Appropriate Branch**: From 809d99b0f6b72877190b1a88436137349a99763a Mon Sep 17 00:00:00 2001 From: Enver Date: Sun, 26 Jan 2025 14:39:08 +0300 Subject: [PATCH 10/28] rename `staging-bastion` to `staging` Co-authored-by: Bohdan Zhuravel --- README-devops.md | 4 ++-- README.md | 6 +++--- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/README-devops.md b/README-devops.md index 33d735452..33889c9cd 100644 --- a/README-devops.md +++ b/README-devops.md @@ -42,10 +42,10 @@ The Talkable documentation stack is a containerized system that uses Docker to s git checkout master ``` - - Use the `staging-bastion` branch for staging. + - Use the `staging` branch for staging. ```bash - git checkout staging-bastion + git checkout staging ``` 3. **Create and Configure the `.env` File**: diff --git a/README.md b/README.md index 45ea5cb41..4a23f2701 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ It's stored in a dedicated GitHub repository ([talkable-docs](https://github.com 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-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion): A staging branch used for testing by QA. It is available at [bastion-docs.talkable.com](https://bastion-docs.talkable.com). +- [staging](https://github.com/talkable/talkable-docs/tree/staging): A staging branch used for testing by QA. It is available at [bastion-docs.talkable.com](https://bastion-docs.talkable.com). - Feature branches created from `master` by individual contributors/developers. ## What is the documentation update workflow? @@ -61,7 +61,7 @@ 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-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion) branch => [bastion-docs.talkable.com](https://bastion-docs.talkable.com/). +- Commit to the [staging](https://github.com/talkable/talkable-docs/tree/staging) branch => [bastion-docs.talkable.com](https://bastion-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? @@ -131,7 +131,7 @@ name "Talkable Section". ## Links -- GitHub "staging" branch: [staging-bastion](https://github.com/talkable/talkable-docs/tree/staging-bastion) +- GitHub "staging" branch: [staging](https://github.com/talkable/talkable-docs/tree/staging) - Staging web server: [bastion-docs.talkable.com](https://bastion-docs.talkable.com/) - GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) - Production web server: [docs.talkable.com](https://docs.talkable.com/) \ No newline at end of file From 5a9074b569a564de5f3e1d00b1e87c48aa88ed61 Mon Sep 17 00:00:00 2001 From: en-ver Date: Sun, 26 Jan 2025 15:10:54 +0300 Subject: [PATCH 11/28] get back PULL_REQUEST_TEMPLATE.md --- .github/PULL_REQUEST_TEMPLATE.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .github/PULL_REQUEST_TEMPLATE.md diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md new file mode 100644 index 000000000..b462e1178 --- /dev/null +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -0,0 +1,5 @@ + + +## Related Stories + +[![](http://proxies.talkable.com/talkable/PR-1234)](https://talkable.atlassian.net/browse/PR-1234) \ No newline at end of file From 05bf23b6d6c6799cb9e03a5ebdb1019fa4bf9fdc Mon Sep 17 00:00:00 2001 From: en-ver Date: Sun, 26 Jan 2025 15:30:28 +0300 Subject: [PATCH 12/28] replace bastion-docs.talkable.com with staging-docs.talkable.com --- .env.template | 2 +- README.md | 6 +++--- nginx/robots/staging.txt | 2 +- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.env.template b/.env.template index 161f4b646..e5e69fe46 100644 --- a/.env.template +++ b/.env.template @@ -6,5 +6,5 @@ ENVIRONMENT=development # Base urls used for the canonical urls compilation DEVELOPMENT_HOST=localhost -STAGING_HOST=bastion-docs.talkable.com +STAGING_HOST=staging-docs.talkable.com PRODUCTION_HOST=docs.talkable.com \ No newline at end of file diff --git a/README.md b/README.md index 4a23f2701..f1069bacf 100644 --- a/README.md +++ b/README.md @@ -12,7 +12,7 @@ It's stored in a dedicated GitHub repository ([talkable-docs](https://github.com 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 [bastion-docs.talkable.com](https://bastion-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. ## What is the documentation update workflow? @@ -61,7 +61,7 @@ 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 => [bastion-docs.talkable.com](https://bastion-docs.talkable.com/). +- 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? @@ -132,6 +132,6 @@ name "Talkable Section". ## Links - GitHub "staging" branch: [staging](https://github.com/talkable/talkable-docs/tree/staging) -- Staging web server: [bastion-docs.talkable.com](https://bastion-docs.talkable.com/) +- Staging web server: [staging-docs.talkable.com](https://staging-docs.talkable.com/) - GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) - Production web server: [docs.talkable.com](https://docs.talkable.com/) \ No newline at end of file diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt index e3c680508..8b7d59f28 100644 --- a/nginx/robots/staging.txt +++ b/nginx/robots/staging.txt @@ -1,4 +1,4 @@ User-agent: * Disallow: / -Sitemap: https://bastion-docs.talkable.com/sitemap.xml \ No newline at end of file +Sitemap: https://staging-docs.talkable.com/sitemap.xml \ No newline at end of file From c0192d8d7b7cf9ec75bd65b733dbd90dba2b39f4 Mon Sep 17 00:00:00 2001 From: Enver Date: Sun, 26 Jan 2025 15:52:37 +0300 Subject: [PATCH 13/28] change nginx:1.27 to nginx:1.27.3 Co-authored-by: Bohdan Zhuravel --- docker-compose.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docker-compose.yml b/docker-compose.yml index bbf7b1bc5..82b089dd3 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -17,7 +17,7 @@ services: command: sphinx-autobuild -b dirhtml /docs/source /docs/_build nginx: - image: nginx:1.27.3-alpine3.20 + image: nginx:1.27-alpine3.20 container_name: nginx volumes: - docs_build:/var/www/html From 0244c2d11576383c51f755756124c4814efe2ad6 Mon Sep 17 00:00:00 2001 From: en-ver Date: Sun, 26 Jan 2025 16:07:02 +0300 Subject: [PATCH 14/28] hardcode the dependencies versions, rename confusing requirements-dev.txt to packages.txt --- packages.txt | 9 +++++++++ requirements-dev.txt | 5 ----- requirements.txt | 37 +++++++++++++++++++++++++++++++++++-- 3 files changed, 44 insertions(+), 7 deletions(-) create mode 100644 packages.txt delete mode 100644 requirements-dev.txt diff --git a/packages.txt b/packages.txt new file mode 100644 index 000000000..666d69ca9 --- /dev/null +++ b/packages.txt @@ -0,0 +1,9 @@ +# This is a list of packages used in the framework +# Used by the Sphinx maintainer +# Add the package name here if you add some to the framework + +furo +sphinx-copybutton +sphinx-sitemap +sphinx-autobuild +sphinx-favicon \ No newline at end of file diff --git a/requirements-dev.txt b/requirements-dev.txt deleted file mode 100644 index 83c514c9d..000000000 --- a/requirements-dev.txt +++ /dev/null @@ -1,5 +0,0 @@ -furo -sphinx-copybutton -sphinx-sitemap -sphinx-autobuild -sphinx-favicon \ No newline at end of file diff --git a/requirements.txt b/requirements.txt index e358dfdf7..7b2f8fbf9 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,38 @@ +alabaster==1.0.0 +anyio==4.8.0 +babel==2.16.0 +beautifulsoup4==4.12.3 +certifi==2024.12.14 +charset-normalizer==3.4.1 +click==8.1.8 +colorama==0.4.6 +docutils==0.21.2 furo==2024.8.6 +h11==0.14.0 +idna==3.10 +imagesize==1.4.1 +Jinja2==3.1.5 +MarkupSafe==3.0.2 +packaging==24.2 +Pygments==2.19.1 +requests==2.32.3 +sniffio==1.3.1 +snowballstemmer==2.2.0 +soupsieve==2.6 +Sphinx==8.1.3 +sphinx-autobuild==2024.10.3 +sphinx-basic-ng==1.0.0b2 sphinx-copybutton==0.5.2 +sphinx-favicon==1.0.1 sphinx-sitemap==2.6.0 -sphinx-autobuild==2024.10.3 -sphinx-favicon==1.0.1 \ No newline at end of file +sphinxcontrib-applehelp==2.0.0 +sphinxcontrib-devhelp==2.0.0 +sphinxcontrib-htmlhelp==2.1.0 +sphinxcontrib-jsmath==1.0.1 +sphinxcontrib-qthelp==2.0.0 +sphinxcontrib-serializinghtml==2.0.0 +starlette==0.45.3 +urllib3==2.3.0 +uvicorn==0.34.0 +watchfiles==1.0.4 +websockets==14.2 From d9167bb199485a5cc26ec4c4191cc88636e40ee8 Mon Sep 17 00:00:00 2001 From: en-ver Date: Sun, 26 Jan 2025 16:10:21 +0300 Subject: [PATCH 15/28] replace deprecated docker-compose with modern "docker compose" --- README-devops.md | 2 +- README.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/README-devops.md b/README-devops.md index 33889c9cd..21068e9ca 100644 --- a/README-devops.md +++ b/README-devops.md @@ -65,7 +65,7 @@ The Talkable documentation stack is a containerized system that uses Docker to s 4. **Deploy the Stack**: ```bash - docker-compose up -d + docker compose up -d ``` ## Environment-Specific Configuration diff --git a/README.md b/README.md index f1069bacf..527e554a4 100644 --- a/README.md +++ b/README.md @@ -47,7 +47,7 @@ The repository consists of the following branches: Run the command: ```bash - docker-compose up -d + 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. From 100b25b4fcfa1fe979a17444c7f2400c1c6930a4 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 10:34:08 +0300 Subject: [PATCH 16/28] gitignore refactoring --- .gitignore | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/.gitignore b/.gitignore index e1d1a7ce4..a89d2a3d7 100644 --- a/.gitignore +++ b/.gitignore @@ -1,8 +1,19 @@ +# IDE .idea -*.log +.vscode +# Local +*.log .env -.venv +# Sphinx _build -__pycache__ \ No newline at end of file + +# Python +.venv +__pycache__ + +# Ruby +/build +/.bundle +Gemfile.lock \ No newline at end of file From 24f89203366b666845757c9cbe771984eb399bf5 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 10:35:46 +0300 Subject: [PATCH 17/28] Maintenance guide intro --- README-maintainer.md | 215 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 215 insertions(+) create mode 100644 README-maintainer.md 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. From c2056489f3e6fbb76997b8b0dc414a251adafe45 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 10:40:12 +0300 Subject: [PATCH 18/28] build option added to make sure if it builds up to date image --- README-devops.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README-devops.md b/README-devops.md index 21068e9ca..b09d2e06a 100644 --- a/README-devops.md +++ b/README-devops.md @@ -65,7 +65,7 @@ The Talkable documentation stack is a containerized system that uses Docker to s 4. **Deploy the Stack**: ```bash - docker compose up -d + docker compose up -d --build ``` ## Environment-Specific Configuration From 05d0c4adf53cc4c002bfd6f3043b7904c9e24049 Mon Sep 17 00:00:00 2001 From: Enver Date: Mon, 27 Jan 2025 10:44:20 +0300 Subject: [PATCH 19/28] Update README.md Changed the docker-compose file extension from .yaml to .yml Co-authored-by: Bohdan Zhuravel --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 527e554a4..8d33922da 100644 --- a/README.md +++ b/README.md @@ -34,7 +34,7 @@ The repository consists of the following branches: 1. **Navigate to the repository root directory.** - Ensure the `docker-compose.yaml` file is located there. + Ensure the `docker-compose.yml` file is located there. 2. **Create an `.env` file by copying `.env.template`.** From c7f5d7badc8973967246e6f71051c91959ac5e7a Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 12:20:46 +0300 Subject: [PATCH 20/28] nginx: add caching policy and compression --- nginx/conf.d/default.conf | 45 ++++++++++++++++++++++++++++++++------- 1 file changed, 37 insertions(+), 8 deletions(-) diff --git a/nginx/conf.d/default.conf b/nginx/conf.d/default.conf index 9e41dbfca..ab632d1be 100644 --- a/nginx/conf.d/default.conf +++ b/nginx/conf.d/default.conf @@ -2,30 +2,59 @@ server { listen 80; server_name _; - root /var/www/html; + root /var/www/html; index index.html; - # Helps preserve the port in redirects when not on :80/:443 + # Basic security headers + add_header X-Frame-Options "SAMEORIGIN"; + add_header X-Content-Type-Options "nosniff"; + add_header X-XSS-Protection "1; mode=block"; + + # Gzip settings + gzip on; + gzip_types + text/plain + text/css + application/json + application/javascript + text/xml + application/xml + application/xml+rss + text/javascript + image/svg+xml + font/ttf + font/woff + font/woff2; + + # Helps preserve the port in redirects when not on 80/443 absolute_redirect off; port_in_redirect on; - # Single location to catch ANY URL ending in .html + # Rewrite .html → no .html location ~* ^/.+\.html$ { # Avoid redirect loops if this is an internal sub-request (fallback) if ($uri != $request_uri) { break; } - # 1) If it ends with "index.html", remove "index.html" - # e.g. /path/index.html => /path/ - # e.g. /index.html => / + # If it ends with "index.html", remove "index.html" + # e.g. /path/index.html => /path/ + # e.g. /index.html => / rewrite "^/(.*)index\.html$" /$1 permanent; - # 2) Otherwise remove ".html" and append slash - # e.g. /overview.html => /overview/ + # Otherwise remove ".html" and append slash + # e.g. /overview.html => /overview/ rewrite "^/(.+)\.html$" /$1/ permanent; } + # Serve static assets with long cache + location ~* \.(?:css|js|jpe?g|png|gif|ico|svg|woff2?|ttf|eot|otf|webp)$ { + expires 1y; + add_header Cache-Control "public, max-age=31536000, immutable"; + access_log off; + try_files $uri =404; + } + # Serve robots.txt from a specific file location = /robots.txt { alias /var/www/robots.txt; From f215c9e97fb4d6ab6917d88cd0a0d5374a293027 Mon Sep 17 00:00:00 2001 From: Bohdan Zhuravel Date: Mon, 27 Jan 2025 11:43:10 +0200 Subject: [PATCH 21/28] Add newline at end of files --- .env.template | 2 +- .github/PULL_REQUEST_TEMPLATE.md | 2 +- .gitignore | 2 +- Dockerfile | 2 +- README.md | 2 +- nginx/conf.d/default.conf | 2 +- nginx/robots/development.txt | 2 +- nginx/robots/production.txt | 2 +- nginx/robots/staging.txt | 2 +- packages.txt | 4 ++-- 10 files changed, 11 insertions(+), 11 deletions(-) diff --git a/.env.template b/.env.template index e5e69fe46..7d3d25946 100644 --- a/.env.template +++ b/.env.template @@ -7,4 +7,4 @@ ENVIRONMENT=development # Base urls used for the canonical urls compilation DEVELOPMENT_HOST=localhost STAGING_HOST=staging-docs.talkable.com -PRODUCTION_HOST=docs.talkable.com \ No newline at end of file +PRODUCTION_HOST=docs.talkable.com diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md index b462e1178..cca63a8aa 100644 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -2,4 +2,4 @@ ## Related Stories -[![](http://proxies.talkable.com/talkable/PR-1234)](https://talkable.atlassian.net/browse/PR-1234) \ No newline at end of file +[![](http://proxies.talkable.com/talkable/PR-1234)](https://talkable.atlassian.net/browse/PR-1234) diff --git a/.gitignore b/.gitignore index a89d2a3d7..c3f20a68f 100644 --- a/.gitignore +++ b/.gitignore @@ -16,4 +16,4 @@ __pycache__ # Ruby /build /.bundle -Gemfile.lock \ No newline at end of file +Gemfile.lock diff --git a/Dockerfile b/Dockerfile index 70a1c498f..147eda903 100644 --- a/Dockerfile +++ b/Dockerfile @@ -3,4 +3,4 @@ FROM python:3.13-alpine3.21 # Install dependencies WORKDIR /docs ADD requirements.txt /docs -RUN python3 -m pip install -r requirements.txt \ No newline at end of file +RUN python3 -m pip install -r requirements.txt diff --git a/README.md b/README.md index 8d33922da..ed2a8ba2c 100644 --- a/README.md +++ b/README.md @@ -134,4 +134,4 @@ name "Talkable Section". - GitHub "staging" branch: [staging](https://github.com/talkable/talkable-docs/tree/staging) - Staging web server: [staging-docs.talkable.com](https://staging-docs.talkable.com/) - GitHub "production" branch: [master](https://github.com/talkable/talkable-docs/tree/master) -- Production web server: [docs.talkable.com](https://docs.talkable.com/) \ No newline at end of file +- Production web server: [docs.talkable.com](https://docs.talkable.com/) diff --git a/nginx/conf.d/default.conf b/nginx/conf.d/default.conf index ab632d1be..5eb1145ca 100644 --- a/nginx/conf.d/default.conf +++ b/nginx/conf.d/default.conf @@ -66,4 +66,4 @@ server { location / { try_files $uri $uri/ =404; } -} \ No newline at end of file +} diff --git a/nginx/robots/development.txt b/nginx/robots/development.txt index 77470cb39..1f53798bb 100644 --- a/nginx/robots/development.txt +++ b/nginx/robots/development.txt @@ -1,2 +1,2 @@ User-agent: * -Disallow: / \ No newline at end of file +Disallow: / diff --git a/nginx/robots/production.txt b/nginx/robots/production.txt index 85f681f30..4acb8d432 100644 --- a/nginx/robots/production.txt +++ b/nginx/robots/production.txt @@ -1,4 +1,4 @@ User-agent: * Allow: / -Sitemap: https://docs.talkable.com/sitemap.xml \ No newline at end of file +Sitemap: https://docs.talkable.com/sitemap.xml diff --git a/nginx/robots/staging.txt b/nginx/robots/staging.txt index 8b7d59f28..ed8d11f01 100644 --- a/nginx/robots/staging.txt +++ b/nginx/robots/staging.txt @@ -1,4 +1,4 @@ User-agent: * Disallow: / -Sitemap: https://staging-docs.talkable.com/sitemap.xml \ No newline at end of file +Sitemap: https://staging-docs.talkable.com/sitemap.xml diff --git a/packages.txt b/packages.txt index 666d69ca9..cca18c7b5 100644 --- a/packages.txt +++ b/packages.txt @@ -1,9 +1,9 @@ # This is a list of packages used in the framework -# Used by the Sphinx maintainer +# Used by the documentation maintainer # Add the package name here if you add some to the framework furo sphinx-copybutton sphinx-sitemap sphinx-autobuild -sphinx-favicon \ No newline at end of file +sphinx-favicon From 83ea286d8c4589f2b63daaba2159f6dbdd02b3e5 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 16:44:55 +0300 Subject: [PATCH 22/28] different modes of Sphinx builder for different environments --- Dockerfile | 8 ++++++++ docker-compose.yml | 1 - 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 147eda903..7697fa96f 100644 --- a/Dockerfile +++ b/Dockerfile @@ -4,3 +4,11 @@ FROM python:3.13-alpine3.21 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/docker-compose.yml b/docker-compose.yml index 82b089dd3..0795ce5c1 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -14,7 +14,6 @@ services: - PRODUCTION_HOST=${PRODUCTION_HOST} - STAGING_HOST=${STAGING_HOST} - DEVELOPMENT_HOST=${DEVELOPMENT_HOST} - command: sphinx-autobuild -b dirhtml /docs/source /docs/_build nginx: image: nginx:1.27-alpine3.20 From 2430bb768a36a9f8a6a593b7023af44fe3f57d24 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 16:48:16 +0300 Subject: [PATCH 23/28] get the circleci back --- .circleci/config.yml | 33 +++++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) create mode 100644 .circleci/config.yml diff --git a/.circleci/config.yml b/.circleci/config.yml new file mode 100644 index 000000000..2f7ff556c --- /dev/null +++ b/.circleci/config.yml @@ -0,0 +1,33 @@ +version: 2.1 + +executors: + default: + docker: + - auth: + username: $DOCKERHUB_USERNAME + password: $DOCKERHUB_ACCESS_TOKEN + image: cimg/python:3.12 + +orbs: + python: circleci/python@2.1.1 + +jobs: + build: + executor: default + + steps: + - checkout + + - python/install-packages: + app-dir: ~/project + + - run: + name: Build documentation + command: sphinx-build -nW -b html -d build/doctrees source build/html + +workflows: + workflow: + jobs: + - build: + context: + - org-global \ No newline at end of file From 29aef03f121e40e7e7aa8854cda3c77d428ca837 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 17:04:53 +0300 Subject: [PATCH 24/28] change the html mode to dirhtml for cicleci run --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 2f7ff556c..93c4369a6 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -23,7 +23,7 @@ jobs: - 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: From d70976d6b093c6581008ea7bcee2b49bbfef7669 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 17:53:33 +0300 Subject: [PATCH 25/28] ENVIRONMENT variable intro to circleci - conditional. and let the job run only on master and staging branches --- .circleci/config.yml | 22 +++++++++++++++++++++- 1 file changed, 21 insertions(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 93c4369a6..1263619e0 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -14,10 +14,25 @@ orbs: jobs: build: executor: default + environment: + LOCAL_PORT: 8080 + DEVELOPMENT_HOST: localhost + STAGING_HOST: staging-docs.talkable.com + PRODUCTION_HOST: docs.talkable.com steps: - checkout + - run: + name: Set ENVIRONMENT variable + command: | + if [ "$CIRCLE_BRANCH" == "staging" ]; then + echo 'export ENVIRONMENT=staging' >> $BASH_ENV + elif [ "$CIRCLE_BRANCH" == "master" ]; then + echo 'export ENVIRONMENT=production' >> $BASH_ENV + fi + echo "ENVIRONMENT set to $ENVIRONMENT" + - python/install-packages: app-dir: ~/project @@ -30,4 +45,9 @@ workflows: jobs: - build: context: - - org-global \ No newline at end of file + - org-global + filters: + branches: + only: + - master + - staging \ No newline at end of file From 71dfa55eb907add577e826645d93f9775846652c Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 18:48:39 +0300 Subject: [PATCH 26/28] remove filter and add fallback env variable value --- .circleci/config.yml | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 1263619e0..011c17565 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -30,6 +30,8 @@ jobs: echo 'export ENVIRONMENT=staging' >> $BASH_ENV elif [ "$CIRCLE_BRANCH" == "master" ]; then echo 'export ENVIRONMENT=production' >> $BASH_ENV + else + echo 'export ENVIRONMENT=development' >> $BASH_ENV fi echo "ENVIRONMENT set to $ENVIRONMENT" @@ -46,8 +48,3 @@ workflows: - build: context: - org-global - filters: - branches: - only: - - master - - staging \ No newline at end of file From f6478aba3dc5df7af63ce6a64a1a5d3713dbf2f1 Mon Sep 17 00:00:00 2001 From: en-ver Date: Mon, 27 Jan 2025 19:00:33 +0300 Subject: [PATCH 27/28] circleci env variable processing improvement --- .circleci/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 011c17565..f7f0f961a 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -33,7 +33,7 @@ jobs: else echo 'export ENVIRONMENT=development' >> $BASH_ENV fi - echo "ENVIRONMENT set to $ENVIRONMENT" + source "$BASH_ENV" - python/install-packages: app-dir: ~/project From 2b70a61132ee88984244dc3eea5e43e4eac8b488 Mon Sep 17 00:00:00 2001 From: Bohdan Zhuravel Date: Mon, 27 Jan 2025 19:10:07 +0200 Subject: [PATCH 28/28] Cleanup --- .circleci/config.yml | 26 +++++--------------------- 1 file changed, 5 insertions(+), 21 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index f7f0f961a..ba6254a5c 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -6,37 +6,21 @@ 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: executor: default - environment: - LOCAL_PORT: 8080 - DEVELOPMENT_HOST: localhost - STAGING_HOST: staging-docs.talkable.com - PRODUCTION_HOST: docs.talkable.com steps: - checkout - - run: - name: Set ENVIRONMENT variable - command: | - if [ "$CIRCLE_BRANCH" == "staging" ]; then - echo 'export ENVIRONMENT=staging' >> $BASH_ENV - elif [ "$CIRCLE_BRANCH" == "master" ]; then - echo 'export ENVIRONMENT=production' >> $BASH_ENV - else - echo 'export ENVIRONMENT=development' >> $BASH_ENV - fi - source "$BASH_ENV" - - - python/install-packages: - app-dir: ~/project + - python/install-packages - run: name: Build documentation